从 TableView 中的可编辑列动态填充 ChoiceBox

2024-02-19

基本上问题标题说明了一切。我有一列字符串(称为type)在表格视图和相应的数字列(称为size), 每行代表一个对象CargoItem,它有两个属性type and size。 两列都是可编辑的。 TableView 与相应的可观察列表相关联CargoItem,称为cargoList.

ChoiceBox 最初设置为可观察列表。我有一个附加到 ChoiceBox 的侦听器,因此它可以使用相应的大小属性更新 TextField。这可以正常工作:如果您在大小列中的单元格中键入新值,则 TextField 中的相应值将更新(选择 ChoiceBox 项后)。

目前,如果我编辑一个单元格type列中,ChoiceBox 中的项目不会更新。我有一个诊断按钮,它将验证可观察列表是否正确更新。我想我需要将侦听器放入类型列的单元格中,但不确定如何操作。

以下是一个最小的工作示例,有 4 个文件:Table_test02.fxml、CargoItem.java、ControllerTest.java, and 测试应用程序.java

表_test02.fxml:

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.text.*?>
<?import javafx.scene.control.*?>
<?import java.lang.*?>
<?import javafx.scene.layout.*?>

<AnchorPane fx:id="root" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="252.0" prefWidth="232.0" xmlns="http://javafx.com/javafx/8" xmlns:fx="http://javafx.com/fxml/1" fx:controller="so_question01.ControllerTest">
   <children>
      <TableView fx:id="cargoTable" editable="true" layoutX="9.0" layoutY="30.0" prefHeight="107.0" prefWidth="213.0">
         <columnResizePolicy>
            <TableView fx:constant="CONSTRAINED_RESIZE_POLICY" />
         </columnResizePolicy>
         <columns>
            <TableColumn fx:id="cargoTableTypeCol" maxWidth="130.0" minWidth="60.0" prefWidth="75.0" sortable="false" text="Type" />
            <TableColumn fx:id="cargoTableSizeCol" maxWidth="118.0" minWidth="60.0" prefWidth="67.0" sortable="false" text="Size" />
         </columns>
      </TableView>
      <Label layoutX="10.0" layoutY="11.0" text="Cargo">
         <font>
            <Font name="System Bold" size="12.0" />
         </font>
      </Label>
      <ChoiceBox fx:id="externalChoiceBox" layoutX="9.0" layoutY="162.0" prefHeight="25.0" prefWidth="99.0" />
      <TextField fx:id="externalTextField" layoutX="123.0" layoutY="162.0" prefHeight="25.0" prefWidth="99.0" />
      <Button fx:id="diagnosticButton" layoutX="83.0" layoutY="210.0" mnemonicParsing="false" text="Diagnostic" textFill="#862828">
         <font>
            <Font name="Arial Black" size="11.0" />
         </font>
      </Button>
      <Label layoutX="10.0" layoutY="143.0" text="Type">
         <font>
            <Font name="System Bold" size="12.0" />
         </font>
      </Label>
      <Label layoutX="123.0" layoutY="144.0" text="Size">
         <font>
            <Font name="System Bold" size="12.0" />
         </font>
      </Label>
   </children>
</AnchorPane>

CargoItem.java:

package so_question01;

import javafx.beans.property.*;

public class CargoItem {

   private final StringProperty type;
   private final FloatProperty size;

   public CargoItem() {
      this.type = new SimpleStringProperty("");
      this.size = new SimpleFloatProperty(0f);
   }

   public CargoItem(String type, float size) {
      this.type = new SimpleStringProperty(type);
      this.size = new SimpleFloatProperty(size);
   }

   public void setType(String type) {
      this.type.set(type);
   }

   public void setSize(float size) {
      this.size.set(size);
   }

   public String getType() {
      if (!(type.get().equals("") || type.get() == null )) {
         return type.get();
      }
      return "";
   }

   public float getSize() {
      return size.get();
   }

   public StringProperty typeProperty() {
      return type;
   }

   public FloatProperty sizeProperty() {
      return size;
   }

   public void clear() {
      this.type.set("");
      this.size.set(0);
   }

   @Override
   public String toString() {  
      return type.get();     
   }

   public String display() {
      return "\nCARGO ITEM:"
              + "\n\ttype: " + type.get()
              + "\n\tsize: " + size.get();
   }
}

控制器测试.java:

package so_question01;

import java.text.DecimalFormat;
import javafx.collections.ListChangeListener.*;
import javafx.collections.ObservableList;
import javafx.collections.FXCollections;
import javafx.collections.*;
import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.ChoiceBox;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.TextField;
import javafx.scene.control.cell.*;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.beans.value.ObservableValue;
import javafx.util.StringConverter;
import javafx.beans.value.*;

public class ControllerTest {

    ObservableList<CargoItem> cargoList = FXCollections.observableArrayList();
   private TestApp mainApp;   // Reference to the main application.

   @FXML private ChoiceBox externalChoiceBox;
   @FXML private TextField externalTextField;
   @FXML private Button diagnosticButton;
    @FXML   private TableView<CargoItem> cargoTable;
    @FXML   private TableColumn<CargoItem, Float> cargoTableSizeCol;
    @FXML   private TableColumn<CargoItem, String> cargoTableTypeCol;

    public ControllerTest() {  }

    @FXML
    private void initialize() {

      //*****DIAGNOSTIC ONLY*******************//
      diagnosticButton.setOnAction(e->{
         for(CargoItem c: cargoList){
            System.out.print(c.display());
         }
      });
      //does not work when changes made via table editing, but may be useful when new list loaded from file?
      cargoList.addListener(new ListChangeListener< CargoItem>(){
         public void onChanged(Change<? extends CargoItem> c){
             // Do your changes here
            System.out.println(c.getList()); 
         }});

      //**** test choicebox***//
      StringConverter cargoItemConverter= new CargoItemStringConverter();
      externalChoiceBox.setConverter(cargoItemConverter);

      //***************** CARGO TABLE  **************//
      cargoTable.setEditable(true);
      // Initialize the cargo table with the 2 columns.
        cargoTableTypeCol.setCellFactory(TextFieldTableCell.forTableColumn());
        cargoTableTypeCol.setCellValueFactory(new PropertyValueFactory<>("type"));

        cargoTableSizeCol.setCellFactory(TextFieldTableCell.forTableColumn(new FloatIntegerStringConverter()));
        cargoTableSizeCol.setCellValueFactory(new PropertyValueFactory<>("size"));
    }

    /**
     * Is called by the main application to give a reference back to itself.
     *
     * @param mainApp
     */
    public void setMainApp(TestApp mainApp) {
        this.mainApp = mainApp;   
      cargoList.setAll(mainApp.getCargoData());

        // Add observable list data to the table
        cargoTable.setItems(cargoList);
        cargoTable.getSelectionModel().setCellSelectionEnabled(true);

      externalChoiceBox.setItems(cargoList); 
      //adds listener to choicebox selection, so size is rendered to text field
      externalChoiceBox.getSelectionModel().selectedItemProperty().addListener(
         new ChangeListener<CargoItem>() {
             @Override public void changed(ObservableValue<? extends CargoItem> observableValue, CargoItem oldChoice, CargoItem newChoice) {
                externalTextField.setText(new FloatIntegerStringConverter().toString(newChoice.getSize()));
            }
         });
    }
}

//*******string converter classes, nothing to see here************//
class CargoItemStringConverter extends StringConverter<CargoItem>{
   @Override
   public String toString(CargoItem c){
      return c.toString();         
   }
   @Override
   public CargoItem fromString(String type){
      return null;
   }
}

class FloatIntegerStringConverter extends StringConverter<Float> 
{
    DecimalFormat decimalFormat = new DecimalFormat("##,###,##0");
    private Float returnVal;

   @Override
   public Float fromString(String value) {       
      if (value == null) {
         return null;
      }
      value = value.trim();
      String digits = value.replaceAll("[^0-9.]", ""); //gets rid of non-numerics

      if (digits.length() < 1) {
         return null;
      }
      try{  //avoids exception by converting invalid strings to zero (which should just be redundant due to regexp above)
      returnVal=  Float.valueOf(Math.round(Float.valueOf(digits)));//have to do this so the float value coresponds to the rounded value in table
      } catch(NumberFormatException e) { returnVal= Float.valueOf(0); }
      return  returnVal;             
    }

    @Override 
    public String toString(Float value) {
      if (value == null) {  // If the specified value is null, return a zero-length String
         return "";
      }
      return decimalFormat.format(value)+" MT"; //truncated integer representation, unit metric tonnes
   }
}

测试应用程序.java:

package so_question01;

import java.io.*;
import javafx.fxml.FXMLLoader;
import javafx.scene.layout.*;
import javafx.collections.ObservableList;
import javafx.collections.FXCollections;
import javafx.application.Application;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class TestApp extends Application {

    private Stage primaryStage;
   private  ObservableList<CargoItem> cargoData = FXCollections.observableArrayList(); 

   public TestApp() {
      // create some sample data
        CargoItem cargoInstance0 = new CargoItem("Toxic waste", 5000f);
        CargoItem cargoInstance1 = new CargoItem("Cement", 10000f);
        CargoItem cargoInstance2 = new CargoItem("Wheat", 20000f);
        cargoData.add(cargoInstance0);
        cargoData.add(cargoInstance1);
        cargoData.add(cargoInstance2);
   }

    /**
     * Returns the data as observable lists of objects. 
     * @return
     */
    public ObservableList<CargoItem> getCargoData() {return cargoData; }   //called in the controller

   @Override
   public void start(Stage primaryStage) { 
        this.primaryStage = primaryStage;
      this.primaryStage.setTitle("**** TEST ****");   
      show();
    }

      /**
     * Initializes the root layout.
     */
    public void show() {
        try {
            // Load root layout from fxml file.
            FXMLLoader loader = new FXMLLoader();
            loader.setLocation(TestApp.class.getResource("Table_test02.fxml"));
            AnchorPane root = (AnchorPane) loader.load();

            // Give the controller access to the main app.
            ControllerTest controller = loader.getController();
            controller.setMainApp(this);

            // Show the scene containing the root layout.
            Scene scene = new Scene(root);
            primaryStage.setScene(scene);
            primaryStage.show();
        } catch (IOException e) { System.out.println(e);}
    } 

   /**
     * Returns the main stage.
     * @return
     */
    public Stage getPrimaryStage() {
        return primaryStage;
    }

    public static void main(String[] args) { 
        Application.launch(TestApp.class, (java.lang.String[])null);
    }   
}

假设我正确理解了这个问题,你需要初始化你的cargoListextractor http://docs.oracle.com/javase/8/javafx/api/javafx/collections/FXCollections.html#observableArrayList-javafx.util.Callback-:

ObservableList<CargoItem> cargoList = FXCollections.observableArrayList(cargo -> 
    new Observable[]{cargo.typeProperty(), cargo.sizeProperty()});

ChoiceBox历史上无法很好地处理其基础列表的更新通知;我不确定最新版本是否已修复所有问题。您可能需要考虑切换到ComboBox反而。

Update

如果您需要使用ChoiceBox,解决方法可能是不直接将项目设置为cargoList,但要随时更新项目cargoList变化。您可以通过进行以下更改来做到这一点setMainApp(...):

//  externalChoiceBox.setItems(cargoList); 
    externalChoiceBox.getItems().setAll(cargoList);

并在列表更改侦听器中添加以下内容:

  cargoList.addListener(new ListChangeListener< CargoItem>(){
     public void onChanged(Change<? extends CargoItem> c){
         // Do your changes here
        System.out.println(c.getList()); 
        externalChoiceBox.getItems().setAll(cargoList);
     }});
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

从 TableView 中的可编辑列动态填充 ChoiceBox 的相关文章

随机推荐

  • Prolog 中的递归乘法

    序言新手 编辑 使用 swi prolog 我想递归地执行乘法方法在序言中已经执行的操作 而不实际使用乘法方法 我想要实现的算法看起来像 multn N1 N2 output if n2 lt 0 return output else mu
  • Ruby 和 Ruby on Rails 之间有什么区别?

    这几天我一直在研究 Ruby 我注意到有 Ruby on Rails 两者有什么区别 他们一起工作吗 语法一样吗 Ruby 是一种编程语言 Ruby on Rails RoR 是一个用 Ruby 实现的 Web 应用程序框架 因此 它们不仅
  • 使用 SQL 后端时维护表单中的自动编号功能

    目前 我正在尝试将数据库从 Access 后端迁移到 SQL 后端 同时仍保留 Access 表单作为前端 这是通过带有 ODBC 连接到我的后端的链接表来完成的 该表单旨在向表中添加新记录 问题出在自动编号上 ID 设置为自动编号 并且在
  • Appium - 在 PageFactory 获取 java.lang.RuntimeException:java.lang.NoSuchMethodException:jdk.proxy2.$Proxy9.proxyClassLookup()

    我在运行 Appium 测试时遇到此异常 PageFactory 上的测试失败 但出现以下异常 My POM
  • ggplot2:使用循环在一页上打印多个绘图

    我有几个主题需要生成一个绘图 因为我有很多主题 我希望在一页上有多个绘图 而不是一个主题图 这是我到目前为止所做的 读取带有主题名称的txt文件 subjs lt scan ListSubjs txt what 创建一个列表来保存绘图对象
  • 膨胀类 android.support.v7.widget.RecyclerView 时出错

    我正在尝试在现有项目中使用 RecyclerView 构建时没有错误 但在膨胀时没有发现 RecyclerView 的类错误 看不到我做错了什么 感谢您的帮助 activity main xml
  • 使用 Paperclip 和 Amazon s3 上传照片时缺少凭据

    我在使用 Paperclip 将照片上传到 S3 时遇到问题 照片正确下载到网站 但尝试上传新照片时会出现 凭据丢失 消息 我尝试重置 S3 密钥 尝试更改编码中变量的访问权限 但没有成功 我几乎可以肯定这是照片问题 因为如果我提交没有照片
  • 使用 jquery 延迟对象链接多个 ajax 调用

    在寻找类似于此处描述的解决方案时 如何使用 jquery 链接 ajax 调用 https stackoverflow com questions 8612894 how to chain ajax calls using jquery l
  • 覆盖CSS属性全部:未设置

    对于我正在开发的 CSS 框架 我正在使用all unset 它本身工作得很好 foo all unset 但是 在某些情况下 我想 撤消 此规则的效果 如 foo hover all auto 然而 这显然行不通 因为没有值auto fo
  • Git 工作流程最佳实践

    愚蠢的问题 把我当作版本控制的完全新手 我是 Git 新手 我以前使用过 subversion 但只是基础知识 我了解 Git 及其分支命令的基础知识 我有一个想象的情况需要您的建议 假设我的软件当前版本是 v1 2 稳定且已发布 My S
  • 在 Visual Studio Code 中运行 python 脚本;如何让“input()”工作?

    我试图抓住一个简单的input using input 但是当我在其中运行脚本时视觉代码每当该行代码被执行时 程序就会挂起 我如何在其中运行代码视觉工作室代码并使用input task version 0 1 0 command pytho
  • 键入时禁用 Visual Studio Code 中的 linting

    前提 当我使用 Visual Studio Code 时 我希望在保存时进行 linting 并且我喜欢智能感知 尽管如此 我对 linting 感到非常恼火 又名错误和警告消息 形式为 ts jshint 这里你缺少一个分号 当我打字的时
  • EF 4.1 Code First 的复合密钥

    我试图弄清楚如何使用 EF code First 4 1 RC 拥有复合密钥 目前 我正在使用 Key 数据注释 但我无法指定多个键 如何指定复合键 这是我的例子 public class ActivityType Key public i
  • java.util.Scanner 读取剩余内容

    Using a java util Scanner例如 它是否可以返回所有剩余内容 我的情况是 一旦我阅读了一个中的多个字段String 我只想得到剩下的东西并将其存储在其他地方 例如我的内容可能包括 1 2 Some Garbage wi
  • Intellij IDEA 正则表达式字符类不能在字符范围内使用

    IDEA 不允许出现此错误 并且我无法找到关闭此类错误的选项 有谁知道如何修复错误或关闭警告 javascript 工作正常 只有 IDEA 认为这是一个问题 您正在使用以下命令创建一个范围连字符 在你的角色课程中 您应该将其移至任一端 另
  • 无符号 32 位整数的水平最小值和 SSE 中的位置

    我正在寻找一种方法来查找无符号 32 位整数的最小值及其在 SSE 中的位置 类似于 mm minpos epu16 我知道我可以通过一系列 mm min epu32 和洗牌 移位找到最小值 但这并不能让我找到位置 有人有任何很酷的方法吗
  • 使 SKScene 的背景透明不起作用...这是一个错误吗?

    有没有一种方法可以使 SKScene 的背景透明 并通过透明度将该场景呈现在另一个场景上 这个想法是让所呈现的场景的背景像这样 self backgroundColor SKColor colorWithRed 0 0f green 0 0
  • 将 HTML 表格粘贴到 Excel 中,如何在单元格中保留换行符

    我有一个简单的html表格 例如 只有一个单元格 但是当我复制dom节点并将其粘贴到excel中时 它将被识别为两行 如何使Excel获取正确的粘贴数据 table tr td 1 br 2 td tr tr tr table 我尝试添加c
  • Angular2 ngFor 跳过第一个索引[重复]

    这个问题在这里已经有答案了 如何跳过数组中的第一个索引 li user name is user age years old li 你可以使用片管 https angular io docs ts latest api common ind
  • 从 TableView 中的可编辑列动态填充 ChoiceBox

    基本上问题标题说明了一切 我有一列字符串 称为type 在表格视图和相应的数字列 称为size 每行代表一个对象CargoItem 它有两个属性type and size 两列都是可编辑的 TableView 与相应的可观察列表相关联Car