Как обнаружить столкновения между объектами в LibGDX - PullRequest
0 голосов
/ 29 апреля 2020

Это мой первый пост о переполнении стека, поэтому я заранее извиняюсь, если нарушаю какие-либо правила публикации, и т. Д. c. Я работал над стрельбой по астероидам - ​​ esque и не могу понять, как заставить обнаружение столкновений работать между камнями и лазером.
Исходный код можно найти здесь . Мне пришлось внести некоторые изменения в метод обновления LevelScreen, потому что исходный код зависит от использования BlueJ IDE. Я нашел исправление в этом посте и получил столкновение, работающее между космическим кораблем и камнями.

Класс LevelScreen

    package com.mygdx.game;

import com.badlogic.gdx.Input.Keys;
import com.badlogic.gdx.scenes.scene2d.actions.Actions;

import java.util.ArrayList;

public class LevelScreen extends BaseScreen {

    private Spaceship spaceship;
    private boolean gameOver;
    private boolean rocksNeedRemoved;
    private Rock toRemove;

    private ArrayList<Rock> rocks;
    private ArrayList<Laser> lasers;

    public void initialize() {

        gameOver = false;
        toRemove = null;
        rocksNeedRemoved = false;

        BaseActor space = new BaseActor(0, 0, mainStage);
        space.loadTexture("space.png");
        space.setSize(800, 600);
        BaseActor.setWorldBounds(space);

        spaceship = new Spaceship(400, 300, mainStage);

        rocks = new ArrayList<Rock>();
        lasers = new ArrayList<Laser>();

        rocks.add(new Rock(600, 500, mainStage));
        rocks.add(new Rock(600, 300, mainStage));
        rocks.add(new Rock(600, 100, mainStage));
        rocks.add(new Rock(400, 100, mainStage));
        rocks.add(new Rock(200, 100, mainStage));
        rocks.add(new Rock(200, 300, mainStage));
        rocks.add(new Rock(200, 500, mainStage));
        rocks.add(new Rock(400, 500, mainStage));

        lasers.add(new Laser(400, 500, mainStage));


    }

    public void update(float dt) {


        //Code from book(Throws class not found error)
        /*
        for (BaseActor rockActor : BaseActor.getList(mainStage, 
            "Rock")) {

            if (rockActor.overlaps(spaceship)) {

                if (spaceship.shieldPower <= 0) {

                    Explosion boom = new Explosion(0, 0, 
                        mainStage);
                    boom.centerAtActor(spaceship);
                    spaceship.remove();
                    spaceship.setPosition(-1000, -1000);

                    BaseActor messageLose = new BaseActor(0, 0, 
                        uiStage);
                    messageLose.loadTexture("message- 
                        lose.png");
                    messageLose.centerAtPosition(400, 300);
                    messageLose.setOpacity(0);
                    messageLose.addAction(Actions.fadeIn(1));
                    gameOver = true;
                }
                else {

                    spaceship.shieldPower -= 34;
                    Explosion boom = new Explosion(0, 0, 
                        mainStage);
                    boom.centerAtActor(rockActor);
                    rockActor.remove();
                }
            }

            for (BaseActor laserActor : 
                BaseActor.getList(mainStage, "Laser")) {

                if (laserActor.overlaps(rockActor)) {

                }
                Explosion boom = new Explosion(0, 0, 
                    mainStage);
                boom.centerAtActor(rockActor);
                laserActor.remove();
                rockActor.remove();
            }
        }

        if (!gameOver && BaseActor.count(mainStage, "Rock") == 
            0) {

            BaseActor messageWin = new BaseActor(0, 0, 
                uiStage);
            messageWin.loadTexture("message-win.png");
            messageWin.centerAtPosition(400, 300);
            messageWin.setOpacity(0);
            messageWin.addAction(Actions.fadeIn(1));
            gameOver = true;
        }

    }

         */





        // loop I used to get collision working between rocks 
            and spaceship

        for (Rock each : rocks)
            if (spaceship.overlaps(each) && !each.crashed && 
                spaceship.shieldPower <= 0) {
                Explosion boom = new Explosion(0, 0, 
                    mainStage);
                Explosion boom2 = new Explosion(0, 0, 
                    mainStage);
                boom.centerAtActor(spaceship);
                boom2.centerAtActor(each);

                spaceship.remove();
                spaceship.setPosition(-1000, -1000);
                each.crashed = true;
                each.clearActions();
                each.addAction(Actions.fadeOut(1));

    each.addAction(Actions.after(Actions.removeActor()));
                rocksNeedRemoved = true;
                toRemove = each;

            } else if (spaceship.overlaps(each) && 
                  !each.crashed) {
                Explosion boom = new Explosion(0, 0, 
                    mainStage);
                boom.centerAtActor(each);
                spaceship.shieldPower -= 34;
                each.crashed = true;
                each.clearActions();
                each.addAction(Actions.fadeOut(1));

    each.addAction(Actions.after(Actions.removeActor()));
                rocksNeedRemoved = true;
                toRemove = each;
            }

        //check for collision between rocks and lasers (Not 
          working correctly)

        for (int i = rocks.size() - 1; i >= 0; i--) {
            Rock rock = rocks.get(i);
            for (int j = lasers.size() - 1; j >= 0; j--) {
                Laser laser = lasers.get(j);

    if(rock.getBounds().overlaps(laser.getBounds())) {
                    Explosion boom = new Explosion(0, 0, 
                        mainStage);
                    boom.centerAtActor(rock);
                    rock.crashed = true;
                    rock.clearActions();
                    rock.addAction(Actions.fadeOut(1));

    rock.addAction(Actions.after(Actions.removeActor()));
                    rocksNeedRemoved = true;
                    toRemove = rock;
                }
            }
        }

    }







    //override default InputProcessor method
    public boolean keyDown(int keycode) {

        if (keycode == Keys.X)
            spaceship.warp();

        if (keycode == Keys.SPACE)
            spaceship.shoot();

        return false;
    }
}

Класс Космического корабля.

    package com.mygdx.game;

import com.badlogic.gdx.scenes.scene2d.Stage;
import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.Input.Keys;
import com.badlogic.gdx.math.MathUtils;

public class Spaceship extends BaseActor {

    private Thrusters thrusters;
    private Shield shield;
    public int shieldPower;

    public Spaceship(float x, float y, Stage s) {

        super(x, y, s);

        loadTexture("spaceship.png");
        setBoundaryPolygon(8);

        setAcceleration(100);
        setMaxSpeed(100);
        setDeceleration(10);

        thrusters = new Thrusters(0, 0, s);
        addActor(thrusters);
        thrusters.setPosition(-thrusters.getWidth(), 
            getHeight() / 2 - thrusters.getHeight() / 2);

        shield = new Shield(0, 0, s);
        addActor(shield);
        shield.centerAtPosition(getWidth() / 2, getHeight() / 
            2);
        shieldPower = 100;

    }

    public void act(float dt) {

        super.act(dt);

        float degreesPerSecond = 160; //rotation speed
        if (Gdx.input.isKeyPressed(Keys.LEFT))
            rotateBy(degreesPerSecond * dt);
        if (Gdx.input.isKeyPressed(Keys.RIGHT))
            rotateBy(-degreesPerSecond * dt);

        if (Gdx.input.isKeyPressed(Keys.UP)) {
            accelerateAtAngle(getRotation());
            thrusters.setVisible(true);
        }
        else {
            thrusters.setVisible(false);
        }

        shield.setOpacity(shieldPower / 100f);
        if (shieldPower <= 0)
            shield.setVisible(false);

        applyPhysics(dt);

        wrapAroundWorld();
    }

    public void warp() {

        if(getStage() == null)
            return;

        Warp warp1 = new Warp(0, 0, this.getStage());
        warp1.centerAtActor(this);
        setPosition(MathUtils.random(800), 
            MathUtils.random(600));
        Warp warp2 = new Warp(0, 0, this.getStage());
        warp2.centerAtActor(this);
    }

    public void shoot() {

        if (getStage() == null)
            return;

        Laser laser = new Laser(0, 0, this.getStage());
        laser.centerAtActor(this);
        laser.setRotation(this.getRotation());
        laser.setMotionAngle(this.getRotation());


    }
}

Класс Лазера.

    package com.mygdx.game;

import com.badlogic.gdx.math.Rectangle;
import com.badlogic.gdx.scenes.scene2d.Stage;
import com.badlogic.gdx.scenes.scene2d.Action;
import com.badlogic.gdx.scenes.scene2d.actions.Actions;

public class Laser extends BaseActor {

    Rectangle bounds;

    public Laser(float x, float y, Stage s) {

        super(x, y, s);
        bounds = new Rectangle((int)getX(), (int)getY(), 
             (int)getWidth(), (int)getHeight());






        loadTexture("laser.png");

        addAction(Actions.delay(1));
        addAction(Actions.after(Actions.fadeOut(0.5f)));
        addAction(Actions.after(Actions.removeActor()));

        setSpeed(400);
        setMaxSpeed(400);
        setDeceleration(0);
    }

    public void act(float dt) {

        super.act(dt);
        applyPhysics(dt);
        wrapAroundWorld();
    }

    public Rectangle getBounds() {
        return bounds;
    }

    private void setXY(float pX,float pY) {
        setPosition(pX, pY);
        bounds.setX((int)pX);
        bounds.setY((int)pY);
    }

}

Класс Рока

    package com.mygdx.game;

import com.badlogic.gdx.math.Rectangle;
import com.badlogic.gdx.scenes.scene2d.Stage;
import com.badlogic.gdx.scenes.scene2d.Action;
import com.badlogic.gdx.scenes.scene2d.actions.Actions;
import com.badlogic.gdx.math.MathUtils;
import com.mygdx.game.BaseActor;

public class Rock extends BaseActor {


    public boolean crashed;
    Rectangle bounds;

    public Rock(float x, float y, Stage s) {

        super(x, y, s);

        bounds = new Rectangle((int)getX(), (int)getY(), 
            (int)getWidth(), (int)getHeight());



        loadTexture("rock.png");

        float random = MathUtils.random(30);

        addAction(Actions.forever(Actions.rotateBy(30 + random, 
            1)));

        setSpeed(50 + random);
        setMaxSpeed(50 + random);
        setDeceleration(0);
        setMotionAngle(MathUtils.random(360));

        crashed = false;
    }

    public void act(float dt) {


        super.act(dt);
        applyPhysics(dt);
        wrapAroundWorld();
    }



    public boolean isCrashed() {
        return crashed;
    }

    public void crashed() {

        crashed = true;
        clearActions();
        addAction(Actions.fadeOut(1));
        addAction(Actions.after(Actions.removeActor()));

    }
    public Rectangle getBounds() {
        return bounds;
    }

    private void setXY(float pX,float pY) {
        setPosition(pX, pY);
        bounds.setX((int)pX);
        bounds.setY((int)pY);
    }
}

Методы границы и перекрытия из класса BaseActor

public void setBoundaryPolygon(int numSides) {

        float w = getWidth();
        float h = getHeight();
        float[] vertices = new float[2 * numSides];

        for(int i = 0; i < numSides; i ++) {

            float angle = i * 6.28f / numSides;
            //x coordinate
            vertices[2 * i] = w / 2 * MathUtils.cos(angle) + w / 2;
            //y coordinate
            vertices[2 * i + 1] = h / 2 * MathUtils.sin(angle) + h / 2;

        }

        boundaryPolygon = new Polygon(vertices);
    }

    public Polygon getBoundaryPolygon() {

        boundaryPolygon.setPosition(getX(), getY());
        boundaryPolygon.setOrigin(getOriginX(), getOriginY());
        boundaryPolygon.setRotation(getRotation());
        boundaryPolygon.setScale(getScaleX(), getScaleY());
        return boundaryPolygon;
    }

    public boolean overlaps(BaseActor other) {

        Polygon poly1 = this.getBoundaryPolygon();
        Polygon poly2 = other.getBoundaryPolygon();

        //initial text to improve performance
        if(!poly1.getBoundingRectangle().overlaps(poly2.getBoundingRectangle()))
            return false;

        return Intersector.overlapConvexPolygons(poly1, poly2);
    }

Так что я думаю, что реальный вопрос будет в том, как бы я go проверил наличие столкновений между камнями ArrayList и лазерами, запущенными игрок? На данный момент я просто хочу завершить игру, даже если она не использует лучшие практики. Я попытался использовать метод, описанный здесь , и не получил никаких ошибок, но также не столкнулся между лазерами и камнем. Даже если я добавлю лазер в лазеры ArrayList. Эта последняя запись , которую я нашел, заставляет меня поверить, что мне нужно что-то вроде getAllLasers (), но я не уверен на 100%, как go узнать об этом. Будет ли проще просто изучить Box2D или Quadtree?

Я понимаю, что это сложная проблема, и хочу заранее поблагодарить вас за то, что вы нашли время, чтобы прочитать ее. Я рад предоставить любую необходимую вам информацию.

Ответы [ 2 ]

1 голос
/ 04 мая 2020

У вас уже есть Rectangle границы для обеих этих сущностей, это все, что вам нужно. Вы можете использовать Rectangle.overlaps()

for(Laser laser: lasers){
   for(Rock rock: rocks){
      if(laser.getBounds().overlaps(rock.getBounds())){
         //collided! 
      }
   }
}

Убедитесь, что вы получаете обновленный прямоугольник / границы. Добавьте эту дополнительную строку в методах Rock и Laser getBounds():

public Rectangle getBounds() {
   bounds.set(getX(),getY(),getWidth(),getHeight());
   return bounds;
}

, если ваши акторы масштабируются или поворачиваются, вы должны обновить bounds соответственно

0 голосов
/ 03 мая 2020

Я думаю, что проблема может быть в обновлении границ ваших актеров, я не могу найти, где вы обновляете это. Я написал похожую игру и меняю Bound актеров на каждом шаге обновления, и все хорошо работает в некоторых строках ...

public void update() {
...
        // changing bounds
        bounds = new Rectangle(position.x,position.y,
                actorWidth,
                actorHeight);
}

Или, если вы используете другой подход, проверьте, что ваши границы меняются во времени

...