Два JXTables с той же пользовательской моделью - Как сделать ячейки редактируемыми в одной таблице, но не редактируемыми во второй?
У меня есть пользовательская модель таблицы, данные которой я хочу редактировать в одной JXTable, но только для просмотра во второй JXTable. Можно ли это сделать без двух отдельных моделей? Есть ли какой-нибудь способ переопределения model.isCellEditable для таблицы только для просмотра?
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import javax.swing.*;
import javax.swing.table.AbstractTableModel;
import org.jdesktop.swingx.JXTable;
import org.jdesktop.swingx.decorator.ColorHighlighter;
import org.jdesktop.swingx.decorator.ComponentAdapter;
import org.jdesktop.swingx.decorator.HighlightPredicate;
public class SSCCE extends JPanel {
private JSplitPane splitPane;
private JXTable viewTable, editTable;
private class CustomModel extends AbstractTableModel {
public static final int SPORT_COL = 0;
public static final int EQUIPMENT_COL = 1;
private final String[] COLUMN_NAMES = {
"Sport",
"Equipment"
};
private Map<String, String> sports;
private List<String> set;
public CustomModel() {
sports = new TreeMap<String, String>();
sports.put("Rugby", "Headguard");
sports.put("Hurling", "Sliotar");
sports.put("Tennis", "Racket");
set = new ArrayList<String>(sports.keySet());
}
public int getRowCount() {
return sports.size();
}
public int getColumnCount() {
return COLUMN_NAMES.length;
}
@Override
public String getColumnName(int columnIndex) {
return COLUMN_NAMES[columnIndex];
}
public Object getValueAt(int rowIndex, int columnIndex) {
String sport = set.get(rowIndex);
switch (columnIndex) {
case SPORT_COL:
return sport;
case EQUIPMENT_COL:
return sports.get(sport);
}
return null;
}
@Override
public void setValueAt(Object aValue, int rowIndex, int columnIndex) {
if (columnIndex == EQUIPMENT_COL) {
if (aValue != null) {
String sport = (String) getValueAt(rowIndex, SPORT_COL);
String equip = (String) aValue;
sports.put(sport, equip);
fireTableDataChanged();
}
}
}
@Override
public boolean isCellEditable(int rowIndex, int columnIndex) {
return columnIndex == EQUIPMENT_COL;
}
}
private CustomModel model;
public SSCCE() {
super();
model = new CustomModel();
viewTable = new JXTable(model);
editTable = new JXTable(model);
editTable.addHighlighter(new ColorHighlighter(new HighlightPredicate() {
@Override
public boolean isHighlighted(Component renderer, ComponentAdapter adapter) {
return adapter.isEditable();
}
}, Color.GREEN.brighter(), Color.BLACK));
JPanel panelLeft = new JPanel(new BorderLayout(0, 10));
panelLeft.add(new JLabel("Editable"), BorderLayout.NORTH);
panelLeft.add(new JScrollPane(editTable));
JPanel panelRight = new JPanel(new BorderLayout(0, 10));
panelRight.add(new JLabel("How to make non-editable?"), BorderLayout.NORTH);
panelRight.add(new JScrollPane(viewTable));
splitPane = new JSplitPane(JSplitPane.HORIZONTAL_SPLIT, panelLeft, panelRight);
add(splitPane);
}
public static void main(String args[]) {
JFrame frame = new JFrame("One Model | Two Views");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(new SSCCE(), BorderLayout.CENTER);
frame.setSize(360, 240);
frame.setVisible(true);
}
}
2 ответа
Решение
SwingX поддерживает настройку редактируемости на уровне представления. Не нужно прикасаться к вашей модели, оставьте ее редактируемой соответствующим образом:
viewTable = new JXTable(model);
viewTable.setEditable(false);
editTable = new JXTable(model);
Создать WrapperTableModel
который делегирует все звонки на внутренний TableModel
но возвращает ложь из isCellEditable().
Установите первую редактируемую модель для table1 и модель оболочки (содержащую ссылку на первую) в table2.