我目前正在为用 Swift 编写的新应用程序实现 MVVM(使用 Rx)架构,并且在编写 ViewModel 时遇到了困难。
这是它的开始:
class GameViewModel {
public let input: Input
public let output: Output
public struct Input {
let slider: AnyObserver<Float>
let buttonCLicked: AnyObserver<Void>
}
public struct Output {
let slider: Observable<Float>
let target: Observable<Int>
let score: Observable<Int>
let round: Observable<Int>
let buttonCLicked: Observable<AlertProperties>
}
private var slider = BehaviorSubject<Float>(value: 0)
private var buttonCLicked = PublishSubject<Void>()
private var target = BehaviorSubject<Int>(value: 0)
private var score = BehaviorSubject<Int>(value: 0)
private var round = BehaviorSubject<Int>(value: 0)
private let disposebag: DisposeBag
private var model: Game!
init(game: Game?) {
disposebag = DisposeBag()
input = Input(slider: slider.asObserver(),
buttonCLicked: buttonCLicked.asObserver())
output = Output(slider: slider.asObservable(),
target: target.asObservable(),
score: score.asObservable(),
round: round.asObservable(),
buttonCLicked: buttonCLicked.map { [weak self] _ in
guard let self = self else {
return AlertProperties(title: "Fail",
message: "!!!",
buttonTitle: "Ok",
handler: nil)
}
return self.getAlertProperties() })
model = game ?? initDefaultGame()
slider.onNext(Float(model.slider))
target.onNext(model.target)
score.onNext(model.score)
round.onNext(model.round)
}
这是一个基本函数,它采用当前值来创建要返回的对象:
func getAlertProperties() -> AlertProperties {
guard let targetValue = try? target.value(),
let currentValue = try? slider.value() else {
return AlertProperties(title: "Fail",
message: "!!!",
buttonTitle: "Ok",
handler: nil)
}
// some code
return AlertProperties(title: title,
message: "\(points) points: \(castCurrentValue)",
buttonTitle: "Ok",
handler: {
self.target.onNext(self.getRandomValue())
self.slider.onNext(50.0)
})
}
我在这里要做的是在我的 init 函数上注册我的输入和输出。对于输出部分,我想用一些当前值(getAlertProperties)传输函数的结果。基本上我的 ViewController 上有一个触发器,当用户单击一个简单的按钮时,它需要这些值显示在 AlertController 中。
对于输出中的 buttonClicked 声明,我收到错误消息Variable 'self.xxx' used before being initialized"。我知道我需要在使用self之前初始化每个非可选属性,但我认为也许使用警卫可以解决问题。显然不是...
如何在不使我的属性输入和输出选项的情况下解决此问题?
谢谢