I need to get an appropriate table size according to a text in a header. It contains abbreviations of czech names of days like: "Po", "Út", "St", atc. but instead of this three dots are displayed.
I have this code:
width
, height
- max of all minimum row/column sizes
allWidth
,allHeight
- should be the total width, height of the whole table
//GET DIMENSION
int width = 0;
int height = 0;
int allWidth, allHeight;
for (int col = 0; col < table.getColumnCount(); col++) {
TableColumn tableColumn = table.getTableHeader().getColumnModel().getColumn(col);
TableCellRenderer renderer = tableColumn.getHeaderRenderer();
if (renderer == null) {
renderer = table.getTableHeader().getDefaultRenderer();
}
Component component = renderer.getTableCellRendererComponent(table,
tableColumn.getHeaderValue(), false, false, -1, col);
width = Math.max(component.getPreferredSize().width, width);
table.getColumnModel().getColumn(col).setPreferredWidth(width);
}
allWidth = table.getColumnCount() * width;
for (int row = 0; row < table.getRowCount(); row++) {
TableCellRenderer renderer = table.getCellRenderer(row, 0);
Component comp = table.prepareRenderer(renderer, row, 0);
height = Math.max(comp.getMinimumSize().height, height);
}
allHeight = table.getRowCount() * height;
//HERE I SET WIDTHS AND HEIGHTS
table.setRowHeight(height);
table.setMinimumSize(new Dimension(allWidth, allHeight));
scrollPane.setMinimumSize(new Dimension(allWidth, allHeight));
I'm using GridBagLayout with this settings, I think this is ok:
this.setBorder(BorderFactory.createTitledBorder("Calendar"));
gbc.fill = GridBagConstraints.NONE;
gbc.gridx = 0;
gbc.gridy = 1; //position x=0, y=0 is for JSpinners, months and years
gbc.weighty = 0;
gbc.weightx = 0;
add(scrollPane,gbc);//add table in a custom Calendar component
Also I changed the look and feel and this is causing those three dots. Otherwise the names are correctly displayed.
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException e) {
}
And this is how I initialize JTable and JScrollPane, I have a feeling like this could be ok too.
setTable(new JTable());
getTable().setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
getTable().setAutoscrolls(false);
getTable().getTableHeader().setResizingAllowed(false);
getTable().getTableHeader().setReorderingAllowed(false);
getTable().setColumnSelectionAllowed(true);
getTable().setRowSelectionAllowed(true);
getTable().setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
tableModel = new CalendarTableModel(); //my class extended from AbstractTableModel
getTable().setModel(tableModel);
scrollPane = new JScrollPane(table,JScrollPane.VERTICAL_SCROLLBAR_NEVER,JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
scrollPane.setAutoscrolls(false);
Problems:
If I don't set the scrollPane with gbc to resize itself, it gets the minimum possible size unless I calculate and set the minimum sizes of table
and scrollPane
by hand. (the left picture)
allHeight = (1+table.getRowCount()+pady) * height; //plus one line of header
allWidth = table.getColumnCount() * width;
I got the picture on the right.
Here is a simple complete compilable class that demonstrates the problem:
import java.awt.*;
import javax.swing.*;
import javax.swing.table.*;
public class Main {
public static void main(String[] args) {
new Main().start();
}
private void start() {
JFrame f = new JFrame();
f.setLayout(new GridBagLayout());
GridBagConstraints gbc = new GridBagConstraints();
initTable();
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException e) {
}
JPanel calendar = new JPanel();
calendar.setLayout(new GridBagLayout());
calendar.setBorder(BorderFactory.createTitledBorder("Calendar"));
gbc.fill = GridBagConstraints.NONE;
gbc.gridx = 0;
gbc.gridy = 0;
gbc.weighty = 0;
gbc.weightx = 0;
calendar.add(scrollPane, gbc);
f.add(calendar,gbc);
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.setVisible(true);
}
JTable table;
JScrollPane scrollPane;
CalendarTableModel tableModel;
private void initTable() {
setTable(new JTable());
getTable().setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
getTable().setAutoscrolls(false);
getTable().getTableHeader().setResizingAllowed(false);
getTable().getTableHeader().setReorderingAllowed(false);
getTable().setColumnSelectionAllowed(true);
getTable().setRowSelectionAllowed(true);
getTable().setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
tableModel = new CalendarTableModel(); //my class extended from AbstractTableModel
getTable().setModel(tableModel);
scrollPane = new JScrollPane(table, JScrollPane.VERTICAL_SCROLLBAR_NEVER, JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
scrollPane.setAutoscrolls(false);
//GET DIMENSION
int width = 0;
int height = 0;
int allWidth, allHeight;
int padx = 2,pady = 2;
for (int col = 0; col < table.getColumnCount(); col++) {
TableColumn tableColumn = table.getTableHeader().getColumnModel().getColumn(col);
TableCellRenderer renderer = tableColumn.getHeaderRenderer();
if (renderer == null) {
renderer = table.getTableHeader().getDefaultRenderer();
}
Component component = renderer.getTableCellRendererComponent(table,
tableColumn.getHeaderValue(), false, false, -1, col);
width = Math.max(component.getPreferredSize().width, width);
table.getColumnModel().getColumn(col).setPreferredWidth(width);
}
allWidth = (table.getColumnCount()+padx) * width;
for (int row = 0; row < table.getRowCount(); row++) {
TableCellRenderer renderer = table.getCellRenderer(row, 0);
Component comp = table.prepareRenderer(renderer, row, 0);
height = Math.max(comp.getMinimumSize().height, height);
}
allHeight = (1+table.getRowCount()+pady) * height;
//HERE I SET WIDTHS AND HEIGHTS
table.setRowHeight(height);
table.setMinimumSize(new Dimension(allWidth, allHeight));
scrollPane.setMinimumSize(new Dimension(allWidth, allHeight));
}
private void setTable(JTable jTable) {
this.table = jTable;
}
private JTable getTable() {
return this.table;
}
private class CalendarTableModel extends AbstractTableModel {
private String[] daysData = {"Po", "Út", "St", "Čt", "Pá", "So", "Ne"};
private int[][] values;
public CalendarTableModel() {
values = new int[7][6];
for (int i = 0; i < 6; i++) {
for (int j = 0; j < 7; j++) {
values[j][i] = 7;
}
}
}
@Override
public int getRowCount() {
return 6;
}
@Override
public int getColumnCount() {
return 7;
}
@Override
public String getColumnName(int column) {
return daysData[column];
}
@Override
public Object getValueAt(int row, int column) {
return this.values[column][row];
}
@Override
public boolean isCellEditable(int row, int column) {
return false;
}
}
}
you can to get this one
by
f.pack()
before f.setVisible(true);
desired widht should be at 22 / 23 pixelx but component.getPreferredSize(
) or SwingUtilities#computeStringWidth(FontMetrics fm, String str)
returns only 17 / 18 and plus table.getIntercellSpacing().width
only one pixel for Po - Ne
note real workaround could be based on TableColumnAdjuster by @camickr
from code
import java.awt.*;
import javax.swing.*;
import javax.swing.table.*;
import javax.swing.text.JTextComponent;
public class Main {
public static void main(String[] args) {
new Main().start();
}
private void start() {
JFrame f = new JFrame();
f.setLayout(new GridBagLayout());
GridBagConstraints gbc = new GridBagConstraints();
initTable();
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException e) {
}
JPanel calendar = new JPanel();
calendar.setLayout(new GridBagLayout());
calendar.setBorder(BorderFactory.createTitledBorder("Calendar"));
gbc.fill = GridBagConstraints.NONE;
gbc.gridx = 0;
gbc.gridy = 0;
gbc.weighty = 0;
gbc.weightx = 0;
calendar.add(scrollPane, gbc);
f.add(calendar, gbc);
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.pack();
f.setVisible(true);
}
JTable table;
JScrollPane scrollPane;
CalendarTableModel tableModel;
private void initTable() {
setTable(new JTable());
getTable().setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
getTable().setAutoscrolls(false);
getTable().getTableHeader().setResizingAllowed(false);
getTable().getTableHeader().setReorderingAllowed(false);
getTable().setColumnSelectionAllowed(true);
getTable().setRowSelectionAllowed(true);
getTable().setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
tableModel = new CalendarTableModel(); //my class extended from AbstractTableModel
getTable().setModel(tableModel);
scrollPane = new JScrollPane(table, JScrollPane.VERTICAL_SCROLLBAR_NEVER, JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
scrollPane.setAutoscrolls(false);
TableColumnModel columnModel = table.getColumnModel();
for (int col = 0; col < table.getColumnCount(); col++) {
int maxWidth = 0;
for (int row = 0; row < table.getRowCount(); row++) {
TableCellRenderer rend = table.getCellRenderer(row, col);
Object value = table.getValueAt(row, col);
Component comp = rend.getTableCellRendererComponent(table, value, false, false, row, col);
maxWidth = Math.max(comp.getPreferredSize().width, maxWidth);
}
TableColumn column = columnModel.getColumn(col);
TableCellRenderer headerRenderer = column.getHeaderRenderer();
if (headerRenderer == null) {
headerRenderer = table.getTableHeader().getDefaultRenderer();
}
Object headerValue = column.getHeaderValue();
Component headerComp = headerRenderer.getTableCellRendererComponent(table, headerValue, false, false, 0, col);
maxWidth = Math.max(maxWidth, headerComp.getPreferredSize().width);
// note some extra padding
column.setPreferredWidth(maxWidth + 6);//IntercellSpacing * 2 + 2 * 2 pixel instead of taking this value from Borders
}
DefaultTableCellRenderer stringRenderer = (DefaultTableCellRenderer) table.getDefaultRenderer(String.class);
stringRenderer.setHorizontalAlignment(SwingConstants.CENTER);
table.setPreferredScrollableViewportSize(table.getPreferredSize());
}
private void setTable(JTable jTable) {
this.table = jTable;
}
private JTable getTable() {
return this.table;
}
private class CalendarTableModel extends AbstractTableModel {
private String[] daysData = {"Pondelok", "Út", "St", "Čt", "Pá", "Sobota", "Ne"};
private int[][] values;
public CalendarTableModel() {
values = new int[7][6];
for (int i = 0; i < 6; i++) {
for (int j = 0; j < 7; j++) {
values[j][i] = 30;
}
}
}
@Override
public int getRowCount() {
return 6;
}
@Override
public int getColumnCount() {
return 7;
}
@Override
public String getColumnName(int column) {
return daysData[column];
}
@Override
public Object getValueAt(int row, int column) {
return this.values[column][row];
}
@Override
public boolean isCellEditable(int row, int column) {
return false;
}
}
}