5

How can I use @AppStorage for a string map in a SwiftUI app?

This is what I want to do:

@AppStorage("ratings") var ratings: [String: Double] = []

But this gives me the error message “No exact matches in call to initializer”. When looking at the documentation, it looks like only a few data types are supported. It is possible to encode it as Data?

beta
  • 2,369
  • 21
  • 34

1 Answers1

15

Looking at the documentation for @AppStorage the only values that you can currently store using this property wrapper are

  • Int
  • Double
  • String
  • Bool
  • URL
  • Data

And their optional counterparts. You can also store values that conform to RawRepresentable, like enums that conform to Int or String.

If you want to store a dictionary using this method then you would have to convert it to data and store it that way.

@AppStorage("ratings")
var ratings: Data = Data() // we need to initialize it with something

Then we can save to it using

let data = ["Hello": 5.0]
guard let ratings = try? JSONEncoder().encode(data) else { return }
self.ratings = ratings

And if we want to retrieve it we can do the following:

guard let decodedRatings = try? JSONDecoder().decode([String:Double].self, from: ratings) else { return }
print(decodedRatings)

Otherwise you will have to use UserDefaults directly, you can always use onChange and State to manage it. See this example of how to use onChange. You may need to create a custom init for your view so as to populate the State the value from UserDefaults.

Though you could write your own property wrapper, this article by John Sundell explains in detail how to do it.

Andrew
  • 23,242
  • 6
  • 75
  • 89