在ObservableObject中引用环境对象

时间:2020-02-01 13:59:10

标签: swiftui combine observableobject

我有一个LoginView,如果用户未登录,则显示一个RegisterView,如果用户未登录,则显示一个ContentView:

struct LoginView: View {
    @EnvironmentObject var userManager: UserManager
    var body: some View {
        Group {
            if userManager.isRegistered {
                ContentView()
            } else {
                RegisterView()
            }
        }
    }
}

ContentView具有三个ObservedObject属性,该属性使用Combine来从具有剩余api的服务器中获取内容。

struct ContentView: View {
    @EnvironmentObject var userManager: UserManager
    @ObservedObject var usersStore = UsersStore()
    @ObservedObject var rolesStore = RolesStore()
    @ObservedObject var elementsStore = ElementsStore()

    var body: some View {
        NavigationView {
            ZStack {
                Image("stell")
                    .resizable()
                    .aspectRatio(contentMode: .fit)
                    .opacity(0.1)

                VStack(alignment: .leading, spacing: 5) {
                    NavigationLink(destination: UsersView(usersStore: usersStore) ) {
                        Text("Users")
                    }
                    NavigationLink(destination: RolesView(rolesStore: rolesStore)) {
                        Text("Roles")
                    }
                    NavigationLink(destination: ElementsView(elements: $elementsStore.elements)) {
                        Text("Elements")
                    }
                }.font(.title).padding(20)
            }.navigationBarTitle(Text("STELL"))
        }
    }
}

我遇到的问题是我想从任何观察到的对象中引用userManager,例如当会话令牌到期时,其余api返回401 Unauthorized时。然后,我希望ObservedObject将isRegistered中的userManager标志设置为false,以便自动显示RegisterView。但是我该怎么做呢?我无法在任何ObservedObject属性初始化程序中设置对userManager的引用,因为编译器抱怨属性初始化程序在self可用之前就运​​行了。

1 个答案:

答案 0 :(得分:2)

在这种情况下,我将通过构造函数使用依赖项注入...下面是在UsersStore示例中展示可能的方法,对于其他示例,则是相同的

UsersStore

中的更改
class UsersStore: ObservableObject {
    var manager: UserManager

    init(manager: UserManager) { // << inject UserManager via constructor
        self.manager = manager 
    }
    ...
}

ContentView

中的更改
struct ContentView: View {
    @EnvironmentObject var userManager: UserManager
    @ObservedObject var usersStore: UsersStore

    init(usersStore: UsersStore) {
        self.usersStore = usersStore // << inject UsersStore via contructor
    }

用法更改

struct LoginView: View {
    @EnvironmentObject var userManager: UserManager
    var body: some View {
        Group {
            if userManager.isRegistered {
                // userManager is valid here some UsersStore can be created
                ContentView(usersStore: UsersStore(manager: self.userManager))