con un clic
boutique-swiftui
// Integrate Boutique with SwiftUI views using onChange, onStoreDidLoad, bindings, and preview stores. Use when building SwiftUI views that display or react to Boutique-persisted data.
// Integrate Boutique with SwiftUI views using onChange, onStoreDidLoad, bindings, and preview stores. Use when building SwiftUI views that display or react to Boutique-persisted data.
| name | boutique-swiftui |
| description | Integrate Boutique with SwiftUI views using onChange, onStoreDidLoad, bindings, and preview stores. Use when building SwiftUI views that display or react to Boutique-persisted data. |
Use this skill when building SwiftUI views that display or react to data from Boutique's Store, @StoredValue, or @SecurelyStoredValue.
Inject an @Observable controller as @State and access items directly.
struct NotesListView: View {
@State var notesController: NotesController
var body: some View {
List(self.notesController.notes) { note in
Text(note.text)
}
}
}
Use .onChange(of:initial:) to run code whenever the stored items change. The initial: true parameter ensures the closure also fires on first appearance.
struct NotesListView: View {
@State var notesController: NotesController
@State private var filteredNotes: [Note] = []
var body: some View {
List(self.filteredNotes) { note in
Text(note.text)
}
.onChange(of: self.notesController.notes, initial: true) { _, newValue in
self.filteredNotes = newValue.filter({ $0.text.count < 280 })
}
}
}
When a Store is initialized synchronously, items load in a background task. Use onStoreDidLoad to show loading states or trigger actions once items are ready.
struct NotesView: View {
@State var notesController: NotesController
@State private var isLoaded = false
var body: some View {
Group {
if self.isLoaded {
NotesList(notes: self.notesController.notes)
} else {
ProgressView()
}
}
.onStoreDidLoad(self.notesController.$notes, onLoad: {
self.isLoaded = true
}, onError: { error in
print("Failed to load notes:", error)
})
}
}
struct NotesView: View {
@State var notesController: NotesController
@State private var hasLoaded = false
var body: some View {
Group {
if self.hasLoaded {
NotesList(notes: self.notesController.notes)
} else {
ProgressView()
}
}
.onStoreDidLoad(self.notesController.$notes, update: self.$hasLoaded)
}
}
Both @StoredValue and @SecurelyStoredValue provide a .binding property for two-way SwiftUI bindings.
@Observable
final class Preferences {
@ObservationIgnored
@StoredValue(key: "hasHapticsEnabled")
var hasHapticsEnabled = false
}
struct SettingsView: View {
@State var preferences: Preferences
var body: some View {
Form {
Toggle(
"Haptics",
isOn: self.preferences.$hasHapticsEnabled.binding
)
}
}
}
struct ThemePickerView: View {
@State var preferences: Preferences
var body: some View {
Picker("Theme", selection: self.preferences.$currentlySelectedTheme.binding) {
Text("Light").tag(Theme.light)
Text("Dark").tag(Theme.dark)
Text("System").tag(Theme.system)
}
}
}
@SecurelyStoredValue bindings are optional (Binding<Item?>).
struct TokenView: View {
@State var securityManager: SecurityManager
var body: some View {
let tokenBinding = Binding(
get: { self.securityManager.authToken ?? "" },
set: { try? self.securityManager.$authToken.set($0.isEmpty ? nil : $0) }
)
SecureField("Auth Token", text: tokenBinding)
}
}
Wrap async store operations in Task blocks from button actions or gestures.
struct NoteRow: View {
let note: Note
@State var notesController: NotesController
var body: some View {
Text(self.note.text)
.swipeActions(edge: .trailing) {
Button("Delete", role: .destructive) {
Task {
try await self.notesController.removeNote(self.note)
}
}
}
}
}
In DEBUG builds, use Store.previewStore(items:) to create in-memory stores for SwiftUI previews.
#Preview {
let store = Store<Note>.previewStore(items: [
Note(id: "1", text: "First note", createdAt: .now),
Note(id: "2", text: "Second note", createdAt: .now),
])
let controller = NotesController(store: store)
NotesListView(notesController: controller)
}
#Preview {
let store = Store<Bookmark>.previewStore(
items: [Bookmark(url: URL(string: "https://example.com")!, title: "Example")],
cacheIdentifier: \.url.absoluteString
)
// ...
}
Preview stores only hold items in memory and do not persist to disk. They are only available in DEBUG builds.
@Observable
final class AppPreferences {
@ObservationIgnored
@StoredValue(key: "hasHapticsEnabled")
var hasHapticsEnabled = true
@ObservationIgnored
@StoredValue(key: "prefersDarkMode")
var prefersDarkMode = false
@ObservationIgnored
@StoredValue(key: "fontSize")
var fontSize: Double = 16.0
}
struct SettingsView: View {
@State var preferences: AppPreferences
var body: some View {
Form {
Section("Experience") {
Toggle("Haptics", isOn: self.preferences.$hasHapticsEnabled.binding)
Toggle("Dark Mode", isOn: self.preferences.$prefersDarkMode.binding)
}
Section("Display") {
Slider(
value: self.preferences.$fontSize.binding,
in: 12...24,
step: 1
) {
Text("Font Size: \(Int(self.preferences.fontSize))")
}
}
}
}
}
@Observable controllers should be injected as @State in SwiftUI views.$controller.property to access the projected value for Boutique operations (insert, remove, set, etc.).onStoreDidLoad is a View modifier that wraps store.itemsHaveLoaded() in a .task block.#if DEBUG only and will not compile in Release builds.boutique-store skill for building @Observable controllers with @Stored.boutique-stored-values skill for all @StoredValue and @SecurelyStoredValue APIs.Best practices for using Boutique with Swift 6 concurrency, @Observable, @ObservationIgnored, Sendable conformance, testing with preview stores, and dependency injection. Use when troubleshooting Boutique issues, migrating to Swift 6, or setting up tests.
Create and use Boutique Store for Swift data persistence, including initialization, @Stored controllers, CRUD operations, operation chaining, and granular event monitoring. Use when persisting arrays of items, building data controllers, or working with Boutique's Store type.
Persist individual values with Boutique's @StoredValue (UserDefaults) and @SecurelyStoredValue (Keychain), including set, reset, toggle, bindings, keypath setters, array and dictionary helpers, and async observation. Use when storing preferences, settings, feature flags, or sensitive data like auth tokens.