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.widget;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.kuali.rice.core.api.CoreApiServiceLocator;
20  import org.kuali.rice.core.web.format.Formatter;
21  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
22  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
23  import org.kuali.rice.krad.service.KRADServiceLocator;
24  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
25  import org.kuali.rice.krad.service.ModuleService;
26  import org.kuali.rice.krad.uif.UifConstants;
27  import org.kuali.rice.krad.uif.UifParameters;
28  import org.kuali.rice.krad.uif.component.BindingInfo;
29  import org.kuali.rice.krad.uif.component.Component;
30  import org.kuali.rice.krad.uif.element.Action;
31  import org.kuali.rice.krad.uif.element.Link;
32  import org.kuali.rice.krad.uif.field.DataField;
33  import org.kuali.rice.krad.uif.field.InputField;
34  import org.kuali.rice.krad.uif.util.LookupInquiryUtils;
35  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
36  import org.kuali.rice.krad.uif.util.ViewModelUtils;
37  import org.kuali.rice.krad.uif.view.View;
38  import org.kuali.rice.krad.util.UrlFactory;
39  
40  import java.security.GeneralSecurityException;
41  import java.util.HashMap;
42  import java.util.List;
43  import java.util.Map;
44  import java.util.Map.Entry;
45  import java.util.Properties;
46  
47  /**
48   * Widget for rendering an Inquiry link or DirectInquiry action field
49   *
50   * <p>
51   * The inquiry widget will render a button for the field value when
52   * that field is editable. When read only the widget will create a link on the display value.
53   * It points to the associated inquiry view for the field. The inquiry can be configured to point to a certain
54   * {@code InquiryView}, or the framework will attempt to associate the
55   * field with a inquiry based on its metadata (in particular its
56   * relationships in the model).
57   * </p>
58   *
59   * @author Kuali Rice Team (rice.collab@kuali.org)
60   */
61  @BeanTag(name = "inquiry", parent = "Uif-Inquiry")
62  public class Inquiry extends WidgetBase {
63      private static final long serialVersionUID = -2154388007867302901L;
64      private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(Inquiry.class);
65  
66      public static final String INQUIRY_TITLE_PREFIX = "title.inquiry.url.value.prependtext";
67  
68      private String baseInquiryUrl;
69  
70      private String dataObjectClassName;
71      private String viewName;
72  
73      private Map<String, String> inquiryParameters;
74  
75      private Link inquiryLink;
76  
77      private Action directInquiryAction;
78      private boolean enableDirectInquiry;
79  
80      private boolean adjustInquiryParameters;
81      private BindingInfo fieldBindingInfo;
82  
83      public Inquiry() {
84          super();
85  
86          inquiryParameters = new HashMap<String, String>();
87      }
88  
89      /**
90       * @see org.kuali.rice.krad.uif.widget.WidgetBase#performFinalize(org.kuali.rice.krad.uif.view.View,
91       *      java.lang.Object, org.kuali.rice.krad.uif.component.Component)
92       */
93      @Override
94      public void performFinalize(View view, Object model, Component parent) {
95          super.performFinalize(view, model, parent);
96  
97          if (!isRender()) {
98              return;
99          }
100 
101         // set render to false until we find an inquiry class
102         setRender(false);
103 
104         // Do checks for inquiry when read only
105         if (isReadOnly()) {
106             if (StringUtils.isBlank(((DataField) parent).getBindingInfo().getBindingPath())) {
107                 return;
108             }
109 
110             // check if field value is null, if so no inquiry
111             Object propertyValue = ObjectPropertyUtils.getPropertyValue(model,
112                     ((DataField) parent).getBindingInfo().getBindingPath());
113             if ((propertyValue == null) || StringUtils.isBlank(propertyValue.toString())) {
114                 return;
115             }
116         }
117 
118         // Do checks for direct inquiry when editable
119         if (!isReadOnly() && parent instanceof InputField) {
120             if (!enableDirectInquiry) {
121                 return;
122             }
123 
124             // determine whether inquiry parameters will need adjusted
125             if (StringUtils.isBlank(getDataObjectClassName())
126                     || (getInquiryParameters() == null)
127                     || getInquiryParameters().isEmpty()) {
128                 // if inquiry parameters not given, they will not be adjusted by super
129                 adjustInquiryParameters = true;
130                 fieldBindingInfo = ((InputField) parent).getBindingInfo();
131             }
132         }
133 
134         setupLink(view, model, (DataField) parent);
135     }
136 
137     /**
138      * Get parent object and field name and build the inquiry link
139      *
140      * <p>
141      * This was moved from the performFinalize because overlapping and to be used
142      * by DirectInquiry.
143      * </p>
144      *
145      * @param view - Container View
146      * @param model - model
147      * @param field - The parent Attribute field
148      */
149     public void setupLink(View view, Object model, DataField field) {
150         String propertyName = field.getBindingInfo().getBindingName();
151 
152         // if class and parameters configured, build link from those
153         if (StringUtils.isNotBlank(getDataObjectClassName()) && (getInquiryParameters() != null) &&
154                 !getInquiryParameters().isEmpty()) {
155             Class<?> inquiryObjectClass;
156             try {
157                 inquiryObjectClass = Class.forName(getDataObjectClassName());
158             } catch (ClassNotFoundException e) {
159                 LOG.error("Unable to get class for: " + getDataObjectClassName());
160                 throw new RuntimeException(e);
161             }
162 
163             updateInquiryParameters(field.getBindingInfo());
164 
165             buildInquiryLink(model, propertyName, inquiryObjectClass, getInquiryParameters());
166         }
167         // get inquiry class and parameters from view helper
168         else {
169             // get parent object for inquiry metadata
170             Object parentObject = ViewModelUtils.getParentObjectForMetadata(view, model, field);
171             view.getViewHelperService().buildInquiryLink(parentObject, propertyName, this);
172         }
173     }
174 
175     /**
176      * Adjusts the path on the inquiry parameter property to match the binding
177      * path prefix of the given {@code BindingInfo}
178      *
179      * @param bindingInfo - binding info instance to copy binding path prefix from
180      */
181     public void updateInquiryParameters(BindingInfo bindingInfo) {
182         Map<String, String> adjustedInquiryParameters = new HashMap<String, String>();
183         for (Entry<String, String> stringEntry : inquiryParameters.entrySet()) {
184             String toField = stringEntry.getValue();
185             String adjustedFromFieldPath = bindingInfo.getPropertyAdjustedBindingPath(stringEntry.getKey());
186 
187             adjustedInquiryParameters.put(adjustedFromFieldPath, toField);
188         }
189 
190         this.inquiryParameters = adjustedInquiryParameters;
191     }
192 
193     /**
194      * Builds the inquiry link based on the given inquiry class and parameters
195      *
196      * @param dataObject - parent object that contains the data (used to pull inquiry
197      * parameters)
198      * @param propertyName - name of the property the inquiry is set on
199      * @param inquiryObjectClass - class of the object the inquiry should point to
200      * @param inquiryParams - map of key field mappings for the inquiry
201      */
202     public void buildInquiryLink(Object dataObject, String propertyName, Class<?> inquiryObjectClass,
203             Map<String, String> inquiryParams) {
204 
205         Properties urlParameters = new Properties();
206 
207         urlParameters.setProperty(UifParameters.DATA_OBJECT_CLASS_NAME, inquiryObjectClass.getName());
208         urlParameters.setProperty(UifParameters.METHOD_TO_CALL, UifConstants.MethodToCallNames.START);
209         if(StringUtils.isNotBlank(this.viewName)){
210           urlParameters.setProperty(UifParameters.VIEW_NAME, this.viewName);
211         }
212         // add inquiry specific parms to url
213         if (getInquiryLink().getLightBox() != null) {
214             getInquiryLink().getLightBox().setAddAppParms(true);
215         }
216 
217         // configure inquiry when read only
218         if (isReadOnly()) {
219             for (Entry<String, String> inquiryParameter : inquiryParams.entrySet()) {
220                 String parameterName = inquiryParameter.getKey();
221 
222                 Object parameterValue = ObjectPropertyUtils.getPropertyValue(dataObject, parameterName);
223 
224                 // TODO: need general format util that uses spring
225                 if (parameterValue == null) {
226                     parameterValue = "";
227                 } else if (parameterValue instanceof java.sql.Date) {
228                     if (Formatter.findFormatter(parameterValue.getClass()) != null) {
229                         Formatter formatter = Formatter.getFormatter(parameterValue.getClass());
230                         parameterValue = formatter.format(parameterValue);
231                     }
232                 } else {
233                     parameterValue = parameterValue.toString();
234                 }
235 
236                 // Encrypt value if it is a field that has restriction that prevents a value from being shown to
237                 // user, because we don't want the browser history to store the restricted attributes value in the URL
238                 if (KRADServiceLocatorWeb.getDataObjectAuthorizationService()
239                         .attributeValueNeedsToBeEncryptedOnFormsAndLinks(inquiryObjectClass,
240                                 inquiryParameter.getValue())) {
241                     try {
242                         parameterValue = CoreApiServiceLocator.getEncryptionService().encrypt(parameterValue);
243                     } catch (GeneralSecurityException e) {
244                         throw new RuntimeException("Exception while trying to encrypted value for inquiry framework.",
245                                 e);
246                     }
247                 }
248 
249                 // add inquiry parameter to URL
250                 urlParameters.put(inquiryParameter.getValue(), parameterValue);
251             }
252 
253             /* build inquiry URL */
254             String inquiryUrl;
255 
256             // check for EBOs for an alternate inquiry URL
257             ModuleService responsibleModuleService =
258                     KRADServiceLocatorWeb.getKualiModuleService().getResponsibleModuleService(inquiryObjectClass);
259             if (responsibleModuleService != null && responsibleModuleService.isExternalizable(inquiryObjectClass)) {
260                 inquiryUrl = responsibleModuleService.getExternalizableDataObjectInquiryUrl(inquiryObjectClass,
261                         urlParameters);
262             } else {
263                 inquiryUrl = UrlFactory.parameterizeUrl(getBaseInquiryUrl(), urlParameters);
264             }
265 
266             getInquiryLink().setHref(inquiryUrl);
267 
268             // set inquiry title
269             String linkTitle = createTitleText(inquiryObjectClass);
270             linkTitle = LookupInquiryUtils.getLinkTitleText(linkTitle, inquiryObjectClass, getInquiryParameters());
271             getInquiryLink().setTitle(linkTitle);
272 
273             setRender(true);
274         }
275 
276         // configure direct inquiry when editable
277         if (!isReadOnly()) {
278             // Direct inquiry
279             String inquiryUrl = UrlFactory.parameterizeUrl(getBaseInquiryUrl(), urlParameters);
280 
281             StringBuilder paramMapString = new StringBuilder();
282 
283             // Build parameter string using the actual names of the fields as on the html page
284             for (Entry<String, String> inquiryParameter : inquiryParams.entrySet()) {
285                 String inquiryParameterFrom = inquiryParameter.getKey();
286 
287                 if (adjustInquiryParameters && (fieldBindingInfo != null)) {
288                     inquiryParameterFrom = fieldBindingInfo.getPropertyAdjustedBindingPath(inquiryParameterFrom);
289                 }
290 
291                 paramMapString.append(inquiryParameterFrom);
292                 paramMapString.append(":");
293                 paramMapString.append(inquiryParameter.getValue());
294                 paramMapString.append(",");
295             }
296             paramMapString.deleteCharAt(paramMapString.length() - 1);
297 
298             // Check if lightbox is set. Get lightbox options.
299             String lightBoxOptions = "";
300             boolean lightBoxShow = (getInquiryLink().getLightBox() != null);
301             if (lightBoxShow) {
302                 lightBoxOptions = getInquiryLink().getLightBox().getTemplateOptionsJSString();
303             }
304 
305             // Create onlick script to open the inquiry window on the click event
306             // of the direct inquiry
307             StringBuilder onClickScript = new StringBuilder("showDirectInquiry(\"");
308             onClickScript.append(inquiryUrl);
309             onClickScript.append("\", \"");
310             onClickScript.append(paramMapString);
311             onClickScript.append("\", ");
312             onClickScript.append(lightBoxShow);
313             onClickScript.append(", ");
314             onClickScript.append(lightBoxOptions);
315             onClickScript.append(");");
316 
317             directInquiryAction.setPerformDirtyValidation(false);
318             directInquiryAction.setActionScript(onClickScript.toString());
319 
320             setRender(true);
321         }
322     }
323 
324     /**
325      * Gets text to prepend to the inquiry link title
326      *
327      * @param dataObjectClass - data object class being inquired into
328      * @return String title prepend text
329      */
330     public String createTitleText(Class<?> dataObjectClass) {
331         String titleText = "";
332 
333         String titlePrefixProp = KRADServiceLocator.getKualiConfigurationService().getPropertyValueAsString(
334                 INQUIRY_TITLE_PREFIX);
335         if (StringUtils.isNotBlank(titlePrefixProp)) {
336             titleText += titlePrefixProp + " ";
337         }
338 
339         String objectLabel = KRADServiceLocatorWeb.getDataDictionaryService().getDataDictionary().getDataObjectEntry(
340                 dataObjectClass.getName()).getObjectLabel();
341         if (StringUtils.isNotBlank(objectLabel)) {
342             titleText += objectLabel + " ";
343         }
344 
345         return titleText;
346     }
347 
348     /**
349      * @see org.kuali.rice.krad.uif.component.ComponentBase#getComponentsForLifecycle()
350      */
351     @Override
352     public List<Component> getComponentsForLifecycle() {
353         List<Component> components = super.getComponentsForLifecycle();
354 
355         components.add(getInquiryLink());
356         components.add(getDirectInquiryAction());
357 
358         return components;
359     }
360 
361     /**
362      * Returns the URL for the inquiry for which parameters will be added
363      *
364      * <p>
365      * The base URL includes the domain, context, and controller mapping for the inquiry invocation. Parameters are
366      * then added based on configuration to complete the URL. This is generally defaulted to the application URL and
367      * internal KRAD servlet mapping, but can be changed to invoke another application such as the Rice standalone
368      * server
369      * </p>
370      *
371      * @return String inquiry base URL
372      */
373     @BeanTagAttribute(name="baseInquiryUrl")
374     public String getBaseInquiryUrl() {
375         return this.baseInquiryUrl;
376     }
377 
378     /**
379      * Setter for the inquiry base url (domain, context, and controller)
380      *
381      * @param baseInquiryUrl
382      */
383     public void setBaseInquiryUrl(String baseInquiryUrl) {
384         this.baseInquiryUrl = baseInquiryUrl;
385     }
386 
387     /**
388      * Full class name the inquiry should be provided for
389      *
390      * <p>
391      * This is passed on to the inquiry request for the data object the lookup should be rendered for. This is then
392      * used by the inquiry framework to select the lookup view (if more than one inquiry view exists for the same
393      * data object class name, the {@link #getViewName()} property should be specified to select the view to render).
394      * </p>
395      *
396      * @return String inquiry class name
397      */
398     @BeanTagAttribute(name="dataObjectClassName")
399     public String getDataObjectClassName() {
400         return this.dataObjectClassName;
401     }
402 
403     /**
404      * Setter for the class name that inquiry should be provided for
405      *
406      * @param dataObjectClassName
407      */
408     public void setDataObjectClassName(String dataObjectClassName) {
409         this.dataObjectClassName = dataObjectClassName;
410     }
411 
412     /**
413      * When multiple target inquiry views exists for the same data object class, the view name can be set to
414      * determine which one to use
415      *
416      * <p>
417      * When creating multiple inquiry views for the same data object class, the view name can be specified for the
418      * different versions (for example 'simple' and 'advanced'). When multiple inquiry views exist the view name must
419      * be sent with the data object class for the request. Note the view id can be alternatively used to uniquely
420      * identify the inquiry view
421      * </p>
422      */
423     @BeanTagAttribute(name="viewName")
424     public String getViewName() {
425         return this.viewName;
426     }
427 
428     /**
429      * Setter for the view name configured on the inquiry view that should be invoked by the inquiry widget
430      *
431      * @param viewName
432      */
433     public void setViewName(String viewName) {
434         this.viewName = viewName;
435     }
436 
437     /**
438      * Map that determines what properties from a calling view will be sent to properties on the inquiry data object
439      *
440      * <p>
441      * When invoking an inquiry view, a query is done against the inquiries configured data object and the resulting
442      * record is display. The values for the properties configured within the inquiry parameters Map will be
443      * pulled and passed along as values for the inquiry data object properties (thus they form the criteria for
444      * the inquiry)
445      * </p>
446      *
447      * @return Map<String, String> mapping of calling view properties to inquiry data object properties
448      */
449     @BeanTagAttribute(name="inquiryParameters",type= BeanTagAttribute.AttributeType.MAPVALUE)
450     public Map<String, String> getInquiryParameters() {
451         return this.inquiryParameters;
452     }
453 
454     /**
455      * Setter for the map that determines what property values on the calling view will be sent to properties on the
456      * inquiry data object
457      *
458      * @param inquiryParameters
459      */
460     public void setInquiryParameters(Map<String, String> inquiryParameters) {
461         this.inquiryParameters = inquiryParameters;
462     }
463 
464     /**
465      * {@code Link} that will be rendered for an inquiry
466      *
467      * @return the inquiry link
468      */
469     @BeanTagAttribute(name="inquiryLink",type= BeanTagAttribute.AttributeType.SINGLEBEAN)
470     public Link getInquiryLink() {
471         return this.inquiryLink;
472     }
473 
474     /**
475      * Setter for the inquiry {@code Link}
476      *
477      * @param inquiryLink - the inquiry {@link Link} object
478      */
479     public void setInquiryLink(Link inquiryLink) {
480         this.inquiryLink = inquiryLink;
481     }
482 
483     /**
484      * {@code Action} that will be rendered next to the field for a direct inquiry
485      *
486      * @return the directInquiryAction
487      */
488     @BeanTagAttribute(name="directInquiryAction",type= BeanTagAttribute.AttributeType.SINGLEBEAN)
489     public Action getDirectInquiryAction() {
490         return this.directInquiryAction;
491     }
492 
493     /**
494      * Setter for the direct inquiry {@code Action}
495      *
496      * @param directInquiryAction the direct inquiry {@link Action}
497      */
498     public void setDirectInquiryAction(Action directInquiryAction) {
499         this.directInquiryAction = directInquiryAction;
500     }
501 
502     /**
503      * Indicates that the direct inquiry will not be rendered
504      *
505      * @return boolean true if the direct inquiry should be rendered, false if not
506      */
507     @BeanTagAttribute(name="enableDirectInquiry")
508     public boolean isEnableDirectInquiry() {
509         return enableDirectInquiry;
510     }
511 
512     /**
513      * Setter for the hideDirectInquiry flag
514      *
515      * @param enableDirectInquiry
516      */
517     public void setEnableDirectInquiry(boolean enableDirectInquiry) {
518         this.enableDirectInquiry = enableDirectInquiry;
519     }
520 }