在 watchOS 中使用 environmentObject

Using environmentObject in watchOS

我正尝试在 watchOS6 应用程序中使用 environmentObject 将我的数据模型绑定到我的视图。

我在 Xcode 11.

中创建了一个简单的独立 Watch 应用程序

我新建了一个DataModelclass

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
    }
}