JavaFx: странное исключение TreeTableView NullpointerException

Я заметил странный NPE в TreeTableView

Это происходит, если вы дважды щелкаете в таблице, где нет строки. Я приложу скриншот, чтобы показать, что именно я имею в виду.

Я пришел исключительно из JavaFx, это не имеет ничего общего с моим кодом. Я также провел исследование и не нашел ничего, связанного с этим.

Вот простой код для его воспроизведения:

Контроллер:

public class Controller implements Initializable {
    @FXML
    private TreeTableColumn<Model, String> column;
    @FXML
    private TreeTableView<Model> treeTable;

    @Override
    public void initialize(URL location, ResourceBundle resources) {
        TreeItem<Model> root = new TreeItem<>(new Model("Root"));
        treeTable.setRoot(root);
        column.setCellValueFactory(value -> value.getValue().getValue().textProperty());

    }

    private class Model {
        private StringProperty text;

        Model(String text) {
            this.text = new SimpleStringProperty(text);
        }

        public StringProperty textProperty() {
            return text;
        }

        public String getText() {
            return text.get();
        }

        @Override
        public String toString() {
            return getText();
        }
    }
}

.fxml:

<?xml version = "1.0" encoding = "UTF-8"?>

<?import javafx.scene.control.TreeTableColumn?>
<?import javafx.scene.control.TreeTableView?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane xmlns = "http://javafx.com/javafx"
            xmlns:fx = "http://javafx.com/fxml"
            fx:controller = "stackoverflow.treeview.Controller">
    <TreeTableView fx:id = "treeTable">
        <columns>
            <TreeTableColumn fx:id = "column" text = "Test"/>
        </columns>
    </TreeTableView>
</AnchorPane>

Если вы дважды щелкнете в красной области, будет выброшено NPE.

JavaFx: странное исключение TreeTableView NullpointerException

Трассировки стека:

Exception in thread "JavaFX Application Thread" java.lang.NullPointerException
    at com.sun.javafx.scene.control.behavior.TreeTableRowBehavior.handleClicks(TreeTableRowBehavior.java:89)
    at com.sun.javafx.scene.control.behavior.CellBehaviorBase.simpleSelect(CellBehaviorBase.java:259)
    at com.sun.javafx.scene.control.behavior.TableRowBehaviorBase.doSelect(TableRowBehaviorBase.java:120)
    at com.sun.javafx.scene.control.behavior.CellBehaviorBase.mousePressed(CellBehaviorBase.java:150)
    at com.sun.javafx.scene.control.behavior.TableRowBehaviorBase.mousePressed(TableRowBehaviorBase.java:64)
    at com.sun.javafx.scene.control.skin.BehaviorSkinBase$1.handle(BehaviorSkinBase.java:95)
    at com.sun.javafx.scene.control.skin.BehaviorSkinBase$1.handle(BehaviorSkinBase.java:89)
    at com.sun.javafx.event.CompositeEventHandler$NormalEventHandlerRecord.handleBubblingEvent(CompositeEventHandler.java:218)
    at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:80)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
    at com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
    at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
    at javafx.event.Event.fireEvent(Event.java:198)
    at javafx.scene.Scene$MouseHandler.process(Scene.java:3757)
    at javafx.scene.Scene$MouseHandler.access$1500(Scene.java:3485)
    at javafx.scene.Scene.impl_processMouseEvent(Scene.java:1762)
    at javafx.scene.Scene$ScenePeerListener.mouseEvent(Scene.java:2494)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:381)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler$MouseEventNotification.run(GlassViewEventHandler.java:295)
    at java.security.AccessController.doPrivileged(Native Method)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler.lambda$handleMouseEvent$354(GlassViewEventHandler.java:417)
    at com.sun.javafx.tk.quantum.QuantumToolkit.runWithoutRenderLock(QuantumToolkit.java:389)
    at com.sun.javafx.tk.quantum.GlassViewEventHandler.handleMouseEvent(GlassViewEventHandler.java:416)
    at com.sun.glass.ui.View.handleMouseEvent(View.java:555)
    at com.sun.glass.ui.View.notifyMouse(View.java:937)
    at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at com.sun.glass.ui.win.WinApplication.lambda$null$148(WinApplication.java:191)
    at java.lang.Thread.run(Thread.java:745)

Java-версия: 1.8u121

В качестве решения я бы принял любой обходной путь, который действительно работает, или любое объяснение / отчет об ошибке javafx, который показывает, что они работают над этим, или в следующей версии есть исправление.

Что ж, довольно плохим решением было бы просто поймать исключение и обработать его.

GamingFelix 05.04.2019 10:00

это ошибка в treeTableRowBehavior.handleClicks (видимая в fx11) - код не может справиться с пустыми строками.. вы можете сообщить об этом :)

kleopatra 05.04.2019 11:50

в качестве обходного пути вы можете попробовать зарегистрировать eventFilter в пользовательской строке, которая использует событие, если это двойной клик.

kleopatra 05.04.2019 11:52

@kleopatra Я пробовал, но тот же NPE. Вы имеете в виду так: pastebin.com/TvT29PPU ?

Sunflame 05.04.2019 11:57

Обратите внимание, что вы можете отправить отчет об ошибке здесь.

Slaw 05.04.2019 12:07

вам нужно перехватить нажатое, как показано в его ответе. Полезно читать источники - просто говорю :)

kleopatra 05.04.2019 12:07
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
2
6
323
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Когда вы щелкаете в пустом месте, где нет столбцов, вы в конечном итоге нажимаете на TreeTableRow. По-видимому, в классе поведения элемента управления есть ошибка, из-за которой он не может обрабатывать строки с null элементами. Вы не увидите эту ошибку, если щелкнете в пустом месте колонны, потому что это щелкнет на TreeTableCell, чей класс поведения, похоже, не страдает от той же проблемы. Не уверен, в чем точная разница, но я только бегло взглянул на исходный код.

Как упомянул @kleopatra, один из способов остановить выдачу исключения — добавить фильтр событий в TreeTableRow, который потребляет двойные щелчки по пустым строкам.

var table = new TreeTableView<String>();
table.setRowFactory(ttv -> {
    var row = new TreeTableRow<String>();
    row.addEventFilter(MouseEvent.MOUSE_PRESSED, event -> {
        if (row.getItem() == null && event.getClickCount() % 2 == 0) {
            event.consume();
        }
    });
    return row;
});

Примечание. Вам нужно clickCount % 2 == 0, потому что класс поведения попытается получить доступ к TreeItem, когда количество кликов будет четным.

Вау, спасибо и за объяснение, и за ответ :) Не могли бы вы предложить сообщить об этом?

Sunflame 05.04.2019 13:01

да, конечно - баги, о которых команда разработчиков не знает, редко исправляются :)

kleopatra 05.04.2019 14:28

Кстати, у этого хака есть недостаток: из-за того, что он является фильтром, двойное нажатие никогда не достигнет пустой treeTableCell, поэтому реализации пользовательских ячеек не могут делать ничего необычного с пустыми ячейками. Я предполагаю, что это довольно редко, но следует иметь в виду

kleopatra 05.04.2019 14:33

спасибо за сообщение :) bugs.openjdk.java.net/browse/JDK-8222454

kleopatra 15.04.2019 11:42

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