在 watchOS 中使用 environmentObject
Using environmentObject in watchOS
我正尝试在 watchOS6 应用程序中使用 environmentObject
将我的数据模型绑定到我的视图。
我在 Xcode 11.
中创建了一个简单的独立 Watch 应用程序
我新建了一个DataModel
class
import Combine
import Foundation
import SwiftUI
final class DataModel: BindableObject {
let didChange = PassthroughSubject<DataModel,Never>()
var aString: String = "" {
didSet {
didChange.send(self)
}
}
}
在我的 ContentView
结构中,我使用 @EnvironmentObject
-
绑定此 class
struct ContentView : View {
@EnvironmentObject private var dataModel: DataModel
var body: some View {
Text($dataModel.aString.value)
}
}
最后,我尝试将 DataModel
的实例注入到 HostingController
class -
的环境中
class HostingController : WKHostingController<ContentView> {
override var body: ContentView {
return ContentView().environmentObject(DataModel())
}
}
但是,我得到一个错误:
Cannot convert return expression of type '_ModifiedContent<ContentView, _EnvironmentKeyWritingModifier<DataModel?>>' to return type 'ContentView'
错误是因为 WKHostingController
是需要具体类型的泛型 - 在本例中为 WKHostingController<ContentView>
。
类似的方法在 iOS 应用程序中与 UIHostingController
完美配合,因为 UIHostingController
不是通用的 class。
是否有其他方法可以将环境注入 watchOS 视图?
你可以使用类型擦除,AnyView
在 SwiftUI 的情况下 View
。
我会将 WKHostingController
重构为 return AnyView
。
这似乎在我这边编译得很好。
class HostingController : WKHostingController<AnyView> {
override var body: AnyView {
return AnyView(ContentView().environmentObject(DataModel()))
}
}
对于像 Brett(在评论中)这样的人
"Property 'body' with type 'AnyView' cannot override a property with type 'ContentView'"
我遇到了同样的错误,因为我没有替换 return 值并包装正在 returned 的 ContentView。
即。这是我第一次尝试的样子..注意
WKHostingController<ContentView>
那应该是
WKHostingController<AnyView>
class HostingController : WKHostingController<ContentView> {
override var body: AnyView {
return AnyView(ContentView().environmentObject(DataModel()))
}
}
添加到 Matteo 的精彩回答中,
如果你想使用委托,那么使用这样的:
class HostingController : WKHostingController<AnyView> {
override var body: AnyView {
var contentView = ContentView()
contentView.environmentObject(DataModel())
contentView.delegate = self
let contentWrapperView = AnyView(contentView)
return contentWrapperView
}
}
我正尝试在 watchOS6 应用程序中使用 environmentObject
将我的数据模型绑定到我的视图。
我在 Xcode 11.
中创建了一个简单的独立 Watch 应用程序我新建了一个DataModel
class
import Combine
import Foundation
import SwiftUI
final class DataModel: BindableObject {
let didChange = PassthroughSubject<DataModel,Never>()
var aString: String = "" {
didSet {
didChange.send(self)
}
}
}
在我的 ContentView
结构中,我使用 @EnvironmentObject
-
struct ContentView : View {
@EnvironmentObject private var dataModel: DataModel
var body: some View {
Text($dataModel.aString.value)
}
}
最后,我尝试将 DataModel
的实例注入到 HostingController
class -
class HostingController : WKHostingController<ContentView> {
override var body: ContentView {
return ContentView().environmentObject(DataModel())
}
}
但是,我得到一个错误:
Cannot convert return expression of type '_ModifiedContent<ContentView, _EnvironmentKeyWritingModifier<DataModel?>>' to return type 'ContentView'
错误是因为 WKHostingController
是需要具体类型的泛型 - 在本例中为 WKHostingController<ContentView>
。
类似的方法在 iOS 应用程序中与 UIHostingController
完美配合,因为 UIHostingController
不是通用的 class。
是否有其他方法可以将环境注入 watchOS 视图?
你可以使用类型擦除,AnyView
在 SwiftUI 的情况下 View
。
我会将 WKHostingController
重构为 return AnyView
。
这似乎在我这边编译得很好。
class HostingController : WKHostingController<AnyView> {
override var body: AnyView {
return AnyView(ContentView().environmentObject(DataModel()))
}
}
对于像 Brett(在评论中)这样的人
"Property 'body' with type 'AnyView' cannot override a property with type 'ContentView'"
我遇到了同样的错误,因为我没有替换 return 值并包装正在 returned 的 ContentView。
即。这是我第一次尝试的样子..注意
WKHostingController<ContentView>
那应该是
WKHostingController<AnyView>
class HostingController : WKHostingController<ContentView> {
override var body: AnyView {
return AnyView(ContentView().environmentObject(DataModel()))
}
}
添加到 Matteo 的精彩回答中,
如果你想使用委托,那么使用这样的:
class HostingController : WKHostingController<AnyView> {
override var body: AnyView {
var contentView = ContentView()
contentView.environmentObject(DataModel())
contentView.delegate = self
let contentWrapperView = AnyView(contentView)
return contentWrapperView
}
}