View Javadoc

1   /**
2    * Copyright 2005-2013 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.krad.uif.layout;
17  
18  import java.util.ArrayList;
19  import java.util.HashMap;
20  import java.util.List;
21  import java.util.Map;
22  
23  import org.apache.commons.lang.StringUtils;
24  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
25  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
26  import org.kuali.rice.krad.datadictionary.parse.BeanTags;
27  import org.kuali.rice.krad.uif.component.Component;
28  import org.kuali.rice.krad.uif.container.Container;
29  import org.kuali.rice.krad.util.KRADUtils;
30  
31  /**
32   * Css Grid Layout manager is a layout manager which creates div "rows" and "cells" to replicate a
33   * table look by using div elements for its items. Items are added into rows based on their colSpan
34   * setting, while each row has a max size of 12 columns. By default, if colSpan is not set on an
35   * item, that item will take a full row.
36   */
37  @BeanTags({@BeanTag(name = "cssGridLayout-bean", parent = "Uif-CssGridLayoutBase"),
38          @BeanTag(name = "fixedCssGridLayout-bean", parent = "Uif-FixedCssGridLayout"),
39          @BeanTag(name = "fluidCssGridLayout-bean", parent = "Uif-FluidCssGridLayout")})
40  public class CssGridLayoutManager extends LayoutManagerBase {
41      private static final long serialVersionUID = 1830635073147703757L;
42  
43      private static final int NUMBER_OF_COLUMNS = 12;
44      private static final String BOOTSTRAP_SPAN_PREFIX = "col-md-";
45  
46      private Map<String, String> conditionalRowCssClasses;
47      private String rowLayoutCssClass;
48      private int defaultItemColSpan;
49  
50      // non-settable
51      protected List<List<Component>> rows;
52      protected List<String> rowCssClassAttributes;
53      protected List<String> cellCssClassAttributes;
54  
55      public CssGridLayoutManager() {
56          rows = new ArrayList<List<Component>>();
57          conditionalRowCssClasses = new HashMap<String, String>();
58          cellCssClassAttributes = new ArrayList<String>();
59          rowCssClassAttributes = new ArrayList<String>();
60      }
61  
62      /**
63       * CssGridLayoutManager's performFinalize method calculates and separates the items into rows
64       * based on their colSpan settings and the defaultItemColSpan setting
65       * 
66       * @see Component#performFinalize(org.kuali.rice.krad.uif.view.View, Object,
67       *      org.kuali.rice.krad.uif.component.Component)
68       */
69      @Override
70      public void performFinalize(Object model, Component component) {
71          super.performFinalize(model, component);
72          
73          Container container = (Container) component;
74  
75          int rowSpaceLeft = NUMBER_OF_COLUMNS;
76          int rowIndex = 0;
77          boolean isOdd = true;
78          List<Component> currentRow = new ArrayList<Component>();
79          for (Component item : container.getItems()) {
80              if (item == null) {
81                  continue;
82              }
83              isOdd = rowIndex % 2 == 0;
84  
85              // set colSpan to default setting (12 is the default)
86              int colSpan = this.defaultItemColSpan;
87  
88              // if the item's set colSpan is greater than 1 set it to that number; 1 is the default colSpan for Component
89              if (item.getColSpan() > 1 && item.getColSpan() <= NUMBER_OF_COLUMNS) {
90                  colSpan = item.getColSpan();
91              }
92  
93              // determine "cell" div css
94              List<String> cellCssClasses = item.getCellCssClasses();
95              if (cellCssClasses == null) {
96                  item.setCellCssClasses(cellCssClasses = new ArrayList<String>());
97              }
98              cellCssClasses.add(0, BOOTSTRAP_SPAN_PREFIX + colSpan);
99              cellCssClassAttributes.add(getCellStyleClassesAsString(cellCssClasses));
100 
101             // calculate space left in row
102             rowSpaceLeft = rowSpaceLeft - colSpan;
103 
104             if (rowSpaceLeft > 0) {
105                 // space is left, just add item to row
106                 currentRow.add(item);
107             } else if (rowSpaceLeft < 0) {
108                 // went over, add item to next new row
109                 rows.add(new ArrayList<Component>(currentRow));
110                 currentRow = new ArrayList<Component>();
111                 currentRow.add(item);
112 
113                 // determine "row" div css
114                 String rowCss = rowLayoutCssClass + " " + KRADUtils.generateRowCssClassString(conditionalRowCssClasses,
115                         rowIndex, isOdd, null, null);
116                 rowCssClassAttributes.add(rowCss);
117 
118                 rowIndex++;
119                 rowSpaceLeft = NUMBER_OF_COLUMNS - colSpan;
120             } else if (rowSpaceLeft == 0) {
121                 // last item in row, create new row
122                 currentRow.add(item);
123                 rows.add(new ArrayList<Component>(currentRow));
124                 currentRow = new ArrayList<Component>();
125 
126                 // determine "row" div css
127                 String rowCss = rowLayoutCssClass + " " + KRADUtils.generateRowCssClassString(conditionalRowCssClasses,
128                         rowIndex, isOdd, null, null);
129                 rowCssClassAttributes.add(rowCss);
130 
131                 rowIndex++;
132                 rowSpaceLeft = NUMBER_OF_COLUMNS;
133             }
134         }
135 
136         isOdd = rowIndex % 2 == 0;
137         // add the last row if it wasn't full (but has items)
138         if (!currentRow.isEmpty()) {
139             // determine "row" div css
140             String rowCss = rowLayoutCssClass + " " + KRADUtils.generateRowCssClassString(conditionalRowCssClasses,
141                     rowIndex, isOdd, null, null);
142             rowCssClassAttributes.add(rowCss);
143 
144             rows.add(currentRow);
145         }
146     }
147 
148     /**
149      * Builds the HTML class attribute string by combining the cellStyleClasses list with a space
150      * delimiter
151      * 
152      * @return class attribute string
153      */
154     private String getCellStyleClassesAsString(List<String> cellCssClasses) {
155         if (cellCssClasses != null) {
156             return StringUtils.join(cellCssClasses, " ");
157         }
158 
159         return "";
160     }
161 
162     /**
163      * Get the rows (which are a list of components each)
164      * 
165      * @return the List of Lists of Components which represents rows for this layout
166      */
167     public List<List<Component>> getRows() {
168         return rows;
169     }
170 
171     /**
172      * List of css class HTML attribute values ordered by index of row
173      * 
174      * @return the list of css class HTML attributes for rows
175      */
176     public List<String> getRowCssClassAttributes() {
177         return rowCssClassAttributes;
178     }
179 
180     /**
181      * List of css class HTML attribute values ordered by the order in which the cell appears
182      * 
183      * @return the list of css class HTML attributes for cells
184      */
185     public List<String> getCellCssClassAttributes() {
186         return cellCssClassAttributes;
187     }
188 
189     /**
190      * The default cell colSpan to use for this layout (max setting, and the bean default, is 12)
191      *
192      * @return int representing the default colSpan for cells in this layout
193      */
194     @BeanTagAttribute(name = "defaultItemColSpan")
195     public int getDefaultItemColSpan() {
196         return defaultItemColSpan;
197     }
198 
199     /**
200      * Set the default colSpan for this layout's items
201      * 
202      * @param defaultItemColSpan
203      */
204     public void setDefaultItemColSpan(int defaultItemColSpan) {
205         this.defaultItemColSpan = defaultItemColSpan;
206     }
207 
208     /**
209      * The row css classes for the rows of this layout
210      * 
211      * <p>
212      * To set a css class on all rows, use "all" as a key. To set a class for even rows, use "even"
213      * as a key, for odd rows, use "odd". Use a one-based index to target a specific row by index.
214      * </p>
215      * 
216      * @return a map which represents the css classes of the rows of this layout
217      */
218     @BeanTagAttribute(name = "conditionalRowCssClasses", type = BeanTagAttribute.AttributeType.MAPVALUE)
219     public Map<String, String> getConditionalRowCssClasses() {
220         return conditionalRowCssClasses;
221     }
222 
223     /**
224      * Set conditionalRowCssClasses
225      * 
226      * @param conditionalRowCssClasses
227      */
228     public void setConditionalRowCssClasses(Map<String, String> conditionalRowCssClasses) {
229         this.conditionalRowCssClasses = conditionalRowCssClasses;
230     }
231 
232     /**
233      * The layout css class used by the framework to represent the row as a row visually (currently
234      * using a bootstrap class), which should not be manually reset in most situations
235      * 
236      * @return the css structure class for the rows of this layout
237      */
238     @BeanTagAttribute(name = "rowLayoutCssClass")
239     public String getRowLayoutCssClass() {
240         return rowLayoutCssClass;
241     }
242 
243     /**
244      * Set the rowLayoutCssClass
245      * 
246      * @param rowLayoutCssClass
247      */
248     public void setRowLayoutCssClass(String rowLayoutCssClass) {
249         this.rowLayoutCssClass = rowLayoutCssClass;
250     }
251 
252     /**
253      * @see org.kuali.rice.krad.datadictionary.DictionaryBeanBase#copyProperties(Object)
254      */
255     @Override
256     protected <T> void copyProperties(T component) {
257         super.copyProperties(component);
258 
259         CssGridLayoutManager cssGridLayoutManagerCopy = (CssGridLayoutManager) component;
260 
261         if (this.rowLayoutCssClass != null) {
262             cssGridLayoutManagerCopy.setRowLayoutCssClass(this.rowLayoutCssClass);
263         }
264 
265         cssGridLayoutManagerCopy.setDefaultItemColSpan(this.defaultItemColSpan);
266 
267         if (this.conditionalRowCssClasses != null) {
268             cssGridLayoutManagerCopy.setConditionalRowCssClasses(new HashMap<String, String>(
269                     this.conditionalRowCssClasses));
270         }
271 
272         if (this.cellCssClassAttributes != null) {
273             cssGridLayoutManagerCopy.cellCssClassAttributes = new ArrayList<String>(this.cellCssClassAttributes);
274         }
275 
276         if (this.rowCssClassAttributes != null) {
277             cssGridLayoutManagerCopy.rowCssClassAttributes = new ArrayList<String>(this.rowCssClassAttributes);
278         }
279 
280         if (this.rows != null) {
281             cssGridLayoutManagerCopy.rows = new ArrayList<List<Component>>();
282             for (List<Component> row : this.rows) {
283                 List<Component> rowCopy = new ArrayList<Component>();
284 
285                 if (row == null) {
286                     cssGridLayoutManagerCopy.rows.add(row);
287                     continue;
288                 }
289 
290                 for (Component cellComp : row) {
291                     if (cellComp == null) {
292                         rowCopy.add(cellComp);
293                         continue;
294                     }
295                     rowCopy.add((Component) cellComp.copy());
296                 }
297 
298                 cssGridLayoutManagerCopy.rows.add(rowCopy);
299             }
300         }
301     }
302 }