View Javadoc

1   /**
2    * Copyright 2005-2013 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.krad.uif.element;
17  
18  import org.apache.commons.lang.StringUtils;
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.datadictionary.validator.ValidationTrace;
23  import org.kuali.rice.krad.datadictionary.validator.Validator;
24  import org.kuali.rice.krad.uif.component.Component;
25  import org.kuali.rice.krad.uif.container.Group;
26  import org.kuali.rice.krad.uif.util.ComponentFactory;
27  import org.kuali.rice.krad.uif.view.View;
28  import org.kuali.rice.krad.util.KRADConstants;
29  
30  import java.util.ArrayList;
31  import java.util.List;
32  
33  /**
34   * Content element that renders a header element and optionally a <code>Group</code> to
35   * present along with the header text
36   *
37   * <p>
38   * Generally the group is used to display content to the right of the header,
39   * such as links for the group or other information
40   * </p>
41   *
42   * @author Kuali Rice Team (rice.collab@kuali.org)
43   */
44  @BeanTags({@BeanTag(name = "header-bean", parent = "Uif-HeaderBase"), @BeanTag(name = "headerOne-bean", parent = "Uif-HeaderOne"),
45          @BeanTag(name = "headerTwo-bean", parent = "Uif-HeaderTwo"),
46          @BeanTag(name = "headerThree-bean", parent = "Uif-HeaderThree"),
47          @BeanTag(name = "headerFour-bean", parent = "Uif-HeaderFour"),
48          @BeanTag(name = "headerFive-bean", parent = "Uif-HeaderFive"),
49          @BeanTag(name = "headerSix-bean", parent = "Uif-HeaderSix"),
50          @BeanTag(name = "pageHeader-bean", parent = "Uif-PageHeader"),
51          @BeanTag(name = "sectionHeader-bean", parent = "Uif-SectionHeader"),
52          @BeanTag(name = "subSectionHeader-bean", parent = "Uif-SubSectionHeader"),
53          @BeanTag(name = "subCollectionHeader-bean", parent = "Uif-SubCollectionHeader"),
54          @BeanTag(name = "editablePageHeader-bean", parent = "Uif-EditablePageHeader"),
55          @BeanTag(name = "readOnlyPageHeader-bean", parent = "Uif-ReadOnlyPageHeader"),
56          @BeanTag(name = "imageCaptionHeader-bean", parent = "Uif-ImageCaptionHeader"),
57          @BeanTag(name = "documentViewHeader-bean", parent = "Uif-DocumentViewHeader"),
58          @BeanTag(name = "lookupPageHeader-bean", parent = "Uif-LookupPageHeader")})
59  public class Header extends ContentElementBase {
60      private static final long serialVersionUID = -6950408292923393244L;
61  
62      private String headerText;
63      private String headerLevel;
64  
65      private String headerTagStyle;
66      private List<String> headerTagCssClasses;
67  
68      private Message richHeaderMessage;
69      private List<Component> inlineComponents;
70  
71      private Group upperGroup;
72      private Group rightGroup;
73      private Group lowerGroup;
74  
75      public Header() {
76          super();
77  
78          headerTagCssClasses = new ArrayList<String>();
79      }
80  
81      /**
82       * Sets up rich message content for the label, if any exists
83       *
84       * @see Component#performApplyModel(org.kuali.rice.krad.uif.view.View, Object,
85       *      org.kuali.rice.krad.uif.component.Component)
86       */
87      @Override
88      public void performApplyModel(View view, Object model, Component parent) {
89          super.performApplyModel(view, model, parent);
90  
91          if (richHeaderMessage == null && headerText != null && headerText.contains(
92                  KRADConstants.MessageParsing.LEFT_TOKEN) && headerText.contains(
93                  KRADConstants.MessageParsing.RIGHT_TOKEN)) {
94              Message message = ComponentFactory.getMessage();
95              view.assignComponentIds(message);
96  
97              message.setMessageText(headerText);
98              message.setInlineComponents(inlineComponents);
99              message.setGenerateSpan(false);
100 
101             view.getViewHelperService().performComponentInitialization(view, model, message);
102 
103             this.setRichHeaderMessage(message);
104         }
105     }
106 
107     /**
108      * The following finalization is performed:
109      *
110      * <ul>
111      * <li>Set render on header group to false if no items are configured</li>
112      * </ul>
113      *
114      * @see org.kuali.rice.krad.uif.component.ComponentBase#performFinalize(org.kuali.rice.krad.uif.view.View,
115      *      java.lang.Object, org.kuali.rice.krad.uif.component.Component)
116      */
117     @Override
118     public void performFinalize(View view, Object model, Component parent) {
119         super.performFinalize(view, model, parent);
120 
121         // don't render header groups if no items were configured
122         if ((getUpperGroup() != null) && (getUpperGroup().getItems().isEmpty())) {
123             getUpperGroup().setRender(false);
124         }
125 
126         if ((getRightGroup() != null) && (getRightGroup().getItems().isEmpty())) {
127             getRightGroup().setRender(false);
128         }
129 
130         if ((getLowerGroup() != null) && (getLowerGroup().getItems().isEmpty())) {
131             getLowerGroup().setRender(false);
132         }
133 
134         //add preset styles to header groups
135         if (getUpperGroup() != null) {
136             getUpperGroup().addStyleClass("uif-header-upperGroup");
137         }
138 
139         if (getRightGroup() != null) {
140             getRightGroup().addStyleClass("uif-header-rightGroup");
141         }
142 
143         if (getLowerGroup() != null) {
144             getLowerGroup().addStyleClass("uif-header-lowerGroup");
145         }
146     }
147 
148     /**
149      * @see org.kuali.rice.krad.uif.component.ComponentBase#getComponentsForLifecycle()
150      */
151     @Override
152     public List<Component> getComponentsForLifecycle() {
153         List<Component> components = super.getComponentsForLifecycle();
154 
155         components.add(richHeaderMessage);
156         components.add(upperGroup);
157         components.add(rightGroup);
158         components.add(lowerGroup);
159 
160         return components;
161     }
162 
163     /**
164      * Text that should be displayed on the header
165      *
166      * @return header text
167      */
168     @BeanTagAttribute(name = "headerText")
169     public String getHeaderText() {
170         return this.headerText;
171     }
172 
173     /**
174      * Setter for the header text
175      *
176      * @param headerText
177      */
178     public void setHeaderText(String headerText) {
179         this.headerText = headerText;
180     }
181 
182     /**
183      * HTML header level (h1 ... h6) that should be applied to the header text
184      *
185      * @return header level
186      */
187     @BeanTagAttribute(name = "headerLevel")
188     public String getHeaderLevel() {
189         return this.headerLevel;
190     }
191 
192     /**
193      * Setter for the header level
194      *
195      * @param headerLevel
196      */
197     public void setHeaderLevel(String headerLevel) {
198         this.headerLevel = headerLevel;
199     }
200 
201     /**
202      * Style classes that should be applied to the header text (h tag)
203      *
204      * <p>
205      * Note the style class given here applies to only the header text. The
206      * style class property inherited from the <code>Component</code> interface
207      * can be used to set the class for the whole field div (which could
208      * include a nested <code>Group</code>)
209      * </p>
210      *
211      * @return list of style classes
212      * @see org.kuali.rice.krad.uif.component.Component#getCssClasses()
213      */
214     @BeanTagAttribute(name = "headerTagCssClasses", type = BeanTagAttribute.AttributeType.LISTVALUE)
215     public List<String> getHeaderTagCssClasses() {
216         return this.headerTagCssClasses;
217     }
218 
219     /**
220      * Setter for the list of classes to apply to the header h tag
221      *
222      * @param headerTagCssClasses
223      */
224     public void setHeaderTagCssClasses(List<String> headerTagCssClasses) {
225         this.headerTagCssClasses = headerTagCssClasses;
226     }
227 
228     /**
229      * Builds the HTML class attribute string by combining the headerStyleClasses list
230      * with a space delimiter
231      *
232      * @return class attribute string
233      */
234     public String getHeaderStyleClassesAsString() {
235         if (headerTagCssClasses != null) {
236             return StringUtils.join(headerTagCssClasses, " ");
237         }
238 
239         return "";
240     }
241 
242     /**
243      * Style that should be applied to the header h tag
244      *
245      * <p>
246      * Note the style given here applies to only the header text. The style
247      * property inherited from the <code>Component</code> interface can be used
248      * to set the style for the whole header div (which could include a nested
249      * <code>Group</code>)
250      * </p>
251      *
252      * @return header style
253      * @see org.kuali.rice.krad.uif.component.Component#getStyle()
254      */
255     @BeanTagAttribute(name = "headerTagStyle")
256     public String getHeaderTagStyle() {
257         return this.headerTagStyle;
258     }
259 
260     /**
261      * Setter for the header h tag style
262      *
263      * @param headerTagStyle
264      */
265     public void setHeaderTagStyle(String headerTagStyle) {
266         this.headerTagStyle = headerTagStyle;
267     }
268 
269     /**
270      * Nested group instance that can be used to render contents above the header text
271      *
272      * <p>
273      * The header group is useful for adding content such as links or actions that is presented with the header
274      * </p>
275      *
276      * @return Group instance
277      */
278     @BeanTagAttribute(name = "upperGroup", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
279     public Group getUpperGroup() {
280         return upperGroup;
281     }
282 
283     /**
284      * Setter for the header group instance that is rendered above the header text
285      *
286      * @param upperGroup
287      */
288     public void setUpperGroup(Group upperGroup) {
289         this.upperGroup = upperGroup;
290     }
291 
292     /**
293      * Nested group instance that can be used to render contents to the right of the header text
294      *
295      * <p>
296      * The header group is useful for adding content such as links or actions that is presented with the header
297      * </p>
298      *
299      * @return Group instance
300      */
301     @BeanTagAttribute(name = "rightGroup", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
302     public Group getRightGroup() {
303         return rightGroup;
304     }
305 
306     /**
307      * Setter for the header group instance that is rendered to the right of the header text
308      *
309      * @param rightGroup
310      */
311     public void setRightGroup(Group rightGroup) {
312         this.rightGroup = rightGroup;
313     }
314 
315     /**
316      * Nested group instance that can be used to render contents below the header text
317      *
318      * <p>
319      * The header group is useful for adding content such as links or actions that is presented with the header
320      * </p>
321      *
322      * @return Group instance
323      */
324     @BeanTagAttribute(name = "lowerGroup", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
325     public Group getLowerGroup() {
326         return lowerGroup;
327     }
328 
329     /**
330      * Setter for the header group instance that is rendered below the header text
331      *
332      * @param lowerGroup
333      */
334     public void setLowerGroup(Group lowerGroup) {
335         this.lowerGroup = lowerGroup;
336     }
337 
338     /**
339      * List of <code>Component</code> instances contained in the lower header group
340      *
341      * <p>
342      * Convenience method for configuration to get the items List from the
343      * lower header group
344      * </p>
345      *
346      * @return List<? extends Component> items
347      */
348     @BeanTagAttribute(name = "items", type = BeanTagAttribute.AttributeType.LISTBEAN)
349     public List<? extends Component> getItems() {
350         if (lowerGroup != null) {
351             return lowerGroup.getItems();
352         }
353 
354         return null;
355     }
356 
357     /**
358      * Setter for the lower group's items
359      *
360      * <p>
361      * Convenience method for configuration to set the items List for the
362      * lower header group
363      * </p>
364      *
365      * @param items
366      */
367     public void setItems(List<? extends Component> items) {
368         if (lowerGroup != null) {
369             lowerGroup.setItems(items);
370         }
371     }
372 
373     /**
374      * Gets the Message that represents the rich message content of the header if headerText is using rich message
375      * tags.
376      * <b>DO NOT set this
377      * property directly unless you need full control over the message structure.</b>
378      *
379      * @return rich message structure, null if no rich message structure
380      */
381     @BeanTagAttribute(name = "richHeaderMessage", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
382     public Message getRichHeaderMessage() {
383         return richHeaderMessage;
384     }
385 
386     /**
387      * Sets the Message that represents the rich message content of the header if headerText is using rich message
388      * tags.
389      * <b>DO
390      * NOT set this
391      * property directly unless you need full control over the message structure.</b>
392      *
393      * @param richHeaderMessage
394      */
395     public void setRichHeaderMessage(Message richHeaderMessage) {
396         this.richHeaderMessage = richHeaderMessage;
397     }
398 
399     /**
400      * Gets the inlineComponents used by index in a Header that has rich message component index tags in its headerText
401      *
402      * @return the Label's inlineComponents
403      */
404     @BeanTagAttribute(name = "inlineComponents", type = BeanTagAttribute.AttributeType.LISTBEAN)
405     public List<Component> getInlineComponents() {
406         return inlineComponents;
407     }
408 
409     /**
410      * Sets the inlineComponents used by index in a Header that has rich message component index tags in its headerText
411      *
412      * @param inlineComponents
413      */
414     public void setInlineComponents(List<Component> inlineComponents) {
415         this.inlineComponents = inlineComponents;
416     }
417 
418     /**
419      * @see org.kuali.rice.krad.uif.component.Component#completeValidation
420      */
421     @Override
422     public void completeValidation(ValidationTrace tracer) {
423         tracer.addBean(this);
424 
425         // Checks that a correct header level is set
426         String headerLevel = getHeaderLevel().toUpperCase();
427         boolean correctHeaderLevel = false;
428         if (headerLevel.compareTo("H1") == 0) {
429             correctHeaderLevel = true;
430         } else if (headerLevel.compareTo("H2") == 0) {
431             correctHeaderLevel = true;
432         } else if (headerLevel.compareTo("H3") == 0) {
433             correctHeaderLevel = true;
434         } else if (headerLevel.compareTo("H4") == 0) {
435             correctHeaderLevel = true;
436         } else if (headerLevel.compareTo("H5") == 0) {
437             correctHeaderLevel = true;
438         } else if (headerLevel.compareTo("H6") == 0) {
439             correctHeaderLevel = true;
440         } else if (headerLevel.compareTo("LABEL") == 0) {
441             correctHeaderLevel = true;
442         }
443         if (!correctHeaderLevel) {
444             String currentValues[] = {"headerLevel =" + getHeaderLevel()};
445             tracer.createError("HeaderLevel must be of values h1, h2, h3, h4, h5, h6, or label", currentValues);
446         }
447 
448         // Checks that header text is set
449         if (getHeaderText() == null) {
450             if (!Validator.checkExpressions(this, "headerText")) {
451                 String currentValues[] = {"headertText =" + getHeaderText()};
452                 tracer.createWarning("HeaderText should be set", currentValues);
453             }
454         }
455 
456         super.completeValidation(tracer.getCopy());
457     }
458 }