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