使用libgdx(java)的碰撞检测Tmx映射

时间:2013-08-04 04:41:22

标签: java libgdx tmx

所以我试图在我的游戏中实现碰撞检测,我在tmx文件中有一个名为Collision的图层。 LIBGDX现场教程不涉及与对象层的交互,很难弄清楚如何首先渲染地图。这就是我渲染屏幕的方式,我想学习如何获取碰撞图层,然后让我的精灵与它进行交互。

@Override
    public void render(float delta) {
        translateCamera();

        Gdx.gl.glClearColor(0, 0, 0, 1);
        Gdx.gl.glClear(GL10.GL_COLOR_BUFFER_BIT);

        camera.update();

        renderer.setView(camera);

        renderer.render(bgLayers);
        // renderer.render();

        batch.begin();

        batch.draw(playerDirect, Gdx.graphics.getWidth() / 2,
                Gdx.graphics.getHeight() / 2);

        batch.end();
        renderer.render(fgLayers);

    }

2 个答案:

答案 0 :(得分:5)

有一种方法可以使用对象图层。不要放弃希望!

此方法优于使用切片属性的一个主要优点是,您可以轻松地生成更少,更大的实体,以提高Box2d的效率。而且,更好的是,这些身体可以是你想要的任何形状!我的游戏中的样本级别现在只有三个有趣的形状(看起来更有机的)基于ChainShape的主体,而不是数十个方形体。

我在GameDev的answered the same question前几天,经过对网络丛林的严重追捕。我发现的tutorial对我来说并不是很有用,所以稍后编辑我想出了这个:

public class MapBodyBuilder {

    // The pixels per tile. If your tiles are 16x16, this is set to 16f
    private static float ppt = 0;

    public static Array<Body> buildShapes(Map map, float pixels, World world) {
        ppt = pixels;
        MapObjects objects = map.getLayers().get("Obstacles").getObjects();

        Array<Body> bodies = new Array<Body>();

        for(MapObject object : objects) {

            if (object instanceof TextureMapObject) {
                continue;
            }

            Shape shape;

            if (object instanceof RectangleMapObject) {
                shape = getRectangle((RectangleMapObject)object);
            }
            else if (object instanceof PolygonMapObject) {
                shape = getPolygon((PolygonMapObject)object);
            }
            else if (object instanceof PolylineMapObject) {
                shape = getPolyline((PolylineMapObject)object);
            }
            else if (object instanceof CircleMapObject) {
                shape = getCircle((CircleMapObject)object);
            }
            else {
                continue;
            }

            BodyDef bd = new BodyDef();
            bd.type = BodyType.StaticBody;
            Body body = world.createBody(bd);
            body.createFixture(shape, 1);

            bodies.add(body);

            shape.dispose();
        }
        return bodies;
    }

    private static PolygonShape getRectangle(RectangleMapObject rectangleObject) {
        Rectangle rectangle = rectangleObject.getRectangle();
        PolygonShape polygon = new PolygonShape();
        Vector2 size = new Vector2((rectangle.x + rectangle.width * 0.5f) / ppt,
                                   (rectangle.y + rectangle.height * 0.5f ) / ppt);
        polygon.setAsBox(rectangle.width * 0.5f / ppt,
                         rectangle.height * 0.5f / ppt,
                         size,
                         0.0f);
        return polygon;
    }

    private static CircleShape getCircle(CircleMapObject circleObject) {
        Circle circle = circleObject.getCircle();
        CircleShape circleShape = new CircleShape();
        circleShape.setRadius(circle.radius / ppt);
        circleShape.setPosition(new Vector2(circle.x / ppt, circle.y / ppt));
        return circleShape;
    }

    private static PolygonShape getPolygon(PolygonMapObject polygonObject) {
        PolygonShape polygon = new PolygonShape();
        float[] vertices = polygonObject.getPolygon().getTransformedVertices();

        float[] worldVertices = new float[vertices.length];

        for (int i = 0; i < vertices.length; ++i) {
            worldVertices[i] = vertices[i] / ppt;
        }

        polygon.set(worldVertices);
        return polygon;
    }

    private static ChainShape getPolyline(PolylineMapObject polylineObject) {
        float[] vertices = polylineObject.getPolyline().getTransformedVertices();
        Vector2[] worldVertices = new Vector2[vertices.length / 2];

        for (int i = 0; i < vertices.length / 2; ++i) {
            worldVertices[i] = new Vector2();
            worldVertices[i].x = vertices[i * 2] / ppt;
            worldVertices[i].y = vertices[i * 2 + 1] / ppt;
        }

        ChainShape chain = new ChainShape(); 
        chain.createChain(worldVertices);
        return chain;
    }
}

假设你已经设置好了你的瓷砖大小相当于你的Box2d World中的1平方米(1平方单位,如果你愿意的话),这会产生静态Body将是你在 Tiled 中绘制它们的确切位置。看到这种情况真是令人满意,请相信你。

答案 1 :(得分:3)

我建议将阻塞属性添加到实际的切片本身 - 您可以通过实际切片集上的Tiled编辑器添加切片属性。您可以在tileset上检索它们的属性。我要引用文档:

  

TiledMap包含一个或多个TiledMapTileSet实例。瓷砖   set包含许多TiledMapTile实例。有多个   瓦片的实现,例如静态瓷砖,动画瓷砖等你   也可以为特殊的

创建自己的实现      

图块层中的单元格引用这些图块。层内的单元格可以   多个图块集的参考图块。但是建议这样做   坚持每层单个图块集以减少纹理切换。

具体来说,在tileset中的tile上调用getProperties。这将检索属性 - 然后您可以与您的自定义属性进行比较,这可以告诉您特定磁贴是否被阻止 - 然后您可以继续并实现您自己的碰撞逻辑。