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.lifecycle;
17  
18  import java.util.List;
19  import java.util.Map;
20  
21  import org.apache.commons.lang.StringUtils;
22  import org.kuali.rice.krad.uif.UifConstants;
23  import org.kuali.rice.krad.uif.UifPropertyPaths;
24  import org.kuali.rice.krad.uif.component.Component;
25  import org.kuali.rice.krad.uif.component.DataBinding;
26  import org.kuali.rice.krad.uif.container.CollectionGroup;
27  import org.kuali.rice.krad.uif.container.Container;
28  import org.kuali.rice.krad.uif.container.Group;
29  import org.kuali.rice.krad.uif.container.LightTable;
30  import org.kuali.rice.krad.uif.container.PageGroup;
31  import org.kuali.rice.krad.uif.field.DataField;
32  import org.kuali.rice.krad.uif.field.Field;
33  import org.kuali.rice.krad.uif.field.FieldGroup;
34  import org.kuali.rice.krad.uif.util.ComponentUtils;
35  import org.kuali.rice.krad.uif.util.ExpressionUtils;
36  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
37  import org.kuali.rice.krad.uif.util.ProcessLogger;
38  import org.kuali.rice.krad.uif.view.View;
39  import org.kuali.rice.krad.uif.view.ViewModel;
40  import org.kuali.rice.krad.web.form.UifFormBase;
41  
42  /**
43   * TODO mark don't forget to fill this in. 
44   * 
45   * @author Kuali Rice Team (rice.collab@kuali.org)
46   */
47  public class ViewLifecycleComponentBuild implements Runnable {
48  
49      private final String origId;
50      private final Component component;
51  
52      /**
53       * Constructor.
54       * 
55       * @param origId The ID of the original component, which component was based on.
56       * @param component The component to build.
57       */
58      public ViewLifecycleComponentBuild(String origId, Component component) {
59          this.origId = origId;
60          this.component = component;
61      }
62  
63      /**
64       * 
65       * 
66       * @see java.lang.Runnable#run()
67       */
68      @Override
69      public void run() {
70          ViewLifecycleProcessor processor = ViewLifecycle.getProcessor();
71          View view = ViewLifecycle.getView();
72          Object model = ViewLifecycle.getModel();
73  
74          if (ViewLifecycle.isTrace()) {
75              ProcessLogger.trace("begin-component-lifecycle:" + component.getId());
76          }
77          
78          Component newComponent = component;
79          Component origComponent = view.getViewIndex().getComponentById(origId);
80  
81          // check if the component is nested in a box layout in order to
82          // reapply the layout item style
83          List<String> origCss = origComponent.getCssClasses();
84          if (origCss != null && (model instanceof UifFormBase)
85                  && ((UifFormBase) model).isUpdateComponentRequest()) {
86  
87              if (origCss.contains(UifConstants.BOX_LAYOUT_HORIZONTAL_ITEM_CSS)) {
88                  component.addStyleClass(UifConstants.BOX_LAYOUT_HORIZONTAL_ITEM_CSS);
89              } else if (origCss.contains(UifConstants.BOX_LAYOUT_VERTICAL_ITEM_CSS)) {
90                  component.addStyleClass(UifConstants.BOX_LAYOUT_VERTICAL_ITEM_CSS);
91              }
92          }
93  
94          Map<String, Object> origContext = origComponent.getContext();
95  
96          Component parent = origContext == null ? null : (Component) origContext
97                  .get(UifConstants.ContextVariableNames.PARENT);
98  
99          // update context on all components within the refresh component to catch context set by parent
100         if (origContext != null) {
101             newComponent.pushAllToContext(origContext);
102 
103             List<Component> nestedComponents = ComponentUtils.getAllNestedComponents(newComponent);
104             for (Component nestedComponent : nestedComponents) {
105                 nestedComponent.pushAllToContext(origContext);
106             }
107         }
108 
109         // make sure the dataAttributes are the same as original
110         newComponent.setDataAttributes(origComponent.getDataAttributes());
111 
112         // initialize the expression evaluator
113         ViewLifecycle.getExpressionEvaluator().initializeEvaluationContext(model);
114 
115         // the expression graph for refreshed components is captured in the view index (initially it might expressions
116         // might have come from a parent), after getting the expression graph then we need to populate the expressions
117         // on the configurable for which they apply
118         Map<String, String> expressionGraph = view.getViewIndex().getComponentExpressionGraphs().get(
119                 newComponent.getBaseId());
120         newComponent.setExpressionGraph(expressionGraph);
121         ExpressionUtils.populatePropertyExpressionsFromGraph(newComponent, false);
122 
123         // binding path should stay the same
124         if (newComponent instanceof DataBinding) {
125             ((DataBinding) newComponent).setBindingInfo(((DataBinding) origComponent).getBindingInfo());
126             ((DataBinding) newComponent).getBindingInfo().setBindingPath(
127                     ((DataBinding) origComponent).getBindingInfo().getBindingPath());
128         }
129 
130         // copy properties that are set by parent components in the full view lifecycle
131         if (newComponent instanceof Field) {
132             ((Field) newComponent).setLabelRendered(((Field) origComponent).isLabelRendered());
133         }
134 
135         if (origComponent.isRefreshedByAction()) {
136             newComponent.setRefreshedByAction(true);
137         }
138 
139         // reset data if needed
140         if (newComponent.isResetDataOnRefresh()) {
141             // TODO: this should handle groups as well, going through nested data fields
142             if (newComponent instanceof DataField) {
143                 // TODO: should check default value
144 
145                 // clear value
146                 ObjectPropertyUtils.initializeProperty(model,
147                         ((DataField) newComponent).getBindingInfo().getBindingPath());
148             }
149         }
150         
151         if (ViewLifecycle.isTrace()) {
152             ProcessLogger.trace("ready:" + newComponent.getId());
153         }
154         
155         processor.performPhase(LifecyclePhaseFactory.initialize(newComponent, model));
156 
157         if (ViewLifecycle.isTrace()) {
158             ProcessLogger.trace("initialize:" + newComponent.getId());
159         }
160 
161         // adjust IDs for suffixes that might have been added by a parent component during the full view lifecycle
162         String suffix = StringUtils.replaceOnce(origComponent.getId(), origComponent.getBaseId(), "");
163         if (StringUtils.isNotBlank(suffix)) {
164             ComponentUtils.updateIdWithSuffix(newComponent, suffix);
165             ComponentUtils.updateChildIdsWithSuffixNested(newComponent, suffix);
166         }
167 
168         // for collections that are nested in the refreshed group, we need to adjust the binding prefix and
169         // set the sub collection id prefix from the original component (this is needed when the group being
170         // refreshed is part of another collection)
171         if (newComponent instanceof Group || newComponent instanceof FieldGroup) {
172             List<CollectionGroup> origCollectionGroups = ComponentUtils.getComponentsOfTypeShallow(
173                     origComponent,
174                     CollectionGroup.class);
175             List<CollectionGroup> collectionGroups = ComponentUtils.getComponentsOfTypeShallow(newComponent,
176                     CollectionGroup.class);
177 
178             for (int i = 0; i < collectionGroups.size(); i++) {
179                 CollectionGroup origCollectionGroup = origCollectionGroups.get(i);
180                 CollectionGroup collectionGroup = collectionGroups.get(i);
181 
182                 String prefix = origCollectionGroup.getBindingInfo().getBindByNamePrefix();
183                 if (StringUtils.isNotBlank(prefix) && StringUtils.isBlank(
184                         collectionGroup.getBindingInfo().getBindByNamePrefix())) {
185                     ComponentUtils.prefixBindingPath(collectionGroup, prefix);
186                 }
187 
188                 String lineSuffix = origCollectionGroup.getSubCollectionSuffix();
189                 collectionGroup.setSubCollectionSuffix(lineSuffix);
190             }
191 
192             // Handle LightTables, as well
193             List<LightTable> origLightTables = ComponentUtils.getComponentsOfTypeShallow(origComponent,
194                     LightTable.class);
195             List<LightTable> lightTables = ComponentUtils.getComponentsOfTypeShallow(newComponent,
196                     LightTable.class);
197 
198             for (int i = 0; i < lightTables.size(); i++) {
199                 LightTable origLightTable = origLightTables.get(i);
200                 LightTable lightTable = lightTables.get(i);
201 
202                 String prefix = origLightTable.getBindingInfo().getBindByNamePrefix();
203                 if (StringUtils.isNotBlank(prefix) && StringUtils.isBlank(
204                         lightTable.getBindingInfo().getBindByNamePrefix())) {
205                     ComponentUtils.prefixBindingPath(lightTable, prefix);
206                 }
207             }
208         }
209 
210         // if disclosed by action and request was made, make sure the component will display
211         if (newComponent.isDisclosedByAction()) {
212             ComponentUtils.setComponentPropertyFinal(newComponent, UifPropertyPaths.RENDER, true);
213             ComponentUtils.setComponentPropertyFinal(newComponent, UifPropertyPaths.HIDDEN, false);
214         }
215 
216         processor.performPhase(LifecyclePhaseFactory.applyModel(newComponent, model, parent));
217 
218         if (ViewLifecycle.isTrace()) {
219             ProcessLogger.trace("apply-model:" + newComponent.getId());
220         }
221 
222         // adjust nestedLevel property on some specific collection cases
223         if (newComponent instanceof Container) {
224             ComponentUtils.adjustNestedLevelsForTableCollections((Container) newComponent, 0);
225         } else if (newComponent instanceof FieldGroup) {
226             ComponentUtils.adjustNestedLevelsForTableCollections(((FieldGroup) newComponent).getGroup(), 0);
227         }
228 
229         processor.performPhase(LifecyclePhaseFactory.finalize(newComponent, model, parent));
230 
231         if (ViewLifecycle.isTrace()) {
232             ProcessLogger.trace("finalize:" + newComponent.getId());
233         }
234 
235         // make sure id, binding, and label settings stay the same as initial
236         // TODO: this currently doesn't work because IDS don't get generated the same, needs reworked
237         // to use paths once that functionality is in palce
238 //        if (newComponent instanceof Group || newComponent instanceof FieldGroup) {
239 //            List<Component> nestedGroupComponents = ComponentUtils.getAllNestedComponents(newComponent);
240 //            List<Component> originalNestedGroupComponents = ComponentUtils
241 //                    .getAllNestedComponents(origComponent);
242 //
243 //            for (Component nestedComponent : nestedGroupComponents) {
244 //                Component origNestedComponent = ComponentUtils.findComponentInList(
245 //                        originalNestedGroupComponents,
246 //                        nestedComponent.getId());
247 //
248 //                if (origNestedComponent != null) {
249 //                    // update binding
250 //                    if (nestedComponent instanceof DataBinding) {
251 //                        ((DataBinding) nestedComponent).setBindingInfo(
252 //                                ((DataBinding) origNestedComponent).getBindingInfo());
253 //                        ((DataBinding) nestedComponent).getBindingInfo().setBindingPath(
254 //                                ((DataBinding) origNestedComponent).getBindingInfo().getBindingPath());
255 //                    }
256 //
257 //                    // update label rendered flag
258 //                    if (nestedComponent instanceof Field) {
259 //                        ((Field) nestedComponent).setLabelRendered(((Field) origNestedComponent)
260 //                                .isLabelRendered());
261 //                    }
262 //
263 //                    if (origNestedComponent.isRefreshedByAction()) {
264 //                        nestedComponent.setRefreshedByAction(true);
265 //                    }
266 //                }
267 //            }
268 //        }
269 
270         // get script for generating growl messages
271         String growlScript = ViewLifecycle.getHelper().buildGrowlScript();
272         ((ViewModel) model).setGrowlScript(growlScript);
273 
274         view.getViewIndex().indexComponent(newComponent);
275 
276         PageGroup page = view.getCurrentPage();
277         // regenerate server message content for page
278         page.getValidationMessages().generateMessages(false, view, model, page);
279 
280         if (ViewLifecycle.isTrace()) {
281             ProcessLogger.trace("end-component-lifecycle:" + newComponent.getId());
282         }
283     }
284 
285 }