繁体   English   中英

无法转换“已发布”类型的值<bool> .Publisher'到预期的参数类型'绑定<bool> '</bool></bool>

[英]Cannot convert value of type 'Published<Bool>.Publisher' to expected argument type 'Binding<Bool>'

尝试编译以下代码时:

class LoginViewModel: ObservableObject, Identifiable {
    @Published var mailAdress: String = ""
    @Published var password: String = ""
    @Published var showRegister = false
    @Published var showPasswordReset = false

    private let applicationStore: ApplicationStore

    init(applicationStore: ApplicationStore) {
        self.applicationStore = applicationStore
    }

    var passwordResetView: some View {
        PasswordResetView(isPresented: $showPasswordReset) // This is where the error happens
    }
}

PasswordResetView 如下所示:

struct PasswordResetView: View {
    @Binding var isPresented: Bool
    @State var mailAddress: String = ""
    
    var body: some View {
            EmptyView()
        }
    }
}

我收到错误编译错误

Cannot convert value of type 'Published<Bool>.Publisher' to expected argument type 'Binding<Bool>'

如果我使用 LoginViewModel class 外部的已发布变量,它就可以正常工作:

struct LoginView: View {
    @ObservedObject var viewModel: LoginViewModel

    init(viewModel: LoginViewModel) {
      self.viewModel = viewModel
    }
    
    var body: some View {
            PasswordResetView(isPresented: self.$viewModel.showPasswordReset)
    }
}

有什么建议我在这里做错了吗? 我有没有机会从拥有的 class 内部将已发布的变量作为绑定传递?

谢谢!

** 组合和 SwiftUI 仍然是新手,所以不确定是否有更好的方法来接近 **

您可以从发布者初始化绑定。

https://developer.apple.com/documentation/swiftui/binding/init(get:set:)-6g3d5

let binding = Binding(
    get: { [weak self] in
        (self?.showPasswordReset ?? false)
    },
    set: { [weak self] in
        self?.showPasswordReset = $0
    }
)

PasswordResetView(isPresented: binding)

这是可能的方法 - 在生成的视图中进行可能的观察并避免工厂和演示者之间的紧密耦合的想法。

使用 Xcode 12 / iOS 14 测试(对于旧系统,可能需要进行一些调整)

protocol ResetViewModel {
    var showPasswordReset: Bool { get set }
}

struct PasswordResetView<Model: ResetViewModel & ObservableObject>: View {
    @ObservedObject var resetModel: Model

    var body: some View {
        if resetModel.showPasswordReset {
            Text("Show password reset")
        } else {
            Text("Show something else")
        }
    }
}

class LoginViewModel: ObservableObject, Identifiable, ResetViewModel {
    @Published var mailAdress: String = ""
    @Published var password: String = ""
    @Published var showRegister = false
    @Published var showPasswordReset = false

    private let applicationStore: ApplicationStore

    init(applicationStore: ApplicationStore) {
        self.applicationStore = applicationStore
    }

    var passwordResetView: some View {
        PasswordResetView(resetModel: self)
    }
}

我认为这里要理解的重要一点是“$”在组合上下文中的作用。

"$" 所做的是将变量 "showPasswordReset" 的变化发布在被观察的地方。

当它在类型之前时,它不代表您为变量声明的类型(在这种情况下为布尔值),它代表发布者,如果您想要类型的值,只需删除“$”。

"$" 用于将变量标记为@ObservedObject 的上下文中,(此处的 ObservableObject 是 LoginViewModel,您订阅它以作为发布者侦听其变量市场的变化)

struct ContentView: View {
       @ObservedObject var loginViewModel: LoginViewModel...

在这种情况下(例如 ContentView),“showPasswordReset”的更改将在其值更新时“发布”,因此视图会使用新值更新。

对于声明:“无法将'Binding'类型的值转换为预期的参数类型'Bool'”的错误,解决方案是使用wrappedValue,如下例所示。

如果您有带有属性 isEnabled 的 MyObject object 并且您需要将其用作 vanilla Bool 类型而不是 'Binding' 然后执行 myView.disabled($myObject.isEnabled.wrappedValue)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM