setCellRenderer()不起作用

时间:2013-10-26 14:04:32

标签: java swing jtable tablecellrenderer abstracttablemodel

我正在尝试为特定的列设置渲染器,但不知何故,此渲染器未用于渲染该列。对此有任何解释吗?

tabledata = new LendDataTable();
table.setModel(tabledata);
TableColumn xx = table.getColumnModel().getColumn(3);
xx.setCellRenderer(new BookBackRenderer());//here it doesn't (there are 7 rows in total)
table.setDefaultRenderer(Integer.class, new BookBackRenderer());// here it works
add(table, BorderLayout.CENTER);



public  class BookBackRenderer extends DefaultTableCellRenderer {

/**
 * 
 */
private static final long serialVersionUID = 1L;

public BookBackRenderer() {
    // TODO Auto-generated constructor stub

}

@Override
public Component getTableCellRendererComponent(JTable table, Object value,
        boolean isSelected, boolean hasFocus, int row, int column) {


    System.out.println(column);
    return super.getTableCellRendererComponent(table, value.toString() + "xy", isSelected, hasFocus, row, column);
}

}

问题似乎是对TableModel的影响。一旦我使用DefaultTableModel,它就可以完美地为我的表模型编码:

/**
 * 
 */
package client.gui;

import java.awt.Component;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.EventObject;
import java.util.LinkedList;
import java.util.List;

import javax.swing.JButton;
import javax.swing.JLabel;
import javax.swing.JTable;
import javax.swing.event.CellEditorListener;
import javax.swing.table.AbstractTableModel;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.TableCellEditor;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableModel;
import javax.swing.text.DateFormatter;

import org.json.JSONArray;

import lbvs.Leiheintrag;

/**
 * @author John
 *
 */
public class LendDataTable extends AbstractTableModel  implements
TableModel{
    private List<Leiheintrag> lendlist;

    /**
     * 
     */
    private static final long serialVersionUID = 1L;

    /**
     * 
     */
    public LendDataTable() {
        // TODO Auto-generated constructor stub
        lendlist = new LinkedList<Leiheintrag> ();
    }
    public void setData (List<Leiheintrag> list)
    {
        lendlist = list;
        this.fireTableStructureChanged();
        this.fireTableDataChanged();



    }

    /* (non-Javadoc)
     * @see javax.swing.table.TableModel#getRowCount()
     */
    @Override
    public int getRowCount() {
        // TODO Auto-generated method stub
        return lendlist.size();
    }

    /* (non-Javadoc)
     * @see javax.swing.table.TableModel#getColumnCount()
     */
    @Override
    public int getColumnCount() {
        // TODO Auto-generated method stub
        return 7;
    }

    /* (non-Javadoc)
     * @see javax.swing.table.TableModel#getValueAt(int, int)
     */
    @Override
    public Object getValueAt(int rowIndex, int columnIndex) {
        // TODO Auto-generated method stub
        Leiheintrag eintr = lendlist.get(rowIndex);
        switch (columnIndex){
        case 0:
            return eintr.getBuch();
        case 1:
            return eintr.getLeihdatum();
        case 2:
            return eintr.getAbgabe_bis();
        case 3:
            return ((Long)eintr.getRueck_datum());
        case 4:
            return eintr.getBezahlt_am();
        case 5:
            return eintr.getKosten();
        case 6 :
            return eintr.getLast_edit_user();
         default:
            return null;


        }
    }
    public Class<?> getColumnClass(int columnIndex){
        switch (columnIndex){
        case 0:
            return Integer.class;
        case 1:
            return Long.class;
        case 2:
            return Long.class;
        case 3:
            return Long.class;
        case 4:
            return Long.class;
        case 5:
            return Float.class;
        case 6 :
            return Integer.class;
         default:
            return null;


        }
    }
}

谢谢你的帮助

2 个答案:

答案 0 :(得分:4)

  

问题似乎是对TableModel的影响。一旦我使用DefaultTableModel,它就可以完美地运行

以下方法似乎是问题所在。

public void setData (List<Leiheintrag> list)
{
    lendlist = list;
    this.fireTableStructureChanged();
    this.fireTableDataChanged();
}

当您调用fireTableStructureChange()时,JTable将重新创建TableColumnModel(以及所有TableColumns),这意味着您的渲染器将不再与TableColumn 3相关联。

我认为您可以使用fireTableDataChanged(),或者如果不起作用则使用fireTableRowsInserted()

如果您愿意,可以查看DefaultTableModel的源代码,了解setDataVector()方法调用的内容,因为两个模型的概念相同。

答案 1 :(得分:1)

@camickr所说的,显然是true。但我认为有一件重要的事情需要提及。

fireTableStructureChanged的JAVA DOC明确指出:

如果设置了 JTable.setModel(model) ,则event通过fireTableStructureChanged来电或在autoCreateColumnsFromModel函数调用时触发true <{1>} JTablefireTableStructureChanged()将丢弃它拥有的任何表列,并按照它们在模型中出现的顺序重新分配默认列。我认为将autoCreateColumnsFromModel标记设置为false就足够了,而不是删除JTable.setAutoCreateColumnsFromModel(boolean) 调用。要设置或取消设置此标志,请使用:

{{1}}