如何滚动以使ScrollPane内容中的一个节点可见?

12

假设一个有多个子节点的巨大AnchorPaneScrollPane的内容,如何滚动以使其中一个当前视口之外的子节点可见?

3个回答

16

编辑:下一个示例中的代码比我的更精确:https://dev59.com/X3DYa4cB1Zd3GeqPB5V8#23518314

您需要找到 content 中此内部节点的坐标,并相应地调整 ScrollPanevValuehValue

请参见下一个小应用程序中的 ensureVisible() 方法:

public class ScrollPaneEnsureVisible extends Application {

    private static final Random random = new Random();

    private static void ensureVisible(ScrollPane pane, Node node) {
        double width = pane.getContent().getBoundsInLocal().getWidth();
        double height = pane.getContent().getBoundsInLocal().getHeight();

        double x = node.getBoundsInParent().getMaxX();
        double y = node.getBoundsInParent().getMaxY();

        // scrolling values range from 0 to 1
        pane.setVvalue(y/height);
        pane.setHvalue(x/width);

        // just for usability
        node.requestFocus();
    }

    @Override
    public void start(Stage primaryStage) {

        final ScrollPane root = new ScrollPane();
        final Pane content = new Pane();
        root.setContent(content);

        // put 10 buttons at random places with same handler
        final EventHandler<ActionEvent> handler = new EventHandler<ActionEvent>() {
            @Override
            public void handle(ActionEvent event) {
                int index = random.nextInt(10);
                System.out.println("Moving to button " + index);
                ensureVisible(root, content.getChildren().get(index));
            }
        };

        for (int i = 0; i < 10; i++) {
            Button btn = new Button("next " + i);
            btn.setOnAction(handler);
            content.getChildren().add(btn);
            btn.relocate(2000 * random.nextDouble(), 2000 * random.nextDouble());
        }

        Scene scene = new Scene(root, 300, 250);
        primaryStage.setScene(scene);
        primaryStage.show();

        // run once to don't search for a first button manually
        handler.handle(null);
    }

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

10

我为这种情况创建了一个稍微更优雅的版本。然而,我只能在y轴上进行测试。希望它有所帮助。

private static void ensureVisible(ScrollPane scrollPane, Node node) {
    Bounds viewport = scrollPane.getViewportBounds();
    double contentHeight = scrollPane.getContent().localToScene(scrollPane.getContent().getBoundsInLocal()).getHeight();
    double nodeMinY = node.localToScene(node.getBoundsInLocal()).getMinY();
    double nodeMaxY = node.localToScene(node.getBoundsInLocal()).getMaxY();

    double vValueDelta = 0;
    double vValueCurrent = scrollPane.getVvalue();

    if (nodeMaxY < 0) {
        // currently located above (remember, top left is (0,0))
        vValueDelta = (nodeMinY - viewport.getHeight()) / contentHeight;
    } else if (nodeMinY > viewport.getHeight()) {
        // currently located below
        vValueDelta = (nodeMinY + viewport.getHeight()) / contentHeight;
    }
    scrollPane.setVvalue(vValueCurrent + vValueDelta);
}

你的有什么不同之处? - Brad Turek
很好的答案,但我注意到if语句需要修复。当用户向下滚动滚动视口时,视口边界将具有负minY值。if语句应该这样说:“if(nodeMaxY < Math.abs(viewport.getMinY())” ,否则应该这样说:“else if(nodeMinY > viewport.getHeight()+ viewport.getMinY())”。至少对我来说这样做可以使其正常工作。 - Dirk Lemmermann

4
为了获得完美的截图,您需要考虑滚动条的起始和停止位置。视口从0(边界滚动窗格)的一半视口开始,并在最大边界的一半视口处停止。 边界与视口图片 因此: 节点Y位于半个视口以下->将Vvalue设置为0 节点Y位于maxheigh-half viewport以上->设置值为1
在此之间,我们需要计算总高度-1视口(一半开始,一半在结束处)和节点y-一半视口。 然后(节点y-半视口)/(总高度-1视口)。
                double heightViewPort = scrollPane.getViewportBounds().getHeight();
                double heightScrollPane = scrollPane.getContent().getBoundsInLocal().getHeight();
                double y = Label.getBoundsInParent().getMaxY();
                if (y<(heightViewPort/2)){
                    scrollPane.setVvalue(0);
                    // below 0 of scrollpane

                }else if ((y>=(heightViewPort/2))&(y<=(heightScrollPane-heightViewPort/2))){
                   // between 0 and 1 of scrollpane
                    scrollPane.setVvalue((y-(heightViewPort/2))/(heightScrollPane-heightViewPort));
                }
                else if(y>= (heightScrollPane-(heightViewPort/2))){
                    // above 1 of scrollpane
                    scrollPane.setVvalue(1);

                }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接