- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我创建了一个包含人员列表的 JTable
。一个 Person 总是可以包含另一个 Person(或者可能是它自己)。因此,用户可以单击该列,然后会出现一个 JComboBox
。此 JComboBox 包含 JTable 的所有人员。
用户完成 JTable 编辑后,可以单击 JButton
“保存”。在此示例中,它将仅将 TableModel 的值打印到控制台。
问题:最后的更改没有“转移”到 TableModel。
例如。用户为每一行选择 Kim。对于他来说,它看起来是正确的,但如果您单击该按钮,您将看到最后一行“Person”列的值仍然是 John。当用户单击另一列时,在更改第三列中的人员后,然后单击按钮,输出是正确的。
我认为我遗漏了一些东西,但我不知道问题出在哪里。
对于 JTable,我使用了本教程: https://www.codejava.net/java-se/swing/how-to-create-jcombobox-cell-editor-for-jtable
这是我的代码:
public class Person {
private String name;
private Person person;
private String job;
public Person(String name, String job) {
this.name = name;
this.job = job;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Person getPerson() {
return person;
}
public void setPerson(Person person) {
this.person = person;
}
public String getJob() {
return job;
}
public void setJob(String job) {
this.job = job;
}
}
public class PersonTable extends JTable {
private PersonTableModel tableModel;
public PersonTable(List<Person> listPerson) {
tableModel = new PersonTableModel(listPerson);
this.setModel(tableModel);
this.setDefaultRenderer(Person.class, new PersonCellRenderer());
this.setDefaultEditor(Person.class, new PersonCellEditor(listPerson));
}
public Set<Person> getAllPersons() {
Set<Person> set = new HashSet<Person>();
for (int i = 0; i < tableModel.getRowCount(); i++) {
set.add(tableModel.getPerson(i));
}
return set;
}
}
public class PersonCellEditor extends AbstractCellEditor
implements TableCellEditor, ActionListener {
private Person Person;
private List<Person> listPerson;
public PersonCellEditor(List<Person> listPerson) {
this.listPerson = listPerson;
}
@Override
public Object getCellEditorValue() {
return this.Person;
}
@Override
public Component getTableCellEditorComponent(JTable table, Object value,
boolean isSelected, int row, int column) {
if (value instanceof Person) {
this.Person = (Person) value;
}
JComboBox<Person> comboPerson = new JComboBox<Person>();
comboPerson.setRenderer(new PersonComboBoxRenderer());
for (Person aPerson : listPerson) {
comboPerson.addItem(aPerson);
}
comboPerson.setSelectedItem(Person);
comboPerson.addActionListener(this);
if (isSelected) {
comboPerson.setBackground(table.getSelectionBackground());
} else {
comboPerson.setBackground(table.getSelectionForeground());
}
return comboPerson;
}
@Override
public void actionPerformed(ActionEvent event) {
JComboBox<Person> comboPerson = (JComboBox<Person>) event.getSource();
this.Person = (Person) comboPerson.getSelectedItem();
}
}
public class PersonCellRenderer extends DefaultTableCellRenderer {
public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column) {
if (value instanceof Person) {
Person Person = (Person) value;
setText(Person.getName());
}
if (isSelected) {
setBackground(table.getSelectionBackground());
} else {
setBackground(table.getSelectionForeground());
}
return this;
}
}
public class PersonTableModel extends AbstractTableModel {
private String[] columnNames = {"No.", "Name", "Person", "Job"};
private List<Person> listPerson = new ArrayList<>();
public PersonTableModel(List<Person> listPerson) {
this.listPerson.addAll(listPerson);
}
@Override
public int getColumnCount() {
return columnNames.length;
}
public String getColumnName(int column) {
return columnNames[column];
}
public Class getColumnClass(int column) {
return getValueAt(0, column).getClass();
}
@Override
public int getRowCount() {
return listPerson.size();
}
@Override
public void setValueAt(Object value, int rowIndex, int columnIndex) {
Person person = listPerson.get(rowIndex);
switch (columnIndex) {
case 1:
person.setName((String) value);
break;
case 2:
person.setPerson((Person) value);
break;
case 3:
person.setJob((String) value);
break;
}
}
@Override
public Object getValueAt(int rowIndex, int columnIndex) {
Object returnValue = null;
Person person = listPerson.get(rowIndex);
switch (columnIndex) {
case 0:
returnValue = rowIndex + 1;
break;
case 1:
returnValue = person.getName();
break;
case 2:
returnValue = person.getPerson();
break;
case 3:
returnValue = person.getJob();
break;
}
return returnValue;
}
public Person getPerson(int row) {
return listPerson.get(row);
}
public boolean isCellEditable(int rowIndex, int columnIndex) {
return columnIndex > 0;
}
}
public class PersonComboBoxRenderer extends DefaultListCellRenderer {
@Override
public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected,
boolean cellHasFocus) {
super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
if(value instanceof Person) {
Person p = (Person) value;
setText(p.getName());
}
return this;
}
}
public class JComboBoxTableCellEditorExample extends JFrame {
public JComboBoxTableCellEditorExample() {
super("JComboBox Cell Editor for JTable Demo");
List<Person> listPerson = new ArrayList<>();
Person p1 = new Person("John", "Developer");
Person p2 = new Person("Kim", "Designer");
Person p3 = new Person("Peter", "Manager");
p1.setPerson(p2);
p2.setPerson(p3);
p3.setPerson(p1);
listPerson.add(p1);
listPerson.add(p2);
listPerson.add(p3);
PersonTable table = new PersonTable(listPerson);
JScrollPane scrollpane = new JScrollPane(table);
scrollpane.setPreferredSize(new Dimension(400, 200));
add(scrollpane, BorderLayout.CENTER);
JButton buttonSave = new JButton("Save");
buttonSave.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
for (Person p : table.getAllPersons()) {
System.out.println(p.getName() + " --> " + p.getPerson().getName());
}
}
});
add(buttonSave, BorderLayout.PAGE_END);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
pack();
setLocationRelativeTo(null);
setVisible(true);
}
public static void main(String[] args) {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (Exception ex) {
ex.printStackTrace();
}
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
new JComboBoxTableCellEditorExample();
}
});
}
}
最佳答案
我已经测试了您的代码并解决了应用 camickr here 解释的解决方案的问题
camickr自己在这个entry中解释了您面临的问题。 。让我引用一段摘录
When editing a cell in a JTable the table doesn’t know when a user isfinished editing the cell. Therefore it is the users responsibility totell the table when to stop editing.
This is normally done by:
- using the enter key
- tabbing to the next cell
- clicking on another cellwith the mouse
如果您编辑第三行 Person [第 2 列] 并立即按“保存”按钮,JTable 不会检测到您已完成对该单元格的编辑。这就是您获得的原因
Kim --> Kim
John --> Kim
Peter --> John
camickr 提供的解决方案之一是当您检测到“保存”按钮已被按下时强制停止编辑。只需更改您的按钮声明
JButton buttonSave = new JButton("Save");
buttonSave.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
if (table.isEditing()) {
table.getCellEditor().stopCellEditing();
}
for (Person p : table.getAllPersons()) {
System.out.println(p.getName() + " --> " + p.getPerson().getName());
}
}
});
现在您应该获得所需的结果。
请注意,如果您稍微更改一下 PersonTable 类,在方法 getAllPersons
中将 Set 替换为 List,您将获得有序结果,从而提高控制台的可读性。
public class PersonTable extends JTable {
private PersonTableModel tableModel;
public PersonTable(List<Person> listPerson) {
tableModel = new PersonTableModel(listPerson);
setModel(tableModel);
setDefaultRenderer(Person.class, new PersonCellRenderer());
setDefaultEditor(Person.class, new PersonCellEditor(listPerson));
}
public List<Person> getAllPersons() {
List<Person> set = new ArrayList<Person>();
for (int i = 0; i < tableModel.getRowCount(); i++) {
set.add(tableModel.getPerson(i));
}
return set;
}
}
使用列表未进行任何更改的控制台输出
John --> Kim
Kim --> Peter
Peter --> John
关于java - JComboBox 并不总是将值保存到 TableModel,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54036283/
我正在尝试组合 2 个 jcomboboxes。 1 组合框用于显示费用类别。第二个组合框正在从文本文件中读取文件以显示产品类型。如果我更改第一个组合框,我希望第二个组合框会根据用户在第一个组合框中选
我希望一个 JComboBox 的值根据另一个 JComboBox 的值进行更改。在一个 JComboBox 中,我显示了从数据库检索的医生名称。根据该名称,我想在另一个 JComboBox 中显示医
我有课 public class Speciality { String specName; String eduProgram; } 和 TreeMap Map> treeM
我有一个 JComboBox。我希望它能够工作,以便如果选择某个项目(“其他”),立即在同一个组合框中显示更多项目(类似于子菜单,但在组合框内)。我很难让这个工作发挥作用。 有人对如何做到这一点有任何
如何从 JComboBox 中进行选择,删除或添加选择到另一个 JComboBox? 更具体地说,我有一个 JComboBox,其中包含一年中的月份作为选择。我还有另一个,里面有这个月的日子。虽然 3
我有一个用例(在 Java Swing 中),其中某个 JTable 列必须可以通过 JComboBox 编辑器进行编辑,但可选择的值列表取决于该行的域对象(我使用自定义 TableModel)。 C
我们可以在执行 jFrame 后根据点击另一个 jcomboBox 的项目来启用 jcomboBox 吗?这可能吗? 最佳答案 JComboBox comboBox_1 = new JComboBox
我有 7 个 jcombobox,它们都以相同的默认选择开头。如果在前面的 jcombobox 中选择了默认值以外的内容,如何启用下一个? if ( ! (custData1.equa
我有 2 个 JComboBox,在第一个 JComboBox 上选择一个项目后,第二个会从数据库中填充。问题是,每次我输入一个字母时,第二个 jcombobox 都会填充。我想让第二个 jcombo
您好,我正在尝试将一个对象从一个组合框传输到另一个组合框,反之亦然。为了实现这一目标,我使用 actionListeners 或 ItemListeners,不幸的是它们没有回答我的问题,或者可能只是
当我从 JComboBox A 中选择项目时,我尝试在 JComboBox B 中显示值。到目前为止,当我从 JComboBox A 中选择一个值时,没有任何反应。这是我的数据和代码。例如,如果我从
我写了这个简单的程序: import java.awt.*; import java.awt.event.*; import javax.swing.*; import javax.swing.tab
我正在制作一个医疗药物软件程序,我希望用户能够从由类别数组填充的 jcombo 框中选择疾病类别(呼吸系统疾病、心血管疾病等)。 选择此类别后,我希望另一个 jcombobox 仅填充与该类别相关的药
这段代码很草率,我也欢迎一些反馈。 我正在尝试根据另一个 JComboBox 的值更改 JComboBox 的值。还有一个额外的复杂性,因为我使用一个额外的类来确定要返回的字符串数组(请参阅我之前的问
我已经加载了对象的 JComboBox,这些对象有自己的“toString”方法。该组合框用于选择并通过该组合框上的“ActionListener”返回该对象。一切工作正常,直到我决定添加通过在此组合
所以我的问题是 JComboBoxs 和 ActionListeners。我将制作一个新的简化代码来尝试代表我的原始代码中的问题。我想要一个 JComboBox 添加一个 JComboBox,然后 J
我的转换器程序中有 3 个 JCombobox,其中一个用于类别,两个用于选择要执行的功能;我希望程序做的是,当用户在一个 JCombobox 中选择一个类别时,两个 JCombobox 将自动更改与
所以,这就是问题所在:我需要 3 个或更多 jComboBox(都具有相同的项目)相互连接,这样当我在任何 jComboBox 中选择一个项目时,该项目将在所有其他 jComboBox 中消失。具体例
我用这样的东西制作了一个图形用户界面: String[] days29 = {"1",....."29"}; String[] days30 = {"1",....."30"}; String[] d
我有一个小应用程序,它通过 JPA 从 MySQL 数据库生成统计图表。为了选择要包含在统计信息中的数据库组件,我安装了 2 个 JComboBoxes。第一个 JComboBox 填充了 Categ
我是一名优秀的程序员,十分优秀!