2007-05-27 12:01:53 -04:00
|
|
|
// $Id$
|
|
|
|
//
|
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
|
|
// Copyright (C) 2006 SuperTuxKart-Team
|
|
|
|
//
|
|
|
|
// 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
|
|
|
//#define DEBUG_SHOW_DRIVEPOINTS //This would place a small sphere in every point
|
|
|
|
//of the driveline, the one at the first point
|
|
|
|
//is purple, the others are yellow.
|
|
|
|
#ifdef DEBUG_SHOW_DRIVEPOINTS
|
|
|
|
#include <plib/ssgAux.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <sstream>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <ctime>
|
|
|
|
|
|
|
|
#include "world.hpp"
|
|
|
|
#include "herring_manager.hpp"
|
|
|
|
#include "projectile_manager.hpp"
|
|
|
|
#include "gui/menu_manager.hpp"
|
|
|
|
#include "loader.hpp"
|
|
|
|
#include "player_kart.hpp"
|
|
|
|
#include "auto_kart.hpp"
|
|
|
|
#include "isect.hpp"
|
|
|
|
#include "track.hpp"
|
|
|
|
#include "kart_properties_manager.hpp"
|
|
|
|
#include "track_manager.hpp"
|
|
|
|
#include "race_manager.hpp"
|
|
|
|
#include "user_config.hpp"
|
|
|
|
#include "callback_manager.hpp"
|
|
|
|
#include "history.hpp"
|
|
|
|
#include "constants.hpp"
|
|
|
|
#include "sound_manager.hpp"
|
|
|
|
#include "ssg_help.hpp"
|
|
|
|
#include "translation.hpp"
|
|
|
|
#include "highscore_manager.hpp"
|
|
|
|
#include "scene.hpp"
|
2007-09-19 06:54:27 -04:00
|
|
|
#include "camera.hpp"
|
2007-05-27 12:01:53 -04:00
|
|
|
#include "robots/default_robot.hpp"
|
2007-09-19 05:23:02 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
# include "replay_player.hpp"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
#if defined(WIN32) && !defined(__CYGWIN__)
|
|
|
|
# define snprintf _snprintf
|
|
|
|
#endif
|
|
|
|
|
|
|
|
World* world = 0;
|
|
|
|
|
|
|
|
World::World(const RaceSetup& raceSetup_) : m_race_setup(raceSetup_)
|
2007-09-19 05:23:02 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
, m_p_replay_player(NULL)
|
|
|
|
#endif
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
|
|
|
delete world;
|
|
|
|
world = this;
|
|
|
|
m_phase = START_PHASE;
|
|
|
|
|
|
|
|
m_track = NULL;
|
|
|
|
|
|
|
|
m_clock = 0.0f;
|
|
|
|
m_fastest_lap = 9999999.9f;
|
|
|
|
m_fastest_kart = 0;
|
|
|
|
|
|
|
|
|
|
|
|
// Grab the track file
|
|
|
|
try
|
|
|
|
{
|
|
|
|
m_track = track_manager->getTrack(m_race_setup.m_track) ;
|
|
|
|
}
|
|
|
|
catch(std::runtime_error)
|
|
|
|
{
|
|
|
|
char msg[MAX_ERROR_MESSAGE_LENGTH];
|
|
|
|
snprintf(msg, sizeof(msg),
|
|
|
|
"Track '%s' not found.\n",m_race_setup.m_track.c_str());
|
|
|
|
throw std::runtime_error(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start building the scene graph
|
|
|
|
m_track_branch = new ssgBranch ;
|
|
|
|
scene->add ( m_track_branch ) ;
|
|
|
|
|
|
|
|
// Create the physics
|
|
|
|
m_physics = new Physics(getGravity());
|
|
|
|
|
|
|
|
assert(m_race_setup.m_karts.size() > 0);
|
|
|
|
|
|
|
|
// Clear all callbacks, which might still be stored there from a previous race.
|
|
|
|
callback_manager->clear(CB_TRACK);
|
|
|
|
|
|
|
|
// Load the track models - this must be done before the karts so that the
|
|
|
|
// karts can be positioned properly on (and not in) the tracks.
|
|
|
|
loadTrack ( ) ;
|
|
|
|
|
2007-09-25 23:43:02 -04:00
|
|
|
m_track->createHash(m_track_branch, 1000);
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
int pos = 0;
|
|
|
|
int playerIndex = 0;
|
|
|
|
for (RaceSetup::Karts::iterator i = m_race_setup.m_karts.begin() ;
|
|
|
|
i != m_race_setup.m_karts.end() ; ++i )
|
|
|
|
{
|
2007-09-25 23:43:02 -04:00
|
|
|
sgCoord init_pos;
|
|
|
|
m_track->getStartCoords(pos, &init_pos);
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
Kart* newkart;
|
|
|
|
if(user_config->m_profile)
|
|
|
|
{
|
|
|
|
// In profile mode, load only the old kart
|
|
|
|
newkart = new DefaultRobot (kart_properties_manager->getKart("tuxkart"), pos,
|
2007-11-03 09:13:26 -04:00
|
|
|
init_pos);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (std::find(m_race_setup.m_players.begin(),
|
|
|
|
m_race_setup.m_players.end(), pos) != m_race_setup.m_players.end())
|
|
|
|
{
|
2007-09-19 06:54:27 -04:00
|
|
|
|
|
|
|
Camera *cam = scene->createCamera(m_race_setup.getNumPlayers(), playerIndex);
|
2007-05-27 12:01:53 -04:00
|
|
|
// the given position belongs to a player
|
|
|
|
newkart = new PlayerKart (kart_properties_manager->getKart(*i), pos,
|
2007-09-19 06:54:27 -04:00
|
|
|
&(user_config->m_player[playerIndex]),
|
|
|
|
init_pos, cam);
|
|
|
|
playerIndex++;
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
newkart = loadRobot(kart_properties_manager->getKart(*i), pos,
|
2007-11-03 09:13:26 -04:00
|
|
|
init_pos);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
} // if !user_config->m_profile
|
|
|
|
if(user_config->m_replay_history)
|
|
|
|
{
|
|
|
|
history->LoadKartData(newkart, pos);
|
|
|
|
}
|
|
|
|
newkart -> getModel () -> clrTraversalMaskBits(SSGTRAV_ISECT|SSGTRAV_HOT);
|
|
|
|
|
|
|
|
scene->add ( newkart -> getModel() ) ;
|
|
|
|
m_kart.push_back(newkart);
|
|
|
|
pos++;
|
|
|
|
} // for i
|
|
|
|
|
|
|
|
resetAllKarts();
|
|
|
|
m_number_collisions = new int[m_race_setup.getNumKarts()];
|
|
|
|
for(unsigned int i=0; i<m_race_setup.getNumKarts(); i++) m_number_collisions[i]=0;
|
|
|
|
|
|
|
|
#ifdef SSG_BACKFACE_COLLISIONS_SUPPORTED
|
|
|
|
//ssgSetBackFaceCollisions ( m_race_setup.mirror ) ;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Highscores::HighscoreType hst = (m_race_setup.m_mode==RaceSetup::RM_TIME_TRIAL)
|
|
|
|
? Highscores::HST_TIMETRIAL_OVERALL_TIME
|
|
|
|
: Highscores::HST_RACE_OVERALL_TIME;
|
|
|
|
|
|
|
|
m_highscores = highscore_manager->getHighscores(hst, m_kart.size(),
|
|
|
|
m_race_setup.m_difficulty,
|
|
|
|
m_track->getName(),
|
|
|
|
m_race_setup.m_num_laps);
|
|
|
|
|
|
|
|
callback_manager->initAll();
|
|
|
|
menu_manager->switchToRace();
|
|
|
|
|
|
|
|
const std::string& MUSIC_NAME= track_manager->getTrack(m_race_setup.m_track)->getMusic();
|
|
|
|
if (MUSIC_NAME.size()>0) sound_manager->playMusic(MUSIC_NAME.c_str());
|
|
|
|
|
|
|
|
if(user_config->m_profile)
|
|
|
|
{
|
|
|
|
m_ready_set_go = -1;
|
|
|
|
m_phase = RACE_PHASE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_phase = START_PHASE; // profile starts without countdown
|
|
|
|
m_ready_set_go = 3;
|
|
|
|
}
|
2007-09-19 05:23:02 -04:00
|
|
|
|
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
m_replay_recorder.initRecorder( m_race_setup.getNumKarts() );
|
2007-09-21 14:02:53 -04:00
|
|
|
|
2007-11-03 09:13:26 -04:00
|
|
|
m_p_replay_player = new ReplayPlayer;
|
2007-09-19 14:28:48 -04:00
|
|
|
if( !loadReplayHumanReadable( "test1" ) )
|
2007-11-03 09:13:26 -04:00
|
|
|
{
|
|
|
|
delete m_p_replay_player;
|
|
|
|
m_p_replay_player = NULL;
|
|
|
|
}
|
|
|
|
if( m_p_replay_player ) m_p_replay_player->showReplayAt( 0.0 );
|
2007-09-19 05:23:02 -04:00
|
|
|
#endif
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
World::~World()
|
|
|
|
{
|
2007-09-19 05:23:02 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
saveReplayHumanReadable( "test" );
|
|
|
|
#endif
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
for ( unsigned int i = 0 ; i < m_kart.size() ; i++ )
|
|
|
|
delete m_kart[i];
|
|
|
|
|
|
|
|
m_kart.clear();
|
|
|
|
projectile_manager->cleanup();
|
|
|
|
delete [] m_number_collisions;
|
|
|
|
delete m_physics;
|
|
|
|
|
|
|
|
sound_manager -> stopMusic();
|
|
|
|
|
|
|
|
sgVec3 sun_pos;
|
|
|
|
sgVec4 ambient_col, specular_col, diffuse_col;
|
|
|
|
sgSetVec3 ( sun_pos, 0.0f, 0.0f, 1.0f );
|
|
|
|
sgSetVec4 ( ambient_col , 0.2f, 0.2f, 0.2f, 1.0f );
|
|
|
|
sgSetVec4 ( specular_col, 1.0f, 1.0f, 1.0f, 1.0f );
|
|
|
|
sgSetVec4 ( diffuse_col , 1.0f, 1.0f, 1.0f, 1.0f );
|
|
|
|
|
|
|
|
ssgGetLight ( 0 ) -> setPosition ( sun_pos ) ;
|
|
|
|
ssgGetLight ( 0 ) -> setColour ( GL_AMBIENT , ambient_col ) ;
|
|
|
|
ssgGetLight ( 0 ) -> setColour ( GL_DIFFUSE , diffuse_col ) ;
|
|
|
|
ssgGetLight ( 0 ) -> setColour ( GL_SPECULAR, specular_col ) ;
|
2007-09-19 05:23:02 -04:00
|
|
|
|
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
m_replay_recorder.destroy();
|
2007-11-03 09:13:26 -04:00
|
|
|
if( m_p_replay_player )
|
|
|
|
{
|
|
|
|
m_p_replay_player->destroy();
|
|
|
|
delete m_p_replay_player;
|
|
|
|
m_p_replay_player = NULL;
|
|
|
|
}
|
2007-09-19 05:23:02 -04:00
|
|
|
#endif
|
2007-11-03 09:13:26 -04:00
|
|
|
} // ~World
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** Waits till each kart is resting on the ground
|
|
|
|
*
|
|
|
|
* Does simulation steps still all karts reach the ground, i.e. are not
|
|
|
|
* moving anymore
|
|
|
|
*/
|
|
|
|
void World::resetAllKarts()
|
|
|
|
{
|
|
|
|
#ifdef BULLET
|
|
|
|
bool all_finished=false;
|
|
|
|
for(int i=0; i<10; i++) m_physics->update(1./60.);
|
|
|
|
while(!all_finished)
|
|
|
|
{
|
2007-11-21 20:20:57 -05:00
|
|
|
m_physics->update(1.f/60.f);
|
2007-05-27 12:01:53 -04:00
|
|
|
all_finished=true;
|
|
|
|
for ( Karts::iterator i=m_kart.begin(); i!=m_kart.end(); i++)
|
|
|
|
{
|
|
|
|
if(!(*i)->isInRest())
|
|
|
|
{
|
|
|
|
all_finished=false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} // while
|
|
|
|
#endif
|
|
|
|
} // resetAllKarts
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::draw()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::update(float delta)
|
|
|
|
{
|
|
|
|
if(user_config->m_replay_history) delta=history->GetNextDelta();
|
|
|
|
|
|
|
|
checkRaceStatus();
|
2007-11-03 09:13:26 -04:00
|
|
|
// this line was before checkRaceStatus. but m_clock is set to 0.0 in
|
|
|
|
// checkRaceStatus on start, so m_clock would not be synchron and the
|
|
|
|
// first delta would not be added .. that would cause a gap in
|
|
|
|
// replay-recording
|
|
|
|
m_clock += delta;
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
// Count the number of collision in the next 'FRAMES_FOR_TRAFFIC_JAM' frames.
|
|
|
|
// If a kart has more than one hit, play 'traffic jam' noise.
|
|
|
|
static int nCount=0;
|
|
|
|
const int FRAMES_FOR_TRAFFIC_JAM=20;
|
|
|
|
nCount++;
|
|
|
|
if(nCount==FRAMES_FOR_TRAFFIC_JAM)
|
|
|
|
{
|
|
|
|
for(unsigned int i=0; i<m_race_setup.getNumKarts(); i++) m_number_collisions[i]=0;
|
|
|
|
nCount=0;
|
|
|
|
}
|
|
|
|
if( getPhase() == FINISH_PHASE )
|
|
|
|
{
|
|
|
|
// Add times to highscore list. First compute the order of karts,
|
|
|
|
// so that the timing of the fastest kart is added first (otherwise
|
|
|
|
// someone might get into the highscore list, only to be kicked out
|
|
|
|
// again by a faster kart in the same race), which might be confusing
|
|
|
|
// if we ever decide to display a message (e.g. during a race)
|
|
|
|
unsigned int *index = new unsigned int[m_kart.size()];
|
|
|
|
for (unsigned int i=0; i<m_kart.size(); i++ )
|
|
|
|
{
|
|
|
|
index[m_kart[i]->getPosition()-1] = i;
|
|
|
|
}
|
|
|
|
|
2007-06-04 23:45:43 -04:00
|
|
|
// Don't record the time for the last kart, since it didn't finish
|
|
|
|
// the race - unless it's timetrial (then there is only one kart)
|
2007-11-03 09:13:26 -04:00
|
|
|
unsigned int karts_to_enter = (m_race_setup.m_mode==RaceSetup::RM_TIME_TRIAL)
|
2007-06-04 23:45:43 -04:00
|
|
|
? m_kart.size() : m_kart.size()-1;
|
|
|
|
for(unsigned int pos=0; pos<karts_to_enter; pos++)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
|
|
|
// Only record times for player karts
|
|
|
|
if(!m_kart[index[pos]]->isPlayerKart()) continue;
|
|
|
|
|
|
|
|
PlayerKart *k = (PlayerKart*)m_kart[index[pos]];
|
|
|
|
|
|
|
|
Highscores::HighscoreType hst = (m_race_setup.m_mode==RaceSetup::RM_TIME_TRIAL)
|
|
|
|
? Highscores::HST_TIMETRIAL_OVERALL_TIME
|
|
|
|
: Highscores::HST_RACE_OVERALL_TIME;
|
|
|
|
if(m_highscores->addData(hst, m_kart.size(),
|
2007-11-03 09:13:26 -04:00
|
|
|
m_race_setup.m_difficulty,
|
|
|
|
m_track->getName(),
|
|
|
|
k->getName(),
|
|
|
|
k->getPlayer()->getName(),
|
|
|
|
k->getFinishTime(),
|
|
|
|
m_race_setup.m_num_laps)>0)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
|
|
|
highscore_manager->Save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete []index;
|
2007-06-09 21:13:39 -04:00
|
|
|
pause();
|
2007-05-27 12:01:53 -04:00
|
|
|
menu_manager->pushMenu(MENUID_RACERESULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
float inc = 0.05f;
|
|
|
|
float dt = delta;
|
|
|
|
while (dt>0.0)
|
|
|
|
{
|
|
|
|
if(dt>=inc)
|
|
|
|
{
|
|
|
|
dt-=inc;
|
|
|
|
if(user_config->m_replay_history) delta=history->GetNextDelta();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
inc=dt;
|
|
|
|
dt=0.0;
|
|
|
|
}
|
|
|
|
// The same delta is stored over and over again! This helps to use
|
|
|
|
// the same index in History:allDeltas, and the history* arrays here,
|
|
|
|
// and makes writing easier, since we had to write delta the first
|
|
|
|
// time, and then inc from then on.
|
|
|
|
if(!user_config->m_replay_history) history->StoreDelta(delta);
|
|
|
|
m_physics->update(inc);
|
|
|
|
for ( Karts::size_type i = 0 ; i < m_kart.size(); ++i)
|
|
|
|
{
|
|
|
|
m_kart[i]->update(inc) ;
|
|
|
|
}
|
|
|
|
} // while dt>0
|
|
|
|
|
|
|
|
projectile_manager->update(delta);
|
|
|
|
herring_manager->update(delta);
|
|
|
|
|
|
|
|
for ( Karts::size_type i = 0 ; i < m_kart.size(); ++i)
|
|
|
|
{
|
|
|
|
if(!m_kart[i]->raceIsFinished()) updateRacePosition((int)i);
|
|
|
|
m_kart[i]->addMessages();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Routine stuff we do even when paused */
|
|
|
|
callback_manager->update(delta);
|
|
|
|
|
|
|
|
// Check for traffic jam. The sound is played even if it's
|
|
|
|
// not a player kart - a traffic jam happens rarely anyway.
|
|
|
|
for(unsigned int i=0; i<m_race_setup.getNumKarts(); i++)
|
|
|
|
{
|
|
|
|
if(m_number_collisions[i]>1)
|
|
|
|
{
|
|
|
|
sound_manager->playSfx(SOUND_TRAFFIC_JAM);
|
|
|
|
nCount = FRAMES_FOR_TRAFFIC_JAM-1; // sets all fields to zero in next frame
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-19 05:23:02 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
2007-11-03 09:13:26 -04:00
|
|
|
// we start recording after START_PHASE, since during start-phase m_clock is incremented
|
|
|
|
// normally, but after switching to RACE_PHASE m_clock is set back to 0.0
|
|
|
|
if( m_phase != START_PHASE )
|
|
|
|
{
|
|
|
|
m_replay_recorder.pushFrame();
|
|
|
|
if( m_p_replay_player ) m_p_replay_player->showReplayAt( m_clock );
|
|
|
|
}
|
2007-09-19 05:23:02 -04:00
|
|
|
#endif
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
|
2007-09-19 05:23:02 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
bool World::saveReplayHumanReadable( std::string const &filename ) const
|
|
|
|
{
|
|
|
|
std::string path;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
path = loader->getPath( ReplayBase::REPLAY_FOLDER );
|
|
|
|
}
|
|
|
|
catch(std::runtime_error& e)
|
|
|
|
{
|
|
|
|
fprintf( stderr, _("Couldn't find replay-path: '%s'\n"), ReplayBase::REPLAY_FOLDER.c_str() );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
path += DIR_SEPARATOR + filename + ".";
|
|
|
|
path += ReplayBase::REPLAY_FILE_EXTENSION_HUMAN_READABLE;
|
|
|
|
|
|
|
|
FILE *fd = fopen( path.c_str(), "w" );
|
|
|
|
if( !fd )
|
|
|
|
{
|
|
|
|
fprintf(stderr, _("Error while opening replay file for writing '%s'\n"), path.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
int nKarts = world->getNumKarts();
|
|
|
|
const char *version = "unknown";
|
|
|
|
#ifdef VERSION
|
|
|
|
version = VERSION;
|
|
|
|
#endif
|
|
|
|
fprintf(fd, "Version: %s\n", version);
|
|
|
|
fprintf(fd, "numkarts: %d\n", m_kart.size());
|
|
|
|
fprintf(fd, "numplayers: %d\n", m_race_setup.getNumPlayers());
|
|
|
|
fprintf(fd, "difficulty: %d\n", m_race_setup.m_difficulty);
|
|
|
|
fprintf(fd, "track: %s\n", m_track->getIdent());
|
2007-09-21 14:02:53 -04:00
|
|
|
|
|
|
|
for (RaceSetup::Karts::const_iterator i = m_race_setup.m_karts.begin() ;
|
|
|
|
i != m_race_setup.m_karts.end() ; ++i )
|
2007-09-19 05:23:02 -04:00
|
|
|
{
|
2007-09-21 14:02:53 -04:00
|
|
|
fprintf(fd, "model %d: %s\n", i-m_race_setup.m_karts.begin(), (*i).c_str());
|
2007-09-19 05:23:02 -04:00
|
|
|
}
|
|
|
|
if( !m_replay_recorder.saveReplayHumanReadable( fd ) )
|
|
|
|
{
|
|
|
|
fclose( fd ); fd = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose( fd ); fd = NULL;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} // saveReplayHumanReadable
|
|
|
|
#endif // HAVE_GHOST_REPLAY
|
|
|
|
|
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
bool World::loadReplayHumanReadable( std::string const &filename )
|
|
|
|
{
|
2007-09-19 14:28:48 -04:00
|
|
|
assert( m_p_replay_player );
|
|
|
|
m_p_replay_player->destroy();
|
2007-09-19 05:23:02 -04:00
|
|
|
|
|
|
|
std::string path = ReplayBase::REPLAY_FOLDER + DIR_SEPARATOR;
|
|
|
|
path += filename + ".";
|
|
|
|
path += ReplayBase::REPLAY_FILE_EXTENSION_HUMAN_READABLE;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
path = loader->getPath(path.c_str());
|
|
|
|
}
|
|
|
|
catch(std::runtime_error& e)
|
|
|
|
{
|
|
|
|
fprintf( stderr, _("Couldn't find replay-file: '%s'\n"), path.c_str() );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *fd = fopen( path.c_str(), "r" );
|
|
|
|
if( !fd )
|
|
|
|
{
|
|
|
|
fprintf(stderr, _("Error while opening replay file for loading '%s'\n"), path.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-09-21 14:02:53 -04:00
|
|
|
bool blnRet = m_p_replay_player->loadReplayHumanReadable( fd );
|
2007-09-19 05:23:02 -04:00
|
|
|
|
|
|
|
fclose( fd ); fd = NULL;
|
|
|
|
|
|
|
|
return blnRet;
|
|
|
|
} // loadReplayHumanReadable
|
|
|
|
#endif // HAVE_GHOST_REPLAY
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::checkRaceStatus()
|
|
|
|
{
|
|
|
|
if (m_clock > 1.0 && m_ready_set_go == 0)
|
|
|
|
{
|
|
|
|
m_ready_set_go = -1;
|
|
|
|
}
|
|
|
|
else if (m_clock > 2.0 && m_ready_set_go == 1)
|
|
|
|
{
|
|
|
|
m_ready_set_go = 0;
|
|
|
|
m_phase = RACE_PHASE;
|
|
|
|
m_clock = 0.0f;
|
|
|
|
sound_manager->playSfx(SOUND_START);
|
2007-09-21 14:02:53 -04:00
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
2007-11-03 09:13:26 -04:00
|
|
|
// push positions at time 0.0 to replay-data
|
|
|
|
m_replay_recorder.pushFrame();
|
2007-09-21 14:02:53 -04:00
|
|
|
#endif
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
else if (m_clock > 1.0 && m_ready_set_go == 2)
|
|
|
|
{
|
|
|
|
sound_manager->playSfx(SOUND_PRESTART);
|
|
|
|
m_ready_set_go = 1;
|
|
|
|
}
|
|
|
|
else if (m_clock > 0.0 && m_ready_set_go == 3)
|
|
|
|
{
|
|
|
|
sound_manager->playSfx(SOUND_PRESTART);
|
|
|
|
m_ready_set_go = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*if all players have finished, or if only one kart is not finished when
|
|
|
|
not in time trial mode, the race is over. Players are the last in the
|
|
|
|
vector, so substracting the number of players finds the first player's
|
|
|
|
position.*/
|
|
|
|
int new_finished_karts = 0;
|
|
|
|
for ( Karts::size_type i = 0; i < m_kart.size(); ++i)
|
|
|
|
{
|
|
|
|
if ((m_kart[i]->getLap () >= m_race_setup.m_num_laps) && !m_kart[i]->raceIsFinished())
|
|
|
|
{
|
|
|
|
m_kart[i]->setFinishingState(m_clock);
|
|
|
|
|
|
|
|
race_manager->addKartScore((int)i, m_kart[i]->getPosition());
|
|
|
|
|
|
|
|
++new_finished_karts;
|
|
|
|
if(m_kart[i]->isPlayerKart())
|
|
|
|
{
|
|
|
|
race_manager->PlayerFinishes();
|
|
|
|
RaceGUI* m=(RaceGUI*)menu_manager->getRaceMenu();
|
|
|
|
if(m)
|
|
|
|
{
|
|
|
|
m->addMessage(m_kart[i]->getPosition()==1
|
|
|
|
? _("You won")
|
|
|
|
: _("You finished") ,
|
|
|
|
m_kart[i], 2.0f, 60);
|
|
|
|
m->addMessage( _("the race!"), m_kart[i], 2.0f, 60);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
race_manager->addFinishedKarts(new_finished_karts);
|
|
|
|
// Different ending conditions:
|
|
|
|
// 1) all players are finished -->
|
|
|
|
// wait TIME_DELAY_TILL_FINISH seconds - if no other end condition
|
|
|
|
// applies, end the game (and make up some artificial times for the
|
|
|
|
// outstanding karts).
|
|
|
|
// 2) If only one kart is playing, finish when one kart is finished.
|
|
|
|
// 3) Otherwise, wait till all karts except one is finished - the last
|
|
|
|
// kart obviously becoming the last
|
|
|
|
if(race_manager->allPlayerFinished() && m_phase == RACE_PHASE)
|
|
|
|
{
|
|
|
|
m_phase = DELAY_FINISH_PHASE;
|
|
|
|
m_finish_delay_start_time = m_clock;
|
|
|
|
}
|
|
|
|
else if(m_phase==DELAY_FINISH_PHASE &&
|
|
|
|
m_clock-m_finish_delay_start_time>TIME_DELAY_TILL_FINISH)
|
|
|
|
{
|
|
|
|
m_phase = FINISH_PHASE;
|
|
|
|
for ( Karts::size_type i = 0; i < m_kart.size(); ++i)
|
|
|
|
{
|
|
|
|
if(!m_kart[i]->raceIsFinished())
|
|
|
|
{
|
|
|
|
// FIXME: Add some tenths to have different times - a better solution
|
|
|
|
// would be to estimate the distance to go and use this to
|
|
|
|
// determine better times.
|
|
|
|
m_kart[i]->setFinishingState(m_clock+m_kart[i]->getPosition()*0.3f);
|
|
|
|
race_manager->addKartScore((int)i, m_kart[i]->getPosition());
|
|
|
|
|
|
|
|
} // if !raceIsFinished
|
|
|
|
} // for i
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(m_race_setup.getNumKarts() == 1)
|
|
|
|
{
|
|
|
|
if(race_manager->getFinishedKarts() == 1) m_phase = FINISH_PHASE;
|
|
|
|
}
|
|
|
|
else if(race_manager->getFinishedKarts() >= m_race_setup.getNumKarts() - 1)
|
|
|
|
{
|
|
|
|
m_phase = FINISH_PHASE;
|
|
|
|
for ( Karts::size_type i = 0; i < m_kart.size(); ++i)
|
|
|
|
{
|
|
|
|
if(!m_kart[i]->raceIsFinished())
|
|
|
|
{
|
|
|
|
m_kart[i]->setFinishingState(m_clock);
|
|
|
|
} // if !raceIsFinished
|
|
|
|
} // for i
|
|
|
|
} // if getFinishedKarts()>=geNumKarts()
|
|
|
|
|
|
|
|
} // checkRaceStatus
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::updateRacePosition ( int k )
|
|
|
|
{
|
|
|
|
int p = 1 ;
|
|
|
|
|
|
|
|
/* Find position of kart 'k' */
|
|
|
|
|
|
|
|
for ( Karts::size_type j = 0 ; j < m_kart.size() ; ++j )
|
|
|
|
{
|
|
|
|
if ( int(j) == k ) continue ;
|
|
|
|
|
|
|
|
// Count karts ahead of the current kart, i.e. kart that are already
|
|
|
|
// finished (the current kart k has not yet finished!!), have done more
|
|
|
|
// laps, or the same number of laps, but a greater distance.
|
|
|
|
if (m_kart[j]->raceIsFinished() ||
|
|
|
|
m_kart[j]->getLap() > m_kart[k]->getLap() ||
|
|
|
|
(m_kart[j]->getLap() == m_kart[k]->getLap() &&
|
|
|
|
m_kart[j]->getDistanceDownTrack() > m_kart[k]->getDistanceDownTrack()) )
|
|
|
|
p++ ;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_kart [ k ] -> setPosition ( p ) ;
|
|
|
|
} // updateRacePosition
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::herring_command (sgVec3 *xyz, char htype, int bNeedHeight )
|
|
|
|
{
|
|
|
|
|
|
|
|
// if only 2d coordinates are given, let the herring fall from very heigh
|
|
|
|
if(bNeedHeight) (*xyz)[2] = 1000000.0f;
|
|
|
|
|
|
|
|
// Even if 3d data are given, make sure that the herring is on the ground
|
|
|
|
(*xyz)[2] = getHeight ( m_track_branch, *xyz ) + 0.06f;
|
|
|
|
herringType type=HE_GREEN;
|
2007-09-25 23:43:02 -04:00
|
|
|
if ( htype=='Y' || htype=='y' ) { type = HE_GOLD ;}
|
2007-05-27 12:01:53 -04:00
|
|
|
if ( htype=='G' || htype=='g' ) { type = HE_GREEN ;}
|
|
|
|
if ( htype=='R' || htype=='r' ) { type = HE_RED ;}
|
|
|
|
if ( htype=='S' || htype=='s' ) { type = HE_SILVER ;}
|
|
|
|
herring_manager->newHerring(type, xyz);
|
|
|
|
} // herring_command
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::loadTrack()
|
|
|
|
{
|
|
|
|
std::string path = "data/";
|
|
|
|
path += m_track->getIdent();
|
|
|
|
path += ".loc";
|
|
|
|
path = loader->getPath(path.c_str());
|
|
|
|
|
|
|
|
// remove old herrings (from previous race), and remove old
|
|
|
|
// track specific herring models
|
|
|
|
herring_manager->cleanup();
|
|
|
|
if(m_race_setup.m_mode == RaceSetup::RM_GRAND_PRIX)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
herring_manager->loadHerringStyle(m_race_setup.getHerringStyle());
|
|
|
|
}
|
|
|
|
catch(std::runtime_error)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "The cup '%s' contains an invalid herring style '%s'.\n",
|
|
|
|
race_manager->getGrandPrix()->getName().c_str(),
|
|
|
|
race_manager->getGrandPrix()->getHerringStyle().c_str());
|
|
|
|
fprintf(stderr, "Please fix the file '%s'.\n",
|
|
|
|
race_manager->getGrandPrix()->getFilename().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
herring_manager->loadHerringStyle(m_track->getHerringStyle());
|
|
|
|
}
|
|
|
|
catch(std::runtime_error)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "The track '%s' contains an invalid herring style '%s'.\n",
|
|
|
|
m_track->getName(), m_track->getHerringStyle().c_str());
|
|
|
|
fprintf(stderr, "Please fix the file '%s'.\n",
|
|
|
|
m_track->getFilename().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *fd = fopen (path.c_str(), "r" );
|
|
|
|
if ( fd == NULL )
|
|
|
|
{
|
|
|
|
char msg[MAX_ERROR_MESSAGE_LENGTH];
|
|
|
|
snprintf(msg, sizeof(msg),"Can't open track location file '%s'.\n",
|
|
|
|
path.c_str());
|
|
|
|
throw std::runtime_error(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
char s [ 1024 ] ;
|
|
|
|
|
|
|
|
while ( fgets ( s, 1023, fd ) != NULL )
|
|
|
|
{
|
|
|
|
if ( *s == '#' || *s < ' ' )
|
|
|
|
continue ;
|
|
|
|
|
|
|
|
int need_hat = false ;
|
|
|
|
int fit_skin = false ;
|
|
|
|
char fname [ 1024 ] ;
|
|
|
|
sgCoord loc ;
|
|
|
|
sgZeroVec3 ( loc.xyz ) ;
|
|
|
|
sgZeroVec3 ( loc.hpr ) ;
|
|
|
|
|
|
|
|
char htype = '\0' ;
|
|
|
|
|
|
|
|
if ( sscanf ( s, "%cHERRING,%f,%f,%f", &htype,
|
|
|
|
&(loc.xyz[0]), &(loc.xyz[1]), &(loc.xyz[2]) ) == 4 )
|
|
|
|
{
|
|
|
|
herring_command(&loc.xyz, htype, false) ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "%cHERRING,%f,%f", &htype,
|
|
|
|
&(loc.xyz[0]), &(loc.xyz[1]) ) == 3 )
|
|
|
|
{
|
|
|
|
herring_command (&loc.xyz, htype, true) ;
|
|
|
|
}
|
|
|
|
else if ( s[0] == '\"' )
|
|
|
|
{
|
|
|
|
if ( sscanf ( s, "\"%[^\"]\",%f,%f,%f,%f,%f,%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]), &(loc.xyz[2]),
|
|
|
|
&(loc.hpr[0]), &(loc.hpr[1]), &(loc.hpr[2]) ) == 7 )
|
|
|
|
{
|
|
|
|
/* All 6 DOF specified */
|
|
|
|
need_hat = false;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,{},%f,%f,%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]),
|
|
|
|
&(loc.hpr[0]), &(loc.hpr[1]), &(loc.hpr[2])) == 6 )
|
|
|
|
{
|
|
|
|
/* All 6 DOF specified - but need height */
|
|
|
|
need_hat = true ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,%f,%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]), &(loc.xyz[2]),
|
|
|
|
&(loc.hpr[0]) ) == 5 )
|
|
|
|
{
|
|
|
|
/* No Roll/Pitch specified - assumed zero */
|
|
|
|
need_hat = false ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,{},%f,{},{}",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]),
|
|
|
|
&(loc.hpr[0]) ) == 3 )
|
|
|
|
{
|
|
|
|
/* All 6 DOF specified - but need height, roll, pitch */
|
|
|
|
need_hat = true ;
|
|
|
|
fit_skin = true ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,{},%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]),
|
|
|
|
&(loc.hpr[0]) ) == 4 )
|
|
|
|
{
|
|
|
|
/* No Roll/Pitch specified - but need height */
|
|
|
|
need_hat = true ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]),
|
|
|
|
&(loc.xyz[2]) ) == 4 )
|
|
|
|
{
|
|
|
|
/* No Heading/Roll/Pitch specified - but need height */
|
|
|
|
need_hat = false ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f,{}",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]) ) == 3 )
|
|
|
|
{
|
|
|
|
/* No Roll/Pitch specified - but need height */
|
|
|
|
need_hat = true ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\",%f,%f",
|
|
|
|
fname, &(loc.xyz[0]), &(loc.xyz[1]) ) == 3 )
|
|
|
|
{
|
|
|
|
/* No Z/Heading/Roll/Pitch specified */
|
|
|
|
need_hat = false ;
|
|
|
|
}
|
|
|
|
else if ( sscanf ( s, "\"%[^\"]\"", fname ) == 1 )
|
|
|
|
{
|
|
|
|
/* Nothing specified */
|
|
|
|
need_hat = false ;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fclose(fd);
|
|
|
|
char msg[MAX_ERROR_MESSAGE_LENGTH];
|
|
|
|
snprintf(msg, sizeof(msg), "Syntax error in '%s': %s",
|
|
|
|
path.c_str(), s);
|
|
|
|
throw std::runtime_error(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( need_hat )
|
|
|
|
{
|
|
|
|
sgVec3 nrm ;
|
|
|
|
|
|
|
|
loc.xyz[2] = 1000.0f ;
|
|
|
|
loc.xyz[2] = getHeightAndNormal ( m_track_branch, loc.xyz, nrm ) ;
|
|
|
|
|
|
|
|
if ( fit_skin )
|
|
|
|
{
|
|
|
|
float sy = sin ( -loc.hpr [ 0 ] * SG_DEGREES_TO_RADIANS ) ;
|
|
|
|
float cy = cos ( -loc.hpr [ 0 ] * SG_DEGREES_TO_RADIANS ) ;
|
|
|
|
|
|
|
|
loc.hpr[2] = SG_RADIANS_TO_DEGREES * atan2 ( nrm[0] * cy -
|
|
|
|
nrm[1] * sy, nrm[2] ) ;
|
|
|
|
loc.hpr[1] = -SG_RADIANS_TO_DEGREES * atan2 ( nrm[1] * cy +
|
|
|
|
nrm[0] * sy, nrm[2] ) ;
|
|
|
|
}
|
|
|
|
} // if need_hat
|
|
|
|
|
|
|
|
ssgEntity *obj = loader->load(fname, CB_TRACK);
|
|
|
|
createDisplayLists(obj);
|
|
|
|
ssgRangeSelector *lod = new ssgRangeSelector ;
|
|
|
|
ssgTransform *trans = new ssgTransform ( & loc ) ;
|
|
|
|
|
|
|
|
float r [ 2 ] = { -10.0f, 2000.0f } ;
|
|
|
|
|
|
|
|
lod -> addKid ( obj ) ;
|
|
|
|
trans -> addKid ( lod ) ;
|
|
|
|
m_track_branch -> addKid ( trans ) ;
|
|
|
|
lod -> setRanges ( r, 2 ) ;
|
|
|
|
if(user_config->m_track_debug)
|
|
|
|
m_track->addDebugToScene(user_config->m_track_debug);
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-08-22 17:31:07 -04:00
|
|
|
// fclose(fd);
|
|
|
|
// char msg[MAX_ERROR_MESSAGE_LENGTH];
|
|
|
|
// snprintf(msg, sizeof(msg), "Syntax error in '%s': %s",
|
|
|
|
fprintf(stderr, "Warning: Syntax error in '%s': %s",
|
2007-05-27 12:01:53 -04:00
|
|
|
path.c_str(), s);
|
2007-08-22 17:31:07 -04:00
|
|
|
// throw std::runtime_error(msg);
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
} // while fgets
|
|
|
|
|
|
|
|
fclose ( fd ) ;
|
|
|
|
#ifdef BULLET
|
|
|
|
m_physics->setTrack(m_track_branch);
|
|
|
|
#endif
|
|
|
|
} // loadTrack
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::restartRace()
|
|
|
|
{
|
|
|
|
m_ready_set_go = 3;
|
|
|
|
m_clock = 0.0f;
|
|
|
|
m_phase = START_PHASE;
|
|
|
|
|
|
|
|
for ( Karts::iterator i = m_kart.begin(); i != m_kart.end() ; ++i )
|
|
|
|
{
|
|
|
|
(*i)->reset();
|
|
|
|
}
|
|
|
|
resetAllKarts();
|
|
|
|
herring_manager->reset();
|
|
|
|
projectile_manager->cleanup();
|
|
|
|
race_manager->reset();
|
2007-09-21 15:17:48 -04:00
|
|
|
|
|
|
|
#ifdef HAVE_GHOST_REPLAY
|
|
|
|
m_replay_recorder.destroy();
|
|
|
|
m_replay_recorder.initRecorder( m_race_setup.getNumKarts() );
|
|
|
|
|
2007-11-03 09:13:26 -04:00
|
|
|
if( m_p_replay_player )
|
2007-09-21 15:17:48 -04:00
|
|
|
{
|
|
|
|
m_p_replay_player->reset();
|
|
|
|
m_p_replay_player->showReplayAt( 0.0 );
|
|
|
|
}
|
|
|
|
#endif
|
2007-05-27 12:01:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
Kart* World::loadRobot(const KartProperties *kart_properties, int position,
|
|
|
|
sgCoord init_pos)
|
|
|
|
{
|
|
|
|
Kart* currentRobot;
|
|
|
|
|
|
|
|
const int NUM_ROBOTS = 1;
|
|
|
|
|
2007-11-03 09:13:26 -04:00
|
|
|
srand((unsigned)std::time(0));
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
switch(rand() % NUM_ROBOTS)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
currentRobot = new DefaultRobot(kart_properties, position,
|
|
|
|
init_pos);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
std::cerr << "Warning: Unknown robot, using default." << std::endl;
|
|
|
|
currentRobot = new DefaultRobot(kart_properties, position,
|
|
|
|
init_pos);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return currentRobot;
|
|
|
|
}
|
|
|
|
|
2007-06-09 21:13:39 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::pause()
|
|
|
|
{
|
|
|
|
sound_manager -> pauseMusic() ;
|
|
|
|
m_phase = LIMBO_PHASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void World::unpause()
|
|
|
|
{
|
|
|
|
sound_manager -> resumeMusic() ;
|
|
|
|
m_phase = RACE_PHASE;
|
|
|
|
}
|
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
/* EOF */
|