// $Id$ // // SuperTuxKart - a fun racing game with go-kart // Copyright (C) 2004-2005 Steve Baker // Copyright (C) 2006 Joerg Henrichs, Steve Baker // // 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. #ifdef __APPLE__ // Necessary for Macs when using SDL without Xwindows: this include // will rename main to SDLmain, and a new main program will be linked // in from the library, causing a correct framework to be set up # include "SDL/SDL.h" #endif #ifdef WIN32 # ifdef __CYGWIN__ # include # endif # include # ifdef _MSC_VER # include # include # endif #else # include #endif #include #include #include #include #include #include "user_config.hpp" #include "track_manager.hpp" #include "track.hpp" #include "kart_properties_manager.hpp" #include "kart.hpp" #include "projectile_manager.hpp" #include "race_manager.hpp" #include "loader.hpp" #include "game_manager.hpp" #include "widget_manager.hpp" #include "material_manager.hpp" #include "sdldrv.hpp" #include "callback_manager.hpp" #include "history.hpp" #include "herring_manager.hpp" #include "attachment_manager.hpp" #include "sound_manager.hpp" #include "stk_config.hpp" #include "translation.hpp" #include "highscore_manager.hpp" #include "gui/menu_manager.hpp" #include "scene.hpp" // Only needed for bullet debug! #ifdef __APPLE__ # include #else # include #endif void cmdLineHelp (char* invocation) { fprintf ( stdout, _("Usage: %s [OPTIONS]\n\n" "Run SuperTuxKart, a racing game with go-kart that features" " the Tux and friends.\n\n" "Options:\n" " -N, --no-start-screen Quick race\n" " -t, --track NAME Start at track NAME (see --list-tracks)\n" " --stk-config FILE use ./data/FILE instead of ./data/stk_config.data\n" " -l, --list-tracks Show available tracks\n" " -k, --numkarts NUM Number of karts on the racetrack\n" " --kart NAME Use kart number NAME (see --list-karts)\n" " --list-karts Show available karts\n" " --laps N Define number of laps to N\n" " --mode N N=1 novice, N=2 driver, N=3 racer\n" " --players n Define number of players to between 1 and 4.\n" //FIXME " --reverse Enable reverse mode\n" //FIXME " --mirror Enable mirror mode (when supported)\n" " --herring STYLE Use STYLE as your herring style\n" " -f, --fullscreen Fullscreen display\n" " -w, --windowed Windowed display (default)\n" " -s, --screensize WxH Set the screen size (e.g. 320x200)\n" " -v, --version Show version\n" // should not be used by unaware users: // " --profile Enable automatic driven profile mode for 20 seconds\n" // " --profile=n Enable automatic driven profile mode for n seconds\n" // " if n<0 --> (-n) = number of laps to drive // " --history Replay history file 'history.dat'\n" " --log=terminal Write messages to screen\n" " --log=file Write messages/warning to log files stdout.log/stderr.log\n" " -h, --help Show this help\n" "\n" "You can visit SuperTuxKart's homepage at " "http://supertuxkart.berlios.de\n\n"), invocation ); } // cmdLineHelp //============================================================================= int handleCmdLine(int argc, char **argv) { int n; for(int i=1; im_keyboard_debug=true; } else if(sscanf(argv[i], "--track-debug=%d",&n)==1) { user_config->m_track_debug=n; } else if(!strcmp(argv[i], "--track-debug")) { user_config->m_track_debug=1; } else if(!strcmp(argv[i], "--bullet-debug")) { user_config->m_bullet_debug=1; } else if( (!strcmp(argv[i], "--kart") && i+1getKartFile(std::string(argv[i+1])+".tkkf"); if(filename!="") { race_manager->setPlayerKart(0, argv[i+1]); fprintf ( stdout, _("You choose to use kart '%s'.\n"), argv[i+1] ) ; } else { fprintf(stdout, _("Kart '%s' not found, ignored.\n"), argv[i+1]); } } else if( (!strcmp(argv[i], "--mode") && i+1setDifficulty(RD_EASY); break; case 2: race_manager->setDifficulty(RD_MEDIUM); break; case 3: race_manager->setDifficulty(RD_HARD); break; } } else if( (!strcmp(argv[i], "--track") || !strcmp(argv[i], "-t")) && i+1setTrack(argv[i+1]); fprintf ( stdout, _("You choose to start in track: %s.\n"), argv[i+1] ) ; } else if( (!strcmp(argv[i], "--stk-config")) && i+1load(loader->getConfigFile(argv[i+1])); fprintf ( stdout, _("STK config will be read from %s.\n"), argv[i+1] ) ; } else if( (!strcmp(argv[i], "--numkarts") || !strcmp(argv[i], "-k")) && i+1m_karts = atoi(argv[i+1]); if(user_config->m_karts>stk_config->m_max_karts) { fprintf(stdout, _("Number of karts reset to maximum number %d\n"), stk_config->m_max_karts); user_config->m_karts = stk_config->m_max_karts; } race_manager->setNumKarts(user_config->m_karts ); fprintf ( stdout, _("%d karts will be used.\n"), user_config->m_karts); i++; } else if( !strcmp(argv[i], "--list-tracks") || !strcmp(argv[i], "-l") ) { fprintf ( stdout, _(" Available tracks:\n") ); for (size_t i = 0; i != track_manager->getTrackCount(); i++) fprintf ( stdout, "\t%10s: %s\n", track_manager->getTrack(i)->getIdent().c_str(), track_manager->getTrack(i)->getName()); fprintf ( stdout, _("Use --track N to choose track.\n\n")); delete track_manager; track_manager = 0; return 0; } else if( !strcmp(argv[i], "--list-karts") ) { bool dont_load_models=true; kart_properties_manager->loadKartData(dont_load_models) ; fprintf ( stdout, _(" Available karts:\n") ); for (unsigned int i = 0; NULL != kart_properties_manager->getKartById(i); i++) { const KartProperties* KP= kart_properties_manager->getKartById(i); fprintf (stdout, "\t%10s: %s\n", KP->getIdent(), KP->getName().c_str()); } fprintf ( stdout, "\n" ); return 0; } else if ( !strcmp(argv[i], "--no-start-screen") || !strcmp(argv[i], "-N") ) { user_config->m_no_start_screen = true; //FIXME} else if ( !strcmp(argv[i], "--reverse") ) { //FIXME:fprintf ( stdout, "Enabling reverse mode.\n" ) ; //FIXME:raceSetup.reverse = 1; } else if ( !strcmp(argv[i], "--mirror") ) { #ifdef SSG_BACKFACE_COLLISIONS_SUPPORTED fprintf ( stdout, _("Enabling mirror mode.\n") ) ; //raceSetup.mirror = 1; #else //raceSetup.mirror = 0 ; #endif } else if ( !strcmp(argv[i], "--laps") && i+1setNumLaps(atoi(argv[i+1])); } /* FIXME: else if ( !strcmp(argv[i], "--players") && i+1 4) { fprintf ( stderr, "You choose an invalid number of players: %d.\n", raceSetup.numPlayers ); cmdLineHelp(argv[0]); return 0; } fprintf ( stdout, _("You choose to have %d players.\n"), atoi(argv[i+1]) ) ; } */ #if !defined(WIN32) && !defined(__CYGWIN) else if ( !strcmp(argv[i], "--fullscreen") || !strcmp(argv[i], "-f")) { // Check that current res is not blacklisted std::ostringstream o; o << user_config->m_width << "x" << user_config->m_height; std::string res = o.str(); if (std::find(user_config->m_blacklist_res.begin(), user_config->m_blacklist_res.end(),res) == user_config->m_blacklist_res.end()) user_config->m_fullscreen = true; else fprintf ( stdout, _("Resolution %s has been blacklisted, so it is not available!\n"), res.c_str()); } else if ( !strcmp(argv[i], "--windowed") || !strcmp(argv[i], "-w")) { user_config->m_fullscreen = false; } #endif else if ( !strcmp(argv[i], "--screensize") || !strcmp(argv[i], "-s") ) { //Check if fullscreen and new res is blacklisted int width, height; if (sscanf(argv[i+1], "%dx%d", &width, &height) == 2) { std::ostringstream o; o << width << "x" << height; std::string res = o.str(); if (!user_config->m_fullscreen || std::find(user_config->m_blacklist_res.begin(), user_config->m_blacklist_res.end(),res) == user_config->m_blacklist_res.end()) { user_config->m_prev_width = user_config->m_width = width; user_config->m_prev_height = user_config->m_height = height; fprintf ( stdout, _("You choose to be in %dx%d.\n"), user_config->m_width, user_config->m_height ); } else fprintf ( stdout, _("Resolution %s has been blacklisted, so it is not available!\n"), res.c_str()); } else { fprintf(stderr, _("Error: --screensize argument must be given as WIDTHxHEIGHT\n")); exit(EXIT_FAILURE); } } else if( !strcmp(argv[i], "--version") || !strcmp(argv[i], "-v") ) { #ifdef VERSION fprintf ( stdout, "SuperTuxKart, %s.\n", VERSION ) ; #endif #ifdef SVNVERSION fprintf ( stdout, "SuperTuxKart, SVN revision number '%s'.\n", SVNVERSION ) ; #endif return 0; } else if( !strcmp(argv[i], "--log=terminal")) { user_config->m_log_errors=false; } else if( !strcmp(argv[i], "--log=file")) { user_config->m_log_errors=true; }else if( sscanf(argv[i], "--profile=%d", &n)==1) { user_config->m_profile=n; if(n<0) { fprintf(stdout,"Profiling %d laps\n",-n); race_manager->setNumLaps(-n); } else { printf("Profiling: %d seconds.\n",user_config->m_profile); race_manager->setNumLaps (999999); // profile end depends on time } } else if( !strcmp(argv[i], "--profile") ) { user_config->m_profile=20; } else if( !strcmp(argv[i], "--history") ) { user_config->m_replay_history=true; } else if( !strcmp(argv[i], "--herring") && i+1setUserFilename(argv[i+1]); } else { fprintf ( stderr, _("Invalid parameter: %s.\n\n"), argv[i] ); cmdLineHelp(argv[0]); return 0; } } // for i m_profile) { user_config->setSFX(UserConfig::UC_DISABLE); // Disable sound effects user_config->setMusic(UserConfig::UC_DISABLE);// and music when profiling } return 1; } /* handleCmdLine */ //============================================================================= void InitTuxkart() { loader = new Loader(); loader->setCreateStateCallback(getAppState); user_config = new UserConfig(); sound_manager = new SoundManager(); // The order here can be important, e.g. KartPropertiesManager needs // defaultKartProperties. history = new History (); material_manager = new MaterialManager (); track_manager = new TrackManager (); stk_config = new STKConfig (); kart_properties_manager = new KartPropertiesManager(); projectile_manager = new ProjectileManager (); collectable_manager = new CollectableManager (); race_manager = new RaceManager (); callback_manager = new CallbackManager (); herring_manager = new HerringManager (); attachment_manager = new AttachmentManager (); highscore_manager = new HighscoreManager (); track_manager ->loadTrackList () ; // default settings for Quickstart race_manager->setNumPlayers(1); race_manager->setNumLaps (3); race_manager->setRaceMode (RaceSetup::RM_QUICK_RACE); race_manager->setDifficulty(RD_MEDIUM); stk_config->load(loader->getConfigFile("stk_config.data")); } //============================================================================= int main(int argc, char *argv[] ) { try { initTranslations(); glutInit(&argc, argv); InitTuxkart(); //handleCmdLine() needs InitTuxkart() so it can't be called first if(!handleCmdLine(argc, argv)) exit(0); if (user_config->m_log_errors) //Enable logging of stdout and stderr to logfile { std::string logoutfile = loader->getLogFile("stdout.log"); std::string logerrfile = loader->getLogFile("stderr.log"); std::cout << "Error messages and other text output will be logged to " ; std::cout << logoutfile << " and "< loadMaterial (); kart_properties_manager -> loadKartData (); projectile_manager -> loadData (); collectable_manager -> loadCollectables (); herring_manager -> loadDefaultHerrings(); attachment_manager -> loadModels (); scene = new Scene(); //For some reason, calling this before the material loading screws //the background picture. fntInit(); init_fonts(); widget_manager = new WidgetManager; menu_manager->switchToMainMenu(); // Replay a race // ============= if(user_config->m_replay_history) { // This will setup the race manager etc. history->Load(); race_manager->start(); game_manager->run(); // well, actually run() will never return, since // it exits after replaying history (see history::GetNextDT()). // So the next line is just to make this obvious here! exit(-3); } // Not replaying // ============= if(!user_config->m_profile) { if(user_config->m_no_start_screen) { // Quickstart (-N) // =============== // all defaults are set in InitTuxkart() //race_manager->setNumPlayers(1); //race_manager->setNumKarts (4); //race_manager->setRaceMode (RaceSetup::RM_QUICK_RACE); //race_manager->setDifficulty(RD_MEDIUM); //race_manager->setPlayerKart(0, kart_properties_manager->getKart("tuxkart")->getIdent()); //race_manager->setNumLaps (3); //race_manager->setTrack ("tuxtrack"); //race_manager->setTrack ("sandtrack"); //race_manager->setTrack ("race"); race_manager->start(); } } else // profile { // Profiling // ========= race_manager->setNumPlayers(1); race_manager->setPlayerKart(0, kart_properties_manager->getKart("tuxkart")->getIdent()); race_manager->setRaceMode (RaceSetup::RM_QUICK_RACE); race_manager->setDifficulty(RD_HARD); race_manager->start(); } game_manager->run(); } // try catch (std::exception &e) { fprintf(stderr,e.what()); fprintf(stderr,_("\nAborting SuperTuxKart\n")); } /* Program closing...*/ user_config->saveConfig(); drv_deinit(); if (user_config->m_log_errors) //close logfiles { fclose(stderr); fclose(stdout); } delete highscore_manager; delete_fonts(); return 0 ; }