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.element;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.kuali.rice.core.api.exception.RiceRuntimeException;
20  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
21  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
22  import org.kuali.rice.krad.datadictionary.parse.BeanTags;
23  import org.kuali.rice.krad.datadictionary.validator.ValidationTrace;
24  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
25  import org.kuali.rice.krad.uif.UifConstants;
26  import org.kuali.rice.krad.uif.UifParameters;
27  import org.kuali.rice.krad.uif.UifPropertyPaths;
28  import org.kuali.rice.krad.uif.component.Component;
29  import org.kuali.rice.krad.uif.component.ComponentSecurity;
30  import org.kuali.rice.krad.uif.container.DialogGroup;
31  import org.kuali.rice.krad.uif.container.Group;
32  import org.kuali.rice.krad.uif.field.DataField;
33  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
34  import org.kuali.rice.krad.uif.util.ComponentFactory;
35  import org.kuali.rice.krad.uif.util.LifecycleElement;
36  import org.kuali.rice.krad.uif.util.ScriptUtils;
37  import org.kuali.rice.krad.uif.util.UrlInfo;
38  import org.kuali.rice.krad.uif.view.ExpressionEvaluator;
39  import org.kuali.rice.krad.uif.view.FormView;
40  import org.kuali.rice.krad.uif.view.View;
41  import org.kuali.rice.krad.util.KRADUtils;
42  
43  import java.util.ArrayList;
44  import java.util.HashMap;
45  import java.util.List;
46  import java.util.Map;
47  
48  /**
49   * Field that presents an action that can be taken on the UI such as submitting the form or invoking a script.
50   *
51   * @author Kuali Rice Team (rice.collab@kuali.org)
52   */
53  @BeanTags({@BeanTag(name = "action", parent = "Uif-Action"),
54          @BeanTag(name = "actionImage", parent = "Uif-ActionImage"),
55          @BeanTag(name = "button", parent = "Uif-PrimaryActionButton"),
56          @BeanTag(name = "secondaryButton", parent = "Uif-SecondaryActionButton"),
57          @BeanTag(name = "buttonLarge", parent = "Uif-PrimaryActionButton-Large"),
58          @BeanTag(name = "secondaryButtonLarge", parent = "Uif-SecondaryActionButton-Large"),
59          @BeanTag(name = "buttonSmall", parent = "Uif-PrimaryActionButton-Small"),
60          @BeanTag(name = "secondaryButtonSmall", parent = "Uif-SecondaryActionButton-Small"),
61          @BeanTag(name = "buttonMini", parent = "Uif-PrimaryActionButton-Mini"),
62          @BeanTag(name = "secondaryButtonMini", parent = "Uif-SecondaryActionButton-Mini"),
63          @BeanTag(name = "actionLink", parent = "Uif-ActionLink"),
64          @BeanTag(name = "navigationActionLink", parent = "Uif-NavigationActionLink"),
65          @BeanTag(name = "navigationButton", parent = "Uif-NavigationActionButton"),
66          @BeanTag(name = "secondaryNavigationActionButton", parent = "Uif-SecondaryNavigationActionButton")})
67  public class Action extends ContentElementBase {
68      private static final long serialVersionUID = 1025672792657238829L;
69  
70      private String methodToCall;
71      private String actionEvent;
72      private String navigateToPageId;
73  
74      private String actionScript;
75      private UrlInfo actionUrl;
76  
77      private String actionLabel;
78      private boolean renderInnerTextSpan;
79  
80      private Image actionImage;
81      private String actionImagePlacement;
82  
83      private String iconClass;
84      private String actionIconPlacement;
85  
86      private String jumpToIdAfterSubmit;
87      private String jumpToNameAfterSubmit;
88      private String focusOnIdAfterSubmit;
89  
90      private boolean performClientSideValidation;
91      private boolean performDirtyValidation;
92      private boolean clearDirtyOnAction;
93      private boolean dirtyOnAction;
94  
95      private String preSubmitCall;
96      private String confirmationPromptText;
97      private DialogGroup confirmationDialog;
98  
99      private String dialogDismissOption;
100     private String dialogResponse;
101 
102     private boolean ajaxSubmit;
103     private String ajaxReturnType;
104     private String refreshId;
105     private String refreshPropertyName;
106 
107     private String successCallback;
108     private String errorCallback;
109 
110     private String loadingMessageText;
111     private boolean disableBlocking;
112 
113     private Map<String, String> additionalSubmitData;
114     private Map<String, String> actionParameters;
115 
116     private boolean evaluateDisabledOnKeyUp;
117 
118     private boolean defaultEnterKeyAction;
119 
120     private boolean disabled;
121     private String disabledReason;
122     private String disabledExpression;
123     private String disabledConditionJs;
124     private List<String> disabledConditionControlNames;
125 
126     private List<String> disabledWhenChangedPropertyNames;
127     private List<String> enabledWhenChangedPropertyNames;
128 
129     /**
130      * Sets initial field values and initializes collections.
131      */
132     public Action() {
133         super();
134 
135         actionImagePlacement = UifConstants.Position.LEFT.name();
136         actionIconPlacement = UifConstants.Position.LEFT.name();
137 
138         ajaxSubmit = true;
139 
140         successCallback = "";
141         errorCallback = "";
142         preSubmitCall = "";
143 
144         additionalSubmitData = new HashMap<String, String>();
145         actionParameters = new HashMap<String, String>();
146 
147         disabled = false;
148         disabledWhenChangedPropertyNames = new ArrayList<String>();
149         enabledWhenChangedPropertyNames = new ArrayList<String>();
150     }
151 
152     /**
153      * Sets the disabledExpression, if any, evaluates it and sets the disabled property.
154      *
155      * @param model top level object containing the data (could be the form or a
156      * @param parent parent component
157      */
158     public void performApplyModel(Object model, LifecycleElement parent) {
159         super.performApplyModel(model, parent);
160 
161         disabledExpression = this.getPropertyExpression("disabled");
162         if (disabledExpression != null) {
163             ExpressionEvaluator expressionEvaluator = ViewLifecycle.getExpressionEvaluator();
164 
165             disabledExpression = expressionEvaluator.replaceBindingPrefixes(ViewLifecycle.getView(), this,
166                     disabledExpression);
167             disabled = (Boolean) expressionEvaluator.evaluateExpression(this.getContext(), disabledExpression);
168         }
169 
170         if (actionUrl != null) {
171             ViewLifecycle.getExpressionEvaluator().populatePropertyExpressionsFromGraph(actionUrl, false);
172             ViewLifecycle.getExpressionEvaluator().evaluateExpressionsOnConfigurable(ViewLifecycle.getView(),
173                     actionUrl, this.getContext());
174         }
175 
176         if (StringUtils.isNotBlank(confirmationPromptText) && (confirmationDialog != null) && StringUtils.isBlank(
177                 confirmationDialog.getPromptText())) {
178             confirmationDialog.setPromptText(confirmationPromptText);
179         }
180 
181         addConfirmDialogToView();
182     }
183 
184     /**
185      * For confirm text without a dialog, add instance of yes no dialog to view so it is already available
186      * on the client for dynamic dialog creation.
187      */
188     protected void addConfirmDialogToView() {
189         if (StringUtils.isBlank(confirmationPromptText) || (confirmationDialog != null)) {
190             return;
191         }
192 
193         boolean containsYesNoDialog = false;
194 
195         List<Group> viewDialogs = ViewLifecycle.getView().getDialogs();
196         if (viewDialogs == null) {
197             viewDialogs = new ArrayList<Group>();
198         } else {
199             for (Group dialogGroup : viewDialogs) {
200                 if (StringUtils.equals(ComponentFactory.YES_NO_DIALOG, dialogGroup.getId())) {
201                     containsYesNoDialog = true;
202                 }
203             }
204         }
205 
206         if (!containsYesNoDialog) {
207             Group confirmDialog = ComponentFactory.getYesNoDialog();
208             confirmDialog.setId(ComponentFactory.YES_NO_DIALOG);
209 
210             viewDialogs.add(confirmDialog);
211         }
212     }
213 
214     /**
215      * The following finalization is performed:
216      *
217      * <ul>
218      * <li>Add methodToCall action parameter if set and setup event code for
219      * setting action parameters</li>
220      * <li>Invoke method to build the data attributes and submit data for the action</li>
221      * <li>Compose the final onclick script for the action</li>
222      * <li>Parses the disabled expressions, if any, to equivalent javascript and evaluates the disable/enable when
223      * changed property names</li>
224      * </ul>
225      *
226      * {@inheritDoc}
227      */
228     @Override
229     public void performFinalize(Object model, LifecycleElement parent) {
230         super.performFinalize(model, parent);
231 
232         View view = ViewLifecycle.getView();
233         ExpressionEvaluator expressionEvaluator = ViewLifecycle.getExpressionEvaluator();
234 
235         if (StringUtils.isNotEmpty(disabledExpression)
236                 && !disabledExpression.equalsIgnoreCase("true")
237                 && !disabledExpression.equalsIgnoreCase("false")) {
238             disabledConditionControlNames = new ArrayList<String>();
239             disabledConditionJs = ViewLifecycle.getExpressionEvaluator().parseExpression(disabledExpression,
240                     disabledConditionControlNames, this.getContext());
241         }
242 
243         List<String> adjustedDisablePropertyNames = new ArrayList<String>();
244         for (String propertyName : disabledWhenChangedPropertyNames) {
245             adjustedDisablePropertyNames.add(expressionEvaluator.replaceBindingPrefixes(view, this, propertyName));
246         }
247         disabledWhenChangedPropertyNames = adjustedDisablePropertyNames;
248 
249         List<String> adjustedEnablePropertyNames = new ArrayList<String>();
250         for (String propertyName : enabledWhenChangedPropertyNames) {
251             adjustedEnablePropertyNames.add(expressionEvaluator.replaceBindingPrefixes(view, this, propertyName));
252         }
253         enabledWhenChangedPropertyNames = adjustedEnablePropertyNames;
254 
255         // clear alt text to avoid screen reader confusion when using image in button with text
256         if (actionImage != null && StringUtils.isNotBlank(actionImagePlacement) && StringUtils.isNotBlank(actionLabel)) {
257             actionImage.setAltText("");
258         }
259 
260         // when icon only is set, add the icon class to the action
261         if (StringUtils.isNotBlank(iconClass) && (UifConstants.ICON_ONLY_PLACEMENT.equals(actionIconPlacement)
262                 || StringUtils.isBlank(actionLabel))) {
263             getCssClasses().add(iconClass);
264 
265             // force icon only placement
266             actionIconPlacement = UifConstants.ICON_ONLY_PLACEMENT;
267         }
268 
269         if (!actionParameters.containsKey(UifConstants.UrlParams.ACTION_EVENT) && StringUtils.isNotBlank(actionEvent)) {
270             actionParameters.put(UifConstants.UrlParams.ACTION_EVENT, actionEvent);
271         }
272 
273         if (StringUtils.isNotBlank(navigateToPageId)) {
274             actionParameters.put(UifParameters.NAVIGATE_TO_PAGE_ID, navigateToPageId);
275             if (StringUtils.isBlank(methodToCall)) {
276                 this.methodToCall = UifConstants.MethodToCallNames.NAVIGATE;
277             }
278         }
279 
280         if (!actionParameters.containsKey(UifConstants.CONTROLLER_METHOD_DISPATCH_PARAMETER_NAME) && StringUtils
281                 .isNotBlank(methodToCall)) {
282             actionParameters.put(UifConstants.CONTROLLER_METHOD_DISPATCH_PARAMETER_NAME, methodToCall);
283         }
284 
285         setupRefreshAction(view);
286 
287         // Apply dirty check if it is enabled for the view and the action requires it
288         if (view instanceof FormView) {
289             performDirtyValidation = performDirtyValidation && ((FormView) view).isApplyDirtyCheck();
290         }
291 
292         if (StringUtils.isBlank(getActionScript()) && (actionUrl != null) && actionUrl.isFullyConfigured()) {
293             String actionScript = ScriptUtils.buildFunctionCall(UifConstants.JsFunctions.REDIRECT, actionUrl.getHref());
294             setActionScript(actionScript);
295         }
296 
297         // add the method to call as an available method
298         if (StringUtils.isNotBlank(methodToCall)) {
299             ViewLifecycle.getViewPostMetadata().addAvailableMethodToCall(methodToCall);
300         }
301 
302         // add additional submit data as accessible binding paths, and method to call as accessible method
303         if (isRender()) {
304             for (String additionalSubmitPath : additionalSubmitData.keySet()) {
305                 ViewLifecycle.getViewPostMetadata().addAccessibleBindingPath(additionalSubmitPath);
306             }
307 
308             if (StringUtils.isNotBlank(methodToCall)) {
309                 ViewLifecycle.getViewPostMetadata().addAccessibleMethodToCall(methodToCall);
310             }
311         }
312 
313         buildActionData(view, model, parent);
314     }
315 
316     /**
317      * When the action is updating a component sets up the refresh script for the component (found by the
318      * given refresh id or refresh property name.
319      *
320      * @param view view instance the action belongs to
321      */
322     protected void setupRefreshAction(View view) {
323         // if refresh property or id is given, make return type update component
324         // TODO: what if the refresh id is the page id? we should set the return type as update page
325         if (StringUtils.isNotBlank(refreshPropertyName) || StringUtils.isNotBlank(refreshId)) {
326             ajaxReturnType = UifConstants.AjaxReturnTypes.UPDATECOMPONENT.getKey();
327         }
328 
329         // if refresh property name is given, adjust the binding and then attempt to find the
330         // component in the view index
331         Component refreshComponent = null;
332         if (StringUtils.isNotBlank(refreshPropertyName)) {
333             // TODO: does this support all binding prefixes?
334             if (refreshPropertyName.startsWith(UifConstants.NO_BIND_ADJUST_PREFIX)) {
335                 refreshPropertyName = StringUtils.removeStart(refreshPropertyName, UifConstants.NO_BIND_ADJUST_PREFIX);
336             } else if (StringUtils.isNotBlank(view.getDefaultBindingObjectPath())) {
337                 refreshPropertyName = view.getDefaultBindingObjectPath() + "." + refreshPropertyName;
338             }
339 
340             DataField dataField = view.getViewIndex().getDataFieldByPath(refreshPropertyName);
341             if (dataField != null) {
342                 refreshComponent = dataField;
343                 refreshId = refreshComponent.getId();
344             }
345         } else if (StringUtils.isNotBlank(refreshId)) {
346             Component component = view.getViewIndex().getComponentById(refreshId);
347             if (component != null) {
348                 refreshComponent = component;
349             }
350         }
351 
352         if (refreshComponent != null) {
353             refreshComponent.setRefreshedByAction(true);
354         }
355     }
356 
357     /**
358      * Builds the data attributes that will be read client side to determine how to
359      * handle the action and the additional data that should be submitted with the action
360      *
361      * <p>
362      * Note these data attributes will be exposed as a data map client side. The simple attributes (non object
363      * value) are also written out as attributes on the action element.
364      * </p>
365      *
366      * @param view view instance the action belongs to
367      * @param model model object containing the view data
368      * @param parent component the holds the action
369      */
370     protected void buildActionData(View view, Object model, LifecycleElement parent) {
371         HashMap<String, String> actionDataAttributes = new HashMap<String, String>();
372 
373         Map<String, String> dataDefaults =
374                 (Map<String, String>) (KRADServiceLocatorWeb.getDataDictionaryService().getDictionaryBean(
375                         UifConstants.ACTION_DEFAULTS_MAP_ID));
376 
377         // map properties to data attributes
378         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.AJAX_SUBMIT,
379                 Boolean.toString(ajaxSubmit));
380         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
381                 UifConstants.ActionDataAttributes.SUCCESS_CALLBACK, this.successCallback);
382         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.ERROR_CALLBACK,
383                 this.errorCallback);
384         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
385                 UifConstants.ActionDataAttributes.PRE_SUBMIT_CALL, this.preSubmitCall);
386         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
387                 UifConstants.ActionDataAttributes.LOADING_MESSAGE, this.loadingMessageText);
388         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
389                 UifConstants.ActionDataAttributes.DISABLE_BLOCKING, Boolean.toString(this.disableBlocking));
390         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
391                 UifConstants.ActionDataAttributes.AJAX_RETURN_TYPE, this.ajaxReturnType);
392         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.REFRESH_ID,
393                 this.refreshId);
394         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.VALIDATE,
395                 Boolean.toString(this.performClientSideValidation));
396         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
397                 UifConstants.ActionDataAttributes.DIRTY_ON_ACTION, Boolean.toString(this.dirtyOnAction));
398         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.CLEAR_DIRTY,
399                 Boolean.toString(this.clearDirtyOnAction));
400         addActionDataSettingsValue(actionDataAttributes, dataDefaults,
401                 UifConstants.ActionDataAttributes.PERFORM_DIRTY_VALIDATION, Boolean.toString(
402                 this.performDirtyValidation));
403 
404         if (confirmationDialog != null) {
405             addDataAttribute(UifConstants.ActionDataAttributes.CONFIRM_DIALOG_ID, confirmationDialog.getId());
406         } else if (StringUtils.isNotBlank(confirmationPromptText)) {
407             addDataAttribute(UifConstants.ActionDataAttributes.CONFIRM_PROMPT_TEXT, confirmationPromptText);
408         }
409 
410         if (StringUtils.isNotBlank(dialogDismissOption)) {
411             addDataAttribute(UifConstants.DataAttributes.DISMISS_DIALOG_OPTION, dialogDismissOption);
412         }
413 
414         if (StringUtils.isNotBlank(dialogResponse)) {
415             addDataAttribute(UifConstants.DataAttributes.DISMISS_RESPONSE, dialogResponse);
416         }
417 
418         // all action parameters should be submitted with action
419         Map<String, String> submitData = new HashMap<String, String>();
420         for (String key : actionParameters.keySet()) {
421             String parameterPath = key;
422             if (!key.equals(UifConstants.CONTROLLER_METHOD_DISPATCH_PARAMETER_NAME)) {
423                 parameterPath = UifPropertyPaths.ACTION_PARAMETERS + "[" + key + "]";
424             }
425             submitData.put(parameterPath, actionParameters.get(key));
426         }
427 
428         for (String key : additionalSubmitData.keySet()) {
429             submitData.put(key, additionalSubmitData.get(key));
430         }
431 
432         // if focus id not set default to focus on action
433         if (focusOnIdAfterSubmit.equalsIgnoreCase(UifConstants.Order.NEXT_INPUT.toString())) {
434             focusOnIdAfterSubmit = UifConstants.Order.NEXT_INPUT.toString() + ":" + this.getId();
435         }
436 
437         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.FOCUS_ID,
438                 focusOnIdAfterSubmit);
439 
440         if (StringUtils.isNotBlank(jumpToIdAfterSubmit)) {
441             addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.ActionDataAttributes.JUMP_TO_ID,
442                     jumpToIdAfterSubmit);
443         } else if (StringUtils.isNotBlank(jumpToNameAfterSubmit)) {
444             addActionDataSettingsValue(actionDataAttributes, dataDefaults,
445                     UifConstants.ActionDataAttributes.JUMP_TO_NAME, jumpToNameAfterSubmit);
446         }
447 
448         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.DataAttributes.SUBMIT_DATA,
449                 ScriptUtils.toJSON(submitData));
450 
451         // build final onclick script
452         String onClickScript = this.getOnClickScript();
453         if (StringUtils.isNotBlank(actionScript)) {
454             onClickScript = ScriptUtils.appendScript(onClickScript, actionScript);
455         } else {
456             onClickScript = ScriptUtils.appendScript(onClickScript, "actionInvokeHandler(this);");
457         }
458 
459         //stop action if the action is disabled
460         if (disabled) {
461             this.addStyleClass("disabled");
462             this.setSkipInTabOrder(true);
463         }
464 
465         // on click script becomes a data attribute for use in a global handler on the client
466         addActionDataSettingsValue(actionDataAttributes, dataDefaults, UifConstants.DataAttributes.ONCLICK,
467                 KRADUtils.convertToHTMLAttributeSafeString(onClickScript));
468 
469         if (!actionDataAttributes.isEmpty()) {
470             this.getDataAttributes().putAll(actionDataAttributes);
471         }
472 
473         this.addDataAttribute(UifConstants.DataAttributes.ROLE, UifConstants.RoleTypes.ACTION);
474 
475         // add data attribute if this is the primary action
476         if (this.isDefaultEnterKeyAction()) {
477             this.addDataAttribute(UifConstants.DataAttributes.DEFAULT_ENTER_KEY_ACTION,
478                     Boolean.toString(this.isDefaultEnterKeyAction()));
479         }
480     }
481 
482     /**
483      * Adds the value passed to the valueMap with the key specified, if the value does not match the
484      * value which already exists in defaults (to avoid having to write out extra data that can later
485      * be derived from the defaults in the js client-side).
486      *
487      * @param valueMap the data map being constructed
488      * @param defaults defaults for validation messages
489      * @param key the variable name being added
490      * @param value the value set on this object as a String equivalent
491      */
492     protected void addActionDataSettingsValue(Map<String, String> valueMap, Map<String, String> defaults, String key,
493             String value) {
494         if (StringUtils.isBlank(value)) {
495             return;
496         }
497 
498         String defaultValue = defaults.get(key);
499         if (defaultValue == null || !value.equals(defaultValue)) {
500             valueMap.put(key, value);
501         }
502     }
503 
504     /**
505      * Name of the method that should be called when the action is selected
506      *
507      * <p>
508      * For a server side call (clientSideCall is false), gives the name of the
509      * method in the mapped controller that should be invoked when the action is
510      * selected. For client side calls gives the name of the script function
511      * that should be invoked when the action is selected
512      * </p>
513      *
514      * @return name of method to call
515      */
516     @BeanTagAttribute
517     public String getMethodToCall() {
518         return this.methodToCall;
519     }
520 
521     /**
522      * Setter for the actions method to call.
523      *
524      * @param methodToCall method to call
525      */
526     public void setMethodToCall(String methodToCall) {
527         this.methodToCall = methodToCall;
528     }
529 
530     /**
531      * Label text for the action
532      *
533      * <p>
534      * The label text is used by the template renderers to give a human readable
535      * label for the action. For buttons this generally is the button text,
536      * while for an action link it would be the links displayed text
537      * </p>
538      *
539      * @return label for action
540      */
541     @BeanTagAttribute
542     public String getActionLabel() {
543         return this.actionLabel;
544     }
545 
546     /**
547      * Setter for the actions label.
548      *
549      * @param actionLabel action label
550      */
551     public void setActionLabel(String actionLabel) {
552         this.actionLabel = actionLabel;
553     }
554 
555     /**
556      * When true, a span will be rendered around the actionLabel text.
557      *
558      * @return true if rendering a span around actionLabel, false otherwise
559      */
560     @BeanTagAttribute
561     public boolean isRenderInnerTextSpan() {
562         return renderInnerTextSpan;
563     }
564 
565     /**
566      * Setter for {@link org.kuali.rice.krad.uif.element.Action#isRenderInnerTextSpan()}.
567      *
568      * @param renderInnerTextSpan property value
569      */
570     public void setRenderInnerTextSpan(boolean renderInnerTextSpan) {
571         this.renderInnerTextSpan = renderInnerTextSpan;
572     }
573 
574     /**
575      * Image to use for the action
576      *
577      * <p>
578      * When the action image component is set (and render is true) the image will be
579      * used to present the action as opposed to the default (input submit). For
580      * action link templates the image is used for the link instead of the
581      * action link text
582      * </p>
583      *
584      * @return action image
585      */
586     @BeanTagAttribute
587     public Image getActionImage() {
588         return this.actionImage;
589     }
590 
591     /**
592      * Setter for the action image field.
593      *
594      * @param actionImage action image
595      */
596     public void setActionImage(Image actionImage) {
597         this.actionImage = actionImage;
598     }
599 
600     /**
601      * The css class (some which exist in bootstrap css) to use to render an icon for this action.
602      *
603      * @return the icon css class
604      */
605     @BeanTagAttribute
606     public String getIconClass() {
607         return iconClass;
608     }
609 
610     /**
611      * Setter for {@link org.kuali.rice.krad.uif.element.Action#getIconClass()}.
612      *
613      * @param iconClass property value
614      */
615     public void setIconClass(String iconClass) {
616         this.iconClass = iconClass;
617     }
618 
619     /**
620      * For an <code>Action</code> that is part of a
621      * <code>NavigationGroup</code>, the navigate to page id can be set to
622      * configure the page that should be navigated to when the action is
623      * selected.
624      *
625      * <p>
626      * Support exists in the <code>UifControllerBase</code> for handling
627      * navigation between pages.
628      * </p>
629      *
630      * @return id of page that should be rendered when the action item is
631      *         selected
632      */
633     @BeanTagAttribute
634     public String getNavigateToPageId() {
635         return this.navigateToPageId;
636     }
637 
638     /**
639      * Setter for {@link #getNavigateToPageId()}.
640      *
641      * @param navigateToPageId property value
642      */
643     public void setNavigateToPageId(String navigateToPageId) {
644         this.navigateToPageId = navigateToPageId;
645     }
646 
647     /**
648      * Name of the event that will be set when the action is invoked
649      *
650      * <p>
651      * Action events can be looked at by the view or components in order to render differently depending on
652      * the action requested.
653      * </p>
654      *
655      * @return action event name
656      * @see org.kuali.rice.krad.uif.UifConstants.ActionEvents
657      */
658     @BeanTagAttribute
659     public String getActionEvent() {
660         return actionEvent;
661     }
662 
663     /**
664      * Setter for {@link #getActionEvent()}.
665      *
666      * @param actionEvent property value
667      */
668     public void setActionEvent(String actionEvent) {
669         this.actionEvent = actionEvent;
670     }
671 
672     /**
673      * Map of additional data that will be posted when the action is invoked.
674      *
675      * <p>
676      * Each entry in this map will be sent as a request parameter when the action is chosen. Note this in
677      * addition to the form data that is sent. For example, suppose the model contained a property named
678      * number and a boolean named showActive, we can send values for this properties by adding the following
679      * entries to this map:
680      * {'number':'a13', 'showActive', 'true'}
681      * </p>
682      *
683      * <p>
684      * The additionalSubmitData map is different from the actionParameters map. All name/value pairs given as
685      * actionParameters populated the form map actionParameters. While name/value pair given in additionalSubmitData
686      * populate different form (model) properties.
687      * </p>
688      *
689      * @return additional key/value pairs to submit
690      */
691     @BeanTagAttribute
692     public Map<String, String> getAdditionalSubmitData() {
693         return additionalSubmitData;
694     }
695 
696     /**
697      * Setter for map holding additional data to post.
698      *
699      * @param additionalSubmitData property value
700      */
701     public void setAdditionalSubmitData(Map<String, String> additionalSubmitData) {
702         this.additionalSubmitData = additionalSubmitData;
703     }
704 
705     /**
706      * Parameters that should be sent when the action is invoked
707      *
708      * <p>
709      * Action renderer will decide how the parameters are sent for the action
710      * (via script generated hiddens, or script parameters, ...)
711      * </p>
712      *
713      * <p>
714      * Can be set by other components such as the <code>CollectionGroup</code>
715      * to provide the context the action is in (such as the collection name and
716      * line the action applies to)
717      * </p>
718      *
719      * @return action parameters
720      */
721     @BeanTagAttribute
722     public Map<String, String> getActionParameters() {
723         return this.actionParameters;
724     }
725 
726     /**
727      * Setter for {@link #getActionParameters()}.
728      *
729      * @param actionParameters property value
730      */
731     public void setActionParameters(Map<String, String> actionParameters) {
732         this.actionParameters = actionParameters;
733     }
734 
735     /**
736      * Convenience method to add a parameter to the action parameters Map.
737      *
738      * @param parameterName name of parameter to add
739      * @param parameterValue value of parameter to add
740      */
741     public void addActionParameter(String parameterName, String parameterValue) {
742         if (actionParameters == null) {
743             this.actionParameters = new HashMap<String, String>();
744         }
745 
746         this.actionParameters.put(parameterName, parameterValue);
747     }
748 
749     /**
750      * Gets an action parameter by name.
751      *
752      * @param parameterName parameter name
753      * @return action parameter
754      */
755     public String getActionParameter(String parameterName) {
756         return this.actionParameters.get(parameterName);
757     }
758 
759     /**
760      * Action Security object that indicates what authorization (permissions) exist for the action.
761      *
762      * @return ActionSecurity instance
763      */
764     public ActionSecurity getActionSecurity() {
765         return (ActionSecurity) super.getComponentSecurity();
766     }
767 
768     /**
769      * Override to assert a {@link ActionSecurity} instance is set.
770      *
771      * @param componentSecurity instance of ActionSecurity
772      */
773     @Override
774     public void setComponentSecurity(ComponentSecurity componentSecurity) {
775         if ((componentSecurity != null) && !(componentSecurity instanceof ActionSecurity)) {
776             throw new RiceRuntimeException("Component security for Action should be instance of ActionSecurity");
777         }
778 
779         super.setComponentSecurity(componentSecurity);
780     }
781 
782     /**
783      * {@inheritDoc}
784      */
785     @Override
786     protected void initializeComponentSecurity() {
787         if (getComponentSecurity() == null) {
788             setComponentSecurity(KRADUtils.createNewObjectFromClass(ActionSecurity.class));
789         }
790     }
791 
792     /**
793      * Indicates whether or not to perform action auth.
794      *
795      * @return true to perform action auth
796      */
797     @BeanTagAttribute
798     public boolean isPerformActionAuthz() {
799         initializeComponentSecurity();
800 
801         return getActionSecurity().isPerformActionAuthz();
802     }
803 
804     /**
805      * Setter for {@link #isPerformActionAuthz()}.
806      *
807      * @param performActionAuthz property value
808      */
809     public void setPerformActionAuthz(boolean performActionAuthz) {
810         initializeComponentSecurity();
811 
812         getActionSecurity().setPerformActionAuthz(performActionAuthz);
813     }
814 
815     /**
816      * Indicates whether or not to perform line action auth.
817      *
818      * @return true to perform line action auth
819      */
820     @BeanTagAttribute
821     public boolean isPerformLineActionAuthz() {
822         initializeComponentSecurity();
823 
824         return getActionSecurity().isPerformLineActionAuthz();
825     }
826 
827     /**
828      * Setter for {@link #isPerformActionAuthz()}.
829      *
830      * @param performLineActionAuthz property value
831      */
832     public void setPerformLineActionAuthz(boolean performLineActionAuthz) {
833         initializeComponentSecurity();
834 
835         getActionSecurity().setPerformLineActionAuthz(performLineActionAuthz);
836     }
837 
838     /**
839      * Gets the id to jump to after submit.
840      *
841      * @return the jumpToIdAfterSubmit
842      */
843     @BeanTagAttribute
844     public String getJumpToIdAfterSubmit() {
845         return this.jumpToIdAfterSubmit;
846     }
847 
848     /**
849      * The id to jump to in the next page, the element with this id will be
850      * jumped to automatically when the new page is retrieved after a submit.
851      * Using "TOP" or "BOTTOM" will jump to the top or the bottom of the
852      * resulting page. Passing in nothing for both jumpToIdAfterSubmit and
853      * jumpToNameAfterSubmit will result in this Action being jumped to by
854      * default if it is present on the new page. WARNING: jumpToIdAfterSubmit
855      * always takes precedence over jumpToNameAfterSubmit, if set.
856      *
857      * @param jumpToIdAfterSubmit the jumpToIdAfterSubmit to set
858      */
859     public void setJumpToIdAfterSubmit(String jumpToIdAfterSubmit) {
860         this.jumpToIdAfterSubmit = jumpToIdAfterSubmit;
861     }
862 
863     /**
864      * The name to jump to in the next page, the element with this name will be
865      * jumped to automatically when the new page is retrieved after a submit.
866      * Passing in nothing for both jumpToIdAfterSubmit and jumpToNameAfterSubmit
867      * will result in this Action being jumped to by default if it is
868      * present on the new page. WARNING: jumpToIdAfterSubmit always takes
869      * precedence over jumpToNameAfterSubmit, if set.
870      *
871      * @return the jumpToNameAfterSubmit
872      */
873     @BeanTagAttribute
874     public String getJumpToNameAfterSubmit() {
875         return this.jumpToNameAfterSubmit;
876     }
877 
878     /**
879      * Setter for {@link #getJumpToIdAfterSubmit()}.
880      *
881      * @param jumpToNameAfterSubmit the jumpToNameAfterSubmit to set
882      */
883     public void setJumpToNameAfterSubmit(String jumpToNameAfterSubmit) {
884         this.jumpToNameAfterSubmit = jumpToNameAfterSubmit;
885     }
886 
887     /**
888      * The element to place focus on in the new page after the new page
889      * is retrieved.
890      *
891      * <p>The following are allowed:
892      * <ul>
893      * <li>A valid element id</li>
894      * <li>"FIRST" will focus on the first visible input element on the form</li>
895      * <li>"SELF" will result in this Action being focused (action bean defaults to "SELF")</li>
896      * <li>"LINE_FIRST" will result in the first input of the collection line to be focused (if available)</li>
897      * <li>"NEXT_INPUT" will result in the next available input that exists after this Action to be focused
898      * (only if this action still exists on the page)</li>
899      * </ul>
900      * </p>
901      *
902      * @return the focusOnAfterSubmit
903      */
904     @BeanTagAttribute
905     public String getFocusOnIdAfterSubmit() {
906         return this.focusOnIdAfterSubmit;
907     }
908 
909     /**
910      * Setter for {@link #getFocusOnIdAfterSubmit()}.
911      *
912      * @param focusOnIdAfterSubmit the focusOnAfterSubmit to set
913      */
914     public void setFocusOnIdAfterSubmit(String focusOnIdAfterSubmit) {
915         this.focusOnIdAfterSubmit = focusOnIdAfterSubmit;
916     }
917 
918     /**
919      * Indicates whether the form data should be validated on the client side.
920      *
921      * @return true if validation should occur, false otherwise
922      */
923     @BeanTagAttribute
924     public boolean isPerformClientSideValidation() {
925         return this.performClientSideValidation;
926     }
927 
928     /**
929      * Setter for the client side validation flag.
930      *
931      * @param performClientSideValidation property value
932      */
933     public void setPerformClientSideValidation(boolean performClientSideValidation) {
934         this.performClientSideValidation = performClientSideValidation;
935     }
936 
937     /**
938      * Client side javascript to be executed when this actionField is clicked.
939      *
940      * <p>
941      * This overrides the default action for this Action so the method
942      * called must explicitly submit, navigate, etc. through js, if necessary.
943      * In addition, this js occurs AFTER onClickScripts set on this field, it
944      * will be the last script executed by the click event. Sidenote: This js is
945      * always called after hidden actionParameters and methodToCall methods are
946      * written by the js to the html form.
947      * </p>
948      *
949      * @return the actionScript
950      */
951     @BeanTagAttribute
952     public String getActionScript() {
953         return this.actionScript;
954     }
955 
956     /**
957      * Setter for {@link #getActionScript()}.
958      *
959      * @param actionScript the actionScript to set
960      */
961     public void setActionScript(String actionScript) {
962         if (StringUtils.isNotBlank(actionScript) && !StringUtils.endsWith(actionScript, ";")) {
963             actionScript = actionScript + ";";
964         }
965 
966         this.actionScript = actionScript;
967     }
968 
969     /**
970      * Url to open when the action item is selected
971      *
972      * <p>
973      * This makes the action behave like a standard link. Instead of posting the form, the configured URL will
974      * simply be opened (using window.open). For using standard post actions these does not need to be configured.
975      * </p>
976      *
977      * @return Url info instance for the configuration action link
978      */
979     @BeanTagAttribute
980     public UrlInfo getActionUrl() {
981         return actionUrl;
982     }
983 
984     /**
985      * Setter for {@link #getActionUrl()}.
986      *
987      * @param actionUrl property value
988      */
989     public void setActionUrl(UrlInfo actionUrl) {
990         this.actionUrl = actionUrl;
991     }
992 
993     /**
994      * Setter for {@link #isPerformDirtyValidation()}.
995      *
996      * @param performDirtyValidation the blockValidateDirty to set
997      */
998     public void setPerformDirtyValidation(boolean performDirtyValidation) {
999         this.performDirtyValidation = performDirtyValidation;
1000     }
1001 
1002     /**
1003      * Indicates whether or not to perform dirty validation.
1004      *
1005      * @return true to perform dirty validation
1006      */
1007     @BeanTagAttribute
1008     public boolean isPerformDirtyValidation() {
1009         return performDirtyValidation;
1010     }
1011 
1012     /**
1013      * True to make this action clear the dirty flag before submitting.
1014      *
1015      * <p>This will clear both the dirtyForm flag on the form and the count of fields considered dirty on the
1016      * client-side.  This will only be performed if this action is a request based action.</p>
1017      *
1018      * @return true if the dirty
1019      */
1020     @BeanTagAttribute
1021     public boolean isClearDirtyOnAction() {
1022         return clearDirtyOnAction;
1023     }
1024 
1025     /**
1026      * Setter for {@link #isClearDirtyOnAction()}.
1027      *
1028      * @param clearDirtyOnAction property value
1029      */
1030     public void setClearDirtyOnAction(boolean clearDirtyOnAction) {
1031         this.clearDirtyOnAction = clearDirtyOnAction;
1032     }
1033 
1034     /**
1035      * When true, this action will mark the form dirty by incrementing the dirty field count, but if this action
1036      * refreshes the entire view this will be lost (most actions only refresh the page)
1037      *
1038      * <p>This will increase count of fields considered dirty on the
1039      * client-side by 1.  This will only be performed if this action is a request based action.</p>
1040      *
1041      * @return true if this action is considered dirty, false otherwise
1042      */
1043     @BeanTagAttribute
1044     public boolean isDirtyOnAction() {
1045         return dirtyOnAction;
1046     }
1047 
1048     /**
1049      * Set to true, if this action is considered one that changes the form's data (makes the form dirty).
1050      *
1051      * @param dirtyOnAction property value
1052      */
1053     public void setDirtyOnAction(boolean dirtyOnAction) {
1054         this.dirtyOnAction = dirtyOnAction;
1055     }
1056 
1057     /**
1058      * Indicates whether the action (input or button) is disabled (doesn't allow interaction).
1059      *
1060      * @return true if the action field is disabled, false if not
1061      */
1062     @BeanTagAttribute
1063     public boolean isDisabled() {
1064         return disabled;
1065     }
1066 
1067     /**
1068      * Setter for the disabled indicator.
1069      *
1070      * @param disabled property value
1071      */
1072     public void setDisabled(boolean disabled) {
1073         this.disabled = disabled;
1074     }
1075 
1076     /**
1077      * If the action field is disabled, gives a reason for why which will be displayed as a tooltip
1078      * on the action field (button).
1079      *
1080      * @return disabled reason text
1081      * @see #isDisabled()
1082      */
1083     @BeanTagAttribute
1084     public String getDisabledReason() {
1085         return disabledReason;
1086     }
1087 
1088     /**
1089      * Setter for the disabled reason text.
1090      *
1091      * @param disabledReason property value
1092      */
1093     public void setDisabledReason(String disabledReason) {
1094         this.disabledReason = disabledReason;
1095     }
1096 
1097     /**
1098      * Gets the action image placement.
1099      *
1100      * @return action image placement
1101      */
1102     @BeanTagAttribute
1103     public String getActionImagePlacement() {
1104         return actionImagePlacement;
1105     }
1106 
1107     /**
1108      * Set to TOP, BOTTOM, LEFT, RIGHT to position image at that location within the button.
1109      * For the subclass ActionLinkField only LEFT and RIGHT are allowed.  When set to blank/null/IMAGE_ONLY, the image
1110      * itself will be the Action, if no value is set the default is ALWAYS LEFT, you must explicitly set
1111      * blank/null/IMAGE_ONLY to use ONLY the image as the Action.
1112      *
1113      * @param actionImagePlacement action image placement indicator
1114      */
1115     public void setActionImagePlacement(String actionImagePlacement) {
1116         this.actionImagePlacement = actionImagePlacement;
1117     }
1118 
1119     /**
1120      * Gets the action icon placement.
1121      *
1122      * @return action icon placement
1123      */
1124     @BeanTagAttribute
1125     public String getActionIconPlacement() {
1126         return actionIconPlacement;
1127     }
1128 
1129     /**
1130      * Setter for {@link #getActionIconPlacement()}.
1131      *
1132      * @param actionIconPlacement property value
1133      */
1134     public void setActionIconPlacement(String actionIconPlacement) {
1135         this.actionIconPlacement = actionIconPlacement;
1136     }
1137 
1138     /**
1139      * Gets the script which needs to be invoked before the form is submitted
1140      *
1141      * <p>
1142      * The preSubmitCall can carry out custom logic for the action before the submit occurs. The value should
1143      * be given as one or more lines of script and should return a boolean. If false is returned from the call,
1144      * the submit is not carried out. Furthermore, the preSubmitCall can refer to the request object through the
1145      * variable 'kradRequest' or 'this'. This gives full access over the request for doing such things as
1146      * adding additional data
1147      * </p>
1148      *
1149      * <p>
1150      * Examples 'return doFunction(kradRequest);', 'var valid=true;return valid;'
1151      * </p>
1152      *
1153      * <p>
1154      * The preSubmit call will be invoked both for ajax and non-ajax submits
1155      * </p>
1156      *
1157      * @return script text that will be invoked before form submission
1158      */
1159     @BeanTagAttribute
1160     public String getPreSubmitCall() {
1161         return preSubmitCall;
1162     }
1163 
1164     /**
1165      * Setter for {@link #getPreSubmitCall()}.
1166      *
1167      * @param preSubmitCall property value
1168      */
1169     public void setPreSubmitCall(String preSubmitCall) {
1170         this.preSubmitCall = preSubmitCall;
1171     }
1172 
1173     /**
1174      * Text to display as a confirmation of the action.
1175      *
1176      * <p>When this text is displayed the user will receive a confirmation when the action is taken. The user
1177      * can then cancel the action, or continue. If set, {@link Action#getConfirmationDialog()} will be used
1178      * to build the dialog. Otherwise, the dialog is created dynamically on the client.</p>
1179      *
1180      * @return text to display in a confirmation for the action
1181      */
1182     public String getConfirmationPromptText() {
1183         return confirmationPromptText;
1184     }
1185 
1186     /**
1187      * @see Action#getConfirmationPromptText()
1188      */
1189     public void setConfirmationPromptText(String confirmationPromptText) {
1190         this.confirmationPromptText = confirmationPromptText;
1191     }
1192 
1193     /**
1194      * Dialog to use an a confirmation for the action.
1195      *
1196      * <p>For custom confirmation dialogs this can be set to any valid dialog group. It is expected that the
1197      * dialog have at least one action with the dialog response of 'true' to continue the action.</p>
1198      *
1199      * @return dialog group instance to use an a confirmation
1200      */
1201     public DialogGroup getConfirmationDialog() {
1202         return confirmationDialog;
1203     }
1204 
1205     /**
1206      * @see Action#getConfirmationDialog()
1207      */
1208     public void setConfirmationDialog(DialogGroup confirmationDialog) {
1209         this.confirmationDialog = confirmationDialog;
1210     }
1211 
1212     /**
1213      * If the action is within a {@link org.kuali.rice.krad.uif.container.DialogGroup} it can be configured to
1214      * dismiss the dialog using this property.
1215      *
1216      * <p>A dialog can be dismissed at various points of the action using the values:
1217      *    IMMEDIATE - dismiss dialog right away (and do nothig further)
1218      *    PRESUBMIT - run the action presubmit (which can include validation), if successful close the dialog and
1219      *                do nothing further
1220      *    REQUEST - carry out the action request as usual and dismiss the dialog when the server request is made
1221      * </p>
1222      *
1223      * <p>Note the id for the dialog that will be dismissed is automatically associated with the action when
1224      * the dialog is shown.</p>
1225      *
1226      * @return String option for dismissing a dialog
1227      */
1228     public String getDialogDismissOption() {
1229         return dialogDismissOption;
1230     }
1231 
1232     /**
1233      * @see Action#getDialogDismissOption()
1234      */
1235     public void setDialogDismissOption(String dialogDismissOption) {
1236         this.dialogDismissOption = dialogDismissOption;
1237     }
1238 
1239     /**
1240      * If the action is within a {@link org.kuali.rice.krad.uif.container.DialogGroup} it can be configured to
1241      * return a response using this property.
1242      *
1243      * <p>Dialogs can be used to get a response from a user, either a simple confirmation (true or false), or to
1244      * choice from a list of options. The responses for the dialog are created with action components. The property
1245      * specifies the action value that should be returned (when chosen) to the dialog response handlers. For example,
1246      * in a simple confirmation one action will have a dialog response 'false', and the other will have a dialog
1247      * response 'true'.</p>
1248      *
1249      * @return String dialog response value
1250      */
1251     public String getDialogResponse() {
1252         return dialogResponse;
1253     }
1254 
1255     /**
1256      * @see Action#getDialogResponse()
1257      */
1258     public void setDialogResponse(String dialogResponse) {
1259         this.dialogResponse = dialogResponse;
1260     }
1261 
1262     /**
1263      * When this property is set to true it will submit the form using Ajax instead of the browser submit. Will default
1264      * to updating the page contents
1265      *
1266      * @return boolean
1267      */
1268     @BeanTagAttribute
1269     public boolean isAjaxSubmit() {
1270         return ajaxSubmit;
1271     }
1272 
1273     /**
1274      * Setter for {@link #isAjaxSubmit()}.
1275      *
1276      * @param ajaxSubmit property value
1277      */
1278     public void setAjaxSubmit(boolean ajaxSubmit) {
1279         this.ajaxSubmit = ajaxSubmit;
1280     }
1281 
1282     /**
1283      * Gets the return type for the ajax call
1284      *
1285      * <p>
1286      * The ajax return type indicates how the response content will be handled in the client. Typical
1287      * examples include updating a component, the page, or doing a redirect.
1288      * </p>
1289      *
1290      * @return return type
1291      * @see org.kuali.rice.krad.uif.UifConstants.AjaxReturnTypes
1292      */
1293     @BeanTagAttribute
1294     public String getAjaxReturnType() {
1295         return this.ajaxReturnType;
1296     }
1297 
1298     /**
1299      * Setter for the type of ajax return.
1300      *
1301      * @param ajaxReturnType property value
1302      */
1303     public void setAjaxReturnType(String ajaxReturnType) {
1304         this.ajaxReturnType = ajaxReturnType;
1305     }
1306 
1307     /**
1308      * Indicates if the action response should be displayed in a lightbox.
1309      *
1310      * @return true if response should be rendered in a lightbox, false if not
1311      */
1312     @BeanTagAttribute
1313     public boolean isDisplayResponseInLightBox() {
1314         return StringUtils.equals(this.ajaxReturnType, UifConstants.AjaxReturnTypes.DISPLAYLIGHTBOX.getKey());
1315     }
1316 
1317     /**
1318      * Setter for indicating the response should be rendered in a lightbox.
1319      *
1320      * @param displayResponseInLightBox property value
1321      */
1322     public void setDisplayResponseInLightBox(boolean displayResponseInLightBox) {
1323         if (displayResponseInLightBox) {
1324             this.ajaxReturnType = UifConstants.AjaxReturnTypes.DISPLAYLIGHTBOX.getKey();
1325         }
1326         // if display lightbox is false and it was previously true, set to default of update page
1327         else if (StringUtils.equals(this.ajaxReturnType, UifConstants.AjaxReturnTypes.DISPLAYLIGHTBOX.getKey())) {
1328             this.ajaxReturnType = UifConstants.AjaxReturnTypes.UPDATEPAGE.getKey();
1329         }
1330     }
1331 
1332     /**
1333      * Gets the script which will be invoked on a successful ajax call
1334      *
1335      * <p>
1336      * The successCallback can carry out custom logic after a successful ajax submission has been made. The
1337      * value can contain one or more script statements. In addition, the response contents can be accessed
1338      * through the variable 'responseContents'
1339      * </p>
1340      *
1341      * <p>
1342      * Examples 'handleSuccessfulUpdate(responseContents);'
1343      * </p>
1344      *
1345      * <p>
1346      * The successCallback may only be specified when {@link #isAjaxSubmit()} is true
1347      * </p>
1348      *
1349      * @return script to be executed when the action is successful
1350      */
1351     @BeanTagAttribute
1352     public String getSuccessCallback() {
1353         return successCallback;
1354     }
1355 
1356     /**
1357      * Setter for successCallback.
1358      *
1359      * @param successCallback property value
1360      */
1361     public void setSuccessCallback(String successCallback) {
1362         this.successCallback = successCallback;
1363     }
1364 
1365     /**
1366      * Gets the script which will be invoked when the action fails due to problems in the ajax call or
1367      * the return of an incident report
1368      *
1369      * <p>
1370      * The errorCallback can carry out custom logic after a failed ajax submission. The
1371      * value can contain one or more script statements. In addition, the response contents can be accessed
1372      * through the variable 'responseContents'
1373      * </p>
1374      *
1375      * <p>
1376      * Examples 'handleFailedUpdate(responseContents);'
1377      * </p>
1378      *
1379      * <p>
1380      * The errorCallback may only be specified when {@link #isAjaxSubmit()} is true
1381      * </p>
1382      *
1383      * @return script to be executed when the action is successful
1384      */
1385     @BeanTagAttribute
1386     public String getErrorCallback() {
1387         return errorCallback;
1388     }
1389 
1390     /**
1391      * Setter for {@link #getErrorCallback()}.
1392      *
1393      * @param errorCallback property value
1394      */
1395     public void setErrorCallback(String errorCallback) {
1396         this.errorCallback = errorCallback;
1397     }
1398 
1399     /**
1400      * Id for the component that should be refreshed after the action completes
1401      *
1402      * <p>
1403      * Either refresh id or refresh property name can be set to configure the component that should
1404      * be refreshed after the action completes. If both are blank, the page will be refreshed
1405      * </p>
1406      *
1407      * @return valid component id
1408      */
1409     @BeanTagAttribute
1410     public String getRefreshId() {
1411         return refreshId;
1412     }
1413 
1414     /**
1415      * Setter for the {@link #getRefreshId()}.
1416      *
1417      * @param refreshId property value
1418      */
1419     public void setRefreshId(String refreshId) {
1420         this.refreshId = refreshId;
1421     }
1422 
1423     /**
1424      * Property name for the {@link org.kuali.rice.krad.uif.field.DataField} that should be refreshed after the action
1425      * completes
1426      *
1427      * <p>
1428      * Either refresh id or refresh property name can be set to configure the component that should
1429      * be refreshed after the action completes. If both are blank, the page will be refreshed
1430      * </p>
1431      *
1432      * <p>
1433      * Property name will be adjusted to use the default binding path unless it contains the form prefix
1434      * </p>
1435      *
1436      * @return valid property name with an associated DataField
1437      * @see org.kuali.rice.krad.uif.UifConstants#NO_BIND_ADJUST_PREFIX
1438      */
1439     @BeanTagAttribute
1440     public String getRefreshPropertyName() {
1441         return refreshPropertyName;
1442     }
1443 
1444     /**
1445      * Setter for the property name of the DataField that should be refreshed.
1446      *
1447      * @param refreshPropertyName property value
1448      */
1449     public void setRefreshPropertyName(String refreshPropertyName) {
1450         this.refreshPropertyName = refreshPropertyName;
1451     }
1452 
1453     /**
1454      * Gets the loading message used by action's blockUI.
1455      *
1456      * @return String if String is not null, used in place of loading message
1457      */
1458     @BeanTagAttribute
1459     public String getLoadingMessageText() {
1460         return loadingMessageText;
1461     }
1462 
1463     /**
1464      * When this property is set, it is used in place of the loading message text used by the blockUI.
1465      *
1466      * @param loadingMessageText property value
1467      */
1468     public void setLoadingMessageText(String loadingMessageText) {
1469         this.loadingMessageText = loadingMessageText;
1470     }
1471 
1472     /**
1473      * Indicates whether blocking for the action should be disabled
1474      *
1475      * <p>
1476      * By default when an action is invoked part of the page or the entire window is blocked until
1477      * the action completes. If this property is set to true the blocking will not be displayed.
1478      * </p>
1479      *
1480      * <p>
1481      * Currently if an action returns a file download, this property should be set to true. If not, the blocking
1482      * will never get unblocked (because the page does not get notification a file was downloaded)
1483      * </p>
1484      *
1485      * @return true if blocking should be disabled, false if not
1486      */
1487     @BeanTagAttribute
1488     public boolean isDisableBlocking() {
1489         return disableBlocking;
1490     }
1491 
1492     /**
1493      * Setter for disabling blocking when the action is invoked.
1494      *
1495      * @param disableBlocking property value
1496      */
1497     public void setDisableBlocking(boolean disableBlocking) {
1498         this.disableBlocking = disableBlocking;
1499     }
1500 
1501     /**
1502      * Evaluate the disable condition on controls which disable it on each key up event.
1503      *
1504      * @return true if evaluate on key up, false otherwise
1505      */
1506     @BeanTagAttribute
1507     public boolean isEvaluateDisabledOnKeyUp() {
1508         return evaluateDisabledOnKeyUp;
1509     }
1510 
1511     /**
1512      * Setter for {@link #isEvaluateDisabledOnKeyUp()}.
1513      *
1514      * @param evaluateDisabledOnKeyUp property value
1515      */
1516     public void setEvaluateDisabledOnKeyUp(boolean evaluateDisabledOnKeyUp) {
1517         this.evaluateDisabledOnKeyUp = evaluateDisabledOnKeyUp;
1518     }
1519 
1520     /**
1521      * Evaluate if this action is the default action for a page, view, group, or section.
1522      *
1523      * @return true if this action is default, false otherwise
1524      */
1525     @BeanTagAttribute(name = "defaultEnterKeyAction")
1526     public boolean isDefaultEnterKeyAction() {
1527         return this.defaultEnterKeyAction;
1528     }
1529 
1530     /**
1531      * @see  #isDefaultEnterKeyAction()
1532      */
1533     public void setDefaultEnterKeyAction(boolean defaultEnterKeyAction) {
1534         this.defaultEnterKeyAction = defaultEnterKeyAction;
1535     }
1536 
1537     /**
1538      * Get the disable condition js derived from the springEL, cannot be set.
1539      *
1540      * @return the disableConditionJs javascript to be evaluated
1541      */
1542     public String getDisabledConditionJs() {
1543         return disabledConditionJs;
1544     }
1545 
1546     /**
1547      * Sets the disabled condition javascript.
1548      *
1549      * @param disabledConditionJs property value
1550      */
1551     protected void setDisabledConditionJs(String disabledConditionJs) {
1552         this.disabledConditionJs = disabledConditionJs;
1553     }
1554 
1555     /**
1556      * Gets a list of control names to add handlers to for disable functionality, cannot be set.
1557      *
1558      * @return control names to add handlers to for disable
1559      */
1560     public List<String> getDisabledConditionControlNames() {
1561         return disabledConditionControlNames;
1562     }
1563 
1564     /**
1565      * Set disabled condition control names.
1566      *
1567      * @param disabledConditionControlNames property value
1568      */
1569     public void setDisabledConditionControlNames(List<String> disabledConditionControlNames) {
1570         this.disabledConditionControlNames = disabledConditionControlNames;
1571     }
1572 
1573     /**
1574      * Gets the property names of fields that when changed, will disable this component.
1575      *
1576      * @return the property names to monitor for change to disable this component
1577      */
1578     @BeanTagAttribute
1579     public List<String> getDisabledWhenChangedPropertyNames() {
1580         return disabledWhenChangedPropertyNames;
1581     }
1582 
1583     /**
1584      * Sets the property names of fields that when changed, will disable this component.
1585      *
1586      * @param disabledWhenChangedPropertyNames property value
1587      */
1588     public void setDisabledWhenChangedPropertyNames(List<String> disabledWhenChangedPropertyNames) {
1589         this.disabledWhenChangedPropertyNames = disabledWhenChangedPropertyNames;
1590     }
1591 
1592     /**
1593      * Gets the property names of fields that when changed, will enable this component.
1594      *
1595      * @return the property names to monitor for change to enable this component
1596      */
1597     @BeanTagAttribute
1598     public List<String> getEnabledWhenChangedPropertyNames() {
1599         return enabledWhenChangedPropertyNames;
1600     }
1601 
1602     /**
1603      * Sets the property names of fields that when changed, will enable this component.
1604      *
1605      * @param enabledWhenChangedPropertyNames property value
1606      */
1607     public void setEnabledWhenChangedPropertyNames(List<String> enabledWhenChangedPropertyNames) {
1608         this.enabledWhenChangedPropertyNames = enabledWhenChangedPropertyNames;
1609     }
1610 
1611     /**
1612      * Sets the disabled expression.
1613      *
1614      * @param disabledExpression property value
1615      */
1616     protected void setDisabledExpression(String disabledExpression) {
1617         this.disabledExpression = disabledExpression;
1618     }
1619 
1620     /**
1621      * {@inheritDoc}
1622      */
1623     @Override
1624     public void completeValidation(ValidationTrace tracer) {
1625         tracer.addBean(this);
1626 
1627         // Checks that an action is set
1628         if (getJumpToIdAfterSubmit() != null && getJumpToNameAfterSubmit() != null) {
1629             String currentValues[] = {"jumpToIdAfterSubmit =" + getJumpToIdAfterSubmit(),
1630                     "jumpToNameAfterSubmit =" + getJumpToNameAfterSubmit()};
1631             tracer.createWarning("Only 1 jumpTo property should be set", currentValues);
1632         }
1633         super.completeValidation(tracer.getCopy());
1634     }
1635 }