JTable autoresize, with a twist?
This is a well known autoresizer for JTable :
public JTable autoResizeColWidth(JTable table, DefaultTableModel model) {
table.setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
table.setModel(model);
int margin = 5;
for (int i = 0; i < table.getColumnCount(); i++) {
int vColIndex = i;
DefaultTableColumnModel colModel = (DefaultTableColumnModel) table.getColumnModel();
TableColumn col = colModel.getColumn(vColIndex);
int width = 0;
// Get width of column header
TableCellRenderer renderer = col.getHeaderRenderer();
if (renderer == null) {
renderer = table.getTableHeader().getDefaultRenderer();
}
Component comp = renderer.getTableCellRenderer开发者_如何转开发Component(table, col.getHeaderValue(), false, false, 0, 0);
width = comp.getPreferredSize().width;
// Get maximum width of column data
for (int r = 0; r < table.getRowCount(); r++) {
renderer = table.getCellRenderer(r, vColIndex);
comp = renderer.getTableCellRendererComponent(table, table.getValueAt(r, vColIndex), false, false,
r, vColIndex);
width = Math.max(width, comp.getPreferredSize().width);
}
// Add margin
width += 2 * margin;
// Set the width
col.setPreferredWidth(width);
}
((DefaultTableCellRenderer) table.getTableHeader().getDefaultRenderer()).setHorizontalAlignment(
SwingConstants.LEFT);
// table.setAutoCreateRowSorter(true);
table.getTableHeader().setReorderingAllowed(false);
return table;
}
The problem here is that the returned table shrinks (in width)
So if my JTable component is 100%, the returned column area may only be 70% (say)
What if i want the autoresize, keeping in mind the original set width of the JTable as a whole. So that even after resizing, all the columns fit in properly.
I have managed to improve the above code in order to consider the table size:
public static void resizeTable(JTable table) {
int margin = 5;
int tableWidth = 0;
for (int i = 0; i < table.getColumnCount(); i++) {
int vColIndex = i;
TableColumn col = table.getColumnModel().getColumn(vColIndex);
int width = 0;
// Get width of column header
TableCellRenderer renderer = col.getHeaderRenderer();
if (renderer == null) {
renderer = table.getTableHeader().getDefaultRenderer();
}
Component comp = renderer.getTableCellRendererComponent(table, col.getHeaderValue(), false, false, 0, 0);
width = comp.getPreferredSize().width;
// Get maximum width of column data
for (int r = 0; r < table.getRowCount(); r++) {
renderer = table.getCellRenderer(r, vColIndex);
comp = renderer.getTableCellRendererComponent(table, table.getValueAt(r, vColIndex), false, false,
r, vColIndex);
width = Math.max(width, comp.getPreferredSize().width);
}
// Add margin
width += 2 * margin;
// Set the width
col.setPreferredWidth(width);
tableWidth += width;
}
if (tableWidth < table.getParent().getWidth()) {
tableWidth = table.getParent().getWidth();
for (int i = 0; i < table.getColumnCount(); i++) {
int vColIndex = i;
TableColumn col = table.getColumnModel().getColumn(vColIndex);
int width = 0;
// Get width of column header
TableCellRenderer renderer = col.getHeaderRenderer();
if (renderer == null) {
renderer = table.getTableHeader().getDefaultRenderer();
}
Component comp = renderer.getTableCellRendererComponent(table, col.getHeaderValue(), false, false, 0, 0);
width = comp.getPreferredSize().width;
// Get maximum width of column data
for (int r = 0; r < table.getRowCount(); r++) {
renderer = table.getCellRenderer(r, vColIndex);
comp = renderer.getTableCellRendererComponent(table, table.getValueAt(r, vColIndex), false, false,
r, vColIndex);
width = Math.max(width, comp.getPreferredSize().width);
}
// Add margin
width += 2 * margin;
// Set the width based on table width
if (width < tableWidth / (table.getColumnCount() - i)) {
width = tableWidth / (table.getColumnCount() - i);
}
tableWidth -= width;
col.setPreferredWidth(width);
}
}
}
Note that the loop in table data is executed twice. Also, I prefer the function to be independent of JTable instance to be able to call it from any JTable in my projects.
精彩评论