25

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

我在 Xcode 11 中创建了一个简单的独立 Watch 应用程序。

我创建了一个新DataModel课程

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 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>在这种情况下。

类似的方法UIHostingController在 iOS 应用程序中非常有效,因为UIHostingController它不是泛型类。

还有其他方法可以将环境注入到 watchOS 视图中吗?

4

3 回答 3

54

AnyView在 SwiftUI 的情况下,您可以使用类型擦除View

我会重构WKHostingController为 return AnyView

这似乎对我来说编译得很好。

class HostingController : WKHostingController<AnyView> {
    override var body: AnyView {
        return AnyView(ContentView().environmentObject(DataModel()))
    }
}
于 2019-06-12T06:28:26.020 回答
2

对于像布雷特(在评论中)这样的人来说

"Property 'body' with type 'AnyView' cannot override a property with type 'ContentView'"

我得到了同样的错误,因为我没有替换返回值并包装了返回的 ContentView。

IE。这是我第一次尝试的样子..注意 WKHostingController<ContentView> 应该是 WKHostingController<AnyView>

class HostingController : WKHostingController<ContentView> {
    override var body: AnyView {
        return AnyView(ContentView().environmentObject(DataModel()))
    }
}
于 2019-09-14T22:36:00.303 回答
0

添加到马特奥的真棒答案,

如果你想使用委托,那么使用这样的:

class HostingController : WKHostingController<AnyView> {
    override var body: AnyView {
        var contentView = ContentView()
        contentView.environmentObject(DataModel())
        contentView.delegate = self
        let contentWrapperView = AnyView(contentView)
        return contentWrapperView
    }
}
于 2020-04-08T20:45:53.403 回答