Added random GP functionality into gp_info screen. Changed GPData

in GPInfoScreen from a pointer to be an object, while this adds
some coping of this object, it removes potential memory leaks.
This commit is contained in:
hiker 2014-09-02 09:32:12 +10:00
parent 4ba8e85169
commit 7dbfc151bf
11 changed files with 337 additions and 89 deletions

View File

@ -25,23 +25,35 @@
</box> </box>
<spacer width="1" height="3%"/> <spacer width="1" height="3%"/>
<box width="95%" height="17%" padding="10" layout="vertical-row"> <box width="95%" height="17%" padding="10" layout="horizontal-row">
<spacer width="1" height="5%"/> <spacer width="1" height="5%"/>
<div width="50%" height="100%" layout="vertical-row">
<div width="75%" height="fit" layout="horizontal-row" > <div width="100%" height="fit" layout="horizontal-row" >
<spacer width="40" height="2" /> <label id="ai-text" height="100%" width="40%" I18N="Number of AI karts" text="AI karts"/>
<label id="ai-text" height="100%" width="40%" I18N="Number of AI karts" text="Number of AI karts"/> <spinner id="ai-spinner" width="40%" min_value="1" max_value="20" align="center"
<spacer with="40" height="2" /> wrap_around="true" />
<spinner id="ai-spinner" width="20%" min_value="1" max_value="20" align="center" <spacer proportion="1" height="2" />
wrap_around="true" /> </div>
<spacer proportion="1" height="2" /> <div width="100%" height="fit" layout="horizontal-row" >
<label id="reverse-text" height="100%" width="40%" I18N="Drive the track reverse" text="Reverse"/>
<spinner id="reverse-spinner" width="50%" align="center" wrap_around="true" />
<spacer proportion="1" height="2" />
</div>
</div> </div>
<div width="75%" height="fit" layout="horizontal-row" > <div width="50%" height="100%" layout="vertical-row">
<spacer width="40" height="2" /> <div width="100%" height="fit" layout="horizontal-row" >
<label id="reverse-text" height="100%" width="40%" I18N="Drive the track reverse" text="Drive in reverse"/> <label id="track-text" height="100%" width="50%" text="Tracks"
<spacer width="40" height="2" /> I18N="Number of tracks to pick in a random Grand Prix."/>
<checkbox id="reverse"/> <spinner id="track-spinner" width="50%" min_value="1" max_value="20" align="center"
<spacer proportion="1" height="2" /> wrap_around="true" />
<spacer proportion="1" height="2" />
</div>
<div width="100%" height="fit" layout="horizontal-row" >
<label id="group-text" height="100%" width="50%" I18N="Number of AI karts" text="Track group"/>
<spinner id="group-spinner" width="50%" align="center" wrap_around="true" />
<spacer proportion="1" height="2" />
</div>
</div> </div>
<spacer width="1" height="2%"/> <spacer width="1" height="2%"/>
</box> </box>

View File

@ -0,0 +1,69 @@
<?xml version="1.0" encoding="UTF-8"?>
<stkgui>
<icon-button id="back" x="0" y="0" height="8%" icon="gui/back.png"/>
<div x="1%" y="1%" width="98%" height="98%" layout="vertical-row">
<header id="name" width="100%" text_align="center"/>
<spacer width="1" height="5%"/>
<box width="95%" height="40%" padding="10" layout="horizontal-row">
<spacer width="10" height="100%"/>
<!-- Left pane -->
<div width="45%" height="95%" align="center" layout="vertical-row">
<placeholder proportion="1" width="100%" height="100%" id="screenshot_div">
</placeholder>
</div>
<spacer width="5%" height="100%"/>
<!-- Right pane -->
<div width="45%" height="95%" align="center" layout="vertical-row">
<list id="tracks" width="100%" height="100%"/>
</div>
</box>
<spacer width="1" height="3%"/>
<box width="95%" height="30%" padding="10" layout="horizontal-row">
<spacer width="1" height="5%"/>
<div width="50%" height="100%" layout="vertical-row">
<div width="100%" height="fit" layout="horizontal-row" >
<label id="ai-text" height="100%" width="50%" I18N="Number of AI karts" text="AI karts"/>
<spinner id="ai-spinner" width="50%" min_value="1" max_value="20" align="center"
wrap_around="true" />
</div>
<spacer height="5%" />
<div width="100%" height="fit" layout="horizontal-row" >
<label id="reverse-text" width="50%" height="100%" I18N="Drive the track reverse" text="Reverse"/>
<spinner id="reverse-spinner" width="50%" align="center" wrap_around="true" />
</div>
<spacer height="5%" />
<div width="100%" height="fit" layout="horizontal-row">
<label id="track-text" width="50%" height="100%" text="Tracks"
I18N="Number of tracks to pick in a random Grand Prix."/>
<spinner id="track-spinner" width="50%" min_value="1" max_value="20" align="center"
wrap_around="true" />
<spacer height="1%" />
</div>
<spacer height="5%" />
<div width="100%" height="fit" layout="horizontal-row" >
<label id="group-text" width="50%" height="100%" I18N="Number of AI karts" text="Track group"/>
<spinner id="group-spinner" width="50%" align="center" wrap_around="true" />
<spacer proportion="1" height="2" />
</div>
</div>
</box>
<spacer width="1" height="2%"/>
<buttonbar id="buttons" height="15%" width="100%" align="center">
<icon-button id="start" width="64" height="64" icon="gui/green_check.png"
I18N="Start race" text="Start Race"/>
<icon-button id="continue" width="64" height="64" icon="gui/green_check.png"
I18N="Continue race" text="Continue Race"/>
</buttonbar>
</div>
</stkgui>

View File

@ -376,7 +376,7 @@ void ChallengeData::setRace(RaceManager::Difficulty d) const
else if(m_mode==CM_GRAND_PRIX) else if(m_mode==CM_GRAND_PRIX)
{ {
race_manager->setMinorMode(m_minor); race_manager->setMinorMode(m_minor);
race_manager->setGrandPrix(grand_prix_manager->getGrandPrix(m_gp_id)); race_manager->setGrandPrix(*grand_prix_manager->getGrandPrix(m_gp_id));
race_manager->setDifficulty(d); race_manager->setDifficulty(d);
race_manager->setNumKarts(m_num_karts[d]); race_manager->setNumKarts(m_num_karts[d]);
race_manager->setNumLocalPlayers(1); race_manager->setNumLocalPlayers(1);

View File

@ -854,7 +854,7 @@ int handleCmdLine()
Log::warn("main", "There is no GP named '%s'.", s.c_str()); Log::warn("main", "There is no GP named '%s'.", s.c_str());
return 0; return 0;
} }
race_manager->setGrandPrix(gp); race_manager->setGrandPrix(*gp);
} // --gp } // --gp
if(CommandLine::has("--numkarts", &n) ||CommandLine::has("-k", &n)) if(CommandLine::has("--numkarts", &n) ||CommandLine::has("-k", &n))

View File

@ -762,12 +762,9 @@ void RaceManager::rerunRace()
//----------------------------------------------------------------------------- //-----------------------------------------------------------------------------
void RaceManager::startGP(const GrandPrixData* gp, bool from_overworld, void RaceManager::startGP(const GrandPrixData &gp, bool from_overworld,
bool continue_saved_gp) bool continue_saved_gp)
{ {
assert(gp != NULL);
//std::cout << gp->getId();
StateManager::get()->enterGameState(); StateManager::get()->enterGameState();
setGrandPrix(gp); setGrandPrix(gp);
setCoinTarget( 0 ); // Might still be set from a previous challenge setCoinTarget( 0 ); // Might still be set from a previous challenge

View File

@ -411,9 +411,9 @@ public:
void setDifficulty(Difficulty diff); void setDifficulty(Difficulty diff);
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
void setGrandPrix(const GrandPrixData *gp) void setGrandPrix(const GrandPrixData &gp)
{ {
m_grand_prix = *gp; m_grand_prix = gp;
m_coin_target = 0; m_coin_target = 0;
} }
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
@ -683,7 +683,7 @@ public:
* \brief Higher-level method to start a GP without having to care about * \brief Higher-level method to start a GP without having to care about
* the exact startup sequence * the exact startup sequence
*/ */
void startGP(const GrandPrixData* gp, bool from_overworld, void startGP(const GrandPrixData &gp, bool from_overworld,
bool continue_saved_gp); bool continue_saved_gp);
/** /**

View File

@ -254,7 +254,7 @@ void GPInfoDialog::onEnterPressedInternal()
ModalDialog::dismiss(); ModalDialog::dismiss();
// Disable accidentally unlocking of a challenge // Disable accidentally unlocking of a challenge
PlayerManager::getCurrentPlayer()->setCurrentChallenge(""); PlayerManager::getCurrentPlayer()->setCurrentChallenge("");
race_manager->startGP(grand_prix_manager->getGrandPrix(gp_id), false, false); race_manager->startGP(*grand_prix_manager->getGrandPrix(gp_id), false, false);
} }
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
@ -270,8 +270,8 @@ GUIEngine::EventPropagation GPInfoDialog::processEvent(const std::string& event_
// becomes invalid! // becomes invalid!
std::string save_source = event_source; std::string save_source = event_source;
ModalDialog::dismiss(); ModalDialog::dismiss();
race_manager->startGP(grand_prix_manager->getGrandPrix(gp_id), false, //race_manager->startGP(grand_prix_manager->getGrandPrix(gp_id), false,
(save_source == "continue")); // (save_source == "continue"));
return GUIEngine::EVENT_BLOCK; return GUIEngine::EVENT_BLOCK;
} }

View File

@ -143,7 +143,7 @@ GUIEngine::EventPropagation RandomGPInfoDialog::processEvent(
// Save GP data, since dismiss will delete this object. // Save GP data, since dismiss will delete this object.
GrandPrixData gp = m_gp; GrandPrixData gp = m_gp;
ModalDialog::dismiss(); ModalDialog::dismiss();
race_manager->startGP(&gp, false, false); race_manager->startGP(gp, false, false);
return GUIEngine::EVENT_BLOCK; return GUIEngine::EVENT_BLOCK;
} }
else if (eventSource == "Number of tracks") else if (eventSource == "Number of tracks")

View File

@ -1,5 +1,6 @@
// SuperTuxKart - a fun racing game with go-kart // SuperTuxKart - a fun racing game with go-kart
// Copyright (C) 2009-2013 Marianne Gagnon // Copyright (C) 2009-2014 Marianne Gagnon
// 2014 Joerg Henrichs, konstin
// //
// This program is free software; you can redistribute it and/or // This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License // modify it under the terms of the GNU General Public License
@ -29,6 +30,7 @@
#include "guiengine/widgets/label_widget.hpp" #include "guiengine/widgets/label_widget.hpp"
#include "guiengine/widgets/list_widget.hpp" #include "guiengine/widgets/list_widget.hpp"
#include "guiengine/widgets/ribbon_widget.hpp" #include "guiengine/widgets/ribbon_widget.hpp"
#include "guiengine/widgets/spinner_widget.hpp"
#include "io/file_manager.hpp" #include "io/file_manager.hpp"
#include "race/grand_prix_manager.hpp" #include "race/grand_prix_manager.hpp"
#include "race/grand_prix_data.hpp" #include "race/grand_prix_data.hpp"
@ -42,88 +44,177 @@
#include <IGUIEnvironment.h> #include <IGUIEnvironment.h>
#include <IGUIStaticText.h> #include <IGUIStaticText.h>
#include <algorithm>
using irr::gui::IGUIStaticText; using irr::gui::IGUIStaticText;
using GUIEngine::PROP_ID; using GUIEngine::PROP_ID;
using namespace GUIEngine; using namespace GUIEngine;
DEFINE_SCREEN_SINGLETON( GPInfoScreen ); DEFINE_SCREEN_SINGLETON( GPInfoScreen );
DEFINE_SCREEN_SINGLETON( RandomGPInfoScreen );
GPInfoScreen::GPInfoScreen() : Screen("gp_info.stkgui") BaseGPInfoScreen::BaseGPInfoScreen(const std::string &name) : Screen(name.c_str())
{ {
m_gp = NULL;
m_curr_time = 0.0f; m_curr_time = 0.0f;
} // GPInfoScreen // Necessary to test if loadedFroMFile() was executed (in setGP)
m_reverse_spinner = NULL;
} // BaseGPInfoScreen
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
GPInfoScreen::~GPInfoScreen() BaseGPInfoScreen::~BaseGPInfoScreen()
{ {
} // ~GPInfoScreen } // ~BaseGPInfoScreen
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
/** Sets the GP to be displayed. void BaseGPInfoScreen::loadedFromFile()
{
// The group spinner is filled in init every time the screen is shown
// (since the groups can change if addons are added/deleted).
m_group_spinner = getWidget<SpinnerWidget>("group-spinner");
m_reverse_spinner = getWidget<SpinnerWidget>("reverse-spinner");
m_reverse_spinner->addLabel(_("No"));
m_reverse_spinner->addLabel(_("Yes"));
m_reverse_spinner->addLabel(_("Random"));
m_reverse_spinner->setValue(0);
m_num_tracks_spinner = getWidget<SpinnerWidget>("track-spinner");
// Only init the number of tracks here, this way the previously selected
// number of tracks will be the default.
m_num_tracks_spinner->setValue(4);
int number_of_tracks = m_num_tracks_spinner->getValue();
} // loadedFromFile
// ----------------------------------------------------------------------------
/** Sets the GP to be displayed. If the identifier is 'random', no gp info
* will be loaded.
*/ */
void GPInfoScreen::setGP(const std::string &gp_ident) void BaseGPInfoScreen::setGP(const std::string &gp_ident)
{ {
m_gp = grand_prix_manager->getGrandPrix(gp_ident); if(gp_ident!="random")
m_gp = *grand_prix_manager->getGrandPrix(gp_ident);
else
{
// Doesn't matter what kind of GP we create, it just gets the
// right id ("random").
m_gp.createRandomGP(1, "standard",
m_reverse_spinner ? getReverse()
: GrandPrixData::GP_NO_REVERSE);
}
} // setGP } // setGP
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GPInfoScreen::init() GrandPrixData::GPReverseType BaseGPInfoScreen::getReverse() const
{
switch (m_reverse_spinner->getValue())
{
case 0: return GrandPrixData::GP_NO_REVERSE; break;
case 1: return GrandPrixData::GP_ALL_REVERSE; break;
case 2: return GrandPrixData::GP_RANDOM_REVERSE; break;
default: assert(false);
} // switch
// Avoid compiler warning
return GrandPrixData::GP_NO_REVERSE;
} // getReverse
// ----------------------------------------------------------------------------
void BaseGPInfoScreen::init()
{ {
Screen::init(); Screen::init();
m_curr_time = 0.0f; m_curr_time = 0.0f;
getWidget<LabelWidget>("name")->setText(m_gp->getName(), false); bool random = m_gp.getId()=="random";
m_gp->checkConsistency(); SpinnerWidget *reverse_spinner = getWidget<SpinnerWidget>("reverse-spinner");
getWidget<LabelWidget >("track-text" )->setVisible(random);
m_num_tracks_spinner->setVisible(random);
getWidget<LabelWidget >("group-text" )->setVisible(random);
m_group_spinner->setVisible(random);
m_over_body = UserConfigParams::m_height/7;
m_lower_bound = UserConfigParams::m_height*6/7; if(random)
{
getWidget<LabelWidget>("name")->setText(_("Random Grand Prix"), false);
getWidget<IconButtonWidget>("continue")->setVisible(false);
// We have to recreate the group spinner, but a new group might have
// been added or deleted since the last time this screen was shown.
m_group_spinner->clearLabels();
m_group_spinner->addLabel("all");
int index_standard;
const std::vector<std::string>& groups = track_manager->getAllTrackGroups();
for (unsigned int i = 0; i < groups.size(); i++)
{
m_group_spinner->addLabel(stringw(groups[i].c_str()));
if (groups[i] == "standard")
index_standard = i + 1;
}
// Try to keep a previously selected group value
if(m_group_spinner->getValue() >= (int)groups.size())
{
m_group_spinner->setValue(index_standard);
m_group_name = "standard";
}
else
m_group_name = stringc(m_group_spinner->getStringValue().c_str()).c_str();
// If there are more tracks selected atm as in the group (which can
// happen if the group has been changed since last time this screen
// was shown), adjust it:
int max_num_tracks =
m_group_name=="all" ? track_manager->getNumberOfTracks()
: track_manager->getTracksInGroup(m_group_name).size();
if(m_num_tracks_spinner->getValue() > max_num_tracks)
{
m_num_tracks_spinner->setValue(max_num_tracks);
}
// Now create the random GP:
m_gp.createRandomGP(m_num_tracks_spinner->getValue(),
m_group_name, getReverse(), true);
}
else
{
getWidget<LabelWidget>("name")->setText(m_gp.getName(), false);
m_gp.checkConsistency();
// Check if there is a saved GP:
SavedGrandPrix* saved_gp =
SavedGrandPrix::getSavedGP(StateManager::get()
->getActivePlayerProfile(0)->getUniqueID(),
m_gp.getId(),
race_manager->getDifficulty(),
race_manager->getNumberOfKarts(),
race_manager->getNumLocalPlayers());
getWidget<IconButtonWidget>("continue")->setVisible(saved_gp != NULL);
}
addTracks(); addTracks();
addScreenshot(); addScreenshot();
CheckBoxWidget *reverse = getWidget<CheckBoxWidget>("reverse");
reverse->setState(false);
// Check if there is a saved GP:
SavedGrandPrix* saved_gp =
SavedGrandPrix::getSavedGP(StateManager::get()
->getActivePlayerProfile(0)->getUniqueID(),
m_gp->getId(),
race_manager->getDifficulty(),
race_manager->getNumberOfKarts(),
race_manager->getNumLocalPlayers());
getWidget<IconButtonWidget>("continue")->setVisible(saved_gp!=NULL);
} // init } // init
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GPInfoScreen::loadedFromFile()
{
} // loadedFromFile
// ---------------------------------------------------------------------------- void BaseGPInfoScreen::addTracks()
void GPInfoScreen::addTracks()
{ {
const std::vector<std::string> tracks = m_gp->getTrackNames(); const std::vector<std::string> tracks = m_gp.getTrackNames();
ListWidget *list = getWidget<ListWidget>("tracks"); ListWidget *list = getWidget<ListWidget>("tracks");
list->clear(); list->clear();
for(unsigned int i=0; i<tracks.size(); i++) for (unsigned int i = 0; i < tracks.size(); i++)
{ {
const Track *track = track_manager->getTrack(tracks[i]); const Track *track = track_manager->getTrack(tracks[i]);
std::string s = StringUtils::toString(i); std::string s = StringUtils::toString(i);
list->addItem(s, translations->fribidize(track->getName()) ); list->addItem(s, translations->fribidize(track->getName()));
} }
} // addTracks } // addTracks
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GPInfoScreen::addScreenshot() void BaseGPInfoScreen::addScreenshot()
{ {
Widget* screenshot_div = getWidget("screenshot_div"); Widget* screenshot_div = getWidget("screenshot_div");
@ -143,7 +234,7 @@ void GPInfoScreen::addScreenshot()
m_screenshot_widget->m_properties[PROP_ICON] = "gui/main_help.png"; m_screenshot_widget->m_properties[PROP_ICON] = "gui/main_help.png";
m_screenshot_widget->add(); m_screenshot_widget->add();
const Track *track = track_manager->getTrack(m_gp->getTrackId(0)); const Track *track = track_manager->getTrack(m_gp.getTrackId(0));
video::ITexture* screenshot = irr_driver->getTexture(track->getScreenshotFile(), video::ITexture* screenshot = irr_driver->getTexture(track->getScreenshotFile(),
"While loading screenshot for track '%s':", "While loading screenshot for track '%s':",
track->getFilename() ); track->getFilename() );
@ -154,19 +245,39 @@ void GPInfoScreen::addScreenshot()
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GPInfoScreen::onEnterPressedInternal() void BaseGPInfoScreen::onEnterPressedInternal()
{ {
// Save the GP id because dismiss() will destroy this instance // Save the GP id because dismiss() will destroy this instance
std::string gp_id = m_gp->getId(); GrandPrixData gp_data = m_gp;
//FIXME ModalDialog::dismiss();
// Disable accidentally unlocking of a challenge // Disable accidentally unlocking of a challenge
PlayerManager::getCurrentPlayer()->setCurrentChallenge(""); PlayerManager::getCurrentPlayer()->setCurrentChallenge("");
race_manager->startGP(grand_prix_manager->getGrandPrix(gp_id), false, false); race_manager->startGP(m_gp, false, false);
} }
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void BaseGPInfoScreen::updateRandomGP()
{
// First get the right track group to use
const std::vector<std::string>& groups = track_manager->getAllTrackGroups();
void GPInfoScreen::eventCallback(GUIEngine::Widget *, const std::string &name, int n = m_group_spinner->getValue();
std::string track_group;
if(n>=0 && n < (int)groups.size())
{
track_group = groups[m_group_spinner->getValue()];
}
else
{
m_group_spinner->setValue(0);
track_group = groups[0];
}
int number_of_tracks = getWidget<SpinnerWidget>("track-spinner")->getValue();
} // updateRandomGP
// ----------------------------------------------------------------------------
void BaseGPInfoScreen::eventCallback(GUIEngine::Widget *, const std::string &name,
const int player_id) const int player_id)
{ {
if(name=="buttons") if(name=="buttons")
@ -177,22 +288,49 @@ void GPInfoScreen::eventCallback(GUIEngine::Widget *, const std::string &name,
if (button == "start" || button=="continue") if (button == "start" || button=="continue")
{ {
// Save GP identifier, since dismiss will delete this object. // Save GP identifier, since dismiss will delete this object.
std::string gp_id = m_gp->getId(); GrandPrixData gp_data = m_gp;
// Also create a copy of the string: it is a reference to data // Also create a copy of the string: it is a reference to data
// in a widget in the dialog - so if we call dismiss, this reference // in a widget in the dialog - so if we call dismiss, this reference
// becomes invalid! // becomes invalid!
std::string save_source = name; std::string save_source = name;
// FIXME ModalDialog::dismiss(); // FIXME ModalDialog::dismiss();
race_manager->startGP(grand_prix_manager->getGrandPrix(gp_id), false, race_manager->startGP(gp_data, false, (save_source == "continue"));
(save_source == "continue"));
} }
} // name=="buttons" } // name=="buttons"
else if (name=="group-spinner")
{
m_group_name = stringc(m_group_spinner->getStringValue()).c_str();
// Update the maximum for the number of tracks since it's depending on
// the current track. The current value in the Number-of-tracks-spinner
// has to be updated, since otherwise the displayed (and used) value
// can be bigger than the maximum. (Might be a TODO to fix this)
unsigned int max = (m_group_name == "all") ?
track_manager->getNumberOfTracks() :
track_manager->getTracksInGroup(m_group_name).size();
m_num_tracks_spinner->setMax(max);
int number_of_tracks = std::min((int)max, m_num_tracks_spinner->getValue());
if (m_num_tracks_spinner->getValue() > (signed)max)
m_num_tracks_spinner->setValue(max);
// Create a new (i.e. with new tracks) random gp, since the old
// tracks might not all belong to the newly selected group.
m_gp.createRandomGP(m_num_tracks_spinner->getValue(), m_group_name,
getReverse(), /*new_tracks*/true);
addTracks();
}
else if (name=="track-spinner")
{
m_gp.changeTrackNumber(m_num_tracks_spinner->getValue(), m_group_name);
addTracks();
}
} // eventCallback } // eventCallback
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
void GPInfoScreen::onUpdate(float dt) void BaseGPInfoScreen::onUpdate(float dt)
{ {
if (dt == 0) if (dt == 0)
return; // if nothing changed, return right now return; // if nothing changed, return right now
@ -200,7 +338,7 @@ void GPInfoScreen::onUpdate(float dt)
m_curr_time += dt; m_curr_time += dt;
int frameAfter = (int)(m_curr_time / 1.5f); int frameAfter = (int)(m_curr_time / 1.5f);
const std::vector<std::string> tracks = m_gp->getTrackNames(); const std::vector<std::string> tracks = m_gp.getTrackNames();
if (frameAfter >= (int)tracks.size()) if (frameAfter >= (int)tracks.size())
{ {
frameAfter = 0; frameAfter = 0;

View File

@ -21,32 +21,36 @@
#define HEADER_GP_INFO_SCREEN_HPP #define HEADER_GP_INFO_SCREEN_HPP
#include "guiengine/screen.hpp" #include "guiengine/screen.hpp"
#include "race/grand_prix_data.hpp"
class GrandPrixData; class GrandPrixData;
namespace GUIEngine namespace GUIEngine
{ {
class IconButtonWidget; class IconButtonWidget;
class SpinnerWidget;
} }
/** /**
* \brief Dialog that shows information about a specific grand prix * \brief Dialog that shows information about a specific grand prix
* \ingroup states_screens * \ingroup states_screens
*/ */
class GPInfoScreen : public GUIEngine::Screen, class BaseGPInfoScreen : public GUIEngine::Screen
public GUIEngine::ScreenSingleton<GPInfoScreen>
{ {
protected: // Necessary for RandomGPInfoScreen private:
GUIEngine::SpinnerWidget *m_group_spinner;
GUIEngine::SpinnerWidget *m_reverse_spinner;
GUIEngine::SpinnerWidget *m_num_tracks_spinner;
/** The currently selected group name. */
std::string m_group_name;
protected: // Necessary for RandomBaseGPInfoScreen
GUIEngine::IconButtonWidget* m_screenshot_widget; GUIEngine::IconButtonWidget* m_screenshot_widget;
float m_curr_time; float m_curr_time;
/** The grand prix data. */ /** The grand prix data. */
GrandPrixData *m_gp; GrandPrixData m_gp;
/** height of the separator over the body */
int m_over_body;
/** height of the seperator over the buttons */
int m_lower_bound;
/** \brief display all the tracks according to the current gp /** \brief display all the tracks according to the current gp
* For a normal gp info dialog, it just creates a label for every track. * For a normal gp info dialog, it just creates a label for every track.
@ -54,12 +58,14 @@ protected: // Necessary for RandomGPInfoScreen
* labels as possible by just changing their text. */ * labels as possible by just changing their text. */
void addTracks(); void addTracks();
void addScreenshot(); void addScreenshot();
void updateRandomGP();
GrandPrixData::GPReverseType getReverse() const;
public: public:
GPInfoScreen(); BaseGPInfoScreen(const std::string &name);
/** Places the focus back on the selected GP, in the case that the dialog /** Places the focus back on the selected GP, in the case that the dialog
* was cancelled and we're returning to the track selection screen */ * was cancelled and we're returning to the track selection screen */
virtual ~GPInfoScreen(); virtual ~BaseGPInfoScreen();
void onEnterPressedInternal(); void onEnterPressedInternal();
virtual void eventCallback(GUIEngine::Widget *, const std::string &name, virtual void eventCallback(GUIEngine::Widget *, const std::string &name,
@ -70,6 +76,28 @@ public:
virtual void onUpdate(float dt); virtual void onUpdate(float dt);
void setGP(const std::string &gp_ident); void setGP(const std::string &gp_ident);
}; }; // BaseGPInfoScreen
// ============================================================================
class GPInfoScreen: public BaseGPInfoScreen,
public GUIEngine::ScreenSingleton<GPInfoScreen>
{
private:
GPInfoScreen() : BaseGPInfoScreen("gp_info.stkgui")
{};
public:
friend class GUIEngine::ScreenSingleton<GPInfoScreen>;
}; // class GPInfoScreen
// ============================================================================
class RandomGPInfoScreen: public BaseGPInfoScreen,
public GUIEngine::ScreenSingleton<RandomGPInfoScreen>
{
private:
RandomGPInfoScreen() : BaseGPInfoScreen("random_gp_info.stkgui")
{};
public:
friend class GUIEngine::ScreenSingleton<RandomGPInfoScreen>;
}; // class RandomGPInfoScreen
#endif #endif

View File

@ -116,7 +116,11 @@ void TracksScreen::eventCallback(Widget* widget, const std::string& name,
else else
{ {
if (selection == "Random Grand Prix") if (selection == "Random Grand Prix")
new RandomGPInfoDialog(); {
RandomGPInfoScreen *gpis = RandomGPInfoScreen::getInstance();
gpis->setGP("random");
StateManager::get()->pushScreen(gpis);
}
else else
{ {
GPInfoScreen *gpis = GPInfoScreen::getInstance(); GPInfoScreen *gpis = GPInfoScreen::getInstance();