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.view;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.kuali.rice.krad.uif.container.CollectionGroup;
20  import org.kuali.rice.krad.uif.component.Component;
21  import org.kuali.rice.krad.uif.field.DataField;
22  import org.kuali.rice.krad.uif.field.InputField;
23  import org.kuali.rice.krad.uif.layout.TableLayoutManager;
24  import org.kuali.rice.krad.uif.util.ComponentUtils;
25  import org.kuali.rice.krad.uif.util.ViewCleaner;
26  
27  import java.beans.PropertyEditor;
28  import java.io.Serializable;
29  import java.util.HashMap;
30  import java.util.HashSet;
31  import java.util.Map;
32  import java.util.Set;
33  
34  /**
35   * Holds field indexes of a <code>View</code> instance for retrieval
36   *
37   * @author Kuali Rice Team (rice.collab@kuali.org)
38   */
39  public class ViewIndex implements Serializable {
40      private static final long serialVersionUID = 4700818801272201371L;
41  
42      protected Map<String, Component> index;
43      protected Map<String, DataField> dataFieldIndex;
44      protected Map<String, CollectionGroup> collectionsIndex;
45  
46      protected Map<String, Component> initialComponentStates;
47  
48      protected Map<String, PropertyEditor> fieldPropertyEditors;
49      protected Map<String, PropertyEditor> secureFieldPropertyEditors;
50      protected Map<String, Integer> idSequenceSnapshot;
51      protected Map<String, Map<String, String>> componentExpressionGraphs;
52  
53      /**
54       * Constructs new instance
55       */
56      public ViewIndex() {
57          index = new HashMap<String, Component>();
58          dataFieldIndex = new HashMap<String, DataField>();
59          collectionsIndex = new HashMap<String, CollectionGroup>();
60          initialComponentStates = new HashMap<String, Component>();
61          fieldPropertyEditors = new HashMap<String, PropertyEditor>();
62          secureFieldPropertyEditors = new HashMap<String, PropertyEditor>();
63          idSequenceSnapshot = new HashMap<String, Integer>();
64          componentExpressionGraphs = new HashMap<String, Map<String, String>>();
65      }
66  
67      /**
68       * Walks through the View tree and indexes all components found. All components
69       * are indexed by their IDs with the special indexing done for certain components
70       *
71       * <p>
72       * <code>DataField</code> instances are indexed by the attribute path.
73       * This is useful for retrieving the InputField based on the incoming
74       * request parameter
75       * </p>
76       *
77       * <p>
78       * <code>CollectionGroup</code> instances are indexed by the collection
79       * path. This is useful for retrieving the CollectionGroup based on the
80       * incoming request parameter
81       * </p>
82       */
83      protected void index(View view) {
84          index = new HashMap<String, Component>();
85          dataFieldIndex = new HashMap<String, DataField>();
86          collectionsIndex = new HashMap<String, CollectionGroup>();
87          fieldPropertyEditors = new HashMap<String, PropertyEditor>();
88          secureFieldPropertyEditors = new HashMap<String, PropertyEditor>();
89  
90          indexComponent(view);
91      }
92  
93      /**
94       * Adds an entry to the main index for the given component. If the component
95       * is of type <code>DataField</code> or <code>CollectionGroup</code> an
96       * entry is created in the corresponding indexes for those types as well. Then
97       * the #indexComponent method is called for each of the component's children
98       *
99       * <p>
100      * If the component is already contained in the indexes, it will be replaced
101      * </p>
102      *
103      * <p>
104      * Special processing is done for DataField instances to register their property editor which will
105      * be used for form binding
106      * </p>
107      *
108      * @param component component instance to index
109      */
110     public void indexComponent(Component component) {
111         if (component == null) {
112             return;
113         }
114 
115         index.put(component.getId(), component);
116 
117         if (component instanceof DataField) {
118             DataField field = (DataField) component;
119             dataFieldIndex.put(field.getBindingInfo().getBindingPath(), field);
120 
121             // pull out information we will need to support the form post
122             if (component.isRender()) {
123                 if (field.hasSecureValue()) {
124                     secureFieldPropertyEditors.put(field.getBindingInfo().getBindingPath(), field.getPropertyEditor());
125                 } else {
126                     fieldPropertyEditors.put(field.getBindingInfo().getBindingPath(), field.getPropertyEditor());
127                 }
128             }
129         } else if (component instanceof CollectionGroup) {
130             CollectionGroup collectionGroup = (CollectionGroup) component;
131             collectionsIndex.put(collectionGroup.getBindingInfo().getBindingPath(), collectionGroup);
132         }
133 
134         for (Component nestedComponent : component.getComponentsForLifecycle()) {
135             indexComponent(nestedComponent);
136         }
137     }
138 
139     /**
140      * Invoked after the view lifecycle or component refresh has run to clear indexes that are not
141      * needed for the post
142      */
143     public void clearIndexesAfterRender() {
144         // build list of factory ids for components whose initial state needs to be keep
145         Set<String> holdIds = new HashSet<String>();
146         Set<String> holdFactoryIds = new HashSet<String>();
147         for (Component component : index.values()) {
148             if (component != null) {
149                 // if component has a refresh condition we need to keep it
150                 if (!component.isDisableSessionPersistence() && (StringUtils.isNotBlank(
151                         component.getProgressiveRender())
152                         || StringUtils.isNotBlank(component.getConditionalRefresh())
153                         || component.getRefreshTimer() > 0
154                         || (component.getRefreshWhenChangedPropertyNames() != null && !component
155                         .getRefreshWhenChangedPropertyNames().isEmpty())
156                         || component.isRefreshedByAction()
157                         || component.isDisclosedByAction())) {
158                     holdFactoryIds.add(component.getBaseId());
159                     holdIds.add(component.getId());
160                 }
161                 // if component is marked as persist in session we need to keep it
162                 else if (component.isForceSessionPersistence()) {
163                     holdFactoryIds.add(component.getBaseId());
164                     holdIds.add(component.getId());
165                 }
166                 // hold collection if being used for table data export
167                 else if (component instanceof CollectionGroup && ((CollectionGroup) component)
168                         .getLayoutManager() instanceof TableLayoutManager && usesTableExportOption(
169                         (TableLayoutManager) ((CollectionGroup) component).getLayoutManager())) {
170                     holdFactoryIds.add(component.getBaseId());
171                     holdIds.add(component.getId());
172                 }
173                 // if component is a collection we need to keep it
174                 else if (component instanceof CollectionGroup && !component.isDisableSessionPersistence()) {
175                     ViewCleaner.cleanCollectionGroup((CollectionGroup) component);
176                     holdFactoryIds.add(component.getBaseId());
177                     holdIds.add(component.getId());
178                 }
179                 // if component is input field and has a query we need to keep the final state
180                 else if ((component instanceof InputField) && !component.isDisableSessionPersistence()) {
181                     InputField inputField = (InputField) component;
182                     if ((inputField.getAttributeQuery() != null) || ((inputField.getSuggest() != null) && inputField
183                             .getSuggest().isRender())) {
184                         holdIds.add(component.getId());
185                     }
186                 }
187             }
188         }
189 
190         // remove initial states for components we don't need for post
191         Map<String, Component> holdInitialComponentStates = new HashMap<String, Component>();
192         for (String factoryId : initialComponentStates.keySet()) {
193             if (holdFactoryIds.contains(factoryId)) {
194                 holdInitialComponentStates.put(factoryId, initialComponentStates.get(factoryId));
195             }
196         }
197         initialComponentStates = holdInitialComponentStates;
198 
199         // remove final states for components we don't need for post
200         Map<String, Component> holdComponentStates = new HashMap<String, Component>();
201         for (String id : index.keySet()) {
202             if (holdIds.contains(id)) {
203                 Component component = index.get(id);
204                 holdComponentStates.put(id, component);
205 
206                 // hold expressions for refresh (since they could have been pushed from a parent)
207                 if (!component.getRefreshExpressionGraph().isEmpty()) {
208                     componentExpressionGraphs.put(component.getBaseId(), component.getRefreshExpressionGraph());
209                 }
210             }
211         }
212         index = holdComponentStates;
213 
214         dataFieldIndex = new HashMap<String, DataField>();
215     }
216 
217     /**
218      * checks the export feature is being used
219      *
220      * @param layoutManager
221      * @return
222      */
223     private boolean usesTableExportOption(TableLayoutManager layoutManager) {
224         return layoutManager.getRichTable() != null && (layoutManager.getRichTable().isShowExportOption()
225                 || layoutManager.getRichTable().isShowSearchAndExportOptions());
226     }
227 
228     /**
229      * Retrieves a <code>Component</code> from the view index by Id
230      *
231      * @param id id for the component to retrieve
232      * @return Component instance found in index, or null if no such component exists
233      */
234     public Component getComponentById(String id) {
235         return index.get(id);
236     }
237 
238     /**
239      * Retrieves a <code>DataField</code> instance from the index
240      *
241      * @param propertyPath full path of the data field (from the form)
242      * @return DataField instance for the path or Null if not found
243      */
244     public DataField getDataFieldByPath(String propertyPath) {
245         return dataFieldIndex.get(propertyPath);
246     }
247 
248     /**
249      * Retrieves a <code>DataField</code> instance that has the given property name
250      * specified (note this is not the full binding path and first match is returned)
251      *
252      * @param propertyName property name for field to retrieve
253      * @return DataField instance found or null if not found
254      */
255     public DataField getDataFieldByPropertyName(String propertyName) {
256         DataField dataField = null;
257 
258         for (DataField field : dataFieldIndex.values()) {
259             if (StringUtils.equals(propertyName, field.getPropertyName())) {
260                 dataField = field;
261                 break;
262             }
263         }
264 
265         return dataField;
266     }
267 
268     /**
269      * Gets the Map that contains attribute field indexing information. The Map
270      * key points to an attribute binding path, and the Map value is the
271      * <code>DataField</code> instance
272      *
273      * @return data fields index map
274      */
275     public Map<String, DataField> getDataFieldIndex() {
276         return this.dataFieldIndex;
277     }
278 
279     /**
280      * Gets the Map that contains collection indexing information. The Map key
281      * gives the binding path to the collection, and the Map value givens the
282      * <code>CollectionGroup</code> instance
283      *
284      * @return collection index map
285      */
286     public Map<String, CollectionGroup> getCollectionsIndex() {
287         return this.collectionsIndex;
288     }
289 
290     /**
291      * Retrieves a <code>CollectionGroup</code> instance from the index
292      *
293      * @param collectionPath full path of the collection (from the form)
294      * @return CollectionGroup instance for the collection path or Null if not
295      *         found
296      */
297     public CollectionGroup getCollectionGroupByPath(String collectionPath) {
298         return collectionsIndex.get(collectionPath);
299     }
300 
301     /**
302      * Preserves initial state of components needed for doing component refreshes
303      *
304      * <p>
305      * Some components, such as those that are nested or created in code cannot be requested from the
306      * spring factory to get new instances. For these a copy of the component in its initial state is
307      * set in this map which will be used when doing component refreshes (which requires running just that
308      * component's lifecycle)
309      * </p>
310      *
311      * <p>
312      * Map entries are added during the perform initialize phase from {@link org.kuali.rice.krad.uif.service.ViewHelperService}
313      * </p>
314      *
315      * @return map with key giving the factory id for the component and the value the component instance
316      */
317     public Map<String, Component> getInitialComponentStates() {
318         return initialComponentStates;
319     }
320 
321     /**
322      * Adds a copy of the given component instance to the map of initial component states keyed
323      *
324      * <p>
325      * Component is only added if its factory id is not set yet (which would happen if it had a spring bean id
326      * and we can get the state from Spring). Once added the factory id will be set to the component id
327      * </p>
328      *
329      * @param component component instance to add
330      */
331     public void addInitialComponentStateIfNeeded(Component component) {
332         if (StringUtils.isBlank(component.getBaseId())) {
333             component.setBaseId(component.getId());
334             initialComponentStates.put(component.getBaseId(), ComponentUtils.copy(component));
335         }
336     }
337 
338     /**
339      * Setter for the map holding initial component states
340      *
341      * @param initialComponentStates
342      */
343     public void setInitialComponentStates(Map<String, Component> initialComponentStates) {
344         this.initialComponentStates = initialComponentStates;
345     }
346 
347     /**
348      * Maintains configuration of properties that have been configured for the view (if render was set to
349      * true) and there corresponding PropertyEdtior (if configured)
350      *
351      * <p>
352      * Information is pulled out of the View during the lifecycle so it can be used when a form post is done
353      * from the View. Note if a field is secure, it will be placed in the {@link #getSecureFieldPropertyEditors()} map
354      * instead
355      * </p>
356      *
357      * @return map of property path (full) to PropertyEditor
358      */
359     public Map<String, PropertyEditor> getFieldPropertyEditors() {
360         return fieldPropertyEditors;
361     }
362 
363     /**
364      * Maintains configuration of secure properties that have been configured for the view (if render was set to
365      * true) and there corresponding PropertyEdtior (if configured)
366      *
367      * <p>
368      * Information is pulled out of the View during the lifecycle so it can be used when a form post is done
369      * from the View. Note if a field is non-secure, it will be placed in the {@link #getFieldPropertyEditors()} map
370      * instead
371      * </p>
372      *
373      * @return map of property path (full) to PropertyEditor
374      */
375     public Map<String, PropertyEditor> getSecureFieldPropertyEditors() {
376         return secureFieldPropertyEditors;
377     }
378 
379     /**
380      * Map of components ids to starting id sequences used for the component refresh process
381      *
382      * @return Map<String, Integer> key is component id and value is id sequence value
383      */
384     public Map<String, Integer> getIdSequenceSnapshot() {
385         return idSequenceSnapshot;
386     }
387 
388     /**
389      * Adds a sequence value to the id snapshot map for the given component id
390      *
391      * @param componentId id for the component the id sequence value is associated it
392      * @param sequenceVal current sequence value to insert into the snapshot
393      */
394     public void addSequenceValueToSnapshot(String componentId, int sequenceVal) {
395         if (!idSequenceSnapshot.containsKey(componentId)){
396             idSequenceSnapshot.put(componentId, sequenceVal);
397         }
398     }
399 
400     /**
401      * Map of components with their associated expression graphs that will be used during
402      * the component refresh process
403      *
404      * <p>
405      * Because expressions that impact a component being refreshed might be on a parent component, a special
406      * map needs to be held around that contains expressions that apply to the component and all its nested
407      * components. This map is populated during the initial view processing and populating of the property
408      * expressions from the initial expression graphs
409      * </p>
410      *
411      * @return Map<String, Map<String, String>> key is component id and value is expression graph map
412      * @see org.kuali.rice.krad.uif.util.ExpressionUtils#populatePropertyExpressionsFromGraph(org.kuali.rice.krad.datadictionary.uif.UifDictionaryBean,
413      *      boolean)
414      */
415     public Map<String, Map<String, String>> getComponentExpressionGraphs() {
416         return componentExpressionGraphs;
417     }
418 
419     /**
420      * Returns a clone of the view index.
421      *
422      * @return ViewIndex clone
423      */
424     public ViewIndex copy() {
425         ViewIndex viewIndexCopy = new ViewIndex();
426 
427         if (this.index != null) {
428             Map<String, Component> indexCopy = new HashMap<String, Component>();
429             for (Map.Entry<String, Component> indexEntry : this.index.entrySet()) {
430                 indexCopy.put(indexEntry.getKey(), (Component) indexEntry.getValue().copy());
431             }
432 
433             viewIndexCopy.index = indexCopy;
434         }
435 
436         if (this.dataFieldIndex != null) {
437             Map<String, DataField> dataFieldIndexCopy = new HashMap<String, DataField>();
438             for (Map.Entry<String, DataField> indexEntry : this.dataFieldIndex.entrySet()) {
439                 dataFieldIndexCopy.put(indexEntry.getKey(), (DataField) indexEntry.getValue().copy());
440             }
441 
442             viewIndexCopy.dataFieldIndex = dataFieldIndexCopy;
443         }
444 
445         if (this.collectionsIndex != null) {
446             Map<String, CollectionGroup> collectionsIndexCopy = new HashMap<String, CollectionGroup>();
447             for (Map.Entry<String, CollectionGroup> indexEntry : this.collectionsIndex.entrySet()) {
448                 collectionsIndexCopy.put(indexEntry.getKey(), (CollectionGroup) indexEntry.getValue().copy());
449             }
450 
451             viewIndexCopy.collectionsIndex = collectionsIndexCopy;
452         }
453 
454         if (this.initialComponentStates != null) {
455             Map<String, Component> initialComponentStatesCopy = new HashMap<String, Component>();
456             for (Map.Entry<String, Component> indexEntry : this.initialComponentStates.entrySet()) {
457                 initialComponentStatesCopy.put(indexEntry.getKey(), (Component) indexEntry.getValue().copy());
458             }
459 
460             viewIndexCopy.initialComponentStates = initialComponentStatesCopy;
461         }
462 
463         if (this.fieldPropertyEditors != null) {
464             viewIndexCopy.fieldPropertyEditors = new HashMap<String, PropertyEditor>(this.fieldPropertyEditors);
465         }
466 
467         if (this.secureFieldPropertyEditors != null) {
468             viewIndexCopy.secureFieldPropertyEditors = new HashMap<String, PropertyEditor>(
469                     this.secureFieldPropertyEditors);
470         }
471 
472         if (this.idSequenceSnapshot != null) {
473             viewIndexCopy.idSequenceSnapshot = new HashMap<String, Integer>(this.idSequenceSnapshot);
474         }
475 
476         if (this.componentExpressionGraphs != null) {
477             viewIndexCopy.componentExpressionGraphs = new HashMap<String, Map<String, String>>(
478                     this.componentExpressionGraphs);
479         }
480 
481         return viewIndexCopy;
482     }
483 
484 }