Прокрутка JavaFX началась и закончилась

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

pane.setOnScroll({myMethod()}).

Проблема в том, что если вы прокручиваете много, он вычисляет все много раз. Поэтому я хочу выполнять свои действия только после завершения прокрутки. Я надеялся использовать setOnScrollStartedсохранить начальное значение и setOnScrollFinished делать мои действия.

Но я не знаю, почему эти два метода никогда не вызываются. В качестве теста я использовал

pane.setOnScroll({System.out.println("proof of action"});

и это явно никогда не называлось.

Любая идея о том, как вызвать мой метод только в конце прокрутки?

Заранее спасибо

1 ответ

Решение

От JavadocScrollEvent (выделение мое):

Когда прокрутка производится жестом касания (например, перетаскиванием пальца по сенсорному экрану), она окружается событиями SCROLL_STARTED и SCROLL_FINISHED. Изменение количества задействованных точек касания во время прокрутки считается новым жестом, поэтому пара уведомлений SCROLL_FINISHED и SCROLL_STARTED доставляется каждый раз при изменении TouchCount. Когда прокрутка вызвана вращением колеса мыши, доставляется только одноразовое событие SCROLL без начального / завершенного окружения.

Возможный обходной путь:

Увеличивайте значение счетчика каждый раз при обнаружении прокрутки. В слушателе запустите новый поток, который ждет 1 секунду и выполняет действие, которое вы хотите, только если счетчик равен 1 (последняя прокрутка), а затем уменьшает счетчик.

Я создал Gist, но я копирую здесь код:

public class ScrollablePane extends Pane {
    private Integer scrollCounter = 0;

    private final ObjectProperty<EventHandler<? super ScrollEvent>> onScrollEnded = new SimpleObjectProperty<>();

    public final ObjectProperty<EventHandler<? super ScrollEvent>> onScrollEndedProperty() {
        return onScrollEnded;
    }

    public ScrollablePane() {
        this.setOnScroll(e -> {
            scrollCounter++;

            Thread th = new Thread(() -> {
                try {
                    Thread.sleep(1000);
                    if (scrollCounter == 1)
                        onScrollEnded.get().handle(e);

                    scrollCounter--;
                } catch (Exception e1) {
                    e1.printStackTrace();
                }
            });
            th.setDaemon(true);
            th.start();
        });
    }

    public void setOnScrollEnded(EventHandler<? super ScrollEvent> handler) {
        onScrollEnded.setValue(handler);
    }
}

Чтобы использовать это:

public class MyApplication extends Application {


    @Override
    public void start(Stage primaryStage) {
        try {
            BorderPane root = new BorderPane();
            Scene scene = new Scene(root, 400, 400);

            ScrollablePane pane = new ScrollablePane();
            pane.setOnScrollEnded(e -> System.out.println("Scroll just has been ended"));

            root.setCenter(pane);
            primaryStage.setScene(scene);
            primaryStage.show();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        launch(args);
    }
}
Другие вопросы по тегам