How to present an alert in SwiftUI in iOS 15

⋅ 15 min read ⋅ SwiftUI Alert

Table of Contents

We can show alerts to the user in SwiftUI with an alert() modifier.

There are three ways to present an alert in iOS 15

  1. Present without data.
  2. Present with data.
  3. Present with an error.

The only difference between these options is the content that use to populate an alert.

Let's start with how to present an alert. Then we will go through three ways to present an alert one by one.

How to present an alert

SwiftUI has many variations of alert modifiers. Each variation accepts slightly different parameters, but they have one thing in common. All alert modifiers take a binding boolean value, isPresented. This binding represents the presentation and dismissal state of an alert.

Set the isPresented value to true will present the alert. When a user taps any alert action buttons, the alert dismisses, and isPresented is set back to false.

Here is an example use of alert.

struct ContentView: View {
// 1
@State private var presentAlert = false

var body: some View {
VStack {
// 2
Text(presentAlert ? "Presenting": "Dismissed")
Button("Alert") {
// 3
presentAlert = true
}
Spacer()
}
.padding()
.alert("Title", isPresented: $presentAlert, actions: {}) // 4
}
}

1 We declare a @State variable that will use as a binding value to control the alert presentation. We set this to true when we want to show an alert.
3 When a user taps the button, we set presentAlert to true. This will present the alert that is bound to this value (4).
2 Once the user tap "OK", the alert will dismiss and set the bound value, isPresented, back to false.

isPresented use to show/dismiss alerts.
isPresented use to show/dismiss alerts.

As mentioned earlier, there are many alert modifiers, but I would like to group them into three categories based on the data you want to present in an alert.

  1. Present without data.
  2. Present with data.
  3. Present with an error.

We will go through all of them one by one.

You can easily support sarunw.com by checking out this sponsor.

Sponsor sarunw.com and reach thousands of iOS developers.

Alert without data

This is the simplest form of an alert. You present an alert without relying on an external source of data. This is useful for a static alert when the content is known beforehand.

Since alert without data is the simplest form, I will use this opportunity to introduce you to three components that make up an alert.
Title.
Message.
Actions.

Title Only

The is the only required field in an alert. You have to specify a title for your alert. You pass this as the first argument of an alert modifier. SwiftUI will automatically add an OK action button for you if you don't specify any actions.

struct ContentView: View {
// 1
@State private var presentAlert = false

var body: some View {
VStack {
Button("Alert") {
// 2
presentAlert = true
}
}
.alert("Title", isPresented: $presentAlert, actions: { // 3

})
}
}

1 We declare a @State variable that will use to control the presentation. We set this to true when we want to show an alert.
2 When a user taps the button, we set presentAlert to true, which will present the alert bound to this value (3).
3 We supply only title argument for this alert.

SwiftUI will automatically add an OK action button for you if you don't specify any actions.
SwiftUI will automatically add an OK action button for you if you don't specify any actions.

Title and Message

You can supply an optional alert body which will show under the title by setting it in the message argument. SwiftUI also added an OK action button if you don't specify any actions.

I will leave the rest of the view out and focus only on .alert modifier for brevity.

.alert("Title", isPresented: $presentAlert, actions: {
// actions
}, message: {
Text("Message")
})
Alert with title and message.
Alert with title and message.

Alert Actions

Alert also has an option for us to provide a custom action button. In our previous example, we leave this empty, and SwiftUI adds a default "OK" action for us, but it doesn't have to be this way. We can come up with our own actions.

Before we can go any further, I want to highlight an important change about alert action in iOS 15.

Alert.Button is gone / New ButtonRole

Before iOS 15, we have a dedicated structure, Alert.Button, in which we can define an action and its style, default, cancel, and destructive.

These styles dictate the appearance (color) and position of the button in an alert.

  • Default: Show action in plain text.
  • Destructive: Show action in red text.
  • Cancel: Show action in bold and always on the left (This varies based on localization).

In iOS 15, Apple decided to remove Alert.Button and Alert altogether. They introduce a new ButtonRole struct to describe the purpose of a button and a new button initializer to assign a role to a button.

init<S>(_ title: S, role: ButtonRole?, action: @escaping () -> Void) where S : StringProtocol

In a moment, you will see how this button role replaces an Alert.Button functionality.

iOS 14 vs iOS 15

Alert used to hold title, message, and actions in iOS 14, but as title and message move into alert parameters, and Alert.Button can become just a Button, we no longer need an Alert struct anymore.

An alert modifier now expected three parameters: title, message, and an actions view builder that expected buttons.

Here is an equivalent alert in iOS 14 and iOS 15.

// iOS 14
.alert(isPresented: $presentAlert, content: {
Alert( // 1
title: Text("Title"),
message: Text("Message"),
dismissButton: .cancel()
)
})

// iOS 15
.alert("Title", isPresented: $presentAlert, actions: { // 2
// 3
Button("Cancel", role: .cancel, action: {})
}, message: {
// 4
Text("Message")
})

1 Alert is a representation of an alert content. In iOS 15, all properties in Alert are move to different part of an alert modifier (2, 4). And we use a simple button with a new role to replace an Alert.Button (3).

Alert in iOS 14/15 still looks the same, even though they are structured differently.
Alert in iOS 14/15 still looks the same, even though they are structured differently.

Now that we understand the changes of related components let's see how to put this into use.

Alert without data (continue)

Let's continue where we left off with an alert presentation without data. We will try to add actions to our alert.

Alert with one button

As you see in the previous example, SwiftUI will automatically add an "OK" action if you don't specify any actions. To opt-out from this "OK" action, you must set any non-destructive button to the actions.

Set default role button.

.alert("Title", isPresented: $presentAlert, actions: {
Button("Okey dokey") {}
}, message: {
Text("Message")
})
Set default role button will opt-out from the default
Set default role button will opt-out from the default "OK" action.

Set cancel role button.

You can also set a button with cancel role to remove the default "OK" action.

.alert("Title", isPresented: $presentAlert, actions: {
Button("Cancel", role: .cancel, action: {})
}, message: {
Text("Message")
})
Set cancel role button will opt-out from the default
Set cancel role button will opt-out from the default "OK" action.

Set destructive role button.
The destructive button will remove the default "OK" action, but SwiftUI automatically adds a cancel action for us instead.

.alert("Title", isPresented: $presentAlert, actions: {
Button("Destructive", role: .destructive, action: {})
}, message: {
Text("Message")
})
The destructive button will remove the default
The destructive button will remove the default "OK" action, but SwiftUI automatically adds a cancel action for us instead.

Alert with one button (Destructive)

Destructive action deserves its own section because the behavior is quite different from others, as seen in the previous section. Adding a destructive action causes SwiftUI to add a companion cancel button automatically.

Having only one destructive action means we give users no choice but to destroy something. I think that's why SwiftUI adds that cancel button for us.

We can't remove that cancel button. The best we can do is customize it. To do that, you add your version of the cancel button.

.alert("Title", isPresented: $presentAlert, actions: {
// 1
Button("Nope", role: .cancel, action: {})
Button("Destructive", role: .destructive, action: {})
}, message: {
Text("Message")
})

1 We add a .cancel role button to override the default one.

Add a cancel role button to override the default one.
Add a cancel role button to override the default one.

Alert with two or more buttons

There is no limitation on the number of buttons with this new API. The actions parameter is a view builder that turns any buttons you specified into alert actions.

Here are some behaviors you should know.

Two buttons:

  • Buttons render from left to right.
  • Cancel button is always on the left regardless of position.

More than two buttons (or two buttons with long title):

  • Buttons render from top to bottom just like a list view.
  • Cancel button is always on the bottom.
.alert("Title", isPresented: $presentAlert, actions: {            
Button("OK 1", role: nil, action: {})
Button("Destructive 1", role: .destructive, action: {})
Button("OK 2", action: {})
// 1
Button("Cancel", role: .cancel, action: {})
Button("Destructive 2", role: .destructive, action: {})
}, message: {
Text("Message")
})

1 Cancel button always render at the bottom regardless of view position.

Cancel button always render at the bottom regardless of view position.
Cancel button always render at the bottom regardless of view position.

That's all you need to know about the new alert. So far, we only populate alert content with static data. Let's see what other options do we have.

Alert with data

The difference between presenting an alert with and without data is, in the former, you pass a binding content value to an alert modifier. This value can be used when you populate alert contents.

We pass binding data to the presenting parameter to present an alert with content populated from the given data.

The example below shows a custom data source, PurchaseDetail, that provides data to populate the alert.

// 1
struct PurchaseDetail {
let trackingId: String
let numberOfItems: Int
}

struct ContentView: View {
// 2
@State private var presentAlert = false
// 3
@State private var purchaseDetail: PurchaseDetail?

var body: some View {
VStack {
// 4
Text(presentAlert ? "Presenting": "Dismissed")
Text(purchaseDetail?.trackingId ?? "nil")
Button("Purchase 1") {
// 5
purchaseDetail = PurchaseDetail(
trackingId: "abc",
numberOfItems: 1)
// 6
presentAlert = true
}
Button("Purchase 2") {
purchaseDetail = PurchaseDetail(
trackingId: "def",
numberOfItems: 2)

presentAlert = true
}
Button("Purchase 3") {
purchaseDetail = PurchaseDetail(
trackingId: "ghi",
numberOfItems: 3)

presentAlert = true
}
Spacer()
}
.padding()
.alert("Thank You",
isPresented: $presentAlert, // 7
presenting: purchaseDetail, // 8
actions: { detail in // 9
Button("OK", action: {})
}, message: { detail in // 10
Text("Tracking ID: \(detail.trackingId)")
})
}
}

1 A data source that we will use to populate our alert.
2 We still need a binding boolean value for trigger presentation.
3 We need an extra @State variable as a data source for our alert.
4 I add two text view to demonstrate value of presentAlert and purchaseDetail. This is for demonstration purposes only.
5 Each button set a purchaseDetail that will use to populate an alert. Setting this value won't present an alert. You still need to manually present it by set binding boolean value (presentAlert) to true 6.
7, 8 An alert that bound to presentAlert and purchaseDetail will show up when both isPresented is true and data must not be nil.
9 actions closure will get a current data as a parameter. You can use this information to populate the alert actions.
10 message closure also get a current data as a parameter. In this example, I use trackingId to populate the alert's body.
4 When the user presses any action button, the system sets the presentAlert back to false, but purchaseDetail is untouched.

Alert with data.
Alert with data.

How is this different from alert without data

You might notice that you can have almost the same behavior using an alert without data.

Here is how it looks.

struct ContentView: View {
@State private var presentAlert = false
@State private var purchaseDetail: PurchaseDetail?

var body: some View {
VStack {
Text(presentAlert ? "Presenting": "Dismissed")
Text(purchaseDetail?.trackingId ?? "nil")
Button("Purchase 1") {
purchaseDetail = PurchaseDetail(
trackingId: "abc",
numberOfItems: 1)

presentAlert = true
}
Button("Purchase 2") {
purchaseDetail = PurchaseDetail(
trackingId: "def",
numberOfItems: 2)

presentAlert = true
}
Button("Purchase 3") {
purchaseDetail = PurchaseDetail(
trackingId: "ghi",
numberOfItems: 3)

presentAlert = true
}
Spacer()
}
.padding()
.alert("Thank You",
isPresented: $presentAlert,
actions: {
Button("OK", action: {})
}, message: {
if let detail = purchaseDetail { // 1
Text("Tracking ID: \(detail.trackingId)")
}
})
}
}

1 The only difference is we have to unwrap optional data by ourselves.

So, there are two differences I see so far.

  1. You need to unwrap the data yourself.
  2. Alert with data will appear only if both isPresented is true and data is non-nil.

Alert with error

Presenting an alert with an error is similar to presenting alert with data. The two differences are as follow:

  1. Instead of passing an arbitrary data object, we pass LocalizedError to an error parameter.
  2. We can't explicitly define an alert title. The title of the alert is inferred from the error’s errorDescription.

Present an alert with error also got the same behavior as present an alert with data. For the alert to appear, both isPresented must be true, and error must not be nil. But again, the statement isn't true (either the documentation or implementation is wrong).

Here is an example where we use ErrorInfo to populate the alert content. Most of the code should be familiar to you. The only difference is we don't need to specify the alert's title.

struct ErrorInfo: LocalizedError {
var errorDescription: String?
var failureReason: String?
var recoverySuggestion: String?
var helpAnchor: String?
}

struct ContentView3: View {
@State private var presentAlert = false
@State private var error: ErrorInfo?

var body: some View {
VStack {
Text(presentAlert ? "Presenting": "Dismissed")
Text(error?.errorDescription ?? "nil")
Button("Alert") {
error = ErrorInfo(
errorDescription: "Error Description",
failureReason: "Failure Reason",
recoverySuggestion: "Try Again",
helpAnchor: "Help Anchor"
)
presentAlert = true
}
Spacer()
}
.padding()
.alert(
isPresented: $presentAlert,
error: error, // 1
actions: { error in // 2
if let suggestion = error.recoverySuggestion {
Button(suggestion, action: {
// Recover from an error
})
}
}, message: { error in // 3
if let failureReason = error.failureReason {
Text(failureReason)
} else {
Text("Something went wrong")
}
})
}
}

1 We pass error instead of data.
2, 3 An error pass in as a closure parameter.

Notice that the errorDescription use as an alert' title.

Alert with error.
Alert with error.

You can easily support sarunw.com by checking out this sponsor.

Sponsor sarunw.com and reach thousands of iOS developers.

Conclusion

As you can see, presenting an alert in iOS 15 is more unified than the one in prior iOS versions. We control the presenting/dismissal with only isPresented with an option to pass data/error to populate our alert content.

The way we craft our alert content is also simpler. We don't need a dedicated Alert representation. Everything is streamlined into method parameters and view builder.

I can't say this is better approach than the first version. I like that it solves a multiple alerts problem and how we can build action with a view builder, but there is something that I don't like, such as we don't have a proper way to set a dynamic title (and the option is even more limited for alert with error). There are also some rough edges and inconsistencies between documentation and the actual result, which I already submit feedbacks.


Read more article about SwiftUI, Alert, or see all available topic

Enjoy the read?

If you enjoy this article, you can subscribe to the weekly newsletter.
Every Friday, you'll get a quick recap of all articles and tips posted on this site. No strings attached. Unsubscribe anytime.

Feel free to follow me on Twitter and ask your questions related to this post. Thanks for reading and see you next time.

If you enjoy my writing, please check out my Patreon https://www.patreon.com/sarunw and become my supporter. Sharing the article is also greatly appreciated.

Become a patron Buy me a coffee Tweet Share
Previous
What does the ?? operator mean in Swift

?? is an operator that has one specific use case. Let's find out what it is.

Next
Should every if statement has an else clause

Every switch statement has a default case. Every do-catch statement has a catch clause. Should every if statement has an else clause?

← Home