View Javadoc

1   /*
2    * Copyright 2011 The Kuali Foundation Licensed under the Educational Community
3    * License, Version 1.0 (the "License"); you may not use this file except in
4    * compliance with the License. You may obtain a copy of the License at
5    * http://www.opensource.org/licenses/ecl1.php Unless required by applicable law
6    * or agreed to in writing, software distributed under the License is
7    * distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
8    * KIND, either express or implied. See the License for the specific language
9    * governing permissions and limitations under the License.
10   */
11  package org.kuali.rice.krad.uif.history;
12  
13  import org.apache.commons.lang.StringUtils;
14  import org.apache.log4j.Logger;
15  import org.kuali.rice.krad.service.DataObjectMetaDataService;
16  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
17  import org.kuali.rice.krad.uif.UifConstants;
18  import org.kuali.rice.krad.uif.container.View;
19  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
20  import org.kuali.rice.krad.uif.util.ViewModelUtils;
21  import org.kuali.rice.krad.web.form.UifFormBase;
22  
23  import javax.servlet.http.HttpServletRequest;
24  import java.io.Serializable;
25  import java.io.UnsupportedEncodingException;
26  import java.net.URLDecoder;
27  import java.net.URLEncoder;
28  import java.util.ArrayList;
29  import java.util.Enumeration;
30  import java.util.List;
31  
32  /**
33   * History class used to keep track of views visited so they can be displayed in the ui
34   * as breadcrumbs - both as homeward path and history path interpretations
35   *
36   * @author Kuali Rice Team (rice.collab@kuali.org)
37   */
38  public class History implements Serializable {
39      private static final long serialVersionUID = -8279297694371557335L;
40      private static final Logger LOG = Logger.getLogger(History.class);
41  
42      public static final String ENTRY_TOKEN = "$";
43      public static final String VAR_TOKEN = ",";
44  
45      private boolean appendHomewardPath;
46      private boolean appendPassedHistory;
47  
48      private HistoryEntry current;
49  
50      private List<HistoryEntry> homewardPath;
51      private List<HistoryEntry> historyEntries;
52  
53      public History() {
54          historyEntries = new ArrayList<HistoryEntry>();
55      }
56  
57      /**
58       * Gets the predetermined homeward path for this view's history.
59       * This is set by the same property in the view's Breadcrumbs configuration.
60       *
61       * @return the homewardPath
62       */
63      public List<HistoryEntry> getHomewardPath() {
64          return this.homewardPath;
65      }
66  
67      /**
68       * @param homewardPath the homewardPath to set
69       */
70      public void setHomewardPath(List<HistoryEntry> homewardPath) {
71          this.homewardPath = homewardPath;
72      }
73  
74      /**
75       * Gets a list of the current HistoryEntries not including the current entry.
76       * This list does not include the "&history=" query parameter on each HistoryEntry's
77       * url variable.  For HistoryEntries that include history information to be passed to the
78       * view they are retrieving, getGeneratedBreadcrumbs is used.
79       *
80       * @return the history
81       */
82      public List<HistoryEntry> getHistoryEntries() {
83          return this.historyEntries;
84      }
85  
86      /**
87       * @param history the history to set
88       */
89      public void setHistoryEntries(List<HistoryEntry> history) {
90          this.historyEntries = history;
91      }
92  
93      /**
94       * Gets the current view's HistoryEntry.
95       * This does not include the "&history=" query parameter on its
96       * url variable.  For the HistoryEntry that includes history information to be passed
97       * on the url it is retrieving, getGeneratedCurrentBreadcrumb is used.
98       *
99       * @return the current
100      */
101     public HistoryEntry getCurrent() {
102         return this.current;
103     }
104 
105     /**
106      * @param current the current to set
107      */
108     private void setCurrent(String viewId, String pageId, String title, String url, String formKey) {
109         HistoryEntry entry = new HistoryEntry(viewId, pageId, title, url, formKey);
110         current = entry;
111     }
112 
113     /**
114      * @param current the current to set
115      */
116     public void setCurrent(HistoryEntry current) {
117         this.current = current;
118     }
119 
120     /**
121      * Takes in the encoded history query parameter string passed on the url and parses it to create
122      * the list of historyEntries.  It will also append any homeward path if appendHomewardPath is true.  This
123      * append will happen after the passedHistory entries are appended so it will not make sense to use both settings
124      * in
125      * most cases.
126      *
127      * @param parameterString
128      */
129     public void buildHistoryFromParameterString(String parameterString) {
130         if (StringUtils.isNotEmpty(parameterString)) {
131             try {
132                 parameterString = URLDecoder.decode(parameterString, "UTF-8");
133             } catch (UnsupportedEncodingException e) {
134                 LOG.error("Error decoding history param", e);
135             }
136             historyEntries = new ArrayList<HistoryEntry>();
137             if (appendPassedHistory) {
138                 String[] historyTokens = parameterString.split("\\" + ENTRY_TOKEN);
139                 for (String token : historyTokens) {
140                     String[] params = token.split(VAR_TOKEN);
141                     pushToHistory(params[0], params[1], params[2], params[3], params[4]);
142                 }
143             }
144         }
145 
146         if (appendHomewardPath) {
147             historyEntries.addAll(homewardPath);
148         }
149     }
150 
151     /**
152      * Gets the encoded and tokenized history parameter string that is representative of the HistoryEntries
153      * currently in History and includes the current view's HistoryEntry.  This parameter should be appended on any
154      * appropriate links which perform view swapping.
155      *
156      * @return
157      */
158     public String getHistoryParameterString() {
159         String historyString = "";
160         for (HistoryEntry e : historyEntries) {
161             if (historyEntries.indexOf(e) == 0) {
162                 historyString = historyString + e.toParam();
163             } else {
164                 historyString = historyString + ENTRY_TOKEN + e.toParam();
165             }
166         }
167         // add current
168         if (current != null) {
169             if (historyString.equals("")) {
170                 historyString = historyString + current.toParam();
171             } else {
172                 historyString = historyString + ENTRY_TOKEN + current.toParam();
173             }
174         }
175         try {
176             historyString = URLEncoder.encode(historyString, "UTF-8");
177         } catch (Exception e) {
178             LOG.error("Error encoding history param", e);
179         }
180         return historyString;
181     }
182 
183     /**
184      * Generates a list of HistoryEntries that can be used as breadcrumbs by the breadcrumb widget.  This
185      * method appends the appropriate history information on the HistoryEntry url variables so when a view is requested
186      * its history can be regenerated for use in its breadcrumbs.  It also sets the the passed showHome variable to
187      * false
188      * to prevent showing the homeward path more than once (as it is passed through the history variable backwards).
189      * This does not include the current HistoryEntry as a breadcrumb.
190      *
191      * @return
192      */
193     public List<HistoryEntry> getGeneratedBreadcrumbs() {
194         List<HistoryEntry> breadcrumbs = new ArrayList<HistoryEntry>();
195         for (int i = 0; i < historyEntries.size(); i++) {
196             if (i == 0) {
197                 breadcrumbs.add(copyEntry(historyEntries.get(i)));
198             } else {
199                 HistoryEntry breadcrumb = copyEntry(historyEntries.get(i));
200                 String historyParam = "";
201                 for (int j = 0; j < i; j++) {
202                     historyParam = historyParam + ENTRY_TOKEN + historyEntries.get(j).toParam();
203                 }
204                 historyParam = historyParam.replaceFirst("\\" + ENTRY_TOKEN, "");
205                 try {
206                     historyParam = URLEncoder.encode(historyParam, "UTF-8");
207                 } catch (Exception e) {
208                     LOG.error("Error encoding history param", e);
209                 }
210 
211                 String url = "";
212                 if (breadcrumb.getUrl().contains("?")) {
213                     url = breadcrumb.getUrl() + "&" + UifConstants.UrlParams.HISTORY + "=" + historyParam;
214                 } else {
215                     url = breadcrumb.getUrl() + "?" + UifConstants.UrlParams.HISTORY + "=" + historyParam;
216                 }
217 
218                 breadcrumb.setUrl(url);
219                 breadcrumbs.add(breadcrumb);
220             }
221         }
222 
223         return breadcrumbs;
224     }
225 
226     /**
227      * Gets the current HistoryEntry in the breadcrumb format described in getGeneratedBreadcrumbs.
228      *
229      * @return
230      */
231     public HistoryEntry getGeneratedCurrentBreadcrumb() {
232         HistoryEntry breadcrumb = copyEntry(current);
233         String historyParam = "";
234         for (int j = 0; j < historyEntries.size(); j++) {
235             historyParam = historyParam + ENTRY_TOKEN + historyEntries.get(j).toParam();
236         }
237         historyParam = historyParam.replaceFirst("\\" + ENTRY_TOKEN, "");
238 
239         try {
240             historyParam = URLEncoder.encode(historyParam, "UTF-8");
241         } catch (Exception e) {
242             LOG.error("Error encoding history param", e);
243         }
244 
245         String url = "";
246         if (breadcrumb.getUrl().contains("?")) {
247             url = breadcrumb.getUrl() + "&" + UifConstants.UrlParams.HISTORY + "=" + historyParam;
248         } else {
249             url = breadcrumb.getUrl() + "?" + UifConstants.UrlParams.HISTORY + "=" + historyParam;
250         }
251         breadcrumb.setUrl(url);
252 
253         return breadcrumb;
254     }
255 
256     /**
257      * Copies a HistoryEntry, for use during breadcrumb generation.
258      *
259      * @param e
260      * @return
261      */
262     private HistoryEntry copyEntry(HistoryEntry e) {
263         return new HistoryEntry(e.getViewId(), e.getPageId(), e.getTitle(), e.getUrl(), e.getFormKey());
264     }
265 
266     /**
267      * Pushes the information passed in to history.
268      * Note: currently only used internally in the class - be cautious about its external use.
269      *
270      * @param viewId
271      * @param pageId
272      * @param title
273      * @param url
274      * @param formKey
275      */
276     public void pushToHistory(String viewId, String pageId, String title, String url, String formKey) {
277         HistoryEntry entry = new HistoryEntry(viewId, pageId, title, url, formKey);
278         historyEntries.add(entry);
279     }
280 
281     /**
282      * When this is set to true, the homeward path will be appended.
283      * Note:  For most cases this should only be on during the first view load.
284      * This setting is set automatically in most cases.
285      *
286      * @param appendHomewardPath the appendHomewardPath to set
287      */
288     public void setAppendHomewardPath(boolean appendHomewardPath) {
289         this.appendHomewardPath = appendHomewardPath;
290     }
291 
292     /**
293      * @return the appendHomewardPath
294      */
295     public boolean isAppendHomewardPath() {
296         return appendHomewardPath;
297     }
298 
299     /**
300      * Appends the passed history as each different view is shown.  This setting should be used when displaying
301      * passed history is relevant to the user (ie inquiry/lookup chains).  This setting is set automatically in
302      * most cases.
303      *
304      * @param appendPassedHistory the appendPassedHistory to set
305      */
306     public void setAppendPassedHistory(boolean appendPassedHistory) {
307         this.appendPassedHistory = appendPassedHistory;
308     }
309 
310     /**
311      * @return the appendPassedHistory
312      */
313     public boolean isAppendPassedHistory() {
314         return appendPassedHistory;
315     }
316 
317     /**
318      * Sets the current HistoryEntry using information from the form and the request.  This history parameter is
319      * extracted out of the url inorder for a "clean" url to be used in history parameter and
320      * breadcrumb generation, as passing history history through the nested urls is unnecessary.
321      *
322      * @param form
323      * @param request
324      */
325     @SuppressWarnings("unchecked")
326     public void setCurrent(UifFormBase form, HttpServletRequest request) {
327         if (!request.getMethod().equals("POST")) {
328             boolean showHomeValue = false;
329             boolean pageIdValue = false;
330             boolean formKeyValue = false;
331 
332             String queryString = "";
333             String url = request.getRequestURL().toString();
334 
335             //remove history attribute
336             Enumeration<String> params = request.getParameterNames();
337             while (params.hasMoreElements()) {
338                 String key = params.nextElement();
339                 if (!key.equals(UifConstants.UrlParams.HISTORY)) {
340                     for (String value : request.getParameterValues(key)) {
341                         queryString = queryString + "&" + key + "=" + value;
342                     }
343                 } else if (key.equals(UifConstants.UrlParams.PAGE_ID)) {
344                     pageIdValue = true;
345                 } else if (key.equals(UifConstants.UrlParams.SHOW_HOME)) {
346                     showHomeValue = true;
347                 } else if (key.equals(UifConstants.UrlParams.FORM_KEY)) {
348                     formKeyValue = true;
349                 }
350             }
351 
352             //add formKey and pageId to url
353             if (StringUtils.isNotBlank(form.getFormKey()) && !formKeyValue) {
354                 queryString = queryString + "&" + UifConstants.UrlParams.FORM_KEY + "=" + form.getFormKey();
355             }
356             if (StringUtils.isNotBlank(form.getPageId()) && !pageIdValue) {
357                 queryString = queryString + "&" + UifConstants.UrlParams.PAGE_ID + "=" + form.getPageId();
358             }
359             if (!showHomeValue) {
360                 queryString = queryString + "&" + UifConstants.UrlParams.SHOW_HOME + "=false";
361             }
362 
363             queryString = queryString.replaceFirst("&", "");
364 
365             if (StringUtils.isNotEmpty(queryString)) {
366                 url = url + "?" + queryString;
367             }
368 
369             this.setCurrent(form.getViewId(), form.getPageId(), buildViewTitle(form), url, form.getFormKey());
370         }
371     }
372 
373     /**
374      * Builds the title for the view to display in history (for example breadcrumbs)
375      *
376      * <p>
377      * Retrieves the viewLabelFieldPropertyName from the view if configured, otherwise attempts
378      * to find the title attribute for the default data object. If view label property is found the
379      * corresponding property value is retrieved and appended to the title for the view
380      * </p>
381      *
382      * TODO: Possibly move so it can be used for the actual view title, not just history
383      *
384      * @param form - form instance containing the view and view data
385      * @return String title string to use
386      */
387     protected String buildViewTitle(UifFormBase form) {
388         View view = form.getView();
389         String title = view.getTitle();
390 
391         // may move this into view logic instead in the future if it is required for the view's title (not just breadcrumb)
392         // if so remove this and just use getTitle - this logic would be in performFinalize instead
393         String viewLabelPropertyName = view.getViewLabelFieldPropertyName();
394 
395         // if view label property name given, try to retrieve the title attribute for the main data object
396         if (StringUtils.isBlank(viewLabelPropertyName)) {
397             Class<?> dataObjectClass;
398             if (StringUtils.isNotBlank(view.getDefaultBindingObjectPath())) {
399                 dataObjectClass = ObjectPropertyUtils.getPropertyType(form, view.getDefaultBindingObjectPath());
400             } else {
401                 dataObjectClass = view.getFormClass();
402             }
403 
404             DataObjectMetaDataService mds = KRADServiceLocatorWeb.getDataObjectMetaDataService();
405             if (dataObjectClass != null) {
406                 viewLabelPropertyName = mds.getTitleAttribute(dataObjectClass);
407             }
408         }
409 
410         String viewLabelPropertyPath = "";
411         if (StringUtils.isNotBlank(viewLabelPropertyName)) {
412             // adjust binding prefix
413             if (!viewLabelPropertyName.startsWith(UifConstants.NO_BIND_ADJUST_PREFIX)) {
414                 if (StringUtils.isNotBlank(view.getDefaultBindingObjectPath())) {
415                     viewLabelPropertyPath = view.getDefaultBindingObjectPath() + "." + viewLabelPropertyName;
416                 }
417             } else {
418                 viewLabelPropertyPath = StringUtils.removeStart(viewLabelPropertyName,
419                         UifConstants.NO_BIND_ADJUST_PREFIX);
420             }
421         }
422         else {
423             // attempt to get title attribute
424             Class<?> dataObjectClass;
425             if (StringUtils.isNotBlank(view.getDefaultBindingObjectPath())) {
426                 dataObjectClass = ViewModelUtils.getObjectClassForMetadata(view, form,
427                         view.getDefaultBindingObjectPath());
428             } else {
429                 dataObjectClass = view.getFormClass();
430             }
431 
432             DataObjectMetaDataService mds = KRADServiceLocatorWeb.getDataObjectMetaDataService();
433             if (dataObjectClass != null) {
434                 String titleAttribute = mds.getTitleAttribute(dataObjectClass);
435                 if (StringUtils.isNotBlank(titleAttribute)) {
436                   viewLabelPropertyPath = view.getDefaultBindingObjectPath() + "." + titleAttribute;
437                 }
438             }
439         }
440 
441         Object viewLabelPropertyValue = null;
442         if (StringUtils.isNotBlank(viewLabelPropertyPath)) {
443             viewLabelPropertyValue = ObjectPropertyUtils.getPropertyValue(form, viewLabelPropertyPath);
444         }
445 
446         String titleAppend = "";
447         if (viewLabelPropertyValue != null) {
448             titleAppend = viewLabelPropertyValue.toString();
449         }
450 
451         if (StringUtils.isNotBlank(titleAppend) && view.getAppendOption() != null) {
452             if (view.getAppendOption().equalsIgnoreCase(UifConstants.TitleAppendTypes.DASH)) {
453                 title = title + " - " + titleAppend;
454             } else if (view.getAppendOption().equalsIgnoreCase(UifConstants.TitleAppendTypes.PARENTHESIS)) {
455                 title = title + "(" + titleAppend + ")";
456             } else if (view.getAppendOption().equalsIgnoreCase(UifConstants.TitleAppendTypes.REPLACE)) {
457                 title = titleAppend;
458             }
459             //else it is none or blank so no title modification will be used
460         }
461 
462         return title;
463     }
464 }