Java 如何正确设置jtable中聚焦单元格的边框?

Java 如何正确设置jtable中聚焦单元格的边框?,java,swing,jtable,border,Java,Swing,Jtable,Border,我有一个jtable,它有三种不同类型的渲染器。所以,我看不到提供自定义单元渲染器的简单方法。但我需要在当前选定的单元格周围自定义边框。为了实现这一点,我使用prepareRenderer方法。这可以工作,但有小错误-只显示顶部和底部边框,但不显示左侧和右侧边框 请复制粘贴此文件以查看问题: public class CellBorderDemo extends JFrame { private JTable dataSearchResultTable; public Cell

我有一个jtable,它有三种不同类型的渲染器。所以,我看不到提供自定义单元渲染器的简单方法。但我需要在当前选定的单元格周围自定义边框。为了实现这一点,我使用prepareRenderer方法。这可以工作,但有小错误-只显示顶部和底部边框,但不显示左侧和右侧边框

请复制粘贴此文件以查看问题:

public class CellBorderDemo extends JFrame
{
    private JTable dataSearchResultTable;

    public CellBorderDemo()
    {
        JPanel panel = new JPanel(new GridLayout(2, 1, 5, 10));
        panel.setPreferredSize(new Dimension(500, 300));
        dataSearchResultTable = new JTable(new MyTableModel())
        {
            private EmptyBorder emptyBorder = new EmptyBorder(0, 1, 0, 1);
            private Border redBorder = new CompoundBorder(new MatteBorder(1, 0, 1, 0, Color.RED), emptyBorder);
            private Border unselectedBorder = super.getBorder();

            @Override
            public Component prepareRenderer(TableCellRenderer renderer, int row, int column)
            {
                Object value = getValueAt(row, column);

                boolean isSelected = false;
                boolean hasFocus = false;

                // Only indicate the selection and focused cell if not printing
                if (!isPaintingForPrint()) {
                    isSelected = isCellSelected(row, column);

                    boolean rowIsLead = (selectionModel.getLeadSelectionIndex() == row);
                    boolean colIsLead = (columnModel.getSelectionModel().getLeadSelectionIndex() == column);

                    hasFocus = (rowIsLead && colIsLead) && isFocusOwner();
                }
                JComponent cellRenderer = (JComponent) renderer.getTableCellRendererComponent(this, value, isSelected,
                        hasFocus, row, column);
                if (isSelected && hasFocus) {
                    cellRenderer.setBorder(redBorder);
                } else {
                    cellRenderer.setBorder(unselectedBorder);
                }
                return cellRenderer;
            }
        };
        dataSearchResultTable.setSelectionBackground(new Color(0xccccff));
        dataSearchResultTable.setFillsViewportHeight(true);
        dataSearchResultTable.setRowSelectionAllowed(true);
        dataSearchResultTable.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
        dataSearchResultTable.setAutoResizeMode(JTable.AUTO_RESIZE_OFF);
        dataSearchResultTable.setRowHeight(25);
        dataSearchResultTable.getColumnModel().setColumnMargin(0);
        dataSearchResultTable.setShowGrid(true);
//      dataSearchResultTable.setIntercellSpacing(new Dimension(0, 0));
        dataSearchResultTable.setCellSelectionEnabled(true);

        panel.add(new JScrollPane(dataSearchResultTable));
        super.getContentPane().add(panel);
        super.pack();
        super.setDefaultCloseOperation(EXIT_ON_CLOSE);
        super.setVisible(true);
    }

    class MyTableModel extends AbstractTableModel
    {
        private String[] columnNames = { "First Name", "Last name", "Vegetarian" };
        private Object[][] data;

        MyTableModel()
        {
            data = new Object[][] { { "Vova", "KipokKipokKipokKipok", false }, { "Olia", "Duo", true },
                    { "Ivan", "Brown", false } };
        }

        public int getColumnCount()
        {
            return columnNames.length;
        }

        public int getRowCount()
        {
            return data.length;
        }

        public String getColumnName(int col)
        {
            return columnNames[col];
        }

        public Object getValueAt(int row, int col)
        {
            if (data.length > 0 && data[0] != null) {
                return data[row][col];
            }
            return null;
        }

        public Class getColumnClass(int c)
        {
            Object valueAt = getValueAt(0, c);
            return valueAt == null ? Object.class : valueAt.getClass();
        }

        public boolean isCellEditable(int row, int col)
        {
            return true;
        }

        public void setValueAt(Object value, int row, int col)
        {
            if (data.length > 0 && data[0] != null) {
                data[row][col] = value;
                fireTableCellUpdated(row, col);
            }
        }
    }

    public static void main(String[] args) throws ParseException
    {
        new CellBorderDemo();
    }
}
关于如何在jtable单元格的每一侧设置边框,您有什么想法吗

谢谢大家!

  • prepareRenderer
    默认为行
    Renderer
    ,用于高亮显示整行

  • 当您想要更改整个单个单元格时,必须设置并测试行和列的坐标


使用isSelected,isCellEditable。在某些特定情况下,如果不重写UIManagerIn方法prepareRenderer,Nimbus将无法工作。我使用GetTableCellRenderComponent方法返回的单元格渲染器,该方法使用行和列。aaach现在我看到您的问题边界更改顺序很简单,可以切换为emptyBorder,然后在复合边界中切换为MatteBorder,否则您必须使用插入,我想会很好的。谢谢。我只是重写MatteBorder的getBorderInsets()方法,它就可以工作了。谢谢不要忘记渲染器在屏幕外绘制,不要对边框中的插入使用零值
我有一个jtable,它有三种不同类型的渲染器。因此,我不认为提供自定义单元格渲染器的简单方法
——我在上一个问题()中给了您答案,但您接受了另一个答案,即使您使用的是我的方法。