View Javadoc

1   /**
2    * Copyright 2005-2012 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.container;
17  
18  import org.kuali.rice.core.api.util.KeyValue;
19  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
20  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
21  import org.kuali.rice.krad.datadictionary.parse.BeanTags;
22  import org.kuali.rice.krad.uif.component.Component;
23  import org.kuali.rice.krad.uif.control.MultiValueControl;
24  import org.kuali.rice.krad.uif.field.InputField;
25  import org.kuali.rice.krad.uif.field.MessageField;
26  import org.kuali.rice.krad.uif.view.View;
27  
28  import java.util.ArrayList;
29  import java.util.Collections;
30  import java.util.List;
31  
32  /**
33   * Special type of <code>Group</code> that presents a the content for a modal dialog
34   *
35   * <p>
36   * This type of group will be hidden when the main view is displayed. It will be used as
37   * content inside the LightBox widget when the modal dialog is displayed.
38   * For convenience, this group contains a standard set of components for commonly used modal dialogs
39   * <ul>
40   * <li>a prompt to display in the lightbox</li>
41   * <li>an optional explanation <code>InputField</code> for holding the user's textual response</li>
42   * <li>a set of response options for the user to choose from</li>
43   * </ul>
44   *
45   * <p>
46   * The DialogGroup may also serve as a base class for more complex dialogs.
47   * The default settings for this DialogGroup is to display a prompt message
48   * with two buttons labeled OK and Cancel.
49   * The optional explanation <code>TextAreaControl</code> is hidden by default.
50   * </p>
51   *
52   * <p>
53   * The prompt text, number of user options and their corresponding label are configurable.
54   * The <code>InputField</code> for the explanation is <code>TextAreaControl</code> by default.
55   * It may be configured to other types of InputFields.
56   * The Component for ResponseInputField is a <code>HorizontalCheckboxGroup</code> by default.
57   * JQuery styling is then used to style the checkboxes as buttons. The ResponseInputField may
58   * be configured to other <code>InputField</code> types.
59   * </p>
60   *
61   * @author Kuali Rice Team (rice.collab@kuali.org)
62   */
63  @BeanTags({@BeanTag(name = "dialogGroup", parent = "Uif-DialogGroup"),
64          @BeanTag(name = "sensitiveData-dialogGroup", parent = "Uif-SensitiveData-DialogGroup"),
65          @BeanTag(name = "ok-cancel-dialogGroup", parent = "Uif-OK-Cancel-DialogGroup"),
66          @BeanTag(name = "yes-no-dialogGroup", parent = "Uif-Yes-No-DialogGroup"),
67          @BeanTag(name = "true-false-dialogGroup", parent = "Uif-True-False-DialogGroup"),
68          @BeanTag(name = "checkbox-dialogGroup", parent = "Uif-Checkbox-DialogGroup"),
69          @BeanTag(name = "radioButton-dialogGroup", parent = "Uif-RadioButton-DialogGroup")})
70  public class DialogGroup extends Group {
71      private static final long serialVersionUID = 1L;
72  
73      private String promptText;
74      private List<KeyValue> availableResponses;
75  
76      private MessageField prompt;
77      private InputField explanation;
78      private InputField responseInputField;
79  
80      private boolean reverseButtonOrder = false;
81      private boolean displayExplanation = false;
82      private boolean useAjaxCallForContent = false;
83  
84      public DialogGroup() {
85          super();
86      }
87  
88      /**
89       * @see org.kuali.rice.krad.uif.component.ComponentBase#getComponentsForLifecycle()
90       */
91      @Override
92      public List<Component> getComponentsForLifecycle() {
93          List<Component> components = super.getComponentsForLifecycle();
94  
95          components.add(prompt);
96          components.add(explanation);
97          components.add(responseInputField);
98  
99          return components;
100     }
101 
102     /**
103      * The following actions are performed:
104      *
105      * <ul>
106      * <li>Move custom dialogGroup properties prompt, explanation, and responseInputField into items collection</li>
107      * </ul>
108      *
109      * @see org.kuali.rice.krad.uif.component.ComponentBase#performInitialization(org.kuali.rice.krad.uif.view.View,
110      *      java.lang.Object)
111      */
112     @Override
113     public void performInitialization(View view, Object model) {
114         super.performInitialization(view, model);
115 
116             // move dialogGroup custom properties into the items property.
117             // where they will be rendered by group.jsp
118             List<Component> myItems = new ArrayList<Component>();
119             List<? extends Component> items = getItems();
120 
121             // do not add the custom properties if they are already present
122             if(!(items.contains(prompt))){
123                 myItems.add(prompt);
124             }
125 
126             if(!(items.contains(explanation))){
127                 myItems.add(explanation);
128             }
129 
130             myItems.addAll(getItems());
131 
132             if(!(items.contains(responseInputField))){
133                 myItems.add(responseInputField);
134             }
135 
136             this.setItems(myItems);
137 
138     }
139 
140     /**
141      * Performs the final phase of the component lifecycle.
142      *
143      * <p>For this DialogGroup component, perform the following:
144      * <ul>
145      * <li>set the promptText in the message</li>
146      * <li>sets whether to render explanation field</li>
147      * <li>set the options for the checkbox control to the availableResponses KeyValue property of
148      * this dialogGroup</li>
149      * <li>orders response buttons</li>
150      * </ul>
151      * </p>
152      *
153      * @param view - view instance that should be finalized for rendering
154      * @param model - top level object containing the data
155      * @param parent - parent component
156      */
157     @Override
158     public void performFinalize(View view, Object model, Component parent) {
159         super.performFinalize(view, model, parent);
160 
161         // if ajax, just render a placeholder
162         if (useAjaxCallForContent) {
163             setProgressiveRenderViaAJAX(useAjaxCallForContent);
164             setProgressiveRender("");
165             setRender(false);
166         }
167         // set the messageTest to the promptText
168         prompt.setMessageText(promptText);
169 
170         // hide or show explanation
171         explanation.setRender(displayExplanation);
172 
173         // add options to checkbox
174         if (responseInputField.getControl() != null && responseInputField.getControl() instanceof MultiValueControl) {
175             MultiValueControl multiValueControl = (MultiValueControl) responseInputField.getControl();
176             if (reverseButtonOrder) {
177                 // reverse the button order (without changing original list)
178                 List<KeyValue> buttonList = new ArrayList<KeyValue>(availableResponses);
179                 Collections.reverse(buttonList);
180                 multiValueControl.setOptions(buttonList);
181             } else {
182                 multiValueControl.setOptions(availableResponses);
183             }
184         }
185     }
186 
187     // Getters and Setters
188 
189     /**
190      * Returns the text to be displayed as the prompt or main message in this simple dialog
191      *
192      * @return String containing the prompt text
193      */
194 
195     @BeanTagAttribute(name = "promptText")
196     public String getPromptText() {
197         return promptText;
198     }
199 
200     /**
201      * Sets the text String to display as the main message in this dialog
202      *
203      * @param promptText - the String to be displayed as the main message
204      */
205     public void setPromptText(String promptText) {
206         this.promptText = promptText;
207     }
208 
209     /**
210      * Retrieves the Message element for this dialog
211      *
212      * @return Message - the text element containing the message string
213      */
214     @BeanTagAttribute(name = "prompt", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
215     public MessageField getPrompt() {
216         return prompt;
217     }
218 
219     /**
220      * Sets the prompt Message for this dialog
221      *
222      * @param prompt - The Message element for this dialog
223      */
224     public void setPrompt(MessageField prompt) {
225         this.prompt = prompt;
226     }
227 
228     /**
229      * Retrieves the explanation InputField used to gather user input text from the dialog
230      *
231      * <p>
232      * By default, the control for this input is configured as a TextAreaControl. It may be configured for
233      * other types of input fields.
234      * </p>
235      *
236      * @return InputField component
237      */
238     @BeanTagAttribute(name = "explanation", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
239     public InputField getExplanation() {
240         return explanation;
241     }
242 
243     /**
244      * Sets the InputField for gathering user text input
245      *
246      * @param explanation - InputField
247      */
248     public void setExplanation(InputField explanation) {
249         this.explanation = explanation;
250     }
251 
252     /**
253      * determines if the explanation InputField is to be displayed in this dialog
254      *
255      * <p>
256      * False by default.
257      * </p>
258      *
259      * @return boolean - true if this user input is to be rendered, false if not.
260      */
261     @BeanTagAttribute(name = "displayExplanation")
262     public boolean isDisplayExplanation() {
263         return displayExplanation;
264     }
265 
266     /**
267      * Sets whether to display the Explanation InputField on this dialog
268      *
269      * @param displayExplanation - true if explanation control is to be displayed, false if not
270      */
271     public void setDisplayExplanation(boolean displayExplanation) {
272         this.displayExplanation = displayExplanation;
273     }
274 
275     /**
276      * Gets the choices provided for user response.
277      *
278      * <p>
279      * A List of KeyValue pairs for each of the choices provided on this dialog.
280      * </p>
281      *
282      * @return the List of response actions to provide the user.
283      */
284     @BeanTagAttribute(name = "availableResponses", type = BeanTagAttribute.AttributeType.LISTBEAN)
285     public List<KeyValue> getAvailableResponses() {
286         return availableResponses;
287     }
288 
289     /**
290      * Sets the list of user responses to provide on this dialog
291      *
292      * @param availableResponses - a List of KeyValue pairs representing the user response choices
293      */
294     public void setAvailableResponses(List<KeyValue> availableResponses) {
295         this.availableResponses = availableResponses;
296     }
297 
298     /**
299      * Retrieves the InputField containing the choices displayed in this dialog
300      *
301      * <p>
302      * By default, this InputField is configured to be a HorizontalCheckboxControl.
303      * Styling is then used to make the checkboxes appear to be buttons.
304      * The values of the availableResponses List are used as labels for the "buttons".
305      * </p>
306      *
307      * @return InputField component within this dialog
308      */
309     @BeanTagAttribute(name = "responseInputField", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
310     public InputField getResponseInputField() {
311         return responseInputField;
312     }
313 
314     /**
315      * Sets the type of InputField used to display the user choices in this dialog
316      *
317      * @param responseInputField - A component used to display the response choices
318      */
319     public void setResponseInputField(InputField responseInputField) {
320         this.responseInputField = responseInputField;
321     }
322 
323     /**
324      * Determines the positioning order of the choices displayed on this dialog
325      *
326      * <p>
327      * Some page designers like the positive choice on the left and the negative choice on the right.
328      * Others, prefer just the opposite. This allows the order to easily be switched.
329      * </p>
330      *
331      * @return - true if choices left to right
332      *         false if choices right to left
333      */
334     @BeanTagAttribute(name = "reverseButtonOrder")
335     public boolean isReverseButtonOrder() {
336         return reverseButtonOrder;
337     }
338 
339     /**
340      * Sets the display order of the choices displayed on this dialog
341      *
342      * <p>
343      * By default, the choices are displayed left to right
344      * </p>
345      *
346      * @param reverseButtonOrder - true if buttons displayed left to right, false if right to left
347      */
348     public void setReverseButtonOrder(boolean reverseButtonOrder) {
349         this.reverseButtonOrder = reverseButtonOrder;
350     }
351 
352     /**
353      * indicates which approach is used to fill the lightbox content for this dialog.
354      *
355      * <p>
356      * Two techniques are used for filling the content of the lightbox when displaying this dialog.
357      * <ul>
358      * <li>a hidden group on the page is used as content</li>
359      * <li>an ajax call is made to the server to get the content</li>
360      * </ul>
361      * The default approach is to use a hidden form.
362      * </p>
363      *
364      * @return
365      */
366     @BeanTagAttribute(name = "useAjaxCallForContent")
367     public boolean isUseAjaxCallForContent() {
368         return useAjaxCallForContent;
369     }
370 
371     /**
372      * Sets whether the content for the dialog will be filled via ajax call or hidden group
373      *
374      * @param useAjaxCallForContent - boolean set to true if ajax call is used to get content,
375      * false if hidden group is used for content.
376      */
377     public void setUseAjaxCallForContent(boolean useAjaxCallForContent) {
378         this.useAjaxCallForContent = useAjaxCallForContent;
379     }
380 
381 }