我目前正在为使用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中显示这些值。
我得到错误消息变量'self.xxx'在初始化之前用于我的输出中的buttonClicked声明。我得到的是我需要在使用self之前初始化每个非可选属性但我认为可能使用警卫就可以了。显然不是...
如何在不进行属性输入和输出选项的情况下解决此问题?
谢谢
好吧,我把我的属性输入和输出作为懒惰,它工作。
static let errorAlertProperties = AlertProperties(title: "Fail",
message: "!!!",
buttonTitle: "Ok",
handler: nil)
lazy var input = Input(slider: slider.asObserver(),
buttonCLicked: buttonCLicked.asObserver(),
resetCLicked: resetClicked.asObserver())
lazy var 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 GameViewModel.errorAlertProperties
}
return self.getAlertProperties() })
我不知道这是否是“正确”的方式,所以如果你有更好的方法,请告诉我。
谢谢