de.bitowl.advent.game2.MyAtlasTmxMapLoader.java Source code

Java tutorial

Introduction

Here is the source code for de.bitowl.advent.game2.MyAtlasTmxMapLoader.java

Source

package de.bitowl.advent.game2;

/*******************************************************************************
 * Copyright 2011 See AUTHORS file.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 ******************************************************************************/

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.util.StringTokenizer;
import java.util.zip.DataFormatException;
import java.util.zip.GZIPInputStream;
import java.util.zip.Inflater;

import com.badlogic.gdx.assets.AssetDescriptor;
import com.badlogic.gdx.assets.AssetLoaderParameters;
import com.badlogic.gdx.assets.AssetManager;
import com.badlogic.gdx.assets.loaders.AsynchronousAssetLoader;
import com.badlogic.gdx.assets.loaders.FileHandleResolver;
import com.badlogic.gdx.assets.loaders.resolvers.InternalFileHandleResolver;
import com.badlogic.gdx.files.FileHandle;
import com.badlogic.gdx.graphics.Texture;
import com.badlogic.gdx.graphics.Texture.TextureFilter;
import com.badlogic.gdx.graphics.g2d.TextureAtlas;
import com.badlogic.gdx.graphics.g2d.TextureAtlas.AtlasRegion;
import com.badlogic.gdx.maps.MapLayer;
import com.badlogic.gdx.maps.MapObject;
import com.badlogic.gdx.maps.MapProperties;
import com.badlogic.gdx.maps.objects.EllipseMapObject;
import com.badlogic.gdx.maps.objects.PolygonMapObject;
import com.badlogic.gdx.maps.objects.PolylineMapObject;
import com.badlogic.gdx.maps.objects.RectangleMapObject;
import com.badlogic.gdx.maps.tiled.TiledMap;
import com.badlogic.gdx.maps.tiled.TiledMapTile;
import com.badlogic.gdx.maps.tiled.TiledMapTileLayer;
import com.badlogic.gdx.maps.tiled.TiledMapTileLayer.Cell;
import com.badlogic.gdx.maps.tiled.TiledMapTileSet;
import com.badlogic.gdx.maps.tiled.TiledMapTileSets;
import com.badlogic.gdx.maps.tiled.tiles.StaticTiledMapTile;
import com.badlogic.gdx.math.Polygon;
import com.badlogic.gdx.math.Polyline;
import com.badlogic.gdx.utils.Array;
import com.badlogic.gdx.utils.Base64Coder;
import com.badlogic.gdx.utils.GdxRuntimeException;
import com.badlogic.gdx.utils.ObjectMap;
import com.badlogic.gdx.utils.XmlReader;
import com.badlogic.gdx.utils.XmlReader.Element;

/** A TiledMap Loader which loads tiles from a TextureAtlas instead of separate images.
 * 
 * It requires a map-level property called 'atlas' with its value being the relative path to the TextureAtlas. The atlas must have
 * in it indexed regions named after the tilesets used in the map. The indexes shall be local to the tileset (not the global id).
 * Strip whitespace and rotation should not be used when creating the atlas.
 * 
 * @author Justin Shapcott
 * @author Manuel Bua */
public class MyAtlasTmxMapLoader
        extends AsynchronousAssetLoader<TiledMap, MyAtlasTmxMapLoader.AtlasTiledMapLoaderParameters> {

    public static class AtlasTiledMapLoaderParameters extends AssetLoaderParameters<TiledMap> {
        /** Whether to load the map for a y-up coordinate system */
        public boolean yUp = true;
        /** force texture filters? **/
        public boolean forceTextureFilters = false;
        /** The TextureFilter to use for minification, if forceTextureFilter is enabled **/
        public TextureFilter textureMinFilter = TextureFilter.Nearest;
        /** The TextureFilter to use for magnification, if forceTextureFilter is enabled **/
        public TextureFilter textureMagFilter = TextureFilter.Nearest;
        /** Whether to convert the objects' pixel position and size to the equivalent in tile space. **/
        public boolean convertObjectToTileSpace = false;
    }

    protected static final int FLAG_FLIP_HORIZONTALLY = 0x80000000;
    protected static final int FLAG_FLIP_VERTICALLY = 0x40000000;
    protected static final int FLAG_FLIP_DIAGONALLY = 0x20000000;
    protected static final int MASK_CLEAR = 0xE0000000;

    protected XmlReader xml = new XmlReader();
    protected Element root;
    protected boolean yUp;
    protected boolean convertObjectToTileSpace;

    protected int mapTileWidth;
    protected int mapTileHeight;
    protected int mapWidthInPixels;
    protected int mapHeightInPixels;

    protected TiledMap map;
    protected Array<Texture> trackedTextures = new Array<Texture>();

    private interface AtlasResolver {

        public TextureAtlas getAtlas(String name);

        public static class DirectAtlasResolver implements AtlasResolver {

            private final ObjectMap<String, TextureAtlas> atlases;

            public DirectAtlasResolver(ObjectMap<String, TextureAtlas> atlases) {
                this.atlases = atlases;
            }

            @Override
            public TextureAtlas getAtlas(String name) {
                return atlases.get(name);
            }

        }

        public static class AssetManagerAtlasResolver implements AtlasResolver {
            private final AssetManager assetManager;

            public AssetManagerAtlasResolver(AssetManager assetManager) {
                this.assetManager = assetManager;
            }

            @Override
            public TextureAtlas getAtlas(String name) {
                return assetManager.get(name, TextureAtlas.class);
            }
        }
    }

    public MyAtlasTmxMapLoader() {
        super(new InternalFileHandleResolver());
    }

    public MyAtlasTmxMapLoader(FileHandleResolver resolver) {
        super(resolver);
    }

    public TiledMap load(String fileName) {
        return load(fileName, new AtlasTiledMapLoaderParameters());
    }

    @Override
    public Array<AssetDescriptor> getDependencies(String fileName, FileHandle tmxFile,
            AtlasTiledMapLoaderParameters parameter) {
        Array<AssetDescriptor> dependencies = new Array<AssetDescriptor>();
        try {
            root = xml.parse(tmxFile);

            Element properties = root.getChildByName("properties");
            if (properties != null) {
                for (Element property : properties.getChildrenByName("property")) {
                    String name = property.getAttribute("name");
                    String value = property.getAttribute("value");
                    if (name.startsWith("atlas")) {
                        FileHandle atlasHandle = getRelativeFileHandle(tmxFile, value);
                        dependencies.add(new AssetDescriptor(atlasHandle, TextureAtlas.class));
                    }
                }
            }
        } catch (IOException e) {
            throw new GdxRuntimeException("Unable to parse .tmx file.");
        }
        return dependencies;
    }

    public TiledMap load(String fileName, AtlasTiledMapLoaderParameters parameter) {
        try {
            if (parameter != null) {
                yUp = parameter.yUp;
                convertObjectToTileSpace = parameter.convertObjectToTileSpace;
            } else {
                yUp = true;
                convertObjectToTileSpace = false;
            }

            FileHandle tmxFile = resolve(fileName);
            root = xml.parse(tmxFile);
            ObjectMap<String, TextureAtlas> atlases = new ObjectMap<String, TextureAtlas>();
            FileHandle atlasFile = loadAtlas(root, tmxFile);
            if (atlasFile == null) {
                throw new GdxRuntimeException("Couldn't load atlas");
            }

            TextureAtlas atlas = new TextureAtlas(atlasFile);
            atlases.put(atlasFile.path(), atlas);

            AtlasResolver.DirectAtlasResolver atlasResolver = new AtlasResolver.DirectAtlasResolver(atlases);
            TiledMap map = loadMap(root, tmxFile, atlasResolver, parameter);
            map.setOwnedResources(atlases.values().toArray());
            setTextureFilters(parameter.textureMinFilter, parameter.textureMagFilter);

            return map;
        } catch (IOException e) {
            throw new GdxRuntimeException("Couldn't load tilemap '" + fileName + "'", e);
        }
    }

    protected FileHandle loadAtlas(Element root, FileHandle tmxFile) throws IOException {
        Element e = root.getChildByName("properties");

        if (e != null) {
            for (Element property : e.getChildrenByName("property")) {
                String name = property.getAttribute("name", null);
                String value = property.getAttribute("value", null);
                if (name.equals("atlas")) {
                    if (value == null) {
                        value = property.getText();
                    }

                    if (value == null || value.length() == 0) {
                        // keep trying until there are no more atlas properties
                        continue;
                    }

                    return getRelativeFileHandle(tmxFile, value);
                }
            }
        }

        return null;
    }

    private void setTextureFilters(TextureFilter min, TextureFilter mag) {
        for (Texture texture : trackedTextures) {
            texture.setFilter(min, mag);
        }
    }

    @Override
    public void loadAsync(AssetManager manager, String fileName, FileHandle tmxFile,
            AtlasTiledMapLoaderParameters parameter) {
        map = null;

        if (parameter != null) {
            yUp = parameter.yUp;
            convertObjectToTileSpace = parameter.convertObjectToTileSpace;
        } else {
            yUp = true;
            convertObjectToTileSpace = false;
        }

        try {
            map = loadMap(root, tmxFile, new AtlasResolver.AssetManagerAtlasResolver(manager), parameter);
        } catch (Exception e) {
            throw new GdxRuntimeException("Couldn't load tilemap '" + fileName + "'", e);
        }
    }

    @Override
    public TiledMap loadSync(AssetManager manager, String fileName, FileHandle file,
            AtlasTiledMapLoaderParameters parameter) {
        if (parameter != null) {
            setTextureFilters(parameter.textureMinFilter, parameter.textureMagFilter);
        }

        return map;
    }

    protected TiledMap loadMap(Element root, FileHandle tmxFile, AtlasResolver resolver,
            AtlasTiledMapLoaderParameters parameter) {
        TiledMap map = new TiledMap();

        String mapOrientation = root.getAttribute("orientation", null);
        int mapWidth = root.getIntAttribute("width", 0);
        int mapHeight = root.getIntAttribute("height", 0);
        int tileWidth = root.getIntAttribute("tilewidth", 0);
        int tileHeight = root.getIntAttribute("tileheight", 0);
        String mapBackgroundColor = root.getAttribute("backgroundcolor", null);

        MapProperties mapProperties = map.getProperties();
        if (mapOrientation != null) {
            mapProperties.put("orientation", mapOrientation);
        }
        mapProperties.put("width", mapWidth);
        mapProperties.put("height", mapHeight);
        mapProperties.put("tilewidth", tileWidth);
        mapProperties.put("tileheight", tileHeight);
        if (mapBackgroundColor != null) {
            mapProperties.put("backgroundcolor", mapBackgroundColor);
        }

        mapTileWidth = tileWidth;
        mapTileHeight = tileHeight;
        mapWidthInPixels = mapWidth * tileWidth;
        mapHeightInPixels = mapHeight * tileHeight;

        for (int i = 0, j = root.getChildCount(); i < j; i++) {
            Element element = root.getChild(i);
            String elementName = element.getName();
            if (elementName.equals("properties")) {
                loadProperties(map.getProperties(), element);
            } else if (elementName.equals("tileset")) {
                loadTileset(map, element, tmxFile, resolver, parameter);
            } else if (elementName.equals("layer")) {
                loadTileLayer(map, element);
            } else if (elementName.equals("objectgroup")) {
                loadObjectGroup(map, element);
            }
        }
        return map;
    }

    protected void loadTileset(TiledMap map, Element element, FileHandle tmxFile, AtlasResolver resolver,
            AtlasTiledMapLoaderParameters parameter) {
        if (element.getName().equals("tileset")) {
            String name = element.get("name", null);
            int firstgid = element.getIntAttribute("firstgid", 1);
            int tilewidth = element.getIntAttribute("tilewidth", 0);
            int tileheight = element.getIntAttribute("tileheight", 0);
            int spacing = element.getIntAttribute("spacing", 0);
            int margin = element.getIntAttribute("margin", 0);
            String source = element.getAttribute("source", null);

            String imageSource = "";
            int imageWidth = 0, imageHeight = 0;

            FileHandle image = null;
            if (source != null) {
                FileHandle tsx = getRelativeFileHandle(tmxFile, source);
                try {
                    element = xml.parse(tsx);
                    name = element.get("name", null);
                    tilewidth = element.getIntAttribute("tilewidth", 0);
                    tileheight = element.getIntAttribute("tileheight", 0);
                    spacing = element.getIntAttribute("spacing", 0);
                    margin = element.getIntAttribute("margin", 0);
                    imageSource = element.getChildByName("image").getAttribute("source");
                    imageWidth = element.getChildByName("image").getIntAttribute("width", 0);
                    imageHeight = element.getChildByName("image").getIntAttribute("height", 0);
                } catch (IOException e) {
                    throw new GdxRuntimeException("Error parsing external tileset.");
                }
            } else {
                imageSource = element.getChildByName("image").getAttribute("source");
                imageWidth = element.getChildByName("image").getIntAttribute("width", 0);
                imageHeight = element.getChildByName("image").getIntAttribute("height", 0);
            }

            if (!map.getProperties().containsKey("atlas")) {
                throw new GdxRuntimeException("The map is missing the 'atlas' property");
            }

            // get the TextureAtlas for this tileset
            FileHandle atlasHandle = getRelativeFileHandle(tmxFile, map.getProperties().get("atlas", String.class));
            atlasHandle = resolve(atlasHandle.path());
            TextureAtlas atlas = resolver.getAtlas(atlasHandle.path());
            //String regionsName = atlasHandle.nameWithoutExtension();
            String regionsName = name;

            if (parameter != null && parameter.forceTextureFilters) {
                for (Texture texture : atlas.getTextures()) {
                    trackedTextures.add(texture);
                }
            }

            TiledMapTileSet tileset = new TiledMapTileSet();
            MapProperties props = tileset.getProperties();
            tileset.setName(name);
            props.put("firstgid", firstgid);
            props.put("imagesource", imageSource);
            props.put("imagewidth", imageWidth);
            props.put("imageheight", imageHeight);
            props.put("tilewidth", tilewidth);
            props.put("tileheight", tileheight);
            props.put("margin", margin);
            props.put("spacing", spacing);

            Array<AtlasRegion> regions = atlas.findRegions(regionsName);
            System.out.println(regions.size);
            for (AtlasRegion region : regions) {
                // handle unused tile ids
                if (region != null) {
                    StaticTiledMapTile tile = new StaticTiledMapTile(region);

                    if (!yUp) {
                        region.flip(false, true);
                    }

                    int tileid = firstgid + region.index;
                    tile.setId(tileid);
                    tileset.putTile(tileid, tile);
                    System.out.println("put tile " + tileid);
                }
            }

            Array<Element> tileElements = element.getChildrenByName("tile");

            for (Element tileElement : tileElements) {
                int localtid = tileElement.getIntAttribute("id", 0);
                TiledMapTile tile = tileset.getTile(firstgid + localtid);
                if (tile != null) {
                    String terrain = tileElement.getAttribute("terrain", null);
                    if (terrain != null) {
                        tile.getProperties().put("terrain", terrain);
                    }
                    String probability = tileElement.getAttribute("probability", null);
                    if (probability != null) {
                        tile.getProperties().put("probability", probability);
                    }
                    Element properties = tileElement.getChildByName("properties");
                    if (properties != null) {
                        loadProperties(tile.getProperties(), properties);
                    }
                }
            }

            Element properties = element.getChildByName("properties");
            if (properties != null) {
                loadProperties(tileset.getProperties(), properties);
            }
            System.out.println("add tileset to map: " + tileset.getName());
            map.getTileSets().addTileSet(tileset);
        }
    }

    protected void loadTileLayer(TiledMap map, Element element) {
        if (element.getName().equals("layer")) {
            String name = element.getAttribute("name", null);
            int width = element.getIntAttribute("width", 0);
            int height = element.getIntAttribute("height", 0);
            int tileWidth = element.getParent().getIntAttribute("tilewidth", 0);
            int tileHeight = element.getParent().getIntAttribute("tileheight", 0);
            boolean visible = element.getIntAttribute("visible", 1) == 1;
            float opacity = element.getFloatAttribute("opacity", 1.0f);
            TiledMapTileLayer layer = new TiledMapTileLayer(width, height, tileWidth, tileHeight);
            layer.setVisible(visible);
            layer.setOpacity(opacity);
            layer.setName(name);

            TiledMapTileSets tilesets = map.getTileSets();

            Element data = element.getChildByName("data");
            String encoding = data.getAttribute("encoding", null);
            String compression = data.getAttribute("compression", null);
            if (encoding == null) { // no 'encoding' attribute means that the encoding is XML
                throw new GdxRuntimeException("Unsupported encoding (XML) for TMX Layer Data");
            }
            if (encoding.equals("csv")) {
                String[] array = data.getText().split(",");
                for (int y = 0; y < height; y++) {
                    for (int x = 0; x < width; x++) {
                        int id = (int) Long.parseLong(array[y * width + x].trim());

                        final boolean flipHorizontally = ((id & FLAG_FLIP_HORIZONTALLY) != 0);
                        final boolean flipVertically = ((id & FLAG_FLIP_VERTICALLY) != 0);
                        final boolean flipDiagonally = ((id & FLAG_FLIP_DIAGONALLY) != 0);

                        id = id & ~MASK_CLEAR;

                        tilesets.getTile(id);

                        TiledMapTile tile = tilesets.getTile(id);
                        if (tile != null) {
                            Cell cell = createTileLayerCell(flipHorizontally, flipVertically, flipDiagonally);
                            cell.setTile(tile);
                            layer.setCell(x, yUp ? height - 1 - y : y, cell);
                        }
                    }
                }
            } else {
                if (encoding.equals("base64")) {
                    byte[] bytes = Base64Coder.decode(data.getText());
                    if (compression == null) {
                        int read = 0;
                        for (int y = 0; y < height; y++) {
                            for (int x = 0; x < width; x++) {

                                int id = unsignedByteToInt(bytes[read++]) | unsignedByteToInt(bytes[read++]) << 8
                                        | unsignedByteToInt(bytes[read++]) << 16
                                        | unsignedByteToInt(bytes[read++]) << 24;

                                final boolean flipHorizontally = ((id & FLAG_FLIP_HORIZONTALLY) != 0);
                                final boolean flipVertically = ((id & FLAG_FLIP_VERTICALLY) != 0);
                                final boolean flipDiagonally = ((id & FLAG_FLIP_DIAGONALLY) != 0);

                                id = id & ~MASK_CLEAR;

                                tilesets.getTile(id);
                                TiledMapTile tile = tilesets.getTile(id);
                                if (tile != null) {
                                    Cell cell = createTileLayerCell(flipHorizontally, flipVertically,
                                            flipDiagonally);
                                    cell.setTile(tile);
                                    layer.setCell(x, yUp ? height - 1 - y : y, cell);
                                }
                            }
                        }
                    } else if (compression.equals("gzip")) {
                        GZIPInputStream GZIS = null;
                        try {
                            GZIS = new GZIPInputStream(new ByteArrayInputStream(bytes), bytes.length);
                        } catch (IOException e) {
                            throw new GdxRuntimeException(
                                    "Error Reading TMX Layer Data - IOException: " + e.getMessage());
                        }

                        byte[] temp = new byte[4];
                        for (int y = 0; y < height; y++) {
                            for (int x = 0; x < width; x++) {
                                try {
                                    GZIS.read(temp, 0, 4);
                                    int id = unsignedByteToInt(temp[0]) | unsignedByteToInt(temp[1]) << 8
                                            | unsignedByteToInt(temp[2]) << 16 | unsignedByteToInt(temp[3]) << 24;

                                    final boolean flipHorizontally = ((id & FLAG_FLIP_HORIZONTALLY) != 0);
                                    final boolean flipVertically = ((id & FLAG_FLIP_VERTICALLY) != 0);
                                    final boolean flipDiagonally = ((id & FLAG_FLIP_DIAGONALLY) != 0);

                                    id = id & ~MASK_CLEAR;

                                    tilesets.getTile(id);
                                    TiledMapTile tile = tilesets.getTile(id);
                                    if (tile != null) {
                                        Cell cell = createTileLayerCell(flipHorizontally, flipVertically,
                                                flipDiagonally);
                                        cell.setTile(tile);
                                        layer.setCell(x, yUp ? height - 1 - y : y, cell);
                                    }
                                } catch (IOException e) {
                                    throw new GdxRuntimeException("Error Reading TMX Layer Data.", e);
                                }
                            }
                        }
                    } else if (compression.equals("zlib")) {
                        Inflater zlib = new Inflater();

                        byte[] temp = new byte[4];

                        zlib.setInput(bytes, 0, bytes.length);

                        for (int y = 0; y < height; y++) {
                            for (int x = 0; x < width; x++) {
                                try {
                                    zlib.inflate(temp, 0, 4);
                                    int id = unsignedByteToInt(temp[0]) | unsignedByteToInt(temp[1]) << 8
                                            | unsignedByteToInt(temp[2]) << 16 | unsignedByteToInt(temp[3]) << 24;

                                    final boolean flipHorizontally = ((id & FLAG_FLIP_HORIZONTALLY) != 0);
                                    final boolean flipVertically = ((id & FLAG_FLIP_VERTICALLY) != 0);
                                    final boolean flipDiagonally = ((id & FLAG_FLIP_DIAGONALLY) != 0);

                                    id = id & ~MASK_CLEAR;

                                    tilesets.getTile(id);
                                    if (id != 0) {
                                        String inset = "NONE";
                                        if (tilesets.getTileSet(0).getTile(id) != null) {
                                            inset = tilesets.getTileSet(0).getName();
                                        } else if (tilesets.getTileSet(1).getTile(id) != null) {
                                            inset = tilesets.getTileSet(1).getName();
                                        }
                                        System.out.println("need tile " + id + inset);
                                    }

                                    TiledMapTile tile = tilesets.getTile(id);
                                    if (tile != null) {
                                        Cell cell = createTileLayerCell(flipHorizontally, flipVertically,
                                                flipDiagonally);
                                        cell.setTile(tile);
                                        layer.setCell(x, yUp ? height - 1 - y : y, cell);
                                    }

                                } catch (DataFormatException e) {
                                    throw new GdxRuntimeException("Error Reading TMX Layer Data.", e);
                                }
                            }
                        }
                    }
                } else {
                    // any other value of 'encoding' is one we're not aware of, probably a feature of a future version of Tiled
                    // or another editor
                    throw new GdxRuntimeException("Unrecognised encoding (" + encoding + ") for TMX Layer Data");
                }
            }
            Element properties = element.getChildByName("properties");
            if (properties != null) {
                loadProperties(layer.getProperties(), properties);
            }
            map.getLayers().add(layer);
        }
    }

    protected void loadObjectGroup(TiledMap map, Element element) {
        if (element.getName().equals("objectgroup")) {
            String name = element.getAttribute("name", null);
            MapLayer layer = new MapLayer();
            layer.setName(name);
            Element properties = element.getChildByName("properties");
            if (properties != null) {
                loadProperties(layer.getProperties(), properties);
            }

            for (Element objectElement : element.getChildrenByName("object")) {
                loadObject(layer, objectElement);
            }

            map.getLayers().add(layer);
        }
    }

    protected void loadObject(MapLayer layer, Element element) {
        if (element.getName().equals("object")) {
            MapObject object = null;

            float scaleX = convertObjectToTileSpace ? 1.0f / mapTileWidth : 1.0f;
            float scaleY = convertObjectToTileSpace ? 1.0f / mapTileHeight : 1.0f;

            float x = element.getIntAttribute("x", 0) * scaleX;
            float y = (yUp ? mapHeightInPixels - element.getIntAttribute("y", 0) : element.getIntAttribute("y", 0))
                    * scaleY;

            float width = element.getIntAttribute("width", 0) * scaleX;
            float height = element.getIntAttribute("height", 0) * scaleY;

            if (element.getChildCount() > 0) {
                Element child = null;
                if ((child = element.getChildByName("polygon")) != null) {
                    String[] points = child.getAttribute("points").split(" ");
                    float[] vertices = new float[points.length * 2];
                    for (int i = 0; i < points.length; i++) {
                        String[] point = points[i].split(",");
                        vertices[i * 2] = Integer.parseInt(point[0]) * scaleX;
                        vertices[i * 2 + 1] = Integer.parseInt(point[1]) * scaleY;
                        if (yUp) {
                            vertices[i * 2 + 1] *= -1;
                        }
                    }
                    Polygon polygon = new Polygon(vertices);
                    polygon.setPosition(x, y);
                    object = new PolygonMapObject(polygon);
                } else if ((child = element.getChildByName("polyline")) != null) {
                    String[] points = child.getAttribute("points").split(" ");
                    float[] vertices = new float[points.length * 2];
                    for (int i = 0; i < points.length; i++) {
                        String[] point = points[i].split(",");
                        vertices[i * 2] = Integer.parseInt(point[0]) * scaleX;
                        vertices[i * 2 + 1] = Integer.parseInt(point[1]) * scaleY;
                        if (yUp) {
                            vertices[i * 2 + 1] *= -1;
                        }
                    }
                    Polyline polyline = new Polyline(vertices);
                    polyline.setPosition(x, y);
                    object = new PolylineMapObject(polyline);
                } else if ((child = element.getChildByName("ellipse")) != null) {
                    object = new EllipseMapObject(x, yUp ? y - height : y, width, height);
                }
            }
            if (object == null) {
                object = new RectangleMapObject(x, yUp ? y - height : y, width, height);
            }
            object.setName(element.getAttribute("name", null));
            String type = element.getAttribute("type", null);
            if (type != null) {
                object.getProperties().put("type", type);
            }
            int gid = element.getIntAttribute("gid", -1);
            if (gid != -1) {
                object.getProperties().put("gid", gid);
            }
            object.getProperties().put("x", x * scaleX);
            object.getProperties().put("y", (yUp ? y - height : y) * scaleY);
            object.setVisible(element.getIntAttribute("visible", 1) == 1);
            Element properties = element.getChildByName("properties");
            if (properties != null) {
                loadProperties(object.getProperties(), properties);
            }
            layer.getObjects().add(object);
        }
    }

    protected void loadProperties(MapProperties properties, Element element) {
        if (element.getName().equals("properties")) {
            for (Element property : element.getChildrenByName("property")) {
                String name = property.getAttribute("name", null);
                String value = property.getAttribute("value", null);
                if (value == null) {
                    value = property.getText();
                }
                properties.put(name, value);
            }
        }
    }

    protected Cell createTileLayerCell(boolean flipHorizontally, boolean flipVertically, boolean flipDiagonally) {
        Cell cell = new Cell();
        if (flipDiagonally) {
            if (flipHorizontally && flipVertically) {
                cell.setFlipHorizontally(true);
                cell.setRotation(yUp ? Cell.ROTATE_270 : Cell.ROTATE_90);
            } else if (flipHorizontally) {
                cell.setRotation(yUp ? Cell.ROTATE_270 : Cell.ROTATE_90);
            } else if (flipVertically) {
                cell.setRotation(yUp ? Cell.ROTATE_90 : Cell.ROTATE_270);
            } else {
                cell.setFlipVertically(true);
                cell.setRotation(yUp ? Cell.ROTATE_270 : Cell.ROTATE_90);
            }
        } else {
            cell.setFlipHorizontally(flipHorizontally);
            cell.setFlipVertically(flipVertically);
        }
        return cell;
    }

    public static FileHandle getRelativeFileHandle(FileHandle file, String path) {
        StringTokenizer tokenizer = new StringTokenizer(path, "\\/");
        FileHandle result = file.parent();
        while (tokenizer.hasMoreElements()) {
            String token = tokenizer.nextToken();
            if (token.equals(".."))
                result = result.parent();
            else {
                result = result.child(token);
            }
        }
        return result;
    }

    protected static int unsignedByteToInt(byte b) {
        return (int) b & 0xFF;
    }
}