json/src/main/java/org/netbeans/html/json/impl/ModelProcessor.java
author Jaroslav Tulach <jtulach@netbeans.org>
Tue, 15 Dec 2015 23:19:35 +0100
branchComponent257162
changeset 1035 3534e15dc446
parent 1030 02568f34628a
permissions -rw-r--r--
Introducing @Component annotation, adding basic annotation and providing its (reflection based) implementation in ko4j module.
jtulach@0
     1
/**
jaroslav@358
     2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
jtulach@0
     3
 *
jaroslav@551
     4
 * Copyright 2013-2014 Oracle and/or its affiliates. All rights reserved.
jtulach@0
     5
 *
jaroslav@358
     6
 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
jaroslav@358
     7
 * Other names may be trademarks of their respective owners.
jtulach@0
     8
 *
jaroslav@358
     9
 * The contents of this file are subject to the terms of either the GNU
jaroslav@358
    10
 * General Public License Version 2 only ("GPL") or the Common
jaroslav@358
    11
 * Development and Distribution License("CDDL") (collectively, the
jaroslav@358
    12
 * "License"). You may not use this file except in compliance with the
jaroslav@358
    13
 * License. You can obtain a copy of the License at
jaroslav@358
    14
 * http://www.netbeans.org/cddl-gplv2.html
jaroslav@358
    15
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
jaroslav@358
    16
 * specific language governing permissions and limitations under the
jaroslav@358
    17
 * License.  When distributing the software, include this License Header
jaroslav@358
    18
 * Notice in each file and include the License file at
jaroslav@358
    19
 * nbbuild/licenses/CDDL-GPL-2-CP.  Oracle designates this
jaroslav@358
    20
 * particular file as subject to the "Classpath" exception as provided
jaroslav@358
    21
 * by Oracle in the GPL Version 2 section of the License file that
jaroslav@358
    22
 * accompanied this code. If applicable, add the following below the
jaroslav@358
    23
 * License Header, with the fields enclosed by brackets [] replaced by
jaroslav@358
    24
 * your own identifying information:
jaroslav@358
    25
 * "Portions Copyrighted [year] [name of copyright owner]"
jaroslav@358
    26
 *
jaroslav@358
    27
 * Contributor(s):
jaroslav@358
    28
 *
jaroslav@358
    29
 * The Original Software is NetBeans. The Initial Developer of the Original
jaroslav@551
    30
 * Software is Oracle. Portions Copyright 2013-2014 Oracle. All Rights Reserved.
jaroslav@358
    31
 *
jaroslav@358
    32
 * If you wish your version of this file to be governed by only the CDDL
jaroslav@358
    33
 * or only the GPL Version 2, indicate your decision by adding
jaroslav@358
    34
 * "[Contributor] elects to include this software in this distribution
jaroslav@358
    35
 * under the [CDDL or GPL Version 2] license." If you do not indicate a
jaroslav@358
    36
 * single choice of license, a recipient has the option to distribute
jaroslav@358
    37
 * your version of this file under either the CDDL, the GPL Version 2 or
jaroslav@358
    38
 * to extend the choice of license to its licensees as provided above.
jaroslav@358
    39
 * However, if you add GPL Version 2 code and therefore, elected the GPL
jaroslav@358
    40
 * Version 2 license, then the option applies only if the new code is
jaroslav@358
    41
 * made subject to such option by the copyright holder.
jtulach@0
    42
 */
jaroslav@362
    43
package org.netbeans.html.json.impl;
jtulach@0
    44
jtulach@0
    45
import java.io.IOException;
jtulach@0
    46
import java.io.OutputStreamWriter;
jtulach@0
    47
import java.io.StringWriter;
jtulach@0
    48
import java.io.Writer;
jtulach@0
    49
import java.lang.annotation.AnnotationTypeMismatchException;
jtulach@0
    50
import java.lang.annotation.IncompleteAnnotationException;
jtulach@0
    51
import java.lang.reflect.Method;
jtulach@0
    52
import java.util.ArrayList;
jaroslav@70
    53
import java.util.Arrays;
jtulach@0
    54
import java.util.Collection;
jtulach@0
    55
import java.util.Collections;
jtulach@0
    56
import java.util.HashMap;
jtulach@0
    57
import java.util.HashSet;
jtulach@0
    58
import java.util.LinkedHashSet;
jtulach@0
    59
import java.util.List;
jtulach@0
    60
import java.util.Map;
jtulach@1035
    61
import java.util.Properties;
jaroslav@70
    62
import java.util.ResourceBundle;
jtulach@0
    63
import java.util.Set;
jtulach@0
    64
import java.util.WeakHashMap;
jaroslav@71
    65
import java.util.logging.Level;
jaroslav@70
    66
import java.util.logging.Logger;
jtulach@0
    67
import javax.annotation.processing.AbstractProcessor;
jaroslav@70
    68
import javax.annotation.processing.Completion;
jaroslav@70
    69
import javax.annotation.processing.Completions;
jtulach@0
    70
import javax.annotation.processing.ProcessingEnvironment;
jtulach@0
    71
import javax.annotation.processing.Processor;
jtulach@0
    72
import javax.annotation.processing.RoundEnvironment;
jtulach@0
    73
import javax.annotation.processing.SupportedAnnotationTypes;
jaroslav@58
    74
import javax.annotation.processing.SupportedSourceVersion;
jaroslav@58
    75
import javax.lang.model.SourceVersion;
jtulach@0
    76
import javax.lang.model.element.AnnotationMirror;
jtulach@0
    77
import javax.lang.model.element.AnnotationValue;
jtulach@0
    78
import javax.lang.model.element.Element;
jtulach@0
    79
import javax.lang.model.element.ElementKind;
jtulach@0
    80
import javax.lang.model.element.ExecutableElement;
jtulach@0
    81
import javax.lang.model.element.Modifier;
jtulach@1035
    82
import javax.lang.model.element.Name;
jtulach@0
    83
import javax.lang.model.element.PackageElement;
jtulach@0
    84
import javax.lang.model.element.TypeElement;
jtulach@0
    85
import javax.lang.model.element.VariableElement;
jtulach@0
    86
import javax.lang.model.type.ArrayType;
jaroslav@29
    87
import javax.lang.model.type.DeclaredType;
jtulach@0
    88
import javax.lang.model.type.MirroredTypeException;
jtulach@0
    89
import javax.lang.model.type.TypeKind;
jtulach@0
    90
import javax.lang.model.type.TypeMirror;
jtulach@0
    91
import javax.lang.model.util.Elements;
jtulach@0
    92
import javax.lang.model.util.Types;
jtulach@0
    93
import javax.tools.Diagnostic;
jtulach@0
    94
import javax.tools.FileObject;
jtulach@1035
    95
import javax.tools.StandardLocation;
jtulach@1035
    96
import net.java.html.json.Component;
jtulach@0
    97
import net.java.html.json.ComputedProperty;
jtulach@938
    98
import net.java.html.json.Function;
jtulach@0
    99
import net.java.html.json.Model;
jaroslav@240
   100
import net.java.html.json.ModelOperation;
jtulach@0
   101
import net.java.html.json.OnPropertyChange;
jtulach@0
   102
import net.java.html.json.OnReceive;
jtulach@0
   103
import net.java.html.json.Property;
jtulach@0
   104
import org.openide.util.lookup.ServiceProvider;
jtulach@0
   105
jaroslav@70
   106
/** Annotation processor to process {@link Model @Model} annotations and
jaroslav@70
   107
 * generate appropriate model classes.
jtulach@0
   108
 *
jtulach@790
   109
 * @author Jaroslav Tulach
jtulach@0
   110
 */
jtulach@0
   111
@ServiceProvider(service=Processor.class)
jaroslav@100
   112
@SupportedSourceVersion(SourceVersion.RELEASE_6)
jtulach@0
   113
@SupportedAnnotationTypes({
jtulach@1
   114
    "net.java.html.json.Model",
jaroslav@240
   115
    "net.java.html.json.ModelOperation",
jtulach@1
   116
    "net.java.html.json.Function",
jtulach@1
   117
    "net.java.html.json.OnReceive",
jtulach@1
   118
    "net.java.html.json.OnPropertyChange",
jtulach@1
   119
    "net.java.html.json.ComputedProperty",
jtulach@1035
   120
    "net.java.html.json.Property",
jtulach@1035
   121
    "net.java.html.json.Component"
jtulach@0
   122
})
jtulach@3
   123
public final class ModelProcessor extends AbstractProcessor {
jaroslav@70
   124
    private static final Logger LOG = Logger.getLogger(ModelProcessor.class.getName());
jaroslav@100
   125
    private final Map<Element,String> models = new WeakHashMap<Element,String>();
jaroslav@100
   126
    private final Map<Element,Prprt[]> verify = new WeakHashMap<Element,Prprt[]>();
jtulach@0
   127
    @Override
jtulach@0
   128
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
jtulach@0
   129
        boolean ok = true;
jtulach@0
   130
        for (Element e : roundEnv.getElementsAnnotatedWith(Model.class)) {
jtulach@0
   131
            if (!processModel(e)) {
jtulach@0
   132
                ok = false;
jtulach@0
   133
            }
jtulach@0
   134
        }
jtulach@1035
   135
        for (Element e : roundEnv.getElementsAnnotatedWith(Component.class)) {
jtulach@1035
   136
            if (!processComponent(e)) {
jtulach@1035
   137
                ok = false;
jtulach@1035
   138
            }
jtulach@1035
   139
        }
jtulach@0
   140
        if (roundEnv.processingOver()) {
jtulach@0
   141
            models.clear();
jtulach@0
   142
            for (Map.Entry<Element, Prprt[]> entry : verify.entrySet()) {
jtulach@0
   143
                TypeElement te = (TypeElement)entry.getKey();
jtulach@926
   144
                String fqn = te.getQualifiedName().toString();
jtulach@0
   145
                Element finalElem = processingEnv.getElementUtils().getTypeElement(fqn);
jtulach@0
   146
                if (finalElem == null) {
jtulach@0
   147
                    continue;
jtulach@0
   148
                }
jtulach@0
   149
                Prprt[] props;
jtulach@0
   150
                Model m = finalElem.getAnnotation(Model.class);
jtulach@3
   151
                if (m == null) {
jtulach@3
   152
                    continue;
jtulach@3
   153
                }
jtulach@0
   154
                props = Prprt.wrap(processingEnv, finalElem, m.properties());
jtulach@0
   155
                for (Prprt p : props) {
jtulach@0
   156
                    boolean[] isModel = { false };
jtulach@0
   157
                    boolean[] isEnum = { false };
jtulach@0
   158
                    boolean[] isPrimitive = { false };
jtulach@0
   159
                    String t = checkType(p, isModel, isEnum, isPrimitive);
jtulach@0
   160
                    if (isEnum[0]) {
jtulach@0
   161
                        continue;
jtulach@0
   162
                    }
jtulach@0
   163
                    if (isPrimitive[0]) {
jtulach@0
   164
                        continue;
jtulach@0
   165
                    }
jtulach@0
   166
                    if (isModel[0]) {
jtulach@0
   167
                        continue;
jtulach@0
   168
                    }
jtulach@0
   169
                    if ("java.lang.String".equals(t)) {
jtulach@0
   170
                        continue;
jtulach@0
   171
                    }
jtulach@0
   172
                    error("The type " + t + " should be defined by @Model annotation", entry.getKey());
jtulach@0
   173
                }
jtulach@0
   174
            }
jtulach@0
   175
            verify.clear();
jtulach@0
   176
        }
jtulach@0
   177
        return ok;
jtulach@0
   178
    }
jtulach@0
   179
jtulach@0
   180
    private void error(String msg, Element e) {
jtulach@0
   181
        processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR, msg, e);
jtulach@0
   182
    }
jtulach@938
   183
jtulach@0
   184
    private boolean processModel(Element e) {
jtulach@0
   185
        boolean ok = true;
jtulach@0
   186
        Model m = e.getAnnotation(Model.class);
jtulach@0
   187
        if (m == null) {
jtulach@0
   188
            return true;
jtulach@0
   189
        }
jtulach@0
   190
        String pkg = findPkgName(e);
jtulach@0
   191
        Writer w;
jtulach@0
   192
        String className = m.className();
jtulach@0
   193
        models.put(e, className);
jtulach@0
   194
        try {
jtulach@0
   195
            StringWriter body = new StringWriter();
jaroslav@386
   196
            StringBuilder onReceiveType = new StringBuilder();
jtulach@823
   197
            List<GetSet> propsGetSet = new ArrayList<GetSet>();
jaroslav@597
   198
            List<Object> functions = new ArrayList<Object>();
jaroslav@567
   199
            Map<String, Collection<String[]>> propsDeps = new HashMap<String, Collection<String[]>>();
jaroslav@100
   200
            Map<String, Collection<String>> functionDeps = new HashMap<String, Collection<String>>();
jtulach@0
   201
            Prprt[] props = createProps(e, m.properties());
jtulach@1017
   202
            final String builderPrefix = findBuilderPrefix(e, m);
jtulach@938
   203
jtulach@951
   204
            if (!generateComputedProperties(className, body, props, e.getEnclosedElements(), propsGetSet, propsDeps)) {
jtulach@0
   205
                ok = false;
jtulach@0
   206
            }
jtulach@0
   207
            if (!generateOnChange(e, propsDeps, props, className, functionDeps)) {
jtulach@0
   208
                ok = false;
jtulach@0
   209
            }
jtulach@1017
   210
            if (!generateProperties(e, builderPrefix, body, className, props, propsGetSet, propsDeps, functionDeps)) {
jtulach@0
   211
                ok = false;
jtulach@0
   212
            }
jtulach@0
   213
            if (!generateFunctions(e, body, className, e.getEnclosedElements(), functions)) {
jtulach@0
   214
                ok = false;
jtulach@0
   215
            }
jaroslav@598
   216
            int functionsCount = functions.size() / 2;
jtulach@648
   217
            for (int i = 0; i < functions.size(); i += 2) {
jtulach@648
   218
                for (Prprt p : props) {
jtulach@648
   219
                    if (p.name().equals(functions.get(i))) {
jtulach@648
   220
                        error("Function cannot have the name of an existing property", e);
jtulach@648
   221
                        ok = false;
jtulach@648
   222
                    }
jtulach@648
   223
                }
jtulach@648
   224
            }
jaroslav@386
   225
            if (!generateReceive(e, body, className, e.getEnclosedElements(), onReceiveType)) {
jaroslav@12
   226
                ok = false;
jaroslav@12
   227
            }
jaroslav@598
   228
            if (!generateOperation(e, body, className, e.getEnclosedElements(), functions)) {
jaroslav@240
   229
                ok = false;
jaroslav@240
   230
            }
jtulach@0
   231
            FileObject java = processingEnv.getFiler().createSourceFile(pkg + '.' + className, e);
jtulach@0
   232
            w = new OutputStreamWriter(java.openOutputStream());
jtulach@0
   233
            try {
jtulach@0
   234
                w.append("package " + pkg + ";\n");
jtulach@2
   235
                w.append("import net.java.html.json.*;\n");
jtulach@1023
   236
                final String inPckName = inPckName(e, false);
jtulach@1014
   237
                w.append("/** Generated for {@link ").append(inPckName).append("}*/\n");
jtulach@0
   238
                w.append("public final class ").append(className).append(" implements Cloneable {\n");
jtulach@1014
   239
                w.append("  private static Class<").append(inPckName).append("> modelFor() { return ").append(inPckName).append(".class; }\n");
jaroslav@373
   240
                w.append("  private static final Html4JavaType TYPE = new Html4JavaType();\n");
jtulach@1023
   241
                if (m.instance()) {
jtulach@1023
   242
                    int cCnt = 0;
jtulach@1023
   243
                    for (Element c : e.getEnclosedElements()) {
jtulach@1023
   244
                        if (c.getKind() != ElementKind.CONSTRUCTOR) {
jtulach@1023
   245
                            continue;
jtulach@1023
   246
                        }
jtulach@1023
   247
                        cCnt++;
jtulach@1023
   248
                        ExecutableElement ec = (ExecutableElement) c;
jtulach@1023
   249
                        if (ec.getParameters().size() > 0) {
jtulach@1023
   250
                            continue;
jtulach@1023
   251
                        }
jtulach@1023
   252
                        if (ec.getModifiers().contains(Modifier.PRIVATE)) {
jtulach@1023
   253
                            continue;
jtulach@1023
   254
                        }
jtulach@1023
   255
                        cCnt = 0;
jtulach@1023
   256
                        break;
jtulach@1023
   257
                    }
jtulach@1023
   258
                    if (cCnt > 0) {
jtulach@1023
   259
                        ok = false;
jtulach@1023
   260
                        error("Needs non-private default constructor when instance=true", e);
jtulach@1023
   261
                        w.append("  private final ").append(inPckName).append(" instance = null;\n");
jtulach@1023
   262
                    } else {
jtulach@1023
   263
                        w.append("  private final ").append(inPckName).append(" instance = new ").append(inPckName).append("();\n");
jtulach@1023
   264
                    }
jtulach@1023
   265
                }
jtulach@838
   266
                w.append("  private final org.netbeans.html.json.spi.Proto proto;\n");
jtulach@0
   267
                w.append(body.toString());
jaroslav@111
   268
                w.append("  private ").append(className).append("(net.java.html.BrwsrCtx context) {\n");
jaroslav@410
   269
                w.append("    this.proto = TYPE.createProto(this, context);\n");
jaroslav@374
   270
                for (Prprt p : props) {
jaroslav@374
   271
                    if (p.array()) {
jtulach@949
   272
                        final String tn = typeName(p);
jaroslav@374
   273
                        String[] gs = toGetSet(p.name(), tn, p.array());
jaroslav@383
   274
                        w.write("    this.prop_" + p.name() + " = proto.createList(\""
jaroslav@374
   275
                            + p.name() + "\"");
jaroslav@383
   276
                        if (functionDeps.containsKey(p.name())) {
jaroslav@383
   277
                            int index = Arrays.asList(functionDeps.keySet().toArray()).indexOf(p.name());
jaroslav@383
   278
                            w.write(", " + index);
jaroslav@383
   279
                        } else {
jaroslav@383
   280
                            w.write(", -1");
jaroslav@383
   281
                        }
jaroslav@567
   282
                        Collection<String[]> dependants = propsDeps.get(p.name());
jaroslav@374
   283
                        if (dependants != null) {
jaroslav@567
   284
                            for (String[] depProp : dependants) {
jaroslav@374
   285
                                w.write(", ");
jaroslav@374
   286
                                w.write('\"');
jaroslav@567
   287
                                w.write(depProp[0]);
jaroslav@374
   288
                                w.write('\"');
jaroslav@374
   289
                            }
jaroslav@374
   290
                        }
jaroslav@374
   291
                        w.write(")");
jaroslav@374
   292
                        w.write(";\n");
jaroslav@379
   293
                    }
jaroslav@379
   294
                }
jaroslav@379
   295
                w.append("  };\n");
jaroslav@379
   296
                w.append("  public ").append(className).append("() {\n");
jaroslav@379
   297
                w.append("    this(net.java.html.BrwsrCtx.findDefault(").append(className).append(".class));\n");
jaroslav@379
   298
                for (Prprt p : props) {
jaroslav@379
   299
                    if (!p.array()) {
jaroslav@374
   300
                        boolean[] isModel = {false};
jaroslav@374
   301
                        boolean[] isEnum = {false};
jaroslav@374
   302
                        boolean isPrimitive[] = {false};
jaroslav@374
   303
                        String tn = checkType(p, isModel, isEnum, isPrimitive);
jaroslav@374
   304
                        if (isModel[0]) {
jaroslav@374
   305
                            w.write("    prop_" + p.name() + " = new " + tn + "();\n");
jaroslav@374
   306
                        }
jaroslav@374
   307
                    }
jaroslav@374
   308
                }
jtulach@0
   309
                w.append("  };\n");
jtulach@1017
   310
                if (props.length > 0 && builderPrefix == null) {
jtulach@1011
   311
                    StringBuilder constructorWithArguments = new StringBuilder();
jtulach@1011
   312
                    constructorWithArguments.append("  public ").append(className).append("(");
jaroslav@284
   313
                    Prprt firstArray = null;
jaroslav@284
   314
                    String sep = "";
jtulach@1011
   315
                    int parameterCount = 0;
jaroslav@284
   316
                    for (Prprt p : props) {
jaroslav@284
   317
                        if (p.array()) {
jaroslav@284
   318
                            if (firstArray == null) {
jaroslav@284
   319
                                firstArray = p;
jaroslav@284
   320
                            }
jaroslav@284
   321
                            continue;
jaroslav@232
   322
                        }
jtulach@949
   323
                        String tn = typeName(p);
jtulach@1011
   324
                        constructorWithArguments.append(sep);
jtulach@1011
   325
                        constructorWithArguments.append(tn);
jtulach@862
   326
                        String[] third = toGetSet(p.name(), tn, false);
jtulach@1011
   327
                        constructorWithArguments.append(" ").append(third[2]);
jaroslav@284
   328
                        sep = ", ";
jtulach@1011
   329
                        parameterCount++;
jaroslav@232
   330
                    }
jaroslav@284
   331
                    if (firstArray != null) {
jaroslav@312
   332
                        String tn;
jaroslav@312
   333
                        boolean[] isModel = {false};
jaroslav@312
   334
                        boolean[] isEnum = {false};
jaroslav@312
   335
                        boolean isPrimitive[] = {false};
jaroslav@312
   336
                        tn = checkType(firstArray, isModel, isEnum, isPrimitive);
jtulach@1011
   337
                        constructorWithArguments.append(sep);
jtulach@1011
   338
                        constructorWithArguments.append(tn);
jtulach@862
   339
                        String[] third = toGetSet(firstArray.name(), tn, true);
jtulach@1011
   340
                        constructorWithArguments.append("... ").append(third[2]);
jtulach@1011
   341
                        parameterCount++;
jaroslav@284
   342
                    }
jtulach@1011
   343
                    constructorWithArguments.append(") {\n");
jtulach@1011
   344
                    constructorWithArguments.append("    this(net.java.html.BrwsrCtx.findDefault(").append(className).append(".class));\n");
jaroslav@284
   345
                    for (Prprt p : props) {
jaroslav@284
   346
                        if (p.array()) {
jaroslav@284
   347
                            continue;
jaroslav@284
   348
                        }
jtulach@862
   349
                        String[] third = toGetSet(p.name(), null, false);
jtulach@1011
   350
                        constructorWithArguments.append("    this.prop_" + p.name() + " = " + third[2] + ";\n");
jaroslav@284
   351
                    }
jaroslav@284
   352
                    if (firstArray != null) {
jtulach@862
   353
                        String[] third = toGetSet(firstArray.name(), null, true);
jtulach@1011
   354
                        constructorWithArguments.append("    proto.initTo(this.prop_" + firstArray.name() + ", " + third[2] + ");\n");
jaroslav@284
   355
                    }
jtulach@1011
   356
                    constructorWithArguments.append("  };\n");
jtulach@1011
   357
                    if (parameterCount < 255) {
jtulach@1011
   358
                        w.write(constructorWithArguments.toString());
jtulach@1011
   359
                    }
jaroslav@232
   360
                }
jtulach@838
   361
                w.append("  private static class Html4JavaType extends org.netbeans.html.json.spi.Proto.Type<").append(className).append("> {\n");
jaroslav@373
   362
                w.append("    private Html4JavaType() {\n      super(").append(className).append(".class, ").
jtulach@1014
   363
                    append(inPckName).append(".class, " + propsGetSet.size() + ", "
jaroslav@598
   364
                    + functionsCount + ");\n");
jaroslav@275
   365
                {
jtulach@823
   366
                    for (int i = 0; i < propsGetSet.size(); i++) {
jtulach@823
   367
                        w.append("      registerProperty(\"").append(propsGetSet.get(i).name).append("\", ");
jtulach@824
   368
                        w.append((i) + ", " + propsGetSet.get(i).readOnly + ");\n");
jaroslav@275
   369
                    }
jaroslav@17
   370
                }
jaroslav@275
   371
                {
jaroslav@603
   372
                    for (int i = 0; i < functionsCount; i++) {
jaroslav@603
   373
                        w.append("      registerFunction(\"").append((String)functions.get(i * 2)).append("\", ");
jaroslav@603
   374
                        w.append(i + ");\n");
jaroslav@275
   375
                    }
jaroslav@20
   376
                }
jaroslav@373
   377
                w.append("    }\n");
jaroslav@374
   378
                w.append("    @Override public void setValue(" + className + " data, int type, Object value) {\n");
jaroslav@17
   379
                w.append("      switch (type) {\n");
jtulach@823
   380
                for (int i = 0; i < propsGetSet.size(); i++) {
jtulach@824
   381
                    final GetSet pgs = propsGetSet.get(i);
jtulach@824
   382
                    if (pgs.readOnly) {
jtulach@824
   383
                        continue;
jtulach@824
   384
                    }
jtulach@824
   385
                    final String set = pgs.setter;
jtulach@824
   386
                    String tn = pgs.type;
jaroslav@198
   387
                    String btn = findBoxedType(tn);
jaroslav@198
   388
                    if (btn != null) {
jaroslav@198
   389
                        tn = btn;
jaroslav@68
   390
                    }
jtulach@824
   391
                    w.append("        case " + i + ": ");
jtulach@824
   392
                    if (pgs.setter != null) {
jtulach@824
   393
                        w.append("data.").append(strip(pgs.setter)).append("(TYPE.extractValue(" + tn + ".class, value)); return;\n");
jtulach@824
   394
                    } else {
jtulach@824
   395
                        w.append("TYPE.replaceValue(data.").append(strip(pgs.getter)).append("(), " + tn + ".class, value); return;\n");
jaroslav@17
   396
                    }
jaroslav@17
   397
                }
jaroslav@17
   398
                w.append("      }\n");
jtulach@824
   399
                w.append("      throw new UnsupportedOperationException();\n");
jaroslav@17
   400
                w.append("    }\n");
jaroslav@374
   401
                w.append("    @Override public Object getValue(" + className + " data, int type) {\n");
jaroslav@17
   402
                w.append("      switch (type) {\n");
jtulach@823
   403
                for (int i = 0; i < propsGetSet.size(); i++) {
jtulach@823
   404
                    final String get = propsGetSet.get(i).getter;
jaroslav@17
   405
                    if (get != null) {
jtulach@823
   406
                        w.append("        case " + i + ": return data." + strip(get) + "();\n");
jaroslav@17
   407
                    }
jaroslav@17
   408
                }
jaroslav@17
   409
                w.append("      }\n");
jaroslav@17
   410
                w.append("      throw new UnsupportedOperationException();\n");
jaroslav@17
   411
                w.append("    }\n");
jaroslav@599
   412
                w.append("    @Override public void call(" + className + " model, int type, Object data, Object ev) throws Exception {\n");
jaroslav@20
   413
                w.append("      switch (type) {\n");
jaroslav@20
   414
                for (int i = 0; i < functions.size(); i += 2) {
jaroslav@597
   415
                    final String name = (String)functions.get(i);
jaroslav@598
   416
                    final Object param = functions.get(i + 1);
jaroslav@598
   417
                    if (param instanceof ExecutableElement) {
jaroslav@598
   418
                        ExecutableElement ee = (ExecutableElement)param;
jaroslav@598
   419
                        w.append("        case " + (i / 2) + ":\n");
jtulach@1023
   420
                        w.append("          ");
jtulach@1023
   421
                        if (m.instance()) {
jtulach@1023
   422
                            w.append("model.instance");
jtulach@1023
   423
                        } else {
jtulach@1023
   424
                            w.append(((TypeElement)e).getQualifiedName());
jtulach@1023
   425
                        }
jtulach@1023
   426
                        w.append(".").append(name).append("(");
jaroslav@598
   427
                        w.append(wrapParams(ee, null, className, "model", "ev", "data"));
jaroslav@598
   428
                        w.append(");\n");
jaroslav@598
   429
                        w.append("          return;\n");
jaroslav@598
   430
                    } else {
jaroslav@598
   431
                        String call = (String)param;
jaroslav@598
   432
                        w.append("        case " + (i / 2) + ":\n"); // model." + name + "(data, ev); return;\n");
jaroslav@598
   433
                        w.append("          ").append(call).append("\n");
jaroslav@598
   434
                        w.append("          return;\n");
jtulach@938
   435
jaroslav@598
   436
                    }
jaroslav@20
   437
                }
jaroslav@20
   438
                w.append("      }\n");
jaroslav@20
   439
                w.append("      throw new UnsupportedOperationException();\n");
jaroslav@20
   440
                w.append("    }\n");
jtulach@838
   441
                w.append("    @Override public org.netbeans.html.json.spi.Proto protoFor(Object obj) {\n");
jaroslav@384
   442
                w.append("      return ((" + className + ")obj).proto;");
jaroslav@384
   443
                w.append("    }\n");
jaroslav@383
   444
                w.append("    @Override public void onChange(" + className + " model, int type) {\n");
jaroslav@383
   445
                w.append("      switch (type) {\n");
jaroslav@383
   446
                {
jaroslav@383
   447
                    String[] arr = functionDeps.keySet().toArray(new String[0]);
jaroslav@383
   448
                    for (int i = 0; i < arr.length; i++) {
jaroslav@383
   449
                        Collection<String> onChange = functionDeps.get(arr[i]);
jaroslav@383
   450
                        if (onChange != null) {
jaroslav@383
   451
                            w.append("      case " + i + ":\n");
jaroslav@383
   452
                            for (String call : onChange) {
jaroslav@383
   453
                                w.append("      ").append(call).append("\n");
jaroslav@383
   454
                            }
jaroslav@383
   455
                            w.write("      return;\n");
jaroslav@383
   456
                        }
jaroslav@383
   457
                    }
jaroslav@383
   458
                }
jaroslav@383
   459
                w.append("    }\n");
jaroslav@383
   460
                w.append("      throw new UnsupportedOperationException();\n");
jaroslav@383
   461
                w.append("    }\n");
jaroslav@386
   462
                w.append(onReceiveType);
jaroslav@374
   463
                w.append("    @Override public " + className + " read(net.java.html.BrwsrCtx c, Object json) { return new " + className + "(c, json); }\n");
jaroslav@412
   464
                w.append("    @Override public " + className + " cloneTo(" + className + " o, net.java.html.BrwsrCtx c) { return o.clone(c); }\n");
jaroslav@17
   465
                w.append("  }\n");
jaroslav@110
   466
                w.append("  private ").append(className).append("(net.java.html.BrwsrCtx c, Object json) {\n");
jaroslav@373
   467
                w.append("    this(c);\n");
jtulach@0
   468
                int values = 0;
jtulach@823
   469
                for (int i = 0; i < propsGetSet.size(); i++) {
jtulach@823
   470
                    Prprt p = findPrprt(props, propsGetSet.get(i).name);
jtulach@0
   471
                    if (p == null) {
jtulach@0
   472
                        continue;
jtulach@0
   473
                    }
jtulach@0
   474
                    values++;
jtulach@0
   475
                }
jtulach@0
   476
                w.append("    Object[] ret = new Object[" + values + "];\n");
jaroslav@385
   477
                w.append("    proto.extract(json, new String[] {\n");
jtulach@823
   478
                for (int i = 0; i < propsGetSet.size(); i++) {
jtulach@823
   479
                    Prprt p = findPrprt(props, propsGetSet.get(i).name);
jtulach@0
   480
                    if (p == null) {
jtulach@0
   481
                        continue;
jtulach@0
   482
                    }
jtulach@823
   483
                    w.append("      \"").append(propsGetSet.get(i).name).append("\",\n");
jtulach@0
   484
                }
jtulach@0
   485
                w.append("    }, ret);\n");
jtulach@823
   486
                for (int i = 0, cnt = 0, prop = 0; i < propsGetSet.size(); i++) {
jtulach@823
   487
                    final String pn = propsGetSet.get(i).name;
jtulach@0
   488
                    Prprt p = findPrprt(props, pn);
jtulach@648
   489
                    if (p == null || prop >= props.length) {
jtulach@0
   490
                        continue;
jtulach@0
   491
                    }
jtulach@0
   492
                    boolean[] isModel = { false };
jtulach@0
   493
                    boolean[] isEnum = { false };
jtulach@0
   494
                    boolean isPrimitive[] = { false };
jtulach@0
   495
                    String type = checkType(props[prop++], isModel, isEnum, isPrimitive);
jtulach@0
   496
                    if (p.array()) {
jtulach@943
   497
                        w.append("    for (Object e : useAsArray(ret[" + cnt + "])) {\n");
jtulach@0
   498
                        if (isModel[0]) {
jtulach@943
   499
                            w.append("      this.prop_").append(pn).append(".add(proto.read");
jaroslav@385
   500
                            w.append("(" + type + ".class, e));\n");
jtulach@0
   501
                        } else if (isEnum[0]) {
jaroslav@66
   502
                            w.append("        this.prop_").append(pn);
jtulach@0
   503
                            w.append(".add(e == null ? null : ");
jaroslav@380
   504
                            w.append(type).append(".valueOf(TYPE.stringValue(e)));\n");
jtulach@0
   505
                        } else {
jtulach@0
   506
                            if (isPrimitive(type)) {
jtulach@1017
   507
                                if (type.equals("char")) {
jtulach@1017
   508
                                    w.append("        this.prop_").append(pn).append(".add((char)TYPE.numberValue(e).");
jtulach@1017
   509
                                    w.append("intValue());\n");
jtulach@1017
   510
                                } else {
jtulach@1017
   511
                                    w.append("        this.prop_").append(pn).append(".add(TYPE.numberValue(e).");
jtulach@1017
   512
                                    w.append(type).append("Value());\n");
jtulach@1017
   513
                                }
jtulach@0
   514
                            } else {
jaroslav@66
   515
                                w.append("        this.prop_").append(pn).append(".add((");
jtulach@0
   516
                                w.append(type).append(")e);\n");
jtulach@0
   517
                            }
jtulach@0
   518
                        }
jaroslav@66
   519
                        w.append("    }\n");
jtulach@0
   520
                    } else {
jtulach@0
   521
                        if (isEnum[0]) {
jaroslav@340
   522
                            w.append("    try {\n");
jtulach@0
   523
                            w.append("    this.prop_").append(pn);
jtulach@0
   524
                            w.append(" = ret[" + cnt + "] == null ? null : ");
jaroslav@380
   525
                            w.append(type).append(".valueOf(TYPE.stringValue(ret[" + cnt + "]));\n");
jaroslav@340
   526
                            w.append("    } catch (IllegalArgumentException ex) {\n");
jaroslav@340
   527
                            w.append("      ex.printStackTrace();\n");
jaroslav@340
   528
                            w.append("    }\n");
jtulach@0
   529
                        } else if (isPrimitive(type)) {
jtulach@0
   530
                            w.append("    this.prop_").append(pn);
jaroslav@100
   531
                            w.append(" = ret[" + cnt + "] == null ? ");
jaroslav@100
   532
                            if ("char".equals(type)) {
jaroslav@380
   533
                                w.append("0 : (TYPE.charValue(");
jaroslav@100
   534
                            } else if ("boolean".equals(type)) {
jaroslav@380
   535
                                w.append("false : (TYPE.boolValue(");
jaroslav@100
   536
                            } else {
jaroslav@380
   537
                                w.append("0 : (TYPE.numberValue(");
jaroslav@100
   538
                            }
jaroslav@197
   539
                            w.append("ret[" + cnt + "])).");
jtulach@0
   540
                            w.append(type).append("Value();\n");
jaroslav@136
   541
                        } else if (isModel[0]) {
jaroslav@385
   542
                            w.append("    this.prop_").append(pn).append(" = proto.read");
jaroslav@385
   543
                            w.append("(" + type + ".class, ");
jaroslav@136
   544
                            w.append("ret[" + cnt + "]);\n");
jaroslav@136
   545
                        }else {
jtulach@0
   546
                            w.append("    this.prop_").append(pn);
jtulach@0
   547
                            w.append(" = (").append(type).append(')');
jtulach@0
   548
                            w.append("ret[" + cnt + "];\n");
jtulach@0
   549
                        }
jtulach@0
   550
                    }
jtulach@0
   551
                    cnt++;
jtulach@0
   552
                }
jaroslav@373
   553
                w.append("  }\n");
jtulach@943
   554
                w.append("  private static Object[] useAsArray(Object o) {\n");
jtulach@943
   555
                w.append("    return o instanceof Object[] ? ((Object[])o) : o == null ? new Object[0] : new Object[] { o };\n");
jtulach@943
   556
                w.append("  }\n");
jtulach@0
   557
                writeToString(props, w);
jtulach@0
   558
                writeClone(className, props, w);
jtulach@920
   559
                String targetId = findTargetId(e);
jtulach@920
   560
                if (targetId != null) {
jtulach@920
   561
                    w.write("  /** Activates this model instance in the current {@link \n"
jtulach@920
   562
                        + "net.java.html.json.Models#bind(java.lang.Object, net.java.html.BrwsrCtx) browser context}. \n"
jtulach@920
   563
                        + "In case of using Knockout technology, this means to \n"
jtulach@920
   564
                        + "bind JSON like data in this model instance with Knockout tags in \n"
jtulach@920
   565
                        + "the surrounding HTML page.\n"
jtulach@920
   566
                    );
jtulach@920
   567
                    if (targetId != null) {
jtulach@920
   568
                        w.write("This method binds to element '" + targetId + "' on the page\n");
jtulach@920
   569
                    }
jtulach@920
   570
                    w.write(""
jtulach@920
   571
                        + "@return <code>this</code> object\n"
jtulach@920
   572
                        + "*/\n"
jtulach@920
   573
                    );
jtulach@920
   574
                    w.write("  public " + className + " applyBindings() {\n");
jtulach@920
   575
                    w.write("    proto.applyBindings();\n");
jtulach@920
   576
    //                w.write("    proto.applyBindings(id);\n");
jtulach@920
   577
                    w.write("    return this;\n");
jtulach@920
   578
                    w.write("  }\n");
jtulach@920
   579
                } else {
jtulach@920
   580
                    w.write("  private " + className + " applyBindings() {\n");
jtulach@920
   581
                    w.write("    throw new IllegalStateException(\"Please specify targetId=\\\"\\\" in your @Model annotation\");\n");
jtulach@920
   582
                    w.write("  }\n");
jtulach@920
   583
                }
jaroslav@15
   584
                w.write("  public boolean equals(Object o) {\n");
jaroslav@15
   585
                w.write("    if (o == this) return true;\n");
jaroslav@281
   586
                w.write("    if (!(o instanceof " + className + ")) return false;\n");
jaroslav@281
   587
                w.write("    " + className + " p = (" + className + ")o;\n");
jaroslav@281
   588
                for (Prprt p : props) {
jaroslav@380
   589
                    w.write("    if (!TYPE.isSame(prop_" + p.name() + ", p.prop_" + p.name() + ")) return false;\n");
jaroslav@281
   590
                }
jaroslav@281
   591
                w.write("    return true;\n");
jaroslav@281
   592
                w.write("  }\n");
jaroslav@281
   593
                w.write("  public int hashCode() {\n");
jaroslav@281
   594
                w.write("    int h = " + className + ".class.getName().hashCode();\n");
jaroslav@281
   595
                for (Prprt p : props) {
jaroslav@380
   596
                    w.write("    h = TYPE.hashPlus(prop_" + p.name() + ", h);\n");
jaroslav@281
   597
                }
jaroslav@281
   598
                w.write("    return h;\n");
jaroslav@15
   599
                w.write("  }\n");
jaroslav@14
   600
                w.write("}\n");
jtulach@0
   601
            } finally {
jtulach@0
   602
                w.close();
jtulach@0
   603
            }
jtulach@0
   604
        } catch (IOException ex) {
jtulach@0
   605
            error("Can't create " + className + ".java", e);
jtulach@0
   606
            return false;
jtulach@0
   607
        }
jtulach@0
   608
        return ok;
jtulach@0
   609
    }
jtulach@938
   610
jtulach@1017
   611
    private static String findBuilderPrefix(Element e, Model m) {
jtulach@1017
   612
        if (!m.builder().isEmpty()) {
jtulach@1017
   613
            return m.builder();
jtulach@1017
   614
        }
jtulach@1017
   615
        for (AnnotationMirror am : e.getAnnotationMirrors()) {
jtulach@1017
   616
            for (Map.Entry<? extends Object, ? extends Object> entry : am.getElementValues().entrySet()) {
jtulach@1017
   617
                if ("builder()".equals(entry.getKey().toString())) {
jtulach@1017
   618
                    return "";
jtulach@1017
   619
                }
jtulach@1017
   620
            }
jtulach@1017
   621
        }
jtulach@1017
   622
        return null;
jtulach@1017
   623
    }
jtulach@1017
   624
jtulach@1017
   625
    private static String builderMethod(String builderPrefix, Prprt p) {
jtulach@1017
   626
        if (builderPrefix.isEmpty()) {
jtulach@1017
   627
            return p.name();
jtulach@1017
   628
        }
jtulach@1017
   629
        return builderPrefix + Character.toUpperCase(p.name().charAt(0)) + p.name().substring(1);
jtulach@1017
   630
    }
jtulach@1017
   631
jtulach@0
   632
    private boolean generateProperties(
jtulach@1017
   633
        Element where, String builderPrefix,
jaroslav@383
   634
        Writer w, String className, Prprt[] properties,
jtulach@938
   635
        List<GetSet> props,
jaroslav@567
   636
        Map<String,Collection<String[]>> deps,
jtulach@0
   637
        Map<String,Collection<String>> functionDeps
jtulach@0
   638
    ) throws IOException {
jtulach@0
   639
        boolean ok = true;
jtulach@0
   640
        for (Prprt p : properties) {
jtulach@0
   641
            final String tn;
jtulach@949
   642
            tn = typeName(p);
jtulach@0
   643
            String[] gs = toGetSet(p.name(), tn, p.array());
jaroslav@17
   644
            String castTo;
jtulach@938
   645
jtulach@0
   646
            if (p.array()) {
jaroslav@383
   647
                w.write("  private final java.util.List<" + tn + "> prop_" + p.name() + ";\n");
jtulach@938
   648
jaroslav@17
   649
                castTo = "java.util.List";
jaroslav@66
   650
                w.write("  public java.util.List<" + tn + "> " + gs[0] + "() {\n");
jtulach@786
   651
                w.write("    proto.accessProperty(\"" + p.name() + "\");\n");
jaroslav@66
   652
                w.write("    return prop_" + p.name() + ";\n");
jaroslav@66
   653
                w.write("  }\n");
jtulach@1017
   654
                if (builderPrefix != null) {
jtulach@1017
   655
                    boolean[] isModel = {false};
jtulach@1017
   656
                    boolean[] isEnum = {false};
jtulach@1017
   657
                    boolean isPrimitive[] = {false};
jtulach@1017
   658
                    String ret = checkType(p, isModel, isEnum, isPrimitive);
jtulach@1017
   659
                    w.write("  public " + className + " " + builderMethod(builderPrefix, p) + "(" + ret + "... v) {\n");
jtulach@1017
   660
                    w.write("    proto.accessProperty(\"" + p.name() + "\");\n");
jtulach@1017
   661
                    w.append("   TYPE.replaceValue(prop_").append(p.name()).append(", " + tn + ".class, v);\n");
jtulach@1017
   662
                    w.write("    return this;\n");
jtulach@1017
   663
                    w.write("  }\n");
jtulach@1017
   664
                }
jtulach@0
   665
            } else {
jaroslav@17
   666
                castTo = tn;
jtulach@950
   667
                boolean isModel[] = { false };
jtulach@950
   668
                boolean isEnum[] = { false };
jtulach@950
   669
                boolean isPrimitive[] = { false };
jtulach@950
   670
                checkType(p, isModel, isEnum, isPrimitive);
jaroslav@66
   671
                w.write("  private " + tn + " prop_" + p.name() + ";\n");
jaroslav@66
   672
                w.write("  public " + tn + " " + gs[0] + "() {\n");
jtulach@786
   673
                w.write("    proto.accessProperty(\"" + p.name() + "\");\n");
jaroslav@66
   674
                w.write("    return prop_" + p.name() + ";\n");
jaroslav@66
   675
                w.write("  }\n");
jaroslav@66
   676
                w.write("  public void " + gs[1] + "(" + tn + " v) {\n");
jaroslav@402
   677
                w.write("    proto.verifyUnlocked();\n");
jaroslav@567
   678
                w.write("    Object o = prop_" + p.name() + ";\n");
jtulach@950
   679
                if (isModel[0]) {
jtulach@990
   680
                    w.write("    if (o == v) return;\n");
jtulach@950
   681
                    w.write("    prop_" + p.name() + " = v;\n");
jtulach@990
   682
                } else {
jtulach@975
   683
                    w.write("    if (TYPE.isSame(o , v)) return;\n");
jtulach@950
   684
                    w.write("    prop_" + p.name() + " = v;\n");
jtulach@950
   685
                }
jaroslav@567
   686
                w.write("    proto.valueHasMutated(\"" + p.name() + "\", o, v);\n");
jaroslav@567
   687
                {
jaroslav@567
   688
                    Collection<String[]> dependants = deps.get(p.name());
jaroslav@567
   689
                    if (dependants != null) {
jaroslav@567
   690
                        for (String[] pair : dependants) {
jtulach@938
   691
                            w.write("    proto.valueHasMutated(\"" + pair[0] + "\", null, " + pair[1] + "());\n");
jaroslav@567
   692
                        }
jaroslav@66
   693
                    }
jaroslav@66
   694
                }
jaroslav@567
   695
                {
jaroslav@567
   696
                    Collection<String> dependants = functionDeps.get(p.name());
jaroslav@567
   697
                    if (dependants != null) {
jtulach@1025
   698
                        w.append("    ");
jaroslav@567
   699
                        w.append(className).append(" model = ").append(className).append(".this;\n");
jaroslav@567
   700
                        for (String call : dependants) {
jaroslav@567
   701
                            w.append("  ").append(call);
jaroslav@567
   702
                        }
jtulach@0
   703
                    }
jtulach@0
   704
                }
jtulach@0
   705
                w.write("  }\n");
jtulach@1017
   706
                if (builderPrefix != null) {
jtulach@1017
   707
                    w.write("  public " + className + " " + builderMethod(builderPrefix, p) + "(" + tn + " v) {\n");
jtulach@1017
   708
                    w.write("    " + gs[1] + "(v);\n");
jtulach@1017
   709
                    w.write("    return this;\n");
jtulach@1017
   710
                    w.write("  }\n");
jtulach@1017
   711
                }
jtulach@0
   712
            }
jtulach@938
   713
jtulach@823
   714
            for (int i = 0; i < props.size(); i++) {
jtulach@823
   715
                if (props.get(i).name.equals(p.name())) {
jtulach@648
   716
                    error("Cannot have the name " + p.name() + " defined twice", where);
jtulach@648
   717
                    ok = false;
jtulach@648
   718
                }
jtulach@648
   719
            }
jtulach@938
   720
jtulach@823
   721
            props.add(new GetSet(
jtulach@823
   722
                p.name(),
jtulach@862
   723
                gs[0],
jtulach@862
   724
                gs[1],
jtulach@824
   725
                tn,
jtulach@824
   726
                gs[3] == null && !p.array()
jtulach@823
   727
            ));
jtulach@0
   728
        }
jtulach@0
   729
        return ok;
jtulach@0
   730
    }
jtulach@0
   731
jtulach@0
   732
    private boolean generateComputedProperties(
jtulach@951
   733
        String className,
jtulach@0
   734
        Writer w, Prprt[] fixedProps,
jtulach@823
   735
        Collection<? extends Element> arr, Collection<GetSet> props,
jaroslav@567
   736
        Map<String,Collection<String[]>> deps
jtulach@0
   737
    ) throws IOException {
jtulach@0
   738
        boolean ok = true;
jtulach@0
   739
        for (Element e : arr) {
jtulach@0
   740
            if (e.getKind() != ElementKind.METHOD) {
jtulach@0
   741
                continue;
jtulach@0
   742
            }
jtulach@776
   743
            final ComputedProperty cp = e.getAnnotation(ComputedProperty.class);
jtulach@787
   744
            final Transitive tp = e.getAnnotation(Transitive.class);
jtulach@776
   745
            if (cp == null) {
jtulach@0
   746
                continue;
jtulach@0
   747
            }
jaroslav@227
   748
            if (!e.getModifiers().contains(Modifier.STATIC)) {
jaroslav@227
   749
                error("Method " + e.getSimpleName() + " has to be static when annotated by @ComputedProperty", e);
jaroslav@227
   750
                ok = false;
jaroslav@227
   751
                continue;
jaroslav@227
   752
            }
jtulach@0
   753
            ExecutableElement ee = (ExecutableElement)e;
jtulach@951
   754
            ExecutableElement write = null;
jtulach@951
   755
            if (!cp.write().isEmpty()) {
jtulach@951
   756
                write = findWrite(ee, (TypeElement)e.getEnclosingElement(), cp.write(), className);
jtulach@951
   757
                ok = write != null;
jtulach@951
   758
            }
jtulach@0
   759
            final TypeMirror rt = ee.getReturnType();
jtulach@0
   760
            final Types tu = processingEnv.getTypeUtils();
jtulach@0
   761
            TypeMirror ert = tu.erasure(rt);
jtulach@0
   762
            String tn = fqn(ert, ee);
jtulach@0
   763
            boolean array = false;
jaroslav@235
   764
            final TypeMirror toCheck;
jtulach@0
   765
            if (tn.equals("java.util.List")) {
jtulach@0
   766
                array = true;
jaroslav@235
   767
                toCheck = ((DeclaredType)rt).getTypeArguments().get(0);
jaroslav@235
   768
            } else {
jaroslav@235
   769
                toCheck = rt;
jtulach@0
   770
            }
jtulach@938
   771
jtulach@0
   772
            final String sn = ee.getSimpleName().toString();
jtulach@938
   773
jaroslav@235
   774
            if (toCheck.getKind().isPrimitive()) {
jaroslav@235
   775
                // OK
jaroslav@235
   776
            } else {
jaroslav@235
   777
                TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
jaroslav@235
   778
                TypeMirror enumType = processingEnv.getElementUtils().getTypeElement("java.lang.Enum").asType();
jaroslav@235
   779
jaroslav@235
   780
                if (tu.isSubtype(toCheck, stringType)) {
jaroslav@235
   781
                    // OK
jaroslav@252
   782
                } else if (tu.isSubtype(tu.erasure(toCheck), tu.erasure(enumType))) {
jaroslav@235
   783
                    // OK
jaroslav@236
   784
                } else if (isModel(toCheck)) {
jaroslav@236
   785
                    // OK
jaroslav@235
   786
                } else {
jtulach@828
   787
                    try {
jtulach@828
   788
                        tu.unboxedType(toCheck);
jtulach@828
   789
                        // boxed types are OK
jtulach@828
   790
                    } catch (IllegalArgumentException ex) {
jtulach@828
   791
                        ok = false;
jtulach@828
   792
                        error(sn + " cannot return " + toCheck, e);
jtulach@828
   793
                    }
jaroslav@235
   794
                }
jaroslav@235
   795
            }
jtulach@938
   796
jtulach@0
   797
            String[] gs = toGetSet(sn, tn, array);
jtulach@938
   798
jaroslav@562
   799
            w.write("  public " + tn);
jaroslav@562
   800
            if (array) {
jaroslav@562
   801
                w.write("<" + toCheck + ">");
jaroslav@562
   802
            }
jaroslav@562
   803
            w.write(" " + gs[0] + "() {\n");
jtulach@0
   804
            int arg = 0;
jtulach@788
   805
            boolean deep = false;
jtulach@0
   806
            for (VariableElement pe : ee.getParameters()) {
jtulach@0
   807
                final String dn = pe.getSimpleName().toString();
jtulach@938
   808
jtulach@0
   809
                if (!verifyPropName(pe, dn, fixedProps)) {
jtulach@0
   810
                    ok = false;
jtulach@0
   811
                }
jtulach@788
   812
                final TypeMirror pt = pe.asType();
jtulach@788
   813
                if (isModel(pt)) {
jtulach@788
   814
                    deep = true;
jtulach@788
   815
                }
jtulach@788
   816
                final String dt = fqn(pt, ee);
jtulach@788
   817
                if (dt.startsWith("java.util.List") && pt instanceof DeclaredType) {
jtulach@788
   818
                    final List<? extends TypeMirror> ptArgs = ((DeclaredType)pt).getTypeArguments();
jtulach@788
   819
                    if (ptArgs.size() == 1 && isModel(ptArgs.get(0))) {
jtulach@788
   820
                        deep = true;
jtulach@788
   821
                    }
jtulach@788
   822
                }
jtulach@0
   823
                String[] call = toGetSet(dn, dt, false);
jaroslav@66
   824
                w.write("    " + dt + " arg" + (++arg) + " = ");
jtulach@0
   825
                w.write(call[0] + "();\n");
jtulach@938
   826
jaroslav@567
   827
                Collection<String[]> depends = deps.get(dn);
jtulach@0
   828
                if (depends == null) {
jaroslav@567
   829
                    depends = new LinkedHashSet<String[]>();
jtulach@0
   830
                    deps.put(dn, depends);
jtulach@0
   831
                }
jaroslav@567
   832
                depends.add(new String[] { sn, gs[0] });
jtulach@0
   833
            }
jaroslav@66
   834
            w.write("    try {\n");
jtulach@788
   835
            if (tp != null) {
jtulach@788
   836
                deep = tp.deep();
jtulach@788
   837
            }
jtulach@788
   838
            if (deep) {
jtulach@776
   839
                w.write("      proto.acquireLock(\"" + sn + "\");\n");
jtulach@776
   840
            } else {
jtulach@776
   841
                w.write("      proto.acquireLock();\n");
jtulach@776
   842
            }
jaroslav@66
   843
            w.write("      return " + fqn(ee.getEnclosingElement().asType(), ee) + '.' + e.getSimpleName() + "(");
jtulach@0
   844
            String sep = "";
jtulach@0
   845
            for (int i = 1; i <= arg; i++) {
jtulach@0
   846
                w.write(sep);
jtulach@0
   847
                w.write("arg" + i);
jtulach@0
   848
                sep = ", ";
jtulach@0
   849
            }
jtulach@0
   850
            w.write(");\n");
jaroslav@66
   851
            w.write("    } finally {\n");
jaroslav@373
   852
            w.write("      proto.releaseLock();\n");
jaroslav@66
   853
            w.write("    }\n");
jtulach@0
   854
            w.write("  }\n");
jtulach@0
   855
jtulach@951
   856
            if (write == null) {
jtulach@951
   857
                props.add(new GetSet(
jtulach@951
   858
                    e.getSimpleName().toString(),
jtulach@951
   859
                    gs[0],
jtulach@951
   860
                    null,
jtulach@951
   861
                    tn,
jtulach@951
   862
                    true
jtulach@951
   863
                ));
jtulach@951
   864
            } else {
jtulach@951
   865
                w.write("  public void " + gs[4] + "(" + write.getParameters().get(1).asType());
jtulach@951
   866
                w.write(" value) {\n");
jtulach@951
   867
                w.write("    " + fqn(ee.getEnclosingElement().asType(), ee) + '.' + write.getSimpleName() + "(this, value);\n");
jtulach@951
   868
                w.write("  }\n");
jtulach@951
   869
jtulach@951
   870
                props.add(new GetSet(
jtulach@951
   871
                    e.getSimpleName().toString(),
jtulach@951
   872
                    gs[0],
jtulach@951
   873
                    gs[4],
jtulach@951
   874
                    tn,
jtulach@951
   875
                    false
jtulach@951
   876
                ));
jtulach@951
   877
            }
jtulach@0
   878
        }
jtulach@938
   879
jtulach@0
   880
        return ok;
jtulach@0
   881
    }
jtulach@0
   882
jtulach@0
   883
    private static String[] toGetSet(String name, String type, boolean array) {
jtulach@0
   884
        String n = Character.toUpperCase(name.charAt(0)) + name.substring(1);
jtulach@862
   885
        boolean clazz = "class".equals(name);
jtulach@862
   886
        String pref = clazz ? "access" : "get";
jtulach@862
   887
        if ("boolean".equals(type) && !array) {
jtulach@862
   888
            pref = "is";
jtulach@0
   889
        }
jtulach@0
   890
        if (array) {
jtulach@938
   891
            return new String[] {
jtulach@862
   892
                pref + n,
jtulach@0
   893
                null,
jtulach@862
   894
                "a" + n,
jtulach@951
   895
                null,
jtulach@951
   896
                "set" + n
jtulach@0
   897
            };
jtulach@0
   898
        }
jtulach@0
   899
        return new String[]{
jtulach@938
   900
            pref + n,
jtulach@938
   901
            "set" + n,
jtulach@862
   902
            "a" + n,
jtulach@951
   903
            "",
jtulach@951
   904
            "set" + n
jtulach@0
   905
        };
jtulach@0
   906
    }
jtulach@0
   907
jtulach@949
   908
    private String typeName(Prprt p) {
jtulach@0
   909
        String ret;
jtulach@0
   910
        boolean[] isModel = { false };
jtulach@0
   911
        boolean[] isEnum = { false };
jtulach@0
   912
        boolean isPrimitive[] = { false };
jtulach@0
   913
        ret = checkType(p, isModel, isEnum, isPrimitive);
jtulach@0
   914
        if (p.array()) {
jtulach@0
   915
            String bt = findBoxedType(ret);
jtulach@0
   916
            if (bt != null) {
jtulach@0
   917
                return bt;
jtulach@0
   918
            }
jtulach@0
   919
        }
jtulach@0
   920
        return ret;
jtulach@0
   921
    }
jtulach@938
   922
jtulach@0
   923
    private static String findBoxedType(String ret) {
jtulach@0
   924
        if (ret.equals("boolean")) {
jtulach@0
   925
            return Boolean.class.getName();
jtulach@0
   926
        }
jtulach@0
   927
        if (ret.equals("byte")) {
jtulach@0
   928
            return Byte.class.getName();
jtulach@0
   929
        }
jtulach@0
   930
        if (ret.equals("short")) {
jtulach@0
   931
            return Short.class.getName();
jtulach@0
   932
        }
jtulach@0
   933
        if (ret.equals("char")) {
jtulach@0
   934
            return Character.class.getName();
jtulach@0
   935
        }
jtulach@0
   936
        if (ret.equals("int")) {
jtulach@0
   937
            return Integer.class.getName();
jtulach@0
   938
        }
jtulach@0
   939
        if (ret.equals("long")) {
jtulach@0
   940
            return Long.class.getName();
jtulach@0
   941
        }
jtulach@0
   942
        if (ret.equals("float")) {
jtulach@0
   943
            return Float.class.getName();
jtulach@0
   944
        }
jtulach@0
   945
        if (ret.equals("double")) {
jtulach@0
   946
            return Double.class.getName();
jtulach@0
   947
        }
jtulach@0
   948
        return null;
jtulach@0
   949
    }
jtulach@0
   950
jtulach@0
   951
    private boolean verifyPropName(Element e, String propName, Prprt[] existingProps) {
jtulach@0
   952
        StringBuilder sb = new StringBuilder();
jtulach@0
   953
        String sep = "";
jtulach@0
   954
        for (Prprt Prprt : existingProps) {
jtulach@0
   955
            if (Prprt.name().equals(propName)) {
jtulach@0
   956
                return true;
jtulach@0
   957
            }
jtulach@0
   958
            sb.append(sep);
jtulach@0
   959
            sb.append('"');
jtulach@0
   960
            sb.append(Prprt.name());
jtulach@0
   961
            sb.append('"');
jtulach@0
   962
            sep = ", ";
jtulach@0
   963
        }
jtulach@0
   964
        error(
jtulach@0
   965
            propName + " is not one of known properties: " + sb
jtulach@0
   966
            , e
jtulach@0
   967
        );
jtulach@0
   968
        return false;
jtulach@0
   969
    }
jtulach@0
   970
jtulach@0
   971
    private static String findPkgName(Element e) {
jtulach@0
   972
        for (;;) {
jtulach@0
   973
            if (e.getKind() == ElementKind.PACKAGE) {
jtulach@0
   974
                return ((PackageElement)e).getQualifiedName().toString();
jtulach@0
   975
            }
jtulach@0
   976
            e = e.getEnclosingElement();
jtulach@0
   977
        }
jtulach@0
   978
    }
jtulach@0
   979
jtulach@0
   980
    private boolean generateFunctions(
jtulach@938
   981
        Element clazz, StringWriter body, String className,
jaroslav@597
   982
        List<? extends Element> enclosedElements, List<Object> functions
jtulach@0
   983
    ) {
jtulach@1023
   984
        boolean instance = clazz.getAnnotation(Model.class).instance();
jtulach@0
   985
        for (Element m : enclosedElements) {
jtulach@0
   986
            if (m.getKind() != ElementKind.METHOD) {
jtulach@0
   987
                continue;
jtulach@0
   988
            }
jtulach@0
   989
            ExecutableElement e = (ExecutableElement)m;
jtulach@0
   990
            Function onF = e.getAnnotation(Function.class);
jtulach@0
   991
            if (onF == null) {
jtulach@0
   992
                continue;
jtulach@0
   993
            }
jtulach@1023
   994
            if (!instance && !e.getModifiers().contains(Modifier.STATIC)) {
jtulach@0
   995
                error("@OnFunction method needs to be static", e);
jtulach@0
   996
                return false;
jtulach@0
   997
            }
jtulach@0
   998
            if (e.getModifiers().contains(Modifier.PRIVATE)) {
jtulach@0
   999
                error("@OnFunction method cannot be private", e);
jtulach@0
  1000
                return false;
jtulach@0
  1001
            }
jtulach@0
  1002
            if (e.getReturnType().getKind() != TypeKind.VOID) {
jtulach@0
  1003
                error("@OnFunction method should return void", e);
jtulach@0
  1004
                return false;
jtulach@0
  1005
            }
jtulach@0
  1006
            String n = e.getSimpleName().toString();
jtulach@0
  1007
            functions.add(n);
jaroslav@597
  1008
            functions.add(e);
jtulach@0
  1009
        }
jtulach@0
  1010
        return true;
jtulach@0
  1011
    }
jtulach@0
  1012
jaroslav@567
  1013
    private boolean generateOnChange(Element clazz, Map<String,Collection<String[]>> propDeps,
jtulach@938
  1014
        Prprt[] properties, String className,
jtulach@0
  1015
        Map<String, Collection<String>> functionDeps
jtulach@0
  1016
    ) {
jtulach@1023
  1017
        boolean instance = clazz.getAnnotation(Model.class).instance();
jtulach@0
  1018
        for (Element m : clazz.getEnclosedElements()) {
jtulach@0
  1019
            if (m.getKind() != ElementKind.METHOD) {
jtulach@0
  1020
                continue;
jtulach@0
  1021
            }
jtulach@0
  1022
            ExecutableElement e = (ExecutableElement) m;
jtulach@0
  1023
            OnPropertyChange onPC = e.getAnnotation(OnPropertyChange.class);
jtulach@0
  1024
            if (onPC == null) {
jtulach@0
  1025
                continue;
jtulach@0
  1026
            }
jtulach@0
  1027
            for (String pn : onPC.value()) {
jtulach@0
  1028
                if (findPrprt(properties, pn) == null && findDerivedFrom(propDeps, pn).isEmpty()) {
jtulach@0
  1029
                    error("No Prprt named '" + pn + "' in the model", clazz);
jtulach@0
  1030
                    return false;
jtulach@0
  1031
                }
jtulach@0
  1032
            }
jtulach@1023
  1033
            if (!instance && !e.getModifiers().contains(Modifier.STATIC)) {
jtulach@0
  1034
                error("@OnPrprtChange method needs to be static", e);
jtulach@0
  1035
                return false;
jtulach@0
  1036
            }
jtulach@0
  1037
            if (e.getModifiers().contains(Modifier.PRIVATE)) {
jtulach@0
  1038
                error("@OnPrprtChange method cannot be private", e);
jtulach@0
  1039
                return false;
jtulach@0
  1040
            }
jtulach@0
  1041
            if (e.getReturnType().getKind() != TypeKind.VOID) {
jtulach@0
  1042
                error("@OnPrprtChange method should return void", e);
jtulach@0
  1043
                return false;
jtulach@0
  1044
            }
jtulach@0
  1045
            String n = e.getSimpleName().toString();
jtulach@938
  1046
jtulach@938
  1047
jtulach@0
  1048
            for (String pn : onPC.value()) {
jtulach@0
  1049
                StringBuilder call = new StringBuilder();
jtulach@1023
  1050
                call.append("  ").append(inPckName(clazz, instance)).append(".").append(n).append("(");
jtulach@0
  1051
                call.append(wrapPropName(e, className, "name", pn));
jtulach@0
  1052
                call.append(");\n");
jtulach@938
  1053
jtulach@0
  1054
                Collection<String> change = functionDeps.get(pn);
jtulach@0
  1055
                if (change == null) {
jaroslav@100
  1056
                    change = new ArrayList<String>();
jtulach@0
  1057
                    functionDeps.put(pn, change);
jtulach@0
  1058
                }
jtulach@0
  1059
                change.add(call.toString());
jtulach@0
  1060
                for (String dpn : findDerivedFrom(propDeps, pn)) {
jtulach@0
  1061
                    change = functionDeps.get(dpn);
jtulach@0
  1062
                    if (change == null) {
jaroslav@100
  1063
                        change = new ArrayList<String>();
jtulach@0
  1064
                        functionDeps.put(dpn, change);
jtulach@0
  1065
                    }
jtulach@0
  1066
                    change.add(call.toString());
jtulach@0
  1067
                }
jtulach@0
  1068
            }
jtulach@0
  1069
        }
jtulach@0
  1070
        return true;
jtulach@0
  1071
    }
jaroslav@240
  1072
jtulach@938
  1073
    private boolean generateOperation(Element clazz,
jtulach@938
  1074
        StringWriter body, String className,
jaroslav@598
  1075
        List<? extends Element> enclosedElements,
jaroslav@598
  1076
        List<Object> functions
jaroslav@240
  1077
    ) {
jtulach@1023
  1078
        boolean instance = clazz.getAnnotation(Model.class).instance();
jaroslav@240
  1079
        for (Element m : enclosedElements) {
jaroslav@240
  1080
            if (m.getKind() != ElementKind.METHOD) {
jaroslav@240
  1081
                continue;
jaroslav@240
  1082
            }
jaroslav@240
  1083
            ExecutableElement e = (ExecutableElement)m;
jaroslav@240
  1084
            ModelOperation mO = e.getAnnotation(ModelOperation.class);
jaroslav@240
  1085
            if (mO == null) {
jaroslav@240
  1086
                continue;
jaroslav@240
  1087
            }
jtulach@1023
  1088
            if (!instance && !e.getModifiers().contains(Modifier.STATIC)) {
jaroslav@240
  1089
                error("@ModelOperation method needs to be static", e);
jaroslav@240
  1090
                return false;
jaroslav@240
  1091
            }
jaroslav@240
  1092
            if (e.getModifiers().contains(Modifier.PRIVATE)) {
jaroslav@240
  1093
                error("@ModelOperation method cannot be private", e);
jaroslav@240
  1094
                return false;
jaroslav@240
  1095
            }
jaroslav@240
  1096
            if (e.getReturnType().getKind() != TypeKind.VOID) {
jaroslav@240
  1097
                error("@ModelOperation method should return void", e);
jaroslav@240
  1098
                return false;
jaroslav@240
  1099
            }
jaroslav@240
  1100
            List<String> args = new ArrayList<String>();
jaroslav@240
  1101
            {
jaroslav@240
  1102
                body.append("  public void ").append(m.getSimpleName()).append("(");
jaroslav@240
  1103
                String sep = "";
jaroslav@240
  1104
                boolean checkFirst = true;
jaroslav@240
  1105
                for (VariableElement ve : e.getParameters()) {
jaroslav@240
  1106
                    final TypeMirror type = ve.asType();
jaroslav@240
  1107
                    CharSequence simpleName;
jaroslav@240
  1108
                    if (type.getKind() == TypeKind.DECLARED) {
jaroslav@240
  1109
                        simpleName = ((DeclaredType)type).asElement().getSimpleName();
jaroslav@240
  1110
                    } else {
jaroslav@240
  1111
                        simpleName = type.toString();
jaroslav@240
  1112
                    }
jtulach@643
  1113
                    if (checkFirst && simpleName.toString().equals(className)) {
jaroslav@240
  1114
                        checkFirst = false;
jaroslav@240
  1115
                    } else {
jaroslav@240
  1116
                        if (checkFirst) {
jaroslav@240
  1117
                            error("First parameter of @ModelOperation method must be " + className, m);
jaroslav@240
  1118
                            return false;
jaroslav@240
  1119
                        }
jaroslav@240
  1120
                        args.add(ve.getSimpleName().toString());
jaroslav@240
  1121
                        body.append(sep).append("final ");
jaroslav@240
  1122
                        body.append(ve.asType().toString()).append(" ");
jaroslav@240
  1123
                        body.append(ve.toString());
jaroslav@240
  1124
                        sep = ", ";
jaroslav@240
  1125
                    }
jaroslav@240
  1126
                }
jaroslav@240
  1127
                body.append(") {\n");
jaroslav@598
  1128
                int idx = functions.size() / 2;
jaroslav@598
  1129
                functions.add(m.getSimpleName().toString());
jaroslav@598
  1130
                body.append("    proto.runInBrowser(" + idx);
jaroslav@240
  1131
                for (String s : args) {
jaroslav@240
  1132
                    body.append(", ").append(s);
jaroslav@240
  1133
                }
jaroslav@240
  1134
                body.append(");\n");
jaroslav@240
  1135
                body.append("  }\n");
jaroslav@598
  1136
jaroslav@598
  1137
                StringBuilder call = new StringBuilder();
jaroslav@598
  1138
                call.append("{ Object[] arr = (Object[])data; ");
jtulach@1023
  1139
                call.append(inPckName(clazz, true)).append(".").append(m.getSimpleName()).append("(");
jaroslav@598
  1140
                int i = 0;
jaroslav@598
  1141
                for (VariableElement ve : e.getParameters()) {
jaroslav@598
  1142
                    if (i++ == 0) {
jaroslav@598
  1143
                        call.append("model");
jaroslav@598
  1144
                        continue;
jaroslav@598
  1145
                    }
jaroslav@601
  1146
                    String type = ve.asType().toString();
jaroslav@601
  1147
                    String boxedType = findBoxedType(type);
jaroslav@601
  1148
                    if (boxedType != null) {
jaroslav@601
  1149
                        type = boxedType;
jaroslav@601
  1150
                    }
jaroslav@601
  1151
                    call.append(", ").append("(").append(type).append(")arr[").append(i - 2).append("]");
jaroslav@598
  1152
                }
jaroslav@598
  1153
                call.append("); }");
jaroslav@598
  1154
                functions.add(call.toString());
jaroslav@240
  1155
            }
jtulach@938
  1156
jaroslav@240
  1157
        }
jaroslav@240
  1158
        return true;
jaroslav@240
  1159
    }
jtulach@938
  1160
jtulach@938
  1161
jtulach@0
  1162
    private boolean generateReceive(
jtulach@938
  1163
        Element clazz, StringWriter body, String className,
jaroslav@386
  1164
        List<? extends Element> enclosedElements, StringBuilder inType
jtulach@0
  1165
    ) {
jtulach@947
  1166
        boolean ret = generateReceiveImpl(clazz, body, className, enclosedElements, inType);
jtulach@947
  1167
        if (!ret) {
jtulach@947
  1168
            inType.setLength(0);
jtulach@947
  1169
        }
jtulach@947
  1170
        return ret;
jtulach@947
  1171
    }
jtulach@947
  1172
    private boolean generateReceiveImpl(
jtulach@947
  1173
        Element clazz, StringWriter body, String className,
jtulach@947
  1174
        List<? extends Element> enclosedElements, StringBuilder inType
jtulach@947
  1175
    ) {
jtulach@650
  1176
        inType.append("  @Override public void onMessage(").append(className).append(" model, int index, int type, Object data, Object[] params) {\n");
jaroslav@386
  1177
        inType.append("    switch (index) {\n");
jaroslav@386
  1178
        int index = 0;
jtulach@947
  1179
        boolean ok = true;
jtulach@1023
  1180
        boolean instance = clazz.getAnnotation(Model.class).instance();
jtulach@0
  1181
        for (Element m : enclosedElements) {
jtulach@0
  1182
            if (m.getKind() != ElementKind.METHOD) {
jtulach@0
  1183
                continue;
jtulach@0
  1184
            }
jtulach@0
  1185
            ExecutableElement e = (ExecutableElement)m;
jtulach@0
  1186
            OnReceive onR = e.getAnnotation(OnReceive.class);
jtulach@0
  1187
            if (onR == null) {
jtulach@0
  1188
                continue;
jtulach@0
  1189
            }
jtulach@1023
  1190
            if (!instance && !e.getModifiers().contains(Modifier.STATIC)) {
jtulach@0
  1191
                error("@OnReceive method needs to be static", e);
jtulach@0
  1192
                return false;
jtulach@0
  1193
            }
jtulach@0
  1194
            if (e.getModifiers().contains(Modifier.PRIVATE)) {
jtulach@0
  1195
                error("@OnReceive method cannot be private", e);
jtulach@0
  1196
                return false;
jtulach@0
  1197
            }
jtulach@0
  1198
            if (e.getReturnType().getKind() != TypeKind.VOID) {
jtulach@0
  1199
                error("@OnReceive method should return void", e);
jtulach@0
  1200
                return false;
jtulach@0
  1201
            }
jaroslav@73
  1202
            if (!onR.jsonp().isEmpty() && !"GET".equals(onR.method())) {
jaroslav@73
  1203
                error("JSONP works only with GET transport method", e);
jaroslav@73
  1204
            }
jaroslav@73
  1205
            String dataMirror = findDataSpecified(e, onR);
jaroslav@72
  1206
            if ("PUT".equals(onR.method()) && dataMirror == null) {
jaroslav@70
  1207
                error("PUT method needs to specify a data() class", e);
jaroslav@70
  1208
                return false;
jaroslav@70
  1209
            }
jaroslav@72
  1210
            if ("POST".equals(onR.method()) && dataMirror == null) {
jaroslav@70
  1211
                error("POST method needs to specify a data() class", e);
jaroslav@70
  1212
                return false;
jaroslav@70
  1213
            }
jtulach@649
  1214
            if (e.getParameters().size() < 2) {
jtulach@649
  1215
                error("@OnReceive method needs at least two parameters", e);
jtulach@649
  1216
            }
jtulach@947
  1217
            final boolean isWebSocket = "WebSocket".equals(onR.method());
jtulach@948
  1218
            if (isWebSocket && dataMirror == null) {
jtulach@948
  1219
                error("WebSocket method needs to specify a data() class", e);
jtulach@948
  1220
            }
jtulach@647
  1221
            int expectsList = 0;
jaroslav@100
  1222
            List<String> args = new ArrayList<String>();
jtulach@650
  1223
            List<String> params = new ArrayList<String>();
jtulach@649
  1224
            // first argument is model class
jtulach@649
  1225
            {
jtulach@649
  1226
                TypeMirror type = e.getParameters().get(0).asType();
jtulach@649
  1227
                CharSequence simpleName;
jtulach@649
  1228
                if (type.getKind() == TypeKind.DECLARED) {
jtulach@649
  1229
                    simpleName = ((DeclaredType) type).asElement().getSimpleName();
jtulach@649
  1230
                } else {
jtulach@649
  1231
                    simpleName = type.toString();
jtulach@649
  1232
                }
jtulach@649
  1233
                if (simpleName.toString().equals(className)) {
jtulach@649
  1234
                    args.add("model");
jtulach@649
  1235
                } else {
jtulach@649
  1236
                    error("First parameter needs to be " + className, e);
jtulach@649
  1237
                    return false;
jtulach@938
  1238
                }
jtulach@649
  1239
            }
jtulach@938
  1240
jtulach@649
  1241
            String modelClass;
jtulach@0
  1242
            {
jtulach@647
  1243
                final Types tu = processingEnv.getTypeUtils();
jtulach@649
  1244
                TypeMirror type = e.getParameters().get(1).asType();
jtulach@649
  1245
                TypeMirror modelType = null;
jtulach@649
  1246
                TypeMirror ert = tu.erasure(type);
jtulach@649
  1247
jtulach@649
  1248
                if (isModel(type)) {
jtulach@649
  1249
                    modelType = type;
jtulach@649
  1250
                } else if (type.getKind() == TypeKind.ARRAY) {
jtulach@649
  1251
                    modelType = ((ArrayType)type).getComponentType();
jtulach@649
  1252
                    expectsList = 1;
jtulach@649
  1253
                } else if ("java.util.List".equals(fqn(ert, e))) {
jtulach@649
  1254
                    List<? extends TypeMirror> typeArgs = ((DeclaredType)type).getTypeArguments();
jtulach@649
  1255
                    if (typeArgs.size() == 1) {
jtulach@649
  1256
                        modelType = typeArgs.get(0);
jtulach@649
  1257
                        expectsList = 2;
jaroslav@29
  1258
                    }
jtulach@649
  1259
                } else if (type.toString().equals("java.lang.String")) {
jtulach@649
  1260
                    modelType = type;
jtulach@0
  1261
                }
jtulach@649
  1262
                if (modelType == null) {
jtulach@649
  1263
                    error("Second arguments needs to be a model, String or array or List of models", e);
jtulach@649
  1264
                    return false;
jtulach@649
  1265
                }
jtulach@649
  1266
                modelClass = modelType.toString();
jtulach@649
  1267
                if (expectsList == 1) {
jtulach@649
  1268
                    args.add("arr");
jtulach@649
  1269
                } else if (expectsList == 2) {
jtulach@649
  1270
                    args.add("java.util.Arrays.asList(arr)");
jtulach@649
  1271
                } else {
jtulach@649
  1272
                    args.add("arr[0]");
jtulach@649
  1273
                }
jtulach@0
  1274
            }
jtulach@0
  1275
            String n = e.getSimpleName().toString();
jtulach@650
  1276
            if (isWebSocket) {
jaroslav@250
  1277
                body.append("  /** Performs WebSocket communication. Call with <code>null</code> data parameter\n");
jaroslav@250
  1278
                body.append("  * to open the connection (even if not required). Call with non-null data to\n");
jaroslav@250
  1279
                body.append("  * send messages to server. Call again with <code>null</code> data to close the socket.\n");
jaroslav@250
  1280
                body.append("  */\n");
jtulach@940
  1281
                if (onR.headers().length > 0) {
jtulach@940
  1282
                    error("WebSocket spec does not support headers", e);
jtulach@940
  1283
                }
jaroslav@250
  1284
            }
jaroslav@66
  1285
            body.append("  public void ").append(n).append("(");
jaroslav@24
  1286
            StringBuilder urlBefore = new StringBuilder();
jaroslav@24
  1287
            StringBuilder urlAfter = new StringBuilder();
jtulach@940
  1288
            StringBuilder headers = new StringBuilder();
jtulach@0
  1289
            String jsonpVarName = null;
jtulach@0
  1290
            {
jtulach@0
  1291
                String sep = "";
jtulach@0
  1292
                boolean skipJSONP = onR.jsonp().isEmpty();
jtulach@940
  1293
                Set<String> receiveParams = new LinkedHashSet<String>();
jtulach@940
  1294
                findParamNames(receiveParams, e, onR.url(), onR.jsonp(), urlBefore, urlAfter);
jtulach@940
  1295
                for (String headerLine : onR.headers()) {
jtulach@940
  1296
                    if (headerLine.contains("\r") || headerLine.contains("\n")) {
jtulach@940
  1297
                        error("Header line cannot contain line separator", e);
jtulach@940
  1298
                    }
jtulach@940
  1299
                    findParamNames(receiveParams, e, headerLine, null, headers);
jtulach@940
  1300
                    headers.append("+ \"\\r\\n\" +\n");
jtulach@940
  1301
                }
jtulach@940
  1302
                if (headers.length() > 0) {
jtulach@940
  1303
                    headers.append("\"\"");
jtulach@940
  1304
                }
jtulach@940
  1305
                for (String p : receiveParams) {
jtulach@0
  1306
                    if (!skipJSONP && p.equals(onR.jsonp())) {
jtulach@0
  1307
                        skipJSONP = true;
jtulach@0
  1308
                        jsonpVarName = p;
jtulach@0
  1309
                        continue;
jtulach@0
  1310
                    }
jtulach@0
  1311
                    body.append(sep);
jtulach@0
  1312
                    body.append("String ").append(p);
jtulach@0
  1313
                    sep = ", ";
jtulach@0
  1314
                }
jtulach@0
  1315
                if (!skipJSONP) {
jtulach@0
  1316
                    error(
jtulach@938
  1317
                        "Name of jsonp attribute ('" + onR.jsonp() +
jtulach@0
  1318
                        "') is not used in url attribute '" + onR.url() + "'", e
jtulach@0
  1319
                    );
jtulach@0
  1320
                }
jaroslav@73
  1321
                if (dataMirror != null) {
jaroslav@73
  1322
                    body.append(sep).append(dataMirror.toString()).append(" data");
jaroslav@73
  1323
                }
jtulach@650
  1324
                for (int i = 2; i < e.getParameters().size(); i++) {
jtulach@650
  1325
                    if (isWebSocket) {
jtulach@650
  1326
                        error("@OnReceive(method=\"WebSocket\") can only have two arguments", e);
jtulach@948
  1327
                        ok = false;
jtulach@650
  1328
                    }
jtulach@938
  1329
jtulach@650
  1330
                    VariableElement ve = e.getParameters().get(i);
jtulach@650
  1331
                    body.append(sep).append(ve.asType().toString()).append(" ").append(ve.getSimpleName());
jtulach@650
  1332
                    final String tp = ve.asType().toString();
jtulach@650
  1333
                    String btn = findBoxedType(tp);
jtulach@650
  1334
                    if (btn == null) {
jtulach@650
  1335
                        btn = tp;
jtulach@650
  1336
                    }
jtulach@650
  1337
                    args.add("(" + btn + ")params[" + (i - 2) + "]");
jtulach@650
  1338
                    params.add(ve.getSimpleName().toString());
jtulach@650
  1339
                    sep = ", ";
jtulach@650
  1340
                }
jtulach@0
  1341
            }
jtulach@0
  1342
            body.append(") {\n");
jaroslav@258
  1343
            boolean webSocket = onR.method().equals("WebSocket");
jaroslav@258
  1344
            if (webSocket) {
jtulach@940
  1345
                if (generateWSReceiveBody(index++, body, inType, onR, e, clazz, className, expectsList != 0, modelClass, n, args, params, urlBefore, jsonpVarName, urlAfter, dataMirror, headers)) {
jaroslav@387
  1346
                    return false;
jaroslav@387
  1347
                }
jaroslav@258
  1348
                body.append("  }\n");
jaroslav@387
  1349
                body.append("  private Object ws_" + e.getSimpleName() + ";\n");
jaroslav@258
  1350
            } else {
jtulach@940
  1351
                if (generateJSONReceiveBody(index++, body, inType, onR, e, clazz, className, expectsList != 0, modelClass, n, args, params, urlBefore, jsonpVarName, urlAfter, dataMirror, headers)) {
jtulach@947
  1352
                    ok = false;
jaroslav@258
  1353
                }
jaroslav@258
  1354
                body.append("  }\n");
jaroslav@258
  1355
            }
jaroslav@258
  1356
        }
jaroslav@386
  1357
        inType.append("    }\n");
jaroslav@386
  1358
        inType.append("    throw new UnsupportedOperationException(\"index: \" + index + \" type: \" + type);\n");
jaroslav@386
  1359
        inType.append("  }\n");
jtulach@947
  1360
        return ok;
jaroslav@258
  1361
    }
jaroslav@258
  1362
jtulach@940
  1363
    private boolean generateJSONReceiveBody(int index, StringWriter method, StringBuilder body, OnReceive onR, ExecutableElement e, Element clazz, String className, boolean expectsList, String modelClass, String n, List<String> args, List<String> params, StringBuilder urlBefore, String jsonpVarName, StringBuilder urlAfter, String dataMirror, StringBuilder headers) {
jtulach@947
  1364
        boolean error = false;
jaroslav@258
  1365
        body.append(
jaroslav@386
  1366
            "    case " + index + ": {\n" +
jaroslav@386
  1367
            "      if (type == 2) { /* on error */\n" +
jaroslav@386
  1368
            "        Exception ex = (Exception)data;\n"
jaroslav@258
  1369
            );
jaroslav@258
  1370
        if (onR.onError().isEmpty()) {
jtulach@0
  1371
            body.append(
jaroslav@386
  1372
                "        ex.printStackTrace();\n"
jaroslav@245
  1373
                );
jaroslav@258
  1374
        } else {
jtulach@947
  1375
            error = !findOnError(e, ((TypeElement)clazz), onR.onError(), className);
jaroslav@258
  1376
            body.append("        ").append(clazz.getSimpleName()).append(".").append(onR.onError()).append("(");
jaroslav@386
  1377
            body.append("model, ex);\n");
jaroslav@258
  1378
        }
jaroslav@258
  1379
        body.append(
jaroslav@386
  1380
            "        return;\n" +
jaroslav@386
  1381
            "      } else if (type == 1) {\n" +
jaroslav@386
  1382
            "        Object[] ev = (Object[])data;\n"
jaroslav@258
  1383
            );
jaroslav@258
  1384
        if (expectsList) {
jaroslav@245
  1385
            body.append(
jaroslav@386
  1386
                "        " + modelClass + "[] arr = new " + modelClass + "[ev.length];\n"
jaroslav@258
  1387
                );
jaroslav@258
  1388
        } else {
jaroslav@258
  1389
            body.append(
jaroslav@258
  1390
                "        " + modelClass + "[] arr = { null };\n"
jaroslav@258
  1391
                );
jaroslav@258
  1392
        }
jaroslav@258
  1393
        body.append(
jaroslav@386
  1394
            "        TYPE.copyJSON(model.proto.getContext(), ev, " + modelClass + ".class, arr);\n"
jaroslav@386
  1395
        );
jaroslav@258
  1396
        {
jaroslav@258
  1397
            body.append("        ").append(clazz.getSimpleName()).append(".").append(n).append("(");
jaroslav@258
  1398
            String sep = "";
jaroslav@258
  1399
            for (String arg : args) {
jaroslav@258
  1400
                body.append(sep);
jaroslav@258
  1401
                body.append(arg);
jaroslav@258
  1402
                sep = ", ";
jaroslav@74
  1403
            }
jaroslav@24
  1404
            body.append(");\n");
jtulach@0
  1405
        }
jaroslav@258
  1406
        body.append(
jaroslav@386
  1407
            "        return;\n" +
jaroslav@258
  1408
            "      }\n" +
jaroslav@258
  1409
            "    }\n"
jaroslav@258
  1410
            );
jtulach@940
  1411
        method.append("    proto.loadJSONWithHeaders(" + index + ",\n        ");
jtulach@940
  1412
        method.append(headers.length() == 0 ? "null" : headers).append(",\n        ");
jaroslav@386
  1413
        method.append(urlBefore).append(", ");
jaroslav@258
  1414
        if (jsonpVarName != null) {
jaroslav@386
  1415
            method.append(urlAfter);
jaroslav@258
  1416
        } else {
jaroslav@386
  1417
            method.append("null");
jaroslav@258
  1418
        }
jaroslav@258
  1419
        if (!"GET".equals(onR.method()) || dataMirror != null) {
jaroslav@386
  1420
            method.append(", \"").append(onR.method()).append('"');
jaroslav@258
  1421
            if (dataMirror != null) {
jaroslav@386
  1422
                method.append(", data");
jaroslav@258
  1423
            } else {
jaroslav@386
  1424
                method.append(", null");
jaroslav@258
  1425
            }
jaroslav@258
  1426
        } else {
jaroslav@386
  1427
            method.append(", null, null");
jaroslav@258
  1428
        }
jtulach@650
  1429
        for (String a : params) {
jtulach@650
  1430
            method.append(", ").append(a);
jtulach@650
  1431
        }
jaroslav@386
  1432
        method.append(");\n");
jtulach@947
  1433
        return error;
jaroslav@258
  1434
    }
jtulach@938
  1435
jtulach@940
  1436
    private boolean generateWSReceiveBody(int index, StringWriter method, StringBuilder body, OnReceive onR, ExecutableElement e, Element clazz, String className, boolean expectsList, String modelClass, String n, List<String> args, List<String> params, StringBuilder urlBefore, String jsonpVarName, StringBuilder urlAfter, String dataMirror, StringBuilder headers) {
jaroslav@258
  1437
        body.append(
jaroslav@387
  1438
            "    case " + index + ": {\n" +
jaroslav@387
  1439
            "      if (type == 0) { /* on open */\n" +
jtulach@1023
  1440
            "        ").append(inPckName(clazz, true)).append(".").append(n).append("(");
jaroslav@258
  1441
        {
jaroslav@258
  1442
            String sep = "";
jaroslav@258
  1443
            for (String arg : args) {
jaroslav@258
  1444
                body.append(sep);
jtulach@650
  1445
                if (arg.startsWith("arr") || arg.startsWith("java.util.Array")) {
jaroslav@258
  1446
                    body.append("null");
jaroslav@258
  1447
                } else {
jaroslav@258
  1448
                    body.append(arg);
jaroslav@258
  1449
                }
jaroslav@258
  1450
                sep = ", ";
jaroslav@258
  1451
            }
jaroslav@258
  1452
        }
jaroslav@258
  1453
        body.append(");\n");
jaroslav@258
  1454
        body.append(
jaroslav@387
  1455
            "        return;\n" +
jaroslav@387
  1456
            "      } else if (type == 2) { /* on error */\n" +
jaroslav@387
  1457
            "        Exception value = (Exception)data;\n"
jaroslav@258
  1458
            );
jaroslav@258
  1459
        if (onR.onError().isEmpty()) {
jaroslav@258
  1460
            body.append(
jaroslav@258
  1461
                "        value.printStackTrace();\n"
jaroslav@258
  1462
                );
jaroslav@258
  1463
        } else {
jaroslav@258
  1464
            if (!findOnError(e, ((TypeElement)clazz), onR.onError(), className)) {
jaroslav@258
  1465
                return true;
jaroslav@258
  1466
            }
jtulach@1023
  1467
            body.append("        ").append(inPckName(clazz, true)).append(".").append(onR.onError()).append("(");
jaroslav@387
  1468
            body.append("model, value);\n");
jaroslav@258
  1469
        }
jaroslav@258
  1470
        body.append(
jaroslav@387
  1471
            "        return;\n" +
jaroslav@387
  1472
            "      } else if (type == 1) {\n" +
jaroslav@387
  1473
            "        Object[] ev = (Object[])data;\n"
jaroslav@258
  1474
        );
jaroslav@258
  1475
        if (expectsList) {
jaroslav@258
  1476
            body.append(
jaroslav@387
  1477
                "        " + modelClass + "[] arr = new " + modelClass + "[ev.length];\n"
jaroslav@258
  1478
                );
jaroslav@258
  1479
        } else {
jaroslav@258
  1480
            body.append(
jaroslav@258
  1481
                "        " + modelClass + "[] arr = { null };\n"
jaroslav@258
  1482
                );
jaroslav@258
  1483
        }
jaroslav@258
  1484
        body.append(
jaroslav@387
  1485
            "        TYPE.copyJSON(model.proto.getContext(), ev, " + modelClass + ".class, arr);\n"
jaroslav@387
  1486
        );
jaroslav@258
  1487
        {
jtulach@1023
  1488
            body.append("        ").append(inPckName(clazz, true)).append(".").append(n).append("(");
jaroslav@258
  1489
            String sep = "";
jaroslav@258
  1490
            for (String arg : args) {
jaroslav@258
  1491
                body.append(sep);
jaroslav@258
  1492
                body.append(arg);
jaroslav@258
  1493
                sep = ", ";
jaroslav@258
  1494
            }
jaroslav@258
  1495
            body.append(");\n");
jaroslav@258
  1496
        }
jaroslav@258
  1497
        body.append(
jaroslav@387
  1498
            "        return;\n" +
jaroslav@387
  1499
            "      }"
jaroslav@258
  1500
        );
jaroslav@258
  1501
        if (!onR.onError().isEmpty()) {
jaroslav@387
  1502
            body.append(" else if (type == 3) { /* on close */\n");
jtulach@1023
  1503
            body.append("        ").append(inPckName(clazz, true)).append(".").append(onR.onError()).append("(");
jaroslav@387
  1504
            body.append("model, null);\n");
jaroslav@258
  1505
            body.append(
jaroslav@397
  1506
                "        return;" +
jaroslav@387
  1507
                "      }"
jaroslav@258
  1508
            );
jaroslav@258
  1509
        }
jaroslav@387
  1510
        body.append("\n");
jaroslav@258
  1511
        body.append("    }\n");
jaroslav@387
  1512
        method.append("    if (this.ws_").append(e.getSimpleName()).append(" == null) {\n");
jaroslav@387
  1513
        method.append("      this.ws_").append(e.getSimpleName());
jaroslav@387
  1514
        method.append("= proto.wsOpen(" + index + ", ");
jaroslav@387
  1515
        method.append(urlBefore).append(", data);\n");
jaroslav@387
  1516
        method.append("    } else {\n");
jtulach@650
  1517
        method.append("      proto.wsSend(this.ws_").append(e.getSimpleName()).append(", ").append(urlBefore).append(", data");
jtulach@650
  1518
        for (String a : params) {
jtulach@650
  1519
            method.append(", ").append(a);
jtulach@650
  1520
        }
jtulach@650
  1521
        method.append(");\n");
jaroslav@387
  1522
        method.append("    }\n");
jaroslav@258
  1523
        return false;
jtulach@0
  1524
    }
jtulach@0
  1525
jtulach@0
  1526
    private CharSequence wrapParams(
jaroslav@597
  1527
        ExecutableElement ee, String id, String className, String classRef, String evName, String dataName
jtulach@0
  1528
    ) {
jtulach@0
  1529
        TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
jtulach@0
  1530
        StringBuilder params = new StringBuilder();
jtulach@0
  1531
        boolean first = true;
jtulach@0
  1532
        for (VariableElement ve : ee.getParameters()) {
jtulach@0
  1533
            if (!first) {
jtulach@0
  1534
                params.append(", ");
jtulach@0
  1535
            }
jtulach@0
  1536
            first = false;
jtulach@0
  1537
            String toCall = null;
jaroslav@22
  1538
            String toFinish = null;
jaroslav@408
  1539
            boolean addNull = true;
jtulach@0
  1540
            if (ve.asType() == stringType) {
jtulach@0
  1541
                if (ve.getSimpleName().contentEquals("id")) {
jtulach@0
  1542
                    params.append('"').append(id).append('"');
jtulach@0
  1543
                    continue;
jtulach@0
  1544
                }
jaroslav@597
  1545
                toCall = classRef + ".proto.toString(";
jtulach@0
  1546
            }
jtulach@0
  1547
            if (ve.asType().getKind() == TypeKind.DOUBLE) {
jaroslav@597
  1548
                toCall = classRef + ".proto.toNumber(";
jaroslav@22
  1549
                toFinish = ".doubleValue()";
jtulach@0
  1550
            }
jtulach@930
  1551
            if (ve.asType().getKind() == TypeKind.FLOAT) {
jtulach@930
  1552
                toCall = classRef + ".proto.toNumber(";
jtulach@930
  1553
                toFinish = ".floatValue()";
jtulach@930
  1554
            }
jtulach@0
  1555
            if (ve.asType().getKind() == TypeKind.INT) {
jaroslav@597
  1556
                toCall = classRef + ".proto.toNumber(";
jaroslav@22
  1557
                toFinish = ".intValue()";
jtulach@0
  1558
            }
jtulach@930
  1559
            if (ve.asType().getKind() == TypeKind.BYTE) {
jtulach@930
  1560
                toCall = classRef + ".proto.toNumber(";
jtulach@930
  1561
                toFinish = ".byteValue()";
jtulach@930
  1562
            }
jtulach@930
  1563
            if (ve.asType().getKind() == TypeKind.SHORT) {
jtulach@930
  1564
                toCall = classRef + ".proto.toNumber(";
jtulach@930
  1565
                toFinish = ".shortValue()";
jtulach@930
  1566
            }
jtulach@930
  1567
            if (ve.asType().getKind() == TypeKind.LONG) {
jtulach@930
  1568
                toCall = classRef + ".proto.toNumber(";
jtulach@930
  1569
                toFinish = ".longValue()";
jtulach@930
  1570
            }
jtulach@930
  1571
            if (ve.asType().getKind() == TypeKind.BOOLEAN) {
jtulach@930
  1572
                toCall = "\"true\".equals(" + classRef + ".proto.toString(";
jtulach@930
  1573
                toFinish = ")";
jtulach@930
  1574
            }
jtulach@930
  1575
            if (ve.asType().getKind() == TypeKind.CHAR) {
jtulach@930
  1576
                toCall = "(char)" + classRef + ".proto.toNumber(";
jtulach@930
  1577
                toFinish = ".intValue()";
jtulach@930
  1578
            }
jtulach@0
  1579
            if (dataName != null && ve.getSimpleName().contentEquals(dataName) && isModel(ve.asType())) {
jaroslav@597
  1580
                toCall = classRef + ".proto.toModel(" + ve.asType() + ".class, ";
jaroslav@408
  1581
                addNull = false;
jtulach@0
  1582
            }
jtulach@0
  1583
jtulach@0
  1584
            if (toCall != null) {
jtulach@0
  1585
                params.append(toCall);
jtulach@0
  1586
                if (dataName != null && ve.getSimpleName().contentEquals(dataName)) {
jtulach@0
  1587
                    params.append(dataName);
jaroslav@408
  1588
                    if (addNull) {
jaroslav@408
  1589
                        params.append(", null");
jaroslav@408
  1590
                    }
jtulach@0
  1591
                } else {
jtulach@0
  1592
                    if (evName == null) {
jtulach@0
  1593
                        final StringBuilder sb = new StringBuilder();
jtulach@0
  1594
                        sb.append("Unexpected string parameter name.");
jtulach@0
  1595
                        if (dataName != null) {
jtulach@0
  1596
                            sb.append(" Try \"").append(dataName).append("\"");
jtulach@0
  1597
                        }
jtulach@0
  1598
                        error(sb.toString(), ee);
jtulach@0
  1599
                    }
jtulach@0
  1600
                    params.append(evName);
jtulach@0
  1601
                    params.append(", \"");
jtulach@0
  1602
                    params.append(ve.getSimpleName().toString());
jtulach@0
  1603
                    params.append("\"");
jtulach@0
  1604
                }
jtulach@0
  1605
                params.append(")");
jaroslav@22
  1606
                if (toFinish != null) {
jaroslav@22
  1607
                    params.append(toFinish);
jaroslav@22
  1608
                }
jtulach@0
  1609
                continue;
jtulach@0
  1610
            }
jtulach@0
  1611
            String rn = fqn(ve.asType(), ee);
jtulach@0
  1612
            int last = rn.lastIndexOf('.');
jtulach@0
  1613
            if (last >= 0) {
jtulach@0
  1614
                rn = rn.substring(last + 1);
jtulach@0
  1615
            }
jtulach@0
  1616
            if (rn.equals(className)) {
jaroslav@597
  1617
                params.append(classRef);
jtulach@0
  1618
                continue;
jtulach@0
  1619
            }
jtulach@930
  1620
            StringBuilder err = new StringBuilder();
jtulach@930
  1621
            err.append("Argument ").
jtulach@930
  1622
                append(ve.getSimpleName()).
jtulach@930
  1623
                append(" is not valid. The annotated method can only accept ").
jtulach@930
  1624
                append(className).
jtulach@930
  1625
                append(" argument");
jtulach@930
  1626
            if (dataName != null) {
jtulach@930
  1627
                err.append(" or argument named '").append(dataName).append("'");
jtulach@930
  1628
            }
jtulach@930
  1629
            err.append(".");
jtulach@930
  1630
            error(err.toString(), ee);
jtulach@0
  1631
        }
jtulach@0
  1632
        return params;
jtulach@0
  1633
    }
jtulach@938
  1634
jtulach@938
  1635
jtulach@0
  1636
    private CharSequence wrapPropName(
jtulach@0
  1637
        ExecutableElement ee, String className, String propName, String propValue
jtulach@0
  1638
    ) {
jtulach@0
  1639
        TypeMirror stringType = processingEnv.getElementUtils().getTypeElement("java.lang.String").asType();
jtulach@0
  1640
        StringBuilder params = new StringBuilder();
jtulach@0
  1641
        boolean first = true;
jtulach@0
  1642
        for (VariableElement ve : ee.getParameters()) {
jtulach@0
  1643
            if (!first) {
jtulach@0
  1644
                params.append(", ");
jtulach@0
  1645
            }
jtulach@0
  1646
            first = false;
jtulach@0
  1647
            if (ve.asType() == stringType) {
jtulach@0
  1648
                if (propName != null && ve.getSimpleName().contentEquals(propName)) {
jtulach@0
  1649
                    params.append('"').append(propValue).append('"');
jtulach@0
  1650
                } else {
jtulach@0
  1651
                    error("Unexpected string parameter name. Try \"" + propName + "\".", ee);
jtulach@0
  1652
                }
jtulach@0
  1653
                continue;
jtulach@0
  1654
            }
jtulach@0
  1655
            String rn = fqn(ve.asType(), ee);
jtulach@0
  1656
            int last = rn.lastIndexOf('.');
jtulach@0
  1657
            if (last >= 0) {
jtulach@0
  1658
                rn = rn.substring(last + 1);
jtulach@0
  1659
            }
jtulach@0
  1660
            if (rn.equals(className)) {
jaroslav@383
  1661
                params.append("model");
jtulach@0
  1662
                continue;
jtulach@0
  1663
            }
jtulach@0
  1664
            error(
jtulach@0
  1665
                "@OnPrprtChange method can only accept String or " + className + " arguments",
jtulach@0
  1666
                ee);
jtulach@0
  1667
        }
jtulach@0
  1668
        return params;
jtulach@0
  1669
    }
jtulach@938
  1670
jtulach@0
  1671
    private boolean isModel(TypeMirror tm) {
jaroslav@13
  1672
        if (tm.getKind() == TypeKind.ERROR) {
jaroslav@13
  1673
            return true;
jaroslav@13
  1674
        }
jtulach@0
  1675
        final Element e = processingEnv.getTypeUtils().asElement(tm);
jtulach@0
  1676
        if (e == null) {
jtulach@0
  1677
            return false;
jtulach@0
  1678
        }
jtulach@0
  1679
        for (Element ch : e.getEnclosedElements()) {
jtulach@0
  1680
            if (ch.getKind() == ElementKind.METHOD) {
jtulach@0
  1681
                ExecutableElement ee = (ExecutableElement)ch;
jtulach@0
  1682
                if (ee.getParameters().isEmpty() && ee.getSimpleName().contentEquals("modelFor")) {
jtulach@0
  1683
                    return true;
jtulach@0
  1684
                }
jtulach@0
  1685
            }
jtulach@0
  1686
        }
jtulach@0
  1687
        return models.values().contains(e.getSimpleName().toString());
jtulach@0
  1688
    }
jtulach@938
  1689
jtulach@0
  1690
    private void writeToString(Prprt[] props, Writer w) throws IOException {
jtulach@0
  1691
        w.write("  public String toString() {\n");
jtulach@0
  1692
        w.write("    StringBuilder sb = new StringBuilder();\n");
jtulach@0
  1693
        w.write("    sb.append('{');\n");
jtulach@0
  1694
        String sep = "";
jtulach@0
  1695
        for (Prprt p : props) {
jtulach@0
  1696
            w.write(sep);
jaroslav@18
  1697
            w.append("    sb.append('\"').append(\"" + p.name() + "\")");
jaroslav@18
  1698
                w.append(".append('\"').append(\":\");\n");
jaroslav@380
  1699
            w.append("    sb.append(TYPE.toJSON(prop_");
jtulach@0
  1700
            w.append(p.name()).append("));\n");
jtulach@0
  1701
            sep =    "    sb.append(',');\n";
jtulach@0
  1702
        }
jtulach@0
  1703
        w.write("    sb.append('}');\n");
jtulach@0
  1704
        w.write("    return sb.toString();\n");
jtulach@0
  1705
        w.write("  }\n");
jtulach@0
  1706
    }
jtulach@0
  1707
    private void writeClone(String className, Prprt[] props, Writer w) throws IOException {
jtulach@0
  1708
        w.write("  public " + className + " clone() {\n");
jaroslav@373
  1709
        w.write("    return clone(proto.getContext());\n");
jaroslav@111
  1710
        w.write("  }\n");
jaroslav@111
  1711
        w.write("  private " + className + " clone(net.java.html.BrwsrCtx ctx) {\n");
jaroslav@111
  1712
        w.write("    " + className + " ret = new " + className + "(ctx);\n");
jtulach@0
  1713
        for (Prprt p : props) {
jtulach@949
  1714
            String tn = typeName(p);
jtulach@949
  1715
            String[] gs = toGetSet(p.name(), tn, p.array());
jtulach@0
  1716
            if (!p.array()) {
jtulach@0
  1717
                boolean isModel[] = { false };
jtulach@0
  1718
                boolean isEnum[] = { false };
jtulach@0
  1719
                boolean isPrimitive[] = { false };
jtulach@0
  1720
                checkType(p, isModel, isEnum, isPrimitive);
jtulach@0
  1721
                if (!isModel[0]) {
jtulach@949
  1722
                    w.write("    ret.prop_" + p.name() + " = " + gs[0] + "();\n");
jtulach@0
  1723
                    continue;
jtulach@0
  1724
                }
jtulach@949
  1725
                w.write("    ret.prop_" + p.name() + " =  " + gs[0] + "()  == null ? null : prop_" + p.name() + ".clone();\n");
jtulach@0
  1726
            } else {
jtulach@949
  1727
                w.write("    proto.cloneList(ret." + gs[0] + "(), ctx, prop_" + p.name() + ");\n");
jtulach@0
  1728
            }
jtulach@0
  1729
        }
jtulach@938
  1730
jtulach@0
  1731
        w.write("    return ret;\n");
jtulach@0
  1732
        w.write("  }\n");
jtulach@0
  1733
    }
jtulach@0
  1734
jtulach@1023
  1735
    private String inPckName(Element e, boolean preferInstance) {
jtulach@1023
  1736
        if (preferInstance && e.getAnnotation(Model.class).instance()) {
jtulach@1023
  1737
            return "model.instance";
jtulach@1023
  1738
        }
jtulach@0
  1739
        StringBuilder sb = new StringBuilder();
jtulach@0
  1740
        while (e.getKind() != ElementKind.PACKAGE) {
jtulach@0
  1741
            if (sb.length() == 0) {
jtulach@0
  1742
                sb.append(e.getSimpleName());
jtulach@0
  1743
            } else {
jtulach@0
  1744
                sb.insert(0, '.');
jtulach@0
  1745
                sb.insert(0, e.getSimpleName());
jtulach@0
  1746
            }
jtulach@0
  1747
            e = e.getEnclosingElement();
jtulach@0
  1748
        }
jtulach@0
  1749
        return sb.toString();
jtulach@0
  1750
    }
jtulach@0
  1751
jtulach@0
  1752
    private String fqn(TypeMirror pt, Element relative) {
jtulach@0
  1753
        if (pt.getKind() == TypeKind.ERROR) {
jtulach@0
  1754
            final Elements eu = processingEnv.getElementUtils();
jtulach@0
  1755
            PackageElement pckg = eu.getPackageOf(relative);
jtulach@0
  1756
            return pckg.getQualifiedName() + "." + pt.toString();
jtulach@0
  1757
        }
jtulach@0
  1758
        return pt.toString();
jtulach@0
  1759
    }
jtulach@0
  1760
jtulach@0
  1761
    private String checkType(Prprt p, boolean[] isModel, boolean[] isEnum, boolean[] isPrimitive) {
jtulach@0
  1762
        TypeMirror tm;
jtulach@0
  1763
        try {
jtulach@0
  1764
            String ret = p.typeName(processingEnv);
jtulach@0
  1765
            TypeElement e = processingEnv.getElementUtils().getTypeElement(ret);
jtulach@0
  1766
            if (e == null) {
jtulach@0
  1767
                isModel[0] = true;
jtulach@0
  1768
                isEnum[0] = false;
jtulach@0
  1769
                isPrimitive[0] = false;
jtulach@0
  1770
                return ret;
jtulach@0
  1771
            }
jtulach@0
  1772
            tm = e.asType();
jtulach@0
  1773
        } catch (MirroredTypeException ex) {
jtulach@0
  1774
            tm = ex.getTypeMirror();
jtulach@0
  1775
        }
jtulach@0
  1776
        tm = processingEnv.getTypeUtils().erasure(tm);
jaroslav@241
  1777
        if (isPrimitive[0] = tm.getKind().isPrimitive()) {
jaroslav@241
  1778
            isEnum[0] = false;
jaroslav@241
  1779
            isModel[0] = false;
jaroslav@241
  1780
            return tm.toString();
jaroslav@241
  1781
        }
jtulach@0
  1782
        final Element e = processingEnv.getTypeUtils().asElement(tm);
jaroslav@241
  1783
        if (e.getKind() == ElementKind.CLASS && tm.getKind() == TypeKind.ERROR) {
jaroslav@241
  1784
            isModel[0] = true;
jaroslav@241
  1785
            isEnum[0] = false;
jaroslav@242
  1786
            return e.getSimpleName().toString();
jaroslav@241
  1787
        }
jtulach@938
  1788
jtulach@0
  1789
        final Model m = e == null ? null : e.getAnnotation(Model.class);
jtulach@0
  1790
        String ret;
jtulach@0
  1791
        if (m != null) {
jtulach@0
  1792
            ret = findPkgName(e) + '.' + m.className();
jtulach@0
  1793
            isModel[0] = true;
jtulach@0
  1794
            models.put(e, m.className());
jtulach@0
  1795
        } else if (findModelForMthd(e)) {
jtulach@0
  1796
            ret = ((TypeElement)e).getQualifiedName().toString();
jtulach@0
  1797
            isModel[0] = true;
jtulach@0
  1798
        } else {
jtulach@0
  1799
            ret = tm.toString();
jtulach@0
  1800
        }
jtulach@0
  1801
        TypeMirror enm = processingEnv.getElementUtils().getTypeElement("java.lang.Enum").asType();
jtulach@0
  1802
        enm = processingEnv.getTypeUtils().erasure(enm);
jtulach@0
  1803
        isEnum[0] = processingEnv.getTypeUtils().isSubtype(tm, enm);
jtulach@0
  1804
        return ret;
jtulach@0
  1805
    }
jtulach@938
  1806
jtulach@0
  1807
    private static boolean findModelForMthd(Element clazz) {
jtulach@0
  1808
        if (clazz == null) {
jtulach@0
  1809
            return false;
jtulach@0
  1810
        }
jtulach@0
  1811
        for (Element e : clazz.getEnclosedElements()) {
jtulach@0
  1812
            if (e.getKind() == ElementKind.METHOD) {
jtulach@0
  1813
                ExecutableElement ee = (ExecutableElement)e;
jtulach@0
  1814
                if (
jtulach@0
  1815
                    ee.getSimpleName().contentEquals("modelFor") &&
jtulach@0
  1816
                    ee.getParameters().isEmpty()
jtulach@0
  1817
                ) {
jtulach@0
  1818
                    return true;
jtulach@0
  1819
                }
jtulach@0
  1820
            }
jtulach@0
  1821
        }
jtulach@0
  1822
        return false;
jtulach@0
  1823
    }
jtulach@0
  1824
jtulach@940
  1825
    private void findParamNames(
jtulach@940
  1826
        Set<String> params, Element e, String url, String jsonParam, StringBuilder... both
jaroslav@24
  1827
    ) {
jaroslav@24
  1828
        int wasJSON = 0;
jtulach@0
  1829
jtulach@0
  1830
        for (int pos = 0; ;) {
jtulach@0
  1831
            int next = url.indexOf('{', pos);
jtulach@0
  1832
            if (next == -1) {
jaroslav@24
  1833
                both[wasJSON].append('"')
jtulach@940
  1834
                    .append(url.substring(pos).replace("\"", "\\\""))
jtulach@0
  1835
                    .append('"');
jtulach@940
  1836
                return;
jtulach@0
  1837
            }
jtulach@0
  1838
            int close = url.indexOf('}', next);
jtulach@0
  1839
            if (close == -1) {
jtulach@0
  1840
                error("Unbalanced '{' and '}' in " + url, e);
jtulach@940
  1841
                return;
jtulach@0
  1842
            }
jtulach@0
  1843
            final String paramName = url.substring(next + 1, close);
jtulach@940
  1844
            params.add(paramName);
jtulach@940
  1845
            if (paramName.equals(jsonParam) && !jsonParam.isEmpty()) {
jtulach@940
  1846
                both[wasJSON].append('"')
jtulach@940
  1847
                    .append(url.substring(pos, next).replace("\"", "\\\""))
jtulach@940
  1848
                    .append('"');
jtulach@940
  1849
                wasJSON = 1;
jtulach@940
  1850
            } else {
jtulach@940
  1851
                both[wasJSON].append('"')
jtulach@940
  1852
                    .append(url.substring(pos, next).replace("\"", "\\\""))
jtulach@940
  1853
                    .append("\" + ").append(paramName).append(" + ");
jaroslav@24
  1854
            }
jtulach@0
  1855
            pos = close + 1;
jtulach@0
  1856
        }
jtulach@0
  1857
    }
jtulach@0
  1858
jtulach@0
  1859
    private static Prprt findPrprt(Prprt[] properties, String propName) {
jtulach@0
  1860
        for (Prprt p : properties) {
jtulach@0
  1861
            if (propName.equals(p.name())) {
jtulach@0
  1862
                return p;
jtulach@0
  1863
            }
jtulach@0
  1864
        }
jtulach@0
  1865
        return null;
jtulach@0
  1866
    }
jtulach@0
  1867
jtulach@0
  1868
    private boolean isPrimitive(String type) {
jtulach@938
  1869
        return
jtulach@0
  1870
            "int".equals(type) ||
jtulach@0
  1871
            "double".equals(type) ||
jtulach@0
  1872
            "long".equals(type) ||
jtulach@0
  1873
            "short".equals(type) ||
jtulach@0
  1874
            "byte".equals(type) ||
jaroslav@100
  1875
            "char".equals(type) ||
jaroslav@100
  1876
            "boolean".equals(type) ||
jtulach@0
  1877
            "float".equals(type);
jtulach@0
  1878
    }
jtulach@0
  1879
jaroslav@567
  1880
    private static Collection<String> findDerivedFrom(Map<String, Collection<String[]>> propsDeps, String derivedProp) {
jaroslav@100
  1881
        Set<String> names = new HashSet<String>();
jaroslav@567
  1882
        for (Map.Entry<String, Collection<String[]>> e : propsDeps.entrySet()) {
jaroslav@567
  1883
            for (String[] pair : e.getValue()) {
jaroslav@567
  1884
                if (pair[0].equals(derivedProp)) {
jaroslav@567
  1885
                    names.add(e.getKey());
jaroslav@567
  1886
                    break;
jaroslav@567
  1887
                }
jtulach@0
  1888
            }
jtulach@0
  1889
        }
jtulach@0
  1890
        return names;
jtulach@0
  1891
    }
jtulach@938
  1892
jtulach@0
  1893
    private Prprt[] createProps(Element e, Property[] arr) {
jtulach@0
  1894
        Prprt[] ret = Prprt.wrap(processingEnv, e, arr);
jtulach@0
  1895
        Prprt[] prev = verify.put(e, ret);
jtulach@0
  1896
        if (prev != null) {
jtulach@0
  1897
            error("Two sets of properties for ", e);
jtulach@0
  1898
        }
jtulach@0
  1899
        return ret;
jtulach@0
  1900
    }
jtulach@938
  1901
jaroslav@17
  1902
    private static String strip(String s) {
jaroslav@17
  1903
        int indx = s.indexOf("__");
jaroslav@17
  1904
        if (indx >= 0) {
jaroslav@17
  1905
            return s.substring(0, indx);
jaroslav@17
  1906
        } else {
jaroslav@17
  1907
            return s;
jaroslav@17
  1908
        }
jaroslav@17
  1909
    }
jaroslav@70
  1910
jaroslav@73
  1911
    private String findDataSpecified(ExecutableElement e, OnReceive onR) {
jaroslav@73
  1912
        try {
jaroslav@73
  1913
            return onR.data().getName();
jaroslav@73
  1914
        } catch (MirroredTypeException ex) {
jaroslav@242
  1915
            final TypeMirror tm = ex.getTypeMirror();
jaroslav@242
  1916
            String name;
jaroslav@242
  1917
            final Element te = processingEnv.getTypeUtils().asElement(tm);
jaroslav@242
  1918
            if (te.getKind() == ElementKind.CLASS && tm.getKind() == TypeKind.ERROR) {
jaroslav@242
  1919
                name = te.getSimpleName().toString();
jaroslav@242
  1920
            } else {
jaroslav@242
  1921
                name = tm.toString();
jaroslav@242
  1922
            }
jaroslav@73
  1923
            return "java.lang.Object".equals(name) ? null : name;
jaroslav@73
  1924
        } catch (Exception ex) {
jaroslav@73
  1925
            // fallback
jaroslav@73
  1926
        }
jtulach@938
  1927
jaroslav@72
  1928
        AnnotationMirror found = null;
jaroslav@72
  1929
        for (AnnotationMirror am : e.getAnnotationMirrors()) {
jaroslav@72
  1930
            if (am.getAnnotationType().toString().equals(OnReceive.class.getName())) {
jaroslav@72
  1931
                found = am;
jaroslav@72
  1932
            }
jaroslav@72
  1933
        }
jaroslav@72
  1934
        if (found == null) {
jaroslav@72
  1935
            return null;
jaroslav@72
  1936
        }
jtulach@938
  1937
jaroslav@72
  1938
        for (Map.Entry<? extends ExecutableElement, ? extends AnnotationValue> entry : found.getElementValues().entrySet()) {
jaroslav@72
  1939
            ExecutableElement ee = entry.getKey();
jaroslav@72
  1940
            AnnotationValue av = entry.getValue();
jaroslav@72
  1941
            if (ee.getSimpleName().contentEquals("data")) {
jaroslav@73
  1942
                List<? extends Object> values = getAnnoValues(processingEnv, e, found);
jaroslav@73
  1943
                for (Object v : values) {
jaroslav@73
  1944
                    String sv = v.toString();
jaroslav@73
  1945
                    if (sv.startsWith("data = ") && sv.endsWith(".class")) {
jaroslav@73
  1946
                        return sv.substring(7, sv.length() - 6);
jaroslav@73
  1947
                    }
jaroslav@73
  1948
                }
jaroslav@73
  1949
                return "error";
jaroslav@72
  1950
            }
jaroslav@72
  1951
        }
jaroslav@72
  1952
        return null;
jaroslav@72
  1953
    }
jaroslav@72
  1954
jaroslav@72
  1955
    static List<? extends Object> getAnnoValues(ProcessingEnvironment pe, Element e, AnnotationMirror am) {
jaroslav@70
  1956
        try {
jaroslav@72
  1957
            Class<?> trees = Class.forName("com.sun.tools.javac.api.JavacTrees");
jaroslav@72
  1958
            Method m = trees.getMethod("instance", ProcessingEnvironment.class);
jaroslav@72
  1959
            Object instance = m.invoke(null, pe);
jaroslav@72
  1960
            m = instance.getClass().getMethod("getPath", Element.class, AnnotationMirror.class);
jaroslav@72
  1961
            Object path = m.invoke(instance, e, am);
jaroslav@72
  1962
            m = path.getClass().getMethod("getLeaf");
jaroslav@72
  1963
            Object leaf = m.invoke(path);
jaroslav@72
  1964
            m = leaf.getClass().getMethod("getArguments");
jaroslav@72
  1965
            return (List) m.invoke(leaf);
jaroslav@72
  1966
        } catch (Exception ex) {
jaroslav@72
  1967
            return Collections.emptyList();
jaroslav@70
  1968
        }
jaroslav@70
  1969
    }
jaroslav@240
  1970
jtulach@920
  1971
    private static String findTargetId(Element e) {
jtulach@920
  1972
        for (AnnotationMirror m : e.getAnnotationMirrors()) {
jtulach@920
  1973
            if (m.getAnnotationType().toString().equals(Model.class.getName())) {
jtulach@920
  1974
                for (Map.Entry<? extends ExecutableElement, ? extends AnnotationValue> entrySet : m.getElementValues().entrySet()) {
jtulach@920
  1975
                    ExecutableElement key = entrySet.getKey();
jtulach@920
  1976
                    AnnotationValue value = entrySet.getValue();
jtulach@920
  1977
                    if ("targetId()".equals(key.toString())) {
jtulach@920
  1978
                        return value.toString();
jtulach@920
  1979
                    }
jtulach@920
  1980
                }
jtulach@920
  1981
            }
jtulach@920
  1982
        }
jtulach@920
  1983
        return null;
jtulach@920
  1984
    }
jtulach@920
  1985
jtulach@0
  1986
    private static class Prprt {
jtulach@0
  1987
        private final Element e;
jtulach@0
  1988
        private final AnnotationMirror tm;
jtulach@0
  1989
        private final Property p;
jtulach@0
  1990
jtulach@0
  1991
        public Prprt(Element e, AnnotationMirror tm, Property p) {
jtulach@0
  1992
            this.e = e;
jtulach@0
  1993
            this.tm = tm;
jtulach@0
  1994
            this.p = p;
jtulach@0
  1995
        }
jtulach@938
  1996
jtulach@0
  1997
        String name() {
jtulach@0
  1998
            return p.name();
jtulach@0
  1999
        }
jtulach@938
  2000
jtulach@0
  2001
        boolean array() {
jtulach@0
  2002
            return p.array();
jtulach@0
  2003
        }
jtulach@0
  2004
jtulach@0
  2005
        String typeName(ProcessingEnvironment env) {
jaroslav@100
  2006
            RuntimeException ex;
jtulach@0
  2007
            try {
jtulach@0
  2008
                return p.type().getName();
jaroslav@100
  2009
            } catch (IncompleteAnnotationException e) {
jaroslav@100
  2010
                ex = e;
jaroslav@100
  2011
            } catch (AnnotationTypeMismatchException e) {
jaroslav@100
  2012
                ex = e;
jaroslav@100
  2013
            }
jaroslav@100
  2014
            for (Object v : getAnnoValues(env, e, tm)) {
jaroslav@100
  2015
                String s = v.toString().replace(" ", "");
jaroslav@100
  2016
                if (s.startsWith("type=") && s.endsWith(".class")) {
jaroslav@100
  2017
                    return s.substring(5, s.length() - 6);
jtulach@0
  2018
                }
jtulach@0
  2019
            }
jaroslav@100
  2020
            throw ex;
jtulach@0
  2021
        }
jtulach@938
  2022
jtulach@938
  2023
jtulach@0
  2024
        static Prprt[] wrap(ProcessingEnvironment pe, Element e, Property[] arr) {
jtulach@0
  2025
            if (arr.length == 0) {
jtulach@0
  2026
                return new Prprt[0];
jtulach@0
  2027
            }
jtulach@938
  2028
jtulach@0
  2029
            if (e.getKind() != ElementKind.CLASS) {
jtulach@0
  2030
                throw new IllegalStateException("" + e.getKind());
jtulach@0
  2031
            }
jtulach@0
  2032
            TypeElement te = (TypeElement)e;
jtulach@0
  2033
            List<? extends AnnotationValue> val = null;
jtulach@0
  2034
            for (AnnotationMirror an : te.getAnnotationMirrors()) {
jtulach@0
  2035
                for (Map.Entry<? extends ExecutableElement, ? extends AnnotationValue> entry : an.getElementValues().entrySet()) {
jtulach@0
  2036
                    if (entry.getKey().getSimpleName().contentEquals("properties")) {
jtulach@0
  2037
                        val = (List)entry.getValue().getValue();
jtulach@0
  2038
                        break;
jtulach@0
  2039
                    }
jtulach@0
  2040
                }
jtulach@0
  2041
            }
jtulach@0
  2042
            if (val == null || val.size() != arr.length) {
jtulach@0
  2043
                pe.getMessager().printMessage(Diagnostic.Kind.ERROR, "" + val, e);
jtulach@0
  2044
                return new Prprt[0];
jtulach@0
  2045
            }
jtulach@0
  2046
            Prprt[] ret = new Prprt[arr.length];
jtulach@0
  2047
            BIG: for (int i = 0; i < ret.length; i++) {
jtulach@0
  2048
                AnnotationMirror am = (AnnotationMirror)val.get(i).getValue();
jtulach@0
  2049
                ret[i] = new Prprt(e, am, arr[i]);
jtulach@938
  2050
jtulach@0
  2051
            }
jtulach@0
  2052
            return ret;
jtulach@0
  2053
        }
jtulach@823
  2054
    } // end of Prprt
jtulach@938
  2055
jtulach@823
  2056
    private static final class GetSet {
jtulach@823
  2057
        final String name;
jtulach@823
  2058
        final String getter;
jtulach@823
  2059
        final String setter;
jtulach@823
  2060
        final String type;
jtulach@824
  2061
        final boolean readOnly;
jtulach@824
  2062
        GetSet(String name, String getter, String setter, String type, boolean readOnly) {
jtulach@823
  2063
            this.name = name;
jtulach@823
  2064
            this.getter = getter;
jtulach@823
  2065
            this.setter = setter;
jtulach@823
  2066
            this.type = type;
jtulach@824
  2067
            this.readOnly = readOnly;
jtulach@823
  2068
        }
jtulach@0
  2069
    }
jaroslav@70
  2070
jaroslav@70
  2071
    @Override
jaroslav@70
  2072
    public Iterable<? extends Completion> getCompletions(Element element, AnnotationMirror annotation, ExecutableElement member, String userText) {
jaroslav@71
  2073
        final Level l = Level.FINE;
jaroslav@71
  2074
        LOG.log(l, " element: {0}", element);
jaroslav@71
  2075
        LOG.log(l, " annotation: {0}", annotation);
jaroslav@71
  2076
        LOG.log(l, " member: {0}", member);
jaroslav@71
  2077
        LOG.log(l, " userText: {0}", userText);
jaroslav@71
  2078
        LOG.log(l, "str: {0}", annotation.getAnnotationType().toString());
jaroslav@70
  2079
        if (annotation.getAnnotationType().toString().equals(OnReceive.class.getName())) {
jaroslav@70
  2080
            if (member.getSimpleName().contentEquals("method")) {
jaroslav@70
  2081
                return Arrays.asList(
jaroslav@70
  2082
                    methodOf("GET"),
jaroslav@70
  2083
                    methodOf("POST"),
jaroslav@70
  2084
                    methodOf("PUT"),
jaroslav@70
  2085
                    methodOf("DELETE"),
jaroslav@247
  2086
                    methodOf("HEAD"),
jaroslav@247
  2087
                    methodOf("WebSocket")
jaroslav@70
  2088
                );
jaroslav@70
  2089
            }
jaroslav@70
  2090
        }
jtulach@938
  2091
jaroslav@70
  2092
        return super.getCompletions(element, annotation, member, userText);
jaroslav@70
  2093
    }
jtulach@938
  2094
jaroslav@70
  2095
    private static final Completion methodOf(String method) {
jaroslav@362
  2096
        ResourceBundle rb = ResourceBundle.getBundle("org.netbeans.html.json.impl.Bundle");
jaroslav@70
  2097
        return Completions.of('"' + method + '"', rb.getString("MSG_Completion_" + method));
jaroslav@70
  2098
    }
jtulach@938
  2099
jaroslav@245
  2100
    private boolean findOnError(ExecutableElement errElem, TypeElement te, String name, String className) {
jaroslav@245
  2101
        String err = null;
jaroslav@245
  2102
        METHODS:
jaroslav@245
  2103
        for (Element e : te.getEnclosedElements()) {
jaroslav@245
  2104
            if (e.getKind() != ElementKind.METHOD) {
jaroslav@245
  2105
                continue;
jaroslav@245
  2106
            }
jaroslav@245
  2107
            if (!e.getSimpleName().contentEquals(name)) {
jaroslav@245
  2108
                continue;
jaroslav@245
  2109
            }
jaroslav@245
  2110
            if (!e.getModifiers().contains(Modifier.STATIC)) {
jaroslav@245
  2111
                errElem = (ExecutableElement) e;
jaroslav@245
  2112
                err = "Would have to be static";
jaroslav@245
  2113
                continue;
jaroslav@245
  2114
            }
jaroslav@245
  2115
            ExecutableElement ee = (ExecutableElement) e;
jaroslav@245
  2116
            TypeMirror excType = processingEnv.getElementUtils().getTypeElement(Exception.class.getName()).asType();
jaroslav@245
  2117
            final List<? extends VariableElement> params = ee.getParameters();
jaroslav@245
  2118
            boolean error = false;
jaroslav@245
  2119
            if (params.size() != 2) {
jaroslav@245
  2120
                error = true;
jaroslav@245
  2121
            } else {
jaroslav@261
  2122
                String firstType = params.get(0).asType().toString();
jaroslav@261
  2123
                int lastDot = firstType.lastIndexOf('.');
jaroslav@261
  2124
                if (lastDot != -1) {
jaroslav@261
  2125
                    firstType = firstType.substring(lastDot + 1);
jaroslav@261
  2126
                }
jaroslav@261
  2127
                if (!firstType.equals(className)) {
jaroslav@245
  2128
                    error = true;
jaroslav@245
  2129
                }
jaroslav@245
  2130
                if (!processingEnv.getTypeUtils().isAssignable(excType, params.get(1).asType())) {
jaroslav@245
  2131
                    error = true;
jaroslav@245
  2132
                }
jaroslav@245
  2133
            }
jaroslav@245
  2134
            if (error) {
jaroslav@245
  2135
                errElem = (ExecutableElement) e;
jaroslav@245
  2136
                err = "Error method first argument needs to be " + className + " and second Exception";
jaroslav@245
  2137
                continue;
jaroslav@245
  2138
            }
jaroslav@245
  2139
            return true;
jaroslav@245
  2140
        }
jaroslav@245
  2141
        if (err == null) {
jaroslav@245
  2142
            err = "Cannot find " + name + "(" + className + ", Exception) method in this class";
jaroslav@245
  2143
        }
jaroslav@245
  2144
        error(err, errElem);
jaroslav@245
  2145
        return false;
jaroslav@245
  2146
    }
jtulach@938
  2147
jtulach@951
  2148
    private ExecutableElement findWrite(ExecutableElement computedPropElem, TypeElement te, String name, String className) {
jtulach@951
  2149
        String err = null;
jtulach@951
  2150
        METHODS:
jtulach@951
  2151
        for (Element e : te.getEnclosedElements()) {
jtulach@951
  2152
            if (e.getKind() != ElementKind.METHOD) {
jtulach@951
  2153
                continue;
jtulach@951
  2154
            }
jtulach@951
  2155
            if (!e.getSimpleName().contentEquals(name)) {
jtulach@951
  2156
                continue;
jtulach@951
  2157
            }
jtulach@951
  2158
            if (e.equals(computedPropElem)) {
jtulach@951
  2159
                continue;
jtulach@951
  2160
            }
jtulach@951
  2161
            if (!e.getModifiers().contains(Modifier.STATIC)) {
jtulach@951
  2162
                computedPropElem = (ExecutableElement) e;
jtulach@951
  2163
                err = "Would have to be static";
jtulach@951
  2164
                continue;
jtulach@951
  2165
            }
jtulach@951
  2166
            ExecutableElement ee = (ExecutableElement) e;
jtulach@955
  2167
            if (ee.getReturnType().getKind() != TypeKind.VOID) {
jtulach@955
  2168
                computedPropElem = (ExecutableElement) e;
jtulach@955
  2169
                err = "Write method has to return void";
jtulach@955
  2170
                continue;
jtulach@955
  2171
            }
jtulach@951
  2172
            TypeMirror retType = computedPropElem.getReturnType();
jtulach@951
  2173
            final List<? extends VariableElement> params = ee.getParameters();
jtulach@951
  2174
            boolean error = false;
jtulach@951
  2175
            if (params.size() != 2) {
jtulach@951
  2176
                error = true;
jtulach@951
  2177
            } else {
jtulach@951
  2178
                String firstType = params.get(0).asType().toString();
jtulach@951
  2179
                int lastDot = firstType.lastIndexOf('.');
jtulach@951
  2180
                if (lastDot != -1) {
jtulach@951
  2181
                    firstType = firstType.substring(lastDot + 1);
jtulach@951
  2182
                }
jtulach@951
  2183
                if (!firstType.equals(className)) {
jtulach@951
  2184
                    error = true;
jtulach@951
  2185
                }
jtulach@951
  2186
                if (!processingEnv.getTypeUtils().isAssignable(retType, params.get(1).asType())) {
jtulach@951
  2187
                    error = true;
jtulach@951
  2188
                }
jtulach@951
  2189
            }
jtulach@951
  2190
            if (error) {
jtulach@951
  2191
                computedPropElem = (ExecutableElement) e;
jtulach@951
  2192
                err = "Write method first argument needs to be " + className + " and second " + retType + " or Object";
jtulach@951
  2193
                continue;
jtulach@951
  2194
            }
jtulach@951
  2195
            return ee;
jtulach@951
  2196
        }
jtulach@951
  2197
        if (err == null) {
jtulach@951
  2198
            err = "Cannot find " + name + "(" + className + ", value) method in this class";
jtulach@951
  2199
        }
jtulach@951
  2200
        error(err, computedPropElem);
jtulach@951
  2201
        return null;
jtulach@951
  2202
    }
jtulach@951
  2203
jtulach@1035
  2204
    private boolean processComponent(Element e) {
jtulach@1035
  2205
        try {
jtulach@1035
  2206
            Component c = e.getAnnotation(Component.class);
jtulach@1035
  2207
            if (c == null) {
jtulach@1035
  2208
                return false;
jtulach@1035
  2209
            }
jtulach@1035
  2210
            if (e.getKind() != ElementKind.METHOD) {
jtulach@1035
  2211
                error("@Component can only annotate a method", e);
jtulach@1035
  2212
                return false;
jtulach@1035
  2213
            }
jtulach@1035
  2214
            ExecutableElement ee = (ExecutableElement) e;
jtulach@1035
  2215
            if (ee.getParameters().size() != 1) {
jtulach@1035
  2216
                error("Method annotated by @Component needs to take one @Model generated parameter", e);
jtulach@1035
  2217
                return false;
jtulach@1035
  2218
            }
jtulach@1035
  2219
            FileObject def = processingEnv.getFiler().createResource(StandardLocation.CLASS_OUTPUT, "", "META-INF/html+java/components/" + c.name(), e);
jtulach@1035
  2220
            OutputStreamWriter w = new OutputStreamWriter(def.openOutputStream());
jtulach@1035
  2221
            Properties p = new Properties();
jtulach@1035
  2222
            final TypeMirror typeParam = ee.getParameters().get(0).asType();
jtulach@1035
  2223
            String fqnParam;
jtulach@1035
  2224
            if (typeParam.getKind() == TypeKind.ERROR) {
jtulach@1035
  2225
                fqnParam = findPkgName(ee) + "." + typeParam;
jtulach@1035
  2226
            } else {
jtulach@1035
  2227
                final TypeElement elem = (TypeElement) processingEnv.getTypeUtils().asElement(typeParam);
jtulach@1035
  2228
                fqnParam = processingEnv.getElementUtils().getBinaryName(elem).toString();
jtulach@1035
  2229
            }
jtulach@1035
  2230
            p.setProperty("paramName", fqnParam);
jtulach@1035
  2231
            p.setProperty("methodName", e.getSimpleName().toString());
jtulach@1035
  2232
            TypeElement clazz = (TypeElement)e.getEnclosingElement();
jtulach@1035
  2233
            final String fqn = clazz.getQualifiedName().toString();
jtulach@1035
  2234
            p.setProperty("className", fqn);
jtulach@1035
  2235
            p.setProperty("template", findPkgName(e).replace('.', '/') + '/' + c.template());
jtulach@1035
  2236
            p.store(w, "Generated by " + e);
jtulach@1035
  2237
            w.close();
jtulach@1035
  2238
            return true;
jtulach@1035
  2239
        } catch (IOException ex) {
jtulach@1035
  2240
            error(ex.getMessage(), e);
jtulach@1035
  2241
            return false;
jtulach@1035
  2242
        }
jtulach@1035
  2243
    }
jtulach@0
  2244
}