/***************************************************************************** * Eliot * Copyright (C) 2012 Olivier Teulière * Authors: Olivier Teulière * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA *****************************************************************************/ #include #include "config.h" #if ENABLE_NLS # include # define _(String) gettext(String) #else # define _(String) String #endif #include "topping.h" #include "dic.h" #include "tile.h" #include "settings.h" #include "rack.h" #include "results.h" #include "pldrack.h" #include "player.h" #include "turn.h" #include "cmd/topping_move_cmd.h" #include "cmd/player_move_cmd.h" #include "cmd/game_move_cmd.h" #include "encoding.h" #include "debug.h" INIT_LOGGER(game, Topping); Topping::Topping(const GameParams &iParams, const Game *iMasterGame) : Game(iParams, iMasterGame) { } void Topping::start() { ASSERT(getNPlayers(), "Cannot start a game without any player"); // Complete the rack try { const PlayedRack &newRack = helperSetRackRandom(getHistory().getCurrentRack(), true, RACK_NEW); setGameAndPlayersRack(newRack); } catch (EndGameException &e) { endGame(); return; } } void Topping::tryWord(const wstring &iWord, const wstring &iCoord, int iElapsed) { m_board.removeTestRound(); // Perform all the validity checks, and fill a move. // We don't really care if the move is valid or not. Move move; checkPlayedWord(iCoord, iWord, move); // Record the try LOG_INFO("Player " << m_currPlayer << " plays topping move after " << iElapsed << "s: " << lfw(move.toString())); Command *pCmd = new ToppingMoveCmd(m_currPlayer, move, iElapsed); accessNavigation().addAndExecute(pCmd); // Find the best score int bestScore = getTopScore(); LOG_DEBUG("Top score to be found: " << bestScore); if (bestScore < 0) { endGame(); return; } ASSERT(move.getScore() <= bestScore, "The player found better than the top"); if (move.getScore() < bestScore) { LOG_INFO("End of the game"); } else { // End the turn // FIXME recordPlayerMove(move, *m_players[m_currPlayer]); // Next turn endTurn(); } } int Topping::play(const wstring &, const wstring &) { ASSERT(false, "The play() method should not be called in topping mode"); throw GameException("The play() method should not be called in topping mode. Please use tryWord() instead."); return 0; } void Topping::recordPlayerMove(const Move &iMove, Player &ioPlayer) { // FIXME: the score of the player should not be the score of the move in topping mode LOG_INFO("Player " << ioPlayer.getId() << " plays: " << lfw(iMove.toString())); // Update the rack and the score of the current player // PlayerMoveCmd::execute() must be called before Game::helperPlayMove() // (called in this class in endTurn()). // See the big comment in game.cpp, line 96 Command *pCmd = new PlayerMoveCmd(ioPlayer, iMove); accessNavigation().addAndExecute(pCmd); } bool Topping::isFinished() const { return !canDrawRack(m_players[0]->getHistory().getCurrentRack(), true); } void Topping::endTurn() { // Play the word on the board const Move &move = m_players[m_currPlayer]->getLastMove(); Command *pCmd = new GameMoveCmd(*this, move, m_currPlayer); accessNavigation().addAndExecute(pCmd); accessNavigation().newTurn(); // Start next turn... start(); } void Topping::endGame() { LOG_INFO("End of the game"); } void Topping::addPlayer(Player *iPlayer) { ASSERT(getNPlayers() == 0, "Only one player can be added in Topping mode"); // Force the name of the player iPlayer->setName(wfl(_("Topping"))); Game::addPlayer(iPlayer); } Move Topping::getTopMove() const { BestResults results; results.search(getDic(), getBoard(), m_players[0]->getCurrentRack().getRack(), getHistory().beforeFirstRound()); ASSERT(results.size() != 0, "No top move found"); return Move(results.get(0)); } int Topping::getTopScore() const { return getTopMove().getScore(); } vector Topping::getTriedMoves() const { vector results; const vector &cmdVect = getNavigation().getCurrentTurn().findAllMatchingCmd(); BOOST_FOREACH(const ToppingMoveCmd * cmd, cmdVect) { results.push_back(cmd->getMove()); } return results; }