skip and kick idle players

This commit is contained in:
Andy Janata 2012-03-16 16:59:50 -07:00
parent 95fe597184
commit e0391b8d4e
7 changed files with 358 additions and 37 deletions

View File

@ -51,6 +51,18 @@ to, for instance, display the number of connected players.
The name you enter and your computer's IP address will <strong>always</strong> be logged when you The name you enter and your computer's IP address will <strong>always</strong> be logged when you
load the game client. Chat and gameplay may also be logged. load the game client. Chat and gameplay may also be logged.
</p> </p>
<p>Recent Changes:</p>
<ul>
<li>17 March, Midnight UTC:<ul>
<li>Initial AFK timer support added. This will skip (or kick, if there are not enough players) a
player that takes longer than 15 + 15 * PICK seconds to play, or skip a judge that takes longer
than 20 + 5 * PICK * PLAYERS seconds to select a winner. If a player is idle for two consecutive
rounds, they will be kicked from the game. All of these numbers are adjustable; if the timeouts
are too long or too short, please let me know!</li>
<li>The game host can specify the Awesome Point goal from 4 to 10.</li>
<li>The game host can specify the maximum number of players in a game from 3 to 10.</li>
</ul></li>
</ul>
<p>Known issues:</p> <p>Known issues:</p>
<ul> <ul>
<li><strong>Do not open the game more than once in the same browser.</strong> Neither instances <li><strong>Do not open the game more than once in the same browser.</strong> Neither instances
@ -80,7 +92,7 @@ to, for instance, display the number of connected players.
game state until the next round begins.</li> game state until the next round begins.</li>
<li>Reloading the page when the winning card is displayed does not display the winning card <li>Reloading the page when the winning card is displayed does not display the winning card
again.</li> again.</li>
<li>Played cards seem to blank when someone joins (or leaves?). You may have to refresh the page <li>Played cards seem to blank when someone joins or leaves. You may have to refresh the page
to see the cards again if you're the Card Czar.</li> to see the cards again if you're the Card Czar.</li>
</ul> </ul>
<p>Current limitations:</p> <p>Current limitations:</p>
@ -100,17 +112,10 @@ to, for instance, display the number of connected players.
</ul> </ul>
</li> </li>
<li>All games and the main lobby share the same chat.</li> <li>All games and the main lobby share the same chat.</li>
<li>There is no play timer to keep the game moving if one person goes idle. However, if their
browser crashes or they lose connection, they will be removed from the game after approximately 45
seconds. An AFK timer is near the top of the priority list to add.</li>
<li>The first player to 8 Awesome Points wins. This is currently hard-coded, but you will be able
to change it later.</li>
<li>You can't bet Awesome Points to play another card, and I am unsure if I will add this.</li> <li>You can't bet Awesome Points to play another card, and I am unsure if I will add this.</li>
</ul> </ul>
<p>Future enhancements:</p> <p>Future enhancements:</p>
<ul> <ul>
<li>There will be host options to limit the number of players and set the target score soon.</li>
<li>There will be a timer to keep the game moving if somebody goes AFK soon.</li>
<li>There may be an option to display who played every card.</li> <li>There may be an option to display who played every card.</li>
<li>A registration system and long-term statistics tracking may be added at some point.</li> <li>A registration system and long-term statistics tracking may be added at some point.</li>
<li>Support for custom Black and White cards will also likely be added, with a game host option to <li>Support for custom Black and White cards will also likely be added, with a game host option to

View File

@ -201,6 +201,9 @@ cah.$.LongPollEvent = function() {
}; };
cah.$.LongPollEvent.prototype.dummyForAutocomplete = undefined; cah.$.LongPollEvent.prototype.dummyForAutocomplete = undefined;
cah.$.LongPollEvent.KICKED = "k"; cah.$.LongPollEvent.KICKED = "k";
cah.$.LongPollEvent.HURRY_UP = "hu";
cah.$.LongPollEvent.KICKED_FROM_GAME_IDLE = "kfgi";
cah.$.LongPollEvent.GAME_JUDGE_SKIPPED = "gjs";
cah.$.LongPollEvent.GAME_PLAYER_LEAVE = "gpl"; cah.$.LongPollEvent.GAME_PLAYER_LEAVE = "gpl";
cah.$.LongPollEvent.NEW_PLAYER = "np"; cah.$.LongPollEvent.NEW_PLAYER = "np";
cah.$.LongPollEvent.GAME_PLAYER_JOIN = "gpj"; cah.$.LongPollEvent.GAME_PLAYER_JOIN = "gpj";
@ -208,13 +211,15 @@ cah.$.LongPollEvent.GAME_LIST_REFRESH = "glr";
cah.$.LongPollEvent.GAME_ROUND_COMPLETE = "grc"; cah.$.LongPollEvent.GAME_ROUND_COMPLETE = "grc";
cah.$.LongPollEvent.NOOP = "_"; cah.$.LongPollEvent.NOOP = "_";
cah.$.LongPollEvent.GAME_PLAYER_INFO_CHANGE = "gpic"; cah.$.LongPollEvent.GAME_PLAYER_INFO_CHANGE = "gpic";
cah.$.LongPollEvent.GAME_PLAYER_KICKED_IDLE = "gpki";
cah.$.LongPollEvent.GAME_BLACK_RESHUFFLE = "gbr"; cah.$.LongPollEvent.GAME_BLACK_RESHUFFLE = "gbr";
cah.$.LongPollEvent.GAME_WHITE_RESHUFFLE = "gwr"; cah.$.LongPollEvent.GAME_WHITE_RESHUFFLE = "gwr";
cah.$.LongPollEvent.GAME_STATE_CHANGE = "gsc"; cah.$.LongPollEvent.GAME_STATE_CHANGE = "gsc";
cah.$.LongPollEvent.GAME_OPTIONS_CHANGED = "goc"; cah.$.LongPollEvent.GAME_OPTIONS_CHANGED = "goc";
cah.$.LongPollEvent.GAME_PLAYER_SKIPPED = "gps";
cah.$.LongPollEvent.PLAYER_LEAVE = "pl"; cah.$.LongPollEvent.PLAYER_LEAVE = "pl";
cah.$.LongPollEvent.CHAT = "c";
cah.$.LongPollEvent.HAND_DEAL = "hd"; cah.$.LongPollEvent.HAND_DEAL = "hd";
cah.$.LongPollEvent.CHAT = "c";
cah.$.LongPollEvent.GAME_JUDGE_LEFT = "gjl"; cah.$.LongPollEvent.GAME_JUDGE_LEFT = "gjl";
cah.$.LongPollResponse = function() { cah.$.LongPollResponse = function() {
@ -222,8 +227,8 @@ cah.$.LongPollResponse = function() {
}; };
cah.$.LongPollResponse.prototype.dummyForAutocomplete = undefined; cah.$.LongPollResponse.prototype.dummyForAutocomplete = undefined;
cah.$.LongPollResponse.WHITE_CARDS = "wc"; cah.$.LongPollResponse.WHITE_CARDS = "wc";
cah.$.LongPollResponse.GAME_ID = "gid";
cah.$.LongPollResponse.REASON = "qr"; cah.$.LongPollResponse.REASON = "qr";
cah.$.LongPollResponse.GAME_ID = "gid";
cah.$.LongPollResponse.HAND = "h"; cah.$.LongPollResponse.HAND = "h";
cah.$.LongPollResponse.INTERMISSION = "i"; cah.$.LongPollResponse.INTERMISSION = "i";
cah.$.LongPollResponse.PLAYER_INFO = "pi"; cah.$.LongPollResponse.PLAYER_INFO = "pi";
@ -231,6 +236,7 @@ cah.$.LongPollResponse.BLACK_CARD = "bc";
cah.$.LongPollResponse.WINNING_CARD = "WC"; cah.$.LongPollResponse.WINNING_CARD = "WC";
cah.$.LongPollResponse.GAME_STATE = "gs"; cah.$.LongPollResponse.GAME_STATE = "gs";
cah.$.LongPollResponse.NICKNAME = "n"; cah.$.LongPollResponse.NICKNAME = "n";
cah.$.LongPollResponse.PLAY_TIMER = "pt";
cah.$.LongPollResponse.MESSAGE = "m"; cah.$.LongPollResponse.MESSAGE = "m";
cah.$.LongPollResponse.ERROR = "e"; cah.$.LongPollResponse.ERROR = "e";
cah.$.LongPollResponse.EVENT = "E"; cah.$.LongPollResponse.EVENT = "E";

View File

@ -749,6 +749,42 @@ cah.Game.prototype.roundComplete = function(data) {
$(".game_show_last_round", this.element_).removeAttr("disabled"); $(".game_show_last_round", this.element_).removeAttr("disabled");
}; };
/**
* Notify the user that they are running out of time to play.
*/
cah.Game.prototype.hurryUp = function() {
cah.log.status("Hurry up! You have less than 10 seconds to decide, or you will be skipped.");
};
/**
* A player was kicked due to being idle.
*
* @param {object}
* data Event data from server.
*/
cah.Game.prototype.playerKickedIdle = function(data) {
cah.log.status(data[cah.$.LongPollResponse.NICKNAME]
+ " was kicked for being idle for too many rounds.");
};
/**
* A player was skipped due to being idle.
*
* @param {obejct}
* data Event data from server.
*/
cah.Game.prototype.playerSkipped = function(data) {
cah.log.status(data[cah.$.LongPollResponse.NICKNAME]
+ " was skipped this round for being idle for too long.");
};
/**
* This player was kicked due to being idle.
*/
cah.Game.prototype.iWasKickedIdle = function() {
};
/** /**
* Notify the player that a deck has been reshuffled. * Notify the player that a deck has been reshuffled.
* *
@ -766,13 +802,21 @@ cah.Game.prototype.reshuffle = function(deck) {
* data Event data from the server. * data Event data from the server.
*/ */
cah.Game.prototype.judgeLeft = function(data) { cah.Game.prototype.judgeLeft = function(data) {
cah.log cah.log.status("The Card Czar has left the game. Cards played this round are being returned to "
.status("The judge has left the game. Cards played this round are being returned to hands."); + "hands.");
cah.log.status("The next round will begin in " cah.log.status("The next round will begin in "
+ (data[cah.$.LongPollResponse.INTERMISSION] / 1000) + " seconds."); + (data[cah.$.LongPollResponse.INTERMISSION] / 1000) + " seconds.");
cah.log.status("(Displayed state will look weird until the next round.)"); cah.log.status("(Displayed state will look weird until the next round.)");
}; };
/**
* The judge was skipped for taking too long.
*/
cah.Game.prototype.judgeSkipped = function() {
cah.log.status("The Card Czar has taken too long to decide and has been skipped. "
+ "Cards played this round are being returned to hands.");
};
/** /**
* Event handler for confirm selection button. * Event handler for confirm selection button.
* *

View File

@ -133,6 +133,37 @@ cah.longpoll.EventHandlers[cah.$.LongPollEvent.GAME_OPTIONS_CHANGED] = function(
"options changed"); "options changed");
}; };
cah.longpoll.EventHandlers[cah.$.LongPollEvent.HURRY_UP] = function(data) {
cah.longpoll.EventHandlers.__gameEvent(data, cah.Game.prototype.hurryUp, "", "hurry up");
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.GAME_PLAYER_KICKED_IDLE] = function(data) {
cah.longpoll.EventHandlers.__gameEvent(data, cah.Game.prototype.playerKickedIdle, data,
"idle kick");
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.GAME_PLAYER_SKIPPED] = function(data) {
cah.longpoll.EventHandlers.__gameEvent(data, cah.Game.prototype.playerSkipped, data,
"player skip");
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.GAME_JUDGE_SKIPPED] = function(data) {
cah.longpoll.EventHandlers.__gameEvent(data, cah.Game.prototype.judgeSkipped, "", "judge skip");
};
cah.longpoll.EventHandlers[cah.$.LongPollEvent.KICKED_FROM_GAME_IDLE] = function(data) {
var game = cah.currentGames[data[cah.$.LongPollResponse.GAME_ID]];
if (game) {
game.dispose();
delete cah.currentGames[data[cah.$.LongPollResponse.GAME_ID]];
}
cah.GameList.instance.update();
cah.GameList.instance.show();
cah.log.error("You were kicked from game " + data[cah.$.LongPollResponse.GAME_ID]
+ " for being idle for too long.");
};
/** /**
* Helper for event handlers for game events. * Helper for event handlers for game events.
* *

View File

@ -317,17 +317,22 @@ public class Constants {
CHAT(AjaxOperation.CHAT), CHAT(AjaxOperation.CHAT),
GAME_BLACK_RESHUFFLE("gbr"), GAME_BLACK_RESHUFFLE("gbr"),
GAME_JUDGE_LEFT("gjl"), GAME_JUDGE_LEFT("gjl"),
GAME_JUDGE_SKIPPED("gjs"),
GAME_LIST_REFRESH("glr"), GAME_LIST_REFRESH("glr"),
GAME_OPTIONS_CHANGED("goc"), GAME_OPTIONS_CHANGED("goc"),
GAME_PLAYER_INFO_CHANGE("gpic"), GAME_PLAYER_INFO_CHANGE("gpic"),
GAME_PLAYER_JOIN("gpj"), GAME_PLAYER_JOIN("gpj"),
GAME_PLAYER_KICKED_IDLE("gpki"),
GAME_PLAYER_LEAVE("gpl"), GAME_PLAYER_LEAVE("gpl"),
GAME_PLAYER_SKIPPED("gps"),
GAME_ROUND_COMPLETE("grc"), GAME_ROUND_COMPLETE("grc"),
GAME_STATE_CHANGE("gsc"), GAME_STATE_CHANGE("gsc"),
GAME_WHITE_RESHUFFLE("gwr"), GAME_WHITE_RESHUFFLE("gwr"),
HAND_DEAL("hd"), HAND_DEAL("hd"),
HURRY_UP("hu"),
@DuplicationAllowed @DuplicationAllowed
KICKED(DisconnectReason.KICKED), KICKED(DisconnectReason.KICKED),
KICKED_FROM_GAME_IDLE("kfgi"),
NEW_PLAYER("np"), NEW_PLAYER("np"),
/** /**
* There has been no other action to inform the client about in a certain timeframe, so inform * There has been no other action to inform the client about in a certain timeframe, so inform
@ -380,6 +385,7 @@ public class Constants {
MESSAGE(AjaxRequest.MESSAGE), MESSAGE(AjaxRequest.MESSAGE),
@DuplicationAllowed @DuplicationAllowed
NICKNAME(AjaxRequest.NICKNAME), NICKNAME(AjaxRequest.NICKNAME),
PLAY_TIMER("pt"),
@DuplicationAllowed @DuplicationAllowed
PLAYER_INFO(AjaxResponse.PLAYER_INFO), PLAYER_INFO(AjaxResponse.PLAYER_INFO),
/** /**

View File

@ -94,6 +94,30 @@ public class Game {
private int maxPlayers = 10; private int maxPlayers = 10;
private int judgeIndex = 0; private int judgeIndex = 0;
private final static int ROUND_INTERMISSION = 8 * 1000; private final static int ROUND_INTERMISSION = 8 * 1000;
/**
* Duration, in milliseconds, for the minimum timeout a player has to choose a card to play.
* Minimum 10 seconds.
*/
private final static int PLAY_TIMEOUT_BASE = 30 * 1000;
/**
* Duration, in milliseconds, for the additional timeout a player has to choose a card to play,
* for each additional card that must be played. For example, on a PICK 2 card, this amount of
* time is added to {@code PLAY_TIMEOUT_BASE}.
*/
private final static int PLAY_TIMEOUT_PER_CARD = 15 * 1000;
/**
* Duration, in milliseconds, for the minimum timeout a judge has to choose a winner.
* Minimum combined of this and 2 * {@code JUDGE_TIMEOUT_PER_CARD} is 10 seconds.
*/
private final static int JUDGE_TIMEOUT_BASE = 20 * 1000;
/**
* Duration, in milliseconds, for the additional timeout a judge has to choose a winning card,
* for each additional card that was played in the round. For example, on a PICK 2 card with
* 3 non-judge players, 6 times this value is added to {@code JUDGE_TIMEOUT_BASE}.
*/
private final static int JUDGE_TIMEOUT_PER_CARD = 5 * 1000;
private final static int MAX_SKIPS_BEFORE_KICK = 2;
private final Object judgeLock = new Object();
private Timer nextRoundTimer; private Timer nextRoundTimer;
private final Object nextRoundTimerLock = new Object(); private final Object nextRoundTimerLock = new Object();
private int scoreGoal = 8; private int scoreGoal = 8;
@ -202,14 +226,7 @@ public class Game {
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_JUDGE_LEFT.toString()); data.put(LongPollResponse.EVENT, LongPollEvent.GAME_JUDGE_LEFT.toString());
data.put(LongPollResponse.INTERMISSION, ROUND_INTERMISSION); data.put(LongPollResponse.INTERMISSION, ROUND_INTERMISSION);
broadcastToPlayers(MessageType.GAME_EVENT, data); broadcastToPlayers(MessageType.GAME_EVENT, data);
synchronized (playedCards) { returnCardsToHand();
for (final Player p : playedCards.playedPlayers()) {
p.getHand().addAll(playedCards.getCards(p));
sendCardsToPlayer(p, playedCards.getCards(p));
}
// prevent startNextRound from discarding cards
playedCards.clear();
}
// startNextRound will advance it again. // startNextRound will advance it again.
judgeIndex--; judgeIndex--;
// Can't start the next round right here. // Can't start the next round right here.
@ -252,13 +269,13 @@ public class Game {
resetState(true); resetState(true);
} else if (wasJudge) { } else if (wasJudge) {
synchronized (nextRoundTimerLock) { synchronized (nextRoundTimerLock) {
nextRoundTimer = new Timer();
final TimerTask task = new TimerTask() { final TimerTask task = new TimerTask() {
@Override @Override
public void run() { public void run() {
startNextRound(); startNextRound();
} }
}; };
nextRoundTimer = new Timer("judge-left-" + id, true);
nextRoundTimer.schedule(task, ROUND_INTERMISSION); nextRoundTimer.schedule(task, ROUND_INTERMISSION);
} }
} }
@ -266,6 +283,17 @@ public class Game {
} }
} }
private void returnCardsToHand() {
synchronized (playedCards) {
for (final Player p : playedCards.playedPlayers()) {
p.getHand().addAll(playedCards.getCards(p));
sendCardsToPlayer(p, playedCards.getCards(p));
}
// prevent startNextRound from discarding cards
playedCards.clear();
}
}
/** /**
* Broadcast a message to all players in this game. * Broadcast a message to all players in this game.
* *
@ -520,30 +548,203 @@ public class Game {
} }
} }
final int playTimer = PLAY_TIMEOUT_BASE + (PLAY_TIMEOUT_PER_CARD * (blackCard.getPick() - 1));
final HashMap<ReturnableData, Object> data = getEventMap(); final HashMap<ReturnableData, Object> data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_STATE_CHANGE.toString()); data.put(LongPollResponse.EVENT, LongPollEvent.GAME_STATE_CHANGE.toString());
data.put(LongPollResponse.BLACK_CARD, getBlackCard()); data.put(LongPollResponse.BLACK_CARD, getBlackCard());
data.put(LongPollResponse.GAME_STATE, GameState.PLAYING.toString()); data.put(LongPollResponse.GAME_STATE, GameState.PLAYING.toString());
data.put(LongPollResponse.PLAY_TIMER, playTimer);
broadcastToPlayers(MessageType.GAME_EVENT, data); broadcastToPlayers(MessageType.GAME_EVENT, data);
synchronized (nextRoundTimerLock) {
killRoundTimer();
final TimerTask task = new TimerTask() {
@Override
public void run() {
warnPlayersToPlay();
}
};
// 10 second warning
nextRoundTimer = new Timer("hurry-up-" + id, true);
nextRoundTimer.schedule(task, playTimer - 10 * 1000);
}
}
private void warnPlayersToPlay() {
// have to do this all synchronized in case they play while we're processing this
synchronized (nextRoundTimerLock) {
killRoundTimer();
synchronized (players) {
for (final Player player : players) {
if (getJudge() == player) {
continue;
}
synchronized (playedCards) {
if (!playedCards.hasPlayer(player)) {
final Map<ReturnableData, Object> data = new HashMap<ReturnableData, Object>();
data.put(LongPollResponse.EVENT, LongPollEvent.HURRY_UP.toString());
data.put(LongPollResponse.GAME_ID, this.id);
final QueuedMessage q = new QueuedMessage(MessageType.GAME_EVENT, data);
player.getUser().enqueueMessage(q);
}
}
}
}
final TimerTask task = new TimerTask() {
@Override
public void run() {
skipIdlePlayers();
}
};
// 10 seconds to finish playing
nextRoundTimer = new Timer("hurry-up-" + id, true);
nextRoundTimer.schedule(task, 10 * 1000);
}
}
private void warnJudgeToJudge() {
// have to do this all synchronized in case they play while we're processing this
synchronized (nextRoundTimerLock) {
killRoundTimer();
if (state == GameState.JUDGING) {
final Map<ReturnableData, Object> data = new HashMap<ReturnableData, Object>();
data.put(LongPollResponse.EVENT, LongPollEvent.HURRY_UP.toString());
data.put(LongPollResponse.GAME_ID, this.id);
final QueuedMessage q = new QueuedMessage(MessageType.GAME_EVENT, data);
getJudge().getUser().enqueueMessage(q);
}
final TimerTask task = new TimerTask() {
@Override
public void run() {
skipIdleJudge();
}
};
// 10 seconds to finish playing
nextRoundTimer = new Timer("hurry-up-" + id, true);
nextRoundTimer.schedule(task, 10 * 1000);
}
}
private void skipIdleJudge() {
killRoundTimer();
synchronized (judgeLock) {
if (state != GameState.JUDGING) {
return;
}
getJudge().skipped();
final HashMap<ReturnableData, Object> data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_JUDGE_SKIPPED.toString());
broadcastToPlayers(MessageType.GAME_EVENT, data);
returnCardsToHand();
startNextRound();
}
}
private void skipIdlePlayers() {
killRoundTimer();
synchronized (players) {
final List<User> playersToRemove = new ArrayList<User>();
final List<Player> playersToUpdateStatus = new ArrayList<Player>();
for (final Player player : players) {
if (getJudge() == player) {
continue;
}
synchronized (playedCards) {
if (!playedCards.hasPlayer(player)) {
player.skipped();
final HashMap<ReturnableData, Object> data = getEventMap();
if (player.getSkipCount() >= MAX_SKIPS_BEFORE_KICK || playedCards.size() < 2) {
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_PLAYER_KICKED_IDLE.toString());
data.put(LongPollResponse.NICKNAME, player.getUser().getNickname());
playersToRemove.add(player.getUser());
} else {
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_PLAYER_SKIPPED.toString());
data.put(LongPollResponse.NICKNAME, player.getUser().getNickname());
playersToUpdateStatus.add(player);
}
broadcastToPlayers(MessageType.GAME_EVENT, data);
}
}
}
for (final User user : playersToRemove) {
removePlayer(user);
final HashMap<ReturnableData, Object> data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.KICKED_FROM_GAME_IDLE.toString());
final QueuedMessage q = new QueuedMessage(MessageType.GAME_PLAYER_EVENT, data);
user.enqueueMessage(q);
}
synchronized (playedCards) {
// not sure how much of this check is actually required
if (players.size() < 3 || playedCards.size() < 2 || state != GameState.PLAYING) {
resetState(true);
} else {
judgingState();
}
}
// have to do this after we move to judging state
for (final Player player : playersToUpdateStatus) {
final HashMap<ReturnableData, Object> data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_PLAYER_INFO_CHANGE.toString());
data.put(LongPollResponse.PLAYER_INFO, getPlayerInfo(player));
broadcastToPlayers(MessageType.GAME_PLAYER_EVENT, data);
}
}
}
private void killRoundTimer() {
synchronized (nextRoundTimerLock) {
if (nextRoundTimer != null) {
nextRoundTimer.cancel();
nextRoundTimer = null;
}
}
} }
/** /**
* Move the game into the {@code JUDGING} state. * Move the game into the {@code JUDGING} state.
*/ */
private void judgingState() { private void judgingState() {
killRoundTimer();
state = GameState.JUDGING; state = GameState.JUDGING;
final int judgeTimer = JUDGE_TIMEOUT_BASE
+ (JUDGE_TIMEOUT_PER_CARD * playedCards.size() * blackCard.getPick());
HashMap<ReturnableData, Object> data = getEventMap(); HashMap<ReturnableData, Object> data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_STATE_CHANGE.toString()); data.put(LongPollResponse.EVENT, LongPollEvent.GAME_STATE_CHANGE.toString());
data.put(LongPollResponse.GAME_STATE, GameState.JUDGING.toString()); data.put(LongPollResponse.GAME_STATE, GameState.JUDGING.toString());
data.put(LongPollResponse.WHITE_CARDS, getWhiteCards()); data.put(LongPollResponse.WHITE_CARDS, getWhiteCards());
data.put(LongPollResponse.PLAY_TIMER, judgeTimer);
broadcastToPlayers(MessageType.GAME_EVENT, data); broadcastToPlayers(MessageType.GAME_EVENT, data);
data = getEventMap(); data = getEventMap();
data.put(LongPollResponse.EVENT, LongPollEvent.GAME_PLAYER_INFO_CHANGE.toString()); data.put(LongPollResponse.EVENT, LongPollEvent.GAME_PLAYER_INFO_CHANGE.toString());
data.put(LongPollResponse.PLAYER_INFO, getPlayerInfo(getJudge())); data.put(LongPollResponse.PLAYER_INFO, getPlayerInfo(getJudge()));
broadcastToPlayers(MessageType.GAME_PLAYER_EVENT, data); broadcastToPlayers(MessageType.GAME_PLAYER_EVENT, data);
synchronized (nextRoundTimerLock) {
killRoundTimer();
final TimerTask task = new TimerTask() {
@Override
public void run() {
warnJudgeToJudge();
}
};
// 10 second warning
nextRoundTimer = new Timer("hurry-up-" + id, true);
nextRoundTimer.schedule(task, judgeTimer - 10 * 1000);
}
} }
/** /**
@ -889,6 +1090,7 @@ public class Game {
public ErrorCode playCard(final User user, final int cardId) { public ErrorCode playCard(final User user, final int cardId) {
final Player player = getPlayerForUser(user); final Player player = getPlayerForUser(user);
if (player != null) { if (player != null) {
player.resetSkipCount();
if (getJudge() == player || state != GameState.PLAYING) { if (getJudge() == player || state != GameState.PLAYING) {
return ErrorCode.NOT_YOUR_TURN; return ErrorCode.NOT_YOUR_TURN;
} }
@ -941,23 +1143,27 @@ public class Game {
* @return Error code if there is an error, or null if success. * @return Error code if there is an error, or null if success.
*/ */
public ErrorCode judgeCard(final User user, final int cardId) { public ErrorCode judgeCard(final User user, final int cardId) {
final Player player = getPlayerForUser(user);
if (getJudge() != player) {
return ErrorCode.NOT_JUDGE;
} else if (state != GameState.JUDGING) {
return ErrorCode.NOT_YOUR_TURN;
}
final Player cardPlayer; final Player cardPlayer;
synchronized (playedCards) { synchronized (judgeLock) {
cardPlayer = playedCards.getPlayerForId(cardId); final Player player = getPlayerForUser(user);
} if (getJudge() != player) {
if (cardPlayer == null) { return ErrorCode.NOT_JUDGE;
return ErrorCode.INVALID_CARD; } else if (state != GameState.JUDGING) {
} return ErrorCode.NOT_YOUR_TURN;
}
cardPlayer.increaseScore(); player.resetSkipCount();
state = GameState.ROUND_OVER;
synchronized (playedCards) {
cardPlayer = playedCards.getPlayerForId(cardId);
}
if (cardPlayer == null) {
return ErrorCode.INVALID_CARD;
}
cardPlayer.increaseScore();
state = GameState.ROUND_OVER;
}
final int clientCardId = playedCards.getCards(cardPlayer).get(0).getId(); final int clientCardId = playedCards.getCards(cardPlayer).get(0).getId();
HashMap<ReturnableData, Object> data = getEventMap(); HashMap<ReturnableData, Object> data = getEventMap();
@ -978,7 +1184,7 @@ public class Game {
broadcastToPlayers(MessageType.GAME_PLAYER_EVENT, data); broadcastToPlayers(MessageType.GAME_PLAYER_EVENT, data);
synchronized (nextRoundTimerLock) { synchronized (nextRoundTimerLock) {
nextRoundTimer = new Timer(); killRoundTimer();
final TimerTask task; final TimerTask task;
// TODO win-by-x option // TODO win-by-x option
if (cardPlayer.getScore() == scoreGoal) { if (cardPlayer.getScore() == scoreGoal) {
@ -996,6 +1202,7 @@ public class Game {
} }
}; };
} }
nextRoundTimer = new Timer("round-intermission-" + id, true);
nextRoundTimer.schedule(task, ROUND_INTERMISSION); nextRoundTimer.schedule(task, ROUND_INTERMISSION);
} }

View File

@ -39,6 +39,7 @@ public class Player {
private final List<WhiteCard> hand = new LinkedList<WhiteCard>(); private final List<WhiteCard> hand = new LinkedList<WhiteCard>();
private int score = 0; private int score = 0;
private int skipCount = 0;
/** /**
* Create a new player object. * Create a new player object.
@ -78,6 +79,27 @@ public class Player {
score = 0; score = 0;
} }
/**
* Increases this player's skipped round count.
*/
public void skipped() {
skipCount++;
}
/**
* Reset this player's skipped round count to 0, because they have been back for a round.
*/
public void resetSkipCount() {
skipCount = 0;
}
/**
* @return This player's skipped round count.
*/
public int getSkipCount() {
return skipCount;
}
/** /**
* @return The backing object for the player's hand (i.e., it can be modified). * @return The backing object for the player's hand (i.e., it can be modified).
*/ */