2013-04-16 221 views
0

好的,我知道这个问题已经被问到堆栈溢出问题了。我几乎读过每篇文章。我绝对不能把这个工作做得太过分。我制作了一张表作为酒店的可用日历,并且我想为单元格的0个房间提供红色,以及0个以上的绿色。我试过实施表格单元格渲染器无济于事。这是我的所有代码。我究竟做错了什么?基于单元格内容的颜色个别JTable单元格?

package gui; 

import entity.Inventory; 
import entity.Room; 
import java.awt.Color; 
import java.awt.Component; 
import java.awt.event.ItemEvent; 
import java.awt.event.ItemListener; 
import java.text.DateFormat; 
import java.text.SimpleDateFormat; 
import java.util.Date; 
import java.util.LinkedList; 
import javax.swing.BorderFactory; 

import javax.swing.JComboBox; 
import javax.swing.JPanel; 
import javax.swing.JTable; 
import javax.swing.table.DefaultTableCellRenderer; 
import javax.swing.table.DefaultTableModel; 
import tools.DateCalc; 

public class Calendar extends JPanel { 

    String[] years = {"2013", "2014"}; 
    JComboBox yearBox = new JComboBox(years); 
    String[] months = {"January", "February", "March", "April", "May", "June", "July", "August", 
     "September", "October", "November", "December"}; 
    JComboBox monthBox = new JComboBox(months); 
    LinkedList<String> lTypes = Room.getRoomTypes(); 
    String[] types = lTypes.toArray(new String[0]); 
    JComboBox typeBox = new JComboBox(types); 

    CalendarModel model = new CalendarModel(); 
    JTable table = new JTable(model); 

    public Calendar() { 
     super(); 



     yearBox.setBounds(10, 10, 100, 30); 
     yearBox.setSelectedIndex(0); 
     yearBox.addItemListener(new ComboHandler()); 

     int selected = DateCalc.getNumMonth(DateCalc.getCurrentDate()); 

     monthBox.setSelectedIndex(selected); 
     monthBox.addItemListener(new ComboHandler()); 
     // table.setBounds(10, 150, 550, 200); 
     model.setMonth(Integer.parseInt((String)yearBox.getSelectedItem()), monthBox.getSelectedIndex(), 0); 
     typeBox.addItemListener(new ComboHandler()); 
     add(monthBox); 
     add(yearBox); 
     add(typeBox); 
     table.setGridColor(Color.black); 
     table.setShowGrid(true); 
     table.setRowHeight(40); 
     table.setBorder(BorderFactory.createLineBorder(Color.black)); 

     table.setDefaultRenderer(String.class, new MyCellRender()); 
     add(table); 



     setSize(600, 800); 
     setVisible(true); 
    } 

    public static void main(String[] args) { 
     Calendar app = new Calendar(); 
    } 

    public class ComboHandler implements ItemListener { 

     public void itemStateChanged(ItemEvent e) { 

      model.setMonth(Integer.parseInt((String)yearBox.getSelectedItem()), monthBox.getSelectedIndex(), typeBox.getSelectedIndex()); 
      table.repaint(); 

     } 
    } 

    public class MyCellRender extends DefaultTableCellRenderer { 

     @Override 
     public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) { 
      Component c = super.getTableCellRendererComponent(table, value, 
              isSelected, hasFocus, 
              row, column); 
      for (int i =0; i < 7; i++){ 
       for (int j = 0; j < j; j++){ 
        String positive = (String)table.getModel().getValueAt(i,j); 
        System.out.println(positive); 
        int val = positive.indexOf(" 0 "); 
        if (val < 0) 
         c.setBackground(Color.green); 
        else{ 
         c.setBackground(Color.red); 
        } 
       } 
      } 

      return c; 
     } 
    } 

    class CalendarModel extends DefaultTableModel { 

     String[] days = {" Sun", " Mon", " Tue", " Wed", " Thu", " Fri", " Sat"}; 
     int[] numDays = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; 
     String[][] calendar = new String[7][7]; 

     public CalendarModel() { 
      for (int i = 0; i < days.length; ++i) { 
       calendar[0][i] = days[i]; 
      } 
      for (int i = 1; i < 7; ++i) { 
       for (int j = 0; j < 7; ++j) { 
        calendar[i][j] = " "; 
       } 
      } 
     } 

     @Override 
     public boolean isCellEditable(int row, int column){ 
      return false; 
     } 
     public int getRowCount() { 
      return 7; 
     } 

     public int getColumnCount() { 
      return 7; 
     } 

     public String getValueAt(int row, int column) { 
      return calendar[row][column]; 
     } 

     public void setValueAt(String value, int row, int column) { 
      calendar[row][column] = value; 
     } 

     public void setMonth(int year, int month, int flag) { 
      for (int i = 1; i < 7; ++i) { 
       for (int j = 0; j < 7; ++j) { 
        calendar[i][j] = " "; 
       } 
      } 
      java.util.Calendar cal = java.util.Calendar.getInstance(); 
      cal.set(year, month, 1); 
      Date date = cal.getTime(); 

      DateFormat df = new SimpleDateFormat("MMddyyyy"); 

      String start = df.format(date); 
      int offset = cal.get(java.util.Calendar.DAY_OF_WEEK)-1; 

      offset += 7; 
      int num = daysInMonth(year, month); 
      for (int i = 0; i < num; ++i) { 
       String target = DateCalc.addDay(start, i); 
       Inventory inventory = Inventory.fromDBObject(target); 
       String type = (String)typeBox.getSelectedItem(); 
       String beds = inventory.getField(type); 
       calendar[offset/7][offset % 7] = " " + Integer.toString(i + 1) 
         + ". " + beds; 
       ++offset; 

      } 
     } 

     public boolean isLeapYear(int year) { 
      if (year % 4 == 0) { 
       return true; 
      } 
      return false; 
     } 

     public int daysInMonth(int year, int month) { 
      int days = numDays[month]; 
      if (month == 1 && isLeapYear(year)) { 
       ++days; 
      } 
      return days; 
     } 
    } 
} 
+1

一些一样的东西:一)模型的实现是无效的:它_must_上的更改通知其侦听器,b)上渲染你混合索引坐标系统:作为参数传入行/列的索引是其_must not_使用视图坐标访问模型C)不知道你想要什么,在渲染器的循环来实现:颜色变化的结果是相当不可预知的,因为你不断改变它D)永永远做组件的任何手动尺寸/定位,这是一个合适的LayoutManager – kleopatra

+0

的独家任务想通了,后来不得不实施模型 –

回答

1

如果String类作为列类

table.setDefaultRenderer(String.class, new MyCellRender()); 

然后,列类TableModel的应符合:

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

此外,还有一个问题,在getTableCellRendererComponent,内循环将不会执行

for (int i = 0; i < 7; i++) { 
    for (int j = 0; j < j; j++) { 
        ^^^^^---------condition always false, loop will not execute 

目前尚不清楚为什么这些循环是必需的,因为无论是为每个单元组件调用getTableCellRendererComponent