// $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 3 // 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. #include "race_manager.hpp" #include #include "modes/world.hpp" #include "user_config.hpp" #include "stk_config.hpp" #include "challenges/unlock_manager.hpp" #include "graphics/scene.hpp" #include "karts/kart_properties_manager.hpp" #include "network/network_manager.hpp" #include "modes/standard_race.hpp" #include "modes/follow_the_leader.hpp" #include "modes/three_strikes_battle.hpp" #include "tracks/track_manager.hpp" RaceManager* race_manager= NULL; //----------------------------------------------------------------------------- World* world = NULL; World* RaceManager::getWorld() { return world; } /** Call to set the world, or call setWorld(NULL) to delete the current world. */ void RaceManager::setWorld(World* world_arg) { if(world != NULL) delete world; world = world_arg; } Track* RaceManager::getTrack() { return world->getTrack(); } Kart* RaceManager::getPlayerKart(const unsigned int n) { return world->getPlayerKart(n); } Kart* RaceManager::getKart(const unsigned int n) { return world->getKart(n); } //----------------------------------------------------------------------------- /** Constructs the race manager. */ RaceManager::RaceManager() { m_num_karts = user_config->getDefaultNumKarts(); m_difficulty = RD_HARD; m_major_mode = MAJOR_MODE_SINGLE; m_minor_mode = MINOR_MODE_QUICK_RACE; m_track_number = 0; m_active_race = false; m_score_for_position = stk_config->m_scores; m_coin_target = 0; setTrack("jungle"); setNumLocalPlayers(1); setLocalKartInfo(0, "tux"); } // RaceManager //----------------------------------------------------------------------------- /** Destroys the race manager. */ RaceManager::~RaceManager() { } // ~RaceManager //----------------------------------------------------------------------------- /** Resets the race manager. It is called by world when restarting a race. */ void RaceManager::reset() { m_num_finished_karts = 0; m_num_finished_players = 0; } // reset //----------------------------------------------------------------------------- /** Sets a player kart (local and non-local). * \param player_id Id of the player. * \param ki Kart info structure for this player. */ void RaceManager::setPlayerKart(unsigned int player_id, const RemoteKartInfo& ki) { m_player_karts[player_id] = ki; } // setPlayerKart // ---------------------------------------------------------------------------- /** Stores the information which local players uses which karts. * \param player_id Id of the local player for which the kart is set. * \param kart Kart name this player is using. */ void RaceManager::setLocalKartInfo(unsigned int player_id, const std::string& kart) { assert(0<=player_id && player_id m_player[player_id].getName(), network_manager->getMyHostId()); } // setLocalKartInfo //----------------------------------------------------------------------------- /** Sets the number of local players playing on this computer (including * split screen). * \param n Number of local players. */ void RaceManager::setNumLocalPlayers(unsigned int n) { m_local_kart_info.resize(n); } // setNumLocalPlayers //----------------------------------------------------------------------------- /** Sets the number of players. * \param num Number of players. */ void RaceManager::setNumPlayers(int num) { m_player_karts.resize(num); } // setNumPlayers //----------------------------------------------------------------------------- /** Sets the difficulty. * \param diff Difficulty. */ void RaceManager::setDifficulty(Difficulty diff) { m_difficulty = diff; } // setDifficulty //----------------------------------------------------------------------------- /** In case of non GP mode set the track to use. * \param track Pointer to the track to use. */ void RaceManager::setTrack(const std::string& track) { m_tracks.clear(); m_tracks.push_back(track); } // setTrack //----------------------------------------------------------------------------- /** Computes the list of random karts to be used for the AI. */ void RaceManager::computeRandomKartList() { int n = m_num_karts - m_player_karts.size(); // If less kart selected than there are player karts, adjust the number of // karts to the minimum if(n<0) { m_num_karts -= n; n = 0; } m_random_kart_list=kart_properties_manager->getRandomKartList(n, m_player_karts); } // computeRandomKartList //----------------------------------------------------------------------------- /** Starts a new race or GP (or other mode). It sets up the list of player * karts, AI karts, laps etc., and then uses startNextRace to actually start * the race. */ void RaceManager::startNew() { if(m_major_mode==MAJOR_MODE_GRAND_PRIX) // GP: get tracks and laps from grand prix { m_tracks = m_grand_prix.getTracks(); m_num_laps = m_grand_prix.getLaps(); } assert(m_player_karts.size() > 0); // command line parameters: negative numbers=all karts if(m_num_karts < 0 ) m_num_karts = stk_config->m_max_karts; if((size_t)m_num_karts < m_player_karts.size()) m_num_karts = (int)m_player_karts.size(); // Create the kart status data structure to keep track of scores, times, ... // ========================================================================== m_kart_status.clear(); // First add the AI karts (randomly chosen) // ---------------------------------------- for(unsigned int i=0; i=0; i--) { KartType kt=(m_player_karts[i].getHostId()==network_manager->getMyHostId()) ? KT_PLAYER : KT_NETWORK_PLAYER; m_kart_status.push_back(KartStatus(m_player_karts[i].getKartName(), i, m_player_karts[i].getLocalPlayerId(), m_player_karts[i].getGlobalPlayerId(), kt ) ); } // Then start the race with the first track // ======================================== m_track_number = 0; startNextRace(); } // startNew //----------------------------------------------------------------------------- /** Starts the next (or first) race. It sorts the kart status data structure * according to the number of points, and then creates the world(). */ void RaceManager::startNextRace() { m_num_finished_karts = 0; m_num_finished_players = 0; // if subsequent race, sort kart status structure // ============================================== if (m_track_number > 0) { // In follow the leader mode do not change the first kart, // since it's always the leader. int offset = (m_minor_mode==MINOR_MODE_FOLLOW_LEADER) ? 1 : 0; std::sort(m_kart_status.begin()+offset, m_kart_status.end()); //reverse kart order if flagged in stk_config if (stk_config->m_grid_order) { std::reverse(m_kart_status.begin()+offset, m_kart_status.end()); } } // not first race // the constructor assigns this object to the global // variable world. Admittedly a bit ugly, but simplifies // handling of objects which get created in the constructor // and need world to be defined. if(m_minor_mode==MINOR_MODE_FOLLOW_LEADER) new FollowTheLeaderRace(); else if(m_minor_mode==MINOR_MODE_QUICK_RACE || m_minor_mode==MINOR_MODE_TIME_TRIAL) new StandardRace(); else if(m_minor_mode==MINOR_MODE_3_STRIKES) new ThreeStrikesBattle(); else{ fprintf(stderr,"Could not create given race mode\n"); assert(0); } m_active_race = true; } // startNextRace //----------------------------------------------------------------------------- /** If there are more races to do, it starts the next race, otherwise it * calls exitRace to finish the race. */ void RaceManager::next() { m_num_finished_karts = 0; m_num_finished_players = 0; m_track_number++; if(m_track_number<(int)m_tracks.size()) { if(network_manager->getMode()==NetworkManager::NW_SERVER) network_manager->beginReadySetGoBarrier(); else network_manager->setState(NetworkManager::NS_WAIT_FOR_RACE_DATA); stk_scene->clear(); startNextRace(); } else { // Back to main menu. Change the state of the state of the // network manager. if(network_manager->getMode()==NetworkManager::NW_SERVER) network_manager->setState(NetworkManager::NS_MAIN_MENU); else network_manager->setState(NetworkManager::NS_WAIT_FOR_AVAILABLE_CHARACTERS); exitRace(); } } // next //----------------------------------------------------------------------------- /** In GP displays the GP result screen, and then deletes the world. */ void RaceManager::exitRace() { // Only display the grand prix result screen if all tracks // were finished, and not when a race is aborted. if(m_major_mode==MAJOR_MODE_GRAND_PRIX && m_track_number==(int)m_tracks.size()) { // calculate the rank of each kart const unsigned int NUM_KARTS = race_manager->getNumKarts(); int *scores = new int[NUM_KARTS]; int *position = new int[NUM_KARTS]; double *race_time = new double[NUM_KARTS]; // Ignore the first kart if it's a follow-the-leader race. int start=(race_manager->getMinorMode()==RaceManager::MINOR_MODE_FOLLOW_LEADER) ? 1 : 0; for( unsigned int kart_id = start; kart_id < NUM_KARTS; ++kart_id ) { position[kart_id] = kart_id; scores[kart_id] = race_manager->getKartScore(kart_id); race_time[kart_id] = race_manager->getOverallTime(kart_id); } if(race_manager->getMinorMode()==RaceManager::MINOR_MODE_FOLLOW_LEADER) { // fill values for leader position[0] = -1; scores[0] = -1; race_time[0] = -1; m_kart_status[0].m_gp_final_rank = -1; } //Bubblesort bool sorted; do { sorted = true; for( unsigned int i = start; i < NUM_KARTS - 1; ++i ) { if( scores[i] < scores[i+1] || (scores[i] == scores[i+1] && race_time[i] > race_time[i+1])) { int tmp_score[2]; double tmp_time; tmp_score[0] = position[i]; tmp_score[1] = scores[i]; tmp_time = race_time[i]; position[i] = position[i+1]; scores[i] = scores[i+1]; race_time[i] = race_time[i+1]; position[i+1] = tmp_score[0]; scores[i+1] = tmp_score[1]; race_time[i+1] = tmp_time; sorted = false; } } } while(!sorted); for(unsigned int i=start; i < NUM_KARTS; ++i) { //printf("setting kart %s to rank %i\n", race_manager->getKartName(position[i]).c_str(), i-start); m_kart_status[position[i]].m_gp_final_rank = i - start; } // printf("kart %s has rank %i\n", 0, m_kart_status[0].m_gp_final_rank); delete []scores; delete []position; delete []race_time; unlock_manager->grandPrixFinished(); // TODO - Grand Prix ending // menu_manager->switchToGrandPrixEnding(); } else { // FIXME - back to main menu // menu_manager->switchToMainMenu(); } stk_scene->clear(); delete world; world = 0; m_track_number = 0; m_active_race = false; } // exitRace //----------------------------------------------------------------------------- /** A kart has finished the race at the specified time (which can be * different from RaceManager::getWorld()->getClock() in case of setting extrapolated arrival * times). * \param kart The kart that finished the race. * \param time Time at which the kart finished the race. */ void RaceManager::RaceFinished(const Kart *kart, float time) { unsigned int id = kart->getWorldKartId(); // In follow the leader mode, kart 0 does not get any points, // so the position of each kart is actually one better --> decrease pos int pos = kart->getPosition(); assert(pos-1 >= 0); assert(pos-1 < (int)m_kart_status.size()); m_kart_status[id].m_last_score = m_kart_status[id].m_score; m_kart_status[id].m_score += m_score_for_position[pos-1]; m_kart_status[id].m_overall_time += time; m_kart_status[id].m_last_time = time; m_num_finished_karts ++; if(kart->isPlayerKart()) m_num_finished_players++; } // raceFinished //----------------------------------------------------------------------------- /** Reruns the last race. This is called after a race is finished, and it will * adjust the number of points and the overall time before restarting the race. */ void RaceManager::rerunRace() { // Subtract last score from all karts: for(int i=0; irestartRace(); } // rerunRace /* EOF */