By the end of this lesson, students should be able to:
- Refactor code to get separation of concerns
- Create views efficiently using Swift's constructs
- Understand the @State property wrapper
- Understand the @Binding property wrapper
Making our code more efficient and organized.
- Structs are immutable
- We use them all over the place with Swift.
- They are fixed values, rarely we will be changing them. (Think about Integers)
struct ContentView: View {
var isAuthenticated = false
var body: some View {
Button(action: {
self.isAuthenticated.toggle() // Error!
}, label: {
Text("auth is: \(isAuthenticated)")
})
}
}
This code produces an error:
Cannot use mutating member on immutable value: 'self' is immutable
Allow us to add extra functionality to existing types
Today we won't go deep on these, but during lab, time check out this resource:
@State
is a property wrapper that tells Swift that we will be changing the value of the property as our program runs.
When a @State
property changes, SwiftUI automatically knows that it should reload the views to reflect the new state.
struct ContentView: View {
@State var isAuthenticated = false // @State property wrapper
var body: some View {
Button(action: {
self.isAuthenticated.toggle()
}, label: {
Text("auth is: \(isAuthenticated.description)")
})
}
}
Using the @State property wrapper here solves the error you saw earlier.
What if the user updates the value of the property? How can we tell Swift to make the updates in the view?
Bind the property to the view and bind the view back to the property.
Two-way binding
Data flows in both ways to stay in sync.
struct ContentView: View {
@State var isAuthenticated = false
@State var password = ""
var body: some View {
VStack{
Button(action: {
self.isAuthenticated.toggle()
}, label: {
Text("auth is: \(isAuthenticated.description)")
})
TextField("Enter your password", text: $password)
.multilineTextAlignment(.center)
Text("You entered:\(password)")
}
}
}
Important concept: @State
is used to define variables that are used in a View. To read a variable use its name, for example: password
. To update a variable you must prefix the name with the $
, for example: $password
.
When you see a property being used by itself:
Text("You entered:\(password)")
This means we want the value to be used here. We read the value.
When you see a property with $
:
TextField("Enter your password", text: $password)
This means the value is being accessed through the property wrapper. Swift handles the two-way binding. We read and update the value.
Use a binding to create a two-way connection between a property that stores data, and a view that displays and changes the data.
A binding connects a property to a source of truth stored elsewhere, instead of storing data directly.
// Create a PlayButton view
struct PlayButton: View {
// Creates a binding to the var in ContentView (below)
@Binding var isPlaying: Bool
var body: some View {
Button(action: {
self.isPlaying.toggle()
}) {
Image(systemName: isPlaying ? "pause.circle" : "play.circle")
}
}
}
Now use the PlayButton in your ContentView.
struct ContentView: View {
@State var isPlaying = false
var body: some View {
VStack {
PlayButton(isPlaying: $isPlaying)
.font(.system(size: 100))
Text(isPlaying ? "Playing" : "Paused")
}
}
}
Important! When a variable exists in another view you'll use @Binding
. This is the case the isPLaying
variable exists in the parent view, ContentView
, but is read and set in the child view, PlayButton
.
Since the isPlaying
property is used by both
struct ContentView: View {
@State var isPlaying = false
var body: some View {
VStack {
PlayButton(isPlaying: $isPlaying)
.font(.system(size: 100))
Text(isPlaying ? "Playing" : "Paused")
}
}
}
When PlayerView
initializes PlayButton
, it passes a binding of its state property into the button's binding property. Whenever the user taps the PlayButton, the PlayerView updates its isPlaying state.
Try and recreate this pizza order form with SwiftUI.
Structure the form like this:
Form {
Section { /* name and address */ }
Section { /* pickup or delivery */ }
Section { /* Pizza size and vege */ }
Section { /* order summary */ }
Section { /* place order button */ }
}
Read about Form: https://developer.apple.com/documentation/swiftui/form/
Read about Section: https://developer.apple.com/documentation/swiftui/section/
Use an @State
var for each of the form input elements.
inputName
String - TextFieldinputAddress
String - TextFieldselectedSize
enum - PickerforPickup
Bool - ToggleisVegetarian
Bool - Toggle
For input text use:
struct ContentView: View {
@State var name = "" // define a var
var body: some View {
TextField("Name", text: $name) // set the var here!
}
}
Read about TextField here: https://developer.apple.com/documentation/swiftui/textfield/
For the Toggles use:
struct ContentView: View {
// Define a variable
@State var pickup = false
var body: some View {
// label, variable (since it can be set use $)
Toggle("Pick up or delivery", isOn: $pickup)
}
}
Read about Toggle here: https://developer.apple.com/documentation/swiftui/toggle/
For the picker, you need to define an enum like this:
// enum of type PizzaSize
enum PizzaSize: String, CaseIterable, Identifiable {
case eight, twelve, twenty
var id: Self { self }
}
Use the enum with the picker like this:
struct ContentView: View {
// selected
@State var selectedSize: PizzaSize = .twelve
var body: some View {
Picker("Pizza Size", selection: $selectedSize, content: {
Text("8\"").tag(PizzaSize.eight)
Text("12\"").tag(PizzaSize.twelve)
Text("20\"").tag(PizzaSize.twenty)
})
}
}
Read about the picker here: https://developer.apple.com/documentation/swiftui/picker/
In the summary section, you will display the results of the form input. To do this use the variables you have defined. When reading a variable you don't use the $.
// Display the name
Text("\(name) ordered")
// Display selectedSize need to use the rawValue
Text("\(selectedSize.rawValue) inch pizza")
// Display isVegetarian, use a ternary
Text("is vegetarian \(isVegetarian ? "Yes" : "No")")
//
- For Each - Hacking with Swift
- Hacking with Swift
- Calculator Logic
- Swift UI documentation in Xcode for @Binding