Java Swing写的支持合并单元格的JTable

年前在网上参加了一个JavaSwing的招聘上机考试。招聘方要求开发一个类似EXCEL支持单元格合并的JTable。差不多用了5天的时间提交代码,最后被告知测试通过,我提出是否可做兼职,对方回复需要到上海做全职开发,最后也就放弃了。最近公司的一个项目中需要用到以前的代码,偶又重构了一次,设计思想来源于ListSelectionModel。

Java Swing写的支持合并单元格的JTable

GridBagModel:抽象模型接口。该接口用于描述表格中单元格的合并状态。
DefaultGridBagTableModel:GridBagModel的默认实现。
GridBagTable:继承自JTable的控制器。通过该类中的方法控制表格单元的合并和拆分。
GridBagTableUI:GridBagTable对应的UI。

TODO:(已合并)行、列的插入,删除操作对应的GridBagModel的修改,不过已留接口。

Java代码 复制代码 收藏代码
  1. packageorg.dxj.guitools.gridbagtable;
  2. importjava.awt.Component;
  3. importjava.awt.Point;
  4. importjava.awt.Rectangle;
  5. importjava.util.Enumeration;
  6. importjava.util.EventObject;
  7. importjavax.swing.DefaultCellEditor;
  8. importjavax.swing.JTable;
  9. importjavax.swing.SwingUtilities;
  10. importjavax.swing.event.TableModelEvent;
  11. importjavax.swing.table.AbstractTableModel;
  12. importjavax.swing.table.TableColumn;
  13. importjavax.swing.table.TableColumnModel;
  14. /**
  15. *@[email protected]
  16. */
  17. publicclassGridBagTableextendsJTable{
  18. GridBagModelgridBagModel;
  19. publicGridBagModelgetGridBagModel(){
  20. returngridBagModel;
  21. }
  22. publicvoidsetGridBagModel(GridBagModelgridBagModel){
  23. if(gridBagModel!=null&&gridBagModel!=this.gridBagModel)
  24. this.gridBagModel=gridBagModel;
  25. }
  26. publicGridBagTable(AbstractTableModeldm){
  27. super(dm);
  28. getTableHeader().setReorderingAllowed(false);
  29. gridBagModel=newDefaultGridBagTableModel(dm);
  30. getColumnModel().setColumnSelectionAllowed(true);
  31. }
  32. privatevoidupdateSubComponentUI(ObjectcomponentShell){
  33. if(componentShell==null){
  34. return;
  35. }
  36. Componentcomponent=null;
  37. if(componentShellinstanceofComponent){
  38. component=(Component)componentShell;
  39. }
  40. if(componentShellinstanceofDefaultCellEditor){
  41. component=((DefaultCellEditor)componentShell).getComponent();
  42. }
  43. if(component!=null){
  44. SwingUtilities.updateComponentTreeUI(component);
  45. }
  46. }
  47. publicvoidupdateUI(){
  48. //UpdatetheUIsofthecellrenderers,celleditorsandheaderrenderers.
  49. TableColumnModelcm=getColumnModel();
  50. for(intcolumn=0;column<cm.getColumnCount();column++){
  51. TableColumnaColumn=cm.getColumn(column);
  52. updateSubComponentUI(aColumn.getCellRenderer());
  53. updateSubComponentUI(aColumn.getCellEditor());
  54. updateSubComponentUI(aColumn.getHeaderRenderer());
  55. }
  56. //UpdatetheUIsofallthedefaultrenderers.
  57. EnumerationdefaultRenderers=defaultRenderersByColumnClass.elements();
  58. while(defaultRenderers.hasMoreElements()){
  59. updateSubComponentUI(defaultRenderers.nextElement());
  60. }
  61. //UpdatetheUIsofallthedefaulteditors.
  62. EnumerationdefaultEditors=defaultEditorsByColumnClass.elements();
  63. while(defaultEditors.hasMoreElements()){
  64. updateSubComponentUI(defaultEditors.nextElement());
  65. }
  66. //UpdatetheUIofthetableheader
  67. if(tableHeader!=null&&tableHeader.getParent()==null){
  68. tableHeader.updateUI();
  69. }
  70. setUI(newGridBagTableUI());
  71. }
  72. publicRectanglegetGridCellRect(introw,intcolumn,booleanincludeSpacing){
  73. returnsuper.getCellRect(row,column,includeSpacing);
  74. }
  75. publicRectanglegetCellRect(introw,intcolumn,booleanincludeSpacing){
  76. RectanglecellRect=super.getCellRect(row,column,includeSpacing);
  77. intcols=gridBagModel.getColumnGrid(row,column);
  78. TableColumnModelcm=getColumnModel();
  79. for(intn=1;n<cols;n++)
  80. cellRect.width+=cm.getColumn(column+n).getWidth();
  81. introws=gridBagModel.getRowGrid(row,column);
  82. for(intn=1;n<rows;n++)
  83. cellRect.height+=getRowHeight(row+n);
  84. returncellRect;
  85. }
  86. publicvoidtableChanged(TableModelEvente){
  87. super.tableChanged(e);
  88. //TODO
  89. }
  90. publicbooleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn){
  91. if(gridBagModel.mergeCells(startRow,endRow,startColumn,endColumn)){
  92. repaint();
  93. returntrue;
  94. }
  95. returnfalse;
  96. }
  97. publicbooleanmergeCells(int[]rows,int[]columns){
  98. if(gridBagModel.mergeCells(rows,columns)){
  99. repaint();
  100. returntrue;
  101. }
  102. returnfalse;
  103. }
  104. publicbooleanspliteCellAt(introw,intcolumn){
  105. if(gridBagModel.spliteCellAt(row,column)){
  106. repaint();
  107. returntrue;
  108. }
  109. returnfalse;
  110. }
  111. publicvoidchangeSelection(introwIndex,intcolumnIndex,booleantoggle,booleanextend){
  112. if(gridBagModel.getCellState(rowIndex,columnIndex)!=GridBagModel.COVERED)
  113. super.changeSelection(rowIndex,columnIndex,toggle,extend);
  114. Pointp;
  115. for(introw=rowIndex;row>=0;row--){
  116. for(intcol=columnIndex;col>=0;col--){
  117. p=gridBagModel.getGrid(row,col);
  118. //p=((Point)((Vector)rowVector.get(row)).get(col));
  119. if(col+p.x>columnIndex&&row+p.y>rowIndex){
  120. rowIndex=row;
  121. columnIndex=col;
  122. break;
  123. }
  124. }
  125. }
  126. super.changeSelection(rowIndex,columnIndex,toggle,extend);
  127. repaint();
  128. }
  129. publicbooleaneditCellAt(introwIndex,intcolumnIndex,EventObjecte){
  130. if(gridBagModel.getCellState(rowIndex,columnIndex)!=GridBagModel.COVERED)
  131. returnsuper.editCellAt(rowIndex,columnIndex,e);
  132. Pointp;
  133. for(introw=rowIndex;row>=0;row--){
  134. for(intcol=columnIndex;col>=0;col--){
  135. p=gridBagModel.getGrid(row,col);
  136. if(col+p.x>columnIndex&&row+p.y>rowIndex){
  137. rowIndex=row;
  138. columnIndex=col;
  139. break;
  140. }
  141. }
  142. }
  143. returnsuper.editCellAt(rowIndex,columnIndex,e);
  144. }
  145. }
Java代码 复制代码 收藏代码
  1. packageorg.dxj.guitools.gridbagtable;
  2. importjava.awt.Point;
  3. publicinterfaceGridBagModel{
  4. //格子处于正常状态
  5. intDEFAULT=0;
  6. //格子合并了其他的格子
  7. intMERGE=1;
  8. //格子被其他格子合并
  9. intCOVERED=-1;
  10. /**
  11. *@paramrow行
  12. *@paramcolumn列
  13. *@return该单元格在行、列的跨度
  14. */
  15. PointgetGrid(introw,intcolumn);
  16. /**
  17. *在Y轴方向的跨度
  18. *@paramrow
  19. *@paramcolumn
  20. *@return
  21. */
  22. intgetRowGrid(introw,intcolumn);
  23. /**
  24. *在X轴方向的跨度
  25. *@paramrow
  26. *@paramcolumn
  27. *@return
  28. */
  29. intgetColumnGrid(introw,intcolumn);
  30. /**
  31. *@paramrows行集合
  32. *@paramcolumns列集合
  33. *@return单元格集合是否可以合并在一起
  34. */
  35. booleancanMergeCells(int[]rows,int[]columns);
  36. /**
  37. *判断该单元格状态
  38. *@paramrow
  39. *@paramcolumn
  40. *@returnMERGE|DEFAULT|COVERED
  41. */
  42. intgetCellState(introw,intcolumn);
  43. /**
  44. *将单元格集合合并
  45. *@paramstartRow开始行
  46. *@paramendRow结束行
  47. *@paramstartColumn开始列
  48. *@paramendColumn结束列
  49. *@return是否合并成功
  50. */
  51. booleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn);
  52. /**
  53. *将单元格集合合并
  54. *@paramrows行集合
  55. *@paramcolumns列集合
  56. *@return是否合并成功
  57. */
  58. booleanmergeCells(int[]rows,int[]columns);
  59. /**
  60. *拆分单元格
  61. *@paramrow行
  62. *@paramcolumn列
  63. *@return是否拆分成功
  64. */
  65. booleanspliteCellAt(introw,intcolumn);
  66. /**
  67. *清除所有合并
  68. */
  69. voidclearMergence();
  70. }
Java代码 复制代码 收藏代码
  1. packageorg.dxj.guitools.gridbagtable;
  2. importjava.awt.Point;
  3. importjava.util.Arrays;
  4. importjava.util.List;
  5. importjava.util.Vector;
  6. importjavax.swing.event.TableModelEvent;
  7. importjavax.swing.event.TableModelListener;
  8. importjavax.swing.table.AbstractTableModel;
  9. publicclassDefaultGridBagTableModelimplementsGridBagModel,TableModelListener{
  10. protectedAbstractTableModelmodel;
  11. protectedList<List<Point>>gridInfo;
  12. DefaultGridBagTableModel(AbstractTableModelmodel){
  13. gridInfo=newVector<List<Point>>();
  14. setTableModel(model);
  15. }
  16. publicvoidsetTableModel(AbstractTableModelmodel){
  17. if(model!=null&&model!=this.model){
  18. if(this.model!=null)
  19. this.model.removeTableModelListener(this);
  20. //防止多次添加监听器
  21. model.removeTableModelListener(this);
  22. model.addTableModelListener(this);
  23. this.model=model;
  24. clearMergence();
  25. }
  26. }
  27. publicvoidclearMergence(){
  28. if(gridInfo==null)
  29. gridInfo=newVector<List<Point>>();
  30. else
  31. gridInfo.clear();
  32. if(model==null)
  33. return;
  34. //初始化,每个格子占的格子数为(1,1);
  35. for(introw=model.getRowCount();--row>=0;){
  36. List<Point>infos=newVector<Point>();
  37. gridInfo.add(infos);
  38. for(intcol=model.getColumnCount();--col>=0;){
  39. infos.add(getDefaultPoint());
  40. }
  41. }
  42. }
  43. publicPointgetDefaultPoint(){
  44. returnnewPoint(1,1);
  45. }
  46. @Override
  47. publicbooleancanMergeCells(int[]rows,int[]columns){
  48. if(rows==null||columns==null)returnfalse;
  49. Arrays.sort(rows);
  50. for(intindex=0;index<rows.length-1;index++){
  51. if(rows[index+1]-rows[index]>1)
  52. returnfalse;
  53. }
  54. Arrays.sort(columns);
  55. for(intindex=0;index<columns.length-1;index++){
  56. if(columns[index+1]-columns[index]>1)
  57. returnfalse;
  58. }
  59. returntrue;
  60. }
  61. @Override
  62. publicintgetCellState(introw,intcolumn){
  63. Pointgrid=getGrid(row,column);
  64. if(grid==null)returnDEFAULT;
  65. if(grid.x>1||grid.y>1)
  66. returnMERGE;
  67. if(grid.x<=0||grid.y<=0)
  68. returnCOVERED;
  69. returnDEFAULT;
  70. }
  71. @Override
  72. publicintgetColumnGrid(introw,intcolumn){
  73. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
  74. List<Point>gridRow=gridInfo.get(row);
  75. if(gridRow!=null&&column>=0&&column<gridRow.size()){
  76. Pointpoint=gridRow.get(column);
  77. if(point!=null)
  78. returnpoint.x;
  79. }
  80. }
  81. return1;
  82. }
  83. @Override
  84. publicPointgetGrid(introw,intcolumn){
  85. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
  86. List<Point>gridRow=gridInfo.get(row);
  87. if(gridRow!=null&&column>=0&&column<gridRow.size()){
  88. returngridRow.get(column);
  89. }
  90. }
  91. returngetDefaultPoint();
  92. }
  93. @Override
  94. publicintgetRowGrid(introw,intcolumn){
  95. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
  96. List<Point>gridRow=gridInfo.get(row);
  97. if(gridRow!=null&&column>=0&&column<gridRow.size()){
  98. Pointpoint=gridRow.get(column);
  99. if(point!=null)
  100. returnpoint.y;
  101. }
  102. }
  103. return1;
  104. }
  105. protectedbooleansetGrid(introw,intcolumn,Pointgrid){
  106. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
  107. List<Point>gridRow=gridInfo.get(row);
  108. if(gridRow!=null&&column>=0&&column<gridRow.size()){
  109. Pointpoint=gridRow.get(column);
  110. if(point!=null){
  111. point.setLocation(grid);
  112. }
  113. else{
  114. gridRow.set(column,grid.getLocation());
  115. }
  116. returntrue;
  117. }
  118. }
  119. returnfalse;
  120. }
  121. @Override
  122. publicbooleanspliteCellAt(introw,intcolumn){
  123. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
  124. List<Point>gridRow=gridInfo.get(row);
  125. if(gridRow!=null&&column>=0&&column<gridRow.size()){
  126. Pointpoint=gridRow.get(column);
  127. if(point!=null){
  128. point=point.getLocation();
  129. for(inta=0;a<point.y;a++){
  130. for(intb=0;b<point.x;b++){
  131. setGrid(row+a,column+b,getDefaultPoint());
  132. }
  133. }
  134. }
  135. else{
  136. gridRow.set(column,getDefaultPoint());
  137. }
  138. returntrue;
  139. }
  140. }
  141. returnfalse;
  142. }
  143. @Override
  144. /**
  145. *table中发生行的添加和删除的时候需要修改该模型
  146. */
  147. publicvoidtableChanged(TableModelEvente){
  148. //TODO
  149. }
  150. @Override
  151. publicbooleanmergeCells(int[]rows,int[]columns){
  152. if(!canMergeCells(rows,columns))
  153. returnfalse;
  154. Arrays.sort(rows);
  155. Arrays.sort(columns);
  156. returnmergeCells(rows[0],rows[rows.length-1],columns[0],columns[columns.length-1]);
  157. }
  158. @Override
  159. publicbooleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn){
  160. setGrid(startRow,startColumn,newPoint(endColumn-startColumn+1,endRow-startRow+1));
  161. for(introw=startRow;row<=endRow;row++){
  162. for(intcol=startColumn;col<=endColumn;col++){
  163. if(row==startRow&&col==startColumn)
  164. continue;
  165. else
  166. setGrid(row,col,newPoint(COVERED,COVERED));
  167. }
  168. }
  169. returntrue;
  170. }
  171. publicStringtoString(){
  172. if(gridInfo==null)
  173. return"";
  174. StringBuffersb=newStringBuffer();
  175. for(List<Point>rowInfo:gridInfo){
  176. for(Pointgrid:rowInfo){
  177. sb.append("["+grid.x+","+grid.y+"],");
  178. }
  179. sb.append("\n");
  180. }
  181. returnsb.toString();
  182. }
  183. }
Java代码 复制代码 收藏代码
  1. packageorg.dxj.guitools.gridbagtable;
  2. importjava.awt.Color;
  3. importjava.awt.Component;
  4. importjava.awt.Dimension;
  5. importjava.awt.Graphics;
  6. importjava.awt.Point;
  7. importjava.awt.Rectangle;
  8. importjava.util.Enumeration;
  9. importjavax.swing.BorderFactory;
  10. importjavax.swing.JComponent;
  11. importjavax.swing.JTable;
  12. importjavax.swing.UIManager;
  13. importjavax.swing.plaf.basic.BasicTableUI;
  14. importjavax.swing.table.JTableHeader;
  15. importjavax.swing.table.TableCellRenderer;
  16. importjavax.swing.table.TableColumn;
  17. importjavax.swing.table.TableColumnModel;
  18. publicclassGridBagTableUIextendsBasicTableUI
  19. {
  20. publicDimensiongetPreferredSize(JComponentc){
  21. longwidth=0;
  22. Enumeration<TableColumn>enumeration=table.getColumnModel().getColumns();
  23. while(enumeration.hasMoreElements()){
  24. TableColumnaColumn=(TableColumn)enumeration.nextElement();
  25. width=width+aColumn.getPreferredWidth();
  26. }
  27. returncreateTableSize(width);
  28. }
  29. privateDimensioncreateTableSize(longwidth){
  30. intheight=0;
  31. introwCount=table.getRowCount();
  32. if(rowCount>0&&table.getColumnCount()>0){
  33. Rectangler=table.getCellRect(rowCount-1,0,true);
  34. height=r.y+r.height;
  35. }
  36. //Widthisalwayspositive.Thecalltoabs()isaworkaroundfor
  37. //abuginthe1.1.6JITonWindows.
  38. longtmp=Math.abs(width);
  39. if(tmp>Integer.MAX_VALUE){
  40. tmp=Integer.MAX_VALUE;
  41. }
  42. returnnewDimension((int)tmp,height);
  43. }
  44. publicvoidpaint(Graphicsg,JComponentc){
  45. Rectangleclip=g.getClipBounds();
  46. Rectanglebounds=table.getBounds();
  47. //accountforthefactthatthegraphicshasalreadybeentranslated
  48. //intothetable'sbounds
  49. bounds.x=bounds.y=0;
  50. if(table.getRowCount()<=0||table.getColumnCount()<=0||
  51. //thischeckpreventsusfrompaintingtheentiretable
  52. //whentheclipdoesn'tintersectourboundsatall
  53. !bounds.intersects(clip)){
  54. paintDropLines(g);
  55. return;
  56. }
  57. booleanltr=table.getComponentOrientation().isLeftToRight();
  58. PointupperLeft=clip.getLocation();
  59. if(!ltr){
  60. upperLeft.x++;
  61. }
  62. PointlowerRight=newPoint(clip.x+clip.width-(ltr?1:0),
  63. clip.y+clip.height);
  64. intrMin=table.rowAtPoint(upperLeft);
  65. intrMax=table.rowAtPoint(lowerRight);
  66. //Thisshouldneverhappen(aslongasourboundsintersecttheclip,
  67. //whichiswhywebailaboveifthatisthecase).
  68. if(rMin==-1){
  69. rMin=0;
  70. }
  71. //Ifthetabledoesnothaveenoughrowstofilltheviewwe'llget-1.
  72. //(Wecouldalsoget-1ifourboundsdon'tintersecttheclip,
  73. //whichiswhywebailaboveifthatisthecase).
  74. //Replacethiswiththeindexofthelastrow.
  75. if(rMax==-1){
  76. rMax=table.getRowCount()-1;
  77. }
  78. intcMin=table.columnAtPoint(ltr?upperLeft:lowerRight);
  79. intcMax=table.columnAtPoint(ltr?lowerRight:upperLeft);
  80. //Thisshouldneverhappen.
  81. if(cMin==-1){
  82. cMin=0;
  83. }
  84. //Ifthetabledoesnothaveenoughcolumnstofilltheviewwe'llget-1.
  85. //Replacethiswiththeindexofthelastcolumn.
  86. if(cMax==-1){
  87. cMax=table.getColumnCount()-1;
  88. }
  89. //Paintthegrid.
  90. //paintGrid(g,rMin,rMax,cMin,cMax);
  91. //Paintthecells.
  92. paintCells(g,rMin,rMax,cMin,cMax);
  93. paintDropLines(g);
  94. }
  95. privatevoidpaintDropLines(Graphicsg){
  96. JTable.DropLocationloc=table.getDropLocation();
  97. if(loc==null){
  98. return;
  99. }
  100. Colorcolor=UIManager.getColor("Table.dropLineColor");
  101. ColorshortColor=UIManager.getColor("Table.dropLineShortColor");
  102. if(color==null&&shortColor==null){
  103. return;
  104. }
  105. Rectanglerect;
  106. rect=getHDropLineRect(loc);
  107. if(rect!=null){
  108. intx=rect.x;
  109. intw=rect.width;
  110. if(color!=null){
  111. extendRect(rect,true);
  112. g.setColor(color);
  113. g.fillRect(rect.x,rect.y,rect.width,rect.height);
  114. }
  115. if(!loc.isInsertColumn()&&shortColor!=null){
  116. g.setColor(shortColor);
  117. g.fillRect(x,rect.y,w,rect.height);
  118. }
  119. }
  120. rect=getVDropLineRect(loc);
  121. if(rect!=null){
  122. inty=rect.y;
  123. inth=rect.height;
  124. if(color!=null){
  125. extendRect(rect,false);
  126. g.setColor(color);
  127. g.fillRect(rect.x,rect.y,rect.width,rect.height);
  128. }
  129. if(!loc.isInsertRow()&&shortColor!=null){
  130. g.setColor(shortColor);
  131. g.fillRect(rect.x,y,rect.width,h);
  132. }
  133. }
  134. }
  135. /*
  136. *Paintsthegridlineswithin<I>aRect</I>,usingthegrid
  137. *colorsetwith<I>setGridColor</I>.Paintsverticallines
  138. *if<code>getShowVerticalLines()</code>returnstrueandpaints
  139. *horizontallinesif<code>getShowHorizontalLines()</code>
  140. *returnstrue.
  141. */
  142. privatevoidpaintGrid(Graphicsg,intrMin,intrMax,intcMin,intcMax){
  143. g.setColor(table.getGridColor());
  144. RectangleminCell=table.getCellRect(rMin,cMin,true);
  145. RectanglemaxCell=table.getCellRect(rMax,cMax,true);
  146. RectangledamagedArea=minCell.union(maxCell);
  147. if(table.getShowHorizontalLines()){
  148. inttableWidth=damagedArea.x+damagedArea.width;
  149. inty=damagedArea.y;
  150. for(introw=rMin;row<=rMax;row++){
  151. y+=table.getRowHeight(row);
  152. g.drawLine(damagedArea.x,y-1,tableWidth-1,y-1);
  153. }
  154. }
  155. if(table.getShowVerticalLines()){
  156. TableColumnModelcm=table.getColumnModel();
  157. inttableHeight=damagedArea.y+damagedArea.height;
  158. intx;
  159. if(table.getComponentOrientation().isLeftToRight()){
  160. x=damagedArea.x;
  161. for(intcolumn=cMin;column<=cMax;column++){
  162. intw=cm.getColumn(column).getWidth();
  163. x+=w;
  164. g.drawLine(x-1,0,x-1,tableHeight-1);
  165. }
  166. }else{
  167. x=damagedArea.x;
  168. for(intcolumn=cMax;column>=cMin;column--){
  169. intw=cm.getColumn(column).getWidth();
  170. x+=w;
  171. g.drawLine(x-1,0,x-1,tableHeight-1);
  172. }
  173. }
  174. }
  175. }
  176. privatevoidpaintCells(Graphicsg,intrMin,intrMax,intcMin,intcMax){
  177. JTableHeaderheader=table.getTableHeader();
  178. TableColumndraggedColumn=(header==null)?null:header.getDraggedColumn();
  179. TableColumnModelcm=table.getColumnModel();
  180. intcolumnMargin=cm.getColumnMargin();
  181. RectanglecellRect;
  182. TableColumnaColumn;
  183. intcolumnWidth;
  184. if(table.getComponentOrientation().isLeftToRight()){
  185. for(introw=rMin;row<=rMax;row++){
  186. if(tableinstanceofGridBagTable)
  187. cellRect=((GridBagTable)table).getGridCellRect(row,cMin,false);
  188. else
  189. cellRect=table.getCellRect(row,cMin,false);
  190. for(intcolumn=cMin;column<=cMax;column++){
  191. aColumn=cm.getColumn(column);
  192. columnWidth=aColumn.getWidth();
  193. //TODO
  194. cellRect.width=columnWidth-columnMargin;
  195. intoldHeight=cellRect.height;
  196. if(tableinstanceofGridBagTable){
  197. if(((GridBagTable)table).getGridBagModel().getCellState(row,column)==GridBagModel.COVERED){
  198. cellRect.width=0;
  199. cellRect.height=0;
  200. }
  201. else{
  202. inth=((GridBagTable)table).getGridBagModel().getColumnGrid(row,column);
  203. if(h>1){
  204. for(intn=1;n<h;n++)
  205. cellRect.width+=cm.getColumn(column+n).getWidth();
  206. }
  207. intv=((GridBagTable)table).getGridBagModel().getRowGrid(row,column);
  208. if(v>1){
  209. for(intn=1;n<v;n++)
  210. cellRect.height+=table.getRowHeight(row+n);
  211. }
  212. }
  213. }
  214. if(aColumn!=draggedColumn){
  215. paintCell(g,cellRect,row,column);
  216. }
  217. cellRect.height=oldHeight;
  218. cellRect.x+=columnWidth;
  219. }
  220. }
  221. }else{
  222. for(introw=rMin;row<=rMax;row++){
  223. cellRect=table.getCellRect(row,cMin,false);
  224. aColumn=cm.getColumn(cMin);
  225. if(aColumn!=draggedColumn){
  226. columnWidth=aColumn.getWidth();
  227. cellRect.width=columnWidth-columnMargin;
  228. paintCell(g,cellRect,row,cMin);
  229. }
  230. for(intcolumn=cMin+1;column<=cMax;column++){
  231. aColumn=cm.getColumn(column);
  232. columnWidth=aColumn.getWidth();
  233. //TODO
  234. cellRect.width=columnWidth-columnMargin;
  235. cellRect.x-=columnWidth;
  236. if(aColumn!=draggedColumn){
  237. paintCell(g,cellRect,row,column);
  238. }
  239. }
  240. }
  241. }
  242. //Paintthedraggedcolumnifwearedragging.
  243. if(draggedColumn!=null){
  244. paintDraggedArea(g,rMin,rMax,draggedColumn,header.getDraggedDistance());
  245. }
  246. //RemoveanyrenderersthatmaybeleftintherendererPane.
  247. rendererPane.removeAll();
  248. }
  249. privatevoidpaintCell(Graphicsg,RectanglecellRect,introw,intcolumn){
  250. if(table.isEditing()&&table.getEditingRow()==row&&
  251. table.getEditingColumn()==column){
  252. Componentcomponent=table.getEditorComponent();
  253. component.setBounds(cellRect);
  254. component.validate();
  255. }
  256. else{
  257. TableCellRendererrenderer=table.getCellRenderer(row,column);
  258. Componentcomponent=table.prepareRenderer(renderer,row,column);
  259. if(componentinstanceofJComponent){
  260. ((JComponent)component).setBorder(BorderFactory.createLineBorder(Color.gray));
  261. }
  262. rendererPane.paintComponent(g,component,table,cellRect.x,cellRect.y,
  263. cellRect.width,cellRect.height,true);
  264. }
  265. }
  266. privateRectanglegetHDropLineRect(JTable.DropLocationloc){
  267. if(!loc.isInsertRow()){
  268. returnnull;
  269. }
  270. introw=loc.getRow();
  271. intcol=loc.getColumn();
  272. if(col>=table.getColumnCount()){
  273. col--;
  274. }
  275. Rectanglerect=table.getCellRect(row,col,true);
  276. if(row>=table.getRowCount()){
  277. row--;
  278. RectangleprevRect=table.getCellRect(row,col,true);
  279. rect.y=prevRect.y+prevRect.height;
  280. }
  281. if(rect.y==0){
  282. rect.y=-1;
  283. }else{
  284. rect.y-=2;
  285. }
  286. rect.height=3;
  287. returnrect;
  288. }
  289. privatevoidpaintDraggedArea(Graphicsg,intrMin,intrMax,TableColumndraggedColumn,intdistance){
  290. intdraggedColumnIndex=viewIndexForColumn(draggedColumn);
  291. RectangleminCell=table.getCellRect(rMin,draggedColumnIndex,true);
  292. RectanglemaxCell=table.getCellRect(rMax,draggedColumnIndex,true);
  293. RectanglevacatedColumnRect=minCell.union(maxCell);
  294. //Paintagraywellinplaceofthemovingcolumn.
  295. g.setColor(table.getParent().getBackground());
  296. g.fillRect(vacatedColumnRect.x,vacatedColumnRect.y,
  297. vacatedColumnRect.width,vacatedColumnRect.height);
  298. //Movetothewherethecellhasbeendragged.
  299. vacatedColumnRect.x+=distance;
  300. //Fillthebackground.
  301. g.setColor(table.getBackground());
  302. g.fillRect(vacatedColumnRect.x,vacatedColumnRect.y,
  303. vacatedColumnRect.width,vacatedColumnRect.height);
  304. //Painttheverticalgridlinesifnecessary.
  305. if(table.getShowVerticalLines()){
  306. g.setColor(table.getGridColor());
  307. intx1=vacatedColumnRect.x;
  308. inty1=vacatedColumnRect.y;
  309. intx2=x1+vacatedColumnRect.width-1;
  310. inty2=y1+vacatedColumnRect.height-1;
  311. //Left
  312. g.drawLine(x1-1,y1,x1-1,y2);
  313. //Right
  314. g.drawLine(x2,y1,x2,y2);
  315. }
  316. for(introw=rMin;row<=rMax;row++){
  317. //Renderthecellvalue
  318. Rectangler=table.getCellRect(row,draggedColumnIndex,false);
  319. r.x+=distance;
  320. paintCell(g,r,row,draggedColumnIndex);
  321. //Paintthe(lower)horizontalgridlineifnecessary.
  322. if(table.getShowHorizontalLines()){
  323. g.setColor(table.getGridColor());
  324. Rectanglercr=table.getCellRect(row,draggedColumnIndex,true);
  325. rcr.x+=distance;
  326. intx1=rcr.x;
  327. inty1=rcr.y;
  328. intx2=x1+rcr.width-1;
  329. inty2=y1+rcr.height-1;
  330. g.drawLine(x1,y2,x2,y2);
  331. }
  332. }
  333. }
  334. privateintviewIndexForColumn(TableColumnaColumn){
  335. TableColumnModelcm=table.getColumnModel();
  336. for(intcolumn=0;column<cm.getColumnCount();column++){
  337. if(cm.getColumn(column)==aColumn){
  338. returncolumn;
  339. }
  340. }
  341. return-1;
  342. }
  343. privateRectangleextendRect(Rectanglerect,booleanhorizontal){
  344. if(rect==null){
  345. returnrect;
  346. }
  347. if(horizontal){
  348. rect.x=0;
  349. rect.width=table.getWidth();
  350. }else{
  351. rect.y=0;
  352. if(table.getRowCount()!=0){
  353. RectanglelastRect=table.getCellRect(table.getRowCount()-1,0,true);
  354. rect.height=lastRect.y+lastRect.height;
  355. }else{
  356. rect.height=table.getHeight();
  357. }
  358. }
  359. returnrect;
  360. }
  361. privateRectanglegetVDropLineRect(JTable.DropLocationloc){
  362. if(!loc.isInsertColumn()){
  363. returnnull;
  364. }
  365. booleanltr=table.getComponentOrientation().isLeftToRight();
  366. intcol=loc.getColumn();
  367. Rectanglerect=table.getCellRect(loc.getRow(),col,true);
  368. if(col>=table.getColumnCount()){
  369. col--;
  370. rect=table.getCellRect(loc.getRow(),col,true);
  371. if(ltr){
  372. rect.x=rect.x+rect.width;
  373. }
  374. }elseif(!ltr){
  375. rect.x=rect.x+rect.width;
  376. }
  377. if(rect.x==0){
  378. rect.x=-1;
  379. }else{
  380. rect.x-=2;
  381. }
  382. rect.width=3;
  383. returnrect;
  384. }
  385. }//EndofClassBasicTableUI

测试代码:

Java代码 复制代码 收藏代码
  1. importjava.awt.BorderLayout;
  2. importjava.awt.event.ActionEvent;
  3. importjava.awt.event.ActionListener;
  4. importjavax.swing.JButton;
  5. importjavax.swing.JFrame;
  6. importjavax.swing.JScrollPane;
  7. importjavax.swing.table.DefaultTableModel;
  8. importcom.jrf.jgrid.guitools.gridbagtable.GridBagTable;
  9. publicclassTestimplementsActionListener{
  10. GridBagTabletable;
  11. publicTest()
  12. {
  13. JFramed=newJFrame();
  14. DefaultTableModelmodel=newDefaultTableModel(5,5);
  15. table=newGridBagTable(model);
  16. table.setRowHeight(20);
  17. JScrollPanepane=newJScrollPane(table);
  18. d.getContentPane().add(pane,BorderLayout.CENTER);
  19. JButtonbtn=newJButton("合并/拆分");
  20. d.getContentPane().add(btn,BorderLayout.NORTH);
  21. btn.addActionListener(this);
  22. d.setBounds(0,0,400,400);
  23. d.setVisible(true);
  24. }
  25. publicstaticvoidmain(String[]fsd){
  26. newTest();
  27. }
  28. publicvoidactionPerformed(ActionEvente){
  29. table.mergeCells(table.getSelectedRows(),table.getSelectedColumns());
  30. }
  31. }年前在网上参加了一个JavaSwing的招聘上机考试。招聘方要求开发一个类似EXCEL支持单元格合并的JTable。差不多用了5天的时间提交代码,最后被告知测试通过,我提出是否可做兼职,对方回复需要到上海做全职开发,最后也就放弃了。最近公司的一个项目中需要用到以前的代码,偶又重构了一次,设计思想来源于ListSelectionModel。

    Java Swing写的支持合并单元格的JTable

    GridBagModel:抽象模型接口。该接口用于描述表格中单元格的合并状态。
    DefaultGridBagTableModel:GridBagModel的默认实现。
    GridBagTable:继承自JTable的控制器。通过该类中的方法控制表格单元的合并和拆分。
    GridBagTableUI:GridBagTable对应的UI。

    TODO:(已合并)行、列的插入,删除操作对应的GridBagModel的修改,不过已留接口。

    Java代码 复制代码 收藏代码
    1. packageorg.dxj.guitools.gridbagtable;
    2. importjava.awt.Component;
    3. importjava.awt.Point;
    4. importjava.awt.Rectangle;
    5. importjava.util.Enumeration;
    6. importjava.util.EventObject;
    7. importjavax.swing.DefaultCellEditor;
    8. importjavax.swing.JTable;
    9. importjavax.swing.SwingUtilities;
    10. importjavax.swing.event.TableModelEvent;
    11. importjavax.swing.table.AbstractTableModel;
    12. importjavax.swing.table.TableColumn;
    13. importjavax.swing.table.TableColumnModel;
    14. /**
    15. *@[email protected]
    16. */
    17. publicclassGridBagTableextendsJTable{
    18. GridBagModelgridBagModel;
    19. publicGridBagModelgetGridBagModel(){
    20. returngridBagModel;
    21. }
    22. publicvoidsetGridBagModel(GridBagModelgridBagModel){
    23. if(gridBagModel!=null&&gridBagModel!=this.gridBagModel)
    24. this.gridBagModel=gridBagModel;
    25. }
    26. publicGridBagTable(AbstractTableModeldm){
    27. super(dm);
    28. getTableHeader().setReorderingAllowed(false);
    29. gridBagModel=newDefaultGridBagTableModel(dm);
    30. getColumnModel().setColumnSelectionAllowed(true);
    31. }
    32. privatevoidupdateSubComponentUI(ObjectcomponentShell){
    33. if(componentShell==null){
    34. return;
    35. }
    36. Componentcomponent=null;
    37. if(componentShellinstanceofComponent){
    38. component=(Component)componentShell;
    39. }
    40. if(componentShellinstanceofDefaultCellEditor){
    41. component=((DefaultCellEditor)componentShell).getComponent();
    42. }
    43. if(component!=null){
    44. SwingUtilities.updateComponentTreeUI(component);
    45. }
    46. }
    47. publicvoidupdateUI(){
    48. //UpdatetheUIsofthecellrenderers,celleditorsandheaderrenderers.
    49. TableColumnModelcm=getColumnModel();
    50. for(intcolumn=0;column<cm.getColumnCount();column++){
    51. TableColumnaColumn=cm.getColumn(column);
    52. updateSubComponentUI(aColumn.getCellRenderer());
    53. updateSubComponentUI(aColumn.getCellEditor());
    54. updateSubComponentUI(aColumn.getHeaderRenderer());
    55. }
    56. //UpdatetheUIsofallthedefaultrenderers.
    57. EnumerationdefaultRenderers=defaultRenderersByColumnClass.elements();
    58. while(defaultRenderers.hasMoreElements()){
    59. updateSubComponentUI(defaultRenderers.nextElement());
    60. }
    61. //UpdatetheUIsofallthedefaulteditors.
    62. EnumerationdefaultEditors=defaultEditorsByColumnClass.elements();
    63. while(defaultEditors.hasMoreElements()){
    64. updateSubComponentUI(defaultEditors.nextElement());
    65. }
    66. //UpdatetheUIofthetableheader
    67. if(tableHeader!=null&&tableHeader.getParent()==null){
    68. tableHeader.updateUI();
    69. }
    70. setUI(newGridBagTableUI());
    71. }
    72. publicRectanglegetGridCellRect(introw,intcolumn,booleanincludeSpacing){
    73. returnsuper.getCellRect(row,column,includeSpacing);
    74. }
    75. publicRectanglegetCellRect(introw,intcolumn,booleanincludeSpacing){
    76. RectanglecellRect=super.getCellRect(row,column,includeSpacing);
    77. intcols=gridBagModel.getColumnGrid(row,column);
    78. TableColumnModelcm=getColumnModel();
    79. for(intn=1;n<cols;n++)
    80. cellRect.width+=cm.getColumn(column+n).getWidth();
    81. introws=gridBagModel.getRowGrid(row,column);
    82. for(intn=1;n<rows;n++)
    83. cellRect.height+=getRowHeight(row+n);
    84. returncellRect;
    85. }
    86. publicvoidtableChanged(TableModelEvente){
    87. super.tableChanged(e);
    88. //TODO
    89. }
    90. publicbooleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn){
    91. if(gridBagModel.mergeCells(startRow,endRow,startColumn,endColumn)){
    92. repaint();
    93. returntrue;
    94. }
    95. returnfalse;
    96. }
    97. publicbooleanmergeCells(int[]rows,int[]columns){
    98. if(gridBagModel.mergeCells(rows,columns)){
    99. repaint();
    100. returntrue;
    101. }
    102. returnfalse;
    103. }
    104. publicbooleanspliteCellAt(introw,intcolumn){
    105. if(gridBagModel.spliteCellAt(row,column)){
    106. repaint();
    107. returntrue;
    108. }
    109. returnfalse;
    110. }
    111. publicvoidchangeSelection(introwIndex,intcolumnIndex,booleantoggle,booleanextend){
    112. if(gridBagModel.getCellState(rowIndex,columnIndex)!=GridBagModel.COVERED)
    113. super.changeSelection(rowIndex,columnIndex,toggle,extend);
    114. Pointp;
    115. for(introw=rowIndex;row>=0;row--){
    116. for(intcol=columnIndex;col>=0;col--){
    117. p=gridBagModel.getGrid(row,col);
    118. //p=((Point)((Vector)rowVector.get(row)).get(col));
    119. if(col+p.x>columnIndex&&row+p.y>rowIndex){
    120. rowIndex=row;
    121. columnIndex=col;
    122. break;
    123. }
    124. }
    125. }
    126. super.changeSelection(rowIndex,columnIndex,toggle,extend);
    127. repaint();
    128. }
    129. publicbooleaneditCellAt(introwIndex,intcolumnIndex,EventObjecte){
    130. if(gridBagModel.getCellState(rowIndex,columnIndex)!=GridBagModel.COVERED)
    131. returnsuper.editCellAt(rowIndex,columnIndex,e);
    132. Pointp;
    133. for(introw=rowIndex;row>=0;row--){
    134. for(intcol=columnIndex;col>=0;col--){
    135. p=gridBagModel.getGrid(row,col);
    136. if(col+p.x>columnIndex&&row+p.y>rowIndex){
    137. rowIndex=row;
    138. columnIndex=col;
    139. break;
    140. }
    141. }
    142. }
    143. returnsuper.editCellAt(rowIndex,columnIndex,e);
    144. }
    145. }
    Java代码 复制代码 收藏代码
    1. packageorg.dxj.guitools.gridbagtable;
    2. importjava.awt.Point;
    3. publicinterfaceGridBagModel{
    4. //格子处于正常状态
    5. intDEFAULT=0;
    6. //格子合并了其他的格子
    7. intMERGE=1;
    8. //格子被其他格子合并
    9. intCOVERED=-1;
    10. /**
    11. *@paramrow行
    12. *@paramcolumn列
    13. *@return该单元格在行、列的跨度
    14. */
    15. PointgetGrid(introw,intcolumn);
    16. /**
    17. *在Y轴方向的跨度
    18. *@paramrow
    19. *@paramcolumn
    20. *@return
    21. */
    22. intgetRowGrid(introw,intcolumn);
    23. /**
    24. *在X轴方向的跨度
    25. *@paramrow
    26. *@paramcolumn
    27. *@return
    28. */
    29. intgetColumnGrid(introw,intcolumn);
    30. /**
    31. *@paramrows行集合
    32. *@paramcolumns列集合
    33. *@return单元格集合是否可以合并在一起
    34. */
    35. booleancanMergeCells(int[]rows,int[]columns);
    36. /**
    37. *判断该单元格状态
    38. *@paramrow
    39. *@paramcolumn
    40. *@returnMERGE|DEFAULT|COVERED
    41. */
    42. intgetCellState(introw,intcolumn);
    43. /**
    44. *将单元格集合合并
    45. *@paramstartRow开始行
    46. *@paramendRow结束行
    47. *@paramstartColumn开始列
    48. *@paramendColumn结束列
    49. *@return是否合并成功
    50. */
    51. booleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn);
    52. /**
    53. *将单元格集合合并
    54. *@paramrows行集合
    55. *@paramcolumns列集合
    56. *@return是否合并成功
    57. */
    58. booleanmergeCells(int[]rows,int[]columns);
    59. /**
    60. *拆分单元格
    61. *@paramrow行
    62. *@paramcolumn列
    63. *@return是否拆分成功
    64. */
    65. booleanspliteCellAt(introw,intcolumn);
    66. /**
    67. *清除所有合并
    68. */
    69. voidclearMergence();
    70. }
    Java代码 复制代码 收藏代码
    1. packageorg.dxj.guitools.gridbagtable;
    2. importjava.awt.Point;
    3. importjava.util.Arrays;
    4. importjava.util.List;
    5. importjava.util.Vector;
    6. importjavax.swing.event.TableModelEvent;
    7. importjavax.swing.event.TableModelListener;
    8. importjavax.swing.table.AbstractTableModel;
    9. publicclassDefaultGridBagTableModelimplementsGridBagModel,TableModelListener{
    10. protectedAbstractTableModelmodel;
    11. protectedList<List<Point>>gridInfo;
    12. DefaultGridBagTableModel(AbstractTableModelmodel){
    13. gridInfo=newVector<List<Point>>();
    14. setTableModel(model);
    15. }
    16. publicvoidsetTableModel(AbstractTableModelmodel){
    17. if(model!=null&&model!=this.model){
    18. if(this.model!=null)
    19. this.model.removeTableModelListener(this);
    20. //防止多次添加监听器
    21. model.removeTableModelListener(this);
    22. model.addTableModelListener(this);
    23. this.model=model;
    24. clearMergence();
    25. }
    26. }
    27. publicvoidclearMergence(){
    28. if(gridInfo==null)
    29. gridInfo=newVector<List<Point>>();
    30. else
    31. gridInfo.clear();
    32. if(model==null)
    33. return;
    34. //初始化,每个格子占的格子数为(1,1);
    35. for(introw=model.getRowCount();--row>=0;){
    36. List<Point>infos=newVector<Point>();
    37. gridInfo.add(infos);
    38. for(intcol=model.getColumnCount();--col>=0;){
    39. infos.add(getDefaultPoint());
    40. }
    41. }
    42. }
    43. publicPointgetDefaultPoint(){
    44. returnnewPoint(1,1);
    45. }
    46. @Override
    47. publicbooleancanMergeCells(int[]rows,int[]columns){
    48. if(rows==null||columns==null)returnfalse;
    49. Arrays.sort(rows);
    50. for(intindex=0;index<rows.length-1;index++){
    51. if(rows[index+1]-rows[index]>1)
    52. returnfalse;
    53. }
    54. Arrays.sort(columns);
    55. for(intindex=0;index<columns.length-1;index++){
    56. if(columns[index+1]-columns[index]>1)
    57. returnfalse;
    58. }
    59. returntrue;
    60. }
    61. @Override
    62. publicintgetCellState(introw,intcolumn){
    63. Pointgrid=getGrid(row,column);
    64. if(grid==null)returnDEFAULT;
    65. if(grid.x>1||grid.y>1)
    66. returnMERGE;
    67. if(grid.x<=0||grid.y<=0)
    68. returnCOVERED;
    69. returnDEFAULT;
    70. }
    71. @Override
    72. publicintgetColumnGrid(introw,intcolumn){
    73. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
    74. List<Point>gridRow=gridInfo.get(row);
    75. if(gridRow!=null&&column>=0&&column<gridRow.size()){
    76. Pointpoint=gridRow.get(column);
    77. if(point!=null)
    78. returnpoint.x;
    79. }
    80. }
    81. return1;
    82. }
    83. @Override
    84. publicPointgetGrid(introw,intcolumn){
    85. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
    86. List<Point>gridRow=gridInfo.get(row);
    87. if(gridRow!=null&&column>=0&&column<gridRow.size()){
    88. returngridRow.get(column);
    89. }
    90. }
    91. returngetDefaultPoint();
    92. }
    93. @Override
    94. publicintgetRowGrid(introw,intcolumn){
    95. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
    96. List<Point>gridRow=gridInfo.get(row);
    97. if(gridRow!=null&&column>=0&&column<gridRow.size()){
    98. Pointpoint=gridRow.get(column);
    99. if(point!=null)
    100. returnpoint.y;
    101. }
    102. }
    103. return1;
    104. }
    105. protectedbooleansetGrid(introw,intcolumn,Pointgrid){
    106. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
    107. List<Point>gridRow=gridInfo.get(row);
    108. if(gridRow!=null&&column>=0&&column<gridRow.size()){
    109. Pointpoint=gridRow.get(column);
    110. if(point!=null){
    111. point.setLocation(grid);
    112. }
    113. else{
    114. gridRow.set(column,grid.getLocation());
    115. }
    116. returntrue;
    117. }
    118. }
    119. returnfalse;
    120. }
    121. @Override
    122. publicbooleanspliteCellAt(introw,intcolumn){
    123. if(gridInfo!=null&&row>=0&&row<gridInfo.size()){
    124. List<Point>gridRow=gridInfo.get(row);
    125. if(gridRow!=null&&column>=0&&column<gridRow.size()){
    126. Pointpoint=gridRow.get(column);
    127. if(point!=null){
    128. point=point.getLocation();
    129. for(inta=0;a<point.y;a++){
    130. for(intb=0;b<point.x;b++){
    131. setGrid(row+a,column+b,getDefaultPoint());
    132. }
    133. }
    134. }
    135. else{
    136. gridRow.set(column,getDefaultPoint());
    137. }
    138. returntrue;
    139. }
    140. }
    141. returnfalse;
    142. }
    143. @Override
    144. /**
    145. *table中发生行的添加和删除的时候需要修改该模型
    146. */
    147. publicvoidtableChanged(TableModelEvente){
    148. //TODO
    149. }
    150. @Override
    151. publicbooleanmergeCells(int[]rows,int[]columns){
    152. if(!canMergeCells(rows,columns))
    153. returnfalse;
    154. Arrays.sort(rows);
    155. Arrays.sort(columns);
    156. returnmergeCells(rows[0],rows[rows.length-1],columns[0],columns[columns.length-1]);
    157. }
    158. @Override
    159. publicbooleanmergeCells(intstartRow,intendRow,intstartColumn,intendColumn){
    160. setGrid(startRow,startColumn,newPoint(endColumn-startColumn+1,endRow-startRow+1));
    161. for(introw=startRow;row<=endRow;row++){
    162. for(intcol=startColumn;col<=endColumn;col++){
    163. if(row==startRow&&col==startColumn)
    164. continue;
    165. else
    166. setGrid(row,col,newPoint(COVERED,COVERED));
    167. }
    168. }
    169. returntrue;
    170. }
    171. publicStringtoString(){
    172. if(gridInfo==null)
    173. return"";
    174. StringBuffersb=newStringBuffer();
    175. for(List<Point>rowInfo:gridInfo){
    176. for(Pointgrid:rowInfo){
    177. sb.append("["+grid.x+","+grid.y+"],");
    178. }
    179. sb.append("\n");
    180. }
    181. returnsb.toString();
    182. }
    183. }
    Java代码 复制代码 收藏代码
    1. packageorg.dxj.guitools.gridbagtable;
    2. importjava.awt.Color;
    3. importjava.awt.Component;
    4. importjava.awt.Dimension;
    5. importjava.awt.Graphics;
    6. importjava.awt.Point;
    7. importjava.awt.Rectangle;
    8. importjava.util.Enumeration;
    9. importjavax.swing.BorderFactory;
    10. importjavax.swing.JComponent;
    11. importjavax.swing.JTable;
    12. importjavax.swing.UIManager;
    13. importjavax.swing.plaf.basic.BasicTableUI;
    14. importjavax.swing.table.JTableHeader;
    15. importjavax.swing.table.TableCellRenderer;
    16. importjavax.swing.table.TableColumn;
    17. importjavax.swing.table.TableColumnModel;
    18. publicclassGridBagTableUIextendsBasicTableUI
    19. {
    20. publicDimensiongetPreferredSize(JComponentc){
    21. longwidth=0;
    22. Enumeration<TableColumn>enumeration=table.getColumnModel().getColumns();
    23. while(enumeration.hasMoreElements()){
    24. TableColumnaColumn=(TableColumn)enumeration.nextElement();
    25. width=width+aColumn.getPreferredWidth();
    26. }
    27. returncreateTableSize(width);
    28. }
    29. privateDimensioncreateTableSize(longwidth){
    30. intheight=0;
    31. introwCount=table.getRowCount();
    32. if(rowCount>0&&table.getColumnCount()>0){
    33. Rectangler=table.getCellRect(rowCount-1,0,true);
    34. height=r.y+r.height;
    35. }
    36. //Widthisalwayspositive.Thecalltoabs()isaworkaroundfor
    37. //abuginthe1.1.6JITonWindows.
    38. longtmp=Math.abs(width);
    39. if(tmp>Integer.MAX_VALUE){
    40. tmp=Integer.MAX_VALUE;
    41. }
    42. returnnewDimension((int)tmp,height);
    43. }
    44. publicvoidpaint(Graphicsg,JComponentc){
    45. Rectangleclip=g.getClipBounds();
    46. Rectanglebounds=table.getBounds();
    47. //accountforthefactthatthegraphicshasalreadybeentranslated
    48. //intothetable'sbounds
    49. bounds.x=bounds.y=0;
    50. if(table.getRowCount()<=0||table.getColumnCount()<=0||
    51. //thischeckpreventsusfrompaintingtheentiretable
    52. //whentheclipdoesn'tintersectourboundsatall
    53. !bounds.intersects(clip)){
    54. paintDropLines(g);
    55. return;
    56. }
    57. booleanltr=table.getComponentOrientation().isLeftToRight();
    58. PointupperLeft=clip.getLocation();
    59. if(!ltr){
    60. upperLeft.x++;
    61. }
    62. PointlowerRight=newPoint(clip.x+clip.width-(ltr?1:0),
    63. clip.y+clip.height);
    64. intrMin=table.rowAtPoint(upperLeft);
    65. intrMax=table.rowAtPoint(lowerRight);
    66. //Thisshouldneverhappen(aslongasourboundsintersecttheclip,
    67. //whichiswhywebailaboveifthatisthecase).
    68. if(rMin==-1){
    69. rMin=0;
    70. }
    71. //Ifthetabledoesnothaveenoughrowstofilltheviewwe'llget-1.
    72. //(Wecouldalsoget-1ifourboundsdon'tintersecttheclip,
    73. //whichiswhywebailaboveifthatisthecase).
    74. //Replacethiswiththeindexofthelastrow.
    75. if(rMax==-1){
    76. rMax=table.getRowCount()-1;
    77. }
    78. intcMin=table.columnAtPoint(ltr?upperLeft:lowerRight);
    79. intcMax=table.columnAtPoint(ltr?lowerRight:upperLeft);
    80. //Thisshouldneverhappen.
    81. if(cMin==-1){
    82. cMin=0;
    83. }
    84. //Ifthetabledoesnothaveenoughcolumnstofilltheviewwe'llget-1.
    85. //Replacethiswiththeindexofthelastcolumn.
    86. if(cMax==-1){
    87. cMax=table.getColumnCount()-1;
    88. }
    89. //Paintthegrid.
    90. //paintGrid(g,rMin,rMax,cMin,cMax);
    91. //Paintthecells.
    92. paintCells(g,rMin,rMax,cMin,cMax);
    93. paintDropLines(g);
    94. }
    95. privatevoidpaintDropLines(Graphicsg){
    96. JTable.DropLocationloc=table.getDropLocation();
    97. if(loc==null){
    98. return;
    99. }
    100. Colorcolor=UIManager.getColor("Table.dropLineColor");
    101. ColorshortColor=UIManager.getColor("Table.dropLineShortColor");
    102. if(color==null&&shortColor==null){
    103. return;
    104. }
    105. Rectanglerect;
    106. rect=getHDropLineRect(loc);
    107. if(rect!=null){
    108. intx=rect.x;
    109. intw=rect.width;
    110. if(color!=null){
    111. extendRect(rect,true);
    112. g.setColor(color);
    113. g.fillRect(rect.x,rect.y,rect.width,rect.height);
    114. }
    115. if(!loc.isInsertColumn()&&shortColor!=null){
    116. g.setColor(shortColor);
    117. g.fillRect(x,rect.y,w,rect.height);
    118. }
    119. }
    120. rect=getVDropLineRect(loc);
    121. if(rect!=null){
    122. inty=rect.y;
    123. inth=rect.height;
    124. if(color!=null){
    125. extendRect(rect,false);
    126. g.setColor(color);
    127. g.fillRect(rect.x,rect.y,rect.width,rect.height);
    128. }
    129. if(!loc.isInsertRow()&&shortColor!=null){
    130. g.setColor(shortColor);
    131. g.fillRect(rect.x,y,rect.width,h);
    132. }
    133. }
    134. }
    135. /*
    136. *Paintsthegridlineswithin<I>aRect</I>,usingthegrid
    137. *colorsetwith<I>setGridColor</I>.Paintsverticallines
    138. *if<code>getShowVerticalLines()</code>returnstrueandpaints
    139. *horizontallinesif<code>getShowHorizontalLines()</code>
    140. *returnstrue.
    141. */
    142. privatevoidpaintGrid(Graphicsg,intrMin,intrMax,intcMin,intcMax){
    143. g.setColor(table.getGridColor());
    144. RectangleminCell=table.getCellRect(rMin,cMin,true);
    145. RectanglemaxCell=table.getCellRect(rMax,cMax,true);
    146. RectangledamagedArea=minCell.union(maxCell);
    147. if(table.getShowHorizontalLines()){
    148. inttableWidth=damagedArea.x+damagedArea.width;
    149. inty=damagedArea.y;
    150. for(introw=rMin;row<=rMax;row++){
    151. y+=table.getRowHeight(row);
    152. g.drawLine(damagedArea.x,y-1,tableWidth-1,y-1);
    153. }
    154. }
    155. if(table.getShowVerticalLines()){
    156. TableColumnModelcm=table.getColumnModel();
    157. inttableHeight=damagedArea.y+damagedArea.height;
    158. intx;
    159. if(table.getComponentOrientation().isLeftToRight()){
    160. x=damagedArea.x;
    161. for(intcolumn=cMin;column<=cMax;column++){
    162. intw=cm.getColumn(column).getWidth();
    163. x+=w;
    164. g.drawLine(x-1,0,x-1,tableHeight-1);
    165. }
    166. }else{
    167. x=damagedArea.x;
    168. for(intcolumn=cMax;column>=cMin;column--){
    169. intw=cm.getColumn(column).getWidth();
    170. x+=w;
    171. g.drawLine(x-1,0,x-1,tableHeight-1);
    172. }
    173. }
    174. }
    175. }
    176. privatevoidpaintCells(Graphicsg,intrMin,intrMax,intcMin,intcMax){
    177. JTableHeaderheader=table.getTableHeader();
    178. TableColumndraggedColumn=(header==null)?null:header.getDraggedColumn();
    179. TableColumnModelcm=table.getColumnModel();
    180. intcolumnMargin=cm.getColumnMargin();
    181. RectanglecellRect;
    182. TableColumnaColumn;
    183. intcolumnWidth;
    184. if(table.getComponentOrientation().isLeftToRight()){
    185. for(introw=rMin;row<=rMax;row++){
    186. if(tableinstanceofGridBagTable)
    187. cellRect=((GridBagTable)table).getGridCellRect(row,cMin,false);
    188. else
    189. cellRect=table.getCellRect(row,cMin,false);
    190. for(intcolumn=cMin;column<=cMax;column++){
    191. aColumn=cm.getColumn(column);
    192. columnWidth=aColumn.getWidth();
    193. //TODO
    194. cellRect.width=columnWidth-columnMargin;
    195. intoldHeight=cellRect.height;
    196. if(tableinstanceofGridBagTable){
    197. if(((GridBagTable)table).getGridBagModel().getCellState(row,column)==GridBagModel.COVERED){
    198. cellRect.width=0;
    199. cellRect.height=0;
    200. }
    201. else{
    202. inth=((GridBagTable)table).getGridBagModel().getColumnGrid(row,column);
    203. if(h>1){
    204. for(intn=1;n<h;n++)
    205. cellRect.width+=cm.getColumn(column+n).getWidth();
    206. }
    207. intv=((GridBagTable)table).getGridBagModel().getRowGrid(row,column);
    208. if(v>1){
    209. for(intn=1;n<v;n++)
    210. cellRect.height+=table.getRowHeight(row+n);
    211. }
    212. }
    213. }
    214. if(aColumn!=draggedColumn){
    215. paintCell(g,cellRect,row,column);
    216. }
    217. cellRect.height=oldHeight;
    218. cellRect.x+=columnWidth;
    219. }
    220. }
    221. }else{
    222. for(introw=rMin;row<=rMax;row++){
    223. cellRect=table.getCellRect(row,cMin,false);
    224. aColumn=cm.getColumn(cMin);
    225. if(aColumn!=draggedColumn){
    226. columnWidth=aColumn.getWidth();
    227. cellRect.width=columnWidth-columnMargin;
    228. paintCell(g,cellRect,row,cMin);
    229. }
    230. for(intcolumn=cMin+1;column<=cMax;column++){
    231. aColumn=cm.getColumn(column);
    232. columnWidth=aColumn.getWidth();
    233. //TODO
    234. cellRect.width=columnWidth-columnMargin;
    235. cellRect.x-=columnWidth;
    236. if(aColumn!=draggedColumn){
    237. paintCell(g,cellRect,row,column);
    238. }
    239. }
    240. }
    241. }
    242. //Paintthedraggedcolumnifwearedragging.
    243. if(draggedColumn!=null){
    244. paintDraggedArea(g,rMin,rMax,draggedColumn,header.getDraggedDistance());
    245. }
    246. //RemoveanyrenderersthatmaybeleftintherendererPane.
    247. rendererPane.removeAll();
    248. }
    249. privatevoidpaintCell(Graphicsg,RectanglecellRect,introw,intcolumn){
    250. if(table.isEditing()&&table.getEditingRow()==row&&
    251. table.getEditingColumn()==column){
    252. Componentcomponent=table.getEditorComponent();
    253. component.setBounds(cellRect);
    254. component.validate();
    255. }
    256. else{
    257. TableCellRendererrenderer=table.getCellRenderer(row,column);
    258. Componentcomponent=table.prepareRenderer(renderer,row,column);
    259. if(componentinstanceofJComponent){
    260. ((JComponent)component).setBorder(BorderFactory.createLineBorder(Color.gray));
    261. }
    262. rendererPane.paintComponent(g,component,table,cellRect.x,cellRect.y,
    263. cellRect.width,cellRect.height,true);
    264. }
    265. }
    266. privateRectanglegetHDropLineRect(JTable.DropLocationloc){
    267. if(!loc.isInsertRow()){
    268. returnnull;
    269. }
    270. introw=loc.getRow();
    271. intcol=loc.getColumn();
    272. if(col>=table.getColumnCount()){
    273. col--;
    274. }
    275. Rectanglerect=table.getCellRect(row,col,true);
    276. if(row>=table.getRowCount()){
    277. row--;
    278. RectangleprevRect=table.getCellRect(row,col,true);
    279. rect.y=prevRect.y+prevRect.height;
    280. }
    281. if(rect.y==0){
    282. rect.y=-1;
    283. }else{
    284. rect.y-=2;
    285. }
    286. rect.height=3;
    287. returnrect;
    288. }
    289. privatevoidpaintDraggedArea(Graphicsg,intrMin,intrMax,TableColumndraggedColumn,intdistance){
    290. intdraggedColumnIndex=viewIndexForColumn(draggedColumn);
    291. RectangleminCell=table.getCellRect(rMin,draggedColumnIndex,true);
    292. RectanglemaxCell=table.getCellRect(rMax,draggedColumnIndex,true);
    293. RectanglevacatedColumnRect=minCell.union(maxCell);
    294. //Paintagraywellinplaceofthemovingcolumn.
    295. g.setColor(table.getParent().getBackground());
    296. g.fillRect(vacatedColumnRect.x,vacatedColumnRect.y,
    297. vacatedColumnRect.width,vacatedColumnRect.height);
    298. //Movetothewherethecellhasbeendragged.
    299. vacatedColumnRect.x+=distance;
    300. //Fillthebackground.
    301. g.setColor(table.getBackground());
    302. g.fillRect(vacatedColumnRect.x,vacatedColumnRect.y,
    303. vacatedColumnRect.width,vacatedColumnRect.height);
    304. //Painttheverticalgridlinesifnecessary.
    305. if(table.getShowVerticalLines()){
    306. g.setColor(table.getGridColor());
    307. intx1=vacatedColumnRect.x;
    308. inty1=vacatedColumnRect.y;
    309. intx2=x1+vacatedColumnRect.width-1;
    310. inty2=y1+vacatedColumnRect.height-1;
    311. //Left
    312. g.drawLine(x1-1,y1,x1-1,y2);
    313. //Right
    314. g.drawLine(x2,y1,x2,y2);
    315. }
    316. for(introw=rMin;row<=rMax;row++){
    317. //Renderthecellvalue
    318. Rectangler=table.getCellRect(row,draggedColumnIndex,false);
    319. r.x+=distance;
    320. paintCell(g,r,row,draggedColumnIndex);
    321. //Paintthe(lower)horizontalgridlineifnecessary.
    322. if(table.getShowHorizontalLines()){
    323. g.setColor(table.getGridColor());
    324. Rectanglercr=table.getCellRect(row,draggedColumnIndex,true);
    325. rcr.x+=distance;
    326. intx1=rcr.x;
    327. inty1=rcr.y;
    328. intx2=x1+rcr.width-1;
    329. inty2=y1+rcr.height-1;
    330. g.drawLine(x1,y2,x2,y2);
    331. }
    332. }
    333. }
    334. privateintviewIndexForColumn(TableColumnaColumn){
    335. TableColumnModelcm=table.getColumnModel();
    336. for(intcolumn=0;column<cm.getColumnCount();column++){
    337. if(cm.getColumn(column)==aColumn){
    338. returncolumn;
    339. }
    340. }
    341. return-1;
    342. }
    343. privateRectangleextendRect(Rectanglerect,booleanhorizontal){
    344. if(rect==null){
    345. returnrect;
    346. }
    347. if(horizontal){
    348. rect.x=0;
    349. rect.width=table.getWidth();
    350. }else{
    351. rect.y=0;
    352. if(table.getRowCount()!=0){
    353. RectanglelastRect=table.getCellRect(table.getRowCount()-1,0,true);
    354. rect.height=lastRect.y+lastRect.height;
    355. }else{
    356. rect.height=table.getHeight();
    357. }
    358. }
    359. returnrect;
    360. }
    361. privateRectanglegetVDropLineRect(JTable.DropLocationloc){
    362. if(!loc.isInsertColumn()){
    363. returnnull;
    364. }
    365. booleanltr=table.getComponentOrientation().isLeftToRight();
    366. intcol=loc.getColumn();
    367. Rectanglerect=table.getCellRect(loc.getRow(),col,true);
    368. if(col>=table.getColumnCount()){
    369. col--;
    370. rect=table.getCellRect(loc.getRow(),col,true);
    371. if(ltr){
    372. rect.x=rect.x+rect.width;
    373. }
    374. }elseif(!ltr){
    375. rect.x=rect.x+rect.width;
    376. }
    377. if(rect.x==0){
    378. rect.x=-1;
    379. }else{
    380. rect.x-=2;
    381. }
    382. rect.width=3;
    383. returnrect;
    384. }
    385. }//EndofClassBasicTableUI

    测试代码:

    Java代码 复制代码 收藏代码
    1. importjava.awt.BorderLayout;
    2. importjava.awt.event.ActionEvent;
    3. importjava.awt.event.ActionListener;
    4. importjavax.swing.JButton;
    5. importjavax.swing.JFrame;
    6. importjavax.swing.JScrollPane;
    7. importjavax.swing.table.DefaultTableModel;
    8. importcom.jrf.jgrid.guitools.gridbagtable.GridBagTable;
    9. publicclassTestimplementsActionListener{
    10. GridBagTabletable;
    11. publicTest()
    12. {
    13. JFramed=newJFrame();
    14. DefaultTableModelmodel=newDefaultTableModel(5,5);
    15. table=newGridBagTable(model);
    16. table.setRowHeight(20);
    17. JScrollPanepane=newJScrollPane(table);
    18. d.getContentPane().add(pane,BorderLayout.CENTER);
    19. JButtonbtn=newJButton("合并/拆分");
    20. d.getContentPane().add(btn,BorderLayout.NORTH);
    21. btn.addActionListener(this);
    22. d.setBounds(0,0,400,400);
    23. d.setVisible(true);
    24. }
    25. publicstaticvoidmain(String[]fsd){
    26. newTest();
    27. }
    28. publicvoidactionPerformed(ActionEvente){
    29. table.mergeCells(table.getSelectedRows(),table.getSelectedColumns());
    30. }
    31. }

你可能感兴趣的:(java,swing)