- 对于应该与应用程序中的许多视图共享的数据,SwiftUI 为我们提供了 @EnvironmentObject 属性包装器,这让我们可以在任何需要的地方共享模型数据,同时还确保我们的视图在数据更改时自动保持更新。
- 可以把 @EnvironmentObject 看作在许多视图上使用,是一种比 @ObservedObject 更智能、更简单的方式。与其在视图 A 中创建一些数据,然后将其传递给视图 B,然后是视图 C,然后是视图 D 才最终使用它,可以在视图 A 中创建它并将其放入环境中,以便视图 B、C 和 D 将自动访问它。
- 就像 @ObservedObject 一样,永远不会为 @EnvironmentObject 属性赋值。相反,它应该从其他地方传入,最终您可能想要使用 @StateObject 在某处创建它。但是,与 @ObservedObject 不同的是,我们不会手动将对象传递到其它视图中。相反,使用将数据发送到一个名为 environmentObject() 的修饰符,这使得该对象在 SwiftUI 的环境中可用于该视图以及其中的任何其它对象(需要注意:环境对象必须由父视图提供,如果 SwiftUI 找不到正确类型的环境对象,程序就会崩溃,这也适用于预览,所以要小心)、
- 为了演示环境对象,我们将定义三件事:
- 一个 GameSettings 类,其中包含一个名为 score 的已发布属性;
- 期望在环境中接收 GameSettings 对象并显示其 score 属性的 ScoreView 视图;
- 一个创建 GameSettings 对象的 ContentView 视图,有一个按钮可以将 1 添加到它的 score 属性,还有一个 NavigationLink 来显示详细信息视图。
- 如下所示:
// Our observable object class
class GameSettings: ObservableObject {
@Published var score = 0
}
// A view that expects to find a GameSettings object
// in the environment, and shows its score.
struct ScoreView: View {
@EnvironmentObject var settings: GameSettings
var body: some View {
Text("Score: \(settings.score)")
}
}
// A view that creates the GameSettings object,
// and places it into the environment for the
// navigation view.
struct ContentView: View {
@StateObject var settings = GameSettings()
var body: some View {
NavigationView {
VStack {
// A button that writes to the environment settings
Button("Increase Score") {
settings.score += 1
}
NavigationLink(destination: ScoreView()) {
Text("Show Detail View")
}
}
.frame(height: 200)
}
.environmentObject(settings)
}
}
- 分析该代码中几个重要部分:
- 就像 @StateObject 和 @ObservedObject 一样,所有与 @EnvironmentObject 一起使用的类都必须符合ObservableObject 协议;
- 将 GameSettings 放入导航视图的环境中,这意味着导航视图中的所有视图都可以根据需要读取该对象,以及导航视图显示的任何视图;
- 当使用 @EnvironmentObject 属性包装器时,声明希望接收的事物类型,但并没有创建它,毕竟我们希望从环境中接收它;
- 因为我们的细节视图显示在导航视图中,所以它可以访问相同的环境,这意味着它可以读取我们创建的 GameSettings 对象;
- 不需要将环境中的 GameSettings 实例与 ScoreView 中的 settings 属性显式关联,SwiftUI 会自动确定它在环境中具有一个 GameSettings 实例,因此它就是它使用的实例。
- 既然我们的视图依赖于存在的环境对象,那么更新预览代码以提供一些要使用的示例设置也很重要。 例如,使用 ScoreView().environmentObject(GameSettings()) 之类的东西进行预览应该这样做。
- 如果需要向环境添加多个对象,应该添加多个 environmentObject() 修饰符,只需一个接一个地调用它们。