【问题标题】:Java. How to paint a specific cell in a JTable?爪哇。如何在 JTable 中绘制特定单元格?
【发布时间】:2017-09-07 01:07:30
【问题描述】:

好的,我处于这种情况...我的班级中有渲染器,但不知道如何使用它来使某些单元格的背景变为红色。这是一个房间出租应用程序,我将 Jtable 作为日历,所以我想要将出租的单元格绘制成红色。所以它应该以某种方式采用特定的列和行并使该单元格变为红色。我的渲染器在下面,但正如我所说,不知道如何使用它,因为我是 Java 新手。真正的问题是如何传递该列和行,我对此有疑问。渲染的单元格与其他一些代码一起使用,但这不是我需要的。

ublic class TableColour extends javax.swing.table.DefaultTableCellRenderer {
@Override
public java.awt.Component getTableCellRendererComponent(javax.swing.JTable table, java.lang.Object value, boolean isSelected, boolean hasFocus, int row, int column) {
    java.awt.Component cellComponent = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
    cellComponent.setBackground(java.awt.Color.RED);
    return cellComponent;
}

}

【问题讨论】:

    标签: java swing colors jtable tablecellrenderer


    【解决方案1】:

    您必须为 JTable 的每一列设置单元格渲染器。 我希望这个例子可以帮助你:

    import javax.swing.JFrame;
    import javax.swing.JScrollPane;
    import javax.swing.JTable;
    import javax.swing.SwingUtilities;
    import javax.swing.table.DefaultTableModel;
    
    public class JTableTest
    {
    
      public static void main( String[] args )
      {
        SwingUtilities.invokeLater( new Runnable()
        {
          @Override
          public void run()
          {
            buildUI();
          }
        } );
      }
    
      public static void buildUI()
      {
        final int w = 500;
        final int h = 200;
        Object colNames[] =
        {
          "COL1", "COL2", "COL3"
        };
        Object[][] data =
        {
        };
        DefaultTableModel dtm = new DefaultTableModel( data, colNames );
        dtm.addRow( new Object[]
        {
          "a", "b", "c"
        } );
        dtm.addRow( new Object[]
        {
          "d", "e", "f"
        } );
        dtm.addRow( new Object[]
        {
          "g", "h", "i"
        } );
        dtm.addRow( new Object[]
        {
          "l", "m", "n"
        } );
        final JTable t = new JTable( dtm );
        final TableColour tce = new TableColour();
        t.getColumnModel().getColumn( 0 ).setCellRenderer( tce );
        t.getColumnModel().getColumn( 1 ).setCellRenderer( tce );
        t.getColumnModel().getColumn( 2 ).setCellRenderer( tce );
        final JFrame f = new JFrame();
        f.setBounds( 0, 0, w, h );
        JScrollPane sp = new JScrollPane( t );
        f.getContentPane().add( sp );
        f.setVisible( true );
    
      }
    }
    
    class TableColour
        extends javax.swing.table.DefaultTableCellRenderer
    {
    
      @Override
      public java.awt.Component getTableCellRendererComponent( javax.swing.JTable table, java.lang.Object value, boolean isSelected, boolean hasFocus, int row, int column )
      {
        java.awt.Component cellComponent = super.getTableCellRendererComponent( table, value, isSelected, hasFocus, row, column );
        cellComponent.setBackground( java.awt.Color.RED );
        return cellComponent;
      }
    
    }
    

    【讨论】:

    • 很奇怪,我把它放在按钮点击动作中,当按钮点击时没有任何反应。 Calendar 是我的 Jtable 的名称 final JTable t = new JTable(Calendar.getModel());最终 TableColour tce = new TableColour(); t.getColumnModel().getColumn(0).setCellRenderer(tce); t.getColumnModel().getColumn(1).setCellRenderer(tce); t.getColumnModel().getColumn(2).setCellRenderer(tce);
    • 上面的代码是一个完整的运行类。如果您想使用按钮,我认为您应该在操作管理中调用方法 buildUI()。
    【解决方案2】:

    好吧,哇哦,我可能很难弄清楚这一点。但也许不知何故。你说你不知道我的代码看起来如何,我有一些基本的渲染器。要记住的一件事我有一个二维数组 ReservedOne,它保存了占用房间的行索引和列索引,还有房间号日期,直到它被保留的时间。所以现在我在看你的例子时有点困惑如何使用我的数组来设置颜色。我希望我不会精神崩溃

    您的 TableModel 应该对这些数据进行建模,这非常重要,因为它允许 API 的其余部分围绕它旋转

    真正的问题是如何传递该列和行,我对此有疑问。渲染的单元格与其他一些代码一起使用,但这不是我需要的。

    这就是为什么让TableModel 包装数据很重要的原因,因为表格API 会将rowcolumn 信息传递给TableCellRenderer,但它也会传递单元格值!

    public class RoomTableModel extends AbstractTableModel {
    
        private Room[][] reservations;
    
        public RoomTableModel(Room[][] reservations) {
            this.reservations = reservations;
        }
    
        @Override
        public int getRowCount() {
            return reservations.length;
        }
    
        @Override
        public int getColumnCount() {
            return reservations[0].length;
        }
    
        @Override
        public Object getValueAt(int rowIndex, int columnIndex) {
            return reservations[rowIndex][columnIndex];
        }
    
        @Override
        public Class<?> getColumnClass(int columnIndex) {
            return Room.class;
        }
    
        @Override
        public void setValueAt(Object aValue, int rowIndex, int columnIndex) {
            if (aValue instanceof Room) {
                Room room = (Room) aValue;
                reservations[rowIndex][columnIndex] = room;
                fireTableCellUpdated(rowIndex, columnIndex);
            }
        }
    
    }
    

    这意味着我们现在可以设置单元格渲染器来显示我们需要的信息

    公共静态类 RoomTableCellRenderer 扩展 DefaultTableCellRenderer {

        private static Color BOOKED_COLOR = Color.RED;
    
        @Override
        public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
            if (value instanceof Room && value != null) {
                if (isSelected) {
                    setBackground(table.getSelectionBackground());
                    setForeground(table.getSelectionForeground());
                } else {
                    setBackground(table.getBackground());
                    setForeground(table.getForeground());
                }
                // Update the details based on the room properties
            } else { //if (value == null) {
                setBackground(BOOKED_COLOR);
                setText(null);
            }
            return this;
        }
    
    }
    

    别忘了,如果你想让表格使用你的渲染器,你需要注册它...

    table.setDefaultRenderer(Room.class, new RoomTableCellRenderer());
    

    更新...

    基于可用数据被存储在二维String数组中(你真的不喜欢我)。

    这有点脏。实际上,数据应该设置在可以传递给TableModel并让它处理细节的地方。您还需要注意如何更新数组,因为在您可以强制刷新之前,表格不会反映更新......这不会很漂亮。

    public class LocalDateTableCellRenderer extends DefaultTableCellRenderer {
    
        protected static DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd");
        private String[][] bookings;
    
        public LocalDateTableCellRenderer(String[][] bookings) {
            this.bookings = bookings;
        }
    
    
        @Override
        public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
            super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
            setBackground(isSelected ? table.getSelectionBackground() : table.getBackground());
            setForeground(isSelected ? table.getSelectionForeground() : table.getForeground());
            if (value instanceof LocalDate) {
                LocalDate date = (LocalDate) value;
                if (hasBookingFor(date)) {
                    setForeground(Color.WHITE);
                    setBackground(Color.RED);
                }
                setText(formatter.format(date));
            } else {
                setText(null);
            }
            return this;
        }
    
        protected boolean hasBookingFor(LocalDate date) {
            for (String[] data : bookings) {
                int day = Integer.parseInt(data[2]);
                int month = Integer.parseInt(data[3]);
                int year = 2017; // Because :P
    
                LocalDate booking = LocalDate.of(year, month, day);
                if (booking.isEqual(date)) {
                    return true;
                }
            }
            return false;
        }
    
    }
    

    基本上,这允许您将预订信息传递给TableCellRenderer,正如我所说,这不是您真正应该这样做的方式,但它需要对您的代码进行重大重组才能使其工作正确。

    现在,我创建一个TableModel,它基本上采用年份和月份值,并为每个单元格返回一个LocalDate(如果单元格超出月份范围,则返回null

    public class CalendarModel extends AbstractTableModel {
    
        public static String[] COLUMN_NAMES = {"Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"};
    
        private int rows = 0;
    
        private LocalDate startOfCalendar;
        private LocalDate firstDayOfMonth;
        private LocalDate lastDayOfMonth;
    
        public CalendarModel(int year, Month month) {
            firstDayOfMonth = LocalDate.of(year, month, 1);
    
            startOfCalendar = firstDayOfMonth.minusDays(firstDayOfMonth.getDayOfWeek().getValue());
            lastDayOfMonth = firstDayOfMonth.with(TemporalAdjusters.lastDayOfMonth());
    
            System.out.println(startOfCalendar.getDayOfWeek());
            System.out.println(firstDayOfMonth);
            System.out.println(lastDayOfMonth);
    
            Duration between = Duration.between(startOfCalendar.atStartOfDay(), lastDayOfMonth.atStartOfDay());
            long days = between.toDays();
            rows = (int) Math.round(days / 7d) + 1;
        }
    
        @Override
        public int getRowCount() {
            return rows;
        }
    
        @Override
        public int getColumnCount() {
            return 7;
        }
    
        @Override
        public Class<?> getColumnClass(int columnIndex) {
            return LocalDate.class;
        }
    
        @Override
        public String getColumnName(int column) {
            return COLUMN_NAMES[column];
        }
    
        @Override
        public Object getValueAt(int rowIndex, int columnIndex) {
    
            LocalDate date = null;
    
            if (startOfCalendar != null) {
                int day = (rowIndex * 7) + columnIndex;
                date = startOfCalendar.plusDays(day);
    
                if (date.isBefore(firstDayOfMonth) || date.isAfter(lastDayOfMonth)) {
                    date = null;
                }
            }
    
            return date;
    
        }
    
    }
    

    这意味着TableCellRenderer 被传递了null 值或LocalDate 值,然后您需要在数组中搜索指定日期的任何可能预订。

    这将可怕地扩展,这就是为什么我一直避免这样做并一直试图让你改变你管理数据的方式,但它就是这样

    最后是一个非常粗略的例子......

    这个例子并不真正关心你要管理的所有信息,它只关心月份和日期信息

    import java.awt.Color;
    import java.awt.Component;
    import java.awt.EventQueue;
    import java.time.Duration;
    import java.time.LocalDate;
    import java.time.Month;
    import java.time.format.DateTimeFormatter;
    import java.time.temporal.TemporalAdjusters;
    import javax.swing.JFrame;
    import javax.swing.JScrollPane;
    import javax.swing.JTable;
    import javax.swing.UIManager;
    import javax.swing.UnsupportedLookAndFeelException;
    import javax.swing.table.AbstractTableModel;
    import javax.swing.table.DefaultTableCellRenderer;
    import javax.swing.table.TableModel;
    
    public class Test {
    
        public static void main(String[] args) {
            new Test();
        }
    
        public Test() {
            EventQueue.invokeLater(new Runnable() {
                @Override
                public void run() {
                    try {
                        UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                    } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                        ex.printStackTrace();
                    }
    
                    String[][] bookings = new String[7][6];
                    bookings[0][2] = "5";
                    bookings[0][3] = "4";
                    bookings[1][2] = "10";
                    bookings[1][3] = "4";
                    bookings[2][2] = "15";
                    bookings[2][3] = "4";
                    bookings[3][2] = "20";
                    bookings[3][3] = "4";
                    bookings[4][2] = "25";
                    bookings[4][3] = "4";
                    bookings[5][2] = "30";
                    bookings[5][3] = "4";
                    bookings[6][2] = "5";
                    bookings[6][3] = "5";
    
                    TableModel model = new CalendarModel(2017, Month.APRIL);
                    JTable table = new JTable(model);
                    table.setDefaultRenderer(LocalDate.class, new LocalDateTableCellRenderer(bookings));
    
                    JFrame frame = new JFrame("Testing");
                    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                    frame.add(new JScrollPane(table));
                    frame.pack();
                    frame.setLocationRelativeTo(null);
                    frame.setVisible(true);
                }
            });
        }
    
    }
    

    【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2011-10-02
    • 2020-09-02
    • 1970-01-01
    • 2013-05-20
    • 2012-03-25
    • 2012-03-05
    • 2015-03-23
    • 1970-01-01
    相关资源
    最近更新 更多