org.destinationsol.game.SolCam.java Source code

Java tutorial

Introduction

Here is the source code for org.destinationsol.game.SolCam.java

Source

/*
 * Copyright 2015 MovingBlocks
 *
 * 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.
 */

package org.destinationsol.game;

import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.graphics.OrthographicCamera;
import com.badlogic.gdx.math.Matrix4;
import com.badlogic.gdx.math.Vector2;
import com.badlogic.gdx.math.Vector3;
import org.destinationsol.Const;
import org.destinationsol.common.SolColor;
import org.destinationsol.common.SolMath;
import org.destinationsol.game.planet.Planet;
import org.destinationsol.game.screens.MainScreen;
import org.destinationsol.game.ship.SolShip;

public class SolCam {
    private static final float VIEWPORT_HEIGHT = 5f;
    private static final float MAX_ZOOM_SPD = 5f;
    private static final float MED_ZOOM_SPD = 3f;
    private static final float ZOOM_CHG_SPD = .1f;
    private static final float MOVE_SPD = 3f;

    private static final float MAX_SHAKE = .07f;
    private static final float SHAKE_DAMP = MAX_SHAKE;
    public static final float CAM_ROT_SPD = 90f;

    private final CamRotStrategy myCamRotStrategy;
    private final OrthographicCamera myCam;
    private final Vector3 myTmpVec;

    private float myPrevHeroLife;
    private float myShake;
    private float myAngle;
    private float myZoom;
    private Vector2 myPos;

    public SolCam(float r) {
        myCamRotStrategy = new CamRotStrategy.ToPlanet();
        myCam = new OrthographicCamera(VIEWPORT_HEIGHT * r, -VIEWPORT_HEIGHT);
        myZoom = calcZoom(Const.CAM_VIEW_DIST_GROUND);
        myPos = new Vector2();
        myTmpVec = new Vector3();
    }

    public Matrix4 getMtx() {
        return myCam.combined;
    }

    public void update(SolGame game) {

        float desiredVd = Const.CAM_VIEW_DIST_GROUND;
        float life = 0;

        SolShip hero = game.getHero();
        float ts = game.getTimeStep();
        if (hero == null) {
            StarPort.Transcendent trans = game.getTranscendentHero();
            if (trans == null) {
                if (DebugOptions.DIRECT_CAM_CONTROL) {
                    applyInput(game);
                }
            } else {
                desiredVd = Const.CAM_VIEW_DIST_SPACE;
                myPos.set(trans.getPosition());
            }
        } else {
            Vector2 heroPos = hero.getHull().getBody().getWorldCenter();
            if (myZoom * VIEWPORT_HEIGHT < heroPos.dst(myPos)) {
                myPos.set(heroPos);
                game.getObjMan().resetDelays();
            } else {
                Vector2 moveDiff = SolMath.getVec(hero.getSpd());
                moveDiff.scl(ts);
                myPos.add(moveDiff);
                SolMath.free(moveDiff);
                float moveSpd = MOVE_SPD * ts;
                myPos.x = SolMath.approach(myPos.x, heroPos.x, moveSpd);
                myPos.y = SolMath.approach(myPos.y, heroPos.y, moveSpd);
            }
            life = hero.getLife();

            float spd = hero.getSpd().len();

            desiredVd = Const.CAM_VIEW_DIST_SPACE;
            Planet np = game.getPlanetMan().getNearestPlanet(myPos);
            if (np.getFullHeight() < np.getPos().dst(myPos) && MAX_ZOOM_SPD < spd) {
                desiredVd = Const.CAM_VIEW_DIST_JOURNEY;
            } else if (np.isNearGround(myPos) && spd < MED_ZOOM_SPD) {
                desiredVd = Const.CAM_VIEW_DIST_GROUND;
            }
            desiredVd += hero.getHull().config.getApproxRadius();
        }

        if (life < myPrevHeroLife) {
            float shakeDiff = .1f * MAX_SHAKE * (myPrevHeroLife - life);
            myShake = SolMath.approach(myShake, MAX_SHAKE, shakeDiff);
        } else {
            myShake = SolMath.approach(myShake, 0, SHAKE_DAMP * ts);
        }
        myPrevHeroLife = life;

        Vector2 pos = SolMath.fromAl(SolMath.rnd(180), myShake);
        pos.add(myPos);
        applyPos(pos.x, pos.y);
        SolMath.free(pos);

        float desiredAngle = myCamRotStrategy.getRotation(myPos, game);
        float rotSpd = CAM_ROT_SPD * ts;
        myAngle = SolMath.approachAngle(myAngle, desiredAngle, rotSpd);
        applyAngle();

        float desiredZoom = calcZoom(desiredVd);
        myZoom = SolMath.approach(myZoom, desiredZoom, ZOOM_CHG_SPD * ts);
        applyZoom(game.getMapDrawer());
        myCam.update();
    }

    private float calcZoom(float vd) {
        float h = vd * SolMath.sqrt(2);
        return h / VIEWPORT_HEIGHT;
    }

    private void applyZoom(MapDrawer mapDrawer) {
        if (mapDrawer.isToggled()) {
            myCam.zoom = mapDrawer.getZoom();
            return;
        }
        if (DebugOptions.ZOOM_OVERRIDE != 0) {
            myCam.zoom = DebugOptions.ZOOM_OVERRIDE;
            return;
        }
        myCam.zoom = myZoom;
    }

    private void applyPos(float posX, float posY) {
        myCam.position.set(posX, posY, 0);
    }

    private void applyInput(SolGame game) {
        MainScreen s = game.getScreens().mainScreen;
        boolean d = s.isDown();
        boolean u = s.isUp();
        boolean l = s.isLeft();
        boolean r = s.isRight();
        Vector2 v = SolMath.getVec();
        if (l != r)
            v.x = SolMath.toInt(r);
        if (d != u)
            v.y = SolMath.toInt(d);
        v.scl(MOVE_SPD * game.getTimeStep());
        SolMath.rotate(v, myAngle);
        myPos.add(v);
        SolMath.free(v);
    }

    private void applyAngle() {
        Vector2 v = SolMath.getVec(0, 1);
        SolMath.rotate(v, myAngle);
        myCam.up.set(v.x, v.y, 0); // up is actually down, fcuk!!
        SolMath.free(v);
    }

    public float getViewDist() {
        return getViewDist(myZoom);
    }

    public float getViewDist(float zoom) {
        float r = myCam.viewportWidth / myCam.viewportHeight;
        return .5f * VIEWPORT_HEIGHT * SolMath.sqrt(1 + r * r) * zoom;
    }

    /**
     * @return angle of a vector pointing right on screen
     */
    public float getAngle() {
        return myAngle;
    }

    public Vector2 getPos() {
        return myPos;
    }

    public void drawDebug(GameDrawer drawer) {
        float hOver2 = VIEWPORT_HEIGHT * myZoom / 2;
        float wOver2 = hOver2 * drawer.r;
        Vector2 dr = SolMath.getVec(wOver2, hOver2);
        SolMath.rotate(dr, myAngle);
        Vector2 dl = SolMath.getVec(-wOver2, hOver2);
        SolMath.rotate(dl, myAngle);
        Vector2 ul = SolMath.getVec(dr);
        ul.scl(-1);
        Vector2 ur = SolMath.getVec(dl);
        ur.scl(-1);
        dr.add(myPos);
        dl.add(myPos);
        ul.add(myPos);
        ur.add(myPos);

        float lw = getRealLineWidth();
        drawer.drawLine(drawer.debugWhiteTex, dr, dl, SolColor.W, lw, false);
        drawer.drawLine(drawer.debugWhiteTex, dl, ul, SolColor.W, lw, false);
        drawer.drawLine(drawer.debugWhiteTex, ul, ur, SolColor.W, lw, false);
        drawer.drawLine(drawer.debugWhiteTex, ur, dr, SolColor.W, lw, false);

        SolMath.free(dr);
        SolMath.free(dl);
        SolMath.free(ul);
        SolMath.free(ur);
    }

    public float getRealLineWidth() {
        return getViewHeight(myCam.zoom) / Gdx.graphics.getHeight();
    }

    public float getViewHeight() {
        return getViewHeight(myZoom);
    }

    public float getViewHeight(float zoom) {
        float r = -myCam.viewportHeight * zoom;
        if (r < 0)
            throw new AssertionError("negative view height");
        return r;
    }

    public float getViewWidth() {
        float r = myCam.viewportWidth * myZoom;
        if (r < 0)
            throw new AssertionError("negative view width");
        return r;
    }

    public float getRealZoom() {
        return myCam.zoom;
    }

    public void setPos(Vector2 pos) {
        myPos.set(pos);
    }

    public boolean isVisible(Vector2 pos) {
        Vector2 rp = SolMath.toRel(pos, myAngle, myPos);
        boolean res = isRelVisible(rp);
        SolMath.free(rp);
        return res;
    }

    public boolean isRelVisible(Vector2 rp) {
        float wHalf = getViewWidth() / 2;
        if (wHalf < SolMath.abs(rp.x))
            return false;
        float hHalf = getViewHeight() / 2;
        if (hHalf < SolMath.abs(rp.y))
            return false;
        return true;
    }

    public float getDebugFontSize() {
        return .04f * getRealZoom();
    }

    public void screenToWorld(Vector2 pos) {
        myTmpVec.set(pos, 0);
        myCam.unproject(myTmpVec);
        pos.x = myTmpVec.x;
        pos.y = myTmpVec.y;
    }
}