Core Data 和 SwiftUI 多态问题

Pav*_*lov 7 swift swiftui

将 SwiftUI 和泛型类型放在一起处理核心数据时遇到了麻烦。

考虑以下示例:

在此处输入图片说明

Parent是抽象的。FooBarParent他们的孩子,他们有一些自定义属性。

现在我想做的,大致是:

protocol EntityWithView {
    associatedtype T: View
    func buildView() -> T
}

extension Parent: EntityWithView {
    func buildView() -> some View {
        fatalError("Re-implement in child")
    }
}

extension Foo {
    override func buildView() -> some View {
        return Text(footribute)
    }
}

extension Bar {
    override func buildView() -> some View {
        return Text(atrribar)
    }
}

struct ViewThatUsesCoreDataAsModel: View {
    let entities: [Parent]

    var body: some View {
        ForEach(entities) { entity in
            entity.buildView()
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我想将多态构建器添加到塑造数据或构建视图的核心数据实体中,以确认通用接口,以便我可以在不进行强制转换/键入的情况下使用它们。

如果我尝试直接修改生成的 Core 数据实体而不是通过扩展,编译器会抛出错误的问题,并确认协议虽然扩展不允许覆盖。

Asp*_*eri 1

好吧,这令人头疼(至少对于预览版来说,它变得疯狂),但它在运行时有效。使用 Xcode 11.4 / iOS 13.4 进行测试。

由于我们需要在扩展中完成所有工作,因此想法是通过 Obj-C 消息传递使用调度,这是在此类要求下实际可用的覆盖实现的通道。

注意:使用模拟器或设备

演示

完整的测试模块

protocol EntityWithView {
    associatedtype T: View
    var buildView: T { get }
}

extension Parent {
    // allows to use Objective-C run-time messaging by complete
    // type erasing.
    // By convention subclasses
    @objc func generateView() -> Any {
        AnyView(EmptyView()) // << safe
        //fatalError("stub in base") // << alternate
    }
}

extension Parent: EntityWithView {
    var buildView: some View {
        // restory SwiftUI view type from dispatched message
        guard let view = self.generateView() as? AnyView else {
            fatalError("Dev error - subview must generate AnyView")
        }
        return view
    }
}

extension Foo {
    @objc override func generateView() -> Any {
        AnyView(Text(footribute ?? ""))
    }
}

extension Bar {
    @objc override func generateView() -> Any {
        AnyView(Text(attribar ?? ""))
    }
}

struct ViewThatUsesCoreDataAsModel: View {
    let entities: [Parent]

    var body: some View {
        VStack {
            ForEach(entities, id: \.self) { entity in
                entity.buildView
            }
        }
    }
}

struct DemoGeneratingViewInCoreDataExtension: View {
    @Environment(\.managedObjectContext) var context
    var body: some View {
        ViewThatUsesCoreDataAsModel(entities: [
           Foo(context: context), 
           Bar(context: context)
        ])
    }
}
Run Code Online (Sandbox Code Playgroud)