您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

JsonBuilderGenerator.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. /*
  2. * Copyright 2011, The gwtquery team.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License"); you may not
  5. * use this file except in compliance with the License. You may obtain a copy of
  6. * the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  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, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations under
  14. * the License.
  15. */
  16. package com.google.gwt.query.rebind;
  17. import com.google.gwt.core.client.JavaScriptObject;
  18. import com.google.gwt.core.ext.Generator;
  19. import com.google.gwt.core.ext.GeneratorContext;
  20. import com.google.gwt.core.ext.TreeLogger;
  21. import com.google.gwt.core.ext.TreeLogger.Type;
  22. import com.google.gwt.core.ext.UnableToCompleteException;
  23. import com.google.gwt.core.ext.typeinfo.JArrayType;
  24. import com.google.gwt.core.ext.typeinfo.JClassType;
  25. import com.google.gwt.core.ext.typeinfo.JMethod;
  26. import com.google.gwt.core.ext.typeinfo.JParameter;
  27. import com.google.gwt.core.ext.typeinfo.JParameterizedType;
  28. import com.google.gwt.core.ext.typeinfo.JType;
  29. import com.google.gwt.core.ext.typeinfo.TypeOracle;
  30. import com.google.gwt.query.client.Function;
  31. import com.google.gwt.query.client.IsProperties;
  32. import com.google.gwt.query.client.Properties;
  33. import com.google.gwt.query.client.builders.JsonBuilder;
  34. import com.google.gwt.query.client.builders.JsonBuilderBase;
  35. import com.google.gwt.query.client.builders.JsonFactory;
  36. import com.google.gwt.query.client.builders.Name;
  37. import com.google.gwt.query.client.js.JsUtils;
  38. import com.google.gwt.user.rebind.ClassSourceFileComposerFactory;
  39. import com.google.gwt.user.rebind.SourceWriter;
  40. import java.io.PrintWriter;
  41. import java.util.ArrayList;
  42. import java.util.Collection;
  43. import java.util.HashSet;
  44. import java.util.Iterator;
  45. import java.util.List;
  46. import java.util.Set;
  47. /**
  48. */
  49. public class JsonBuilderGenerator extends Generator {
  50. static JClassType functionType;
  51. static JClassType jsonBuilderType;
  52. static JClassType settingsType;
  53. static JClassType jsType;
  54. static JClassType listType;
  55. static JClassType stringType;
  56. static JClassType jsonFactoryType;
  57. public static String capitalize(String s) {
  58. if (s.length() == 0)
  59. return s;
  60. return s.substring(0, 1).toUpperCase() + s.substring(1).toLowerCase();
  61. }
  62. public static String classNameToJsonName(String name) {
  63. return deCapitalize(name.replaceAll("^.*[\\.\\$_]", ""));
  64. }
  65. public static String deCapitalize(String s) {
  66. return s == null || s.isEmpty() ? s :
  67. (s.substring(0, 1).toLowerCase() + (s.length() > 1 ? s.substring(1) : ""));
  68. }
  69. TypeOracle oracle;
  70. public String generate(TreeLogger treeLogger,
  71. GeneratorContext generatorContext, String requestedClass)
  72. throws UnableToCompleteException {
  73. oracle = generatorContext.getTypeOracle();
  74. JClassType clazz = oracle.findType(requestedClass);
  75. jsonBuilderType = oracle.findType(JsonBuilder.class.getName());
  76. settingsType = oracle.findType(IsProperties.class.getName());
  77. stringType = oracle.findType(String.class.getName());
  78. jsType = oracle.findType(JavaScriptObject.class.getName());
  79. listType = oracle.findType(List.class.getName());
  80. functionType = oracle.findType(Function.class.getName());
  81. jsonFactoryType = oracle.findType(JsonFactory.class.getName());
  82. String t[] = generateClassName(clazz);
  83. boolean isFactory = clazz.isAssignableTo(jsonFactoryType);
  84. SourceWriter sw =
  85. getSourceWriter(treeLogger, generatorContext, t[0], t[1], isFactory, requestedClass);
  86. if (sw != null) {
  87. if (isFactory) {
  88. generateCreateMethod(sw, treeLogger);
  89. } else {
  90. Set<String> attrs = new HashSet<String>();
  91. for (JMethod method : clazz.getInheritableMethods()) {
  92. String methName = method.getName();
  93. // skip method from JsonBuilder
  94. if (jsonBuilderType.findMethod(method.getName(), method.getParameterTypes()) != null ||
  95. settingsType.findMethod(method.getName(), method.getParameterTypes()) != null) {
  96. continue;
  97. }
  98. Name nameAnnotation = method.getAnnotation(Name.class);
  99. String name = nameAnnotation != null
  100. ? nameAnnotation.value()
  101. : methName.replaceFirst("^(get|set)", "");
  102. if (nameAnnotation == null) {
  103. name = name.substring(0, 1).toLowerCase() + name.substring(1);
  104. }
  105. attrs.add(name);
  106. generateMethod(sw, method, name, treeLogger);
  107. }
  108. generateFieldNamesMethod(sw, attrs, treeLogger);
  109. generateToJsonMethod(sw, t[3], treeLogger);
  110. }
  111. sw.commit(treeLogger);
  112. }
  113. return t[2];
  114. }
  115. public String[] generateClassName(JType t) {
  116. String[] ret = new String[4];
  117. JClassType c = t.isClassOrInterface();
  118. ret[0] = c.getPackage().getName();
  119. ret[1] = c.getName().replace('.', '_') + "_JsonBuilder";
  120. ret[2] = ret[0] + "." + ret[1];
  121. ret[3] = classNameToJsonName(c.getName());
  122. return ret;
  123. }
  124. public void generateFieldNamesMethod(SourceWriter sw, Collection<String> attrs, TreeLogger logger) {
  125. String ret = "";
  126. for (Iterator<String> it = attrs.iterator(); it.hasNext();) {
  127. ret += (ret.isEmpty() ? "" : ",") + "\"" + it.next() + "\"";
  128. }
  129. sw.println("{ fieldNames = new String[]{" + ret + "}; }");
  130. }
  131. public void generateMethod(SourceWriter sw, JMethod method, String name, TreeLogger logger)
  132. throws UnableToCompleteException {
  133. String ifaceName = method.getEnclosingType().getQualifiedSourceName();
  134. String retType = method.getReturnType().getParameterizedQualifiedSourceName();
  135. sw.print("public final " + retType + " " + method.getName());
  136. JParameter[] params = method.getParameters();
  137. if (params.length == 0) {
  138. JArrayType arr = method.getReturnType().isArray();
  139. JParameterizedType list = method.getReturnType().isParameterized();
  140. sw.println("() {");
  141. sw.indent();
  142. if (retType.matches("(java.lang.Boolean|boolean)")) {
  143. sw.println("return p.getBoolean(\"" + name + "\");");
  144. } else if (retType.matches("java.util.Date")) {
  145. sw.println("return new Date(java.lang.Long.parseLong(p.getStr(\"" + name + "\")));");
  146. } else if (method.getReturnType().isPrimitive() != null) {
  147. sw.println("return (" + retType + ")p.getFloat(\"" + name + "\");");
  148. } else if (retType.equals("java.lang.Character")) {
  149. sw.println("return (char) p.getFloat(\"" + name + "\");");
  150. } else if (retType.equals("java.lang.Byte")) {
  151. sw.println("return (byte) p.getFloat(\"" + name + "\");");
  152. } else if (retType.equals("java.lang.Integer")) {
  153. sw.println("return (int) p.getFloat(\"" + name + "\");");
  154. } else if (retType.equals("java.lang.Float")) {
  155. sw.println("return p.getFloat(\"" + name + "\");");
  156. } else if (retType.equals("java.lang.Double")) {
  157. sw.println("return (double) p.getFloat(\"" + name + "\");");
  158. } else if (retType.equals("java.lang.Long")) {
  159. sw.println("return (long) p.getFloat(\"" + name + "\");");
  160. } else if (retType.equals("java.lang.Byte")) {
  161. sw.println("return (byte) p.getFloat(\"" + name + "\");");
  162. } else if (isTypeAssignableTo(method.getReturnType(), stringType)) {
  163. sw.println("return p.getStr(\"" + name + "\");");
  164. } else if (isTypeAssignableTo(method.getReturnType(), jsonBuilderType)) {
  165. String q = method.getReturnType().getQualifiedSourceName();
  166. sw.println("return " + "getIsPropertiesBase(getPropertiesBase(\"" + name + "\")," + q + ".class);");
  167. } else if (isTypeAssignableTo(method.getReturnType(), settingsType)) {
  168. String q = method.getReturnType().getQualifiedSourceName();
  169. sw.println("return " + "((" + q + ")getPropertiesBase(\"" + name + "\"));");
  170. } else if (retType.equals(Properties.class.getName())) {
  171. sw.println("return getPropertiesBase(\"" + name + "\");");
  172. } else if (isTypeAssignableTo(method.getReturnType(), jsType)) {
  173. sw.println("return p.getJavaScriptObject(\"" + name + "\");");
  174. } else if (isTypeAssignableTo(method.getReturnType(), functionType)) {
  175. sw.println("return p.getFunction(\"" + name + "\");");
  176. } else if (arr != null || list != null) {
  177. JType type = arr != null ? arr.getComponentType()
  178. : list.getTypeArgs()[0];
  179. boolean buildType = isTypeAssignableTo(type, jsonBuilderType);
  180. String t = type.getQualifiedSourceName();
  181. sw.println("JsArrayMixed a = p.getArray(\"" + name + "\");");
  182. sw.println("int l = a == null ? 0 : a.length();");
  183. String ret;
  184. if (buildType) {
  185. ret = "getIsPropertiesArrayBase(a, new " + t + "[l], " + t + ".class)";
  186. } else {
  187. ret = "getArrayBase(\"" + name + "\", new " + t + "[l], " + t + ".class)";
  188. }
  189. if (arr != null) {
  190. sw.println("return " + ret + ";");
  191. } else {
  192. sw.println("return (a == null ? null : Arrays.asList(" + ret + "));");
  193. }
  194. } else if (method.getReturnType().isEnum() != null) {
  195. sw.println("return " + method.getReturnType().getQualifiedSourceName()
  196. + ".valueOf(p.getStr(\"" + name + "\"));");
  197. } else {
  198. sw.println("System.err.println(\"JsonBuilderGenerator WARN: unknown return type "
  199. + retType + " " + ifaceName + "." + name + "()\"); ");
  200. // We return the object because probably the user knows how to handle it
  201. sw.println("return p.get(\"" + name + "\");");
  202. }
  203. sw.outdent();
  204. sw.println("}");
  205. } else if (params.length == 1) {
  206. JType type = params[0].getType();
  207. JArrayType arr = type.isArray();
  208. JParameterizedType list = type.isParameterized();
  209. sw.print("(" + type.getParameterizedQualifiedSourceName() + " a)");
  210. sw.println("{");
  211. sw.indent();
  212. if (arr != null || list != null) {
  213. String a = "a";
  214. if (list != null) {
  215. a = "a.toArray(new " + list.getTypeArgs()[0].getQualifiedSourceName()
  216. + "[0])";
  217. }
  218. sw.println("setArrayBase(\"" + name + "\", " + a + ");");
  219. } else if (type.getParameterizedQualifiedSourceName().matches("java.util.Date")) {
  220. sw.println("p.setNumber(\"" + name + "\", a.getTime());");
  221. } else if (type
  222. .getParameterizedQualifiedSourceName()
  223. .matches(
  224. "(java.lang.(Character|Long|Double|Integer|Float|Byte)|(char|long|double|int|float|byte))")) {
  225. sw.println("p.setNumber(\"" + name + "\", a);");
  226. } else if (type.getParameterizedQualifiedSourceName().matches("(java.lang.Boolean|boolean)")) {
  227. sw.println("p.setBoolean(\"" + name + "\", a);");
  228. } else if (type.getParameterizedQualifiedSourceName().matches(
  229. "com.google.gwt.query.client.Function")) {
  230. sw.println("p.setFunction(\"" + name + "\", a);");
  231. } else if (type.isEnum() != null) {
  232. sw.println("p.set(\"" + name + "\", a.name());");
  233. } else {
  234. sw.println("set(\"" + name + "\", a);");
  235. }
  236. if (!"void".equals(retType)) {
  237. if (isTypeAssignableTo(method.getReturnType(),
  238. method.getEnclosingType())) {
  239. sw.println("return this;");
  240. } else {
  241. sw.println("return null;");
  242. }
  243. }
  244. sw.outdent();
  245. sw.println("}");
  246. }
  247. }
  248. public void generateToJsonMethod(SourceWriter sw, String name, TreeLogger logger) {
  249. sw.println("public final String getJsonName() {return \"" + name + "\";}");
  250. }
  251. protected SourceWriter getSourceWriter(TreeLogger logger,
  252. GeneratorContext context, String packageName, String className, boolean isFactory,
  253. String... interfaceNames) {
  254. PrintWriter printWriter = context.tryCreate(logger, packageName, className);
  255. if (printWriter == null) {
  256. return null;
  257. }
  258. ClassSourceFileComposerFactory composerFactory = new ClassSourceFileComposerFactory(
  259. packageName, className);
  260. if (!isFactory) {
  261. composerFactory.setSuperclass(JsonBuilderBase.class.getName() +
  262. "<" + packageName + "." + className + ">");
  263. }
  264. composerFactory.addImport("com.google.gwt.query.client.js.*");
  265. composerFactory.addImport("com.google.gwt.query.client.*");
  266. composerFactory.addImport("com.google.gwt.core.client.*");
  267. composerFactory.addImport("com.google.gwt.dom.client.*");
  268. composerFactory.addImport("java.util.*");
  269. for (String interfaceName : interfaceNames) {
  270. composerFactory.addImplementedInterface(interfaceName);
  271. }
  272. return composerFactory.createSourceWriter(context, printWriter);
  273. }
  274. public boolean isTypeAssignableTo(JType t, JClassType o) {
  275. JClassType c = t.isClassOrInterface();
  276. return (c != null && c.isAssignableTo(o));
  277. }
  278. private void generateCreateMethod(SourceWriter sw, TreeLogger logger) {
  279. sw.println("public <T extends " + JsonBuilder.class.getName() + "> T create(Class<T> clz) {");
  280. sw.indent();
  281. ArrayList<JClassType> types = new ArrayList<JClassType>();
  282. for (JClassType t : oracle.getTypes()) {
  283. if (t.isInterface() != null && t.isAssignableTo(jsonBuilderType)) {
  284. if (t.isPublic()) {
  285. sw.println("if (clz == " + t.getQualifiedSourceName() + ".class) return GWT.<T>create("
  286. + t.getQualifiedSourceName() + ".class);");
  287. } else {
  288. logger.log(Type.WARN, t.getQualifiedSourceName() + " is not public");
  289. }
  290. types.add(t);
  291. }
  292. }
  293. sw.println("GQuery.console.error(\"GQ.create: not registered class :\" + clz);");
  294. sw.println("return null;");
  295. sw.outdent();
  296. sw.println("}");
  297. sw.println("public " + IsProperties.class.getName() + " create(String s) {");
  298. sw.indent();
  299. sw.println("return (" + IsProperties.class.getName() + ")" + JsUtils.class.getName()
  300. + ".parseJSON(s);");
  301. sw.outdent();
  302. sw.println("}");
  303. sw.println("public " + IsProperties.class.getName() + " create() {");
  304. sw.indent();
  305. sw.println("return " + Properties.class.getName() + ".create();");
  306. sw.outdent();
  307. sw.println("}");
  308. }
  309. }