使用 @ObservedObject 时是否可以在 Swift 中嵌套属性包装器?

Nic*_*oft 2 swift swiftui combine property-wrapper

我最近开始深入研究 SwiftUI、Combine 和属性包装器的奇妙世界,并且正在努力将 @ObservedObject 与我编写的 @Injected 属性包装器结合起来,以将依赖项注入到我的视图中。大多数时候,我的 @Injected 包装器工作正常,但是当与 @ObservedObject 配合使用来管理我的视图模型时,我收到“属性类型与‘wrappedValue’属性的类型不匹配”错误。

目前我的 @Injected 属性包装器如下所示:

@propertyWrapper
public struct Injected<Service> {

    private var service: Service

    public init() {
        self.service = assembler.resolver.resolve(Service.self)!
    }

    public init(name: String? = nil, container: Resolver? = nil) {
        // `assembler` here referring to my global Swinject assembler
        self.service = container?.resolve(Service.self, name: name) ??
            assembler.resolver.resolve(Service.self, name: name)!
    }

    public var wrappedValue: Service {
        get {
            return service
        }

        mutating set {
            service = newValue
        }
    }

    public var projectedValue: Injected<Service> {
        get {
            return self
        }

        mutating set {
            self = newValue
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

这是我当前的用法:

struct MyModalView: View {
    
    @ObservedObject @Injected var viewModel: MyModalViewModel
    
    var body: some View {
        Text("Hello World")
    }
}
Run Code Online (Sandbox Code Playgroud)

以这种方式排序包装器,我收到:“属性类型‘MyModalViewModel’与其包装器类型‘ObservedObject’的‘wrappedValue’属性不匹配”,而MyModalViewModel类确实从ObservableObject扩展。

如果我翻转包装器,它会编译,但 Swinject 尝试解析包装的 ObservedObject 类,并且由于容器只是注册原始 MyModalViewModel 类,因此此解析失败并且应用程序崩溃。

同时,通过直接赋值来分配@ObservedObject值是可行的:

@ObservedObject var viewModel: MyModalViewModel = assembler.resolver.resolve(MyModalViewModel.self)!
Run Code Online (Sandbox Code Playgroud)

我认为原始代码应该编译,看到 @Injected 将返回一个符合 ObservableObject 的包装值,就像 @ObservedObject 期望的那样,尽管所有这些对我来说仍然相当新,所以可能有一些我遗漏的东西。如有任何意见,我们将不胜感激。谢谢!!

Nic*_*oft 5

对于将来遇到此问题的任何人,我最终将两个属性包装器合并在一个屋檐下,借用了Resolver项目的实现:

@propertyWrapper
public struct InjectedObject<Service>: DynamicProperty where Service: ObservableObject {
 
    @ObservedObject private var service: Service
    
    public init() {
        self.service = assembler.resolver.resolve(Service.self)!
    }
    
    public init(name: String? = nil, container: Resolver? = nil) {
        self.service = container?.resolve(Service.self, name: name) ??
            assembler.resolver.resolve(Service.self, name: name)!
    }
    
    public var wrappedValue: Service {
        get { return service }
        mutating set { service = newValue }
    }
    
    public var projectedValue: ObservedObject<Service>.Wrapper {
        return self.$service
    }
}
Run Code Online (Sandbox Code Playgroud)