我正在使用TornadoFX 1.7.5,但我似乎无法使绑定属性工作。我有以下项目视图模型
class DynamicMenuViewModel : ItemViewModel<DynamicMenu>(DynamicMenu()) {
val title = bind { item?.title?.toProperty() }
val isBold = bind { item?.isBold?.toProperty() }
val routes = bind { item?.routes?.toProperty() }
}
data class DynamicMenu(var title: String = "", var isBold: Boolean = false, var routes: MutableList<MenuRouteViewModel> = mutableListOf())
class MenuRouteViewModel : ItemViewModel<MenuRoute>(MenuRoute()) {
val url = bind { item?.url?.toProperty() }
val title = bind { item?.title?.toProperty() }
val isBold = bind { item?.isBold?.toProperty() }
val showNew = bind { item?.showNew?.toProperty() }
}
data class MenuRoute(var url: String = "", var title: String = "", var showNew: Boolean = false, var isBold: Boolean = false)
绑定如下:
//routesController.dynamicMenu is an instance of DynamicMenuViewModel()
textfield(property = routesController.dynamicMenu.title) {
prefWidth = formWidth * .5
gridpaneConstraints {
columnRowIndex(0, 1)
marginLeft = 10.0
columnSpan = 2
marginBottom = 20.0
}
}
checkbox(property = routesController.dynamicMenu.isBold){
gridpaneConstraints {
columnRowIndex(2, 1)
marginLeft = 15.0
marginBottom = 20.0
}
}
然后,以下函数提交模型并在我单击按钮时将它们打印到屏幕上:
fun onClick(){
commitModel()
println(dynamicMenu.item.toString())
dynamicMenu.item.routes.forEach {
println(it.item.toString())
}
}
fun commitModel(){
dynamicMenu.item.routes.forEach {
it.commit()
}
dynamicMenu.commit()
}
问题是,当我运行程序并编辑文本字段和复选框,然后单击运行onClick()
的按钮时,支持项似乎没有更新。因此,不会将任何更新的值打印到控制台。
我在这里做错了什么?
您可能知道,ViewModel 只能与 JavaFX 属性进行双向绑定。您的域对象不包含 JavaFX 属性,因此您需要转换它们。但是,您使用的toProperty()
函数仅对值进行操作,并将其转换为属性。此属性无法知道其字段所有者,因此无法写回域对象。
幸运的是,您可以使用observable
函数使域对象属性也可写:
val url = bind { item?.observable(MenuRoute::url) }
由于observable
函数对MenuRoute
对象的特定实例进行操作,因此当您commit()
模型时,它现在有足够的信息写回该实例。
如果您考虑将域对象中的属性更改为可观察的,则可以编写:
val url = bind(MenuRoute::url)
您可以使用TornadoFX IDEA插件检查"将所有属性转换为TornadoFX属性"来自动返工您的属性。这会将MenuRoute
对象转换为:
class MenuRoute {
val isBoldProperty = SimpleBooleanProperty(false)
var isBold by isBoldProperty
val showNewProperty = SimpleBooleanProperty(false)
var showNew by showNewProperty
val urlProperty = SimpleStringProperty("")
var url by urlProperty
val titleProperty = SimpleStringProperty("")
var title by titleProperty
}
(您必须手动删除类上的data
修饰符。还要注意,当前版本的插件在转换函数中有一个错误,会保留旧参数 - 新版本将很快发布)。
如果您出于各种原因不想这样做,即使对于像您这样的可变变量,我也只能支持这种不错的语法,因此从 TornadoFX 1.7.6 开始,即使您不想更改数据类,您也可以在绑定语句中使用此语法:
val url = bind(MenuRoute::url)