View Javadoc

1   /**
2    * Copyright 2005-2013 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.uif.util;
17  
18  import org.apache.commons.lang.StringEscapeUtils;
19  import org.apache.commons.lang.StringUtils;
20  import org.apache.commons.lang.math.NumberUtils;
21  import org.kuali.rice.core.api.util.type.TypeUtils;
22  import org.springframework.util.Assert;
23  
24  import java.beans.PropertyDescriptor;
25  import java.util.ArrayList;
26  import java.util.List;
27  import java.util.Map;
28  import java.util.Set;
29  
30  /**
31   * Utility class for generating JavaScript
32   *
33   * @author Kuali Rice Team (rice.collab@kuali.org)
34   */
35  public class ScriptUtils {
36  
37      /**
38       * Translates an Object to a String for representing the given Object as
39       * a JavaScript value
40       *
41       * <p>
42       * Handles null, List, Map, and Set collections, along with non quoting for numeric and
43       * boolean types. Complex types are treated as a String value using toString
44       * </p>
45       *
46       * @param value Object instance to translate
47       * @return JS value
48       */
49      public static String translateValue(Object value) {
50          String jsValue = "";
51  
52          if (value == null) {
53              jsValue = "null";
54              return jsValue;
55          }
56  
57          if (value instanceof List) {
58              jsValue = "[";
59  
60              List<Object> list = (List<Object>) value;
61              for (Object listItem : list) {
62                  jsValue += translateValue(listItem);
63                  jsValue += ",";
64              }
65              jsValue = StringUtils.removeEnd(jsValue, ",");
66  
67              jsValue += "]";
68          } else if (value instanceof Set) {
69              jsValue = "[";
70  
71              Set<Object> set = (Set<Object>) value;
72              for (Object setItem : set) {
73                  jsValue += translateValue(setItem);
74                  jsValue += ",";
75              }
76              jsValue = StringUtils.removeEnd(jsValue, ",");
77  
78              jsValue += "]";
79          } else if (value instanceof Map) {
80              jsValue = "{";
81  
82              Map<Object, Object> map = (Map<Object, Object>) value;
83              for (Map.Entry<Object, Object> mapEntry : map.entrySet()) {
84                  jsValue += "\"" + mapEntry.getKey().toString() + "\":";
85                  jsValue += translateValue(mapEntry.getValue());
86                  jsValue += ",";
87              }
88              jsValue = StringUtils.removeEnd(jsValue, ",");
89  
90              jsValue += "}";
91          } else {
92              Class<?> valueClass = value.getClass();
93              if (TypeUtils.isSimpleType(valueClass) || TypeUtils.isClassClass(valueClass)) {
94                  boolean quoteValue = true;
95  
96                  if (TypeUtils.isBooleanClass(valueClass) ||
97                          TypeUtils.isDecimalClass(valueClass) ||
98                          TypeUtils.isIntegralClass(valueClass)) {
99                      quoteValue = false;
100                 }
101 
102                 if (quoteValue) {
103                     jsValue = "\"";
104                 }
105 
106                 // TODO: should this go through property editors?
107                 jsValue += value.toString();
108 
109                 if (quoteValue) {
110                     jsValue += "\"";
111                 }
112             } else {
113                 // treat as data object
114                 jsValue = "{";
115 
116                 PropertyDescriptor[] propertyDescriptors = ObjectPropertyUtils.getPropertyDescriptors(value);
117                 for (int i = 0; i < propertyDescriptors.length; i++) {
118                     PropertyDescriptor descriptor = propertyDescriptors[i];
119                     if ((descriptor.getReadMethod() != null) && !"class".equals(descriptor.getName())) {
120                         Object propertyValue = ObjectPropertyUtils.getPropertyValue(value, descriptor.getName());
121 
122                         jsValue += descriptor.getName() + ":";
123                         jsValue += translateValue(propertyValue);
124                         jsValue += ",";
125                     }
126                 }
127                 jsValue = StringUtils.removeEnd(jsValue, ",");
128 
129                 jsValue += "}";
130             }
131         }
132 
133         return jsValue;
134     }
135 
136     /**
137      * Builds a JSON string form the given map
138      *
139      * @param map map to translate
140      * @return String in JSON format
141      */
142     public static String toJSON(Map<String, String> map) {
143         StringBuffer sb = new StringBuffer("{");
144 
145         for (String key : map.keySet()) {
146             String optionValue = map.get(key);
147             if (sb.length() > 1) {
148                 sb.append(",");
149             }
150             sb.append("\"" + key + "\"");
151 
152             sb.append(":");
153             sb.append("\"" + escapeJSONString(optionValue) + "\"");
154         }
155         sb.append("}");
156 
157         return sb.toString();
158     }
159 
160     /**
161      * Escapes double quotes present in the given string
162      *
163      * @param jsonString string to escape
164      * @return escaped string
165      */
166     public static String escapeJSONString(String jsonString) {
167         if (jsonString != null) {
168             jsonString = jsonString.replace("\"", "&quot;");
169         }
170 
171         return jsonString;
172     }
173 
174     /**
175      * Converts a map of string values to a json equivalent by converting the string values through the
176      * convertToJsValue(String) method; this will output a string representation of the map
177      * in json format
178      *
179      * @param stringMap the map of String values to convert to a simple json object representation
180      * @return the json object as a string (for use in js)
181      */
182     public static String convertToJsValue(Map<String,String> stringMap){
183         if (stringMap == null || stringMap.isEmpty()){
184             return "{}";
185         }
186 
187         String convertedValue = "{";
188         for (String key: stringMap.keySet()){
189             convertedValue = convertedValue + "\"" + key + "\":" + convertToJsValue(stringMap.get(key)) + ",";
190         }
191 
192         convertedValue = StringUtils.removeEnd(convertedValue, ",");
193         convertedValue = convertedValue + "}";
194 
195         return convertedValue;
196     }
197 
198     /**
199      * Convert a string to a javascript value - especially for use for options used to initialize widgets such as the
200      * tree and rich table
201      *
202      * @param value the string to be converted
203      * @return the converted value
204      */
205     public static String convertToJsValue(String value) {
206         boolean isNumber = false;
207 
208         // save input value to preserve any whitespace formatting
209         String originalValue = value;
210 
211         // remove whitespace for correct string matching
212         value = StringUtils.strip(value);
213         isNumber = NumberUtils.isNumber(value);
214 
215         // If an option value starts with { or [, it would be a nested value
216         // and it should not use quotes around it
217         if (StringUtils.startsWith(value, "{") || StringUtils.startsWith(value, "[")) {
218             return originalValue;
219         }
220         // need to be the base boolean value "false" is true in js - a non
221         // empty string
222         else if (value.equalsIgnoreCase("false") || value.equalsIgnoreCase("true")) {
223             return originalValue;
224         }
225         // if it is a call back function, do not add the quotes
226         else if (StringUtils.startsWith(value, "function") && StringUtils.endsWith(value, "}")) {
227             return originalValue;
228         }
229         // for numerics
230         else if (isNumber) {
231             return originalValue;
232         } else {
233             // use single quotes since hidden scripts are placed in the value attribute which surrounds the script with double quotes
234             return "'" + originalValue + "'";
235         }
236     }
237 
238     /**
239      * Escapes the ' character present in collection names so it can be properly used in js without causing
240      * javascript errors due to an early completion of a ' string.
241      *
242      * @param name
243      * @return
244      */
245     public static String escapeName(String name) {
246         name = name.replace("'", "\\'");
247         return name;
248     }
249 
250     /**
251      * Converts a list of string to a valid js string array
252      *
253      * @param list list of Strings to be converted
254      * @return String representing the js array
255      */
256     public static String convertStringListToJsArray(List<String> list) {
257         String array = "[";
258 
259         if (list != null) {
260             for (String s : list) {
261                 array = array + "'" + s + "',";
262             }
263             array = StringUtils.removeEnd(array, ",");
264         }
265         array = array + "]";
266 
267         return array;
268     }
269 
270     /**
271      * escapes a string using {@link org.apache.commons.lang.StringEscapeUtils#escapeHtml(String)}
272      *
273      * <p>The apostrophe character is included as <code>StringEscapeUtils#escapeHtml(String)</code>
274      * does not consider it a legal entity. </p>
275      *
276      * @param string the string to be escaped
277      * @return the escaped string - useful for embedding in server side generated JS scripts
278      */
279     public static String escapeHtml(String string) {
280         if (string == null) {
281             return null;
282         } else {
283             return StringEscapeUtils.escapeHtml(string).replace("'", "&apos;").replace("&quot;", "\\u0022");
284         }
285     }
286 
287     /**
288      * escape an array of strings
289      *
290      * @param strings an array of strings to escape
291      * @return the array, with the strings escaped
292      */
293     public static List<String> escapeHtml(List<String> strings) {
294         if (strings == null) {
295             return null;
296         } else if (strings.isEmpty()) {
297             return strings;
298         } else {
299             List<String> result = new ArrayList<String>(strings.size());
300             for (String string : strings) {
301                 result.add(escapeHtml(string));
302             }
303             return result;
304         }
305     }
306 
307     /**
308      * Will append the second script parameter to the first if the first is not empty, also checks to see if the
309      * first script needs an end semi-colon added
310      *
311      * @param script script that will be added to (null is allowed and converted to empty string)
312      * @param appendScript script to append
313      * @return String result of appending the two script parameters
314      */
315     public static String appendScript(String script, String appendScript) {
316         String appendedScript = script;
317         if (appendedScript == null) {
318             appendedScript = "";
319         } else if (StringUtils.isNotBlank(appendedScript) && !appendedScript.trim().endsWith(";")) {
320             appendedScript += "; ";
321         }
322 
323         appendedScript += appendScript;
324 
325         return appendedScript;
326     }
327 
328     /**
329      * Helper method to build a JS string that will invoke the given function with the given arguments
330      * 
331      * @param functionName name of the JS function to invoke
332      * @param arguments zero or more arguments to pass, each will be converted to the corresponding JS type
333      * @return JS String for invoking the function
334      */
335     public static String buildFunctionCall(String functionName, Object... arguments) {
336         StringBuffer sb = new StringBuffer(functionName).append("(");
337 
338         if (arguments != null) {
339             List<String> jsArguments = new ArrayList<String>();
340             for (Object argument : arguments) {
341                 jsArguments.add(translateValue(argument));
342             }
343 
344             sb.append(StringUtils.join(jsArguments, ","));
345         }
346 
347         sb.append(");");
348 
349         return sb.toString();
350     }
351 
352     /**
353      * Builds the JavaScript string for binding the given script to the component with the given id
354      * for the given event name (using jQuery)
355      *
356      * @param id id of the element to handle the event for
357      * @param eventName name of the event the script will handle
358      * @param eventScript script to be executed when the event is thrown, if blank an empty string will
359      * be returned
360      * @return JS event handler script
361      */
362     public static String buildEventHandlerScript(String id, String eventName, String eventScript) {
363         if (StringUtils.isBlank(eventScript)) {
364             return "";
365         }
366 
367         Assert.hasLength(id, "Id is required for building event handler script");
368         Assert.hasLength(eventName, "Event name is required for building event handler script");
369 
370         StringBuffer sb = new StringBuffer();
371 
372         sb.append("jQuery('#" + id + "').on('");
373         sb.append(eventName);
374         sb.append("', function(e) {");
375         sb.append(eventScript);
376         sb.append("}); ");
377 
378         return sb.toString();
379     }
380 }