java - 在 CellRenderer 中显示选定的行和列

标签 java swing jtable cells tablecellrenderer

我有一个JTable,每次选择一个单元格时,我都想打印它的行和列索引。由于这个原因,我使用 getSelectedRow()getSelectedColumn() 方法。运行下面的代码:

测试时间表类

public class TestTimeTable extends JFrame{
    private final int rows = 10;
    private final int cols = 8;
    private final String daysOfTheWeek[] = {"MONDAY", "TUESDAY", "WEDNESDAY", "THURSDAY", "FRIDAY"};
    private final JPanel jTablePanel;
    private final JScrollPane scrollPane;
    private final JTable timeTable;
    private final Object[][] rowData;

    public TestTimeTable(){
        this.rowData = new Object[this.rows][this.cols];
        this.timeTable = new JTable(this.rowData,this.daysOfTheWeek){
            @Override
            public boolean isCellEditable(int row, int column) {
                return false;
            }
        };

        this.timeTable.setRowHeight(200, 200);
        this.timeTable.setFillsViewportHeight(true);
        this.timeTable.setOpaque(true);
        this.timeTable.setColumnSelectionAllowed(true);
        this.timeTable.setRowSelectionAllowed(true);
        this.timeTable.setCellSelectionEnabled(true);
        this.timeTable.setDefaultRenderer(Object.class, new BoardTableCellRenderer());
        this.scrollPane = new JScrollPane(this.timeTable, JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED, JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);

        this.jTablePanel = new JPanel();
        this.jTablePanel.add(this.scrollPane);
        getContentPane().add(new JScrollPane(this.timeTable), BorderLayout.CENTER);
    }

    public int getColumn(){
        return this.timeTable.getSelectedColumn();
    }

    public int getRow(){
        return this.timeTable.getSelectedRow();
    }

    public JTable getTimeTable(){
        return this.timeTable;
    }

    public void createAndShowUI(){
        setSize(600, 600);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setVisible(true);
    }

    public static void main(String[] argas){
        TestTimeTable tt = new TestTimeTable();
        tt.createAndShowGUI();
    }
}

类 BoardTableCellRenderer

class BoardTableCellRenderer extends DefaultTableCellRenderer {

    Component c;
    private static final long serialVersionUID = 1L;
    private final Color selectionBlue = new Color(131,166,198);
    private final MatteBorder border = new MatteBorder(1, 1, 0, 0, Color.BLACK);

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
        c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);

        if (table.isCellSelected(row, column)){
            c.setBackground(this.selectionBlue);
            setBorder(this.border);
        } else {
            c.setBackground(Color.WHITE);
        }

        if(isSelected){
            JOptionPane.showMessageDialog(null, table.getSelectedRow() + ""+table.getSelectedColumn(),null, JOptionPane.INFORMATION_MESSAGE);
        }
        return c;
    }
}

每次我选择一个单元格时,都会出现 JOptionPane 框架,然后应用程序就会卡住。所以我必须一直打断它。有人可以告诉我为什么会发生这种情况吗?我可以做什么来修复它?

最佳答案

  • 使用、覆盖 getTableCellRendererComponent 中的 boolean isSelected、boolean hasFocus(内置方法),而不是引用回 JTable - table.isCellSelected(row, column), ba 重置 else 语句中其余单元格的颜色,例如简单的躯干

.

   private class StatusRenderer implements TableCellRenderer {

        private static final long serialVersionUID = 1L;

        @Override
        public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
            if (isSelected) {
                setBackground(table.getSelectionBackground());
            } else {
                setBackground(table.getBackground());
            }
            return this;
        }
    }
  • 更改 ListSelectionMode for JTable

  • 把所有的this.Xxx去掉,都没用

  • 不要设置大小,而是使用 JFrame.pack() 和/或设置 JScrollPane 的大小,例如table.setPreferredScrollableViewportSize(table.getPreferredSize());(对于合理的数字或行和列,否则使用 Dimension(x, x))

编辑

import java.awt.Component;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.JCheckBox;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.border.Border;
import javax.swing.border.EmptyBorder;
import javax.swing.plaf.UIResource;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableCellRenderer;

public class TableRolloverDemo {

    private JFrame frame = new JFrame("TableRolloverDemo");
    private JTable table = new JTable();
    private String[] columnNames = new String[]{"Column"};
    private Object[][] data = new Object[][]{{false}, {false}, {true}, {true},
        {false}, {false}, {true}, {true}, {false}, {false}, {true}, {true}};

    public TableRolloverDemo() {
        final DefaultTableModel model = new DefaultTableModel(data, columnNames) {
            //private boolean ImInLoop = false;

            @Override
            public Class<?> getColumnClass(int columnIndex) {
                return Boolean.class;
            }

            @Override
            public boolean isCellEditable(int rowIndex, int columnIndex) {
                if (columnIndex == 0) {
                    return true;
                } else {
                    return false;
                }
            }

            @Override
            public void setValueAt(Object aValue, int rowIndex, int columnIndex) {
                if (columnIndex == 0) {
                    //if (!ImInLoop) {
                    //    ImInLoop = true;
                        Boolean bol = (Boolean) aValue;
                        super.setValueAt(aValue, rowIndex, columnIndex);
                        for (int i = 0; i < this.getRowCount(); i++) {
                            if (i != rowIndex) {
                                super.setValueAt(!bol, i, columnIndex);
                            }
                        }
                    //    ImInLoop = false;
                    //}
                } else {
                    super.setValueAt(aValue, rowIndex, columnIndex);
                }
            }
        };
        RolloverMouseAdapter rolloverAdapter = new RolloverMouseAdapter(table);
        RolloverBooleanRenderer renderer = new RolloverBooleanRenderer(rolloverAdapter);
        table.addMouseListener(rolloverAdapter);
        table.addMouseMotionListener(rolloverAdapter);
        table.setDefaultRenderer(Boolean.class, renderer);
        table.setModel(model);
        table.setPreferredScrollableViewportSize(table.getPreferredSize());
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.add(new JScrollPane(table));
        frame.pack();
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
    }

    private class RolloverMouseAdapter extends MouseAdapter {

        private int row = -1;
        private int column = -1;
        private JTable table;

        public RolloverMouseAdapter(JTable table) {
            this.table = table;
        }

        public boolean isRolloverCell(int row, int column) {
            return this.row == row && this.column == column;
        }

        @Override
        public void mouseMoved(MouseEvent e) {
            int lastRow = row;
            int lastColumn = column;
            row = table.rowAtPoint(e.getPoint());
            column = table.columnAtPoint(e.getPoint());
            if (row == lastRow && column == lastColumn) {
                return;
            }
            if (row >= 0 && column >= 0) {
                table.repaint(table.getCellRect(row, column, false));
            }
            if (lastRow >= 0 && lastColumn >= 0) {
                table.repaint(table.getCellRect(lastRow, lastColumn, false));
            }
        }

        @Override
        public void mouseExited(MouseEvent e) {
            if (row >= 0 && column >= 0) {
                table.repaint(table.getCellRect(row, column, false));
            }
            row = column = -1;
        }
    }

    private class RolloverBooleanRenderer extends JCheckBox implements
            TableCellRenderer, UIResource {

        private final Border noFocusBorder = new EmptyBorder(1, 1, 1, 1);
        private RolloverMouseAdapter adapter;

        public RolloverBooleanRenderer(RolloverMouseAdapter adapter) {
            super();
            this.adapter = adapter;
            setHorizontalAlignment(JLabel.CENTER);
            setBorderPainted(true);
        }

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

            getModel().setRollover(adapter.isRolloverCell(row, column));
            if (isSelected) {
                setForeground(table.getSelectionForeground());
                super.setBackground(table.getSelectionBackground());
            } else {
                setForeground(table.getForeground());
                setBackground(table.getBackground());
            }
            setSelected((value != null && ((Boolean) value).booleanValue()));
            if (hasFocus) {
                setBorder(UIManager.getBorder("Table.focusCellHighlightBorder"));
            } else {
                setBorder(noFocusBorder);
            }
            return this;
        }
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                TableRolloverDemo tableRolloverDemo = new TableRolloverDemo();
            }
        });
    }
}

关于java - 在 CellRenderer 中显示选定的行和列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25991531/

相关文章:

java - 更新两个 JProgressBars

java - 具有固定标题的可滚动内容,无需使用 JTable

Java - 单击第二个单元格时如何防止 JTable 中的水平自动滚动?

java - 当我尝试通过拖放运行 Android 应用程序时出现意外错误

java - java中的HTML解析库

java - 尽管指定了 'key-type',但 Spring 3.2.4 无法在 <util :map. ../> 中将 String 转换为 Enum

Java EE 安全领域

java - 如何结合 JLabel 和 JFrame?

java - 线程中的异常 "AWT-EventQueue-0"java.lang.IllegalArgumentException : input == null

java - 如何使 Swing jTable 单元格在 View 和内联编辑状态下都显示 "YY-MM-DD"格式的日期对象?