JXTreeTable: как использовать ComponentProvider для установки средства визуализации для одного столбца
Я использую JXTreeTable для отображения некоторых данных, и я хочу использовать предоставленные механизмы SwingX для изменения рендерера для некоторых столбцов.
Ранее я использовал JXTable и пользовательские реализации TableCellRenderer, но это больше не работает (я вижу строки, в которых должны быть индикаторы выполнения, кнопки,...).
Таким образом я пытался достичь того, чего хочу, делая:
examsTable.getColumn(6).setCellRenderer(new DefaultTableRenderer(new ButtonProvider()));
Но переопределенный метод createRenderer
ComponentProvider вызывается один раз (даже если в моем JXTreeTable несколько строк), и кнопка не отображается (метод содержит только return new JButton();
).
Спасибо!
Правка> Надеюсь, что вы ответите Клеопатра и, таким образом, счастлив, что вы сделали.
Я сделал все возможное, но почему-то таблица не отображается. Наверное, я что-то забыл (я новичок в Java на C++), но это не серьезная проблема, и, вероятно, она не связана с моей основной проблемой.
import java.awt.Dimension;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import org.jdesktop.swingx.JXTreeTable;
import org.jdesktop.swingx.decorator.HighlighterFactory;
import org.jdesktop.swingx.renderer.CellContext;
import org.jdesktop.swingx.renderer.ComponentProvider;
import org.jdesktop.swingx.renderer.DefaultTableRenderer;
import org.jdesktop.swingx.treetable.AbstractTreeTableModel;
public class Test extends JFrame {
JXTreeTable table = new JXTreeTable();
JPanel panel = new JPanel();
public Test() {
setMinimumSize(new Dimension(400, 400));
table.setEditable(false);
table.setDragEnabled(false);
table.setColumnSelectionAllowed(false);
table.setHighlighters(HighlighterFactory.createAlternateStriping());
table.setRowHeight(20);
table.setMinimumSize(new Dimension(200, 200));
class Series {
public String seriesInstanceUID;
public String patientName;
public String patientBirthDate;
public String securityToken;
public Series(String seriesInstanceUID, String patientName, String patientBirthDate, String securityToken) {
this.seriesInstanceUID = seriesInstanceUID;
this.patientName = patientName;
this.patientBirthDate = patientBirthDate;
this.securityToken = securityToken;
}
}
class Study {
public List<Series> series = new ArrayList<Series>();
}
class Root {
public List<Study> studies = new ArrayList<Study>();
}
AbstractTreeTableModel model = new AbstractTreeTableModel() {
Root root = new Root() {
{
studies.add(new Study() {
{
series.add(new Series("Series 1.1", "Mr. X", "1988-10-23", "sec-xx-1"));
series.add(new Series("Series 1.2", "Mr. X", "1988-10-23", "sec-xx-2"));
series.add(new Series("Series 1.3", "Mr. X", "1988-10-23", "sec-xx-3"));
}
});
studies.add(new Study() {
{
series.add(new Series("Series 2.1", "Mrs. Y", "1960-02-11", "sec-yy-1"));
}
});
studies.add(new Study() {
{
series.add(new Series("Series 3.1", "HAL 9000", "1975-04-21", "sec-zz-1"));
series.add(new Series("Series 3.2", "HAL 9000", "1975-04-21", "sec-zz-2"));
}
});
}
};
@Override
public String getColumnName(int columnIndex) {
switch (columnIndex) {
case 0:
return "Series Instance UID";
case 1:
return "Patient Name";
case 2:
return "Patient Birth Date";
case 3:
return "View online";
default:
return "";
}
}
@Override
public int getIndexOfChild(Object parent, Object child) {
if (parent == root) {
return root.studies.indexOf(child);
}
if (parent instanceof Study) {
Study study = (Study) parent;
return study.series.indexOf(child);
}
return -1;
}
@Override
public int getChildCount(Object parent) {
if (parent == root) {
return root.studies.size();
}
if (parent instanceof Study) {
Study study = (Study) parent;
return study.series.size();
}
return 0;
}
@Override
public Object getChild(Object parent, int index) {
if (parent == root) {
return root.studies.get(index);
}
if (parent instanceof Study) {
Study study = (Study) parent;
return study.series.get(index);
}
return null;
}
@Override
public Object getValueAt(Object node, int columnIndex) {
if (!(node instanceof Series) && !(node instanceof Study))
return null;
if (columnIndex < 0 || columnIndex >= getColumnCount())
return null;
if (root == null)
return null;
if (node instanceof Series) {
Series series = (Series) node;
if (columnIndex == 0)
return series.seriesInstanceUID;
else if (columnIndex == 1)
return series.patientName;
else if (columnIndex == 2)
return series.patientBirthDate;
else if (columnIndex == 3)
return series.securityToken;
} else if (node instanceof Study) {
// Empty for now
}
return null;
}
@Override
public int getColumnCount() {
return 4;
}
@Override
public Object getRoot() {
return root;
}
public void update() {
modelSupport.fireNewRoot();
}
};
table.setTreeTableModel(model);
table.getColumnModel().getColumn(3).setCellRenderer(new DefaultTableRenderer(new ComponentProvider<JButton>() {
{
rendererComponent.setHorizontalAlignment(JButton.CENTER);
}
@Override
protected void format(CellContext context) {
rendererComponent.setText(getValueAsString(context));
}
@Override
protected void configureState(CellContext context) {
rendererComponent.setHorizontalAlignment(getHorizontalAlignment());
}
@Override
protected JButton createRendererComponent() {
return new JButton("View online");
}
}));
panel.add(table);
this.setContentPane(panel);
}
public static void main(String[] args) {
(new Test()).setVisible(true);
}
}
РЕДАКТИРОВАТЬ> Я на самом деле заметил, что это работает. Но не так, как я хочу. Я хочу видеть реальную кнопку (теперь она выглядит немного иначе, чем остальная часть строки), знать путь к ней (индекс родительского объекта и столбца) и видеть эффекты при нажатии или наведении на нее (кнопка должна выглядеть нажатой,...).
Как мне этого добиться?
1 ответ
Кнопка используется, она появляется только в некоторых LAF:-)
Технически причины
- маркер устанавливает фон кнопки
- визуальные элементы по умолчанию (используемые поставщиком) устанавливают границу по умолчанию, возвращаемую cellContext
В сочетании кнопка не похожа на кнопку для металла (в то время как в Windows она неизменна)
Нет удовлетворительного решения, варианты
- не использовать чередование
- пусть у кнопки есть свойство dont-touch-my-background, расширьте ColorHighlighter, чтобы учесть это, и используйте этот пользовательский подсветчик для чередования (в тестовом пакете есть пример)