2024-03-30

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. ¯\_(ツ)_/¯