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.modifier;
17  
18  import java.util.ArrayList;
19  import java.util.HashMap;
20  import java.util.HashSet;
21  import java.util.List;
22  import java.util.Map;
23  import java.util.Set;
24  
25  import org.apache.commons.lang.StringUtils;
26  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
27  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
28  import org.kuali.rice.krad.datadictionary.parse.BeanTags;
29  import org.kuali.rice.krad.uif.UifConstants;
30  import org.kuali.rice.krad.uif.UifPropertyPaths;
31  import org.kuali.rice.krad.uif.component.Component;
32  import org.kuali.rice.krad.uif.container.Group;
33  import org.kuali.rice.krad.uif.element.Header;
34  import org.kuali.rice.krad.uif.field.DataField;
35  import org.kuali.rice.krad.uif.field.Field;
36  import org.kuali.rice.krad.uif.field.SpaceField;
37  import org.kuali.rice.krad.uif.layout.GridLayoutManager;
38  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
39  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycleUtils;
40  import org.kuali.rice.krad.uif.util.ComponentFactory;
41  import org.kuali.rice.krad.uif.util.ComponentUtils;
42  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
43  import org.kuali.rice.krad.uif.view.ExpressionEvaluator;
44  import org.kuali.rice.krad.uif.view.View;
45  
46  /**
47   * Generates <code>Field</code> instances to produce a comparison view among
48   * objects of the same type
49   *
50   * <p>
51   * Modifier is initialized with a List of <code>ComparableInfo</code> instances.
52   * For each comparable info, a copy of the configured group field is made and
53   * adjusted to the binding object path for the comparable. The comparison fields
54   * are ordered based on the configured order property of the comparable. In
55   * addition, a <code>HeaderField<code> can be generated to label each group
56   * of comparison fields.
57   * </p>
58   *
59   * @author Kuali Rice Team (rice.collab@kuali.org)
60   */
61  @BeanTags({@BeanTag(name = "compareFieldCreate-modifier-bean", parent = "Uif-CompareFieldCreate-Modifier"),
62          @BeanTag(name = "maintenanceCompare-modifier-bean", parent = "Uif-MaintenanceCompare-Modifier")})
63  public class CompareFieldCreateModifier extends ComponentModifierBase {
64  
65      private static final long serialVersionUID = -6285531580512330188L;
66  
67      private int defaultOrderSequence;
68      private boolean generateCompareHeaders;
69  
70      private Header headerFieldPrototype;
71      private List<ComparableInfo> comparables;
72  
73      public CompareFieldCreateModifier() {
74          defaultOrderSequence = 1;
75          generateCompareHeaders = true;
76  
77          comparables = new ArrayList<ComparableInfo>();
78      }
79  
80      /**
81       * Calls <code>ViewHelperService</code> to initialize the header field prototype
82       *
83       * {@inheritDoc}
84       */
85      @Override
86      public void performInitialization(Object model, Component component) {
87          super.performInitialization(model, component);
88      }
89  
90      /**
91       * Generates the comparison fields
92       *
93       * <p>
94       * First the configured List of <code>ComparableInfo</code> instances are
95       * sorted based on their order property. Then if generateCompareHeaders is
96       * set to true, a <code>HeaderField</code> is created for each comparable
97       * using the headerFieldPrototype and the headerText given by the
98       * comparable. Finally for each field configured on the <code>Group</code>,
99       * a corresponding comparison field is generated for each comparable and
100      * adjusted to the binding object path given by the comparable in addition
101      * to suffixing the id and setting the readOnly property
102      * </p>
103      *
104      * {@inheritDoc}
105      */
106     @Override
107     public void performModification(Object model, Component component) {
108         if ((component != null) && !(component instanceof Group)) {
109             throw new IllegalArgumentException(
110                     "Compare field initializer only support Group components, found type: " + component.getClass());
111         }
112 
113         if (component == null) {
114             return;
115         }
116         
117         Group group = (Group) component;
118 
119         // list to hold the generated compare items
120         List<Component> comparisonItems = new ArrayList<Component>();
121 
122         // sort comparables by their order property
123         List<ComparableInfo> groupComparables = (List<ComparableInfo>) ComponentUtils.sort(comparables,
124                 defaultOrderSequence);
125 
126         // evaluate expressions on comparables
127         Map<String, Object> context = new HashMap<String, Object>();
128         
129         View view = ViewLifecycle.getView();
130         
131         Map<String, Object> viewContext = view.getContext();
132         if (viewContext != null) {
133             context.putAll(view.getContext());
134         }
135         
136         context.put(UifConstants.ContextVariableNames.COMPONENT, component);
137 
138         ExpressionEvaluator expressionEvaluator = ViewLifecycle.getExpressionEvaluator();
139 
140         for (ComparableInfo comparable : groupComparables) {
141             expressionEvaluator.evaluateExpressionsOnConfigurable(view, comparable, context);
142         }
143 
144         // generate compare header
145         if (isGenerateCompareHeaders()) {
146             // add space field for label column
147             SpaceField spaceField = ComponentFactory.getSpaceField();
148             comparisonItems.add(spaceField);
149 
150             for (ComparableInfo comparable : groupComparables) {
151                 Header compareHeaderField = ComponentUtils.copy(headerFieldPrototype, comparable.getIdSuffix());
152                 compareHeaderField.setHeaderText(comparable.getHeaderText());
153                 comparisonItems.add(compareHeaderField);
154             }
155             
156             // if group is using grid layout, make first row a header
157             if (group.getLayoutManager() instanceof GridLayoutManager) {
158                 ((GridLayoutManager) group.getLayoutManager()).setRenderFirstRowHeader(true);
159             }
160         }
161 
162         // find the comparable to use for comparing value changes (if
163         // configured)
164         boolean performValueChangeComparison = false;
165         String compareValueObjectBindingPath = null;
166         for (ComparableInfo comparable : groupComparables) {
167             if (comparable.isCompareToForValueChange()) {
168                 performValueChangeComparison = true;
169                 compareValueObjectBindingPath = comparable.getBindingObjectPath();
170             }
171         }
172 
173         // generate the compare items from the configured group
174         boolean changeIconShowedOnHeader = false;
175         for (Component item : group.getItems()) {
176             int defaultSuffix = 0;
177             boolean suppressLabel = false;
178 
179             for (ComparableInfo comparable : groupComparables) {
180                 String idSuffix = comparable.getIdSuffix();
181                 if (StringUtils.isBlank(idSuffix)) {
182                     idSuffix = UifConstants.IdSuffixes.COMPARE + defaultSuffix;
183                 }
184 
185                 Component compareItem = ComponentUtils.copy(item, idSuffix);
186 
187                 ComponentUtils.setComponentPropertyDeep(compareItem, UifPropertyPaths.BIND_OBJECT_PATH,
188                         comparable.getBindingObjectPath());
189                 if (comparable.isReadOnly()) {
190                     compareItem.setReadOnly(true);
191                     if (compareItem.getPropertyExpressions().containsKey("readOnly")) {
192                         compareItem.getPropertyExpressions().remove("readOnly");
193                     }
194                 }
195 
196                 // label will be enabled for first comparable only
197                 if (suppressLabel && (compareItem instanceof Field)) {
198                     ((Field) compareItem).getFieldLabel().setRender(false);
199                 }
200 
201                 // do value comparison
202                 if (performValueChangeComparison && comparable.isHighlightValueChange() && !comparable
203                         .isCompareToForValueChange()) {
204                     boolean valueChanged = performValueComparison(group, compareItem, model,
205                             compareValueObjectBindingPath);
206 
207                     // add icon to group header if not done so yet
208                     if (valueChanged && !changeIconShowedOnHeader && isGenerateCompareHeaders()) {
209                         Group groupToSetHeader = null;
210                         if (group.getDisclosure() != null && group.getDisclosure().isRender()) {
211                             groupToSetHeader = group;
212                         } else if (group.getContext().get(UifConstants.ContextVariableNames.PARENT) != null) {
213                             // use the parent group to set the notification if available
214                             groupToSetHeader = (Group) group.getContext().get(UifConstants.ContextVariableNames.PARENT);
215                         }
216 
217                         if (groupToSetHeader.getDisclosure().isRender()) {
218                             groupToSetHeader.getDisclosure().setOnDocumentReadyScript(
219                                     "showChangeIconOnDisclosure('" + groupToSetHeader.getId() + "');");
220                         } else if (groupToSetHeader.getHeader() != null) {
221                             groupToSetHeader.getHeader().setOnDocumentReadyScript(
222                                     "showChangeIconOnHeader('" + groupToSetHeader.getHeader().getId() + "');");
223                         }
224 
225                         changeIconShowedOnHeader = true;
226                     }
227                 }
228 
229                 comparisonItems.add(compareItem);
230                 defaultSuffix++;
231 
232                 suppressLabel = true;
233             }
234         }
235         
236         // update the group's list of components
237         group.setItems(comparisonItems);
238     }
239 
240     /**
241      * For each attribute field in the compare item, retrieves the field value and compares against the value for the
242      * main comparable. If the value is different, adds script to the field on ready event to add the change icon to
243      * the field and the containing group header
244      *
245      * @param group group that contains the item and whose header will be highlighted for changes
246      * @param compareItem the compare item being generated and to pull attribute fields from
247      * @param model object containing the data
248      * @param compareValueObjectBindingPath object path for the comparison item
249      * @return true if the value in the field represented by compareItem is equal to the comparison items value, false
250      *         otherwise
251      */
252     protected boolean performValueComparison(Group group, Component compareItem, Object model,
253             String compareValueObjectBindingPath) {
254         // get any attribute fields for the item so we can compare the values
255         List<DataField> itemFields = ViewLifecycleUtils.getElementsOfTypeDeep(compareItem, DataField.class);
256         boolean valueChanged = false;
257         for (DataField field : itemFields) {
258             String fieldBindingPath = field.getBindingInfo().getBindingPath();
259             Object fieldValue = ObjectPropertyUtils.getPropertyValue(model, fieldBindingPath);
260 
261             String compareBindingPath = StringUtils.replaceOnce(fieldBindingPath,
262                     field.getBindingInfo().getBindingObjectPath(), compareValueObjectBindingPath);
263             Object compareValue = ObjectPropertyUtils.getPropertyValue(model, compareBindingPath);
264 
265             if (!((fieldValue == null) && (compareValue == null))) {
266                 // if one is null then value changed
267                 if ((fieldValue == null) || (compareValue == null)) {
268                     valueChanged = true;
269                 } else {
270                     // both not null, compare values
271                     valueChanged = !fieldValue.equals(compareValue);
272                 }
273             }
274             if (valueChanged) {
275                 // add script to show change icon
276                 String onReadyScript = "showChangeIcon('" + field.getId() + "');";
277                 field.setRenderMarkerIconSpan(true);
278                 field.setOnDocumentReadyScript(onReadyScript);
279             }
280             // TODO: add script for value changed?
281         }
282         return valueChanged;
283     }
284 
285     /**
286      * Generates an id suffix for the comparable item
287      *
288      * <p>
289      * If the idSuffix to use if configured on the <code>ComparableInfo</code>
290      * it will be used, else the given integer index will be used with an
291      * underscore
292      * </p>
293      *
294      * @param comparable comparable info to check for id suffix
295      * @param index sequence integer
296      * @return id suffix
297      * @see org.kuali.rice.krad.uif.modifier.ComparableInfo#getIdSuffix()
298      */
299     protected String getIdSuffix(ComparableInfo comparable, int index) {
300         String idSuffix = comparable.getIdSuffix();
301         if (StringUtils.isBlank(idSuffix)) {
302             idSuffix = "_" + index;
303         }
304 
305         return idSuffix;
306     }
307 
308     /**
309      * {@inheritDoc}
310      */
311     @Override
312     public Set<Class<? extends Component>> getSupportedComponents() {
313         Set<Class<? extends Component>> components = new HashSet<Class<? extends Component>>();
314         components.add(Group.class);
315 
316         return components;
317     }
318 
319     /**
320      * @see org.kuali.rice.krad.uif.modifier.ComponentModifierBase#getComponentPrototypes()
321      */
322     public List<Component> getComponentPrototypes() {
323         List<Component> components = new ArrayList<Component>();
324 
325         components.add(headerFieldPrototype);
326 
327         return components;
328     }
329 
330     /**
331      * Indicates the starting integer sequence value to use for
332      * <code>ComparableInfo</code> instances that do not have the order property
333      * set
334      *
335      * @return default sequence starting value
336      */
337     @BeanTagAttribute(name = "defaultOrderSequence")
338     public int getDefaultOrderSequence() {
339         return this.defaultOrderSequence;
340     }
341 
342     /**
343      * Setter for the default sequence starting value
344      *
345      * @param defaultOrderSequence
346      */
347     public void setDefaultOrderSequence(int defaultOrderSequence) {
348         this.defaultOrderSequence = defaultOrderSequence;
349     }
350 
351     /**
352      * Indicates whether a <code>HeaderField</code> should be created for each
353      * group of comparison fields
354      *
355      * <p>
356      * If set to true, for each group of comparison fields a header field will
357      * be created using the headerFieldPrototype configured on the modifier with
358      * the headerText property of the comparable
359      * </p>
360      *
361      * @return true if the headers should be created, false if no
362      *         headers should be created
363      */
364     @BeanTagAttribute(name = "generateCompareHeaders")
365     public boolean isGenerateCompareHeaders() {
366         return this.generateCompareHeaders;
367     }
368 
369     /**
370      * Setter for the generate comparison headers indicator
371      *
372      * @param generateCompareHeaders
373      */
374     public void setGenerateCompareHeaders(boolean generateCompareHeaders) {
375         this.generateCompareHeaders = generateCompareHeaders;
376     }
377 
378     /**
379      * Prototype instance to use for creating the <code>HeaderField</code> for
380      * each group of comparison fields (if generateCompareHeaders is true)
381      *
382      * @return header field prototype
383      */
384     @BeanTagAttribute(name = "headerFieldPrototype", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
385     public Header getHeaderFieldPrototype() {
386         return this.headerFieldPrototype;
387     }
388 
389     /**
390      * Setter for the header field prototype
391      *
392      * @param headerFieldPrototype
393      */
394     public void setHeaderFieldPrototype(Header headerFieldPrototype) {
395         this.headerFieldPrototype = headerFieldPrototype;
396     }
397 
398     /**
399      * List of <code>ComparableInfo</code> instances the compare fields should
400      * be generated for
401      *
402      * <p>
403      * For each comparable, a copy of the fields configured for the
404      * <code>Group</code> will be created for the comparison view
405      * </p>
406      *
407      * @return comparables to generate fields for
408      */
409     @BeanTagAttribute(name = "comparables", type = BeanTagAttribute.AttributeType.LISTBEAN)
410     public List<ComparableInfo> getComparables() {
411         return this.comparables;
412     }
413 
414     /**
415      * Setter for the list of comparable info instances
416      *
417      * @param comparables
418      */
419     public void setComparables(List<ComparableInfo> comparables) {
420         this.comparables = comparables;
421     }
422 
423 }