swiftui - 具有 ObservableObject 属性的 ObservableObject 未触发 SwiftUI 更新
问题描述
// NavigationState.swift
class NavigationState: ObservableObject {
@Published var contentViewNavigation = ContentViewNavigationState()
//@Published var selectedTab = 0
}
// App.swift
@main
struct SwiftUIDeepLinkApp: App {
@StateObject private var navigationAppState = NavigationState()
var body: some Scene {
WindowGroup {
ContentView()
.environmentObject(navigationAppState)
.onAppear(perform: {
DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
print("Dispatch called")
navigationAppState.contentViewNavigation.selectedTab = 1
//navigationAppState.selectedTab = 1
}
})
}
}
}
// ContentView.swift
class ContentViewNavigationState: ObservableObject {
@Published var selectedTab = 0
}
struct ContentView: View {
@EnvironmentObject var navigationState: NavigationState
var body: some View {
TabView(selection: $navigationState.contentViewNavigation.selectedTab) {
HomeContainerFlow1()
.tabItem { Text("Flow 1") }
.tag(0)
HomeContainerFlow2()
.embededInNavigation()
.tabItem { Text("Flow 2") }
.tag(1)
}.onReceive(navigationState.contentViewNavigation.$selectedTab, perform: { output in
print("Value changed: \(output)")
})
}
}
如您所见,我正在定义一个全局状态(“NavigationState”)来处理应用程序的导航(即选项卡选择)。因此,我有一个 ObservableObject (NavigationState),它将具有每个屏幕导航状态的属性(即 ContentViewNavigationState - 另一个 ObservableObject)。
实际行为
在这个例子中,我在 DispatchQueue.asyncAfter 中修改 App.swift 文件中的状态,状态正在改变(因为 ContentView 的 onReceive 修饰符正在显示更新值)但没有触发 ContentView 中的任何更新。
预期行为
应根据状态更改所选选项卡。
- 有趣的是,如果我还修改了另一个属性(即 commented
selectedTab
),在这种情况下会触发更新,但不是仅更新 contentViewNavigation 的属性时。
解决方案
如果要在对象已更改时进行更新,则视图必须显式观察 ObservableObject。
在您的情况下,您观察到可观察但不是内部可观察的父级。您的contentViewNavigation
场景中的 不会自行更改,因为它只是对内部对象的引用。
所以解决方案可能如下演示
struct ContentView: View {
@EnvironmentObject var navigationState: NavigationState
var body: some View {
// separate view which observe inner ObservableObject
MainView(contentNavigation: navigationState.contentViewNavigation)
}
}
struct MainView: View {
@ObservedObject var contentNavigation: ContentViewNavigationState
var body: some View {
TabView(selection: $contentNavigation.selectedTab) {
HomeContainerFlow1()
.tabItem { Text("Flow 1") }
.tag(0)
HomeContainerFlow2()
.embededInNavigation()
.tabItem { Text("Flow 2") }
.tag(1)
}
}
}
推荐阅读
- excel - 更改 Access 表中的一个单元格
- html - 如何使用css在我的垂直导航栏中的联系人下添加底部边框?
- c++ - c++科学记数法,改变指数的位数
- python - 最长的字母子串不返回最后一个索引字符
- java - 如何在 Spring 项目中创建包含 main 方法的类的实例
- c# - 尝试围绕固定点旋转对象
- python-3.x - 安装 Tensorflow 2 在 pywrap_tensorflow.py 中加载 dll 失败
- android - 如何在 Android Kotlin 上充气动态布局?
- r - 删除重复的行并将具有特定模式的行保留在另一列中
- javascript - 为什么我从 Javascript 中的日期开始计算小时数是错误的?