我有以下代码:
class ExampleView :View("My Example view") {
val model:ExampleModel by inject()
override val root= vbox {
textfield(model.data)
button("Commit") {
setOnAction {
model.commit()
closeModal()
}
}
button("Rollback") {
setOnAction {
model.rollback()
closeModal()
}
}
button("Just quit") {
setOnAction {
closeModal()
}
}
}
}
class Example() {
var data by property<String>()
fun dataProperty() = getProperty(Example::data)
}
class ExampleModel(example: Example) : ItemViewModel<Example>() {
init {
item = example
}
val data = bind { item?.dataProperty() }
}
class MainView : View() {
val example:Example
override val root = BorderPane()
init {
example = Example()
example.data = "Data for example"
val exampleModel = ExampleModel(example)
with(root){
top {
menubar {
menu("Test") {
menuitem("Example - 1") {
val scope = Scope()
setInScope(exampleModel, scope)
find<ExampleView>(scope).openWindow()
}
menuitem("Example - 2") {
val scope = Scope()
setInScope(exampleModel, scope)
find<ExampleView>(scope).openWindow()
}
}
}
}
}
}
}
这个例子我有两个问题:
1)如果我更改了值并在没有提交的情况下关闭窗口(用户可以使用帮助[X]按钮执行此操作),则只有ViewModel将存储更改(即使在重新打开后它也会显示在GUI中) ,但模型POJO对象将保留旧数据。
如果我使用了Example类的实例(没有DI),则此实例立即收到所有更改。
例如,我不想提交/回滚功能,但我想要DI并立即更新。我该做什么? (ofcource我可以打电话给#34;提交&#34;用于&#34;文本字段更改值事件&#34;)
2)ViewModel有带参数的构造函数,如果我尝试像这样打开ExampleView
find<ExampleView>(Scope()).openWindow()
然后我得到了一个明显的RuntimeException。我可以通过编译器警告(或其他东西)来避免这种情况吗?
答案 0 :(得分:1)
1)这是ViewModel的正确默认行为。如果将视图模型的属性绑定到输入,则更改会立即反映在该绑定属性中,但只有在提交后才会刷新到基础模型对象中。
如果要将视图模型属性中的更改自动提交回基础模型对象,则可以创建绑定,并将autocommit
属性设置为true:
val data = bind(true) { item?.dataProperty() }
如果您看起来更清楚,也可以写bind(autocommit = true)
。这将导致任何更改自动刷回底层对象。
我还想让您意识到,通过在视图模型的构造函数中要求一个项目,您可以有效地防止它与注入一起使用,除非您像使用setInScope
一样使用它。这可能适用于您的用例,但值得注意。
2)如果您忘记传递参数,即将推出的TornadoFX 1.5.10将为您提供更好的运行时错误消息。它还引入了参数的默认值。有关详细信息,请参阅https://github.com/edvin/tornadofx/pull/227。