【问题标题】:JavaFX 2 User Idle DetectionJavaFX 2 用户空闲检测
【发布时间】:2015-01-25 13:28:35
【问题描述】:


我正在尝试使用 JavaFX 作为 UI 制作一个简单的 Java 事务应用程序。
我现在要做的是从我的应用程序中检测用户空闲状态,该应用程序具有 1 个主要阶段和许多场景。

示例:如果用户空闲 3 分钟,则返回主菜单。

我已经在网上尝试了一些关于如何检测 JavaFX 空闲状态的示例,但我发现总是 - 一个功能空闲状态检测在所有场景中都发生 - 方法(我认为)对事务应用程序很危险(例如:应用程序检测交易过程中的空闲状态)。
可以在每个场景中检测到用户空闲状态吗?如何?
谢谢。

编辑:

我已经尝试过的例子:

http://tomasmikula.github.io/blog/2014/06/04/timers-in-javafx-and-reactfx.html

http://ochafik.com/blog/?p=98

【问题讨论】:

  • 你的问题很不清楚。您能否展示您尝试过的解决方案并更清楚地解释为什么它不适合您?

标签: java javafx javafx-2 idle-processing


【解决方案1】:

我不太明白您对交易行为的看法。事务涉及对数据的保证,并且您的事务行为应该在数据级别定义,并且不应受到 UI 中发生的事情的影响。换句话说,即使 UI 由于用户空闲而重置,您的原子行为也应该完成或回滚。

不过,也许这会有所帮助。 (注意我在这些示例中使用了 Java 8 代码,但如果需要,您可以很容易地使其符合 JavaF 2.2。)这遵循 Tomas Mikula 的一般方法,因为它使用Timeline 来实现空闲检查。我没有使用 Tomas 的 FX Timer 包装器,但如果你愿意,你当然可以这样做。这个类封装了一个用户是否空闲的监视器。您可以注册任何节点(或场景)和事件类型:如果该类型的事件发生在该节点(或场景)上,则确定用户不空闲。如果指定的时间过去了而没有发生任何注册事件,则执行提供的可运行文件(在 FX 应用程序线程上)。这使您可以灵活地创建多个监视器(如果需要),并为每个监视器注册一个或多个节点。

import javafx.animation.Animation;
import javafx.animation.KeyFrame;
import javafx.animation.Timeline;
import javafx.event.Event;
import javafx.event.EventHandler;
import javafx.event.EventType;
import javafx.scene.Node;
import javafx.scene.Scene;

import javafx.util.Duration;

public class IdleMonitor {

    private final Timeline idleTimeline ;

    private final EventHandler<Event> userEventHandler ;

    public IdleMonitor(Duration idleTime, Runnable notifier, boolean startMonitoring) {
        idleTimeline = new Timeline(new KeyFrame(idleTime, e -> notifier.run()));
        idleTimeline.setCycleCount(Animation.INDEFINITE);

        userEventHandler = e -> notIdle() ; 

        if (startMonitoring) {
            startMonitoring();
        }
    }

    public IdleMonitor(Duration idleTime, Runnable notifier) {
        this(idleTime, notifier, false);
    }

    public void register(Scene scene, EventType<? extends Event> eventType) {
        scene.addEventFilter(eventType, userEventHandler);
    }

    public void register(Node node, EventType<? extends Event> eventType) {
        node.addEventFilter(eventType, userEventHandler);
    }

    public void unregister(Scene scene, EventType<? extends Event> eventType) {
        scene.removeEventFilter(eventType, userEventHandler);
    }

    public void unregister(Node node, EventType<? extends Event> eventType) {
        node.removeEventFilter(eventType, userEventHandler);
    }

    public void notIdle() {
        if (idleTimeline.getStatus() == Animation.Status.RUNNING) {
            idleTimeline.playFromStart();
        }
    }

    public void startMonitoring() {
        idleTimeline.playFromStart();
    }

    public void stopMonitoring() {
        idleTimeline.stop();
    }
}

这是一个测试。 “开始”按钮可能是登录的替代品。主 UI 有一个带有两个选项卡的选项卡窗格:每个单独的选项卡都以自己的“开始”按钮开头,然后主要内容有一个标签、文本字段和按钮.

每个选项卡内容都有一个(简称,用于测试)空闲监视器与之关联。选项卡内容上的任何事件都会重置空闲监视器,但选项卡内容之外的事件不会重置它。还有一个用于整个窗口的“全局”空闲监视器,它会在 30 秒后重置整个 UI。

请注意,数据会被保留:即,如果您因空闲而超时,您在文本字段中键入的任何文本都会被正确保留。这就是为什么我认为“交易”的问题根本不重要。

import javafx.application.Application;
import javafx.event.Event;
import javafx.geometry.Pos;
import javafx.scene.Node;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.Tab;
import javafx.scene.control.TabPane;
import javafx.scene.control.TextField;
import javafx.scene.layout.StackPane;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;
import javafx.util.Duration;

public class IdleTest extends Application {

    @Override
    public void start(Stage primaryStage) {

        StackPane root = new StackPane();

        Parent mainUI = buildMainUI();
        Scene scene = new Scene(root, 350, 150);
        Parent startUI = buildStartUI(() -> root.getChildren().setAll(mainUI));
        root.getChildren().add(startUI);

        IdleMonitor idleMonitor = new IdleMonitor(Duration.seconds(30),
                () -> root.getChildren().setAll(startUI), true);
        idleMonitor.register(scene, Event.ANY);

        primaryStage.setScene(scene);
        primaryStage.show();
    }

    private Parent buildStartUI(Runnable start) {
        Button button = new Button("Start");
        button.setOnAction(e -> start.run());
        StackPane root = new StackPane(button);
        return root ;
    }

    private Parent buildMainUI() {
        TabPane tabPane = new TabPane();
        Tab tab1 = new Tab("One");
        Parent tab1Content = buildTabUI("Tab 1");
        Parent tab1StartContent = buildStartUI(() -> tab1.setContent(tab1Content));
        tab1.setContent(tab1StartContent);
        IdleMonitor tab1IdleMonitor = new IdleMonitor(Duration.seconds(5), 
                () -> tab1.setContent(tab1StartContent), true);
        tab1IdleMonitor.register(tab1Content, Event.ANY);

        Tab tab2 = new Tab("Two");
        Parent tab2Content = buildTabUI("Tab 2") ;
        Parent tab2StartContent = buildStartUI(() -> tab2.setContent(tab2Content));
        tab2.setContent(tab2StartContent);
        IdleMonitor tab2IdleMonitor = new IdleMonitor(Duration.seconds(10),
                () -> tab2.setContent(tab2StartContent), true);
        tab2IdleMonitor.register(tab2Content, Event.ANY);

        tabPane.getTabs().addAll(tab1, tab2);
        return tabPane ;
    }

    private Parent buildTabUI(String text) {
        Button button = new Button("Click here");
        button.setOnAction(e -> System.out.println("Click in "+text));
        VBox content = new VBox(10, new Label(text), new TextField(), button);
        content.setAlignment(Pos.CENTER);
        return content ;
    }

    public static void main(String[] args) {
        launch(args);
    }
}

【讨论】:

  • 我想在 javaFX 2.0 中做同样的事情,那么如何在主类中传递 runnable?你能帮助我吗?我被困在这里IdleMonitor idleMonitor = new IdleMonitor(Duration.seconds(30),"*****" true);
  • 只需使用anonymous inner class 实现Runnable 代替lambda expression
猜你喜欢
  • 2017-05-07
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-03-28
  • 2011-03-07
  • 1970-01-01
  • 2011-03-06
  • 2019-07-29
相关资源
最近更新 更多