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