1

我最近遇到了编写 a 的 Mock 的需要Class,因为它会导致 SwiftUIpreview无法工作。不幸的是,我得到了错误:

Property type 'T' does not match that of the 'wrappedValue' property of its wrapper type 'EnvironmentObject'

在视图结构中:

struct ContentView<T>: View {
    @EnvironmentObject var mockFoobar: T
    ...
}

还有错误:

Type of expression is ambiguous without more context

对于 Preview 结构:

struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        let mockFoobar: MockFoobar = MockFoobar()
        return ContentView<MockFoobar>()
            .environmentObject(mockFoobar)
    }
}

MockFoobar 类是:

class MockFoobar: ObservableObject {
  ...
}

正如用户@Aspergi 所提供的,按照建议测试了以下内容:

class Foobar: ObservableObject {
    @Published var param: Bool = false
    func start() {
        self.param = true
    }
}

struct MyFoobarView<T: ObservableObject>: View {
    @EnvironmentObject var foobar: T

    var body: some View {
        VStack {
            Text("Hello Foobar")
        }
        .onAppear {
            self.foobar.start()
        }
    }
}

struct MyFoobarView_Previews: PreviewProvider {
    static var previews: some View {
        let foobar: Foobar = Foobar()
        return MyFoobarView()
            .environmentObject(foobar)
    }
}

但我收到以下错误(.onAppearPreviewProvider 中的第一个和第二个):

Cannot call value of non-function type 'Binding<Subject>'

Generic parameter 'T' could not be inferred
4

1 回答 1

4

EnvironmentObject 必须是 ObservableObject,所以这里是修复

struct ContentView<T: ObservableObject>: View {
    @EnvironmentObject var mockFoobar: T

    // .. other code here

更新:添加了引入模型协议的演示

protocol Foobaring {
    var param: Bool { get set }
    func start()
}

class Foobar: ObservableObject, Foobaring {
    @Published var param: Bool = false
    func start() {
        self.param = true
    }
}

struct MyFoobarView<T: ObservableObject & Foobaring>: View {
    @EnvironmentObject var foobar: T

    var body: some View {
        VStack {
            Text("Hello Foobar")
        }
        .onAppear {
            self.foobar.start()
        }
    }
}

struct MyFoobarView_Previews: PreviewProvider {
    static var previews: some View {
        let foobar: Foobar = Foobar()
        return MyFoobarView<Foobar>()
            .environmentObject(foobar)
    }
}
于 2020-05-23T13:31:38.213 回答