Classes in this File | Line Coverage | Branch Coverage | Complexity | ||||
Formatter |
|
| 3.9642857142857144;3.964 |
1 | /* | |
2 | * Copyright 2004 Jonathan M. Lehr | |
3 | * | |
4 | * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. | |
5 | * You may obtain a copy of the License at | |
6 | * | |
7 | * http://www.apache.org/licenses/LICENSE-2.0 | |
8 | * | |
9 | * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" | |
10 | * BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language | |
11 | * governing permissions and limitations under the License. | |
12 | * | |
13 | * MODIFIED BY THE KUALI FOUNDATION | |
14 | */ | |
15 | // begin Kuali Foundation modification | |
16 | package org.kuali.rice.kns.web.format; | |
17 | // end Kuali Foundation modification | |
18 | ||
19 | import java.io.Serializable; | |
20 | import java.lang.reflect.Array; | |
21 | import java.math.BigDecimal; | |
22 | import java.sql.Date; | |
23 | import java.sql.Timestamp; | |
24 | import java.util.ArrayList; | |
25 | import java.util.Collection; | |
26 | import java.util.Collections; | |
27 | import java.util.HashMap; | |
28 | import java.util.HashSet; | |
29 | import java.util.Iterator; | |
30 | import java.util.List; | |
31 | import java.util.Map; | |
32 | import java.util.Set; | |
33 | ||
34 | import javax.servlet.http.HttpServletRequest; | |
35 | ||
36 | import org.apache.commons.beanutils.PropertyUtils; | |
37 | import org.apache.commons.lang.StringUtils; | |
38 | import org.kuali.rice.kns.util.AbstractKualiDecimal; | |
39 | import org.kuali.rice.kns.util.KualiDecimal; | |
40 | import org.kuali.rice.kns.util.KualiInteger; | |
41 | import org.kuali.rice.kns.util.KualiPercent; | |
42 | import org.kuali.rice.kns.web.struts.pojo.ArrayUtils; | |
43 | ||
44 | ||
45 | // begin Kuali Foundation modification | |
46 | /** | |
47 | * This is the base class for all other Formatters. | |
48 | */ | |
49 | /** | |
50 | * It provides default formatting and conversion behavior for most value types, including primitives, arrays, and instances of most | |
51 | * {@link Collection}types. <code>Formatter</code> and its subclasses were designed primarily to be used by web app framework | |
52 | * components, though they can also be used in other contexts. | |
53 | * <p> | |
54 | * During request processing, the {@link PojoActionForm}uses <code>Formatter</code> instances to convert inbound request values | |
55 | * to JavaBean property types. Whenever a given value cannot be converted to its target type, the conversion method | |
56 | * {@link PropertyUtils#getProperty(Object, String)}throws a {@link FormatException}to signal this condition to the | |
57 | * calling code. | |
58 | * <p> | |
59 | * During the response phase, Struts tags make calls to the {@link PojoRequestProcessor}in order to access bean property values. | |
60 | * The <code>PojoRequestProcessor</code> then uses <code>Formatter</code> instances to format the bean values for presentation | |
61 | * in the user interface. | |
62 | * <p> | |
63 | * In either case, <code>Formatter</code> instances are obtained by calling {@link #getFormatter(Class)}, which looks in an | |
64 | * internal registry to determine which <code>Formatter</code> class to instantiate, and returns a new instance. The StrutsLive | |
65 | * framework includes a number of <code>Formatter</code> classes that are registered statically; additional | |
66 | * <code>Formatter classes can be registered at compile | |
67 | * time or at run time. | |
68 | * <p> | |
69 | * Subclasses of <code>Formatter</code> typically override the callback methods | |
70 | * {@link #convertToObject(String)} and {@link #formatObject(Object)}, which | |
71 | * otherwise provide default conversion and formmating behavior needed for | |
72 | * atomic values (i.e., an ordinary bean property such as a <code>String</code> | |
73 | * or <code>Integer</code>, or else an element of a property typed as | |
74 | * array or Collection). | |
75 | * | |
76 | * @see PojoActionForm#populate(HttpServletRequest) | |
77 | * @see PojoPropertyUtilsBean#getProperty(Object, String) | |
78 | */ | |
79 | // end Kuali Foundation modification | |
80 | 0 | public class Formatter implements Serializable { |
81 | // begin Kuali Foundation modification | |
82 | // removed serialVersionUID and logger members | |
83 | // end Kuali Foundation modification | |
84 | ||
85 | static final String CREATE_MSG = "Couldn't create an instance of class "; | |
86 | // begin Kuali Foundation modification | |
87 | // registry changed from AppLocal instance to a Map | |
88 | 0 | private static Map registry = Collections.synchronizedMap(new HashMap()); |
89 | // end Kuali Foundation modification | |
90 | ||
91 | protected Map settings; | |
92 | ||
93 | // begin Kuali Foundation modification | |
94 | // removed keypath and rootObject variables | |
95 | // end Kuali Foundation modification | |
96 | ||
97 | protected Class propertyType; | |
98 | ||
99 | static { | |
100 | // begin Kuali Foundation modification | |
101 | 0 | registerFormatter(String.class, Formatter.class); |
102 | 0 | registerFormatter(String[].class, Formatter.class); |
103 | 0 | registerFormatter(AbstractKualiDecimal.class, BigDecimalFormatter.class); |
104 | 0 | registerFormatter(KualiDecimal.class, CurrencyFormatter.class); |
105 | 0 | registerFormatter(KualiInteger.class, KualiIntegerCurrencyFormatter.class); |
106 | 0 | registerFormatter(KualiPercent.class, PercentageFormatter.class); |
107 | 0 | registerFormatter(BigDecimal.class, BigDecimalFormatter.class); |
108 | 0 | registerFormatter(Date.class, DateFormatter.class); |
109 | 0 | registerFormatter(Integer.class, IntegerFormatter.class); |
110 | 0 | registerFormatter(int.class, IntegerFormatter.class); |
111 | 0 | registerFormatter(int[].class, IntegerFormatter.class); |
112 | 0 | registerFormatter(Boolean.class, BooleanFormatter.class); |
113 | 0 | registerFormatter(Boolean.TYPE, BooleanFormatter.class); |
114 | 0 | registerFormatter(boolean[].class, BooleanFormatter.class); |
115 | 0 | registerFormatter(Long.class, LongFormatter.class); |
116 | 0 | registerFormatter(Timestamp.class, DateViewTimestampObjectFormatter.class); |
117 | 0 | registerFormatter(boolean.class, LittleBooleanFormatter.class); |
118 | 0 | registerFormatter(Collection.class, ArrayFormatter.class); |
119 | // end Kuali Foundation modification | |
120 | 0 | } |
121 | ||
122 | public static Formatter getFormatter(Class aType) { | |
123 | 0 | return getFormatter(aType, null); |
124 | } | |
125 | ||
126 | // begin Kuali Foundation modification | |
127 | // param aType was valueType, comment changes, major code changes | |
128 | /** | |
129 | * Returns an instance of the Formatter class to be used to format the provided value type. | |
130 | * | |
131 | * @param type the class of the value to be formatted | |
132 | * @param settings parameters used by subclasses to customize behavior | |
133 | * @return an instance of Formatter or one of its subclasses | |
134 | */ | |
135 | public static Formatter getFormatter(Class aType, Map settings) { | |
136 | // original code: return createFormatter(formatterForType(valueType), valueType, settings); | |
137 | ||
138 | 0 | Class type = formatterForType(aType); |
139 | 0 | Formatter formatter = null; |
140 | try { | |
141 | 0 | formatter = (Formatter) type.newInstance(); |
142 | } | |
143 | 0 | catch (InstantiationException e) { |
144 | 0 | throw new FormatException(CREATE_MSG + type, e); |
145 | } | |
146 | 0 | catch (IllegalAccessException e) { |
147 | 0 | throw new FormatException(CREATE_MSG + type, e); |
148 | 0 | } |
149 | ||
150 | 0 | if (settings != null) |
151 | 0 | formatter.setSettings(Collections.unmodifiableMap(settings)); |
152 | 0 | formatter.propertyType = aType; |
153 | ||
154 | 0 | return formatter; |
155 | } | |
156 | ||
157 | // removed getFormatterByName, formatterClassForName, createFormatter methods | |
158 | // end Kuali Foundation modification | |
159 | ||
160 | /** | |
161 | * Binds the provided value type to a Formatter type. Note that a single Formatter class can be associated with more than one | |
162 | * type. | |
163 | * | |
164 | * @param type a value type | |
165 | * @param formatterType a Formatter type | |
166 | */ | |
167 | public static void registerFormatter(Class type, Class formatterType) { | |
168 | 0 | registry.put(type, formatterType); |
169 | 0 | } |
170 | ||
171 | /** | |
172 | * Returns <code>true</code> if the provided class is an array type, implements either the {@link List}or {@link Set} | |
173 | * interfaces, or is one of the Formatter classes currently registered. | |
174 | * | |
175 | * @see registerFormatter(Class, Class) | |
176 | */ | |
177 | public static boolean isSupportedType(Class type) { | |
178 | // begin Kuali Foundation modification | |
179 | 0 | if (type == null) |
180 | 0 | return false; |
181 | // end Kuali Foundation modification | |
182 | 0 | if (List.class.isAssignableFrom(type)) |
183 | 0 | return true; |
184 | 0 | if (Set.class.isAssignableFrom(type)) |
185 | 0 | return true; |
186 | ||
187 | 0 | return findFormatter(type) != null; |
188 | } | |
189 | ||
190 | /** | |
191 | * Return the Formatter associated with the given type, by consulting an internal registry. Additional associations can be made | |
192 | * by calling {@link registerFormatter(Class, Class)}. | |
193 | * | |
194 | * @return a new Formatter instance | |
195 | */ | |
196 | public static Class formatterForType(Class type) { | |
197 | 0 | if (type == null) |
198 | 0 | throw new IllegalArgumentException("Type can not be null"); |
199 | ||
200 | 0 | Class formatterType = findFormatter(type); |
201 | ||
202 | 0 | return formatterType == null ? Formatter.class : formatterType; |
203 | } | |
204 | ||
205 | // Kuali Foundation modification: comment removed | |
206 | public static Class findFormatter(Class type) { | |
207 | // begin Kuali Foundation modification | |
208 | 0 | if (type == null) |
209 | 0 | return null; |
210 | ||
211 | 0 | if (registry.containsKey(type)) { |
212 | 0 | return (Class) registry.get(type); |
213 | } | |
214 | ||
215 | ||
216 | 0 | Iterator typeIter = registry.keySet().iterator(); |
217 | 0 | while (typeIter.hasNext()) { |
218 | 0 | Class currType = (Class) typeIter.next(); |
219 | 0 | if (currType.isAssignableFrom(type)) { |
220 | 0 | Class currFormatter = (Class) registry.get(currType); |
221 | 0 | registerFormatter(type, currFormatter); |
222 | 0 | return currFormatter; |
223 | } | |
224 | 0 | } |
225 | ||
226 | 0 | return null; |
227 | // end Kuali Foundation modification | |
228 | } | |
229 | ||
230 | // begin Kuali Foundation modification | |
231 | public String getImplementationClass() { | |
232 | 0 | return this.getClass().getName(); |
233 | } | |
234 | // end Kuali Foundation modification | |
235 | ||
236 | public Class getPropertyType() { | |
237 | 0 | return propertyType; |
238 | } | |
239 | ||
240 | public void setPropertyType(Class propertyType) { | |
241 | 0 | this.propertyType = propertyType; |
242 | 0 | } |
243 | ||
244 | public Map getSettings() { | |
245 | 0 | return settings; |
246 | } | |
247 | ||
248 | public void setSettings(Map settings) { | |
249 | 0 | this.settings = settings; |
250 | 0 | } |
251 | ||
252 | // begin Kuali Foundation modification | |
253 | // removed getKeypath, setKeyPath, getRootObject, setRootObject, hasSettingForKey, settingForKey, typeForKey, getErrorKey | |
254 | // end Kuali Foundation modification | |
255 | ||
256 | /** | |
257 | * begin Kuali Foundation modification | |
258 | * Returns a String representation of the given value. May be overridden by subclasses to provide customized behavior for | |
259 | * different types, though generally the callback method {@link #format(Object)}provides a better customization hook. | |
260 | * <p> | |
261 | * Provides default handling for properties typed as array or Collection. Subclass implementations of this method must invoke | |
262 | * <code>super.formatForPresentation()</code> to take advantage of this built-in behavior. | |
263 | * <p> | |
264 | * Delegates to callback method {@link formatObject}for all other types. This method in turn invokes the callback method | |
265 | * <code>format</code>, which serves as an extension point for subclasses; the default implementation simply returns its | |
266 | * argument. Overriding <code>format</code> allows subclasses to take advantage of all of the array, primitive type, and | |
267 | * Collection handling functionality provided by the base class. | |
268 | * | |
269 | * @param value the object to be formatted | |
270 | * @return a formatted string representation of the given object | |
271 | * @see #formatObject(Object) | |
272 | * end Kuali Foundation modification | |
273 | */ | |
274 | public Object formatForPresentation(Object value) { | |
275 | 0 | if (isNullValue(value)) |
276 | 0 | return formatNull(); |
277 | ||
278 | // begin Kuali Foundation modification | |
279 | // removed code | |
280 | /* | |
281 | // TODO: add registry for non-navigable classes so there's a way to | |
282 | // disable formatting selectively for given types contained in arrays | |
283 | // or Collections. | |
284 | if (Collection.class.isAssignableFrom(value.getClass())) | |
285 | return formatCollection((Collection) value); | |
286 | | |
287 | if (propertyType != null && propertyType.isArray()) | |
288 | return formatArray(value); | |
289 | */ | |
290 | // end Kuali Foundation modification | |
291 | ||
292 | 0 | return formatObject(value); |
293 | } | |
294 | ||
295 | /** | |
296 | * May be overridden by subclasses to provide special handling for <code>null</code> values when formatting a bean property | |
297 | * value for presentation. The default implementation simply returns <code>null</code> | |
298 | */ | |
299 | protected Object formatNull() { | |
300 | 0 | return null; |
301 | } | |
302 | ||
303 | /** | |
304 | * May be overridden by subclasses to provide custom formatting behavior. Provides default formatting implementation for | |
305 | * primitive types. (Note that primitive types are will always be wrapped in an array in order to be passed as an argument of | |
306 | * type <code>Object</code>). | |
307 | */ | |
308 | public Object formatObject(Object value) { | |
309 | 0 | if (value == null) |
310 | 0 | return formatNull(); |
311 | ||
312 | // Collections and arrays have already been handled at this point, so | |
313 | // if value is an array, assume it's a wrapper for a primitive type. | |
314 | 0 | Class type = value.getClass(); |
315 | 0 | if (type.isArray()) |
316 | // begin Kuali Foundation modification | |
317 | 0 | return ArrayUtils.toString(value, type.getComponentType()); |
318 | // end begin Kuali Foundation modification | |
319 | ||
320 | 0 | if (!(isSupportedType(value.getClass()))) |
321 | // begin Kuali Foundation modification | |
322 | 0 | formatBean(value); |
323 | // end Kuali Foundation modification | |
324 | ||
325 | 0 | return format(value); |
326 | } | |
327 | ||
328 | /** | |
329 | * If an element of the Collection isn't a supported type, assume it's a JavaBean, and format each of its properties. Returns a | |
330 | * Map containing the formatted properties keyed by property name. | |
331 | */ | |
332 | protected Object formatBean(Object bean) { | |
333 | 0 | Map properties = null; |
334 | try { | |
335 | // begin Kuali Foundation modification | |
336 | 0 | properties = PropertyUtils.describe(bean); |
337 | // end Kuali Foundation modification | |
338 | } | |
339 | 0 | catch (Exception e) { |
340 | 0 | throw new FormatException("Unable to format values for bean " + bean, e); |
341 | 0 | } |
342 | ||
343 | 0 | Map formattedVals = new HashMap(); |
344 | // begin Kuali Foundation modification | |
345 | 0 | Iterator propIter = properties.entrySet().iterator(); |
346 | ||
347 | 0 | while (propIter.hasNext()) { |
348 | 0 | Map.Entry entry = (Map.Entry) propIter.next(); |
349 | 0 | Object value = entry.getValue(); |
350 | 0 | if (value != null && isSupportedType(value.getClass())) { |
351 | 0 | Formatter formatter = getFormatter(value.getClass()); |
352 | 0 | formattedVals.put(entry.getKey(), formatter.formatForPresentation(value)); |
353 | } | |
354 | 0 | } |
355 | // end Kuali Foundation modification | |
356 | 0 | return formattedVals; |
357 | } | |
358 | ||
359 | public Object format(Object value) { | |
360 | 0 | return value; |
361 | } | |
362 | ||
363 | public Object formatArray(Object value) { | |
364 | // begin Kuali Foundation modification | |
365 | 0 | Class elementType = value.getClass().getComponentType(); |
366 | 0 | if (!isSupportedType(elementType)) |
367 | 0 | return value; |
368 | ||
369 | 0 | int length = Array.getLength(value); |
370 | 0 | Object[] formattedVals = new String[length]; |
371 | ||
372 | 0 | for (int i = 0; i < length; i++) { |
373 | 0 | Object element = Array.get(value, i); |
374 | 0 | Object objValue = ArrayUtils.toObject(element); |
375 | 0 | Formatter elementFormatter = getFormatter(elementType); |
376 | 0 | formattedVals[i] = elementFormatter.formatForPresentation(objValue); |
377 | } | |
378 | ||
379 | 0 | return formattedVals; |
380 | // end Kuali Foundation modification | |
381 | } | |
382 | ||
383 | public Object formatCollection(Collection value) { | |
384 | 0 | List stringVals = new ArrayList(); |
385 | 0 | Iterator iter = value.iterator(); |
386 | 0 | while (iter.hasNext()) { |
387 | 0 | Object obj = iter.next(); |
388 | 0 | Formatter formatter = getFormatter(obj.getClass()); |
389 | // begin Kuali Foundation modification | |
390 | 0 | stringVals.add(formatter.formatForPresentation(obj)); |
391 | // end Kuali Foundation modification | |
392 | 0 | } |
393 | 0 | return stringVals.toArray(); |
394 | } | |
395 | ||
396 | /** | |
397 | * Returns an object representation of the provided string after first removing any extraneous formatting characters. If the | |
398 | * argument is a native array wrapping the actual value, the value is removed (unwrapped) from the array prior to invoking the | |
399 | * callback method {@link #convertToObject(String)}, which performs the actual conversion. | |
400 | * <p> | |
401 | * If the provided object is <code>null</code>, a blank <code>String</code>, or a <code>String[]</code> of length <b>0 | |
402 | * </b> or that has <code>null</code> or a blank <code>String</code> in the first position, returns <code>null</code>. | |
403 | * Otherwise, If the destination property is a <code>Collection</code>, returns an instance of that type containing the | |
404 | * string values of the array elements. | |
405 | * <p> | |
406 | * If the provided object is an array, uses a Formatter corresponding to the array's component type to convert each of its | |
407 | * elements, and returns a new array containing the converted values. | |
408 | * | |
409 | * May be overidden by subclasses to customize conversion, though ordinarily {@link #convertToObject(String)}is a better choice | |
410 | * since it takes advantage of <code>convertFromPresentationFormat</code>'s built-in behavior. | |
411 | * | |
412 | * @param value the string value to be converted | |
413 | * @return the object value corresponding to the provided string value | |
414 | * @see convertToObject(String) | |
415 | */ | |
416 | public Object convertFromPresentationFormat(Object value) { | |
417 | 0 | if (isEmptyValue(value)) |
418 | 0 | return getNullObjectValue(); |
419 | ||
420 | 0 | Class type = value.getClass(); |
421 | 0 | boolean isArray = propertyType != null && propertyType.isArray(); |
422 | 0 | boolean isCollection = propertyType != null && Collection.class.isAssignableFrom(propertyType); |
423 | ||
424 | 0 | if (!(isArray || isCollection)) { |
425 | 0 | value = unwrapString(value); |
426 | 0 | return convertToObject((String) value); |
427 | } | |
428 | ||
429 | 0 | String[] strings = type.isArray() ? (String[]) value : new String[] { (String) value }; |
430 | ||
431 | 0 | return isArray ? convertToArray(strings) : convertToCollection(strings); |
432 | } | |
433 | ||
434 | /** | |
435 | * May be overridden by subclasses to provide special handling for <code>null</code> values when converting from presentation | |
436 | * format to a bean property type. The default implementation simply returns <code>null</code> | |
437 | */ | |
438 | protected Object getNullObjectValue() { | |
439 | 0 | return null; |
440 | } | |
441 | ||
442 | /** | |
443 | * May be orverridden by subclasses to customize its behavior. The default implementation simply trims and returns the provided | |
444 | * string. | |
445 | */ | |
446 | protected Object convertToObject(String string) { | |
447 | 0 | return string == null ? null : string.replace( "\r\n", "\n" ).trim(); |
448 | } | |
449 | ||
450 | /** | |
451 | * Converts an array of strings to a Collection type corresponding to the value of <code>propertyType</code>. Since we don't | |
452 | * have type information for the elements of the collection, no attempt is made to convert the elements from <code>String</code> | |
453 | * to other types. However, subclasses can override this method if they need to provide the ability to convert the elements to a | |
454 | * given type. | |
455 | */ | |
456 | protected Collection convertToCollection(String[] strings) { | |
457 | 0 | Collection collection = null; |
458 | 0 | Class type = propertyType; |
459 | ||
460 | 0 | if (propertyType.isAssignableFrom(List.class)) |
461 | 0 | type = ArrayList.class; |
462 | 0 | else if (propertyType.isAssignableFrom(Set.class)) |
463 | 0 | type = HashSet.class; |
464 | ||
465 | try { | |
466 | 0 | collection = (Collection) type.newInstance(); |
467 | } | |
468 | 0 | catch (Exception e) { |
469 | 0 | throw new FormatException(CREATE_MSG + propertyType, e); |
470 | 0 | } |
471 | ||
472 | 0 | for (int i = 0; i < strings.length; i++) |
473 | 0 | collection.add(strings[i]); |
474 | ||
475 | 0 | return collection; |
476 | } | |
477 | ||
478 | /** | |
479 | * Converts an array of strings to an array of objects by calling {@link #convertToObject(String)}on each element of the | |
480 | * provided array in turn, using instances of a Formatter class that corresponds to this Formatter's property type. | |
481 | * | |
482 | * @see #propertyType | |
483 | */ | |
484 | protected Object convertToArray(String[] strings) { | |
485 | 0 | Class type = propertyType.getComponentType(); |
486 | // begin Kuali Foundation modification | |
487 | 0 | Formatter formatter = getFormatter(type); |
488 | // end Kuali Foundation modification | |
489 | 0 | Object array = null; |
490 | try { | |
491 | 0 | array = Array.newInstance(type, strings.length); |
492 | } | |
493 | 0 | catch (Exception e) { |
494 | 0 | throw new FormatException(CREATE_MSG + type, e); |
495 | 0 | } |
496 | ||
497 | 0 | for (int i = 0; i < strings.length; i++) { |
498 | 0 | Object value = formatter.convertToObject(strings[i]); |
499 | // begin Kuali Foundation modification | |
500 | 0 | ArrayUtils.setArrayValue(array, type, value, i); |
501 | // end Kuali Foundation modification | |
502 | } | |
503 | ||
504 | 0 | return array; |
505 | } | |
506 | ||
507 | public static String unwrapString(Object target) { | |
508 | ||
509 | 0 | if (target.getClass().isArray()) { |
510 | 0 | String wrapper[] = (String[]) target; |
511 | 0 | return wrapper.length > 0 ? wrapper[0] : null; |
512 | } | |
513 | // begin Kuali Foundation modification | |
514 | // if target object is null, return a null String | |
515 | 0 | else if (target == null) { |
516 | 0 | return new String(); |
517 | } | |
518 | ||
519 | // otherwise, return the string value of the object, with the hope | |
520 | // that the toString() has been meaningfully overriden | |
521 | else { | |
522 | 0 | return target.toString(); |
523 | } | |
524 | // end Kuali Foundation modification | |
525 | } | |
526 | ||
527 | public static boolean isNullValue(Object obj) { | |
528 | 0 | if (obj == null) |
529 | 0 | return true; |
530 | ||
531 | // begin Kuali Foundation modification | |
532 | 0 | if ((obj instanceof String) && StringUtils.isEmpty((String) obj)) |
533 | 0 | return true; |
534 | // end Kuali Foundation modification | |
535 | ||
536 | 0 | return false; |
537 | } | |
538 | ||
539 | public static boolean isEmptyValue(Object obj) { | |
540 | 0 | if (obj == null) |
541 | 0 | return true; |
542 | // begin Kuali Foundation modification | |
543 | 0 | if ((obj instanceof String) && StringUtils.isEmpty((String) obj)) |
544 | 0 | return true; |
545 | // end Kuali Foundation modification | |
546 | 0 | Class type = obj.getClass(); |
547 | 0 | if (type.isArray()) { |
548 | 0 | Class compType = type.getComponentType(); |
549 | 0 | if (compType.isPrimitive()) |
550 | 0 | return false; |
551 | 0 | if (((Object[]) obj).length == 0) |
552 | 0 | return true; |
553 | 0 | if (((Object[]) obj)[0] == null) |
554 | 0 | return true; |
555 | 0 | if (String.class.isAssignableFrom(compType)) { |
556 | // begin Kuali Foundation modification | |
557 | 0 | return StringUtils.isEmpty(((String[]) obj)[0]); |
558 | // end Kuali Foundation modification | |
559 | } | |
560 | } | |
561 | 0 | return false; |
562 | } | |
563 | ||
564 | protected String trimString(Object target) { | |
565 | 0 | String stringValue = null; |
566 | try { | |
567 | 0 | stringValue = (String) target; |
568 | } | |
569 | 0 | catch (ClassCastException e) { |
570 | 0 | throw new FormatException("Can't cast " + target + " to String", e); |
571 | 0 | } |
572 | 0 | return stringValue == null ? null : stringValue.trim(); |
573 | } | |
574 | ||
575 | /** | |
576 | * @deprecated in favor of {@link StringUtils#isEmptyString(String)} | |
577 | */ | |
578 | protected boolean isBlank(String string) { | |
579 | 0 | return string == null || string.trim().length() == 0; |
580 | } | |
581 | } |