2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
4 * Copyright 1997-2010 Oracle and/or its affiliates. All rights reserved.
6 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
7 * Other names may be trademarks of their respective owners.
9 * The contents of this file are subject to the terms of either the GNU
10 * General Public License Version 2 only ("GPL") or the Common
11 * Development and Distribution License("CDDL") (collectively, the
12 * "License"). You may not use this file except in compliance with the
13 * License. You can obtain a copy of the License at
14 * http://www.netbeans.org/cddl-gplv2.html
15 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
16 * specific language governing permissions and limitations under the
17 * License. When distributing the software, include this License Header
18 * Notice in each file and include the License file at
19 * nbbuild/licenses/CDDL-GPL-2-CP. Oracle designates this
20 * particular file as subject to the "Classpath" exception as provided
21 * by Oracle in the GPL Version 2 section of the License file that
22 * accompanied this code. If applicable, add the following below the
23 * License Header, with the fields enclosed by brackets [] replaced by
24 * your own identifying information:
25 * "Portions Copyrighted [year] [name of copyright owner]"
29 * The Original Software is NetBeans. The Initial Developer of the Original
30 * Software is Oracle. Portions Copyright 2013-2013 Oracle. All Rights Reserved.
32 * If you wish your version of this file to be governed by only the CDDL
33 * or only the GPL Version 2, indicate your decision by adding
34 * "[Contributor] elects to include this software in this distribution
35 * under the [CDDL or GPL Version 2] license." If you do not indicate a
36 * single choice of license, a recipient has the option to distribute
37 * your version of this file under either the CDDL, the GPL Version 2 or
38 * to extend the choice of license to its licensees as provided above.
39 * However, if you add GPL Version 2 code and therefore, elected the GPL
40 * Version 2 license, then the option applies only if the new code is
41 * made subject to such option by the copyright holder.
43 package org.netbeans.html.json.impl;
45 import java.io.IOException;
46 import java.io.OutputStreamWriter;
47 import java.io.StringWriter;
48 import java.io.Writer;
49 import java.lang.annotation.AnnotationTypeMismatchException;
50 import java.lang.annotation.IncompleteAnnotationException;
51 import java.lang.reflect.Method;
52 import java.util.ArrayList;
53 import java.util.Arrays;
54 import java.util.Collection;
55 import java.util.Collections;
56 import java.util.HashMap;
57 import java.util.HashSet;
58 import java.util.LinkedHashSet;
59 import java.util.List;
61 import java.util.ResourceBundle;
63 import java.util.WeakHashMap;
64 import java.util.logging.Level;
65 import java.util.logging.Logger;
66 import javax.annotation.processing.AbstractProcessor;
67 import javax.annotation.processing.Completion;
68 import javax.annotation.processing.Completions;
69 import javax.annotation.processing.ProcessingEnvironment;
70 import javax.annotation.processing.Processor;
71 import javax.annotation.processing.RoundEnvironment;
72 import javax.annotation.processing.SupportedAnnotationTypes;
73 import javax.annotation.processing.SupportedSourceVersion;
74 import javax.lang.model.SourceVersion;
75 import javax.lang.model.element.AnnotationMirror;
76 import javax.lang.model.element.AnnotationValue;
77 import javax.lang.model.element.Element;
78 import javax.lang.model.element.ElementKind;
79 import javax.lang.model.element.ExecutableElement;
80 import javax.lang.model.element.Modifier;
81 import javax.lang.model.element.PackageElement;
82 import javax.lang.model.element.TypeElement;
83 import javax.lang.model.element.VariableElement;
84 import javax.lang.model.type.ArrayType;
85 import javax.lang.model.type.DeclaredType;
86 import javax.lang.model.type.MirroredTypeException;
87 import javax.lang.model.type.TypeKind;
88 import javax.lang.model.type.TypeMirror;
89 import javax.lang.model.util.Elements;
90 import javax.lang.model.util.Types;
91 import javax.tools.Diagnostic;
92 import javax.tools.FileObject;
93 import net.java.html.json.ComputedProperty;
94 import net.java.html.json.Model;
95 import net.java.html.json.Function;
96 import net.java.html.json.ModelOperation;
97 import net.java.html.json.OnPropertyChange;
98 import net.java.html.json.OnReceive;
99 import net.java.html.json.Property;
100 import org.openide.util.lookup.ServiceProvider;
102 /** Annotation processor to process {@link Model @Model} annotations and
103 * generate appropriate model classes.
105 * @author Jaroslav Tulach <jtulach@netbeans.org>
107 @ServiceProvider(service=Processor.class)
108 @SupportedSourceVersion(SourceVersion.RELEASE_6)
109 @SupportedAnnotationTypes({
110 "net.java.html.json.Model",
111 "net.java.html.json.ModelOperation",
112 "net.java.html.json.Function",
113 "net.java.html.json.OnReceive",
114 "net.java.html.json.OnPropertyChange",
115 "net.java.html.json.ComputedProperty",
116 "net.java.html.json.Property"
118 public final class ModelProcessor extends AbstractProcessor {
119 private static final Logger LOG = Logger.getLogger(ModelProcessor.class.getName());
120 private final Map<Element,String> models = new WeakHashMap<Element,String>();
121 private final Map<Element,Prprt[]> verify = new WeakHashMap<Element,Prprt[]>();
123 public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
125 for (Element e : roundEnv.getElementsAnnotatedWith(Model.class)) {
126 if (!processModel(e)) {
130 if (roundEnv.processingOver()) {
132 for (Map.Entry<Element, Prprt[]> entry : verify.entrySet()) {
133 TypeElement te = (TypeElement)entry.getKey();
134 String fqn = processingEnv.getElementUtils().getBinaryName(te).toString();
135 Element finalElem = processingEnv.getElementUtils().getTypeElement(fqn);
136 if (finalElem == null) {
140 Model m = finalElem.getAnnotation(Model.class);
144 props = Prprt.wrap(processingEnv, finalElem, m.properties());
145 for (Prprt p : props) {
146 boolean[] isModel = { false };
147 boolean[] isEnum = { false };
148 boolean[] isPrimitive = { false };
149 String t = checkType(p, isModel, isEnum, isPrimitive);
153 if (isPrimitive[0]) {
159 if ("java.lang.String".equals(t)) {
162 error("The type " + t + " should be defined by @Model annotation", entry.getKey());
170 private void error(String msg, Element e) {
171 processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, msg, e);
174 private boolean processModel(Element e) {
176 Model m = e.getAnnotation(Model.class);
180 String pkg = findPkgName(e);
182 String className = m.className();
183 models.put(e, className);
185 StringWriter body = new StringWriter();
186 List<String> propsGetSet = new ArrayList<String>();
187 List<String> functions = new ArrayList<String>();
188 Map<String, Collection<String>> propsDeps = new HashMap<String, Collection<String>>();
189 Map<String, Collection<String>> functionDeps = new HashMap<String, Collection<String>>();
190 Prprt[] props = createProps(e, m.properties());
192 if (!generateComputedProperties(body, props, e.getEnclosedElements(), propsGetSet, propsDeps)) {
195 if (!generateOnChange(e, propsDeps, props, className, functionDeps)) {
198 if (!generateProperties(e, body, props, propsGetSet, propsDeps, functionDeps)) {
201 if (!generateFunctions(e, body, className, e.getEnclosedElements(), functions)) {
204 if (!generateReceive(e, body, className, e.getEnclosedElements(), functions)) {
207 if (!generateOperation(e, body, className, e.getEnclosedElements())) {
210 FileObject java = processingEnv.getFiler().createSourceFile(pkg + '.' + className, e);
211 w = new OutputStreamWriter(java.openOutputStream());
213 w.append("package " + pkg + ";\n");
214 w.append("import net.java.html.json.*;\n");
215 w.append("public final class ").append(className).append(" implements Cloneable {\n");
216 w.append(" private boolean locked;\n");
217 w.append(" private net.java.html.BrwsrCtx context;\n");
218 w.append(" private org.netbeans.html.json.impl.Bindings[] ko = { null };\n");
219 w.append(body.toString());
220 w.append(" private static Class<" + inPckName(e) + "> modelFor() { return null; }\n");
221 w.append(" private ").append(className).append("(net.java.html.BrwsrCtx context) {\n");
222 w.append(" this.context = context;\n");
224 w.append(" public ").append(className).append("() {\n");
225 w.append(" this(net.java.html.BrwsrCtx.findDefault(").append(className).append(".class));\n");
226 for (Prprt p : props) {
230 boolean[] isModel = {false};
231 boolean[] isEnum = {false};
232 boolean isPrimitive[] = {false};
233 String tn = checkType(p, isModel, isEnum, isPrimitive);
235 w.write(" prop_" + p.name() + " = new " + tn + "();\n");
239 if (props.length > 0) {
240 w.append(" public ").append(className).append("(");
241 Prprt firstArray = null;
243 for (Prprt p : props) {
245 if (firstArray == null) {
250 String tn = typeName(e, p);
253 w.write(" " + p.name());
256 if (firstArray != null) {
258 boolean[] isModel = {false};
259 boolean[] isEnum = {false};
260 boolean isPrimitive[] = {false};
261 tn = checkType(firstArray, isModel, isEnum, isPrimitive);
264 w.write("... " + firstArray.name());
267 w.append(" this(net.java.html.BrwsrCtx.findDefault(").append(className).append(".class));\n");
268 for (Prprt p : props) {
272 w.write(" this.prop_" + p.name() + " = " + p.name() + ";\n");
274 if (firstArray != null) {
275 w.write(" this.prop_" + firstArray.name() + ".init(" + firstArray.name() + ");\n");
279 w.append(" private org.netbeans.html.json.impl.Bindings intKnckt() {\n");
280 w.append(" if (ko[0] != null) return ko[0];\n");
281 w.append(" ko[0] = org.netbeans.html.json.impl.Bindings.apply(context, this);\n");
283 w.append(" org.apidesign.html.json.spi.PropertyBinding[] propArr = {\n");
284 for (int i = 0; i < propsGetSet.size(); i += 5) {
285 w.append(" ko[0].registerProperty(\"").append(propsGetSet.get(i)).append("\", this, new P(");
286 w.append((i / 5) + "), " + (propsGetSet.get(i + 2) == null) + "),\n");
291 w.append(" org.apidesign.html.json.spi.FunctionBinding[] funcArr = {\n");
292 for (int i = 0; i < functions.size(); i += 2) {
293 w.append(" ko[0].registerFunction(\"").append(functions.get(i)).append("\", this, new P(");
294 w.append((i / 2) + ")),\n");
298 w.append(" ko[0].finish(this, propArr, funcArr);\n");
299 w.append(" return ko[0];\n");
301 w.append(" private static final class P implements org.netbeans.html.json.impl.SetAndGet<" + className + ">,\n");
302 w.append(" org.netbeans.html.json.impl.Callback<" + className + ">,\n");
303 w.append(" org.netbeans.html.json.impl.FromJSON<" + className + "> {\n");
304 w.append(" private final int type;\n");
305 w.append(" P(int t) { type = t; };\n");
306 w.append(" public void setValue(" + className + " data, Object value) {\n");
307 w.append(" switch (type) {\n");
308 for (int i = 0; i < propsGetSet.size(); i += 5) {
309 final String set = propsGetSet.get(i + 2);
310 String tn = propsGetSet.get(i + 4);
311 String btn = findBoxedType(tn);
316 w.append(" case " + (i / 5) + ": data." + strip(set) + "(org.netbeans.html.json.impl.JSON.extractValue(" + tn + ".class, value)); return;\n");
321 w.append(" public Object getValue(" + className + " data) {\n");
322 w.append(" switch (type) {\n");
323 for (int i = 0; i < propsGetSet.size(); i += 5) {
324 final String get = propsGetSet.get(i + 1);
326 w.append(" case " + (i / 5) + ": return data." + strip(get) + "();\n");
330 w.append(" throw new UnsupportedOperationException();\n");
332 w.append(" public void call(" + className + " model, Object data, Object ev) {\n");
333 w.append(" switch (type) {\n");
334 for (int i = 0; i < functions.size(); i += 2) {
335 final String name = functions.get(i);
336 w.append(" case " + (i / 2) + ": model." + name + "(data, ev); return;\n");
339 w.append(" throw new UnsupportedOperationException();\n");
341 w.append(" public Class<" + className + "> factoryFor() { return " + className + ".class; }\n");
342 w.append(" public " + className + " read(net.java.html.BrwsrCtx c, Object json) { return new " + className + "(c, json); }\n");
343 w.append(" public " + className + " cloneTo(Object o, net.java.html.BrwsrCtx c) { return ((" + className + ")o).clone(c); }\n");
345 w.append(" static { org.netbeans.html.json.impl.JSON.register(new P(0)); }\n");
346 w.append(" private ").append(className).append("(net.java.html.BrwsrCtx c, Object json) {\n");
347 w.append(" this.context = c;\n");
349 for (int i = 0; i < propsGetSet.size(); i += 5) {
350 Prprt p = findPrprt(props, propsGetSet.get(i));
356 w.append(" Object[] ret = new Object[" + values + "];\n");
357 w.append(" org.netbeans.html.json.impl.JSON.extract(context, json, new String[] {\n");
358 for (int i = 0; i < propsGetSet.size(); i += 5) {
359 Prprt p = findPrprt(props, propsGetSet.get(i));
363 w.append(" \"").append(propsGetSet.get(i)).append("\",\n");
365 w.append(" }, ret);\n");
366 for (int i = 0, cnt = 0, prop = 0; i < propsGetSet.size(); i += 5) {
367 final String pn = propsGetSet.get(i);
368 Prprt p = findPrprt(props, pn);
372 boolean[] isModel = { false };
373 boolean[] isEnum = { false };
374 boolean isPrimitive[] = { false };
375 String type = checkType(props[prop++], isModel, isEnum, isPrimitive);
377 w.append(" if (ret[" + cnt + "] instanceof Object[]) {\n");
378 w.append(" for (Object e : ((Object[])ret[" + cnt + "])) {\n");
380 w.append(" this.prop_").append(pn).append(".add(org.netbeans.html.json.impl.JSON.read");
381 w.append("(c, " + type + ".class, e));\n");
382 } else if (isEnum[0]) {
383 w.append(" this.prop_").append(pn);
384 w.append(".add(e == null ? null : ");
385 w.append(type).append(".valueOf(org.netbeans.html.json.impl.JSON.stringValue(e)));\n");
387 if (isPrimitive(type)) {
388 w.append(" this.prop_").append(pn).append(".add(org.netbeans.html.json.impl.JSON.numberValue(e).");
389 w.append(type).append("Value());\n");
391 w.append(" this.prop_").append(pn).append(".add((");
392 w.append(type).append(")e);\n");
399 w.append(" try {\n");
400 w.append(" this.prop_").append(pn);
401 w.append(" = ret[" + cnt + "] == null ? null : ");
402 w.append(type).append(".valueOf(org.netbeans.html.json.impl.JSON.stringValue(ret[" + cnt + "]));\n");
403 w.append(" } catch (IllegalArgumentException ex) {\n");
404 w.append(" ex.printStackTrace();\n");
406 } else if (isPrimitive(type)) {
407 w.append(" this.prop_").append(pn);
408 w.append(" = ret[" + cnt + "] == null ? ");
409 if ("char".equals(type)) {
410 w.append("0 : (org.netbeans.html.json.impl.JSON.charValue(");
411 } else if ("boolean".equals(type)) {
412 w.append("false : (org.netbeans.html.json.impl.JSON.boolValue(");
414 w.append("0 : (org.netbeans.html.json.impl.JSON.numberValue(");
416 w.append("ret[" + cnt + "])).");
417 w.append(type).append("Value();\n");
418 } else if (isModel[0]) {
419 w.append(" this.prop_").append(pn).append(" = org.netbeans.html.json.impl.JSON.read");
420 w.append("(c, " + type + ".class, ");
421 w.append("ret[" + cnt + "]);\n");
423 w.append(" this.prop_").append(pn);
424 w.append(" = (").append(type).append(')');
425 w.append("ret[" + cnt + "];\n");
431 writeToString(props, w);
432 writeClone(className, props, w);
433 w.write(" /** Activates this model instance in the current {@link \n"
434 + "net.java.html.json.Models#bind(java.lang.Object, net.java.html.BrwsrCtx) browser context}. \n"
435 + "In case of using Knockout technology, this means to \n"
436 + "bind JSON like data in this model instance with Knockout tags in \n"
437 + "the surrounding HTML page.\n"
440 w.write(" public " + className + " applyBindings() {\n");
441 w.write(" intKnckt().applyBindings();\n");
442 w.write(" return this;\n");
444 w.write(" public boolean equals(Object o) {\n");
445 w.write(" if (o == this) return true;\n");
446 w.write(" if (o instanceof org.netbeans.html.json.impl.WrapperObject) {\n");
447 w.write(" ((org.netbeans.html.json.impl.WrapperObject)o).setRealObject(intKnckt().koData());\n");
448 w.write(" return false;\n");
450 w.write(" if (!(o instanceof " + className + ")) return false;\n");
451 w.write(" " + className + " p = (" + className + ")o;\n");
452 for (Prprt p : props) {
453 w.write(" if (!org.netbeans.html.json.impl.JSON.isSame(prop_" + p.name() + ", p.prop_" + p.name() + ")) return false;\n");
455 w.write(" return true;\n");
457 w.write(" public int hashCode() {\n");
458 w.write(" int h = " + className + ".class.getName().hashCode();\n");
459 for (Prprt p : props) {
460 w.write(" h = org.netbeans.html.json.impl.JSON.hashPlus(prop_" + p.name() + ", h);\n");
462 w.write(" return h;\n");
468 } catch (IOException ex) {
469 error("Can't create " + className + ".java", e);
475 private boolean generateProperties(
477 Writer w, Prprt[] properties,
478 Collection<String> props,
479 Map<String,Collection<String>> deps,
480 Map<String,Collection<String>> functionDeps
481 ) throws IOException {
483 for (Prprt p : properties) {
485 tn = typeName(where, p);
486 String[] gs = toGetSet(p.name(), tn, p.array());
490 w.write(" private org.netbeans.html.json.impl.JSONList<" + tn + "> prop_" + p.name() + " = new org.netbeans.html.json.impl.JSONList<" + tn + ">(ko, \""
492 Collection<String> dependants = deps.get(p.name());
493 if (dependants != null) {
494 for (String depProp : dependants) {
503 dependants = functionDeps.get(p.name());
504 if (dependants != null) {
505 w.write(".onChange(new Runnable() { public void run() {\n");
506 for (String call : dependants) {
507 w.append(" ").append(call);
513 castTo = "java.util.List";
514 w.write(" public java.util.List<" + tn + "> " + gs[0] + "() {\n");
515 w.write(" if (locked) throw new IllegalStateException();\n");
516 w.write(" return prop_" + p.name() + ";\n");
520 w.write(" private " + tn + " prop_" + p.name() + ";\n");
521 w.write(" public " + tn + " " + gs[0] + "() {\n");
522 w.write(" if (locked) throw new IllegalStateException();\n");
523 w.write(" return prop_" + p.name() + ";\n");
525 w.write(" public void " + gs[1] + "(" + tn + " v) {\n");
526 w.write(" if (locked) throw new IllegalStateException();\n");
527 w.write(" if (org.netbeans.html.json.impl.JSON.isSame(prop_" + p.name() + ", v)) return;\n");
528 w.write(" prop_" + p.name() + " = v;\n");
529 w.write(" org.netbeans.html.json.impl.Bindings b = ko[0];\n");
530 w.write(" if (b != null) {\n");
531 w.write(" b.valueHasMutated(\"" + p.name() + "\");\n");
532 Collection<String> dependants = deps.get(p.name());
533 if (dependants != null) {
534 for (String depProp : dependants) {
535 w.write(" b.valueHasMutated(\"" + depProp + "\");\n");
539 dependants = functionDeps.get(p.name());
540 if (dependants != null) {
541 for (String call : dependants) {
542 w.append(" ").append(call);
557 private boolean generateComputedProperties(
558 Writer w, Prprt[] fixedProps,
559 Collection<? extends Element> arr, Collection<String> props,
560 Map<String,Collection<String>> deps
561 ) throws IOException {
563 for (Element e : arr) {
564 if (e.getKind() != ElementKind.METHOD) {
567 if (e.getAnnotation(ComputedProperty.class) == null) {
570 if (!e.getModifiers().contains(Modifier.STATIC)) {
571 error("Method " + e.getSimpleName() + " has to be static when annotated by @ComputedProperty", e);
575 ExecutableElement ee = (ExecutableElement)e;
576 final TypeMirror rt = ee.getReturnType();
577 final Types tu = processingEnv.getTypeUtils();
578 TypeMirror ert = tu.erasure(rt);
579 String tn = fqn(ert, ee);
580 boolean array = false;
581 final TypeMirror toCheck;
582 if (tn.equals("java.util.List")) {
584 toCheck = ((DeclaredType)rt).getTypeArguments().get(0);
589 final String sn = ee.getSimpleName().toString();
591 if (toCheck.getKind().isPrimitive()) {
594 TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
595 TypeMirror enumType = processingEnv.getElementUtils().getTypeElement("java.lang.Enum").asType();
597 if (tu.isSubtype(toCheck, stringType)) {
599 } else if (tu.isSubtype(tu.erasure(toCheck), tu.erasure(enumType))) {
601 } else if (isModel(toCheck)) {
605 error(sn + " cannot return " + toCheck, e);
609 String[] gs = toGetSet(sn, tn, array);
611 w.write(" public " + tn + " " + gs[0] + "() {\n");
612 w.write(" if (locked) throw new IllegalStateException();\n");
614 for (VariableElement pe : ee.getParameters()) {
615 final String dn = pe.getSimpleName().toString();
617 if (!verifyPropName(pe, dn, fixedProps)) {
621 final String dt = fqn(pe.asType(), ee);
622 String[] call = toGetSet(dn, dt, false);
623 w.write(" " + dt + " arg" + (++arg) + " = ");
624 w.write(call[0] + "();\n");
626 Collection<String> depends = deps.get(dn);
627 if (depends == null) {
628 depends = new LinkedHashSet<String>();
629 deps.put(dn, depends);
634 w.write(" locked = true;\n");
635 w.write(" return " + fqn(ee.getEnclosingElement().asType(), ee) + '.' + e.getSimpleName() + "(");
637 for (int i = 1; i <= arg; i++) {
643 w.write(" } finally {\n");
644 w.write(" locked = false;\n");
648 props.add(e.getSimpleName().toString());
658 private static String[] toGetSet(String name, String type, boolean array) {
659 String n = Character.toUpperCase(name.charAt(0)) + name.substring(1);
660 String bck2brwsrType = "L" + type.replace('.', '_') + "_2";
661 if ("int".equals(type)) {
664 if ("double".equals(type)) {
668 if ("boolean".equals(type)) {
672 final String nu = n.replace('.', '_');
674 return new String[] {
677 "get" + nu + "__Ljava_util_List_2",
684 pref + nu + "__" + bck2brwsrType,
685 "set" + nu + "__V" + bck2brwsrType
689 private String typeName(Element where, Prprt p) {
691 boolean[] isModel = { false };
692 boolean[] isEnum = { false };
693 boolean isPrimitive[] = { false };
694 ret = checkType(p, isModel, isEnum, isPrimitive);
696 String bt = findBoxedType(ret);
704 private static String findBoxedType(String ret) {
705 if (ret.equals("boolean")) {
706 return Boolean.class.getName();
708 if (ret.equals("byte")) {
709 return Byte.class.getName();
711 if (ret.equals("short")) {
712 return Short.class.getName();
714 if (ret.equals("char")) {
715 return Character.class.getName();
717 if (ret.equals("int")) {
718 return Integer.class.getName();
720 if (ret.equals("long")) {
721 return Long.class.getName();
723 if (ret.equals("float")) {
724 return Float.class.getName();
726 if (ret.equals("double")) {
727 return Double.class.getName();
732 private boolean verifyPropName(Element e, String propName, Prprt[] existingProps) {
733 StringBuilder sb = new StringBuilder();
735 for (Prprt Prprt : existingProps) {
736 if (Prprt.name().equals(propName)) {
741 sb.append(Prprt.name());
746 propName + " is not one of known properties: " + sb
752 private static String findPkgName(Element e) {
754 if (e.getKind() == ElementKind.PACKAGE) {
755 return ((PackageElement)e).getQualifiedName().toString();
757 e = e.getEnclosingElement();
761 private boolean generateFunctions(
762 Element clazz, StringWriter body, String className,
763 List<? extends Element> enclosedElements, List<String> functions
765 for (Element m : enclosedElements) {
766 if (m.getKind() != ElementKind.METHOD) {
769 ExecutableElement e = (ExecutableElement)m;
770 Function onF = e.getAnnotation(Function.class);
774 if (!e.getModifiers().contains(Modifier.STATIC)) {
775 error("@OnFunction method needs to be static", e);
778 if (e.getModifiers().contains(Modifier.PRIVATE)) {
779 error("@OnFunction method cannot be private", e);
782 if (e.getReturnType().getKind() != TypeKind.VOID) {
783 error("@OnFunction method should return void", e);
786 String n = e.getSimpleName().toString();
787 body.append(" private void ").append(n).append("(Object data, Object ev) {\n");
788 body.append(" ").append(((TypeElement)clazz).getQualifiedName()).append(".").append(n).append("(");
789 body.append(wrapParams(e, null, className, "ev", "data"));
794 functions.add(n + "__VLjava_lang_Object_2Ljava_lang_Object_2");
799 private boolean generateOnChange(Element clazz, Map<String,Collection<String>> propDeps,
800 Prprt[] properties, String className,
801 Map<String, Collection<String>> functionDeps
803 for (Element m : clazz.getEnclosedElements()) {
804 if (m.getKind() != ElementKind.METHOD) {
807 ExecutableElement e = (ExecutableElement) m;
808 OnPropertyChange onPC = e.getAnnotation(OnPropertyChange.class);
812 for (String pn : onPC.value()) {
813 if (findPrprt(properties, pn) == null && findDerivedFrom(propDeps, pn).isEmpty()) {
814 error("No Prprt named '" + pn + "' in the model", clazz);
818 if (!e.getModifiers().contains(Modifier.STATIC)) {
819 error("@OnPrprtChange method needs to be static", e);
822 if (e.getModifiers().contains(Modifier.PRIVATE)) {
823 error("@OnPrprtChange method cannot be private", e);
826 if (e.getReturnType().getKind() != TypeKind.VOID) {
827 error("@OnPrprtChange method should return void", e);
830 String n = e.getSimpleName().toString();
833 for (String pn : onPC.value()) {
834 StringBuilder call = new StringBuilder();
835 call.append(" ").append(clazz.getSimpleName()).append(".").append(n).append("(");
836 call.append(wrapPropName(e, className, "name", pn));
839 Collection<String> change = functionDeps.get(pn);
840 if (change == null) {
841 change = new ArrayList<String>();
842 functionDeps.put(pn, change);
844 change.add(call.toString());
845 for (String dpn : findDerivedFrom(propDeps, pn)) {
846 change = functionDeps.get(dpn);
847 if (change == null) {
848 change = new ArrayList<String>();
849 functionDeps.put(dpn, change);
851 change.add(call.toString());
858 private boolean generateOperation(Element clazz,
859 StringWriter body, String className,
860 List<? extends Element> enclosedElements
862 for (Element m : enclosedElements) {
863 if (m.getKind() != ElementKind.METHOD) {
866 ExecutableElement e = (ExecutableElement)m;
867 ModelOperation mO = e.getAnnotation(ModelOperation.class);
871 if (!e.getModifiers().contains(Modifier.STATIC)) {
872 error("@ModelOperation method needs to be static", e);
875 if (e.getModifiers().contains(Modifier.PRIVATE)) {
876 error("@ModelOperation method cannot be private", e);
879 if (e.getReturnType().getKind() != TypeKind.VOID) {
880 error("@ModelOperation method should return void", e);
883 List<String> args = new ArrayList<String>();
885 body.append(" public void ").append(m.getSimpleName()).append("(");
887 boolean checkFirst = true;
888 for (VariableElement ve : e.getParameters()) {
889 final TypeMirror type = ve.asType();
890 CharSequence simpleName;
891 if (type.getKind() == TypeKind.DECLARED) {
892 simpleName = ((DeclaredType)type).asElement().getSimpleName();
894 simpleName = type.toString();
896 if (simpleName.toString().equals(className)) {
900 error("First parameter of @ModelOperation method must be " + className, m);
903 args.add(ve.getSimpleName().toString());
904 body.append(sep).append("final ");
905 body.append(ve.asType().toString()).append(" ");
906 body.append(ve.toString());
910 body.append(") {\n");
911 body.append(" org.netbeans.html.json.impl.JSON.runInBrowser(this.context, new Runnable() { public void run() {\n");
912 body.append(" ").append(clazz.getSimpleName()).append(".").append(m.getSimpleName()).append("(");
913 body.append(className).append(".this");
914 for (String s : args) {
915 body.append(", ").append(s);
918 body.append(" }});\n");
927 private boolean generateReceive(
928 Element clazz, StringWriter body, String className,
929 List<? extends Element> enclosedElements, List<String> functions
931 for (Element m : enclosedElements) {
932 if (m.getKind() != ElementKind.METHOD) {
935 ExecutableElement e = (ExecutableElement)m;
936 OnReceive onR = e.getAnnotation(OnReceive.class);
940 if (!e.getModifiers().contains(Modifier.STATIC)) {
941 error("@OnReceive method needs to be static", e);
944 if (e.getModifiers().contains(Modifier.PRIVATE)) {
945 error("@OnReceive method cannot be private", e);
948 if (e.getReturnType().getKind() != TypeKind.VOID) {
949 error("@OnReceive method should return void", e);
952 if (!onR.jsonp().isEmpty() && !"GET".equals(onR.method())) {
953 error("JSONP works only with GET transport method", e);
955 String dataMirror = findDataSpecified(e, onR);
956 if ("PUT".equals(onR.method()) && dataMirror == null) {
957 error("PUT method needs to specify a data() class", e);
960 if ("POST".equals(onR.method()) && dataMirror == null) {
961 error("POST method needs to specify a data() class", e);
964 String modelClass = null;
965 boolean expectsList = false;
966 List<String> args = new ArrayList<String>();
968 for (VariableElement ve : e.getParameters()) {
969 TypeMirror modelType = null;
970 final TypeMirror type = ve.asType();
971 CharSequence simpleName;
972 if (type.getKind() == TypeKind.DECLARED) {
973 simpleName = ((DeclaredType)type).asElement().getSimpleName();
975 simpleName = type.toString();
977 if (simpleName.toString().equals(className)) {
978 args.add(className + ".this");
979 } else if (isModel(ve.asType())) {
980 modelType = ve.asType();
981 } else if (ve.asType().getKind() == TypeKind.ARRAY) {
982 modelType = ((ArrayType)ve.asType()).getComponentType();
984 } else if (ve.asType().toString().equals("java.lang.String")) {
985 modelType = ve.asType();
987 if (modelType != null) {
988 if (modelClass != null) {
989 error("There can be only one model class among arguments", e);
991 modelClass = modelType.toString();
1001 if (modelClass == null) {
1002 error("The method needs to have one @Model class as parameter", e);
1004 String n = e.getSimpleName().toString();
1005 if ("WebSocket".equals(onR.method())) {
1006 body.append(" /** Performs WebSocket communication. Call with <code>null</code> data parameter\n");
1007 body.append(" * to open the connection (even if not required). Call with non-null data to\n");
1008 body.append(" * send messages to server. Call again with <code>null</code> data to close the socket.\n");
1009 body.append(" */\n");
1011 body.append(" public void ").append(n).append("(");
1012 StringBuilder urlBefore = new StringBuilder();
1013 StringBuilder urlAfter = new StringBuilder();
1014 String jsonpVarName = null;
1017 boolean skipJSONP = onR.jsonp().isEmpty();
1018 for (String p : findParamNames(e, onR.url(), onR.jsonp(), urlBefore, urlAfter)) {
1019 if (!skipJSONP && p.equals(onR.jsonp())) {
1025 body.append("String ").append(p);
1030 "Name of jsonp attribute ('" + onR.jsonp() +
1031 "') is not used in url attribute '" + onR.url() + "'", e
1034 if (dataMirror != null) {
1035 body.append(sep).append(dataMirror.toString()).append(" data");
1038 body.append(") {\n");
1039 boolean webSocket = onR.method().equals("WebSocket");
1041 if (generateWSReceiveBody(body, onR, e, clazz, className, expectsList, modelClass, n, args, urlBefore, jsonpVarName, urlAfter, dataMirror)) {
1044 body.append(" }\n");
1045 body.append(" private org.netbeans.html.json.impl.JSON.WS ws_" + e.getSimpleName() + ";\n");
1047 if (generateJSONReceiveBody(body, onR, e, clazz, className, expectsList, modelClass, n, args, urlBefore, jsonpVarName, urlAfter, dataMirror)) {
1050 body.append(" }\n");
1056 private boolean generateJSONReceiveBody(StringWriter body, OnReceive onR, ExecutableElement e, Element clazz, String className, boolean expectsList, String modelClass, String n, List<String> args, StringBuilder urlBefore, String jsonpVarName, StringBuilder urlAfter, String dataMirror) {
1058 " class ProcessResult extends org.netbeans.html.json.impl.RcvrJSON {\n" +
1060 " public void onError(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n" +
1061 " Exception value = ev.getException();\n"
1063 if (onR.onError().isEmpty()) {
1065 " value.printStackTrace();\n"
1068 if (!findOnError(e, ((TypeElement)clazz), onR.onError(), className)) {
1071 body.append(" ").append(clazz.getSimpleName()).append(".").append(onR.onError()).append("(");
1072 body.append(className).append(".this, value);\n");
1077 " public void onMessage(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n"
1081 " " + modelClass + "[] arr = new " + modelClass + "[ev.dataSize()];\n"
1085 " " + modelClass + "[] arr = { null };\n"
1089 " ev.dataRead(context, " + modelClass + ".class, arr);\n"
1092 body.append(" ").append(clazz.getSimpleName()).append(".").append(n).append("(");
1094 for (String arg : args) {
1099 body.append(");\n");
1105 body.append(" ProcessResult pr = new ProcessResult();\n");
1106 body.append(" org.netbeans.html.json.impl.JSON.loadJSON(context, pr,\n ");
1107 body.append(urlBefore).append(", ");
1108 if (jsonpVarName != null) {
1109 body.append(urlAfter);
1111 body.append("null");
1113 if (!"GET".equals(onR.method()) || dataMirror != null) {
1114 body.append(", \"").append(onR.method()).append('"');
1115 if (dataMirror != null) {
1116 body.append(", data");
1118 body.append(", null");
1121 body.append(", null, null");
1123 body.append(");\n");
1127 private boolean generateWSReceiveBody(StringWriter body, OnReceive onR, ExecutableElement e, Element clazz, String className, boolean expectsList, String modelClass, String n, List<String> args, StringBuilder urlBefore, String jsonpVarName, StringBuilder urlAfter, String dataMirror) {
1129 " class ProcessResult extends org.netbeans.html.json.impl.RcvrJSON {\n" +
1131 " public void onOpen(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n"
1133 body.append(" ").append(clazz.getSimpleName()).append(".").append(n).append("(");
1136 for (String arg : args) {
1138 if (arg.startsWith("arr")) {
1139 body.append("null");
1146 body.append(");\n");
1150 " public void onError(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n" +
1151 " Exception value = ev.getException();\n"
1153 if (onR.onError().isEmpty()) {
1155 " value.printStackTrace();\n"
1158 if (!findOnError(e, ((TypeElement)clazz), onR.onError(), className)) {
1161 body.append(" ").append(clazz.getSimpleName()).append(".").append(onR.onError()).append("(");
1162 body.append(className).append(".this, value);\n");
1167 " public void onMessage(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n"
1171 " " + modelClass + "[] arr = new " + modelClass + "[ev.dataSize()];\n"
1175 " " + modelClass + "[] arr = { null };\n"
1179 " ev.dataRead(context, " + modelClass + ".class, arr);\n"
1182 body.append(" ").append(clazz.getSimpleName()).append(".").append(n).append("(");
1184 for (String arg : args) {
1189 body.append(");\n");
1194 if (!onR.onError().isEmpty()) {
1197 + " public void onClose(org.netbeans.html.json.impl.RcvrJSON.MsgEvnt ev) {\n"
1199 body.append(" ").append(clazz.getSimpleName()).append(".").append(onR.onError()).append("(");
1200 body.append(className).append(".this, null);\n");
1205 body.append(" }\n");
1206 body.append(" if (this.ws_").append(e.getSimpleName()).append(" == null) {\n");
1207 body.append(" ProcessResult pr = new ProcessResult();\n");
1208 body.append(" this.ws_").append(e.getSimpleName());
1209 body.append("= org.netbeans.html.json.impl.JSON.openWS(context, pr,\n ");
1210 body.append(urlBefore).append(", data);\n");
1211 body.append(" } else {\n");
1212 body.append(" this.ws_").append(e.getSimpleName()).append(".send(context, ").append(urlBefore).append(", data);\n");
1213 body.append(" }\n");
1217 private CharSequence wrapParams(
1218 ExecutableElement ee, String id, String className, String evName, String dataName
1220 TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
1221 StringBuilder params = new StringBuilder();
1222 boolean first = true;
1223 for (VariableElement ve : ee.getParameters()) {
1225 params.append(", ");
1228 String toCall = null;
1229 String toFinish = null;
1230 if (ve.asType() == stringType) {
1231 if (ve.getSimpleName().contentEquals("id")) {
1232 params.append('"').append(id).append('"');
1235 toCall = "org.netbeans.html.json.impl.JSON.toString(context, ";
1237 if (ve.asType().getKind() == TypeKind.DOUBLE) {
1238 toCall = "org.netbeans.html.json.impl.JSON.toNumber(context, ";
1239 toFinish = ".doubleValue()";
1241 if (ve.asType().getKind() == TypeKind.INT) {
1242 toCall = "org.netbeans.html.json.impl.JSON.toNumber(context, ";
1243 toFinish = ".intValue()";
1245 if (dataName != null && ve.getSimpleName().contentEquals(dataName) && isModel(ve.asType())) {
1246 toCall = "org.netbeans.html.json.impl.JSON.toModel(context, " + ve.asType() + ".class, ";
1249 if (toCall != null) {
1250 params.append(toCall);
1251 if (dataName != null && ve.getSimpleName().contentEquals(dataName)) {
1252 params.append(dataName);
1253 params.append(", null");
1255 if (evName == null) {
1256 final StringBuilder sb = new StringBuilder();
1257 sb.append("Unexpected string parameter name.");
1258 if (dataName != null) {
1259 sb.append(" Try \"").append(dataName).append("\"");
1261 error(sb.toString(), ee);
1263 params.append(evName);
1264 params.append(", \"");
1265 params.append(ve.getSimpleName().toString());
1266 params.append("\"");
1269 if (toFinish != null) {
1270 params.append(toFinish);
1274 String rn = fqn(ve.asType(), ee);
1275 int last = rn.lastIndexOf('.');
1277 rn = rn.substring(last + 1);
1279 if (rn.equals(className)) {
1280 params.append(className).append(".this");
1284 "The annotated method can only accept " + className + " argument or argument named 'data'",
1292 private CharSequence wrapPropName(
1293 ExecutableElement ee, String className, String propName, String propValue
1295 TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
1296 StringBuilder params = new StringBuilder();
1297 boolean first = true;
1298 for (VariableElement ve : ee.getParameters()) {
1300 params.append(", ");
1303 if (ve.asType() == stringType) {
1304 if (propName != null && ve.getSimpleName().contentEquals(propName)) {
1305 params.append('"').append(propValue).append('"');
1307 error("Unexpected string parameter name. Try \"" + propName + "\".", ee);
1311 String rn = fqn(ve.asType(), ee);
1312 int last = rn.lastIndexOf('.');
1314 rn = rn.substring(last + 1);
1316 if (rn.equals(className)) {
1317 params.append(className).append(".this");
1321 "@OnPrprtChange method can only accept String or " + className + " arguments",
1327 private boolean isModel(TypeMirror tm) {
1328 if (tm.getKind() == TypeKind.ERROR) {
1331 final Element e = processingEnv.getTypeUtils().asElement(tm);
1335 for (Element ch : e.getEnclosedElements()) {
1336 if (ch.getKind() == ElementKind.METHOD) {
1337 ExecutableElement ee = (ExecutableElement)ch;
1338 if (ee.getParameters().isEmpty() && ee.getSimpleName().contentEquals("modelFor")) {
1343 return models.values().contains(e.getSimpleName().toString());
1346 private void writeToString(Prprt[] props, Writer w) throws IOException {
1347 w.write(" public String toString() {\n");
1348 w.write(" StringBuilder sb = new StringBuilder();\n");
1349 w.write(" sb.append('{');\n");
1351 for (Prprt p : props) {
1353 w.append(" sb.append('\"').append(\"" + p.name() + "\")");
1354 w.append(".append('\"').append(\":\");\n");
1355 w.append(" sb.append(org.netbeans.html.json.impl.JSON.toJSON(prop_");
1356 w.append(p.name()).append("));\n");
1357 sep = " sb.append(',');\n";
1359 w.write(" sb.append('}');\n");
1360 w.write(" return sb.toString();\n");
1363 private void writeClone(String className, Prprt[] props, Writer w) throws IOException {
1364 w.write(" public " + className + " clone() {\n");
1365 w.write(" return clone(context);\n");
1367 w.write(" private " + className + " clone(net.java.html.BrwsrCtx ctx) {\n");
1368 w.write(" " + className + " ret = new " + className + "(ctx);\n");
1369 for (Prprt p : props) {
1371 boolean isModel[] = { false };
1372 boolean isEnum[] = { false };
1373 boolean isPrimitive[] = { false };
1374 checkType(p, isModel, isEnum, isPrimitive);
1376 w.write(" ret.prop_" + p.name() + " = prop_" + p.name() + ";\n");
1379 w.write(" ret.prop_" + p.name() + " = prop_" + p.name() + " == null ? null : prop_" + p.name() + ".clone();\n");
1381 w.write(" ret.prop_" + p.name() + ".cloneAll(ctx, prop_" + p.name() + ");\n");
1385 w.write(" return ret;\n");
1389 private String inPckName(Element e) {
1390 StringBuilder sb = new StringBuilder();
1391 while (e.getKind() != ElementKind.PACKAGE) {
1392 if (sb.length() == 0) {
1393 sb.append(e.getSimpleName());
1396 sb.insert(0, e.getSimpleName());
1398 e = e.getEnclosingElement();
1400 return sb.toString();
1403 private String fqn(TypeMirror pt, Element relative) {
1404 if (pt.getKind() == TypeKind.ERROR) {
1405 final Elements eu = processingEnv.getElementUtils();
1406 PackageElement pckg = eu.getPackageOf(relative);
1407 return pckg.getQualifiedName() + "." + pt.toString();
1409 return pt.toString();
1412 private String checkType(Prprt p, boolean[] isModel, boolean[] isEnum, boolean[] isPrimitive) {
1415 String ret = p.typeName(processingEnv);
1416 TypeElement e = processingEnv.getElementUtils().getTypeElement(ret);
1420 isPrimitive[0] = false;
1424 } catch (MirroredTypeException ex) {
1425 tm = ex.getTypeMirror();
1427 tm = processingEnv.getTypeUtils().erasure(tm);
1428 if (isPrimitive[0] = tm.getKind().isPrimitive()) {
1431 return tm.toString();
1433 final Element e = processingEnv.getTypeUtils().asElement(tm);
1434 if (e.getKind() == ElementKind.CLASS && tm.getKind() == TypeKind.ERROR) {
1437 return e.getSimpleName().toString();
1440 final Model m = e == null ? null : e.getAnnotation(Model.class);
1443 ret = findPkgName(e) + '.' + m.className();
1445 models.put(e, m.className());
1446 } else if (findModelForMthd(e)) {
1447 ret = ((TypeElement)e).getQualifiedName().toString();
1450 ret = tm.toString();
1452 TypeMirror enm = processingEnv.getElementUtils().getTypeElement("java.lang.Enum").asType();
1453 enm = processingEnv.getTypeUtils().erasure(enm);
1454 isEnum[0] = processingEnv.getTypeUtils().isSubtype(tm, enm);
1458 private static boolean findModelForMthd(Element clazz) {
1459 if (clazz == null) {
1462 for (Element e : clazz.getEnclosedElements()) {
1463 if (e.getKind() == ElementKind.METHOD) {
1464 ExecutableElement ee = (ExecutableElement)e;
1466 ee.getSimpleName().contentEquals("modelFor") &&
1467 ee.getParameters().isEmpty()
1476 private Iterable<String> findParamNames(
1477 Element e, String url, String jsonParam, StringBuilder... both
1479 List<String> params = new ArrayList<String>();
1482 for (int pos = 0; ;) {
1483 int next = url.indexOf('{', pos);
1485 both[wasJSON].append('"')
1486 .append(url.substring(pos))
1490 int close = url.indexOf('}', next);
1492 error("Unbalanced '{' and '}' in " + url, e);
1495 final String paramName = url.substring(next + 1, close);
1496 params.add(paramName);
1497 if (paramName.equals(jsonParam) && !jsonParam.isEmpty()) {
1498 both[wasJSON].append('"')
1499 .append(url.substring(pos, next))
1503 both[wasJSON].append('"')
1504 .append(url.substring(pos, next))
1505 .append("\" + ").append(paramName).append(" + ");
1511 private static Prprt findPrprt(Prprt[] properties, String propName) {
1512 for (Prprt p : properties) {
1513 if (propName.equals(p.name())) {
1520 private boolean isPrimitive(String type) {
1522 "int".equals(type) ||
1523 "double".equals(type) ||
1524 "long".equals(type) ||
1525 "short".equals(type) ||
1526 "byte".equals(type) ||
1527 "char".equals(type) ||
1528 "boolean".equals(type) ||
1529 "float".equals(type);
1532 private static Collection<String> findDerivedFrom(Map<String, Collection<String>> propsDeps, String derivedProp) {
1533 Set<String> names = new HashSet<String>();
1534 for (Map.Entry<String, Collection<String>> e : propsDeps.entrySet()) {
1535 if (e.getValue().contains(derivedProp)) {
1536 names.add(e.getKey());
1542 private Prprt[] createProps(Element e, Property[] arr) {
1543 Prprt[] ret = Prprt.wrap(processingEnv, e, arr);
1544 Prprt[] prev = verify.put(e, ret);
1546 error("Two sets of properties for ", e);
1551 private static String strip(String s) {
1552 int indx = s.indexOf("__");
1554 return s.substring(0, indx);
1560 private String findDataSpecified(ExecutableElement e, OnReceive onR) {
1562 return onR.data().getName();
1563 } catch (MirroredTypeException ex) {
1564 final TypeMirror tm = ex.getTypeMirror();
1566 final Element te = processingEnv.getTypeUtils().asElement(tm);
1567 if (te.getKind() == ElementKind.CLASS && tm.getKind() == TypeKind.ERROR) {
1568 name = te.getSimpleName().toString();
1570 name = tm.toString();
1572 return "java.lang.Object".equals(name) ? null : name;
1573 } catch (Exception ex) {
1577 AnnotationMirror found = null;
1578 for (AnnotationMirror am : e.getAnnotationMirrors()) {
1579 if (am.getAnnotationType().toString().equals(OnReceive.class.getName())) {
1583 if (found == null) {
1587 for (Map.Entry<? extends ExecutableElement, ? extends AnnotationValue> entry : found.getElementValues().entrySet()) {
1588 ExecutableElement ee = entry.getKey();
1589 AnnotationValue av = entry.getValue();
1590 if (ee.getSimpleName().contentEquals("data")) {
1591 List<? extends Object> values = getAnnoValues(processingEnv, e, found);
1592 for (Object v : values) {
1593 String sv = v.toString();
1594 if (sv.startsWith("data = ") && sv.endsWith(".class")) {
1595 return sv.substring(7, sv.length() - 6);
1604 static List<? extends Object> getAnnoValues(ProcessingEnvironment pe, Element e, AnnotationMirror am) {
1606 Class<?> trees = Class.forName("com.sun.tools.javac.api.JavacTrees");
1607 Method m = trees.getMethod("instance", ProcessingEnvironment.class);
1608 Object instance = m.invoke(null, pe);
1609 m = instance.getClass().getMethod("getPath", Element.class, AnnotationMirror.class);
1610 Object path = m.invoke(instance, e, am);
1611 m = path.getClass().getMethod("getLeaf");
1612 Object leaf = m.invoke(path);
1613 m = leaf.getClass().getMethod("getArguments");
1614 return (List) m.invoke(leaf);
1615 } catch (Exception ex) {
1616 return Collections.emptyList();
1620 private static class Prprt {
1621 private final Element e;
1622 private final AnnotationMirror tm;
1623 private final Property p;
1625 public Prprt(Element e, AnnotationMirror tm, Property p) {
1639 String typeName(ProcessingEnvironment env) {
1640 RuntimeException ex;
1642 return p.type().getName();
1643 } catch (IncompleteAnnotationException e) {
1645 } catch (AnnotationTypeMismatchException e) {
1648 for (Object v : getAnnoValues(env, e, tm)) {
1649 String s = v.toString().replace(" ", "");
1650 if (s.startsWith("type=") && s.endsWith(".class")) {
1651 return s.substring(5, s.length() - 6);
1658 static Prprt[] wrap(ProcessingEnvironment pe, Element e, Property[] arr) {
1659 if (arr.length == 0) {
1660 return new Prprt[0];
1663 if (e.getKind() != ElementKind.CLASS) {
1664 throw new IllegalStateException("" + e.getKind());
1666 TypeElement te = (TypeElement)e;
1667 List<? extends AnnotationValue> val = null;
1668 for (AnnotationMirror an : te.getAnnotationMirrors()) {
1669 for (Map.Entry<? extends ExecutableElement, ? extends AnnotationValue> entry : an.getElementValues().entrySet()) {
1670 if (entry.getKey().getSimpleName().contentEquals("properties")) {
1671 val = (List)entry.getValue().getValue();
1676 if (val == null || val.size() != arr.length) {
1677 pe.getMessager().printMessage(Diagnostic.Kind.ERROR, "" + val, e);
1678 return new Prprt[0];
1680 Prprt[] ret = new Prprt[arr.length];
1681 BIG: for (int i = 0; i < ret.length; i++) {
1682 AnnotationMirror am = (AnnotationMirror)val.get(i).getValue();
1683 ret[i] = new Prprt(e, am, arr[i]);
1691 public Iterable<? extends Completion> getCompletions(Element element, AnnotationMirror annotation, ExecutableElement member, String userText) {
1692 final Level l = Level.FINE;
1693 LOG.log(l, " element: {0}", element);
1694 LOG.log(l, " annotation: {0}", annotation);
1695 LOG.log(l, " member: {0}", member);
1696 LOG.log(l, " userText: {0}", userText);
1697 LOG.log(l, "str: {0}", annotation.getAnnotationType().toString());
1698 if (annotation.getAnnotationType().toString().equals(OnReceive.class.getName())) {
1699 if (member.getSimpleName().contentEquals("method")) {
1700 return Arrays.asList(
1706 methodOf("WebSocket")
1711 return super.getCompletions(element, annotation, member, userText);
1714 private static final Completion methodOf(String method) {
1715 ResourceBundle rb = ResourceBundle.getBundle("org.netbeans.html.json.impl.Bundle");
1716 return Completions.of('"' + method + '"', rb.getString("MSG_Completion_" + method));
1719 private boolean findOnError(ExecutableElement errElem, TypeElement te, String name, String className) {
1722 for (Element e : te.getEnclosedElements()) {
1723 if (e.getKind() != ElementKind.METHOD) {
1726 if (!e.getSimpleName().contentEquals(name)) {
1729 if (!e.getModifiers().contains(Modifier.STATIC)) {
1730 errElem = (ExecutableElement) e;
1731 err = "Would have to be static";
1734 ExecutableElement ee = (ExecutableElement) e;
1735 TypeMirror excType = processingEnv.getElementUtils().getTypeElement(Exception.class.getName()).asType();
1736 final List<? extends VariableElement> params = ee.getParameters();
1737 boolean error = false;
1738 if (params.size() != 2) {
1741 String firstType = params.get(0).asType().toString();
1742 int lastDot = firstType.lastIndexOf('.');
1743 if (lastDot != -1) {
1744 firstType = firstType.substring(lastDot + 1);
1746 if (!firstType.equals(className)) {
1749 if (!processingEnv.getTypeUtils().isAssignable(excType, params.get(1).asType())) {
1754 errElem = (ExecutableElement) e;
1755 err = "Error method first argument needs to be " + className + " and second Exception";
1761 err = "Cannot find " + name + "(" + className + ", Exception) method in this class";
1763 error(err, errElem);