swiftui state management
i was looking into some how swiftui handles state management. there are 2 mutually exclusive ways i found so far that works with reference type and value type in swift.
in this article, alice the cat will assist me with explaining how state management works in swiftui.
reference type
ObservableObject
is a protocol that a reference type object can conform to to
allow the view layer to bind and react to changes within its values.
to illustrate this protocol, i will be observing alice to see what she/he can do, and abstracted such capability inside of a view model reference object:
final class ViewModel: ObservableObject {
@Published var say = ""
func say() -> String {
"meow"
}
}
now, i will observe alice through a view:
struct ContentView: View {
@ObservedObject var viewModel = ViewModel()
var body: some View {
VStack {
Text("alice say: \(viewModel.say)")
Button("poke") {
viewModel.say()
}
}
}
}
technically, if i poke alice, alice will say "meow". great, this cat is working properly... for now.
in the ContentView
, @ObservedObject
is a swiftui macro that handles all
code generation for observing alice's say
string and then re-rendering the
Text
view itself when updates are sent from the view model.
another macro that i can use is @StateObject
, which also works for reference
type. at this point, i really don't know the difference between these 2 macro,
because according to the latest doc on StateObject
, it does the same thing as
ObservedObject
, although article such as this
one
seem to assume observed objects marked with @StateObject
wrapper don't get
destroyed and re-instantiated when its containing view struct is redrawned.
well, he's correct. but since neither does @ObservedObject
get redrawn in the
2024 swiftui, so his article is already outdated at this point.
value type
@Observable
is a newer macro introduced after ObservableObject
. it's brought
in not as a replacement, but an alternative for a different reason - the value
type.
as usual like most other swiftui macro, @Observable
comes in a pair along with
its Observable
protocol, which a struct can manually conform to. but i won't
do it here, because... "vaguely gesturing the sky...".
to make alice adopt the @Observable
:
@Observable
struct ViewModel {
var say = ""
func say() -> String {
"meow"
}
}
i have updated the view model to be of value type and along a few other things. next, update the view layer:
struct ContentView: View {
@State var viewModel = ViewModel()
var body: some View {
VStack {
Text("alice say: \(viewModel.say)")
Button("poke") {
viewModel.say()
}
}
}
}
choosing between the 2
well, most codebase are legacy codebase after 6 months or so in ios development,
since apple is fixing and deprecating their apis frequently. if legacy codebase
is already using one of the macros to manage states, be either reference type or
value type, then it makes more sense to continue doing so without changing its
underlying swift types. otherwise, why not try the new and shiny @Observable
macro apple is heavily advertising for. ¯\_(ツ)_/¯