Tornadofx просмотр таблицы с использованием comboBox и добавление FXEvent на кнопки
Мне нужна помощь по следующим основным вопросам:
Требование:
У меня есть табличное представление, которое может быть отредактировано (встроено), и строки также могут быть удалены, а также при необходимости некоторые сложные операции по нажатию кнопки.
Ниже приведены 3 небольших вопроса:
- Я создал табличное представление с isEditable = true и сделал имя столбца редактируемым. Но когда я редактирую, он не привязывается к модели. (Должно быть, не хватает очень простой вещи)
- При редактировании comboBox показывает опции, но при выборе значения выдает исключение.
java.lang.ClassCastException: javafx.beans.property.SimpleStringProperty не может быть приведен к javafx.beans.property.ObjectProperty
Затем я добавил кнопку удаления, которая прекрасно работает, когда я раскомментирую tableView.items.removeAt(index)
Но так как мне нужны некоторые дополнительные функции, я решил пойти с огнем FXEvent. Но как я должен использовать это здесь.
class MyView : View() {
val warriorModel : WarriorModel by inject()
val persons = FXCollections.observableArrayList<Warrior>(
Warrior(1,"Tyrion Lannister", "M"),
Warrior(2,"Ned Stark", "M"),
Warrior(3,"Daenerys Targaryen", "F"),
Warrior(4,"Arya Stark", "F")
)
override val root = vbox {
tableview(persons) {
isEditable = true
column("ID", Warrior::idProperty)
column("Name", Warrior::nameProperty).makeEditable()
column("Gender", Warrior::genderProperty).useComboBox(FXCollections.observableArrayList("M", "F"))
column("Action", Warrior::dummyProperty).setCellFactory { DeleteButton<Warrior>() }
bindSelected(warriorModel)
subscribe<DeleteEvent> { event ->
items.removeAt(event.index)
}
}
}
}
class DeleteButton<Warrior>() : TableCell<Warrior, String?>() {
internal val btn = Button("Delete")
override fun updateItem(item: String?, empty: Boolean) {
super.updateItem(item, empty)
if (empty) {
graphic = null
text = null
} else {
btn.setOnAction { event: ActionEvent ->
//tableView.items.removeAt(index)
fire(DeleteEvent(index))
}
graphic = btn
text = null
}
}
}
class Warrior(id: Int, name: String, gender: String) {
val idProperty = SimpleIntegerProperty(id)
var id by idProperty
val nameProperty = SimpleStringProperty(name)
var name by nameProperty
val genderProperty = SimpleStringProperty(gender)
var gender by genderProperty
val dummyProperty = SimpleStringProperty("")
}
class WarriorModel : ItemViewModel<Warrior>() {
val id = bind { item?.idProperty }
val name = bind { item?.nameProperty }
val gender = bind { item?.genderProperty }
}
class DeleteEvent(val index: Int) : FXEvent()
1 ответ
Чтобы получить доступ к EventBus извне Component
, использовать FX.eventbus
переменная, и запустить событие на этом:
FX.eventbus.fire(DeleteEvent(index))
Причиной появления ошибки при изменении значения в выпадающем списке является ошибка в фреймворке. Я только что исправил это, но вы можете обойти его, добавив эту функцию расширения в свое приложение до выпуска 1.5.10:
fun <S, T> TableColumn<S, T?>.useComboBoxWorking(items: ObservableList<T>, afterCommit: ((TableColumn.CellEditEvent<S, T?>) -> Unit)? = null): TableColumn<S, T?> {
cellFactory = ComboBoxTableCell.forTableColumn(items)
setOnEditCommit {
val property = it.tableColumn.getCellObservableValue(it.rowValue) as Property<T?>
property.value = it.newValue
afterCommit?.invoke(it)
}
return this
}
Просто позвони .useComboBoxWorking()
вместо этого пока и ты должен быть золотым:)
Если вы не хотите использовать обходной путь, вы можете временно изменить genderProperty
вашей Warrior
быть типом SimpleObjectProperty<String>
вместо SimpleStringProperty
как альтернатива.