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 org.apache.commons.lang.StringUtils;
19  import org.kuali.rice.core.api.exception.RiceRuntimeException;
20  import org.kuali.rice.core.api.util.RiceConstants;
21  import org.kuali.rice.core.api.util.RiceKeyConstants;
22  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
23  import org.kuali.rice.krad.service.ModuleService;
24  import org.kuali.rice.krad.uif.UifConstants;
25  import org.kuali.rice.krad.uif.UifParameters;
26  import org.kuali.rice.krad.uif.UifPropertyPaths;
27  import org.kuali.rice.krad.util.GlobalVariables;
28  import org.kuali.rice.krad.util.KRADConstants;
29  import org.kuali.rice.krad.util.KRADUtils;
30  import org.kuali.rice.krad.util.UrlFactory;
31  import org.kuali.rice.krad.web.form.UifFormBase;
32  import org.kuali.rice.krad.web.service.ModelAndViewService;
33  import org.kuali.rice.krad.web.service.impl.ControllerServiceImpl;
34  import org.springframework.web.servlet.ModelAndView;
35  import org.springframework.web.servlet.mvc.support.RedirectAttributes;
36  
37  import javax.servlet.http.HttpServletRequest;
38  import java.util.ArrayList;
39  import java.util.Collection;
40  import java.util.Collections;
41  import java.util.HashMap;
42  import java.util.HashSet;
43  import java.util.List;
44  import java.util.Map;
45  import java.util.Properties;
46  import java.util.Set;
47  
48  /**
49   * Default implementation of the lookup controller service.
50   *
51   * @author Kuali Rice Team (rice.collab@kuali.org)
52   */
53  public class LookupControllerServiceImpl extends ControllerServiceImpl implements LookupControllerService {
54  
55      private ModelAndViewService modelAndViewService;
56  
57      /**
58       * {@inheritDoc}
59       */
60      @Override
61      public ModelAndView start(UifFormBase form) {
62          LookupForm lookupForm = (LookupForm) form;
63  
64          Lookupable lookupable = lookupForm.getLookupable();
65          if (lookupable == null) {
66              throw new RuntimeException("Lookupable is null");
67          }
68  
69          HttpServletRequest request = form.getRequest();
70          if (request.getParameter(UifParameters.MESSAGE_TO_DISPLAY) != null) {
71              GlobalVariables.getMessageMap().putErrorForSectionId(UifConstants.MessageKeys.LOOKUP_RESULT_MESSAGES,
72                      request.getParameter(UifParameters.MESSAGE_TO_DISPLAY));
73          }
74  
75          if (!lookupForm.isRedirectedLookup()) {
76              ModelAndView redirectModelAndView = checkForModuleLookupRedirect(lookupForm, request);
77              if (redirectModelAndView != null) {
78                  return redirectModelAndView;
79              }
80          }
81  
82          return super.start(lookupForm);
83      }
84  
85      /**
86       * Checks for a module service that claims the lookup class as an EBO, and if found redirects to the URL
87       * given by the module service.
88       *
89       * @param lookupForm form instance containing the lookup data
90       * @param request http request being handled
91       * @return ModelAndView instance for redirecting to the lookup, or null if a redirect is not needed
92       */
93      protected ModelAndView checkForModuleLookupRedirect(LookupForm lookupForm, HttpServletRequest request) {
94          Class<?> lookupObjectClass;
95          try {
96              lookupObjectClass = Class.forName(lookupForm.getDataObjectClassName());
97          } catch (ClassNotFoundException e) {
98              throw new RiceRuntimeException("Unable to get class for name: " + lookupForm.getDataObjectClassName(),
99                      e);
100         }
101 
102         ModuleService responsibleModuleService =
103                 KRADServiceLocatorWeb.getKualiModuleService().getResponsibleModuleService(lookupObjectClass);
104         if (responsibleModuleService != null && responsibleModuleService.isExternalizable(lookupObjectClass)) {
105             String lookupUrl = responsibleModuleService.getExternalizableDataObjectLookupUrl(lookupObjectClass,
106                     KRADUtils.convertRequestMapToProperties(request.getParameterMap()));
107 
108             Properties redirectUrlProps = new Properties();
109             redirectUrlProps.setProperty(UifParameters.REDIRECTED_LOOKUP, "true");
110 
111             // clear current form from session
112             GlobalVariables.getUifFormManager().removeSessionForm(lookupForm);
113 
114             return getModelAndViewService().performRedirect(lookupForm, lookupUrl, redirectUrlProps);
115         }
116 
117         return null;
118     }
119 
120     /**
121      * Carries out the search action by invoking the {@link Lookupable#performSearch)} method on the
122      * configured lookupable (view helper) and then setting the results onto the given form.
123      *
124      * {@inheritDoc}
125      */
126     @Override
127     public ModelAndView search(LookupForm lookupForm) {
128         Lookupable lookupable = lookupForm.getLookupable();
129         if (lookupable == null) {
130             throw new RuntimeException("Lookupable is null.");
131         }
132 
133         Collection<?> displayList = lookupable.performSearch(lookupForm, lookupForm.getLookupCriteria(), true);
134 
135         lookupForm.setLookupResults(displayList);
136 
137         return getModelAndViewService().getModelAndView(lookupForm);
138     }
139 
140     /**
141      * Carries out the clear values action by invoking the {@link Lookupable#performClear)} method on the
142      * configured lookupable (view helper) and then setting the cleared criteria onto the given form.
143      *
144      * {@inheritDoc}
145      */
146     @Override
147     public ModelAndView clearValues(LookupForm lookupForm) {
148         Lookupable lookupable = lookupForm.getLookupable();
149         if (lookupable == null) {
150             throw new RuntimeException("Lookupable is null.");
151         }
152 
153         Map<String, String> resetLookupCriteria = lookupable.performClear(lookupForm, lookupForm.getLookupCriteria());
154 
155         lookupForm.setLookupCriteria(resetLookupCriteria);
156 
157         return getModelAndViewService().getModelAndView(lookupForm);
158     }
159 
160     /**
161      * Loops through all the lookup results generating the line identifier for each and adding the
162      * resulting set of identifies to the form property
163      * {@link org.kuali.rice.krad.web.form.UifFormBase#getSelectedLookupResultsCache()}.
164      *
165      * {@inheritDoc}
166      */
167     @Override
168     public ModelAndView selectAllPages(LookupForm lookupForm) {
169         List<? extends Object> lookupResults = (List<? extends Object>) lookupForm.getLookupResults();
170 
171         List<String> fromFieldNames = new ArrayList<String>(lookupForm.getFieldConversions().keySet());
172 
173         // loop through  the lookup results and store identifiers for all items in the set
174         Set<String> selectedValues = new HashSet<String>();
175         for (Object lineItem : lookupResults) {
176             String lineIdentifier = LookupUtils.generateMultiValueKey(lineItem, fromFieldNames);
177 
178             selectedValues.add(lineIdentifier);
179         }
180 
181         lookupForm.setSelectedLookupResultsCache(selectedValues);
182 
183         return getModelAndViewService().getModelAndView(lookupForm);
184     }
185 
186     /**
187      * Clears the form property {@link org.kuali.rice.krad.web.form.UifFormBase#getSelectedLookupResultsCache()}
188      * and the selected lines property.
189      *
190      * {@inheritDoc}
191      */
192     @Override
193     public ModelAndView deselectAllPages(LookupForm lookupForm) {
194         lookupForm.getSelectedLookupResultsCache().clear();
195 
196         Set<String> selectedLines = lookupForm.getSelectedCollectionLines().get(UifPropertyPaths.LOOKUP_RESULTS);
197         if (selectedLines != null) {
198             selectedLines.clear();
199         }
200 
201         return getModelAndViewService().getModelAndView(lookupForm);
202     }
203 
204     /**
205      * Builds the URL for returning back to the calling view and passing the selected line values.
206      *
207      * <p>We attempt to pass back all the selected line identifiers as a request parameter on the return URL.
208      * However, this could result in an URL longer than the max length supported by browsers (the most restrictive
209      * is used). If this happens, for local lookups we use Spring flash attributes. In the case of a remote
210      * lookup, there is nothing we can do and return an error message.</p>
211      *
212      * {@inheritDoc}
213      */
214     @Override
215     public String returnSelected(LookupForm lookupForm, RedirectAttributes redirectAttributes) {
216         LookupUtils.refreshLookupResultSelections(lookupForm);
217 
218         Properties urlParams = buildReturnSelectedParameters(lookupForm);
219         String returnUrl = UrlFactory.parameterizeUrl(lookupForm.getReturnLocation(), urlParams);
220 
221         boolean lookupCameFromDifferentServer = KRADUtils.areDifferentDomains(lookupForm.getReturnLocation(),
222                 lookupForm.getRequestUrl());
223 
224         boolean urlGreaterThanMaxLength = returnUrl.length() > RiceConstants.MAXIMUM_URL_LENGTH;
225         if (urlGreaterThanMaxLength) {
226             // removed selected values parameter from the return url
227             urlParams.remove(UifParameters.SELECTED_LINE_VALUES);
228 
229             // if lookup was on a different server, we can't return the selected lines and instead
230             // will return an error message
231             if (lookupCameFromDifferentServer) {
232                 urlParams.setProperty(UifParameters.REFRESH_STATUS, UifConstants.RefreshStatus.ERROR);
233                 urlParams.setProperty(UifParameters.MESSAGE_TO_DISPLAY,
234                         RiceKeyConstants.INFO_LOOKUP_RESULTS_MV_RETURN_EXCEEDS_LIMIT);
235             } else {
236                 // otherwise use flash attributes instead of the URL to return the selected line identifiers
237                 String selectedLineValues = getSelectedLineValues(lookupForm);
238                 redirectAttributes.addFlashAttribute(UifParameters.SELECTED_LINE_VALUES, selectedLineValues);
239             }
240         }
241 
242         GlobalVariables.getUifFormManager().removeSessionForm(lookupForm);
243 
244         // rebuild url based on updated parameters
245         returnUrl = UrlFactory.parameterizeUrl(lookupForm.getReturnLocation(), urlParams);
246 
247         return UifConstants.REDIRECT_PREFIX + returnUrl;
248     }
249 
250     /**
251      * Builds all the request parameters for the return URL.
252      *
253      * @param lookupForm form instance containing the lookup data
254      * @return Properties contains the request parameters key/value pairs
255      */
256     protected Properties buildReturnSelectedParameters(LookupForm lookupForm) {
257         Properties parameters = new Properties();
258 
259         String multiValueReturnFieldsParam = getMultiValueReturnFields(lookupForm);
260         parameters.setProperty(UifParameters.MULIT_VALUE_RETURN_FILEDS, multiValueReturnFieldsParam);
261 
262         String selectedLineValues = getSelectedLineValues(lookupForm);
263         parameters.setProperty(UifParameters.SELECTED_LINE_VALUES, selectedLineValues);
264 
265         parameters.setProperty(KRADConstants.DISPATCH_REQUEST_PARAMETER, KRADConstants.RETURN_METHOD_TO_CALL);
266         parameters.setProperty(UifParameters.FORM_KEY, lookupForm.getReturnFormKey());
267         parameters.setProperty(KRADConstants.REFRESH_CALLER, lookupForm.getView().getId());
268         parameters.setProperty(KRADConstants.REFRESH_CALLER_TYPE, UifConstants.RefreshCallerTypes.MULTI_VALUE_LOOKUP);
269         parameters.setProperty(KRADConstants.REFRESH_DATA_OBJECT_CLASS, lookupForm.getDataObjectClassName());
270 
271         if (StringUtils.isNotBlank(lookupForm.getQuickfinderId())) {
272             parameters.setProperty(UifParameters.QUICKFINDER_ID, lookupForm.getQuickfinderId());
273         }
274 
275         if (StringUtils.isNotBlank(lookupForm.getLookupCollectionName())) {
276             parameters.setProperty(UifParameters.LOOKUP_COLLECTION_NAME, lookupForm.getLookupCollectionName());
277         }
278 
279         if (StringUtils.isNotBlank(lookupForm.getLookupCollectionId())) {
280             parameters.setProperty(UifParameters.LOOKUP_COLLECTION_ID, lookupForm.getLookupCollectionId());
281         }
282 
283         if (StringUtils.isNotBlank(lookupForm.getReferencesToRefresh())) {
284             parameters.setProperty(KRADConstants.REFERENCES_TO_REFRESH, lookupForm.getReferencesToRefresh());
285         }
286 
287         return parameters;
288     }
289 
290     /**
291      * Builds a string containing the names of the fields being returned separated by a comma.
292      *
293      * @param lookupForm form instance containing the lookup data
294      * @return String names of return fields separated by a comma
295      */
296     protected String getMultiValueReturnFields(LookupForm lookupForm) {
297         String multiValueReturnFieldsParam = "";
298 
299         List<String> multiValueReturnFields = lookupForm.getMultiValueReturnFields();
300         Collections.sort(multiValueReturnFields);
301         if (multiValueReturnFields != null && !multiValueReturnFields.isEmpty()) {
302             for (String field : multiValueReturnFields) {
303                 multiValueReturnFieldsParam += field + ",";
304             }
305 
306             multiValueReturnFieldsParam = StringUtils.removeEnd(multiValueReturnFieldsParam, ",");
307         }
308 
309         return multiValueReturnFieldsParam;
310     }
311 
312     /**
313      * Builds a string containing the selected line identifiers separated by a comma.
314      *
315      * @param lookupForm form instance containing the lookup data
316      * @return String selected line identifiers separated by a comma
317      */
318     protected String getSelectedLineValues(LookupForm lookupForm) {
319         String selectedLineValues = "";
320 
321         Set<String> selectedLines = lookupForm.getSelectedCollectionLines().get(UifPropertyPaths.LOOKUP_RESULTS);
322         if (selectedLines != null) {
323             for (String selectedLine : selectedLines) {
324                 selectedLineValues += selectedLine + ",";
325             }
326 
327             selectedLineValues = StringUtils.removeEnd(selectedLineValues, ",");
328         }
329 
330         return selectedLineValues;
331     }
332 
333     /**
334      * Instance of model and view service to use within the collection service.
335      *
336      * @return ModelAndViewService instance
337      */
338     protected ModelAndViewService getModelAndViewService() {
339         return modelAndViewService;
340     }
341 
342     /**
343      * @see LookupControllerServiceImpl#getModelAndViewService()
344      */
345     public void setModelAndViewService(ModelAndViewService modelAndViewService) {
346         this.modelAndViewService = modelAndViewService;
347     }
348 }