All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.threerings.presents.tools.GenActionScriptStreamableTask Maven / Gradle / Ivy

The newest version!
//
// $Id$
//
// Narya library - tools for developing networked games
// Copyright (C) 2002-2012 Three Rings Design, Inc., All Rights Reserved
// http://code.google.com/p/narya/
//
// This library is free software; you can redistribute it and/or modify it
// under the terms of the GNU Lesser General Public License as published
// by the Free Software Foundation; either version 2.1 of the License, or
// (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA

package com.threerings.presents.tools;

import java.io.File;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.tools.ant.Project;

import com.google.common.base.Charsets;
import com.google.common.base.Joiner;
import com.google.common.primitives.Primitives;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.common.io.Files;

import com.samskivert.util.StringUtil;

import com.threerings.io.ObjectInputStream;
import com.threerings.io.ObjectOutputStream;
import com.threerings.io.Streamable;
import com.threerings.presents.data.InvocationMarshaller;
import com.threerings.presents.dobj.DObject;
import com.threerings.presents.dobj.DSet;
import com.threerings.presents.dobj.OidList;

public class GenActionScriptStreamableTask extends GenTask
{
    /**
     * Configures the path to our ActionScript source files.
     */
    public void setAsroot (File asroot)
    {
        _asroot = asroot;
    }

    @Override
    protected void processClass (File javaSource, Class sclass)
        throws Exception
    {
        boolean streamable = Streamable.class.isAssignableFrom(sclass) || sclass.isEnum();
        if (!streamable
            || InvocationMarshaller.class.isAssignableFrom(sclass)
            || Modifier.isInterface(sclass.getModifiers())
            || ActionScriptUtils.hasOmitAnnotation(sclass)) {
            log("Skipping " + sclass.getName(), Project.MSG_VERBOSE);
            return;
        }
        log("Generating " + sclass.getName(), Project.MSG_VERBOSE);

        // Read the existing file in, if it exists
        File outputLocation = ActionScriptUtils.createActionScriptPath(_asroot, sclass);
        String existing = null;
        if (outputLocation.exists()) {
            existing = Files.toString(outputLocation, Charsets.UTF_8);
        }

        // Generate the current version of the streamable
        StreamableClassRequirements reqs = new StreamableClassRequirements(sclass);

        ImportSet imports = new ImportSet();
        String extendsName = "";
        if (sclass.isEnum()) {
            imports.add("com.threerings.util.Enum");
            extendsName = "Enum";
        } else {
            imports.add(ObjectInputStream.class.getName());
            imports.add(ObjectOutputStream.class.getName());
        }
        if (!sclass.getSuperclass().equals(Object.class)) {
            extendsName =
                ActionScriptUtils.addImportAndGetShortType(sclass.getSuperclass(), false, imports);
        }

        // Read the existing file's imports in. Eclipse's import organization makes
        // it a pain to keep additional imports out of the GENERATED PREAMBLE section of
        // class, so we just merge all the imports we find.
        if (existing != null) {
            ActionScriptUtils.addExistingImports(existing, imports);
        }

        boolean isDObject = DObject.class.isAssignableFrom(sclass);
        if (isDObject) {
            imports.add("org.osflash.signals.Signal");
        }

        Set implemented = Sets.newLinkedHashSet();
        for (Class iface : sclass.getInterfaces()) {
            implemented.add(ActionScriptUtils.addImportAndGetShortType(iface, false, imports));
        }
        List pubFields = Lists.newArrayList();
        List protFields = Lists.newArrayList();
        for (Field f : reqs.streamedFields) {
            int mods = f.getModifiers();
            if (Modifier.isPublic(mods)) {
                pubFields.add(new ASField(f, imports));
            } else if (Modifier.isProtected(mods)) {
                protFields.add(new ASField(f, imports));
            } else {
                // don't care about private
                continue;
            }
        }
        List enumFields = Lists.newArrayList();
        if (sclass.isEnum()) {
            Object[] enums = sclass.getEnumConstants();
            for (Object e : enums) {
                enumFields.add(new ASEnum((Enum)e));
            }
        }

        imports.removeGlobals();

        String template = sclass.isEnum() ? "enum_as.tmpl" : "streamable_as.tmpl";
        String output = mergeTemplate("com/threerings/presents/tools/" + template,
            "header", existing == null ? _header : "",
            "package", sclass.getPackage().getName(),
            "classname", ActionScriptUtils.toSimpleName(sclass),
            "importGroups", imports.toGroups(),
            "extends", extendsName,
            "implements", Joiner.on(", ").join(implemented),
            "superclassStreamable", reqs.superclassStreamable,
            "pubFields", pubFields,
            "enumFields", enumFields,
            "protFields", protFields,
            "dobject", isDObject);

        if (existing != null) {
            // Merge in the previously generated version
            output = new GeneratedSourceMerger().merge(output, existing);
        }
        writeFile(outputLocation.getAbsolutePath(), output);

        // generate inner enums
        for (Class inner : sclass.getDeclaredClasses()) {
            if (inner.isEnum()) {
                processClass(javaSource, inner);
            }
        }
    }

    protected static class ASField
    {
        public final String name;
        public final String capitalName;
        public final String simpleType;
        public final String parameterTypes;
        public final String reader;
        public final String writer;
        public final String dobjectField;
        public boolean dset;
        public boolean array;
        public boolean oidList;
        public boolean hasTypeParameters;

        public ASField (Field f, ImportSet imports)
        {
            name = f.getName();
            capitalName = StringUtil.capitalize(name);
            dobjectField = StringUtil.unStudlyName(name).toUpperCase();
            simpleType = ActionScriptUtils.addImportAndGetShortType(f.getType(), true, imports);

            List parameters = Lists.newLinkedList();
            Type genType = f.getGenericType();
            if (genType instanceof ParameterizedType) {
                for (Type param : ((ParameterizedType) genType).getActualTypeArguments()) {
                    if (param instanceof Class) {
                        // Convert any box classes to primitives to avoid having to import them
                        Class primitive = Primitives.unwrap((Class) param);
                        parameters.add(ActionScriptUtils.addImportAndGetShortType(
                            primitive, false, imports));
                    }
                }
            }
            if (!parameters.isEmpty()) {
                parameterTypes = Joiner.on(", ").join(parameters);
                hasTypeParameters = true;
            } else {
                parameterTypes = "";
            }

            // Lists and Maps use their Streamers directly
            if (List.class.isAssignableFrom(f.getType())) {
                imports.add("com.threerings.io.streamers.ArrayStreamer");
            }  else if (Map.class.isAssignableFrom(f.getType())) {
                imports.add("com.threerings.io.streamers.MapStreamer");
            } else if (Set.class.isAssignableFrom(f.getType())) {
                imports.add("com.threerings.io.streamers.SetStreamer");
            } else if (DSet.class.isAssignableFrom(f.getType())) {
                dset = true;
            } else if (OidList.class.isAssignableFrom(f.getType())) {
                oidList = true;
            }
            array = f.getType().isArray();
            reader = ActionScriptUtils.toReadObject(f.getType());
            writer = ActionScriptUtils.toWriteObject(f.getType(), name);
        }
    }

    protected static class ASEnum
    {
        public final String name;

        public ASEnum (Enum e)
        {
            name = e.name();
        }
    }

    /** The path to our ActionScript source files. */
    protected File _asroot;
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy