JavaFX TreeView:TreeCells的访问列表
JavaFX TreeView: Access list of TreeCells
我正在尝试从我的 TreeView 中获取当前存在的 TreeCell 列表。我目前的尝试是这样的:
private Set<MyTreeCell> getMyTreeCells(){
try {
@SuppressWarnings("unchecked")
Field f = ((VirtualContainerBase<TreeView<myNode>, TreeViewBehavior<myNode>, MyTreeCell>) myTreeView.skinProperty().get()).getClass().getDeclaredField("flow");
f.setAccessible(true);
@SuppressWarnings("unchecked")
Field g = ((VirtualFlow<MyTreeCell>) f.get(((VirtualContainerBase<TreeView<myNode>, TreeViewBehavior<myNode>, ProofTreeCell>) proofTreeView.skinProperty().get()))).getClass().getDeclaredField("cells");
g.setAccessible(true);
@SuppressWarnings("unchecked")
ArrayLinkedList<MyTreeCell> l = (ArrayLinkedList<MyTreeCell>) g.get(((VirtualFlow<MyTreeCell>) f.get(((TreeViewSkin<myNode>) myTreeView.skinProperty().get()))));
Set<myTreeCell> s = new HashSet<>();
s.addAll(l);
return s;
}
catch (NoSuchFieldException e) {
e.printStackTrace();
}
catch (SecurityException e) {
e.printStackTrace();
}
catch (IllegalArgumentException e) {
e.printStackTrace();
}
catch (IllegalAccessException e) {
e.printStackTrace();
}
return null;
}
不幸的是它抛出 java.lang.NoSuchFieldException: flow
。但为什么?显然字段 flow
存在于 Class VirtualContainerBase 中。
请注意,我写这篇文章只是为了好玩,而不是实际用于生产代码中。我已经有了更清晰的问题解决方案。
VirtualContainerBase
声明 flow
,但 myTreeView.skinProperty().get().getClass()
不声明 return VirtualContainerBase.class
,它 return 是 TreeViewSkin.class
。由于 getDeclaredField()
只有 return 在 class 中声明的字段,而不是继承的字段,因此您会得到异常。
我觉得你可以做到
Field f = VirtualContainerBase.class.getDeclaredField("flow");
Field g = VirtualFlow.class.getDeclaredField("cells");
您还可以消除大部分的转换,如本例所示:
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import com.sun.javafx.scene.control.skin.VirtualContainerBase;
import com.sun.javafx.scene.control.skin.VirtualFlow;
import javafx.application.Application;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.TreeCell;
import javafx.scene.control.TreeItem;
import javafx.scene.control.TreeView;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
public class GetTreeCellsByReflection extends Application {
@SuppressWarnings({ "restriction", "unchecked" })
@Override
public void start(Stage primaryStage) {
TreeView<Integer> tree = createRandomTree(100);
Button findCells = new Button("Find cells");
findCells.setOnAction(e -> {
try {
Field flowField = VirtualContainerBase.class.getDeclaredField("flow");
flowField.setAccessible(true);
Field cellsField = VirtualFlow.class.getDeclaredField("cells");
cellsField.setAccessible(true);
Object flow = flowField.get(tree.getSkin());
((Iterable<TreeCell<?>>) cellsField.get(flow)).forEach(System.out::println);;
} catch (Exception exc) {
exc.printStackTrace();
}
});
BorderPane root = new BorderPane(tree);
BorderPane.setAlignment(findCells, Pos.CENTER);
BorderPane.setMargin(findCells, new Insets(5));
root.setBottom(findCells);
Scene scene = new Scene(root, 600, 600);
primaryStage.setScene(scene);
primaryStage.show();
}
private TreeView<Integer> createRandomTree(int numItems) {
Random rng = new Random();
TreeItem<Integer> root = new TreeItem<>(0);
List<TreeItem<Integer>> items = new ArrayList<>();
items.add(root);
for (int i = 1 ; i < numItems ; i++) {
TreeItem<Integer> item = new TreeItem<>(i);
items.get(rng.nextInt(items.size())).getChildren().add(item);
items.add(item);
}
return new TreeView<>(root);
}
public static void main(String[] args) {
launch(args);
}
}
我正在尝试从我的 TreeView 中获取当前存在的 TreeCell 列表。我目前的尝试是这样的:
private Set<MyTreeCell> getMyTreeCells(){
try {
@SuppressWarnings("unchecked")
Field f = ((VirtualContainerBase<TreeView<myNode>, TreeViewBehavior<myNode>, MyTreeCell>) myTreeView.skinProperty().get()).getClass().getDeclaredField("flow");
f.setAccessible(true);
@SuppressWarnings("unchecked")
Field g = ((VirtualFlow<MyTreeCell>) f.get(((VirtualContainerBase<TreeView<myNode>, TreeViewBehavior<myNode>, ProofTreeCell>) proofTreeView.skinProperty().get()))).getClass().getDeclaredField("cells");
g.setAccessible(true);
@SuppressWarnings("unchecked")
ArrayLinkedList<MyTreeCell> l = (ArrayLinkedList<MyTreeCell>) g.get(((VirtualFlow<MyTreeCell>) f.get(((TreeViewSkin<myNode>) myTreeView.skinProperty().get()))));
Set<myTreeCell> s = new HashSet<>();
s.addAll(l);
return s;
}
catch (NoSuchFieldException e) {
e.printStackTrace();
}
catch (SecurityException e) {
e.printStackTrace();
}
catch (IllegalArgumentException e) {
e.printStackTrace();
}
catch (IllegalAccessException e) {
e.printStackTrace();
}
return null;
}
不幸的是它抛出 java.lang.NoSuchFieldException: flow
。但为什么?显然字段 flow
存在于 Class VirtualContainerBase 中。
请注意,我写这篇文章只是为了好玩,而不是实际用于生产代码中。我已经有了更清晰的问题解决方案。
VirtualContainerBase
声明 flow
,但 myTreeView.skinProperty().get().getClass()
不声明 return VirtualContainerBase.class
,它 return 是 TreeViewSkin.class
。由于 getDeclaredField()
只有 return 在 class 中声明的字段,而不是继承的字段,因此您会得到异常。
我觉得你可以做到
Field f = VirtualContainerBase.class.getDeclaredField("flow");
Field g = VirtualFlow.class.getDeclaredField("cells");
您还可以消除大部分的转换,如本例所示:
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import com.sun.javafx.scene.control.skin.VirtualContainerBase;
import com.sun.javafx.scene.control.skin.VirtualFlow;
import javafx.application.Application;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.TreeCell;
import javafx.scene.control.TreeItem;
import javafx.scene.control.TreeView;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
public class GetTreeCellsByReflection extends Application {
@SuppressWarnings({ "restriction", "unchecked" })
@Override
public void start(Stage primaryStage) {
TreeView<Integer> tree = createRandomTree(100);
Button findCells = new Button("Find cells");
findCells.setOnAction(e -> {
try {
Field flowField = VirtualContainerBase.class.getDeclaredField("flow");
flowField.setAccessible(true);
Field cellsField = VirtualFlow.class.getDeclaredField("cells");
cellsField.setAccessible(true);
Object flow = flowField.get(tree.getSkin());
((Iterable<TreeCell<?>>) cellsField.get(flow)).forEach(System.out::println);;
} catch (Exception exc) {
exc.printStackTrace();
}
});
BorderPane root = new BorderPane(tree);
BorderPane.setAlignment(findCells, Pos.CENTER);
BorderPane.setMargin(findCells, new Insets(5));
root.setBottom(findCells);
Scene scene = new Scene(root, 600, 600);
primaryStage.setScene(scene);
primaryStage.show();
}
private TreeView<Integer> createRandomTree(int numItems) {
Random rng = new Random();
TreeItem<Integer> root = new TreeItem<>(0);
List<TreeItem<Integer>> items = new ArrayList<>();
items.add(root);
for (int i = 1 ; i < numItems ; i++) {
TreeItem<Integer> item = new TreeItem<>(i);
items.get(rng.nextInt(items.size())).getChildren().add(item);
items.add(item);
}
return new TreeView<>(root);
}
public static void main(String[] args) {
launch(args);
}
}