View Javadoc

1   /**
2    * Copyright 2005-2011 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.kns.web.struts.form;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.apache.struts.action.ActionMapping;
20  import org.kuali.rice.core.framework.services.CoreFrameworkServiceLocator;
21  import org.kuali.rice.core.web.format.Formatter;
22  import org.kuali.rice.kns.datadictionary.HeaderNavigation;
23  import org.kuali.rice.kns.util.ActionFormUtilMap;
24  import org.kuali.rice.kns.util.WebUtils;
25  import org.kuali.rice.kns.web.struts.form.pojo.PojoFormBase;
26  import org.kuali.rice.kns.web.ui.ExtraButton;
27  import org.kuali.rice.kns.web.ui.HeaderField;
28  import org.kuali.rice.krad.util.KRADConstants;
29  import org.kuali.rice.krad.util.ObjectUtils;
30  import org.springframework.util.AutoPopulatingList;
31  
32  import javax.servlet.ServletRequest;
33  import javax.servlet.http.HttpServletRequest;
34  import java.util.ArrayList;
35  import java.util.HashMap;
36  import java.util.List;
37  import java.util.Map;
38  
39  /**
40   * This class common properites for all action forms.
41   */
42  public class KualiForm extends PojoFormBase {
43      private static final long serialVersionUID = 1L;
44  
45      private static final String literalPrefixAndDelimiter =
46      	KRADConstants.LOOKUP_PARAMETER_LITERAL_PREFIX+ KRADConstants.LOOKUP_PARAMETER_LITERAL_DELIMITER;
47  
48      private String backLocation;
49      private String methodToCall;
50      private String refreshCaller;
51      private String anchor;
52      private Map<String, String> tabStates;
53      private Map actionFormUtilMap;
54      private Map displayedErrors = new HashMap();
55      private Map<String, Object> displayedWarnings = new HashMap<String, Object>();
56      private Map<String, Object> displayedInfo = new HashMap<String, Object>();
57      private int currentTabIndex = 0;
58      private int arbitrarilyHighIndex = 1000000;
59  
60      private String navigationCss;
61      private HeaderNavigation[] headerNavigationTabs;
62      protected List<ExtraButton> extraButtons = new AutoPopulatingList( ExtraButton.class ) ;
63  
64      private boolean fieldLevelHelpEnabled;
65      
66      private List<HeaderField> docInfo;
67      private int numColumns = 2;
68      
69      private String fieldNameToFocusOnAfterSubmit;
70      
71      /**
72       * @see org.kuali.rice.krad.web.struts.pojo.PojoFormBase#addRequiredNonEditableProperties()
73       */
74      @Override
75      public void addRequiredNonEditableProperties(){
76      	super.addRequiredNonEditableProperties();
77      	registerRequiredNonEditableProperty(KRADConstants.REFRESH_CALLER);
78      }
79      
80      public int getNumColumns() {
81          return this.numColumns;
82      }
83  
84      public void setNumColumns(int numColumns) {
85          this.numColumns = numColumns;
86      }
87  
88      public List<HeaderField> getDocInfo() {
89          return this.docInfo;
90      }
91  
92      public void setDocInfo(List<HeaderField> docInfo) {
93          this.docInfo = docInfo;
94      }
95  
96      /**
97       * no args constructor which must init our tab states list
98       */
99      public KualiForm() {
100         this.tabStates = new HashMap<String, String>();
101         this.actionFormUtilMap = new ActionFormUtilMap();
102         this.docInfo = new ArrayList<HeaderField>();
103     }
104 
105     /**
106      * Checks for methodToCall parameter, and if not populated in form calls utility method to parse the string from the request.
107      */
108     public void populate(HttpServletRequest request) {
109         setMethodToCall(WebUtils.parseMethodToCall(this, request));
110         
111         super.populate(request);
112 
113         populateBackLocation(request);
114         populateFieldLevelHelpEnabled(request);
115         
116         if (actionFormUtilMap instanceof ActionFormUtilMap) {
117             ((ActionFormUtilMap) actionFormUtilMap).setCacheValueFinderResults(true);
118         }
119     }
120         
121     private static Boolean ENABLE_FIELD_LEVEL_HELP_IND;
122 
123     protected void populateBackLocation(HttpServletRequest request){
124         if (getParameter(request, "returnLocation") != null) {
125             setBackLocation(getParameter(request, "returnLocation"));
126         }
127     }
128     
129     /**
130      * Populates whether the each field will have field-level help associated with it.  Depending on how the jsp/tags are implemented, the value
131      * populated by this method may be overruled by other settings
132      * 
133      * @param request
134      */
135     protected void populateFieldLevelHelpEnabled(HttpServletRequest request) {
136     	if ( ENABLE_FIELD_LEVEL_HELP_IND == null ) {
137     		ENABLE_FIELD_LEVEL_HELP_IND = CoreFrameworkServiceLocator.getParameterService().getParameterValueAsBoolean(
138                     KRADConstants.KRAD_NAMESPACE,
139                     KRADConstants.DetailTypes.ALL_DETAIL_TYPE, KRADConstants.SystemGroupParameterNames.ENABLE_FIELD_LEVEL_HELP_IND, Boolean.FALSE);
140     	}
141     	setFieldLevelHelpEnabled( ENABLE_FIELD_LEVEL_HELP_IND);
142     }
143 
144     public Map getDisplayedErrors() {
145         return displayedErrors;
146     }
147 
148     /**
149 	 * @return the displayedWarnings
150 	 */
151 	public Map<String, Object> getDisplayedWarnings() {
152 		return this.displayedWarnings;
153 	}
154 
155 	/**
156 	 * @return the displayedInfo
157 	 */
158 	public Map<String, Object> getDisplayedInfo() {
159 		return this.displayedInfo;
160 	}
161 
162 	/**
163      * Used by the dispatch action to determine which action method to call into.
164      *
165      * @return Returns the methodToCall.
166      */
167     public String getMethodToCall() {
168         return methodToCall;
169     }
170 
171 
172     /**
173      * @param methodToCall The methodToCall to set.
174      */
175     public void setMethodToCall(String methodToCall) {
176         this.methodToCall = methodToCall;
177     }
178 
179 
180     /**
181      * Can be used by actions refresh method to determine what called the the refresh method.
182      *
183      * @return Returns the refreshCaller.
184      */
185     public String getRefreshCaller() {
186         return refreshCaller;
187     }
188 
189 
190     /**
191      * @param refreshCaller The refreshCaller to set.
192      */
193     public void setRefreshCaller(String refreshCaller) {
194         this.refreshCaller = refreshCaller;
195     }
196 
197     /**
198      * @return the tab state list
199      */
200     public Map<String, String> getTabStates() {
201         return tabStates;
202     }
203 
204     /**
205      * simple setter for the tab state Map
206      *
207      * @param tabStates
208      */
209     public void setTabStates(Map<String, String> tabStates) {
210         this.tabStates = tabStates;
211     }
212 
213     /**
214      * Special getter based on key to work with multi rows for tab state objects
215      */
216     public String getTabState(String key) {
217         String state = KRADConstants.EMPTY_STRING;
218         if (tabStates.containsKey(key)) {
219             if (tabStates.get(key) instanceof String) {
220             	state = tabStates.get(key);
221             }
222             else {
223             	//This is the case where the value is an Array of String,
224             	//so we'll have to get the first element
225             	Object result = tabStates.get(key);
226             	result.getClass();
227             	state = ((String[])result)[0];
228             }
229         }
230 
231         return state;
232     }
233 
234     public int getCurrentTabIndex() {
235         return currentTabIndex;
236     }
237 
238     public void setCurrentTabIndex(int currentTabIndex) {
239         this.currentTabIndex = currentTabIndex;
240     }
241 
242     public void incrementTabIndex() {
243         this.currentTabIndex++;
244     }
245 
246     public int getNextArbitrarilyHighIndex() {
247         return this.arbitrarilyHighIndex++;
248     }
249     
250 	public String getFieldNameToFocusOnAfterSubmit() {
251 		return this.fieldNameToFocusOnAfterSubmit;
252 	}
253 
254 	public void setFieldNameToFocusOnAfterSubmit(String fieldNameToFocusOnAfterSubmit) {
255 		this.fieldNameToFocusOnAfterSubmit = fieldNameToFocusOnAfterSubmit;
256 	}
257 
258 	/**
259      * @return Returns the validOptionsMap.
260      */
261     public Map getActionFormUtilMap() {
262         return actionFormUtilMap;
263     }
264 
265     /**
266      * @param validOptionsMap The validOptionsMap to set.
267      */
268     public void setActionFormUtilMap(Map validOptionsMap) {
269         this.actionFormUtilMap = validOptionsMap;
270     }
271 
272     /**
273      * Gets the headerNavigationTabs attribute.
274      *
275      * @return Returns the headerNavigationTabs.
276      */
277     public HeaderNavigation[] getHeaderNavigationTabs() {
278         return headerNavigationTabs;
279     }
280 
281     /**
282      * Sets the headerNavigationTabs attribute value.
283      *
284      * @param headerNavigationTabs The headerNavigationTabs to set.
285      */
286     public void setHeaderNavigationTabs(HeaderNavigation[] headerNavigationTabs) {
287         this.headerNavigationTabs = headerNavigationTabs;
288     }
289 
290     /**
291      * Gets the navigationCss attribute.
292      *
293      * @return Returns the navigationCss.
294      */
295     public String getNavigationCss() {
296         return navigationCss;
297     }
298 
299     /**
300      * Sets the navigationCss attribute value.
301      *
302      * @param navigationCss The navigationCss to set.
303      */
304     public void setNavigationCss(String navigationCss) {
305         this.navigationCss = navigationCss;
306     }
307 
308     public String getAnchor() {
309         return anchor;
310     }
311 
312     public void setAnchor(String anchor) {
313         this.anchor = anchor;
314     }
315 
316     public List<ExtraButton> getExtraButtons() {
317         return extraButtons;
318     }
319 
320     public void setExtraButtons(List<ExtraButton> extraButtons) {
321         if ( extraButtons instanceof AutoPopulatingList ) {
322             this.extraButtons = extraButtons;
323         } else {
324             this.extraButtons.clear();
325             this.extraButtons.addAll( extraButtons );
326         }
327     }
328 
329     public ExtraButton getExtraButton( int index ) {
330         return extraButtons.get( index );
331     }
332 
333     public void setExtraButton( int index, ExtraButton extraButton ) {
334         extraButtons.set( index, extraButton );
335     }
336 
337     /**
338      * Returns whether field level help is enabled for this form.
339      * 
340      * @return
341      */
342     public boolean isFieldLevelHelpEnabled() {
343         return this.fieldLevelHelpEnabled;
344     }
345 
346     public void setFieldLevelHelpEnabled(boolean fieldLevelHelpEnabled) {
347         this.fieldLevelHelpEnabled = fieldLevelHelpEnabled;
348     }
349     
350     
351     /**
352      * Retrieves a value from the form for the purposes of passing it as a parameter into the lookup or inquiry frameworks 
353      * 
354      * @param parameterName the name of the parameter, as expected by the lookup or inquiry frameworks
355      * @param parameterValueLocation the name of the property containing the value of the parameter
356      * @return the value of the parameter
357      */
358     public String retrieveFormValueForLookupInquiryParameters(String parameterName, String parameterValueLocation) {
359     	// dereference literal values by simply trimming of the prefix
360     	if (parameterValueLocation.startsWith(literalPrefixAndDelimiter)) {
361     		return parameterValueLocation.substring(literalPrefixAndDelimiter.length());
362     	}
363 
364     	Object value = ObjectUtils.getPropertyValue(this, parameterValueLocation);
365 		if (value == null) {
366 			return null;
367 		}
368 		if (value instanceof String) {
369 			return (String) value;
370 		}
371 		Formatter formatter = Formatter.getFormatter(value.getClass());
372 		return (String) formatter.format(value);	
373     }
374 
375 	/**
376 	 * This overridden method ...
377 	 * 
378 	 * @see org.kuali.rice.krad.web.struts.pojo.PojoFormBase#shouldPropertyBePopulatedInForm(java.lang.String, javax.servlet.http.HttpServletRequest)
379 	 */
380 	@Override
381 	public boolean shouldPropertyBePopulatedInForm(
382 			String requestParameterName, HttpServletRequest request) {
383 		if (requestParameterName.startsWith(KRADConstants.TAB_STATES)) {
384 			return true;
385 		}
386 		
387 		if (requestParameterName.equals(KRADConstants.DISPATCH_REQUEST_PARAMETER)) {
388 			String methodToCallParameterName = request.getParameter(KRADConstants.DISPATCH_REQUEST_PARAMETER);
389 			if(StringUtils.equals(methodToCallParameterName, KRADConstants.RETURN_METHOD_TO_CALL)){
390 				return true;
391 			}
392 		}
393 		
394 		return super.shouldPropertyBePopulatedInForm(requestParameterName, request);
395 	}
396     
397     public boolean shouldMethodToCallParameterBeUsed(String methodToCallParameterName, String methodToCallParameterValue, HttpServletRequest request) {
398     	if ("GET".equalsIgnoreCase(request.getMethod())) {
399     		return true;
400     	}
401     	if (shouldPropertyBePopulatedInForm(methodToCallParameterName, request)) {
402     		return true;
403     	}
404     	if (methodToCallParameterName != null && WebUtils.endsWithCoordinates(methodToCallParameterName)) {
405     		methodToCallParameterName = methodToCallParameterName.substring(0, WebUtils.getIndexOfCoordinateExtension(methodToCallParameterName));
406         	if (shouldPropertyBePopulatedInForm(methodToCallParameterName, request)) {
407         		return true;
408         	}
409     	}
410     	if (KRADConstants.METHOD_TO_CALL_PATH.equals(methodToCallParameterName)) {
411     		if (shouldPropertyBePopulatedInForm(methodToCallParameterValue, request)) {
412     			return true;
413     		}
414     		if (methodToCallParameterValue != null && WebUtils.endsWithCoordinates(methodToCallParameterName)) {
415     			methodToCallParameterValue = methodToCallParameterValue.substring(0, WebUtils
416                         .getIndexOfCoordinateExtension(methodToCallParameterName));
417     			if (shouldPropertyBePopulatedInForm(methodToCallParameterValue, request)) {
418         			return true;
419         		}
420     		}
421     	}
422     	return false;
423     }
424 
425 	/**
426 	 * @see org.kuali.rice.krad.web.struts.pojo.PojoFormBase#clearEditablePropertyInformation()
427 	 */
428 	@Override
429 	public void clearEditablePropertyInformation() {
430 		super.clearEditablePropertyInformation();
431 	}
432 	
433 	public void setDerivedValuesOnForm(HttpServletRequest request) {
434 	}
435 
436 	/**
437 	 * @see org.apache.struts.action.ActionForm#reset(org.apache.struts.action.ActionMapping, javax.servlet.http.HttpServletRequest)
438 	 */
439 	@Override
440 	public void reset(ActionMapping mapping, HttpServletRequest request) {
441 		super.reset(mapping, request);
442 		if (extraButtons != null) {
443 			extraButtons.clear();
444 		}
445 		//fieldNameToFocusOnAfterSubmit = "";
446 		clearDisplayedMessages();
447 	}
448 
449 	/**
450 	 * @see org.apache.struts.action.ActionForm#reset(org.apache.struts.action.ActionMapping, javax.servlet.ServletRequest)
451 	 */
452 	@Override
453 	public void reset(ActionMapping mapping, ServletRequest request) {
454 		super.reset(mapping, request);
455 		if (extraButtons != null) {
456 			extraButtons.clear();
457 		}
458 		//fieldNameToFocusOnAfterSubmit = "";
459 		clearDisplayedMessages();
460 	}
461 	
462 	private void clearDisplayedMessages() {
463 		if (displayedErrors != null) {
464 			displayedErrors.clear();
465 		}
466 		if (displayedWarnings != null) {
467 			displayedWarnings.clear();
468 		}
469 		if (displayedInfo != null) {
470 			displayedInfo.clear();
471 		}
472 	}
473 	
474     /**
475 	 * @return the backLocation
476 	 */
477 	public String getBackLocation() {
478 		return this.backLocation;
479 	}
480 
481 	/**
482 	 * @param backLocation the backLocation to set
483 	 */
484 	public void setBackLocation(String backLocation) {
485 		this.backLocation = backLocation;
486 	}
487 
488 }