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.lookup;
17  
18  import java.util.ArrayList;
19  import java.util.Arrays;
20  import java.util.HashMap;
21  import java.util.List;
22  import java.util.Map;
23  
24  import org.apache.commons.lang.StringUtils;
25  import org.kuali.rice.core.api.mo.common.active.Inactivatable;
26  import org.kuali.rice.kim.api.identity.Person;
27  import org.kuali.rice.krad.datadictionary.AttributeDefinition;
28  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
29  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
30  import org.kuali.rice.krad.datadictionary.uif.UifDictionaryBean;
31  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
32  import org.kuali.rice.krad.uif.UifConstants;
33  import org.kuali.rice.krad.uif.UifConstants.ViewType;
34  import org.kuali.rice.krad.uif.UifParameters;
35  import org.kuali.rice.krad.uif.UifPropertyPaths;
36  import org.kuali.rice.krad.uif.component.Component;
37  import org.kuali.rice.krad.uif.component.RequestParameter;
38  import org.kuali.rice.krad.uif.container.CollectionGroup;
39  import org.kuali.rice.krad.uif.container.Group;
40  import org.kuali.rice.krad.uif.control.Control;
41  import org.kuali.rice.krad.uif.control.FilterableLookupCriteriaControl;
42  import org.kuali.rice.krad.uif.control.FilterableLookupCriteriaControlPostData;
43  import org.kuali.rice.krad.uif.control.TextAreaControl;
44  import org.kuali.rice.krad.uif.control.TextControl;
45  import org.kuali.rice.krad.uif.element.Message;
46  import org.kuali.rice.krad.uif.field.FieldGroup;
47  import org.kuali.rice.krad.uif.field.InputField;
48  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
49  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycleRestriction;
50  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycleUtils;
51  import org.kuali.rice.krad.uif.lifecycle.ViewPostMetadata;
52  import org.kuali.rice.krad.uif.lifecycle.initialize.AssignIdsTask;
53  import org.kuali.rice.krad.uif.util.ComponentFactory;
54  import org.kuali.rice.krad.uif.util.ComponentUtils;
55  import org.kuali.rice.krad.uif.util.LifecycleElement;
56  import org.kuali.rice.krad.uif.view.FormView;
57  import org.kuali.rice.krad.uif.view.View;
58  import org.kuali.rice.krad.util.GlobalVariables;
59  import org.kuali.rice.krad.util.KRADConstants;
60  
61  /**
62   * View type for lookups.
63   *
64   * <p>Supports doing a search against a data object class or performing a more advanced query. The view
65   * type is primarily made up of two groups, the search (or criteria) group and the results group. Many
66   * options are supported on the view to enable/disable certain features, like what actions are available
67   * on the search results.</p>
68   *
69   * <p>Works in conjunction with {@link org.kuali.rice.krad.lookup.Lookupable} which customizes the view and
70   * carries out the business functionality</p>
71   *
72   * @author Kuali Rice Team (rice.collab@kuali.org)
73   */
74  @BeanTag(name = "lookupView-bean", parent = "Uif-LookupView")
75  public class LookupView extends FormView {
76      private static final long serialVersionUID = 716926008488403616L;
77  
78      private Class<?> dataObjectClass;
79  
80      private List<Component> criteriaFields;
81      private Group criteriaGroup;
82  
83      @RequestParameter
84      private boolean hideCriteriaOnSearch;
85  
86      private List<Component> resultFields;
87      private CollectionGroup resultsGroup;
88  
89      private List<String> defaultSortAttributeNames;
90      private boolean defaultSortAscending;
91  
92      @RequestParameter
93      private Boolean renderReturnLink;
94  
95      @RequestParameter
96      private boolean renderResultActions;
97  
98      @RequestParameter
99      private Boolean renderMaintenanceLinks;
100 
101     @RequestParameter
102     private boolean multipleValuesSelect;
103 
104     @RequestParameter
105     private boolean renderLookupCriteria;
106 
107     @RequestParameter
108     private boolean renderCriteriaActions;
109 
110     private Integer resultSetLimit;
111     private Integer multipleValuesSelectResultSetLimit;
112 
113     private String maintenanceUrlMapping;
114 
115     private FieldGroup rangeFieldGroupPrototype;
116     private Message rangedToMessage;
117 
118     private boolean autoAddActiveCriteria;
119 
120     private List<String> additionalSecurePropertyNames;
121 
122     public LookupView() {
123         super();
124 
125         setViewTypeName(ViewType.LOOKUP);
126 
127         defaultSortAscending = true;
128         autoAddActiveCriteria = true;
129         renderLookupCriteria = true;
130         renderCriteriaActions = true;
131         renderResultActions = true;
132 
133         additionalSecurePropertyNames = new ArrayList<String>();
134     }
135 
136     /**
137      * Initializes Lookupable with data object class and sets the abstractTypeClasses map for the
138      * lookup object path.
139      *
140      * {@inheritDoc}
141      */
142     @Override
143     public void performInitialization(Object model) {
144         // init the view helper with the data object class
145         Lookupable lookupable = (Lookupable) getViewHelperService();
146         lookupable.setDataObjectClass(dataObjectClass);
147 
148         initializeGroups();
149 
150         super.performInitialization(model);
151 
152         getObjectPathToConcreteClassMapping().put(UifPropertyPaths.LOOKUP_CRITERIA, getDataObjectClass());
153         if (StringUtils.isNotBlank(getDefaultBindingObjectPath())) {
154             getObjectPathToConcreteClassMapping().put(getDefaultBindingObjectPath(), getDataObjectClass());
155         }
156     }
157 
158     /**
159      * Reads the convenience render flags and sets the corresponding component property, processing the criteria
160      * fields for any adjustments, and invokes the lookup authorizer to determine whether maintenance links should
161      * be shown.
162      *
163      * {@inheritDoc}
164      */
165     @Override
166     public void performApplyModel(Object model, LifecycleElement parent) {
167         LookupForm lookupForm = (LookupForm) model;
168 
169         if (!renderCriteriaActions) {
170             criteriaGroup.getFooter().setRender(false);
171         }
172 
173         if (!renderLookupCriteria || (hideCriteriaOnSearch && lookupForm.isDisplayResults())) {
174             criteriaGroup.setRender(false);
175         }
176 
177         if (hideCriteriaOnSearch && !lookupForm.isDisplayResults()) {
178             resultsGroup.setRender(false);
179         }
180 
181         boolean returnLinkAllowed = false;
182         boolean maintenanceLinksAllowed = false;
183 
184         // neither return nor maintenance links are shown for multi-value select
185         if (!multipleValuesSelect) {
186             // if coming from a quickfinder we will show the return URL
187             if ((lookupForm.getInitialRequestParameters() != null) && lookupForm.getInitialRequestParameters()
188                     .containsKey(UifParameters.QUICKFINDER_ID)) {
189                 returnLinkAllowed = true;
190             } else {
191                 maintenanceLinksAllowed = true;
192             }
193         } else {
194             renderResultActions = false;
195         }
196 
197         // only override view properties if they were not manually configured
198         if (renderReturnLink == null) {
199             renderReturnLink = returnLinkAllowed;
200         }
201 
202         if (renderMaintenanceLinks == null) {
203             renderMaintenanceLinks = maintenanceLinksAllowed;
204         }
205 
206         // if maintenance links enabled, verify the user had permission
207         if (renderMaintenanceLinks) {
208             LookupViewAuthorizerBase lookupAuthorizer = (LookupViewAuthorizerBase) getAuthorizer();
209 
210             Person user = GlobalVariables.getUserSession().getPerson();
211             renderMaintenanceLinks = lookupAuthorizer.canInitiateMaintenanceDocument(getDataObjectClass().getName(),
212                     user);
213         }
214 
215         convertLookupCriteriaFields(criteriaGroup);
216 
217         super.performApplyModel(model, parent);
218     }
219 
220     /**
221      * Forces session persistence on the criteria fields so the search criteria can be validated on post.
222      *
223      * {@inheritDoc}
224      */
225     @Override
226     public void performFinalize(Object model, LifecycleElement parent) {
227         super.performFinalize(model, parent);
228 
229         LookupForm lookupForm = (LookupForm) model;
230         String viewId = lookupForm.getViewId();
231 
232         Map<String, FilterableLookupCriteriaControlPostData> filterableLookupCriteria = new HashMap<String, FilterableLookupCriteriaControlPostData>();
233 
234         List<InputField> fields = ViewLifecycleUtils.getElementsOfTypeDeep(criteriaGroup, InputField.class);
235 
236         for (InputField field : fields) {
237             field.setForceSessionPersistence(true);
238 
239             String propertyName = field.getPropertyName();
240 
241             if (field.getControl() instanceof FilterableLookupCriteriaControl) {
242                 FilterableLookupCriteriaControl control = (FilterableLookupCriteriaControl) field.getControl();
243                 filterableLookupCriteria.put(propertyName, control.getPostData(propertyName));
244             }
245         }
246 
247         ViewPostMetadata viewPostMetadata = ViewLifecycle.getViewPostMetadata();
248         viewPostMetadata.addComponentPostData(viewId, UifConstants.PostMetadata.FILTERABLE_LOOKUP_CRITERIA, filterableLookupCriteria);
249 
250         if (lookupForm.isReturnByScript()) {
251             getAdditionalHiddenValues().put(UifParameters.RETURN_BY_SCRIPT, "true");
252         }
253     }
254 
255     /**
256      * Adds the 'active' property criteria to the criteria fields if the BO is inactivatable and their is
257      * not already a lookup field for the active property.
258      */
259     protected void addActiveCriteriaIfNecessary() {
260         boolean isInactivatableClass = Inactivatable.class.isAssignableFrom(dataObjectClass);
261 
262         if (!autoAddActiveCriteria || !isInactivatableClass) {
263             return;
264         }
265 
266         boolean hasActiveCriteria = false;
267         for (Component field : getCriteriaFields()) {
268             if (((InputField) field).getPropertyName().equals(UifPropertyPaths.ACTIVE)) {
269                 hasActiveCriteria = true;
270             }
271         }
272 
273         if (hasActiveCriteria) {
274             return;
275         }
276 
277         AttributeDefinition attributeDefinition =
278                 KRADServiceLocatorWeb.getDataDictionaryService().getAttributeDefinition(
279                         dataObjectClass.getName(), UifPropertyPaths.ACTIVE);
280 
281         LookupInputField activeLookupField;
282         if (attributeDefinition == null) {
283             activeLookupField = (LookupInputField) ComponentFactory.getNewComponentInstance(
284                     ComponentFactory.LOOKUP_ACTIVE_INPUT_FIELD);
285         } else {
286             activeLookupField = (LookupInputField) ComponentFactory.getNewComponentInstance(
287                     ComponentFactory.LOOKUP_INPUT_FIELD);
288 
289             activeLookupField.setPropertyName(UifPropertyPaths.ACTIVE);
290             activeLookupField.copyFromAttributeDefinition(attributeDefinition);
291         }
292 
293         getCriteriaFields().add(activeLookupField);
294     }
295 
296     /**
297      * Adds the list of criteria and result fields to their group prototypes, then adds the criteria and result
298      * groups to the items for the view.
299      */
300     protected void initializeGroups() {
301         if ((getCriteriaGroup() != null) && (getCriteriaGroup().getItems().isEmpty())) {
302             getCriteriaGroup().setItems(getCriteriaFields());
303         }
304 
305         if (getResultsGroup() != null) {
306             if ((getResultsGroup().getItems().isEmpty()) && (getResultFields() != null)) {
307                 getResultsGroup().setItems(getResultFields());
308             }
309 
310             if (getResultsGroup().getCollectionObjectClass() == null) {
311                 getResultsGroup().setCollectionObjectClass(getDataObjectClass());
312             }
313         }
314 
315         if (getItems().isEmpty()) {
316             setItems(Arrays.asList(getCriteriaGroup(), getResultsGroup()));
317         }
318     }
319 
320     /**
321      * Performs conversions of the lookup criteria fields within the given group's items.
322      *
323      * <p>Max lengths are removed on text controls so wildcards can be added. Ranged date fields are
324      * converted to field groups with the from/to date fields</p>
325      */
326     protected void convertLookupCriteriaFields(Group lookupGroup) {
327         @SuppressWarnings("unchecked")
328         List<Component> criteriaGroupItems = (List<Component>) lookupGroup.getItems();
329 
330         // holds the index and range field group for replacement into the items
331         HashMap<Integer, Component> dateRangeFieldMap = new HashMap<Integer, Component>();
332 
333         int rangeIndex = 0;
334         for (Component component : criteriaGroupItems) {
335             if (component == null) {
336                 continue;
337             }
338 
339             if (Group.class.isAssignableFrom(component.getClass())) {
340                 convertLookupCriteriaFields((Group) component);
341             } else if (FieldGroup.class.isAssignableFrom(component.getClass())) {
342                 convertLookupCriteriaFields(((FieldGroup) component).getGroup());
343             } else if (LookupInputField.class.isAssignableFrom(component.getClass())) {
344                 LookupInputField lookupInputField = (LookupInputField) component;
345 
346                 // set the max length on the controls to allow for wildcards
347                 Control control = lookupInputField.getControl();
348 
349                 if (control instanceof TextControl) {
350                     ((TextControl) control).setMaxLength(null);
351                 } else if (control instanceof TextAreaControl) {
352                     ((TextAreaControl) control).setMaxLength(null);
353                 }
354 
355                 if (lookupInputField.isRanged()) {
356                     FieldGroup rangeFieldGroup = createDateRangeFieldGroup(lookupInputField);
357 
358                     dateRangeFieldMap.put(rangeIndex, rangeFieldGroup);
359                 }
360             }
361 
362             rangeIndex++;
363         }
364 
365         // replace original fields with range field groups
366         for (Integer index : dateRangeFieldMap.keySet()) {
367             criteriaGroupItems.set(index, dateRangeFieldMap.get(index));
368         }
369 
370         criteriaGroup.setItems(criteriaGroupItems);
371     }
372 
373     /**
374      * Creates a {@link FieldGroup} instance to replace the given lookup input field as a
375      * date criteria range.
376      *
377      * <p>The field group is created by copying {@link LookupView#rangeFieldGroupPrototype}. This can be
378      * used to configure how the field group will appear. In addition, the two lookup fields are separated
379      * with a message that can be configured with {@link LookupView#rangedToMessage}</p>
380      *
381      * @param toDate lookup input field that field group should be build for
382      *
383      * @return field group that contains a from and to lookup input field for searching a date range
384      *
385      * @see LookupView#rangeFieldGroupPrototype
386      * @see LookupView#rangedToMessage
387      */
388     protected FieldGroup createDateRangeFieldGroup(LookupInputField toDate) {
389         // Generate an ID when the "to date" field is out of the normal lifecycle flow
390         if (toDate.getId() == null) {
391             toDate.setId(AssignIdsTask.generateId(toDate, ViewLifecycle.getView()));
392         }
393 
394         FieldGroup rangeFieldGroup = ComponentUtils.copy(getRangeFieldGroupPrototype());
395 
396         // Copy some properties from the "to date" field to the field group
397         rangeFieldGroup.setFieldLabel(ComponentUtils.copy(toDate.getFieldLabel()));
398         rangeFieldGroup.setPropertyExpressions(toDate.getPropertyExpressions());
399         rangeFieldGroup.setProgressiveRender(toDate.getProgressiveRender());
400         rangeFieldGroup.setProgressiveRenderViaAJAX(toDate.isProgressiveRenderViaAJAX());
401         rangeFieldGroup.setConditionalRefresh(toDate.getConditionalRefresh());
402         rangeFieldGroup.setRefreshWhenChangedPropertyNames(toDate.getRefreshWhenChangedPropertyNames());
403         rangeFieldGroup.setForceSessionPersistence(true);
404 
405         // Reset some fields for the "to date" field
406         toDate.getFieldLabel().setRender(false);
407         toDate.setRefreshWhenChangedPropertyNames(null);
408         toDate.setForceSessionPersistence(true);
409 
410         // Create a "from date" field from the "to date" field
411         LookupInputField fromDate = ComponentUtils.copy(toDate,
412                 KRADConstants.LOOKUP_DEFAULT_RANGE_SEARCH_LOWER_BOUND_LABEL);
413         fromDate.getBindingInfo().setBindingName(
414                 KRADConstants.LOOKUP_RANGE_LOWER_BOUND_PROPERTY_PREFIX + fromDate.getPropertyName());
415         fromDate.setPropertyName(
416                 KRADConstants.LOOKUP_RANGE_LOWER_BOUND_PROPERTY_PREFIX + fromDate.getPropertyName());
417         fromDate.setOrder(0);
418 
419         // add the criteria fields to the field group
420         List<Component> fieldGroupItems = new ArrayList<Component>();
421         fieldGroupItems.add(fromDate);
422         fieldGroupItems.add(rangedToMessage);
423         fieldGroupItems.add(toDate);
424         rangeFieldGroup.setItems(fieldGroupItems);
425 
426         return rangeFieldGroup;
427     }
428 
429     /**
430      * Class for the data object the lookup applies to.
431      *
432      * <p>The object class name is used to pick up a dictionary entry which will feed the attribute field
433      * definitions and other configuration. In addition it is to configure the
434      * {@link org.kuali.rice.krad.lookup.Lookupable} which will carry out the search action</p>
435      *
436      * @return lookup data object class
437      */
438     @BeanTagAttribute(name = "dataObjectClass")
439     public Class<?> getDataObjectClass() {
440         return this.dataObjectClass;
441     }
442 
443     /**
444      * @see LookupView#getDataObjectClass()
445      */
446     public void setDataObjectClass(Class<?> dataObjectClass) {
447         this.dataObjectClass = dataObjectClass;
448     }
449 
450     /**
451      * Convenience setter to configure the lookup data object class by class name.
452      *
453      * @param dataObjectClassName full class name for the lookup data object
454      */
455     public void setDataObjectClassName(String dataObjectClassName) {
456         try {
457             this.dataObjectClass = Class.forName(dataObjectClassName);
458         } catch (ClassNotFoundException e) {
459             throw new RuntimeException("Unable to set class for class name: " + dataObjectClassName, e);
460         }
461     }
462 
463     /**
464      * Indicates whether a return value link should be rendered for each result row.
465      *
466      * <p>When the lookup is called from a view (using a {@link org.kuali.rice.krad.uif.widget.QuickFinder} the return
467      * link can be returned to allow the user to return a value(s) for a selected row. Note, if this is not manually
468      * set the framework will determine when the lookup is called from a quickfinder and turn this flag on</p>
469      *
470      * @return boolean true if the return link should be rendered for each result row, false if not
471      */
472     @BeanTagAttribute(name = "renderReturnLink")
473     public Boolean isRenderReturnLink() {
474         return this.renderReturnLink;
475     }
476 
477     /**
478      * @see LookupView#isRenderReturnLink()
479      */
480     public void setRenderReturnLink(Boolean renderReturnLink) {
481         this.renderReturnLink = renderReturnLink;
482     }
483 
484     /**
485      * Indicates whether the actions column for the search results collection group should be rendered (default
486      * is true).
487      *
488      * <p>Note this is a convenience property for setting the render property on the result collection group</p>
489      *
490      * @return boolean true if the result actions column should be rendered, false if not
491      */
492     @BeanTagAttribute(name = "isRenderResultActions")
493     public boolean isRenderResultActions() {
494         return this.renderResultActions;
495     }
496 
497     /**
498      * @see LookupView#isRenderResultActions()
499      */
500     public void setRenderResultActions(boolean renderResultActions) {
501         this.renderResultActions = renderResultActions;
502     }
503 
504     /**
505      * Indicates whether links for maintenance actions (new, edit, copy, delete) should be rendered.
506      *
507      * <p>When this property is not manually set it will be enabled by the framework when a lookup is not invoked
508      * from a quickfinder (for example a standard link from a menu). Regardless if the flag is manually enabled
509      * or enabled by the framework, an additional authorization check will be performed to determine if the user
510      * has initiate permission for the maintenance document associated with the lookup data object class. If not,
511      * this flag will be disabled</p>
512      *
513      * @return boolean true if maintenance links should be rendered, false if not
514      */
515     @BeanTagAttribute(name = "renderMaintenanceLinks")
516     public Boolean isRenderMaintenanceLinks() {
517         return this.renderMaintenanceLinks;
518     }
519 
520     /**
521      * @see LookupView#isRenderMaintenanceLinks()
522      */
523     public void setRenderMaintenanceLinks(Boolean renderMaintenanceLinks) {
524         this.renderMaintenanceLinks = renderMaintenanceLinks;
525     }
526 
527     /**
528      * Indicates whether multiple values select should be enabled for the lookup.
529      *
530      * <p>When set to true, the select field is enabled for the lookup results group that allows the user
531      * to select one or more rows for returning. The framework will also set the {@link #isRenderReturnLink()}
532      * and {@link #isRenderMaintenanceLinks()} properties to false (unless manually overridden)</p>
533      *
534      * @return true if multiple values select should be enabled, false otherwise
535      */
536     @BeanTagAttribute(name = "multipleValueSelect")
537     public boolean isMultipleValuesSelect() {
538         return multipleValuesSelect;
539     }
540 
541     /**
542      * @see LookupView#isMultipleValuesSelect()
543      */
544     public void setMultipleValuesSelect(boolean multipleValuesSelect) {
545         this.multipleValuesSelect = multipleValuesSelect;
546     }
547 
548     /**
549      * List of fields that will be rendered for the lookup criteria.
550      *
551      * <p>This is a convenience property for setting the items in {@link #getCriteriaGroup()}, which is the
552      * group the criteria for the lookup is rendered in. This property can be bypassed and the items set
553      * directly in the criteria group (for more flexibility)</p>
554      *
555      * @return List of components to render as the lookup criteria
556      */
557     @ViewLifecycleRestriction
558     @BeanTagAttribute(name = "criteriaFields", type = BeanTagAttribute.AttributeType.LISTBEAN)
559     public List<Component> getCriteriaFields() {
560         return this.criteriaFields;
561     }
562 
563     /**
564      * @see LookupView#getCriteriaFields()
565      */
566     public void setCriteriaFields(List<Component> criteriaFields) {
567         this.criteriaFields = criteriaFields;
568     }
569 
570     /**
571      * Component {@link Group} instance to render as search criteria.
572      *
573      * <p>Fields that make up the criteria for the lookup will be rendered in this group. This can be used in a few
574      * different ways:
575      *
576      * <ul>
577      * <li>Set the group to have the desired layout, style, and other general group properties. Note this
578      * is done in the base lookup view. The actual criteria fields can then be configured using
579      * {@link #getCriteriaFields()}</li>
580      * <li>Configure the criteria group entirely (ignoring criteria fields). This would allow you to do things
581      * like have multiple groups for the criteria.</li>
582      * </ul></p>
583      *
584      * <p>Note the footer for the criteria group can contain actions (such as search, clear, custom actions)</p>
585      *
586      * @return group instance that will hold the search criteria fields
587      */
588     @ViewLifecycleRestriction
589     @BeanTagAttribute(name = "criteriaGroup", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
590     public Group getCriteriaGroup() {
591         return this.criteriaGroup;
592     }
593 
594     /**
595      * @see LookupView#getCriteriaGroup()
596      */
597     public void setCriteriaGroup(Group criteriaGroup) {
598         this.criteriaGroup = criteriaGroup;
599     }
600 
601     public boolean isHideCriteriaOnSearch() {
602         return hideCriteriaOnSearch;
603     }
604 
605     public void setHideCriteriaOnSearch(boolean hideCriteriaOnSearch) {
606         this.hideCriteriaOnSearch = hideCriteriaOnSearch;
607     }
608 
609     /**
610      * List of fields that will be rendered for the result collection group, each field will be a column
611      * (assuming table layout is used).
612      *
613      * <p>This is a convenience property for setting the items in {@link #getResultsGroup()}, which is the
614      * collection group the results for the lookup is rendered in. This property can be bypassed and the items set
615      * directly in the results group (for more flexibility)</p>
616      *
617      * @return List of components to render in the results group
618      */
619     @ViewLifecycleRestriction
620     @BeanTagAttribute(name = "resultFields", type = BeanTagAttribute.AttributeType.LISTBEAN)
621     public List<Component> getResultFields() {
622         return this.resultFields;
623     }
624 
625     /**
626      * @see LookupView#getResultFields()
627      */
628     public void setResultFields(List<Component> resultFields) {
629         this.resultFields = resultFields;
630     }
631 
632     /**
633      * Component {@link CollectionGroup} instance to render for the lookup results.
634      *
635      * <p>After a search is performed, the resulting data objects will be rendered in this collection group. This
636      * collection group can be used in two ways:
637      *
638      * <ul>
639      * <li>Set the desired layout, style, and other general collection group properties. Note this is done
640      * in the base lookup view. Then the actual fields that are rendered in the collection group can be
641      * configured using {@link #getResultFields()}</li>
642      * <li>Configure the results group entirely (ignoring result fields)</li>
643      * </ul></p>
644      *
645      * <p>Note actions that are presented for the results can be configured using the
646      * {@link org.kuali.rice.krad.uif.container.CollectionGroup#getLineActions()} property</p>
647      *
648      * @return collection group instance to render for the lookup results
649      */
650     @ViewLifecycleRestriction
651     @BeanTagAttribute(name = "resultsGroup", type = BeanTagAttribute.AttributeType.SINGLEBEAN)
652     public CollectionGroup getResultsGroup() {
653         return this.resultsGroup;
654     }
655 
656     /**
657      * @see LookupView#getResultsGroup()
658      */
659     public void setResultsGroup(CollectionGroup resultsGroup) {
660         this.resultsGroup = resultsGroup;
661     }
662 
663     /**
664      * List of property names on the configured data object class that will be used to perform the initial
665      * sorting of the search results.
666      *
667      * @return list of property names valid for the configured data object class
668      * @see LookupView#isDefaultSortAscending()
669      */
670     @BeanTagAttribute(name = "defaultSortAttributeNames", type = BeanTagAttribute.AttributeType.LISTVALUE)
671     public List<String> getDefaultSortAttributeNames() {
672         return this.defaultSortAttributeNames;
673     }
674 
675     /**
676      * @see LookupView#getDefaultSortAttributeNames()
677      */
678     public void setDefaultSortAttributeNames(List<String> defaultSortAttributeNames) {
679         this.defaultSortAttributeNames = defaultSortAttributeNames;
680     }
681 
682     /**
683      * Indicates whether the initial sort performed using {@link #getDefaultSortAttributeNames()} is done based
684      * on ascending or descending order (default is true, ascending).
685      *
686      * @return boolean true if ascending sort should be performed, false if descending sort should be
687      *         performed
688      */
689     @BeanTagAttribute(name = "defaultSortAscending")
690     public boolean isDefaultSortAscending() {
691         return this.defaultSortAscending;
692     }
693 
694     /**
695      * @see LookupView#isDefaultSortAscending()
696      */
697     public void setDefaultSortAscending(boolean defaultSortAscending) {
698         this.defaultSortAscending = defaultSortAscending;
699     }
700 
701     /**
702      * Retrieves the maximum number of records that will be listed as a result of the lookup search.
703      *
704      * @return Integer result set limit
705      */
706     @BeanTagAttribute(name = "resultSetLimit")
707     public Integer getResultSetLimit() {
708         return resultSetLimit;
709     }
710 
711     /**
712      * @see LookupView#getResultSetLimit()
713      */
714     public void setResultSetLimit(Integer resultSetLimit) {
715         this.resultSetLimit = resultSetLimit;
716     }
717 
718     /**
719      * Retrieves the maximum number of records that will be listed as a result of the multiple
720      * values select lookup search.
721      *
722      * @return multiple values select result set limit
723      */
724     @BeanTagAttribute(name = "multipleValuesSelectResultSetLimit")
725     public Integer getMultipleValuesSelectResultSetLimit() {
726         return multipleValuesSelectResultSetLimit;
727     }
728 
729     /**
730      * @see LookupView#getMultipleValuesSelectResultSetLimit()
731      */
732     public void setMultipleValuesSelectResultSetLimit(Integer multipleValuesSelectResultSetLimit) {
733         this.multipleValuesSelectResultSetLimit = multipleValuesSelectResultSetLimit;
734     }
735 
736     /**
737      * String that maps to the maintenance controller for the maintenance document (if any) associated with the
738      * lookup data object class.
739      *
740      * <p>Mapping will be used to build the maintenance action links (such as edit, copy, and new). If not given, the
741      * default maintenance mapping will be used</p>
742      *
743      * @return mapping string
744      */
745     @BeanTagAttribute(name = "maintenanceUrlMapping")
746     public String getMaintenanceUrlMapping() {
747         return maintenanceUrlMapping;
748     }
749 
750     /**
751      * @see LookupView#getMaintenanceUrlMapping()
752      */
753     public void setMaintenanceUrlMapping(String maintenanceUrlMapping) {
754         this.maintenanceUrlMapping = maintenanceUrlMapping;
755     }
756 
757     /**
758      * Indicates whether the action buttons like search in the criteria group footer should be rendered,
759      * defaults to true.
760      *
761      * @return boolean true if the criteria actions should be rendered, false if not
762      */
763     public boolean isRenderCriteriaActions() {
764         return renderCriteriaActions;
765     }
766 
767     /**
768      * @see LookupView#isRenderCriteriaActions()
769      */
770     public void setRenderCriteriaActions(boolean renderCriteriaActions) {
771         this.renderCriteriaActions = renderCriteriaActions;
772     }
773 
774     /**
775      * Indicates whether the lookup criteria group should be rendered, default to true.
776      *
777      * <p>Hiding the criteria group can be useful in cases where the criteria is passed in through the request and
778      * also the search is executed on the initial request</p>
779      *
780      * @return boolean true if criteria group should be rendered, false if not
781      */
782     public boolean isRenderLookupCriteria() {
783         return renderLookupCriteria;
784     }
785 
786     /**
787      * @see LookupView#isRenderLookupCriteria()
788      */
789     public void setRenderLookupCriteria(boolean renderLookupCriteria) {
790         this.renderLookupCriteria = renderLookupCriteria;
791     }
792 
793     /**
794      * Field group prototype that will be copied to create any date range field groups.
795      *
796      * @return field group instance to use for creating range field groups
797      */
798     @ViewLifecycleRestriction(UifConstants.ViewPhases.INITIALIZE)
799     public FieldGroup getRangeFieldGroupPrototype() {
800         return rangeFieldGroupPrototype;
801     }
802 
803     /**
804      * @see LookupView#getRangeFieldGroupPrototype()
805      */
806     public void setRangeFieldGroupPrototype(FieldGroup rangeFieldGroupPrototype) {
807         this.rangeFieldGroupPrototype = rangeFieldGroupPrototype;
808     }
809 
810     /**
811      * Component {@link Message} instance to render between the range criteria fields within a range
812      * field group.
813      *
814      * @return message instance for range field group
815      */
816     @ViewLifecycleRestriction(UifConstants.ViewPhases.INITIALIZE)
817     public Message getRangedToMessage() {
818         return rangedToMessage;
819     }
820 
821     /**
822      * @see LookupView#getRangedToMessage()
823      */
824     public void setRangedToMessage(Message rangedToMessage) {
825         this.rangedToMessage = rangedToMessage;
826     }
827 
828     /**
829      * Indicates whether the 'active' criteria field must be added automatically for Inactivatable business
830      * objects.
831      *
832      * @return boolean true if active criteria should be added
833      */
834     public boolean isAutoAddActiveCriteria() {
835         return autoAddActiveCriteria;
836     }
837 
838     /**
839      * @see LookupView#isAutoAddActiveCriteria()
840      */
841     public void setAutoAddActiveCriteria(boolean autoAddActiveCriteria) {
842         this.autoAddActiveCriteria = autoAddActiveCriteria;
843     }
844 
845     /**
846      * List of secure property names that are in addition to the
847      * {@link org.kuali.rice.krad.uif.component.ComponentSecurity} or
848      * {@link org.kuali.rice.krad.datadictionary.AttributeSecurity} attributes.
849      *
850      * @return list of secure property names
851      */
852     @BeanTagAttribute(name = "additionalSecurePropertyNames", type = BeanTagAttribute.AttributeType.LISTVALUE)
853     public List<String> getAdditionalSecurePropertyNames() {
854         return additionalSecurePropertyNames;
855     }
856 
857     /**
858      * @see LookupView#getAdditionalSecurePropertyNames()
859      */
860     public void setAdditionalSecurePropertyNames(List<String> additionalSecurePropertyNames) {
861         this.additionalSecurePropertyNames = additionalSecurePropertyNames;
862     }
863 
864     /**
865      * Clones the {@code LookupView} with a deep copy.
866      *
867      * @return a clone of the current {@code LookupView}
868      *
869      * @see org.kuali.rice.krad.uif.component.ComponentBase#clone()
870      */
871     @Override
872     public LookupView clone() throws CloneNotSupportedException {
873         LookupView lookupViewCopy = (LookupView) super.clone();
874 
875         if (getViewHelperService() != null) {
876             lookupViewCopy.setViewHelperService(((LookupableImpl) getViewHelperService()).copy());
877         }
878 
879         return lookupViewCopy;
880     }
881 }