Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Initialize optional @AppStorage property with non-nil value

I need an optional @AppStorage String property (for a NavigationLink selection, which required optional), so I declared

@AppStorage("navItemSelected") var navItemSelected: String?

I need it to start with a default value that's non-nil, so I tried:

@AppStorage("navItemSelected") var navItemSelected: String? = "default"

but that doesn't compile.

I also tried:

init() {
    if navItemSelected == nil { navItemSelected = "default" }
}

But this just overwrites the actual persisted value whenever the app starts.

Is there a way to start it with a default non-nil value and then have it persisted as normal?

like image 858
Steve M Avatar asked Oct 23 '25 14:10

Steve M


2 Answers

Here is a simple demo of possible approach based on inline Binding (follow-up of my comment above).

Tested with Xcode 13 / iOS 15

demo

struct DemoAppStoreNavigation: View {
    static let defaultNav = "default"
    @AppStorage("navItemSelected") var navItemSelected = Self.defaultNav

    var body: some View {
        NavigationView {
            Button("Go Next") {
                navItemSelected = "next"
            }.background(
                NavigationLink(isActive: Binding(
                    get: { navItemSelected != Self.defaultNav },
                    set: { _ in }
                ), destination: {
                    Button("Return") {
                        navItemSelected = Self.defaultNav
                    }
                    .onDisappear {
                        navItemSelected = Self.defaultNav   // << for the case of `<Back`
                    }
                }) { EmptyView() }
            )
        }
    }
}
like image 165
Asperi Avatar answered Oct 26 '25 06:10

Asperi


@AppStorage is a wrapper for UserDefaults, so you can simply register a default the old-fashioned way:

UserDefaults.standard.register(defaults: ["navItemSelected" : "default"])

You will need to call register(defaults:) before your view loads, so I’d recommend calling it in your App’s init or in application(_:didFinishLaunchingWithOptions:).

like image 23
Adam Avatar answered Oct 26 '25 06:10

Adam



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!