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);
}
}