Tornadofx просмотр таблицы с использованием comboBox и добавление FXEvent на кнопки

Мне нужна помощь по следующим основным вопросам:

Требование:
У меня есть табличное представление, которое может быть отредактировано (встроено), и строки также могут быть удалены, а также при необходимости некоторые сложные операции по нажатию кнопки.

Ниже приведены 3 небольших вопроса:

  1. Я создал табличное представление с isEditable = true и сделал имя столбца редактируемым. Но когда я редактирую, он не привязывается к модели. (Должно быть, не хватает очень простой вещи)
  2. При редактировании 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 как альтернатива.

Другие вопросы по тегам