- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的目标是简单地编辑 TreeView 的单元格(为此,我阅读了 JavaFX 的文档)。
我的问题是我的 TreeView 不包含字符串
,而是我自己的数据类型,即调用:数据库
。我收到错误:Database$Variable 无法转换为 core.Database
。
是否可以用我自己的类型实现 TreeView ?
这是我在 James_D 的帮助下使用的代码:
public class DatabaseStructureView extends TreeView<Object> {
private Database db;
@FXML
private TreeView<Database> databaseStructureView;
public DatabaseStructureView() {
FXMLLoader fxmlLoader = new FXMLLoader(getClass().getResource("DatabaseStructureView.fxml"));
fxmlLoader.setRoot(this);
fxmlLoader.setController(this);
try {
fxmlLoader.load();
} catch (IOException exception) {
throw new RuntimeException(exception);
}
databaseStructureView.setEditable(true);
databaseStructureView.setCellFactory(new Callback<TreeView<Database>,TreeCell<Database>>(){
@Override
public TreeCell<Database> call(TreeView<Database> param) {
return new TextFieldTreeCellImpl();
}
});
}
private final class TextFieldTreeCellImpl extends TreeCell<Database> {
private TextField textField;
public TextFieldTreeCellImpl() {
}
@Override
public void startEdit() {
super.startEdit();
if (textField == null) {
createTextField();
}
setText(null);
setGraphic(textField);
textField.selectAll();
}
@Override
public void cancelEdit() {
super.cancelEdit();
setText(getString());
setGraphic(getTreeItem().getGraphic());
}
@Override
public void updateItem(Database item, boolean empty) {
super.updateItem(item, empty);
if (empty) {
setText(null);
setGraphic(null);
} else {
if (isEditing()) {
if (textField != null) {
textField.setText(getString());
}
setText(null);
setGraphic(textField);
} else {
setText(getString());
setGraphic(getTreeItem().getGraphic());
}
}
}
private void createTextField() {
textField = new TextField(getString());
textField.setOnKeyReleased(new EventHandler<KeyEvent>() {
@Override
public void handle(KeyEvent t) {
if (t.getCode() == KeyCode.ENTER) {
commitEdit(getDatabase(textField.getText()));
} else if (t.getCode() == KeyCode.ESCAPE) {
cancelEdit();
}
}
});
}
private Database getDatabase(String text) {
return new Database(text);
}
private String getString(){
return getItem() == null ? "" : getItem().toString();
}
}
public void setDatabase(Database db) {
this.db = db;
TreeItem<Object> dbRoot = new TreeItem<>(db);
dbRoot.setExpanded(true);
for (Variable var : db.getVariables()) {
dbRoot.getChildren().add(new TreeItem<Object>(var));
}
setRoot(dbRoot);
}
}
这是我的 DataBase 类:
public class Database {
public static class Variable {
public static enum VariableType {
NUMERIC, INTEGER, LOGIC, ORDONEE, NOMINAL, DATE
}
private String name;
private VariableType type;
private List<String> values;
public Variable(String name) {
this.name = name;
values = new ArrayList<>();
}
private void addValue(String value) {
values.add(value);
}
public String getValue(int index) {
return values.get(index);
}
public void addValue(int index, String value) {
values.add(index, value);
}
public VariableType getType() {
return type;
}
public String getName() {
return name;
}
public List<String> getValues() {
return values;
}
public String calculerMinimum (){
if(this.getType().equals(VariableType.INTEGER)){
// Je convertit en entier les strings
List<String> maColonneString = this.getValues();
List<Integer> maColonneInt = new ArrayList<Integer>();
for(int i = 0; i < maColonneString.size(); i++){
if(!maColonneString.get(i).equals("NULL")){
maColonneInt.set(i, Integer.parseInt(maColonneString.get(i)));
}
}
return(Integer.toString(Collections.min(maColonneInt)));
}
else {
return null;
}
}
public String calculerMaximum (){
if(this.getType().equals(VariableType.INTEGER)){
// Je convertit en entier les strings
List<String> maColonneString = this.getValues();
List<Integer> maColonneInt = new ArrayList<Integer>();
for(int i = 0; i < maColonneString.size(); i++){
if(!maColonneString.get(i).equals("NULL")){
maColonneInt.set(i, Integer.parseInt(maColonneString.get(i)));
}
}
return(Integer.toString(Collections.max(maColonneInt)));
}
else {
return null;
}
}
public String calculerFrequence (String maValeurTemoin){
List<String> maColonneString = this.getValues();
int compteur = 0;
for(int i = 0; i < maColonneString.size(); i++){
if(maColonneString.get(i).equals(maValeurTemoin)){
compteur ++;
}
}
return Integer.toString(compteur);
}
public String calculerMoyenne (){
if(this.getType().equals(VariableType.INTEGER)){
List<String> maColonneString = this.getValues();
int somme = 0;
int valeurNull=0;
for(int i = 0; i < maColonneString.size(); i++){
if(!maColonneString.get(i).equals("NULL")){
somme += Integer.parseInt(maColonneString.get(i));
}
else{
valeurNull ++;
}
}
return (Integer.toString(somme/(maColonneString.size()-valeurNull)));
}
else {
return null;
}
}
public String calculerVariance (){
if(this.getType().equals(VariableType.INTEGER)){
List<String> maColonneString = this.getValues();
int sommeMul=0;
int variance=0;
int valeurNull=0;
for(int i = 0; i < maColonneString.size(); i++){
if(!maColonneString.get(i).equals("NULL")){
sommeMul += Integer.parseInt(maColonneString.get(i))*Integer.parseInt(maColonneString.get(i));
}
else{
valeurNull ++;
}
}
variance = sommeMul/(maColonneString.size()-valeurNull)-Integer.parseInt(calculerMoyenne())*Integer.parseInt(calculerMoyenne());
return Integer.toString(variance);
}
else {
return null;
}
}
public String calculerEcartType (){
if(this.getType().equals(VariableType.INTEGER)){
return (Double.toString( Math.sqrt(Integer.parseInt(calculerVariance()))));
}
else {
return null;
}
}
public List<String> calculerAnalyseUnivarieRestreinte (){
if(this.getType().equals(VariableType.INTEGER)){
List<String> analyseUniList = new ArrayList<String>();
analyseUniList.add(calculerMinimum());
analyseUniList.add(calculerMaximum());
analyseUniList.add(calculerMoyenne());
return analyseUniList;
}
else{
return null;
}
}
public List<String> calculerAnalyseUnivarieComplete (){
if(this.getType().equals(VariableType.INTEGER)){
List<String> analyseUniList = new ArrayList<String>();
analyseUniList.add(calculerMinimum());
analyseUniList.add(calculerMaximum());
analyseUniList.add(calculerMoyenne());
analyseUniList.add(calculerVariance());
analyseUniList.add(calculerEcartType());
return analyseUniList;
}
else{
return null;
}
}
public boolean rechercherValeur (String maValeurTemoin){
List<String> maColonneString = this.getValues();
return(maColonneString.contains(maValeurTemoin));
}
public void setTypeVariable (VariableType monNouveauType){
this.type = monNouveauType;
}
public void setNomVariable (String monNouveauNom){
this.name = monNouveauNom;
}
public void modifierValeur (String ancienneValeur, String nouvelValeur){
List<String> maColonneString = this.getValues();
for(int i = 0; i < maColonneString.size(); i++){
if(maColonneString.get(i).equals(ancienneValeur)){
this.values.set(i, nouvelValeur);
}
}
}
@Override
public String toString() {
return name;
}
}
private List<Variable> variables;
private String name;
public Database(String name) {
this.name = name;
variables = new ArrayList<>();
}
public void addVariable(String name) {
variables.add(new Variable(name));
}
public String print() {
StringBuilder sb = new StringBuilder();
int i = 0;
while (i != variables.get(0).values.size()) {
for (Variable variable : variables) {
sb.append(variable.values.get(i)).append(" ");
}
sb.append(System.getProperty("line.separator"));
i++;
}
return sb.toString();
}
@Override
public String toString() {
return name;
}
public Variable getVariable(Variable maVariable) {
for (Variable variable : variables) {
if (variable.getName().equals(maVariable.getName())) {
return variable;
}
}
return null;
}
public List<Variable> getVariables() {
return new ArrayList<>(variables);
}
public Variable getVariable(int index) {
return variables.get(index);
}
public boolean addVariable(Variable variableAdd) {
return (this.variables.add(variableAdd));
}
public boolean removeVariable(Variable variableRemove) {
return (this.variables.remove(variableRemove));
}
}
这是我收到的错误:
java.lang.ClassCastException: core.Database$Variable cannot be cast to core.Database
at components.databaseStructureView.DatabaseStructureView$TextFieldTreeCellImpl.updateItem(DatabaseStructureView.java:53)
at javafx.scene.control.TreeCell.updateItem(TreeCell.java:515)
at javafx.scene.control.TreeCell.indexChanged(TreeCell.java:473)
at javafx.scene.control.IndexedCell.updateIndex(IndexedCell.java:116)
最佳答案
您发布的代码有一个奇怪的结构:您有两个 TreeView
引用, TreeView<Object>
的实例子类 ( DatabaseStructureView
),它被设置为 FXML 的动态根,然后是 TreeView<Database>
您声明的引用:databaseStructureView
.
根据您的描述,我只能假设(因为您没有创建 MCVE )您已通过设置 fx:id
将这两个引用分配给同一个对象。根上的属性,如下所示:
<fx:root type="TreeView" fx:id="databaseStructureView" ... >
所以这里发生的是 this
(在 DatabaseStructureView
的上下文中)和 databaseStructureView
引用同一个对象。
这是一个非常令人困惑的设置,您不应该这样做。 (具体来说,当根也是 Controller 时,不要在 fx:id
上设置 <fx:root ... >
。)
发生的情况大致如下:
在某个地方你做类似的事情
TreeView<Object> treeView = new DatabaseStructureView<>();
这会导致 fxml 加载程序加载。因为你这样做setRoot(this)
,它有效地做到了
Object root = treeView ;
(类型为 Object
因为 FXML 实际上是无类型的)。然后因为你有 setController(this)
FXMLLoader
的 Controller 也设置为 treeView
。然后当它解析 FXML 时,它会看到 fx:id
并且还设置 databaseStructureView
至treeView
。所以你实际上有 treeView.databaseStructureView == treeView
。您可以通过添加
System.out.println(this == databaseStructureView);
在FXMLLoader
之后的load()
方法已成功完成。
因为类型删除,并且因为 FXMLLoader
中的所有赋值是通过反射生成的,泛型类型是不可见的,因此 FXML 根的编译时类型为 TreeView<Object>
和databaseStructureView
,其编译时类型为 TreeView<Database>
,两者都有原始 TreeView
的运行时类型。所以这些反射(reflection)性作业都是成功的。
但是发生的情况是,通过在 FXML 加载器中使用反射,您现在已经进行了非类型安全分配,因此所有对泛型类型的赌注都消失了。因为databaseStructureView
编译时类型为 TreeView<Database>
,编译器认为可以安全地假设该树中的所有树项都必须是 TreeItem<Database>
。所以你的updateItem(...)
方法需要 Database
作为参数。但是,由于当前对象是 TreeView<Object>
,编译器将允许您添加任何 TreeView<Object>
到树结构。但由于它们指的是同一件事,当你这样做时
dbRoot.getChildren().add(new TreeItem<Object>(var));
您正在添加 TreeView<Object>
到databaseStructureView
的树结构。当 TreeView 稍后检索包装的对象(实际上是 DatabaseStructure.Variable
)并将其传递给 updateItem(...)
时方法时,您会收到类转换异常,因为它是错误的类型。
这显然非常令人困惑:这种困惑来自于不必要的引用 databaseStructure
这与当前对象确实相同。如果您完全删除该引用,并且 fx:id
属性,那么您只需替换对 databaseStructureView
的任何引用即可与 this
。然后会发生的事情是,您的类型错误将在编译时而不是在运行时被检测到(例如,它不会让您将 Callback<TableView<Database>, TableCell<DataBase>>
设置为单元工厂,因为它现在(正确地)期望Callback<TableView<Object>, TableCell<Object>>
)。
阻止您做您想做的事情的根本问题是 TreeView
中的所有项目必须是同一类型。您正在尝试混合类型:一些 TreeItem
包含 Database
对象,其他包含 Database.Variable
对象。您需要设计对象模型,以便所有内容都具有通用类型,或者将单元实现编写为 TreeCell<Object>
并以某种方式找出在提交编辑时要创建的正确对象。
关于java - 编辑 TreeView 中不包含字符串的项目,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28723389/
我在一页上有两个 Kendo UI TreeViews。例如: var data1 = new kendo.data.HierarchicalDataSource({ data: [
使用 Vuetify 的 TreeView 组件时,我试图能够选择父级 没有 让它也选择所有的后代( child )。我尝试了可选、可激活等的各种组合……但似乎找不到合适的组合。 任何人都有实现这一预
在我的应用程序中,左侧有一个 TreeView,我根据 TreeView 中的选择更新右侧的 Pane 。一个非常直接的场景。当选择为空时,我会在 Pane 中显示一条类似“请进行选择”的消息,即我还
我有一个我自己无法解决的问题。请帮忙。 我有(有条件地): /** @mainpage A @subpage B */ /** @page B @subpage C */ /** @page C */
我制作了一个高度为 40 px 的自定义树单元。 这里的问题披露三角形没有垂直居中对齐。 这里是树单元的代码: public static class TestObjectCell extends A
我正在学习如何使用 kotlin 并已开始使用tornadoFX。我正在阅读该指南以尝试学习它,但是我无法弄清楚“具有不同类型的 TreeView”中的含义。似乎是说我应该使用星形投影,据我所知,当您
如何在 JavaFX 2 TreeView 中过滤节点? 我有一个 TextField,我想根据 TextField 的内容过滤所有节点(例如节点标签)。 谢谢。 最佳答案 这是我编写的可重用的可过滤
我正在通过查询 sharepoint 用户配置文件构建一个 asp.net TreeView 。要选择的帐户名和根节点帐户名正在从查询字符串中读取。 我还需要为树配置可配置的扩展深度。 如果节点属于第
我使用的是 JavaFX 8,目前正在进行一些 GUI 开发。我的 TreeView 有点问题,我需要你的帮助。 您知道在 TreeView 中是否可以只选择标签而不是 TreeCell 的整个宽度吗
我有很多(分层的)数据显示在 TreeView 中(可能是大约 20K 项或更多,包括子项)。我的数据的特殊问题是 TreeView 中显示的每个对象都可以存在于许多 TreeView 项目中。我的意
有没有什么简单的方法可以让 Gtk.Treeview 在编辑时更新它的列? 我基于 Gtk.ListStore 模型构建了 Treeview。我这样初始化单元格: Gtk.CellRendererTe
我开始使用 javafx。我有一个问题。我有一个树 View ,其中节点通过外部命令改变了他的位置,但它只是看不到树。我必须最小化父级并重新展开才能看到效果。 Altem 对该树 View 的任何建议
如何在 Kendo Treeview 中取消选择节点? 我尝试从节点中删除类“k-state-selected”。它工作正常,但有没有直接的方法可以做到这一点。 最佳答案 现在有一种更好的方法可以取消
我有以下情况,我有一个带有许多嵌套子节点的父节点。只有父节点应该有一个复选框,我发现的唯一例子是只有子节点有一个复选框。这可以使用 Kendo 模板吗? http://dojo.telerik.com
我正在尝试向 TreeView 数据项添加一些内联图标,但是 k-template 指令似乎没有呈现任何内容。 我基于在线文档在 http://demos.telerik.com/k
我有一个带有复选框和父节点和子节点的 Kendo Treeview 。 我需要将选中节点的完整层次结构复制到另一个 Treeview 中。 ex - 根节点、父节点和选中的子节点。 下面是我的代码,但
我需要在较大的滚动 Pane 中使用 JavaFX 2.2 TreeView 控件,该滚动 Pane 具有多个不属于 Treeview 的其他元素。问题是 TreeView 有它自己的内置滚动 Pan
当我通过单击 TreeView 节点右侧的加号来展开该节点时,该节点将被选中。我怎样才能避免这种情况?我希望能够在不更改所选节点的情况下展开节点(例如在 RegEdit.exe 中),并且仅在单击节点
我正在尝试设置一个 Treeview 对象,设置节点,然后更新控件以使值具有适当的格式。现在我有以下代码,当我设置一个控件时,它可以工作,但不是来自变量的控件。如何从变量设置本地控件? Private
如何将节点填充到作为另一个 treeview1 实例的 newtreeview1 中?添加到“newtreeview1”的节点应该在 treeview1 的第一个实例中可用? 例如;如果 treevi
我是一名优秀的程序员,十分优秀!