Classes in this File | Line Coverage | Branch Coverage | Complexity | ||||
DocumentRouteHeaderValueLookupableHelperServiceImpl |
|
| 5.375;5.375 |
1 | /* | |
2 | * Copyright 2006-2011 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 | ||
17 | package org.kuali.rice.kew.bo.lookup; | |
18 | ||
19 | import org.apache.commons.beanutils.PropertyUtils; | |
20 | import org.apache.commons.lang.StringUtils; | |
21 | import org.kuali.rice.core.api.datetime.DateTimeService; | |
22 | import org.kuali.rice.core.api.config.property.ConfigContext; | |
23 | import org.kuali.rice.core.web.format.BooleanFormatter; | |
24 | import org.kuali.rice.core.web.format.CollectionFormatter; | |
25 | import org.kuali.rice.core.web.format.DateFormatter; | |
26 | import org.kuali.rice.core.web.format.Formatter; | |
27 | import org.kuali.rice.kew.docsearch.DocSearchCriteriaDTO; | |
28 | import org.kuali.rice.kew.docsearch.DocumentLookupCriteriaProcessor; | |
29 | import org.kuali.rice.kew.docsearch.DocumentLookupCriteriaProcessorKEWAdapter; | |
30 | import org.kuali.rice.kew.docsearch.DocumentSearchResult; | |
31 | import org.kuali.rice.kew.docsearch.DocumentSearchResultComponents; | |
32 | import org.kuali.rice.kew.docsearch.StandardDocumentSearchCriteriaProcessor; | |
33 | import org.kuali.rice.kew.doctype.bo.DocumentType; | |
34 | import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue; | |
35 | import org.kuali.rice.kew.service.KEWServiceLocator; | |
36 | import org.kuali.rice.kew.util.KEWConstants; | |
37 | import org.kuali.rice.kew.util.KEWPropertyConstants; | |
38 | import org.kuali.rice.kew.web.KeyValueSort; | |
39 | import org.kuali.rice.kim.bo.Person; | |
40 | import org.kuali.rice.kns.bo.BusinessObject; | |
41 | import org.kuali.rice.kns.lookup.HtmlData; | |
42 | import org.kuali.rice.kns.lookup.HtmlData.AnchorHtmlData; | |
43 | import org.kuali.rice.kns.lookup.KualiLookupableHelperServiceImpl; | |
44 | import org.kuali.rice.kns.util.GlobalVariables; | |
45 | import org.kuali.rice.kns.util.KNSConstants; | |
46 | import org.kuali.rice.kns.web.comparator.CellComparatorHelper; | |
47 | import org.kuali.rice.kns.web.struts.form.LookupForm; | |
48 | import org.kuali.rice.kns.web.ui.Column; | |
49 | import org.kuali.rice.kns.web.ui.Field; | |
50 | import org.kuali.rice.kns.web.ui.ResultRow; | |
51 | import org.kuali.rice.kns.web.ui.Row; | |
52 | ||
53 | import java.lang.reflect.InvocationTargetException; | |
54 | import java.sql.Date; | |
55 | import java.util.ArrayList; | |
56 | import java.util.Collection; | |
57 | import java.util.HashMap; | |
58 | import java.util.List; | |
59 | import java.util.Map; | |
60 | ||
61 | /** | |
62 | * This is a description of what this class does - chris don't forget to fill this in. | |
63 | * | |
64 | * @author Kuali Rice Team (rice.collab@kuali.org) | |
65 | * | |
66 | */ | |
67 | 0 | public class DocumentRouteHeaderValueLookupableHelperServiceImpl extends |
68 | KualiLookupableHelperServiceImpl { | |
69 | ||
70 | private static final long serialVersionUID = -5162419674659967408L; | |
71 | DateTimeService dateTimeService; | |
72 | DocumentLookupCriteriaProcessor processor; | |
73 | ||
74 | ||
75 | /** | |
76 | * This overridden method ... | |
77 | * | |
78 | * @see org.kuali.rice.kew.bo.lookup.DocumentRouteHeaderValueLookupableHelperService#setDateTimeService(org.kuali.rice.core.api.datetime.DateTimeService) | |
79 | */ | |
80 | public void setDateTimeService(DateTimeService dateTimeService) { | |
81 | 0 | this.dateTimeService = dateTimeService; |
82 | 0 | } |
83 | ||
84 | ||
85 | /** | |
86 | * This overridden method ... | |
87 | * | |
88 | * @see org.kuali.rice.kns.lookup.AbstractLookupableHelperServiceImpl#getCustomActionUrls(org.kuali.rice.kns.bo.BusinessObject, java.util.List) | |
89 | */ | |
90 | @Override | |
91 | public List<HtmlData> getCustomActionUrls(BusinessObject businessObject, | |
92 | List pkNames) { | |
93 | // TODO chris - THIS METHOD NEEDS JAVADOCS | |
94 | 0 | return super.getCustomActionUrls(businessObject, pkNames); |
95 | } | |
96 | ||
97 | ||
98 | ||
99 | ||
100 | ||
101 | ||
102 | ||
103 | /** | |
104 | * This overridden method ... | |
105 | * | |
106 | * @see org.kuali.rice.kns.lookup.AbstractLookupableHelperServiceImpl#performLookup(org.kuali.rice.kns.web.struts.form.LookupForm, java.util.Collection, boolean) | |
107 | */ | |
108 | @Override | |
109 | public Collection performLookup(LookupForm lookupForm, | |
110 | Collection resultTable, boolean bounded) { | |
111 | ||
112 | //TODO: KNS updates to make this not require code from the parent | |
113 | ||
114 | ||
115 | 0 | Map<String,String> fieldsForLookup = lookupForm.getFieldsForLookup(); |
116 | 0 | DocSearchCriteriaDTO criteria = constructCriteria(fieldsForLookup); |
117 | ||
118 | //TODO: move this into actual adapter as well | |
119 | 0 | Collection displayList=null; |
120 | 0 | DocumentSearchResultComponents components = KEWServiceLocator.getDocumentSearchService().getList(GlobalVariables.getUserSession().getPrincipalId(), criteria); |
121 | 0 | List<DocumentSearchResult> result = components.getSearchResults(); |
122 | // for (DocumentSearchResult documentSearchResult : result) { | |
123 | 0 | displayList = result;//.getResultContainers(); |
124 | // } | |
125 | ||
126 | //####BEGIN COPIED CODE######### | |
127 | 0 | setBackLocation((String) lookupForm.getFieldsForLookup().get(KNSConstants.BACK_LOCATION)); |
128 | 0 | setDocFormKey((String) lookupForm.getFieldsForLookup().get(KNSConstants.DOC_FORM_KEY)); |
129 | ||
130 | //###COMENTED OUT | |
131 | // Collection displayList; | |
132 | // // call search method to get results | |
133 | // if (bounded) { | |
134 | // displayList = getSearchResults(lookupForm.getFieldsForLookup()); | |
135 | // } | |
136 | // else { | |
137 | // displayList = getSearchResultsUnbounded(lookupForm.getFieldsForLookup()); | |
138 | // } | |
139 | //##COMENTED OUT | |
140 | ||
141 | 0 | HashMap<String,Class> propertyTypes = new HashMap<String, Class>(); |
142 | ||
143 | 0 | boolean hasReturnableRow = false; |
144 | ||
145 | 0 | List returnKeys = getReturnKeys(); |
146 | 0 | List pkNames = getBusinessObjectMetaDataService().listPrimaryKeyFieldNames(getBusinessObjectClass()); |
147 | 0 | Person user = GlobalVariables.getUserSession().getPerson(); |
148 | ||
149 | // iterate through result list and wrap rows with return url and action urls | |
150 | ||
151 | //COMMENTING THIS OUT FOR NOW | |
152 | // for (Iterator iter = displayList.iterator(); iter.hasNext();) { | |
153 | // BusinessObject element = (BusinessObject) iter.next(); | |
154 | // if(element instanceof PersistableBusinessObject){ | |
155 | // lookupForm.setLookupObjectId(((PersistableBusinessObject)element).getObjectId()); | |
156 | // } | |
157 | // BusinessObject element = null; | |
158 | // BusinessObjectRestrictions businessObjectRestrictions = getBusinessObjectAuthorizationService().getLookupResultRestrictions(element, user); | |
159 | ||
160 | // HtmlData returnUrl = getReturnUrl(element, lookupForm, returnKeys, businessObjectRestrictions); | |
161 | // String actionUrls = getActionUrls(element, pkNames, businessObjectRestrictions); | |
162 | //ADDED (4 lines) | |
163 | 0 | for (DocumentSearchResult aResult : result) |
164 | { | |
165 | //TODO: where to get these from? | |
166 | 0 | HtmlData returnUrl = new AnchorHtmlData(); |
167 | 0 | String actionUrls = ""; |
168 | ||
169 | //TODO: convert columns either here or in the getColumns method | |
170 | //ADDED (3) | |
171 | 0 | List<? extends Column> columns = components.getColumns();//getColumns(); |
172 | 0 | List<KeyValueSort> keyValues = aResult.getResultContainers(); |
173 | 0 | for (int i = 0; i < columns.size(); i++) |
174 | { | |
175 | ||
176 | // for (Iterator iterator = columns.iterator(); iterator.hasNext();) { | |
177 | ||
178 | // Column col = (Column) iterator.next(); | |
179 | //ADDED 3 | |
180 | 0 | Column col = (Column) columns.get(i); |
181 | 0 | KeyValueSort keyValue = keyValues.get(i); |
182 | //Set values from keyvalue on column | |
183 | 0 | col.setPropertyValue(keyValue.getUserDisplayValue()); |
184 | ||
185 | 0 | Formatter formatter = col.getFormatter(); |
186 | ||
187 | // pick off result column from result list, do formatting | |
188 | 0 | String propValue = KNSConstants.EMPTY_STRING; |
189 | // Object prop = ObjectUtils.getPropertyValue(element, col.getPropertyName()); | |
190 | //ADDED | |
191 | 0 | Object prop = col.getPropertyValue(); |
192 | ||
193 | // set comparator and formatter based on property type | |
194 | 0 | Class propClass = propertyTypes.get(col.getPropertyName()); |
195 | 0 | if (propClass == null) |
196 | { | |
197 | try | |
198 | { | |
199 | // propClass = ObjectUtils.getPropertyType( element, col.getPropertyName(), getPersistenceStructureService() ); | |
200 | // propertyTypes.put( col.getPropertyName(), propClass ); | |
201 | } catch (Exception e) | |
202 | { | |
203 | // throw new RuntimeException("Cannot access PropertyType for property " + "'" + col.getPropertyName() + "' " + " on an instance of '" + element.getClass().getName() + "'.", e); | |
204 | 0 | } |
205 | } | |
206 | ||
207 | // formatters | |
208 | 0 | if (prop != null) |
209 | { | |
210 | // for Booleans, always use BooleanFormatter | |
211 | 0 | if (prop instanceof Boolean) |
212 | { | |
213 | 0 | formatter = new BooleanFormatter(); |
214 | } | |
215 | ||
216 | // for Dates, always use DateFormatter | |
217 | 0 | if (prop instanceof Date) |
218 | { | |
219 | 0 | formatter = new DateFormatter(); |
220 | } | |
221 | ||
222 | // for collection, use the list formatter if a formatter hasn't been defined yet | |
223 | 0 | if (prop instanceof Collection && formatter == null) |
224 | { | |
225 | 0 | formatter = new CollectionFormatter(); |
226 | } | |
227 | ||
228 | 0 | if (formatter != null) |
229 | { | |
230 | 0 | propValue = (String) formatter.format(prop); |
231 | } else | |
232 | { | |
233 | 0 | propValue = prop.toString(); |
234 | } | |
235 | } | |
236 | ||
237 | // comparator | |
238 | 0 | col.setComparator(CellComparatorHelper.getAppropriateComparatorForPropertyClass(propClass)); |
239 | 0 | col.setValueComparator(CellComparatorHelper.getAppropriateValueComparatorForPropertyClass(propClass)); |
240 | ||
241 | // propValue = maskValueIfNecessary(element.getClass(), col.getPropertyName(), propValue, businessObjectRestrictions); | |
242 | ||
243 | 0 | col.setPropertyValue(propValue); |
244 | ||
245 | 0 | if (StringUtils.isNotBlank(propValue)) |
246 | { | |
247 | // col.setColumnAnchor(getInquiryUrl(element, col.getPropertyName())); | |
248 | //ADDED (3 lines) | |
249 | 0 | AnchorHtmlData anchor = new AnchorHtmlData(KNSConstants.EMPTY_STRING, KNSConstants.EMPTY_STRING); |
250 | //TODO: change to grab URL from config variable | |
251 | 0 | if (StringUtils.isNotEmpty(keyValue.getValue()) && StringUtils.equals("routeHeaderId", keyValue.getKey())) |
252 | { | |
253 | 0 | anchor.setHref(StringUtils.substringBetween(keyValue.getValue(), "<a href=\"", "docId=") + "docId=" + keyValue.getUserDisplayValue()); |
254 | 0 | col.setMaxLength(100); //for now force this |
255 | } | |
256 | ||
257 | 0 | col.setColumnAnchor(anchor); |
258 | } | |
259 | } | |
260 | ||
261 | 0 | ResultRow row = new ResultRow((List<Column>) columns, returnUrl.constructCompleteHtmlTag(), actionUrls); |
262 | 0 | row.setRowId(returnUrl.getName()); |
263 | // because of concerns of the BO being cached in session on the ResultRow, | |
264 | // let's only attach it when needed (currently in the case of export) | |
265 | 0 | if (getBusinessObjectDictionaryService().isExportable(getBusinessObjectClass())) |
266 | { | |
267 | // row.setBusinessObject(element); | |
268 | } | |
269 | ||
270 | // boolean rowReturnable = isResultReturnable(element); | |
271 | //ADDED | |
272 | 0 | boolean rowReturnable = true; |
273 | 0 | row.setRowReturnable(rowReturnable); |
274 | 0 | if (rowReturnable) |
275 | { | |
276 | 0 | hasReturnableRow = true; |
277 | } | |
278 | 0 | resultTable.add(row); |
279 | 0 | } |
280 | ||
281 | 0 | lookupForm.setHasReturnableRow(hasReturnableRow); |
282 | ||
283 | 0 | return displayList; |
284 | //####END COPIED CODE######### | |
285 | } | |
286 | ||
287 | ||
288 | /** | |
289 | * This method ... | |
290 | * | |
291 | * @param lookupForm | |
292 | * @return | |
293 | */ | |
294 | private DocSearchCriteriaDTO constructCriteria(Map<String,String> fieldsForLookup) { | |
295 | //TODO: move this into adapter | |
296 | 0 | DocSearchCriteriaDTO criteria = new DocSearchCriteriaDTO(); |
297 | 0 | Map<String,String> fieldsToSet = new HashMap<String,String>(); |
298 | 0 | for (String formKey : fieldsForLookup.keySet()) { |
299 | 0 | if(!(formKey.equalsIgnoreCase(KNSConstants.BACK_LOCATION) || |
300 | formKey.equalsIgnoreCase(KNSConstants.DOC_FORM_KEY)) && StringUtils.isNotEmpty(fieldsForLookup.get(formKey))) { | |
301 | 0 | fieldsToSet.put(formKey, fieldsForLookup.get(formKey)); |
302 | } | |
303 | } | |
304 | //if we use DocSearchCriteriaDTO as object we shouldn't need this conversion stuff | |
305 | 0 | for (String fieldToSet : fieldsToSet.keySet()) { |
306 | //need translation code here for certain fields | |
307 | 0 | String valueToSet = fieldsToSet.get(fieldToSet); |
308 | try { | |
309 | //TODO: temporary work around until is a criteria | |
310 | 0 | if(fieldToSet.equals("documentType.name")) { |
311 | 0 | fieldToSet = "docTypeFullName"; |
312 | } | |
313 | 0 | PropertyUtils.setNestedProperty(criteria, fieldToSet, valueToSet); |
314 | 0 | } catch (IllegalAccessException e) { |
315 | 0 | e.printStackTrace(); |
316 | 0 | } catch (InvocationTargetException e) { |
317 | 0 | e.printStackTrace(); |
318 | 0 | } catch (NoSuchMethodException e) { |
319 | 0 | e.printStackTrace(); |
320 | 0 | } |
321 | 0 | } |
322 | 0 | return criteria; |
323 | } | |
324 | ||
325 | ||
326 | /** | |
327 | * This overridden method ... | |
328 | * | |
329 | * @see org.kuali.rice.kns.lookup.AbstractLookupableHelperServiceImpl#getInquiryUrl(org.kuali.rice.kns.bo.BusinessObject, java.lang.String) | |
330 | */ | |
331 | @Override | |
332 | public HtmlData getInquiryUrl(BusinessObject bo, String propertyName) { | |
333 | //FIXME: ctk - make sure and check that it's ok to do this here. I may move this out to the doc search processor | |
334 | 0 | if(KEWPropertyConstants.DOC_SEARCH_RESULT_PROPERTY_NAME_ROUTE_HEADER_ID.equals(propertyName)) { |
335 | ||
336 | 0 | AnchorHtmlData link = new AnchorHtmlData(); |
337 | 0 | DocumentRouteHeaderValue doc = (DocumentRouteHeaderValue)bo; |
338 | //if !superuser | |
339 | 0 | Long routeHeaderId = doc.getRouteHeaderId(); |
340 | 0 | link.setDisplayText(routeHeaderId+""); |
341 | ||
342 | 0 | String href = ConfigContext.getCurrentContextConfig().getKRBaseURL()+"/"+ |
343 | KEWConstants.DOC_HANDLER_REDIRECT_PAGE + "?" + KEWConstants.COMMAND_PARAMETER + "=" + | |
344 | KEWConstants.DOCSEARCH_COMMAND + "&" + KEWConstants.ROUTEHEADER_ID_PARAMETER + "=" + routeHeaderId; | |
345 | 0 | link.setHref(href); |
346 | ||
347 | 0 | return link; |
348 | } | |
349 | ||
350 | 0 | return super.getInquiryUrl(bo, propertyName); |
351 | } | |
352 | ||
353 | ||
354 | /** | |
355 | * This overridden method ... | |
356 | * | |
357 | * @see org.kuali.rice.kns.lookup.AbstractLookupableHelperServiceImpl#setRows() | |
358 | */ | |
359 | @Override | |
360 | protected void setRows() { | |
361 | // TODO chris - this method should call the criteria processor adapter which will | |
362 | //call the criteria processor (either standard or custom) and massage the data into the proper format | |
363 | //this is called by setbo in super(which is called by form) so should be called when the page needs refreshing | |
364 | ||
365 | //TODO: move over code that checks for doctype (actually should that be in the refresh, since that's where the doc type will be coming back to?) | |
366 | ||
367 | ||
368 | //###START LOOKUP ROW CODE Not sure if we need these but they may be valuable for eventually forcing all standard field customization in the xml | |
369 | 0 | super.setRows(); |
370 | 0 | List<Row> lookupRows = new ArrayList<Row>(); |
371 | //copy the current rows | |
372 | 0 | for (Row row : super.getRows()) { |
373 | 0 | lookupRows.add(row); |
374 | } | |
375 | //clear out | |
376 | 0 | super.getRows().clear(); |
377 | //###END LOOKUP ROW CODE TODO: do something with lookupRows or delete above code | |
378 | ||
379 | ||
380 | 0 | processor = new DocumentLookupCriteriaProcessorKEWAdapter(); |
381 | ||
382 | ||
383 | ||
384 | //TODO: get this from parameters (or bo?) | |
385 | 0 | DocumentType docType = null; |
386 | 0 | if(processor != null) { |
387 | //TODO: same Doc type | |
388 | // if(docType==oldDocType) { | |
389 | 0 | ((DocumentLookupCriteriaProcessorKEWAdapter)processor).setCriteriaProcessor(new StandardDocumentSearchCriteriaProcessor()); |
390 | // } else { | |
391 | ||
392 | // } | |
393 | } else { | |
394 | //TODO: same Doc type | |
395 | 0 | if(docType==null) { |
396 | 0 | ((DocumentLookupCriteriaProcessorKEWAdapter)processor).setCriteriaProcessor(new StandardDocumentSearchCriteriaProcessor()); |
397 | }// else { | |
398 | ||
399 | // } | |
400 | } | |
401 | ||
402 | //call get rows | |
403 | 0 | List<Row> rows = processor.getRows(docType,lookupRows,false,false); |
404 | 0 | super.getRows().addAll(rows); |
405 | ||
406 | //are we in basic or detailed, are we in super. | |
407 | //TODO: ctk Add this code back in when KNS impacting changes are worked back | |
408 | ||
409 | ||
410 | ||
411 | ||
412 | ||
413 | 0 | } |
414 | ||
415 | ||
416 | /** | |
417 | * This overridden method allows for overriding what the clear logic does. | |
418 | * | |
419 | * @see org.kuali.rice.kns.lookup.AbstractLookupableHelperServiceImpl#performClear() | |
420 | */ | |
421 | @Override | |
422 | public void performClear(LookupForm lookupForm) { | |
423 | 0 | Map<String,String> fieldsToClear = new HashMap<String,String>(); |
424 | ||
425 | 0 | for (Row row : this.getRows()) { |
426 | 0 | for (Field field : row.getFields()) { |
427 | 0 | fieldsToClear.put(field.getPropertyName(), field.getPropertyValue()); |
428 | } | |
429 | } | |
430 | //TODO: also check if standard here (maybe from object if use criteria) | |
431 | 0 | if(this.processor==null) { |
432 | 0 | super.performClear(lookupForm); |
433 | } else { | |
434 | 0 | DocSearchCriteriaDTO docCriteria = constructCriteria(fieldsToClear); |
435 | 0 | if(StringUtils.isNotEmpty("documentType.name")) { |
436 | //TODO: move these strings to constants (probably after moving this to criteria | |
437 | 0 | getValidDocumentType(fieldsToClear.get("documentType.name")).getDocumentSearchGenerator().clearSearch(docCriteria);; |
438 | } | |
439 | } | |
440 | 0 | } |
441 | /** | |
442 | * | |
443 | * This method is taken from DocSearch to retrieve a document type | |
444 | * | |
445 | * @param docTypeName | |
446 | * @return | |
447 | */ | |
448 | private static DocumentType getValidDocumentType(String docTypeName) { | |
449 | 0 | DocumentType documentType = KEWServiceLocator.getDocumentTypeService().findByName(docTypeName); |
450 | 0 | if (documentType == null) { |
451 | 0 | throw new RuntimeException("Document Type invalid : " + docTypeName); |
452 | } | |
453 | 0 | return documentType; |
454 | } | |
455 | } |