Package jbrickbreaker.model

Source Code of jbrickbreaker.model.Game

package jbrickbreaker.model;

import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.util.*;

import jbrickbreaker.JBrickBreaker;
import jbrickbreaker.controller.*;
import jbrickbreaker.model.bonuses.Bonus;

/**
* This class represents the model of our game.
*
* Date: 12 Dec 2010 Time: 16:07:51
*
* @author Thomas Michel
*/
public class Game implements InfoManager, GameManager {
    private int score;
    private int lives;

    private Pad pad;

    /**
     * This is the list of all the balls that are in the game. It requires a
     * synchronized list so that when we have a lot of balls (more than 100 with
     * the tests we performed, on our development computers), we won't encounter
     * issues with the drawing and different concurrent accesses to the list.
     */
    public List<Ball> balls = Collections.synchronizedList(new Vector<Ball>());
    private List<Level> levels = new ArrayList<Level>(10);

    private Player player;
   
    /**
     * index (in {@link Game#levels} of the level being currently played
     */
    private int currentLevel;

    /**
     * This is the list of all the bonuses that are currently moving in the
     * game. It requires a synchronized list so that when we have a lot of
     * bonuses (more than 100 with the tests we performed, on our development
     * computers), we won't encounter issues with the drawing and different
     * concurrent accesses to the list.
     */
    private List<Bonus> movingBonus = Collections
            .synchronizedList(new Vector<Bonus>());

    private List<ScoreListener> scoreListeners = new ArrayList<ScoreListener>();
    private List<LivesListener> livesListeners = new ArrayList<LivesListener>();
    private List<BonusListener> bonusListeners = new ArrayList<BonusListener>();

    private static final int POINTS_PER_BRICK = 20;

    public boolean collision = false;

    private static final int POINTS_PER_BONUS = 50;
    private static int hitCount = 0;
    private JBrickBreaker brickbreaker;

    /**
     *
     * @author Thomas Michel
     */
    public Game(Player p, JBrickBreaker brickbreaker) {
        this.brickbreaker = brickbreaker;
        levels = new ArrayList<Level>(10);
        currentLevel = 0;
        this.player = p;
        balls = new ArrayList<Ball>(1);
        pad = Pad.getInstance();
        pad.reset();
        collision = false;
        Ball.setSpeedRatio(1.0d);

        balls.add(getNewBall());
        levels.add(new Level0());
        levels.add(new Level1());
        levels.add(new Level2());
        score = 0;
        lives = 3;
        movingBonus = new ArrayList<Bonus>();
    }

    /**
     * This method is called internally, it returns a new ball positioned at the
     * middle of the pad.
     */
    private Ball getNewBall() {
        Ball b = new Ball(pad.getX() + pad.getWidth() / 2, pad.getY()
                - pad.getHeight() - Ball.RADIUS / 2);
        return b;
    }

    public void unstuckBalls() {
        pad.setStuck(false);
    }

    @Override
    public int getScore() {
        return score;
    }

    @Override
    public void setScore(int score) {
        this.score = score;
        fireScoreEvent(new ScoreEvent(this, score));
    }

    @Override
    public int getLives() {
        return lives;
    }

    public void setPadColor(Color c) {
        pad.setColor(c);
    }

    @Override
    public void setLives(int l) {
        this.lives = l;
        fireLivesEvent(new LivesEvent(this, l));
    }

    @Override
    public void goToNextLevel() {
        currentLevel++;
        hitCount = 0;
        balls.clear();
        balls.add(getNewBall());
        pad.reset();
        player.clearBonuses();
        Ball.reset();
        movingBonus.clear();
        if (currentLevel >= levels.size()) {
            // player wins!
            brickbreaker.playerWins();
        }else
            brickbreaker.setDisplayingStartHelp(true);
    }

    public double getPadSpeedX() {
        return pad.getSpeedX() + pad.getCumulativeSpeed();
    }

    public void setPadSpeedX(double x) {
        pad.setSpeedX(x);
    }

    public void addToPadXPos(double x) {
        pad.setX(pad.getX() + x);
    }

    public double getPadXPos() {
        return pad.getX();
    }

    public void setPadXPos(double x) {
        pad.setX(x);
    }

    @Override
    public void drawElements(Graphics2D g) {
        // for (Bonus b : movingBonus)
        for (int i = 0; i < movingBonus.size(); i++) {
            Bonus b = movingBonus.get(i);
            if (b != null)
                b.draw(g);
        }
        pad.draw(g);
        // for (Ball b : balls){
        for (int i = 0; i < balls.size(); i++) {
            Ball b = balls.get(i);
            if (b != null)
                b.draw(g);
        }
        if (levels.size() > 0 && currentLevel < levels.size()) {
            List<Brick> bricks = levels.get(currentLevel).getBricks();
            for (Brick b : bricks)
                b.draw(g);
        }
    }

    @Override
    public void save() {
        throw new UnsupportedOperationException();
    }

    @Override
    public void load() {
        throw new UnsupportedOperationException();
    }

    @Override
    public void removeLIvesListener(LivesListener l) {
        livesListeners.remove(l);
    }

    @Override
    public void addLivesListener(LivesListener l) {
        if (!livesListeners.contains(l)) {
            livesListeners.add(l);
        }
    }

    @Override
    public void removeScoreListener(ScoreListener s) {
        scoreListeners.remove(s);
    }

    @Override
    public void addScoreListener(ScoreListener s) {
        if (!scoreListeners.contains(s)) {
            scoreListeners.add(s);
        }
    }

    @Override
    public void addBonusListener(BonusListener b) {
        if (!bonusListeners.contains(b)) {
            bonusListeners.add(b);
        }
    }

    @Override
    public void removeBonusListener(BonusListener b) {
        bonusListeners.remove(b);
    }

    private void fireBonusEvent(BonusEvent e) {
        for (BonusListener bL : bonusListeners) {
            bL.bonusChanged(e);
        }
    }

    private void fireLivesEvent(LivesEvent e) {
        for (LivesListener lL : livesListeners) {
            lL.livesChanged(e);
        }
    }

    private void fireScoreEvent(ScoreEvent e) {
        for (ScoreListener sL : scoreListeners) {
            sL.scoreChanged(e);
        }
    }

    @Override
    public void next() {
        collision = false;
        updateBonus();
        List<Ball> tmp = new ArrayList<Ball>(balls);
        boolean tmpChanged = false;
        boolean died = false;
        int i = 0;
        for (Ball b : balls) {
            boolean coll = checkCollisionPad(b);
            collision = coll;
            if (!coll || !pad.isStuck()) {
                b.setX(b.x + b.getSpeed().getX() * Ball.getSpeedRatio());
                b.setY(b.y + b.getSpeed().getY() * Ball.getSpeedRatio());

                // Only do this on the last ball
                pad.setStuck(player.hasStuckBonus() && i == balls.size() - 1);
            }
            checkCollisionGameBounds(b);
            checkCollisionsBrick(b);

            if ((b.getY() > pad.getY())) {
                tmp.remove(b);
                tmpChanged = true;
                if (tmp.size() == 0) {
                    movingBonus.clear();
                    pad.reset();
                    player.clearBonuses();
                    died = true;
                    setLives(lives - 1);
                }

            }
            i++;
        }
        if (tmpChanged) {
            balls.clear();
            balls.addAll(tmp);
        }
        if (died) {
            if (lives >= 0) {
                pad.reset();
                Ball.setSpeedRatio(1.0d);
                balls.add(getNewBall());
                hitCount = 0;
            } else {
                brickbreaker.gameOver();
            }
        }
        if (levels.get(currentLevel).getBricks().size() == 0)
            goToNextLevel();
    }

    private void updateBonus() {
        List<Bonus> movingTemp = new ArrayList<Bonus>(movingBonus);
        for (Bonus b : movingBonus) {
            if (pad.intersects(b)) {
                b.execute(this);
                BonusEvent bE = new BonusEvent(this, b);
                fireBonusEvent(bE);
                movingTemp.remove(b);

                setScore(score + POINTS_PER_BONUS);
            }
            b.setX(b.getX() + b.getSpeed().getX());
            b.setY(b.getY() + b.getSpeed().getY());
            if (b.getY() > pad.getY())
                movingTemp.remove(b);
        }
        movingBonus = movingTemp;
    }

    private void checkCollisionGameBounds(Ball b) {
        Point2D speed = b.getSpeed();
        double sx = Math.abs(speed.getX());
        if (b.getMinX() <= 0) {
            speed.setLocation(sx, speed.getY());
        } else if (b.getMaxX() > JBrickBreaker.GAME_WIDTH - 1)
            speed.setLocation(-sx, speed.getY());
        else if (b.getMinY() <= 0) {
            speed.setLocation(speed.getX(), Math.abs(speed.getY()));
        }
    }

    public boolean checkCollisionPad(Ball b) {
        boolean result = false;
        if (pad.intersects(b.getBounds2D())) {
            result = true;
            Point2D ballSpeed = b.getSpeed();
            Rectangle2D.Double out = new Rectangle2D.Double();
            Rectangle2D.intersect(pad.getBounds2D(), b.getBounds2D(), out);

            double ballX = b.getMinX();
            double padX = pad.getMinX();
            double sub = ballX - padX;

            double hitPercent = sub / (pad.getWidth() - b.getWidth()) - 0.5;
            double ballSpeedX = hitPercent * 5;

            /*
             *
             * To fix the issue with the balls bouncing within the pad, make
             * sure the ball speed is < 0 at this point (i.e when a collision
             * with the pad occurs).
             */
            double ballSpeedY = -Math.abs(ballSpeed.getY());
            ballSpeed.setLocation(ballSpeedX, ballSpeedY);
        }
        return result;
    }

    private void checkCollisionsBrick(Ball b) {
        Map<Brick, Double> intersections = new HashMap<Brick, Double>(4);
        for (Brick brick : levels.get(currentLevel).getBricks()) {
            if (brick.intersects(b.getBounds2D())) {
                Rectangle2D.Double r1 = brick;
                Rectangle2D r2 = b.getBounds2D();
                Rectangle2D inter = new Rectangle2D.Double();
                Rectangle2D.intersect(r1, r2, inter);
                double area = inter.getWidth() * inter.getHeight();
                intersections.put(brick, area);
            }
        }
        double max = Double.MIN_VALUE;
        Brick theBrick = null;
        for (Brick brick : intersections.keySet()) {
            double d = intersections.get(brick);
            if (d > max) {
                theBrick = brick;
                max = d;
            }
        }
        if (theBrick != null) {
            if (theBrick.getBonus() != null) {
                movingBonus.add(theBrick.getBonus());
                theBrick.getBonus().setSpeed(new Point2D.Double(0, 2));
            }
            levels.get(currentLevel).removeBrick(theBrick);
            setScore(score + POINTS_PER_BRICK);
            brickBouncing(b, theBrick);
            hitCount++;
            if (hitCount % 10 == 0 && hitCount != 0)
                Ball.setSpeedRatio(Ball.getSpeedRatio() * 1.1);
        }
    }

    private void brickBouncing(Ball b, Brick brick) {
        double baMaxX = b.getMaxX();
        double baMinX = b.getMinX();
        double baMaxY = b.getMaxY();
        double baMinY = b.getMinY();

        double brMinX = brick.getMinX();
        double brMaxX = brick.getMaxX();
        double brMinY = brick.getMinY();
        double brMaxY = brick.getMaxY();

        Point2D speed = b.getSpeed();

        Rectangle2D r2 = b.getBounds2D();
        Rectangle2D inter = new Rectangle2D.Double();
        Rectangle2D.intersect(brick, r2, inter);
        if (inter.getWidth() > inter.getHeight()) {
            // Ball from bottom
            if (baMinY <= brMaxY && baMaxY >= brMaxY) {
                speed.setLocation(speed.getX(), -speed.getY());
            }
            // Ball from top
            else if (baMaxY >= brMinY && baMinY <= brMinY) {
                speed.setLocation(speed.getX(), -speed.getY());
            }

        } else {
            // Ball from left
            if (baMaxX >= brMinX && baMinX <= brMinX) {
                speed.setLocation(-speed.getX(), speed.getY());
            }
            // Ball from right
            else if (baMinX <= brMaxX && baMaxX >= brMaxX) {
                speed.setLocation(-speed.getX(), speed.getY());
            }
        }
    }

    public void addBall(Ball b) {
        balls.add(b);
    }

    public Player getPlayer() {
        return player;
    }

    public List<Ball> getBalls() {
        return new ArrayList<Ball>(balls);
    }

    public void destroyBalls() {
        Ball b = balls.get(0);
        balls.clear();
        balls.add(b);
    }

}
TOP

Related Classes of jbrickbreaker.model.Game

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.