【问题标题】:Get visible state of Node in JavaFX 8在 JavaFX 8 中获取节点的可见状态
【发布时间】:2017-09-04 11:00:30
【问题描述】:

我需要检测一个节点当前是否正在显示。 IE。如果我的节点在 TabPane 中,我需要知道它是否在选定的选项卡中。

在示例中,我想知道 HBox 何时显示。Node 的 visibleProperty 和 managedProperty,似乎对我没有帮助:

public class VisibleTest extends Application {

@Override
public void start(Stage primaryStage) throws Exception {

    TabPane tabpane = new TabPane();
    tabpane.getTabs().add(new Tab("Tab1", new Label("Label1")));

    HBox hbox = new HBox(new Label("Label2"));
    hbox.setStyle("-fx-background-color: aquamarine;");

    hbox.visibleProperty().addListener((observable, oldValue, newValue) -> {
        System.out.println("Hbox visible changed. newValue: " + newValue);
    });

    hbox.managedProperty().addListener((observable, oldValue, newValue) -> {
        System.out.println("Hbox managed changed. newValue: " + newValue);
    });

    Tab tab2 = new Tab("tab2", hbox);
    tabpane.getTabs().add(tab2);

    primaryStage.setScene(new Scene(tabpane));
    primaryStage.setWidth(600);
    primaryStage.setHeight(500);
    primaryStage.show();
}

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

我知道,可以在选项卡的selectedProperty 状态下收听,但这并不能解决我真正的问题。

Node.impl_isTreeVisible() 做我想做的事,但这是废弃的 API。

有什么想法吗?

- - - - - - - - - - - - - - - - - - 更新 - - - - - - - --------
我意识到上面的代码示例并不能很好地解释我想要完成的工作。
下面是一些 Swing 代码,它展示了我想要在 JavaFX 中完成的工作。检测 JComponent/Node 是否可见/显示,并根据该状态启动或停止后台进程。如果它是一个 javaFX 类,构造函数会是什么样子。

public class SwingVisible extends JComponent {

    String instanceNR;
    Thread instanceThread;
    boolean doExpensiveStuff = false;

    public SwingVisible(String instanceNR) {
        this.instanceNR = instanceNR;
        this.setLayout(new FlowLayout());
        this.add(new JLabel(instanceNR));

        instanceThread = new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                    if (doExpensiveStuff) {
                        /*
                         * do expensive stuff.
                         */
                        System.out.println(instanceNR + " is visible " + isVisible());
                    }
                }
            }
        });

        /*
         * How to do this in FX?
         */
        addComponentListener(new ComponentAdapter() {
            @Override
            public void componentShown(ComponentEvent e) {
                if (!instanceThread.isAlive()) {
                    instanceThread.start();
                }
                doExpensiveStuff = true;
            }

            @Override
            public void componentHidden(ComponentEvent e) {
                doExpensiveStuff = false;
            }
        });
    }

    public static void main(String[] args) {    
        /*
         * This block represents code that is external to my library. End user
         * can put instances of SwingVisible in JTabbedPanes, JFrames, JWindows,
         * or other JComponents. How many instances there will bee is not in my
         * control.
         */
        JTabbedPane jtp = new JTabbedPane();
        jtp.add("tab1", new SwingVisible("1"));
        jtp.add("tab2", new SwingVisible("2"));
        jtp.add("tab3", new SwingVisible("3"));

        JFrame f = new JFrame("test");
        f.setContentPane(jtp);
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setSize(300, 300);
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
}

选择tab1时的输出:

1 可见为真
1 可见 true
1 可见 true

...

选择tab2时的输出:

2 可见为真
2 可见真
2 可见真
...

【问题讨论】:

  • 如果tab2正在显示,你不知道hbox正在显示吗?
  • 这可能是真的,但是如果您使用我的原始答案,您就会知道选择 tab2 时 hbox 正在屏幕上显示/可见。
  • HBox 的可见性相当于它所在标签的 selected 属性,为什么不使用呢?
  • 正确的做法是使用标签的selectedProperty()。任何其他方法的问题在于它依赖于选项卡窗格的实现。目前,选项卡窗格的工作方式是将所有选项卡的内容布局在一个窗格中,并根据是否选择选项卡使每个选项卡的内容可见或不可见。但是,不能保证此实现将保留在未来的版本中:因此您唯一的选择是响应选项卡的 selected 属性。
  • 您说“这不是真正的问题”。我们非常感谢您为制作一个最小、完整的示例所做的努力,但它似乎并不能完全描述您要解决的问题。看看你是否能找出一个描述实际问题的例子。请注意,知道“标签何时可见”非常困难,因为您真的需要知道它及其所有祖先是否可见,如果它是显示在正在显示的窗口中的场景的一部分,以及如果在其顶部显示任何其他节点。

标签: java javafx javafx-8


【解决方案1】:

您可以使用TabselectedProperty 来了解它是否被选中,以及其内容是否可见。它是一个布尔属性。

我已根据您最初的 JavaFX 示例将您的 Swing 代码转换为 JavaFX:

public class VisibleTest extends Application {

    public class FXVisible extends Tab {

        FXVisible(String id) {
            super(id, new Label(id));

            Timeline thread = new Timeline(
                    new KeyFrame(Duration.ZERO, e -> { 
                        if (isSelected()) {
                            // do expensive stuff
                            System.out.println(id + " is visible");
                        }
                    }),
                    new KeyFrame(Duration.seconds(1))
            );
            thread.setCycleCount(Timeline.INDEFINITE);

            selectedProperty().addListener((selectedProperty, wasSelected, isSelected) -> {
                if (isSelected) {
                    if (thread.getStatus() != Status.RUNNING) {
                        System.out.println(id + " starting thread");
                        thread.play();
                    }
                }
                // else, it is not selected -> content not shown
            });
        }
    }

    @Override
    public void start(Stage primaryStage) throws Exception {
        TabPane tabpane = new TabPane();
        tabpane.getTabs().add(new FXVisible("1"));
        tabpane.getTabs().add(new FXVisible("2"));
        tabpane.getTabs().add(new FXVisible("3"));
        // add as many as you want

        primaryStage.setScene(new Scene(tabpane));
        primaryStage.setWidth(600);
        primaryStage.setHeight(500);
        primaryStage.show();
    }

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

我用 JavaFX Timeline 替换了您的线程。你的问题与这个主题无关,所以我不会在这里详细说明,尽管它是不言自明的。

我不明白为什么在 Swing 示例中,当您可以直接在线程中调用 isVisible() 时,您有一个侦听器更改了一个布尔值,该布尔值指示组件是否可见(有关线程的注释,请参见下面的 cmets )。这就是为什么在我上面的代码中,我采用了直接检查isSelected() 的方法,没有自我声明的布尔值。如果您需要恢复到您的设计,这很简单。只是为了清楚起见而注意到这一点。

ComponentListener 可以替换为selectedProperty() 上的更改侦听器并查询新值。只要确保您的示例完成了它应该做的事情:第一次选择选项卡时,线程/计时器就会启动。之后,线程/计时器什么也不做。您可能想暂停非显示内容的计算。再说一次,只是注意到它,因为这对我来说似乎是一个潜在的错误,否则你很好。

【讨论】:

  • 线程注释:对于 Swing 代码,isVisible() 只能从 AWT 事件调度线程调用:它不是线程安全的方法。 visible 属性将由 Swing 框架在 AWT 事件分派线程上更改:无法保证其他线程何时,甚至是否会“看到”这些更改。 OP 发布的代码也存在缺陷,但可以通过将 doExpensiveStuff 布尔值替换为 AtomicBoolean 或将其标记为 volatile 来修复。
  • @James_D 好点。我用 Swing Timer 做了一些测试,isVisible() 总是正确更新,但如果你说的是真的,它们需要一些同步。 javaFX的isSelected()也是这样吗?
  • 是的,JavaFX 也是单线程的。 (而且,正如我想你知道的那样,仅仅因为它恰好在测试中正常工作并不等于知道它总是能正常工作。其他平台或 JDK 版本,甚至只是偶尔不走运的时机,都可能导致这样的事情中断.)
  • @James_D 更新了答案以在线程问题上参考 cmets。
  • 谢谢 user1803551。我已经测试了你的代码。并试图修改它。我的问题是,我仍然必须依靠选项卡来通知我的节点它的可见/显示状态。选项卡(或其他类型的父项)在我的库之外。
【解决方案2】:

更新答案。

tab2.getContent().isVisible();

【讨论】:

  • 感谢您的回复!让我把我的问题重新表述得更具体一些:节点如何知道它当前是否正在显示?在 Swing 中,我可以使用 Component.isVisible() 或 ComponentListener 来获取此信息。在 JavaFX 中,isVisible 的行为有所不同。
  • Swing node.isVisible() 与 JavaFx node.isVisible() 有何不同?
  • 在 JavaFX 中这似乎是真的,即使节点没有显示。
  • “节点如何知道它当前是否正在显示”:它不知道。布局和渲染是“自上而下”执行的。因此,场景的根布局了它的控件,并通过渲染它计算出的需要渲染的子节点(或可能的部分子节点)来渲染自己。每个子节点,如果它是Parent,递归地对其布局执行相同的操作。所以一个节点不知道它是否可见(在你的意思上),它只是被告知在必要时由它的父节点执行它自己的布局。
  • 感谢您的解释!因此,节点本身无法检测它是否在屏幕上显示/可见。
【解决方案3】:

在我看来,我原来的答案是正确的。如果没有,您需要以更好的方式提出您的问题。您想知道 hbox 何时可见(意味着您可以在屏幕上看到 hbox)。

tabpane.getSelectionModel().selectedItemProperty().addListener((obsVal, oldTab, newTab)->{
    System.out.println(newTab.getText());
    if(newTab.getText().equals("tab2"))
    {
        //You can use this code to set the hbox visibility, that way you can force the behavior you are looking for.
        hbox.setVisible(true);
        System.out.println("hbox is visible!");
    }
    else
    {
        //You can use this code to set the hbox visibility, that way you can force the behavior you are looking for.
        hbox.setVisible(false);
        System.out.println("hbox is not visible!");
    }            
});

【讨论】:

  • 感谢您的建议!
猜你喜欢
  • 2015-10-26
  • 1970-01-01
  • 2020-02-17
  • 2016-07-09
  • 2015-04-30
  • 1970-01-01
  • 2014-10-12
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多