我有一个包含6列的JXTable
复合词,其中两列是JCheckBox
。我想有以下行为:
我使用Photoshop编辑了一个图像以显示所需的结果:
对于CheckOne
和CheckTwo
列,我使用自定义TableCellEditor
和TableCellRenderer
:
public class CheckBoxCellEditor extends AbstractCellEditor implements TableCellEditor {
private static final long serialVersionUID = 1L;
private JCheckBox checkBox = new JCheckBox();
public CheckBoxCellEditor() {
checkBox.setHorizontalAlignment(SwingConstants.CENTER);
}
@Override
public Component getTableCellEditorComponent(JTable table,
Object value, boolean isSelected, int row, int column) {
checkBox.setSelected(value==null ? false : (boolean)value);
return checkBox;
}
@Override
public Object getCellEditorValue() {
return checkBox.isSelected();
}
}
public class CheckBoxCellRenderer extends JCheckBox implements TableCellRenderer{
private static final long serialVersionUID = 1L;
public CheckBoxCellRenderer() {
setHorizontalAlignment(SwingConstants.CENTER);
}
@Override
public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column) {
setSelected(value==null ? false : (boolean)value);
return this;
}
}
这就是我设置它们的方式:
//CheckOne
table.getColumn(4).setCellEditor(new CheckBoxCellEditor());
table.getColumn(4).setCellRenderer(new CheckBoxCellRenderer());
//CheckTwo
table.getColumn(5).setCellEditor(new CheckBoxCellEditor());
table.getColumn(5).setCellRenderer(new CheckBoxCellRenderer());
我尝试向PropertyChangeListener
添加JXTable
并从那里实施行为,但我无法完成。
然后我意识到我的自定义编辑器和渲染器正在同时更改所有列组件,而不是仅更改所需的组件。
所以,我尝试在TableCellEditor
和TableCellRenderer
以及PropertyChangeListener
中进行更改,但是,再一次,我无法弄明白。
答案 0 :(得分:3)
首先请注意您的问题与JXTable无关,但与渲染器/编辑器/型号无关。
正如@mKorbel所指出的那样JCheckBox是布尔值的默认渲染器/编辑器,一般来说你不需要重新发明轮子:只需正确覆盖getColumnClass()以在5号和6号返回布尔值列。
但是,由于这个要求:
如果未选中第一个复选框,则第二个复选框必须禁用并取消选中。
这不是默认渲染器的行为,因此您实际上需要自己的渲染器。 但只有渲染器,你不需要编辑器:因为@mKorbel指出你需要对表模型进行一些工作。
class CheckBoxCellRenderer implements TableCellRenderer {
private final JCheckBox renderer;
public CheckBoxCellRenderer() {
renderer = new JCheckBox();
renderer.setHorizontalAlignment(SwingConstants.CENTER);
}
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
Color bg = isSelected
? table.getSelectionBackground() : table.getBackground();
renderer.setBackground(bg);
renderer.setEnabled(table.isCellEditable(row, column));
renderer.setSelected(value != null && (Boolean)value);
return renderer;
}
}
您需要同时使用isCellEditable()和setValueAt()方法,根据第一个布尔值的值正确更新您的第二个布尔值列。例如,请考虑以下代码段:
// A default model with 5 rows and 6 columns
DefaultTableModel model = new DefaultTableModel(5, 6) {
@Override
public Class<?> getColumnClass(int columnIndex) {
switch(columnIndex) {
case 4:
case 5: return Boolean.class; // Both 5th and 6th columns are booleans
}
return super.getColumnClass(columnIndex);
}
@Override
public boolean isCellEditable(int row, int column) {
/*
* In order to know if 6th column is editable, you have to check
* 5th column's value.
*/
if(column == 5) {
Object value = getValueAt(row, 4);
return value != null && (Boolean)value;
}
return super.isCellEditable(row, column);
}
@Override
public void setValueAt(Object aValue, int row, int column) {
/*
* If 5th column value is updated to FALSE, you need to
* set 6th column's value to FALSE as well
*/
if(column == 4) {
super.setValueAt(Boolean.FALSE, row, 5);
}
super.setValueAt(aValue, row, column);
}
};
这是一个完整的测试用例。希望它有所帮助。
import java.awt.Color;
import java.awt.Component;
import javax.swing.JCheckBox;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingConstants;
import javax.swing.SwingUtilities;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableCellRenderer;
/**
* @author dic19
*/
public class Demo {
private void createAndShowGui() {
DefaultTableModel model = new DefaultTableModel(5, 6) {
@Override
public Class<?> getColumnClass(int columnIndex) {
switch(columnIndex) {
case 4:
case 5: return Boolean.class;
}
return super.getColumnClass(columnIndex);
}
@Override
public boolean isCellEditable(int row, int column) {
if(column == 5) {
Object value = getValueAt(row, 4);
return value != null && (Boolean)value;
}
return super.isCellEditable(row, column);
}
@Override
public void setValueAt(Object aValue, int row, int column) {
if(column == 4) {
super.setValueAt(false, row, 5);
}
super.setValueAt(aValue, row, column);
}
};
JTable table = new JTable(model);
table.getDefaultRenderer(null);
table.getColumnModel().getColumn(5).setCellRenderer(new CheckBoxCellRenderer());
JScrollPane scrollPane = new JScrollPane(table);
JFrame frame = new JFrame("Demo");
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
frame.add(scrollPane);
frame.pack();
frame.setLocationByPlatform(true);
frame.setVisible(true);
}
class CheckBoxCellRenderer implements TableCellRenderer {
private final JCheckBox renderer;
public CheckBoxCellRenderer() {
renderer = new JCheckBox();
renderer.setHorizontalAlignment(SwingConstants.CENTER);
}
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
Color bg = isSelected
? table.getSelectionBackground() : table.getBackground();
renderer.setBackground(bg);
renderer.setEnabled(table.isCellEditable(row, column));
renderer.setSelected(value != null && (Boolean)value);
return renderer;
}
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
new Demo().createAndShowGui();
}
});
}
}