java fx 中的root_Java fx 坐标揭秘

本文深入探讨了Java FX的场景图(Scene Graph),强调了根节点(root)的重要性。详细介绍了节点(node)的坐标系统,包括逻辑边界(Layout Bounds)、现实边界(BoundsInLocal)和虚幻边界(BoundsInParent)。此外,还讨论了不同坐标系中的bounds、布局属性(如LayoutX和LayoutY)以及如何通过relocate()方法设置节点位置。同时,文章还涉及了可调整大小的节点和它们的尺寸属性,以及managed节点的概念。
摘要由CSDN通过智能技术生成

scene graph

一张呈现为树结构的数据结构,java fx 的渲染系统 是通过该数据结构来完成图形的渲染。

所有在树结构的每一个节点称为一个node

根节点(root)是唯一一个没有父母节点的node

没有子节点的node的称为leaf node(叶子节点)

1 . 在scene graph中的每一个node,都有自己的笛卡尔坐标系

44eda99fe197

Paste_Image.png

2 . 每个node 的视图“呈现”除了几何图形外,其实还包括effect(特效),clip(裁剪),transformation(变换)等特性

3 . Fx 针对每个node 对所包含的特性,提供了不同bound框来囊括这些特性,不同bound的范围

layoutBounds : node geometry

boundsInLocal: node geometry + effects + clip

boundsInParent: node geometry + effects + clip + transformations

44eda99fe197

Paste_Image.png

public class BoundTest extends Application {

public static void main(String[] args) {

Application.launch(args);

}

@Override

public void start(Stage primaryStage) throws Exception {

VBox root = new VBox();

Scene scene = new Scene(root);

primaryStage.setScene(scene);

// 1. only a simple button

System.out.println("only a simple button");

Button button1 = new Button();

System.out.println("button1=" + button1.getLayoutBounds());

System.out.println("button1=" + button1.getBoundsInLocal());

System.out.println("button1=" + button1.getBoundsInParent());

System.out.println();

System.out.println("add shadow");

// 2. add shadow

Button button2 = new Button();

button2.setEffect(new BoxBlur());

System.out.println("button2=" + button2.getLayoutBounds());

System.out.println("button2=" + button2.getBoundsInLocal());

System.out.println("button2=" + button2.getBoundsInParent());

System.out.println();

System.out.println("add transformation");

// 3. add transformation

Button button3 = new Button();

//button3.setRotate(60);

button3.getTransforms().add(new Translate(150, 75));

System.out.println("button3=" + button3.getLayoutBounds());

System.out.println("button3=" + button3.getBoundsInLocal());

System.out.println("button3=" + button3.getBoundsInParent());

root.getChildren().addAll(button1, new Group(button2), button3);

//primaryStage.show();

}

}

4 . 别名

Layout Bounds: 逻辑bound(logic bound)

BoundsInLocal: 现实上的bound(physical bound) (希望当前node包含Effect,Clip时候用的,另一种情况检测两个node有无碰撞)

BoundsInParent: 虚幻的bound(visual bound) (较少在code中使用到)

5 .这些Bound 处于不同的坐标系中

Layout Bounds: 在该node 的坐标系

BoundsInLocas:在该node 的坐标系

BoundsInParent:在该node 的parent的坐标系

了解这些不同的Bounds

6 . 每一个node 都有一对LayoutX和LayoutY以及一对translateX和translateY 来描述他们在坐标系中的位置

LayoutX 和LayoutY : 固定(stable) layout

translateX和translateY: 动态(dynamic) layout (animaiton中)

finalTranslationX = layoutX + translateX

finalTranslationY = layoutY + translateY

但是layoutX and layoutY 并不是就是直接确定了node的位置,它通常需要转为node的坐标系中的位置,通常需要补偿一个layoutBound.minX和layoutBound.minY

layoutX = finalX - node.getLayoutBounds().getMinX();

layoutY = finalY - node.getLayoutBounds().getMinY();

7 . 由于Bound 的存在,当你想设置坐标时候,记得加上边框左上角的位置的横纵坐标,推荐使用relocate() ,它会帮你补偿minX 和minY of layoutBound,自动完成6中提到的转换

8 . 当Parenty 为region 类时候,父节点有自己的位置策略,这时候忽略node layoutX,layoutY. 事实上只有当你的parent节点(layout manager)为Group或着Pane的时候,通过修改layoutX以及layoutY才能修改Node的位置

Node 的Size

当一个node可以随着父节点layout的调整自己的大小,该节点就是可以resize的.Node 的分为resizable和non-resize node.

re-sizable node: Regions, Controls, andWebView

non-resizable node : Group, Text, and Shapes

三个“大小”

Preferred size 最合适大小,当layout 足够满足的时候呈现的size

Minimum size 最小的大小,当resize的时候的最小大小

Maximum size 最大的大小, 当resize的时候的最大大小

对着三个size可以设置两个常量,

USE_COMPUTED_SIZE :依据的node的内容和属性计算

USE_PREF_SIZE:通常用来设置Minimum size和Maximum size保持和Preferred size一致

每个node也有6个属性来对应这个三个特性,这些属性的默认值为:USE_COMPUTED_SIZE

prefWidth,prefHeight

minWidth,minHeight

maxWidth,maxHeight

不要相信getXXX(getMaxWidth,getMinWidth,getPre...),通常这些方法返回的值并没有真实反映node的大小

使用以下方法:

double prefWidth(double height)

double prefHeight(double width)

double minWidth(double height)

double minHeight(double width)

double maxWidth(double height)

double maxHeight(double width)

通过传入-1来获得content bias 边的长度,再根据该长度算得相应的宽或者高

44eda99fe197

Paste_Image.png

public class NodeSizes extends Application {

public static void main(String[] args) {

Application.launch(args);

}

@Override

public void start(Stage stage) {

Button btn = new Button("Hello JavaFX!");

HBox root = new HBox();

root.getChildren().addAll(btn);

Scene scene = new Scene(root);

stage.setScene(scene);

stage.setTitle("Sizes of a Node");

stage.show();

// Print button's sizes

System.out.println("Before changing button properties:");

printSizes(btn);

// Change button's properties

btn.setWrapText(true);

btn.setPrefWidth(80);

stage.sizeToScene();

// Print button's sizes

System.out.println("\nAfter changing button properties:");

printSizes(btn);

}

public void printSizes(Button btn) {

System.out.println("btn.getContentBias() = " + btn.getContentBias());

System.out.println("btn.getPrefWidth() = " + btn.getPrefWidth() +

", btn.getPrefHeight() = " + btn.getPrefHeight());

System.out.println("btn.getMinWidth() = " + btn.getMinWidth() +

", btn.getMinHeight() = " + btn.getMinHeight());

System.out.println("btn.getMaxWidth() = " + btn.getMaxWidth() +

", btn.getMaxHeight() = " + btn.getMaxHeight());

double prefWidth = btn.prefWidth(-1);

System.out.println("btn.prefWidth(-1) = " + prefWidth +

", btn.prefHeight(prefWidth) = " + btn.prefHeight(prefWidth));

double minWidth = btn.minWidth(-1);

System.out.println("btn.minWidth(-1) = " + minWidth +

", btn.minHeight(minWidth) = " + btn.minHeight(minWidth));

double maxWidth = btn.maxWidth(-1);

System.out.println("btn.maxWidth(-1) = " + maxWidth +

", btn.maxHeight(maxWidth) = " + btn.maxHeight(maxWidth));

System.out.println("btn.getWidth() = " + btn.getWidth() +

", btn.getHeight() = " + btn.getHeight());

}

}

7 对于non-resizable node

prefWidth(double h), minWidth(double h), maxWidth(double h) 返回layoutBound的width

prefHeight(double w), minHeight(double w), maxHeight(double w) 返回layoutBound的height

8 关于managed node

当子节点的managedProperty设置为false的时候,父节点在layout布局的时候,就不会帮把计算如layoutboundbox. 这与visibleProperty的区别,尽管它不可见,父节点仍考虑它的布局。

public class SlidingLeftNodeTest extends Application {

public static void main(String[] args) {

Application.launch(args);

}

@Override

public void start(Stage stage) {

Button b1 = new Button("B1");

Button b2 = new Button("B2");

Button b3 = new Button("B3");

Button visibleBtn = new Button("Make Invisible");

// Add an action listener to the button to make b2 visible

// if it is invisible and invisible if it is visible

visibleBtn.setOnAction(e -> b2.setVisible(!b2.isVisible()));

// Bind the text property of the button to the visible

// property of the b2 button

visibleBtn.textProperty().bind(new When(b2.visibleProperty()).then("Make Invisible").otherwise("Make Visible"));

b2.managedProperty().bind(b2.visibleProperty());

HBox root = new HBox();

root.getChildren().addAll(visibleBtn, b1, b2, b3);

Scene scene = new Scene(root);

stage.setScene(scene);

stage.setTitle("Sliding to the Left");

stage.show();

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值