SwiftUI - TabView/NavigationLink 导航在使用自定义绑定时中断

时间:2021-07-21 20:45:26

标签: swiftui swiftui-navigationlink swiftui-navigationview

我认为可能存在错误,但很可能是我做错了什么。

我的模型中有一个稍微复杂的导航状态变量,当在 iPad 上进行多任务处理时,我使用它来跟踪/设置选项卡和侧边栏演示之间的状态。一切正常,除了在选项卡模式下,一旦我使用导航链接,一旦我似乎无法再次使用,无论绑定是在我的选项卡视图上还是在列表中的导航链接。

enter image description here

非常感谢您对此的任何想法, 干杯!

示例

NavigationItem.swift

enum SubNavigationItem: Hashable {
    case overview, user, hobby
}

enum NavigationItem: Hashable {
    case home(SubNavigationItem)
    case settings
}

Model.swift

final class Model: ObservableObject {
    @Published var selectedTab: NavigationItem = .home(.overview)
}

SwiftUIApp.swift

@main
struct SwiftUIApp: App {
    @StateObject var model = Model()
    
    var body: some Scene {
        WindowGroup {
            ContentView()
                .environmentObject(model)
        }
    }
}

ContentView.swift

struct ContentView: View {
    var body: some View {
        AppTabNavigation()
    }
}

AppTabNavigation.swift

struct AppTabNavigation: View {
    @EnvironmentObject private var model: Model

    var body: some View {
        TabView(selection: $model.selectedTab) {
            NavigationView {
                HomeView()
            }
            .tabItem {
                Label("Home", systemImage: "house")
            }
            .tag(NavigationItem.home(.overview))

            NavigationView {
                Text("Settings View")
            }
            .tabItem {
                Label("Settings", systemImage: "gear")
            }
            .tag(NavigationItem.settings)
        }
    }
}

HomeView.swift

我在这里创建了一个绑定,因为选择需要一个可选的 而不是

struct HomeView: View {
    @EnvironmentObject private var model: Model
    
    var body: some View {
        let binding = Binding<NavigationItem?>(
            get: { 
                model.selectedTab 
            },
            set: {
                guard let item = $0 else { return }
                model.selectedTab = item
            }
        )
        
        List {
            NavigationLink(
                destination: Text("Users"),
                tag: .home(.user),
                selection: binding
            ) {
                Text("Users")
            }
            NavigationLink(
                destination: Text("Hobbies"),
                tag: .home(.hobby),
                selection: binding
            ) {
                Text("Hobbies")
            }
        }
        .navigationTitle("Home")
    }
}

第二次尝试

我尝试按照@Lorem Ipsum 的建议将 selectedTab 属性设为可选。这意味着我可以删除那里的绑定。但是随后 TabView 不适用于该属性。所以我为此创建了一个绑定,但遇到了同样的问题,但使用了标签栏!

enter image description here

1 个答案:

答案 0 :(得分:2)

使选定的选项卡可选

@Published var selectedTab: NavigationItem? = .home(.overview)

并摆脱那个临时绑定变量。只需使用变量

$model.selectedTab

如果变量永远不能为 nil,那么总是使用临时变量选择 IAW,它只会保留最后一个值。