1

How can I use @AppStorage in my ViewModel to save an array of custom types? How do I use @AppStorage with an @Published property?

Here is my example code:

MODEL

struct MyModel: Hashable, Codable {
    var text: String
    var number: Int
}

VIEW MODEL

class ViewModel: ObservableObject {

    @Published private(set) var myModels = [MyModel]()
    @Published private(set) var subscribedMyModels = [MyModel]() // TODO: sync this with AppStorage

    func fetchMyModels() {
        myModels = [
            MyModel(text: "Setting", number: 1),
            MyModel(text: "Setting", number: 2),
            MyModel(text: "Setting", number: 3),
            MyModel(text: "Setting", number: 4),
            MyModel(text: "Setting", number: 5),
            MyModel(text: "Setting", number: 6),
            MyModel(text: "Setting", number: 7),
            MyModel(text: "Setting", number: 8)
        ]
    }

    func toggleSubscription(for myModel: MyModel) {
        if subscribedMyModels.contains(myModel) {
            subscribedMyModels.removeAll(where: { $0 == myModel } )
        } else {
            subscribedMyModels.append(myModel)
        }
    }

}

VIEW

struct ContentView: View {

    @StateObject var viewModel = ViewModel()
    @AppStorage(wrappedValue: false, "settings0") private var settings0

    var body: some View {
        List {

            Section(header: Text("Some Settings")) {
                Toggle(isOn: $settings0, label: {
                    Text("Setting 0")
                })
            }
            Section(header: Text("Some other Settings")) {
                ForEach(viewModel.myModels, id: \.self) { myModel in
                    Button(action: {
                        viewModel.toggleSubscription(for: myModel)
                    }, label: {
                        HStack {
                            HStack {
                                Text(myModel.text)
                                Text(String(myModel.number))
                            }
                            Spacer()

                            if viewModel.subscribedMyModels.contains(myModel) {
                                Image(systemName: "checkmark.circle.fill")
                            } else {
                                Image(systemName: "circle")
                            }

                        }

                    })
                }
                .foregroundColor(.black)
            }
        }
        .listStyle(InsetGroupedListStyle())
        .onAppear {
            viewModel.fetchMyModels()
        }
    }
}

UI: ui

TODO: Store checked settings of list "some other settings" in AppStorage / UserDefauls.

WalterBeiter
  • 2,021
  • 3
  • 23
  • 48
  • Encode your custom types either to String or to Data before storing in AppStorage (supported types by NSUserDefaults) and decode on loading. – Asperi Sep 16 '21 at 10:19
  • 1
    [This question](https://stackoverflow.com/questions/62713150/how-to-observe-changes-in-userdefaults) shows you how to subscribe and [here is](https://stackoverflow.com/questions/69148980/trouble-implementing-userdefaults/69154210#69154210) one option for encoding – lorem ipsum Sep 16 '21 at 14:50

1 Answers1

0

First of all I recommend using CoreData instead of AppStorage

but to answer your question :

add this to ViewModel :

// this is a computed variable which changes with subscribedMyModels changes
var encodedSubscribedMyModels : Data? {
    let encoder = JSONEncoder()
    return try? encoder.encode(subscribedMyModels)
}

in your View :

    .onChange(of: viewModel.encodedSubscribedMyModels, perform: { newValue in
        // change @AppStore to newValue
        settings0 =  newValue
    })

Note : you will need to decode back the data to your data model using a JSONDecoder