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.lifecycle;
17  
18  import java.util.List;
19  import java.util.Map;
20  import java.util.Queue;
21  import java.util.Set;
22  
23  import org.apache.commons.lang.StringUtils;
24  import org.kuali.rice.krad.uif.UifConstants;
25  import org.kuali.rice.krad.uif.component.Component;
26  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle.LifecycleEvent;
27  import org.kuali.rice.krad.uif.lifecycle.finalize.SetReadOnlyOnDataBindingTask;
28  import org.kuali.rice.krad.uif.lifecycle.model.AfterEvaluateExpressionTask;
29  import org.kuali.rice.krad.uif.lifecycle.model.ApplyAuthAndPresentationLogicTask;
30  import org.kuali.rice.krad.uif.lifecycle.model.ComponentDefaultApplyModelTask;
31  import org.kuali.rice.krad.uif.lifecycle.model.EvaluateExpressionsTask;
32  import org.kuali.rice.krad.uif.lifecycle.model.HelperCustomApplyModelTask;
33  import org.kuali.rice.krad.uif.lifecycle.model.PopulateComponentContextTask;
34  import org.kuali.rice.krad.uif.lifecycle.model.RefreshStateModifyTask;
35  import org.kuali.rice.krad.uif.lifecycle.model.SuffixIdFromContainerTask;
36  import org.kuali.rice.krad.uif.lifecycle.model.SyncClientSideStateTask;
37  import org.kuali.rice.krad.uif.util.LifecycleElement;
38  
39  /**
40   * Lifecycle phase processing task for applying the model to a component.
41   *
42   * <p>
43   * During the apply model phase each component of the tree if invoked to setup any state based on
44   * the given model data
45   * </p>
46   *
47   * <p>
48   * Part of the view lifecycle that applies the model data to the view. Should be called after the
49   * model has been populated before the view is rendered. The main things that occur during this
50   * phase are:
51   * <ul>
52   * <li>Generation of dynamic fields (such as collection rows)</li>
53   * <li>Execution of conditional logic (hidden, read-only, required settings based on model values)</li>
54   * </ul>
55   * </p>
56   *
57   * <p>
58   * The update phase can be called multiple times for the view's lifecycle (typically only once per
59   * request)
60   * </p>
61   *
62   * @author Kuali Rice Team (rice.collab@kuali.org)
63   */
64  public class ApplyModelComponentPhase extends ViewLifecyclePhaseBase {
65  
66      /**
67       * Set of IDs that have been visited during the view's apply model phase.
68       *
69       * <p>
70       * This reference is typically shared by all component apply model phases.
71       * </p>
72       */
73      private Set<String> visitedIds;
74  
75      /**
76       * {@inheritDoc}
77       */
78      @Override
79      protected void recycle() {
80          super.recycle();
81          visitedIds = null;
82      }
83  
84      /**
85       * Create a new lifecycle phase processing task for applying the model to a element.
86       *
87       * @param element The element the model should be applied to
88       * @param model Top level object containing the data
89       * @param path The path to the element relative to the parent element
90       * @param refreshPaths list of paths to run lifecycle on when executing a refresh lifecycle
91       * @param parent The parent element
92       * @param nextPhase The phase to queue directly upon completion of this phase, if applicable
93       * @param visitedIds Tracks components ids that have been seen for adjusting duplicates
94       */
95      protected void prepare(LifecycleElement element, Object model, String path, List<String> refreshPaths,
96              Component parent, ViewLifecyclePhaseBase nextPhase, Set<String> visitedIds) {
97          super.prepare(element, model, path, refreshPaths, parent, nextPhase);
98  
99          this.visitedIds = visitedIds;
100     }
101 
102     /**
103      * {@inheritDoc}
104      */
105     @Override
106     public String getViewPhase() {
107         return UifConstants.ViewPhases.APPLY_MODEL;
108     }
109 
110     /**
111      * {@inheritDoc}
112      */
113     @Override
114     public String getStartViewStatus() {
115         return UifConstants.ViewStatus.INITIALIZED;
116     }
117 
118     /**
119      * {@inheritDoc}
120      */
121     @Override
122     public String getEndViewStatus() {
123         return UifConstants.ViewStatus.MODEL_APPLIED;
124     }
125 
126     /**
127      * {@inheritDoc}
128      */
129     @Override
130     public LifecycleEvent getEventToNotify() {
131         return null;
132     }
133 
134     /**
135      * Visit a lifecycle element.
136      *
137      * @param element The lifecycle element (component or layout manager) to mark as visisted.
138      * @return True if the element has been visited before, false if this was the first visit.
139      */
140     public boolean visit(LifecycleElement element) {
141         if (visitedIds.contains(element.getId())) {
142             return true;
143         }
144 
145         synchronized (visitedIds) {
146             return !visitedIds.add(element.getId());
147         }
148     }
149 
150     /**
151      * Applies the model data to a component of the View instance
152      *
153      * <p>
154      * TODO: Revise - The component is invoked to to apply the model data. Here the component can
155      * generate any additional fields needed or alter the configured fields. After the component is
156      * invoked a hook for custom helper service processing is invoked. Finally the method is
157      * recursively called for all the component children
158      * </p>
159      *
160      * {@inheritDoc}
161      */
162     @Override
163     protected void initializePendingTasks(Queue<ViewLifecycleTask<?>> tasks) {
164         if ((getParent() != null) && StringUtils.isNotBlank(getParent().getContainerIdSuffix())) {
165             tasks.add(LifecycleTaskFactory.getTask(SuffixIdFromContainerTask.class, this));
166         }
167 
168         tasks.add(LifecycleTaskFactory.getTask(PopulateComponentContextTask.class, this));
169         tasks.add(LifecycleTaskFactory.getTask(EvaluateExpressionsTask.class, this));
170         tasks.add(LifecycleTaskFactory.getTask(AfterEvaluateExpressionTask.class, this));
171         
172         tasks.add(LifecycleTaskFactory.getTask(SyncClientSideStateTask.class, this));
173         tasks.add(LifecycleTaskFactory.getTask(ApplyAuthAndPresentationLogicTask.class, this));
174 
175         if (ViewLifecycle.isRefreshComponent(getViewPhase(), getViewPath())) {
176             tasks.add(LifecycleTaskFactory.getTask(RefreshStateModifyTask.class, this));
177         }
178 
179         tasks.add(LifecycleTaskFactory.getTask(ComponentDefaultApplyModelTask.class, this));
180         getElement().initializePendingTasks(this, tasks);
181         tasks.offer(LifecycleTaskFactory.getTask(RunComponentModifiersTask.class, this));
182         tasks.add(LifecycleTaskFactory.getTask(HelperCustomApplyModelTask.class, this));
183         tasks.add(LifecycleTaskFactory.getTask(SetReadOnlyOnDataBindingTask.class, this));
184     }
185 
186     /**
187      * Define all nested lifecycle components, and component prototypes, as successors.
188      * 
189      * {@inheritDoc}
190      */
191     @Override
192     protected ViewLifecyclePhase initializeSuccessor(LifecycleElement nestedElement, String nestedPath,
193             Component parent) {
194         ApplyModelComponentPhase applyModelPhase = LifecyclePhaseFactory.applyModel(nestedElement, getModel(),
195                 nestedPath, getRefreshPaths(), parent, null, visitedIds);
196 
197         if (nestedElement.isInitialized()) {
198             return applyModelPhase;
199         }
200 
201         return LifecyclePhaseFactory.initialize(nestedElement, getModel(), nestedPath, getRefreshPaths(), parent,
202                 applyModelPhase);
203     }
204 
205 }