根据动态内容自动调整窗口大小

时间:2018-11-28 00:23:11

标签: javafx javafx-8

我正在检查一项功能,可以根据内容自动调整窗口大小。我已经知道Window.sizeToScene()方法。但这是如此麻烦,以至于我需要在每个地方都注意调用sizeToScene()。例如,当我添加/删除节点时,当我展开titlePanes等时...

有人可以让我知道是否可以根据内容自动调整窗口。我正在为普通和透明窗口寻找此功能。

任何提示/建议都受到高度赞赏。

请在下面找到我要寻找的快速演示。如果我考虑调用sizeToScene(),一切都会按预期进行。我的目标是要在不针对每种情况调用sizeToScene的情况下获得相同的行为。

enter image description here

import javafx.application.Application;
import javafx.application.Platform;
import javafx.beans.property.DoubleProperty;
import javafx.beans.property.SimpleDoubleProperty;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.CheckBox;
import javafx.scene.control.Label;
import javafx.scene.control.TitledPane;
import javafx.scene.layout.HBox;
import javafx.scene.layout.StackPane;
import javafx.scene.layout.VBox;
import javafx.scene.paint.Color;
import javafx.stage.Stage;
import javafx.stage.StageStyle;

public class TransparentWindowResizeDemo extends Application {
    @Override
    public void start(Stage stage) throws Exception {
        VBox root = new VBox();
        root.setSpacing(15);
        root.setAlignment(Pos.CENTER);

        Scene sc = new Scene(root, 300, 300);
        stage.setScene(sc);
        stage.show();

        Button showNormal = new Button("Show Normal Window");
        showNormal.setOnAction(e -> showWindow(false));

        Button showTransparent = new Button("Show Transparent Window");
        showTransparent.setOnAction(e -> showWindow(true));

        root.getChildren().addAll(showNormal, showTransparent);
    }

    private void showWindow(boolean isTransparent) {
        Stage window = new Stage();
        VBox root = new VBox();
        root.setStyle("-fx-background-color: grey; -fx-border-width:2px;-fx-border-color:black;");
        Scene scene = new Scene(root, Color.TRANSPARENT);
        window.setScene(scene);

        VBox layout = new VBox();
        layout.setSpacing(5);
        layout.setPadding(new Insets(5));

        CheckBox sizeToScene = new CheckBox("sizeToScene");
        sizeToScene.setSelected(true);

        StackPane box = new StackPane();
        box.setStyle("-fx-background-color:yellow; -fx-border-width:1px;-fx-border-color:black;");
        box.setMinSize(200, 100);
        box.setMaxSize(200, 100);

        Button add = new Button("Add Pane");
        Button remove = new Button("Remove Pane");
        remove.setDisable(true);
        add.setOnAction(e -> {
            layout.getChildren().add(box);
            add.setDisable(true);
            remove.setDisable(false);
            if (sizeToScene.isSelected()) {
                window.sizeToScene();
            }
        });
        remove.setOnAction(e -> {
            layout.getChildren().remove(box);
            add.setDisable(false);
            remove.setDisable(true);
            if (sizeToScene.isSelected()) {
                window.sizeToScene();
            }
        });
        HBox btnLayout = new HBox();
        btnLayout.setSpacing(5);
        btnLayout.getChildren().addAll(add, remove);

        StackPane tpContent = new StackPane();
        tpContent.setStyle("-fx-background-color:pink; -fx-border-width:1px;-fx-border-color:black;");
        tpContent.setMinSize(200, 100);
        tpContent.setMaxSize(200, 100);

        TitledPane tp = new TitledPane("Titled Pane", tpContent);
        tp.setAnimated(false);
        tp.expandedProperty().addListener((obs, oldVal, newVal) -> {
            if (sizeToScene.isSelected()) {
                window.sizeToScene();
            }
        });

        if (isTransparent) {
            window.initStyle(StageStyle.TRANSPARENT);

            StackPane close = new StackPane();
            close.setMaxWidth(30);
            close.setStyle("-fx-background-color:red;");
            close.getChildren().add(new Label("X"));
            close.setOnMouseClicked(e -> window.hide());

            DoubleProperty x = new SimpleDoubleProperty();
            DoubleProperty y = new SimpleDoubleProperty();
            StackPane header = new StackPane();
            header.setOnMousePressed(e -> {
                x.set(e.getSceneX());
                y.set(e.getSceneY());
            });
            header.setOnMouseDragged(e -> {
                window.setX(e.getScreenX() - x.get());
                window.setY(e.getScreenY() - y.get());
            });
            header.setStyle("-fx-border-width:0px 0px 2px 0px;-fx-border-color:black;");
            header.setMinHeight(30);
            header.setAlignment(Pos.CENTER_RIGHT);
            Label lbl = new Label("I am draggable !!");
            lbl.setStyle("-fx-text-fill:white;-fx-font-weight:bold;");
            StackPane.setAlignment(lbl,Pos.CENTER_LEFT);
            header.getChildren().addAll(lbl,close);
            root.getChildren().add(header);
        }

        layout.getChildren().addAll(sizeToScene, btnLayout, tp);
        root.getChildren().add(layout);
        window.show();

    }

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

编辑:: 这与我在其他问题中找到的内容或注释中的链接不同。指定的解决方案是我所知道的,并且我已经在问题中提到了。我有一个沉重的屏幕,其中包含许多不同的节点。我正在检查所有通用解决方案是否都包含在一个地方,而不是从每个节点调用添加/删除方案。

1 个答案:

答案 0 :(得分:2)

我会说这不是一个很好的解决方案(它更像是一个hack),但至少在您的示例中它已经起作用:

VBox root = new VBox() {
    private boolean needsResize = false;

    @Override
    protected void layoutChildren()
    {
        super.layoutChildren();

        if (!needsResize) {
            needsResize = true;

            Platform.runLater(() -> {
                if (needsResize) {
                    window.sizeToScene();
                    needsResize = false;
                }
            });
        }
    }
};

当然,您应该添加sizeToScene.isSelected()部分,还可以使其成为实际的子类。