Прыгающий мяч в libgdx с использованием физики bullet3d

Я пытаюсь создать программу для прыгающих шариков с помощью libgdx в 3d. Я новичок, только что начал получать libgdx. Я могу обнаружить столкновение между битой и мячом, но не могу отскочить назад

Here is my  complete code

    public class Main implements ApplicationListener {
        final static short GROUND_FLAG = 1 << 8;
        final static short OBJECT_FLAG = 1 << 9;
        final static short ALL_FLAG = -1;

        class MyContactListener extends ContactListener {
            @Override
            public boolean onContactAdded(int userValue0, int partId0, int index0,
                    int userValue1, int partId1, int index1) {
                //instances.get(userValue0).moving = false;
                //instances.get(userValue1).moving = false;
                return true;
            }
        }

        static class MyMotionState extends btMotionState {
            Matrix4 transform;

            @Override
            public void getWorldTransform(Matrix4 worldTrans) {
                worldTrans.set(transform);
            }

            @Override
            public void setWorldTransform(Matrix4 worldTrans) {
                transform.set(worldTrans);
            }
        }

        static class GameObject extends ModelInstance implements Disposable {

            public final btRigidBody body;
            public boolean moving;
            public final MyMotionState motionState;

            public GameObject(Model model, String node,
                    btRigidBody.btRigidBodyConstructionInfo constructionInfo) {
                super(model, node);
                motionState = new MyMotionState();
                motionState.transform = transform;
                body = new btRigidBody(constructionInfo);
                body.setMotionState(motionState);
            }

            @Override
            public void dispose() {
                // TODO Auto-generated method stub
                body.dispose();
                motionState.dispose();
            }

            static class Constructor implements Disposable {
                public final Model model;
                public final String node;
                public final btCollisionShape shape;
                public final btRigidBody.btRigidBodyConstructionInfo constructionInfo;
                public static Vector3 inertia = new Vector3();

                public Constructor(Model model, String node,
                        btCollisionShape shape, float mass) {
                    this.model = model;
                    this.node = node;
                    this.shape = shape;
                    if (mass > 0f) {
                        shape.calculateLocalInertia(mass, inertia);
                    } else {
                        inertia.set(0, 0, 0);
                    }
                    this.constructionInfo = new btRigidBody.btRigidBodyConstructionInfo(
                            mass, null, shape, inertia);
                }

                public GameObject construct() {
                    return new GameObject(model, node, constructionInfo);
                }

                @Override
                public void dispose() {
                    shape.dispose();
                    constructionInfo.dispose();
                }
            }
        }

        PerspectiveCamera camera;
        Environment environment;
        ModelBatch modelBatch;
        Model model;
        Array<GameObject> instances;
        ArrayMap<String, GameObject.Constructor> constructors;
        btCollisionConfiguration configuration;
        btDispatcher dispatcher;
        btBroadphaseInterface broadphaseInterface;
        btDynamicsWorld dynamicWorld;
        btConstraintSolver solver;
        MyContactListener contactListener;
        float spawTimer;

        @Override
        public void create() {
            // TODO Auto-generated method stub
            Bullet.init();

            modelBatch = new ModelBatch();
            environment = new Environment();
            environment.set(new ColorAttribute(ColorAttribute.AmbientLight, 0.4f,
                    0.4f, 0.4f, 1f));
            environment.add(new DirectionalLight().set(0.8f, 0.8f, 0.8f, -1f,
                    -0.8f, -0.2f));

            camera = new PerspectiveCamera(67, Gdx.graphics.getWidth(),
                    Gdx.graphics.getHeight());
            camera.position.set(3f, 7f, 10f);
            camera.lookAt(0, 4f, 0);
            camera.near = 1f;
            camera.far = 300f;
            camera.update();

            ModelBuilder builder = new ModelBuilder();

            builder.begin();
            builder.node().id = "ground";
            builder.part("ground", GL20.GL_TRIANGLES,
                    Usage.Position | Usage.Normal,
                    new Material(ColorAttribute.createDiffuse(Color.RED))).box(5f,
                    1f, 5f);
            builder.node().id = "sphere";
            builder.part("sphere", GL20.GL_TRIANGLES,
                    Usage.Position | Usage.Normal,
                    new Material(ColorAttribute.createDiffuse(Color.GREEN)))
                    .sphere(1f, 1f, 1f, 10, 10);
            model = builder.end();
            constructors = new ArrayMap<String, Main.GameObject.Constructor>(
                    String.class, GameObject.Constructor.class);

            constructors.put("ground", new GameObject.Constructor(model, "ground",
                    new btBoxShape(new Vector3(2.5f, .5f, 2.5f)), 0f));
            constructors.put("ball", new GameObject.Constructor(model, "sphere",
                    new btSphereShape(0.5f), 3f));

            configuration = new btDefaultCollisionConfiguration();
            dispatcher = new btCollisionDispatcher(configuration);
            solver = new btSequentialImpulseConstraintSolver();
            broadphaseInterface = new btDbvtBroadphase();
            dynamicWorld = new btDiscreteDynamicsWorld(dispatcher,
                    broadphaseInterface, solver, configuration);
            dynamicWorld.setGravity(new Vector3(0, -10, 0));
            contactListener = new MyContactListener();

            instances = new Array<Main.GameObject>();
            GameObject object = constructors.get("ground").construct();
            object.body.setCollisionFlags(object.body.getCollisionFlags()
                    | btCollisionObject.CollisionFlags.CF_KINEMATIC_OBJECT);
            instances.add(object);
            dynamicWorld.addRigidBody(object.body);
            object.body.setActivationState(Collision.DISABLE_DEACTIVATION);
            createBall();

        }

        public void createBall() {
            GameObject obj = constructors.get("ball").construct();
            //obj.moving = true;
            obj.transform.setFromEulerAngles(MathUtils.random(360f),
                    MathUtils.random(360f), MathUtils.random(360f));
            obj.transform.trn(0f, 9f, 0f);
            //obj.body.proceedToTransform(obj.transform);
            obj.body.setRestitution(1.0f);
            obj.body.setFriction(1.0f);
            obj.body.setWorldTransform(obj.transform);
            obj.body.setUserValue(instances.size);
            obj.body.setCollisionFlags(obj.body.getCollisionFlags()
                    | btCollisionObject.CollisionFlags.CF_CUSTOM_MATERIAL_CALLBACK);
            instances.add(obj);
            // dynamicWorld.addCollisionObject(obj.body, OBJECT_FLAG, GROUND_FLAG);
            dynamicWorld.addRigidBody(obj.body);
        }

        @Override
        public void resize(int width, int height) {

        }

        float angle, speed = 10f;

        @Override
        public void render() {
            // TODO Auto-generated method stub
            final float delta = Gdx.graphics.getDeltaTime();

            angle = (angle + delta+speed) % 360;
            instances.get(0).transform.setTranslation(0f,MathUtils.sinDeg(angle),0f);
            //instances.get(0).body.setWorldTransform(instances.get(0).transform);
            dynamicWorld.stepSimulation(delta, 5, 1 / 60f);


            Gdx.gl.glClearColor(0.3f, 0.3f, 0.3f, 1.f);
            Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT | GL20.GL_DEPTH_BUFFER_BIT);

            modelBatch.begin(camera);
            modelBatch.render(instances, environment);
            modelBatch.end();
        }

        @Override
        public void pause() {

        }

        @Override
        public void resume() {

        }

        @Override
        public void dispose() {
            // TODO Auto-generated method stub
            for (GameObject obj : instances)
                obj.dispose();
            instances.clear();

            for (GameObject.Constructor ctor : constructors.values())
                ctor.dispose();
            constructors.clear();

            dynamicWorld.dispose();
            solver.dispose();
            broadphaseInterface.dispose();
            dispatcher.dispose();
            configuration.dispose();

            contactListener.dispose();

            modelBatch.dispose();
            model.dispose();
        }

    }
}

Вот мой обновленный код, сейчас я могу отскочить от земли, что не является моей целью, но моя цель состоит в том, чтобы отскочить от мяча, когда он ударяет по коду, применяемому для перемещения по земле, который невозможно применить к мячу. Пожалуйста помоги.

2 ответа

Решение

Вот код. Наконец мяч отскок

public class Main implements ApplicationListener {
    final static short GROUND_FLAG = 1 << 8;
    final static short OBJECT_FLAG = 1 << 9;
    final static short ALL_FLAG = -1;
    public float delta;

    class MyContactListener extends ContactListener {
        @Override
        public boolean onContactAdded(int userValue0, int partId0, int index0,
                int userValue1, int partId1, int index1) {
            // Gdx.app.log("onContact", "onContact Added");
            instances.get(1).body.setLinearVelocity(new Vector3(0, 10f, 0f));
            return true;
        }
    }

    static class MyMotionState extends btMotionState {
        Matrix4 transform;

        @Override
        public void getWorldTransform(Matrix4 worldTrans) {
            worldTrans.set(transform);
        }

        @Override
        public void setWorldTransform(Matrix4 worldTrans) {
            transform.set(worldTrans);
        }
    }

    static class GameObject extends ModelInstance implements Disposable {

        public final btRigidBody body;
        public boolean moving;
        public final MyMotionState motionState;

        public GameObject(Model model, String node,
                btRigidBody.btRigidBodyConstructionInfo constructionInfo) {
            super(model, node);
            motionState = new MyMotionState();
            motionState.transform = transform;
            body = new btRigidBody(constructionInfo);
            body.setMotionState(motionState);
        }

        @Override
        public void dispose() {
            // TODO Auto-generated method stub
            body.dispose();
            motionState.dispose();
        }

        static class Constructor implements Disposable {
            public final Model model;
            public final String node;
            public final btCollisionShape shape;
            public final btRigidBody.btRigidBodyConstructionInfo constructionInfo;
            public static Vector3 inertia = new Vector3();


            public Constructor(Model model, String node,
                    btCollisionShape shape, float mass) {
                this.model = model;
                this.node = node;
                this.shape = shape;
                if (mass > 0f) {
                    shape.calculateLocalInertia(mass, inertia);

                } else {
                    inertia.set(0, 0, 0);
                }
                this.constructionInfo = new btRigidBody.btRigidBodyConstructionInfo(
                        mass, null, shape, inertia);
            }

            public GameObject construct() {
                return new GameObject(model, node, constructionInfo);
            }

            @Override
            public void dispose() {
                shape.dispose();
                constructionInfo.dispose();
            }
        }
    }

    PerspectiveCamera camera;
    Environment environment;
    ModelBatch modelBatch;
    Model model;
    Array<GameObject> instances;
    ArrayMap<String, GameObject.Constructor> constructors;
    btCollisionConfiguration configuration;
    btDispatcher dispatcher;
    btBroadphaseInterface broadphaseInterface;
    btDynamicsWorld dynamicWorld;
    btConstraintSolver solver;
    MyContactListener contactListener;
    float spawTimer;
    GameObject obj;

    @Override
    public void create() {
        // TODO Auto-generated method stub
        Bullet.init();

        modelBatch = new ModelBatch();
        environment = new Environment();
        environment.set(new ColorAttribute(ColorAttribute.AmbientLight, 0.4f,
                0.4f, 0.4f, 1f));
        environment.add(new DirectionalLight().set(0.8f, 0.8f, 0.8f, -1f,
                -0.8f, -0.2f));

        camera = new PerspectiveCamera(67, Gdx.graphics.getWidth(),
                Gdx.graphics.getHeight());
        camera.position.set(3f, 7f, 10f);
        camera.lookAt(0, 4f, 0);
        camera.near = 1f;
        camera.far = 300f;
        camera.update();

        ModelBuilder builder = new ModelBuilder();

        builder.begin();
        builder.node().id = "ground";
        builder.part("ground", GL20.GL_TRIANGLES,
                Usage.Position | Usage.Normal,
                new Material(ColorAttribute.createDiffuse(Color.RED))).box(5f,
                1f, 5f);
        builder.node().id = "sphere";
        builder.part("sphere", GL20.GL_TRIANGLES,
                Usage.Position | Usage.Normal,
                new Material(ColorAttribute.createDiffuse(Color.GREEN)))
                .sphere(1f, 1f, 1f, 10, 10);
        model = builder.end();
        constructors = new ArrayMap<String, Main.GameObject.Constructor>(
                String.class, GameObject.Constructor.class);

        constructors.put("ground", new GameObject.Constructor(model, "ground",
                new btBoxShape(new Vector3(2.5f, .5f, 2.5f)), 0f));
        constructors.put("ball", new GameObject.Constructor(model, "sphere",
                new btSphereShape(0.5f), 0.5f));

        configuration = new btDefaultCollisionConfiguration();
        dispatcher = new btCollisionDispatcher(configuration);
        solver = new btSequentialImpulseConstraintSolver();
        broadphaseInterface = new btDbvtBroadphase();
        dynamicWorld = new btDiscreteDynamicsWorld(dispatcher,
                broadphaseInterface, solver, configuration);
        dynamicWorld.setGravity(new Vector3(0, -8f, 0));
        contactListener = new MyContactListener();

        instances = new Array<Main.GameObject>();
        GameObject object = constructors.get("ground").construct();
        object.body.setCollisionFlags(object.body.getCollisionFlags()
                | btCollisionObject.CollisionFlags.CF_KINEMATIC_OBJECT);
        instances.add(object);
        dynamicWorld.addRigidBody(object.body);
        object.body.setActivationState(Collision.DISABLE_DEACTIVATION);
        createBall();

    }

    public void createBall() {
        obj = constructors.get("ball").construct();

        obj.transform.trn(0f, 11f, 0f);
        obj.body.setWorldTransform(obj.transform);
        obj.body.setUserValue(instances.size);
        obj.body.setCollisionFlags(obj.body.getCollisionFlags()
                | btCollisionObject.CollisionFlags.CF_CUSTOM_MATERIAL_CALLBACK);
        obj.body.setFriction(1.0f);
        obj.body.setRestitution(3.0f);
        instances.add(obj);

        dynamicWorld.addRigidBody(obj.body);
    }

    @Override
    public void resize(int width, int height) {

    }

    float angle, speed = 10f, yTranslate = 0f, zAngle = 0f;

    @Override
    public void render() {
        // TODO Auto-generated method stub
        delta = Gdx.graphics.getDeltaTime();
        angle = (angle + delta + speed) % 360;

        if (Gdx.input.isTouched()) {

            if (yTranslate > -60f) {
                zAngle -= 0.2f;
            } else {
                // yTranslate = 30f;
            }
            instances.get(0).transform.setToRotation(0f, 0f, 2f, zAngle);


        } else {
            instances.get(0).transform.setToRotation(0f, 0f, 1f, 30f);
            instances.get(0).transform.setTranslation(0f, 0f, 0f);
        }

        //instances.get(0).transform.setTranslation(0f,0f,0f);
        dynamicWorld.stepSimulation(delta, 5, 1 / 60f);

        Gdx.gl.glClearColor(0.3f, 0.3f, 0.3f, 1.f);
        Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT | GL20.GL_DEPTH_BUFFER_BIT);
        modelBatch.begin(camera);
        modelBatch.render(instances, environment);
        modelBatch.end();
    }

    @Override
    public void pause() {

    }

    @Override
    public void resume() {

    }

    @Override
    public void dispose() {
        // TODO Auto-generated method stub
        for (GameObject obj : instances)
            obj.dispose();
        instances.clear();

        for (GameObject.Constructor ctor : constructors.values())
            ctor.dispose();
        constructors.clear();

        dynamicWorld.dispose();
        solver.dispose();
        broadphaseInterface.dispose();
        dispatcher.dispose();
        configuration.dispose();

        contactListener.dispose();

        modelBatch.dispose();
        model.dispose();
    }

}

Ты читаешь collision от if утверждение, что тесты против !collision и когда произойдет столкновение, дальнейшее чтение не произойдет, и столкновение останется истинным навсегда.

Что вы должны сделать, это:

  • иметь некоторые переменные для хранения координат x, y и z шара, но также хранить vX, vY и vZ для скорости шара.
  • каждый раз проверять наличие коллизий (убрать чтение значения коллизий из оператора if)
  • когда происходит столкновение, вы должны проверить, в какую сторону попал мяч, и инвертировать соответствующую переменную скорости.
  • если вы перемещаете мяч в зависимости от времени, прошедшего с последнего кадра, может случиться так, что после того, как мяч отскочил в первый раз, прошло меньше времени, и мяч переместился меньше, чем в предыдущем кадре, поэтому мяч все еще сталкивается с плоскостью (поверхностью, объектом), отскакивает снова и продолжает подпрыгивая (застрять). Чтобы решить это, вы можете:

    а) не просто перевернуть скорость мяча, но установить новую скорость в зависимости от того, с какой стороны ударили. Т.е. если мяч попадает в левую сторону, всегда устанавливайте vX в + что-то, а не просто инвертируйте его.

    б) заранее проверьте столкновение для следующего кадра (если возможно), чтобы мяч отскочил даже до того, как он войдет в поверхность.

    в) каким-то образом обеспечить постоянную скорость мяча, поэтому, если мяч войдет в поверхность и отскочит в следующем кадре, он будет находиться на втором кадре раньше и вне поверхности

Другие вопросы по тегам