View Javadoc
1   /**
2    * Copyright 2005-2014 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.container;
17  
18  import java.io.Serializable;
19  import java.util.ArrayList;
20  import java.util.Collection;
21  import java.util.HashMap;
22  import java.util.List;
23  import java.util.Map;
24  
25  import org.apache.commons.collections.ListUtils;
26  import org.apache.commons.lang.StringUtils;
27  import org.apache.commons.logging.Log;
28  import org.apache.commons.logging.LogFactory;
29  import org.kuali.rice.core.api.mo.common.active.Inactivatable;
30  import org.kuali.rice.krad.uif.UifConstants;
31  import org.kuali.rice.krad.uif.UifParameters;
32  import org.kuali.rice.krad.uif.UifPropertyPaths;
33  import org.kuali.rice.krad.uif.component.Component;
34  import org.kuali.rice.krad.uif.container.collections.LineBuilderContext;
35  import org.kuali.rice.krad.uif.element.Action;
36  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
37  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycleUtils;
38  import org.kuali.rice.krad.uif.util.ComponentUtils;
39  import org.kuali.rice.krad.uif.util.ContextUtils;
40  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
41  import org.kuali.rice.krad.uif.util.ScriptUtils;
42  import org.kuali.rice.krad.uif.view.ExpressionEvaluator;
43  import org.kuali.rice.krad.uif.view.FormView;
44  import org.kuali.rice.krad.uif.view.View;
45  import org.kuali.rice.krad.uif.view.ViewModel;
46  import org.kuali.rice.krad.util.KRADUtils;
47  import org.kuali.rice.krad.web.form.UifFormBase;
48  
49  /**
50   * Builds out the {@link org.kuali.rice.krad.uif.field.Field} instances for a collection group with a
51   * series of steps that interact with the configured {@link org.kuali.rice.krad.uif.layout.CollectionLayoutManager}
52   * to assemble the fields as necessary for the layout.
53   *
54   * @author Kuali Rice Team (rice.collab@kuali.org)
55   */
56  public class CollectionGroupBuilder implements Serializable {
57      private static final long serialVersionUID = -4762031957079895244L;
58      private static Log LOG = LogFactory.getLog(CollectionGroupBuilder.class);
59  
60      /**
61       * Invoked within the lifecycle to carry out the collection build process.
62       *
63       * <p>The corresponding collection is retrieved from the model and iterated
64       * over to create the necessary fields. The binding path for fields that
65       * implement {@code DataBinding} is adjusted to point to the collection
66       * line it is apart of. For example, field 'number' of collection 'accounts'
67       * for line 1 will be set to 'accounts[0].number', and for line 2
68       * 'accounts[1].number'. Finally parameters are set on the line's action
69       * fields to indicate what collection and line they apply to.</p>
70       *
71       * <p>Only the lines that are to be rendered (as specified by the displayStart
72       * and displayLength properties of the CollectionGroup) will be built.</p>
73       *
74       * @param view View instance the collection belongs to
75       * @param model Top level object containing the data
76       * @param collectionGroup CollectionGroup component for the collection
77       */
78      public void build(View view, Object model, CollectionGroup collectionGroup) {
79          // create add line
80          if (collectionGroup.isRenderAddLine() && !Boolean.TRUE.equals(collectionGroup.getReadOnly()) &&
81                  !collectionGroup.isRenderAddBlankLineButton()) {
82              buildAddLine(view, model, collectionGroup);
83          }
84  
85          // if add line button enabled setup to refresh the collection group
86          if (collectionGroup.isRenderAddBlankLineButton() && (collectionGroup.getAddBlankLineAction() != null)) {
87              collectionGroup.getAddBlankLineAction().setRefreshId(collectionGroup.getId());
88          }
89  
90          // get the collection for this group from the model
91          List<Object> modelCollection = ObjectPropertyUtils.getPropertyValue(model,
92                  collectionGroup.getBindingInfo().getBindingPath());
93  
94          if (modelCollection == null) {
95              return;
96          }
97  
98          // filter inactive model
99          List<Integer> showIndexes = performCollectionFiltering(view, model, collectionGroup, modelCollection);
100 
101         if (collectionGroup.getDisplayCollectionSize() != -1 && showIndexes.size() > collectionGroup
102                 .getDisplayCollectionSize()) {
103             // remove all indexes in showIndexes beyond the collection's size limitation
104             List<Integer> newShowIndexes = new ArrayList<Integer>();
105             Integer counter = 0;
106 
107             for (int index = 0; index < showIndexes.size(); index++) {
108                 newShowIndexes.add(showIndexes.get(index));
109 
110                 counter++;
111 
112                 if (counter == collectionGroup.getDisplayCollectionSize()) {
113                     break;
114                 }
115             }
116 
117             showIndexes = newShowIndexes;
118         }
119 
120         // dataTables needs to know the number of filtered elements for rendering purposes
121         List<IndexedElement> filteredIndexedElements = buildFilteredIndexedCollection(showIndexes, modelCollection);
122         collectionGroup.setFilteredCollectionSize(filteredIndexedElements.size());
123 
124         buildLinesForDisplayedRows(filteredIndexedElements, view, model, collectionGroup);
125     }
126 
127     /**
128      * Build a filtered and indexed version of the model collection based on showIndexes.
129      *
130      * <p>The items in the returned collection contain
131      * <ul>
132      * <li>an <b>index</b> property which refers to the original position within the unfiltered model collection</li>
133      * <li>an <b>element</b> property which is a reference to the element in the model collection</li>
134      * </ul>
135      * </p>
136      *
137      * @param showIndexes A List of indexes to model collection elements that were not filtered out
138      * @param modelCollection the model collection
139      * @return a filtered and indexed version of the model collection
140      * @see IndexedElement
141      */
142     private List<IndexedElement> buildFilteredIndexedCollection(List<Integer> showIndexes,
143             List<Object> modelCollection) {
144         // apply the filtering in a way that preserves the original indices for binding path use
145         List<IndexedElement> filteredIndexedElements = new ArrayList<IndexedElement>(modelCollection.size());
146 
147         for (Integer showIndex : showIndexes) {
148             filteredIndexedElements.add(new IndexedElement(showIndex, modelCollection.get(showIndex)));
149         }
150 
151         return filteredIndexedElements;
152     }
153 
154     /**
155      * Build the lines for the collection rows to be rendered.
156      *
157      * @param filteredIndexedElements a filtered and indexed list of the model collection elements
158      * @param view View instance the collection belongs to
159      * @param model Top level object containing the data
160      * @param collectionGroup CollectionGroup component for the collection
161      */
162     protected void buildLinesForDisplayedRows(List<IndexedElement> filteredIndexedElements, View view, Object model,
163             CollectionGroup collectionGroup) {
164 
165         // if we are doing server paging, but the display length wasn't set (which will be the case on the page render)
166         // then only render one line.  Needed to force the table to show up in the page.
167         if (collectionGroup.isUseServerPaging() && collectionGroup.getDisplayLength() == -1) {
168             collectionGroup.setDisplayLength(1);
169         }
170 
171         int displayStart = (collectionGroup.getDisplayStart() != -1 && collectionGroup.isUseServerPaging()) ?
172                 collectionGroup.getDisplayStart() : 0;
173 
174         int displayLength = (collectionGroup.getDisplayLength() != -1 && collectionGroup.isUseServerPaging()) ?
175                 collectionGroup.getDisplayLength() : filteredIndexedElements.size() - displayStart;
176 
177         // make sure we don't exceed the size of our collection
178         int displayEndExclusive =
179                 (displayStart + displayLength > filteredIndexedElements.size()) ? filteredIndexedElements.size() :
180                         displayStart + displayLength;
181 
182         // get a view of the elements that will be displayed on the page (if paging is enabled)
183         List<IndexedElement> renderedIndexedElements = filteredIndexedElements.subList(displayStart,
184                 displayEndExclusive);
185 
186         // for each unfiltered collection row to be rendered, build the line fields
187         for (IndexedElement indexedElement : renderedIndexedElements) {
188             Object currentLine = indexedElement.element;
189 
190             String bindingPathPrefix =
191                     collectionGroup.getBindingInfo().getBindingPrefixForNested() + "[" + indexedElement.index + "]";
192 
193             List<? extends Component> lineActions = initializeLineActions(collectionGroup.getLineActions(), view, model,
194                     collectionGroup, currentLine, indexedElement.index);
195 
196             LineBuilderContext lineBuilderContext = new LineBuilderContext(indexedElement.index, currentLine,
197                     bindingPathPrefix, false, (ViewModel) model, collectionGroup, lineActions);
198 
199             getCollectionGroupLineBuilder(lineBuilderContext).buildLine();
200         }
201     }
202 
203     /**
204      * Performs any filtering necessary on the collection before building the collection fields.
205      *
206      * <p>If showInactive is set to false and the collection line type implements {@code Inactivatable},
207      * invokes the active collection filter. Then any {@link CollectionFilter} instances configured for the collection
208      * group are invoked to filter the collection. Collections lines must pass all filters in order to be
209      * displayed</p>
210      *
211      * @param view view instance that contains the collection
212      * @param model object containing the views data
213      * @param collectionGroup collection group component instance that will display the collection
214      * @param collection collection instance that will be filtered
215      */
216     protected List<Integer> performCollectionFiltering(View view, Object model, CollectionGroup collectionGroup,
217             Collection<?> collection) {
218         List<Integer> filteredIndexes = new ArrayList<Integer>();
219         for (int i = 0; i < collection.size(); i++) {
220             filteredIndexes.add(Integer.valueOf(i));
221         }
222 
223         if (Inactivatable.class.isAssignableFrom(collectionGroup.getCollectionObjectClass()) && !collectionGroup
224                 .isShowInactiveLines()) {
225             List<Integer> activeIndexes = collectionGroup.getActiveCollectionFilter().filter(view, model,
226                     collectionGroup);
227             filteredIndexes = ListUtils.intersection(filteredIndexes, activeIndexes);
228         }
229 
230         for (CollectionFilter collectionFilter : collectionGroup.getFilters()) {
231             List<Integer> indexes = collectionFilter.filter(view, model, collectionGroup);
232             filteredIndexes = ListUtils.intersection(filteredIndexes, indexes);
233             if (filteredIndexes.isEmpty()) {
234                 break;
235             }
236         }
237 
238         return filteredIndexes;
239     }
240 
241     /**
242      * Builds the fields for holding the collection add line and if necessary makes call to setup
243      * the new line instance.
244      *
245      * @param view view instance the collection belongs to
246      * @param collectionGroup collection group the layout manager applies to
247      * @param model Object containing the view data, should extend UifFormBase
248      * if using framework managed new lines
249      */
250     protected void buildAddLine(View view, Object model, CollectionGroup collectionGroup) {
251         // initialize new line if one does not already exist
252         initializeNewCollectionLine(view, model, collectionGroup, false);
253 
254         String addLineBindingPath = collectionGroup.getAddLineBindingInfo().getBindingPath();
255         List<? extends Component> actionComponents = getAddLineActionComponents(view, model, collectionGroup);
256 
257         Object addLine = ObjectPropertyUtils.getPropertyValue(model, addLineBindingPath);
258 
259         boolean bindToForm = false;
260         if (StringUtils.isBlank(collectionGroup.getAddLinePropertyName())) {
261             bindToForm = true;
262         }
263 
264         LineBuilderContext lineBuilderContext = new LineBuilderContext(-1, addLine, addLineBindingPath, bindToForm,
265                 (ViewModel) model, collectionGroup, actionComponents);
266 
267         getCollectionGroupLineBuilder(lineBuilderContext).buildLine();
268     }
269 
270     /**
271      * Creates new {@code Action} instances for the line.
272      *
273      * <p>Adds context to the action fields for the given line so that the line the action was performed on can be
274      * determined when that action is selected</p>
275      *
276      * @param lineActions the actions to copy
277      * @param view view instance the collection belongs to
278      * @param model top level object containing the data
279      * @param collectionGroup collection group component for the collection
280      * @param collectionLine object instance for the current line
281      * @param lineIndex index of the line the actions should apply to
282      */
283     protected List<? extends Component> initializeLineActions(List<? extends Component> lineActions, View view,
284             Object model, CollectionGroup collectionGroup, Object collectionLine, int lineIndex) {
285         List<? extends Component> actionComponents = ComponentUtils.copy(lineActions);
286 
287         for (Component actionComponent : actionComponents) {
288             view.getViewHelperService().setElementContext(actionComponent, collectionGroup);
289         }
290 
291         String lineSuffix = UifConstants.IdSuffixes.LINE + Integer.toString(lineIndex);
292         ContextUtils.updateContextsForLine(actionComponents, collectionGroup, collectionLine, lineIndex, lineSuffix);
293 
294         ExpressionEvaluator expressionEvaluator = ViewLifecycle.getExpressionEvaluator();
295         for (Component actionComponent : actionComponents) {
296             expressionEvaluator.evaluatePropertyExpression(view, actionComponent.getContext(), actionComponent,
297                     UifPropertyPaths.ID, true);
298         }
299 
300         ComponentUtils.updateIdsWithSuffixNested(actionComponents, lineSuffix);
301 
302         List<Action> actions = ViewLifecycleUtils.getElementsOfTypeDeep(actionComponents, Action.class);
303         initializeActions(actions, collectionGroup, lineIndex);
304 
305         return actionComponents;
306     }
307 
308     /**
309      * Updates the action parameters, jump to, refresh id, and validation configuration for the list of actions
310      * associated with the given collection group and line index.
311      *
312      * @param actions list of action components to update
313      * @param collectionGroup collection group instance the actions belong to
314      * @param lineIndex index of the line the actions are associate with
315      */
316     public void initializeActions(List<Action> actions, CollectionGroup collectionGroup, int lineIndex) {
317         for (Action action : actions) {
318             if (ComponentUtils.containsPropertyExpression(action, UifPropertyPaths.ACTION_PARAMETERS, true)) {
319                 // need to update the actions expressions so our settings do not get overridden
320                 action.getPropertyExpressions().put(
321                         UifPropertyPaths.ACTION_PARAMETERS + "['" + UifParameters.SELECTED_COLLECTION_PATH + "']",
322                         UifConstants.EL_PLACEHOLDER_PREFIX + "'" + collectionGroup.getBindingInfo().getBindingPath() +
323                                 "'" + UifConstants.EL_PLACEHOLDER_SUFFIX);
324                 action.getPropertyExpressions().put(
325                         UifPropertyPaths.ACTION_PARAMETERS + "['" + UifParameters.SELECTED_COLLECTION_ID + "']",
326                         UifConstants.EL_PLACEHOLDER_PREFIX + "'" + collectionGroup.getId() +
327                                 "'" + UifConstants.EL_PLACEHOLDER_SUFFIX);
328                 action.getPropertyExpressions().put(
329                         UifPropertyPaths.ACTION_PARAMETERS + "['" + UifParameters.SELECTED_LINE_INDEX + "']",
330                         UifConstants.EL_PLACEHOLDER_PREFIX + "'" + Integer.toString(lineIndex) +
331                                 "'" + UifConstants.EL_PLACEHOLDER_SUFFIX);
332             } else {
333                 action.addActionParameter(UifParameters.SELECTED_COLLECTION_PATH,
334                         collectionGroup.getBindingInfo().getBindingPath());
335                 action.addActionParameter(UifParameters.SELECTED_COLLECTION_ID,
336                                         collectionGroup.getId());
337                 action.addActionParameter(UifParameters.SELECTED_LINE_INDEX, Integer.toString(lineIndex));
338             }
339 
340             if (StringUtils.isBlank(action.getRefreshId()) && StringUtils.isBlank(action.getRefreshPropertyName())) {
341                 action.setRefreshId(collectionGroup.getId());
342             }
343 
344             // if marked for validation, add call to validate the line and set validation flag to false
345             // so the entire form will not be validated
346             if (action.isPerformClientSideValidation()) {
347                 String preSubmitScript = "var valid=" + UifConstants.JsFunctions.VALIDATE_LINE + "('" +
348                         collectionGroup.getBindingInfo().getBindingPath() + "'," + Integer.toString(lineIndex) +
349                         ");";
350 
351                 // prepend custom presubmit script which should evaluate to a boolean
352                 if (StringUtils.isNotBlank(action.getPreSubmitCall())) {
353                     preSubmitScript = ScriptUtils.appendScript(preSubmitScript,
354                             "if(valid){valid=function(){" + action.getPreSubmitCall() + "}();}");
355                 }
356 
357                 preSubmitScript += " return valid;";
358 
359                 action.setPreSubmitCall(preSubmitScript);
360                 action.setPerformClientSideValidation(false);
361             }
362         }
363     }
364 
365     /**
366      * Creates new {@code Component} instances for the add line
367      *
368      * <p>
369      * Adds context to the action fields for the add line so that the collection
370      * the action was performed on can be determined
371      * </p>
372      *
373      * @param view view instance the collection belongs to
374      * @param model top level object containing the data
375      * @param collectionGroup collection group component for the collection
376      */
377     protected List<? extends Component> getAddLineActionComponents(View view, Object model,
378             CollectionGroup collectionGroup) {
379         String lineSuffix = UifConstants.IdSuffixes.ADD_LINE;
380 
381 
382         List<? extends Component> lineActionComponents = ComponentUtils.copyComponentList(
383                 collectionGroup.getAddLineActions(), lineSuffix);
384 
385         List<Action> actions = ViewLifecycleUtils.getElementsOfTypeDeep(lineActionComponents, Action.class);
386 
387         if (collectionGroup.isAddWithDialog() && (collectionGroup.getAddLineDialog().getFooter() != null) &&
388                 !collectionGroup.getAddLineDialog().getFooter().getItems().isEmpty()) {
389             List<Action> addLineDialogActions = ViewLifecycleUtils.getElementsOfTypeDeep(
390                     collectionGroup.getAddLineDialog().getFooter().getItems(), Action.class);
391 
392             if (addLineDialogActions != null) {
393                 actions.addAll(addLineDialogActions);
394             }
395         }
396 
397         for (Action action : actions) {
398             action.addActionParameter(UifParameters.SELECTED_COLLECTION_PATH,
399                     collectionGroup.getBindingInfo().getBindingPath());
400             action.addActionParameter(UifParameters.SELECTED_COLLECTION_ID,
401                                 collectionGroup.getId());
402             action.setJumpToIdAfterSubmit(collectionGroup.getId());
403             action.addActionParameter(UifParameters.ACTION_TYPE, UifParameters.ADD_LINE);
404             action.setRefreshId(collectionGroup.getId());
405 
406             if (collectionGroup.isAddWithDialog() && view instanceof FormView && ((FormView) view).isValidateClientSide()) {
407                 action.setPerformClientSideValidation(true);
408             }
409 
410             if (action.isPerformClientSideValidation()) {
411                 String preSubmitScript = "var valid=" + UifConstants.JsFunctions.VALIDATE_ADD_LINE + "('" +
412                         collectionGroup.getId() + "');";
413 
414                 // prepend custom presubmit script which should evaluate to a boolean
415                 if (StringUtils.isNotBlank(action.getPreSubmitCall())) {
416                     preSubmitScript = ScriptUtils.appendScript(preSubmitScript,
417                             "if(valid){valid=function(){" + action.getPreSubmitCall() + "}();}");
418                 }
419 
420                 preSubmitScript += "return valid;";
421 
422                 action.setPreSubmitCall(preSubmitScript);
423                 action.setPerformClientSideValidation(false);
424             } else if (collectionGroup.isAddWithDialog()) {
425                 action.setPreSubmitCall("closeLightbox(); return true;");
426             }
427         }
428 
429         // get add line for context
430         String addLinePath = collectionGroup.getAddLineBindingInfo().getBindingPath();
431         Object addLine = ObjectPropertyUtils.getPropertyValue(model, addLinePath);
432 
433         ContextUtils.updateContextsForLine(lineActionComponents, collectionGroup, addLine, -1, lineSuffix);
434 
435         return lineActionComponents;
436     }
437 
438     /**
439      * Initializes a new instance of the collection data object class for the add line.
440      *
441      * <p>If the add line property was not specified for the collection group the new lines will be
442      * added to the generic map on the {@code UifFormBase}, else it will be added to the property given by
443      * the addLineBindingInfo</p>
444      *
445      * <p>New line will only be created if the current line property is null or clearExistingLine is true.
446      * In the case of a new line default values are also applied</p>
447      */
448     public void initializeNewCollectionLine(View view, Object model, CollectionGroup collectionGroup,
449             boolean clearExistingLine) {
450         Object newLine = null;
451 
452         // determine if we are binding to generic form map or a custom property
453         if (StringUtils.isBlank(collectionGroup.getAddLinePropertyName())) {
454             // bind to form map
455             if (!(model instanceof UifFormBase)) {
456                 throw new RuntimeException("Cannot create new collection line for group: "
457                         + collectionGroup.getPropertyName()
458                         + ". Model does not extend "
459                         + UifFormBase.class.getName());
460             }
461 
462             // get new collection line map from form
463             Map<String, Object> newCollectionLines = ObjectPropertyUtils.getPropertyValue(model,
464                     UifPropertyPaths.NEW_COLLECTION_LINES);
465             if (newCollectionLines == null) {
466                 newCollectionLines = new HashMap<String, Object>();
467                 ObjectPropertyUtils.setPropertyValue(model, UifPropertyPaths.NEW_COLLECTION_LINES, newCollectionLines);
468             }
469 
470             // set binding path for add line
471             String newCollectionLineKey = KRADUtils.translateToMapSafeKey(
472                     collectionGroup.getBindingInfo().getBindingPath());
473             String addLineBindingPath = UifPropertyPaths.NEW_COLLECTION_LINES + "['" + newCollectionLineKey + "']";
474             collectionGroup.getAddLineBindingInfo().setBindingPath(addLineBindingPath);
475 
476             // if there is not an instance available or we need to clear create a new instance
477             if (!newCollectionLines.containsKey(newCollectionLineKey) || (newCollectionLines.get(newCollectionLineKey)
478                     == null) || clearExistingLine) {
479                 // create new instance of the collection type for the add line
480                 newLine = KRADUtils.createNewObjectFromClass(collectionGroup.getCollectionObjectClass());
481                 newCollectionLines.put(newCollectionLineKey, newLine);
482             }
483         } else {
484             // bind to custom property
485             Object addLine = ObjectPropertyUtils.getPropertyValue(model,
486                     collectionGroup.getAddLineBindingInfo().getBindingPath());
487             if ((addLine == null) || clearExistingLine) {
488                 newLine = KRADUtils.createNewObjectFromClass(collectionGroup.getCollectionObjectClass());
489                 ObjectPropertyUtils.setPropertyValue(model, collectionGroup.getAddLineBindingInfo().getBindingPath(),
490                         newLine);
491             }
492         }
493 
494         // apply default values if a new line was created
495         if (newLine != null) {
496             ViewLifecycle.getHelper().applyDefaultValuesForCollectionLine(collectionGroup, newLine);
497         }
498     }
499 
500     /**
501      * Returns an instance of {@link CollectionGroupLineBuilder} for building the line.
502      *
503      * @param lineBuilderContext context of line for initializing line builder
504      * @return CollectionGroupLineBuilder instance
505      */
506     public CollectionGroupLineBuilder getCollectionGroupLineBuilder(LineBuilderContext lineBuilderContext) {
507         return new CollectionGroupLineBuilder(lineBuilderContext);
508     }
509 
510     /**
511      * Wrapper object to enable filtering of a collection while preserving original indices
512      */
513     private static class IndexedElement {
514 
515         /**
516          * The index associated with the given element
517          */
518         final int index;
519 
520         /**
521          * The element itself
522          */
523         final Object element;
524 
525         /**
526          * Constructs an {@link org.kuali.rice.krad.uif.container.CollectionGroupBuilder.IndexedElement}
527          *
528          * @param index the index to associate with the element
529          * @param element the element itself
530          */
531         private IndexedElement(int index, Object element) {
532             this.index = index;
533             this.element = element;
534         }
535     }
536 
537 }