eliot/game/navigation.cpp

178 lines
4 KiB
C++

/*******************************************************************
* Eliot
* Copyright (C) 2008 Olivier Teulière
* Authors: Olivier Teulière <ipkiss @@ gmail.com>
*
* 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 <boost/foreach.hpp>
#include <iostream>
#include "navigation.h"
#include "turn_cmd.h"
#include "debug.h"
#include "encoding.h"
Navigation::Navigation()
: m_currTurn(0)
{
newTurn();
}
Navigation::~Navigation()
{
BOOST_FOREACH(Command *c, m_turnCommands)
{
delete c;
}
}
void Navigation::newTurn()
{
lastTurn();
m_turnCommands.push_back(new TurnCmd);
++m_currTurn;
}
void Navigation::addAndExecute(Command *iCmd)
{
ASSERT(isLastTurn(), "Trying to add a command to an old turn!");
ASSERT(m_currTurn >= 1, "Bug in the turns vector");
ASSERT(m_currTurn - 1 < m_turnCommands.size(), "Bug in the turns vector");
m_turnCommands[m_currTurn - 1]->addAndExecute(iCmd);
}
unsigned int Navigation::getCurrTurn() const
{
unsigned int currTurn = m_currTurn;
if (isLastTurn() && m_turnCommands.back()->isEmpty())
--currTurn;
return currTurn;
}
unsigned int Navigation::getNbTurns() const
{
unsigned int count = m_turnCommands.size();
if (m_turnCommands.back()->isEmpty())
--count;
return count;
}
bool Navigation::isFirstTurn() const
{
return m_currTurn == 1 ||
(m_currTurn == 2 && m_turnCommands[1]->isEmpty());
}
bool Navigation::isLastTurn() const
{
return m_currTurn == m_turnCommands.size();
}
void Navigation::prevTurn()
{
if (m_currTurn > 1)
{
--m_currTurn;
m_turnCommands[m_currTurn]->undo();
// Special case: when the last turn is empty, automatically
// undo the previous turn as well
if (m_currTurn + 1 == m_turnCommands.size() &&
m_turnCommands[m_currTurn]->isEmpty())
{
prevTurn();
}
}
}
void Navigation::nextTurn()
{
if (m_currTurn < m_turnCommands.size())
{
m_turnCommands[m_currTurn]->execute();
++m_currTurn;
// Special case: when the last turn is empty, automatically
// execute it
if (m_currTurn + 1 == m_turnCommands.size() &&
m_turnCommands[m_currTurn]->isEmpty())
{
nextTurn();
}
}
}
void Navigation::firstTurn()
{
while (m_currTurn > 1)
{
prevTurn();
}
}
void Navigation::lastTurn()
{
while (m_currTurn < m_turnCommands.size())
{
nextTurn();
}
}
void Navigation::clearFuture()
{
// When there is no future, don't do anything
if (isLastTurn())
return;
for (unsigned int i = m_currTurn; i < m_turnCommands.size(); ++i)
{
delete m_turnCommands[i];
}
while (m_turnCommands.size() > m_currTurn)
{
m_turnCommands.pop_back();
}
newTurn();
// Sanity check
ASSERT(isLastTurn(),
"After removing the next turns, we should be at the last turn");
}
void Navigation::print() const
{
cout << "=== Commands history ===" << endl;
cout << "Current position right after turn " << m_currTurn - 1<< endl;
int index = 0;
BOOST_FOREACH(Command *c, m_turnCommands)
{
cout << index << " " << convertToMb(c->toString()) << endl;
++index;
}
}