Files
stk-code_catmod/src/states_screens/track_info_screen.cpp
CodingJellyfish db2cc69d89 Refactor sprite scaling in STKModifiedSpriteBank
Make it proportional to font height so that it can be auto-resized
2024-04-23 15:30:35 +08:00

810 lines
30 KiB
C++

// SuperTuxKart - a fun racing game with go-kart
// Copyright (C) 2009-2015 Marianne Gagnon
// (C) 2014-2015 Joerg Henrichs
//
// 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 "states_screens/track_info_screen.hpp"
#include "challenges/unlock_manager.hpp"
#include "config/player_manager.hpp"
#include "config/user_config.hpp"
#include "graphics/irr_driver.hpp"
#include "graphics/material.hpp"
#include "graphics/stk_tex_manager.hpp"
#include "guiengine/CGUISpriteBank.hpp"
#include "guiengine/engine.hpp"
#include "guiengine/screen.hpp"
#include "guiengine/widgets/button_widget.hpp"
#include "guiengine/widgets/check_box_widget.hpp"
#include "guiengine/widgets/icon_button_widget.hpp"
#include "guiengine/widgets/label_widget.hpp"
#include "guiengine/widgets/list_widget.hpp"
#include "guiengine/widgets/ribbon_widget.hpp"
#include "guiengine/widgets/spinner_widget.hpp"
#include "io/file_manager.hpp"
#include "karts/kart_properties.hpp"
#include "karts/kart_properties_manager.hpp"
#include "race/highscores.hpp"
#include "race/highscore_manager.hpp"
#include "race/race_manager.hpp"
#include "states_screens/state_manager.hpp"
#include "tracks/track.hpp"
#include "tracks/track_manager.hpp"
#include "utils/string_utils.hpp"
#include "utils/translation.hpp"
#include <IGUIEnvironment.h>
#include <IGUIImage.h>
#include <IGUIStaticText.h>
using namespace irr::gui;
using namespace irr::video;
using namespace irr::core;
using namespace GUIEngine;
// ----------------------------------------------------------------------------
/** Constructor, which loads the corresponding track_info.stkgui file. */
TrackInfoScreen::TrackInfoScreen()
: Screen("track_info.stkgui")
{
} // TrackInfoScreen
// ----------------------------------------------------------------------------
/* Saves some often used pointers. */
void TrackInfoScreen::loadedFromFile()
{
m_target_type_spinner = getWidget<SpinnerWidget>("target-type-spinner");
m_target_type_label = getWidget <LabelWidget>("target-type-text");
m_ai_blue_spinner = getWidget<SpinnerWidget>("ai-blue-spinner");
m_ai_blue_label = getWidget <LabelWidget>("ai-blue-text");
m_ai_blue_div = getWidget<Widget>("ai-blue-div");
m_target_type_div = getWidget<Widget>("target-type-div");
m_target_value_spinner = getWidget<SpinnerWidget>("target-value-spinner");
m_target_value_label = getWidget<LabelWidget>("target-value-text");
m_ai_kart_spinner = getWidget<SpinnerWidget>("ai-spinner");
m_ai_kart_label = getWidget<LabelWidget>("ai-text");
m_option = getWidget<CheckBoxWidget>("option");
m_record_race = getWidget<CheckBoxWidget>("record");
m_option->setState(false);
m_record_race->setState(false);
m_icon_bank = new irr::gui::STKModifiedSpriteBank( GUIEngine::getGUIEnv());
for (unsigned int i=0; i < kart_properties_manager->getNumberOfKarts(); i++)
{
const KartProperties* prop = kart_properties_manager->getKartById(i);
m_icon_bank->addTextureAsSprite(prop->getIconMaterial()->getTexture());
}
video::ITexture* kart_not_found = irr_driver->getTexture(
file_manager->getAsset(FileManager::GUI_ICON, "random_kart.png"));
m_icon_unknown_kart = m_icon_bank->addTextureAsSprite(kart_not_found);
m_highscore_label = getWidget<LabelWidget>("highscores");
for (unsigned int i=0;i<HIGHSCORE_COUNT;i++)
{
m_highscore_entries = getWidget<ListWidget>("highscore_entries");
}
GUIEngine::IconButtonWidget* screenshot = getWidget<IconButtonWidget>("screenshot");
screenshot->setFocusable(false);
screenshot->m_tab_stop = false;
m_is_soccer = false;
m_show_ffa_spinner = false;
} // loadedFromFile
void TrackInfoScreen::beforeAddingWidget()
{
m_is_soccer = RaceManager::get()->isSoccerMode();
m_is_lap_trial = RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_LAP_TRIAL;
m_show_ffa_spinner = RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_3_STRIKES
|| RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_FREE_FOR_ALL;
m_target_type_div->setCollapsed(!m_is_soccer && !m_show_ffa_spinner, this);
// show 'Number of blue team AI karts' if soccer
m_ai_blue_div->setCollapsed(!(RaceManager::get()->isSoccerMode()), this);
} // beforeAddingWidget
// ----------------------------------------------------------------------------
void TrackInfoScreen::setTrack(Track *track)
{
m_track = track;
} // setTrack
// ----------------------------------------------------------------------------
/** Initialised the display. The previous screen has to setup m_track before
* pushing this screen using TrackInfoScreen::getInstance()->setTrack().
*/
void TrackInfoScreen::init()
{
m_record_this_race = false;
const int max_arena_players = m_track->getMaxArenaPlayers();
const int local_players = RaceManager::get()->getNumLocalPlayers();
const bool has_laps = RaceManager::get()->modeHasLaps();
const bool has_highscores = RaceManager::get()->modeHasHighscores();
getWidget<LabelWidget>("name")->setText(m_track->getName(), false);
//I18N: when showing who is the author of track '%s'
//I18N: (place %s where the name of the author should appear)
getWidget<LabelWidget>("author")->setText( _("Track by %s", m_track->getDesigner()),
false );
LabelWidget* max_players = getWidget<LabelWidget>("max-arena-players");
max_players->setVisible(m_track->isArena());
if (m_track->isArena())
{
//I18N: the max players supported by an arena.
max_players->setText( _("Max players supported: %d", max_arena_players), false );
}
// ---- Track screenshot
GUIEngine::IconButtonWidget* screenshot = getWidget<IconButtonWidget>("screenshot");
ITexture* image = STKTexManager::getInstance()
->getTexture(m_track->getScreenshotFile(),
"While loading screenshot for track '%s':", m_track->getFilename());
if(!image)
{
image = STKTexManager::getInstance()->getTexture(GUIEngine::getSkin()->getThemedIcon("gui/icons/track_unknown.png"),
"While loading screenshot for track '%s':", m_track->getFilename());
}
if (image != NULL)
screenshot->setImage(image);
m_target_value_spinner->setVisible(false);
m_target_value_label->setVisible(false);
m_ai_blue_spinner->setVisible(false);
m_ai_blue_label->setVisible(false);
m_ai_blue_label->setActive(false);
// Number of AIs
// -------------
const bool has_AI =
( RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_3_STRIKES
|| RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_FREE_FOR_ALL
|| m_is_soccer ?
m_track->hasNavMesh() && (max_arena_players - local_players) > 0 :
RaceManager::get()->hasAI());
m_ai_kart_spinner->setVisible(has_AI);
m_ai_kart_label->setVisible(has_AI);
if (has_AI)
{
m_ai_kart_label->setText(_("Number of AI karts"), false); // change to label back for other modes
// Set the default number of AIs
int num_ai = int(UserConfigParams::m_num_karts_per_gamemode
[RaceManager::get()->getMinorMode()]) - local_players;
// Avoid negative numbers (which can happen if e.g. the number of karts
// in a previous race was lower than the number of players now.
if (num_ai < 0) num_ai = 0;
m_ai_kart_spinner->setValue(num_ai);
RaceManager::get()->setNumKarts(num_ai + local_players);
// Set the max karts supported based on the selected battle arena
if( RaceManager::get()->isBattleMode() || m_is_soccer)
{
m_ai_kart_spinner->setMax(max_arena_players - local_players);
m_ai_blue_spinner->setMax(max_arena_players - local_players);
}
else
m_ai_kart_spinner->setMax(stk_config->m_max_karts - local_players);
// A ftl race needs at least three karts to make any sense
if(RaceManager::get()->isFollowMode())
{
m_ai_kart_spinner->setMin(std::max(0, 3 - local_players));
}
// In single player battle, make sure there is at least one
// AI against the player. Soccer is handled in setSoccerWidgets
else if( local_players == 1
&& !UserConfigParams::m_artist_debug_mode
&& RaceManager::get()->isBattleMode())
{
m_ai_kart_spinner->setMin(1);
}
else
{
m_ai_kart_spinner->setMin(0);
m_ai_blue_spinner->setMin(0);
}
} // has_AI
else
RaceManager::get()->setNumKarts(local_players);
// Soccer options
// -------------
if (m_is_soccer)
setSoccerWidgets(has_AI);
// options for free-for-all and three strikes battle
// -------------
if (m_show_ffa_spinner)
{
m_target_type_label->setText(_("Game mode"), false);
m_target_type_spinner->clearLabels();
m_target_type_spinner->addLabel(_("3 Strikes Battle"));
m_target_type_spinner->addLabel(_("Free-For-All"));
m_target_type_spinner->setValue(UserConfigParams::m_use_ffa_mode ? 1 : 0);
m_target_value_label->setText(_("Maximum time (min.)"), false);
m_target_value_spinner->setValue(UserConfigParams::m_ffa_time_limit);
m_target_value_label->setVisible(UserConfigParams::m_use_ffa_mode);
m_target_value_spinner->setVisible(UserConfigParams::m_use_ffa_mode);
}
// Lap count m_lap_spinner
// -----------------------
if (has_laps)
{
m_target_value_spinner->setVisible(true);
m_target_value_label->setVisible(true);
if (UserConfigParams::m_artist_debug_mode)
m_target_value_spinner->setMin(0);
else
m_target_value_spinner->setMin(1);
m_target_value_spinner->setValue(m_track->getActualNumberOfLap());
RaceManager::get()->setNumLaps(m_target_value_spinner->getValue());
m_target_value_label->setText(_("Number of laps"), false);
}
if (m_is_lap_trial)
{
m_target_value_spinner->setVisible(true);
m_target_value_label->setVisible(true);
m_target_value_label->setText(_("Maximum time (min.)"), false);
m_target_value_spinner->setValue(UserConfigParams::m_lap_trial_time_limit);
}
// Reverse track or random item in arena
// -------------
const bool reverse_available = m_track->reverseAvailable()
&& !(RaceManager::get()->isEggHuntMode());
const bool random_item = m_track->hasNavMesh();
m_option->setVisible(reverse_available || random_item);
getWidget<LabelWidget>("option-text")->setVisible(reverse_available || random_item);
if (reverse_available)
{
//I18N: In the track info screen
getWidget<LabelWidget>("option-text")->setText(_("Drive in reverse"), false);
}
else if (random_item)
{
//I18N: In the track info screen
getWidget<LabelWidget>("option-text")->setText(_("Random item location"), false);
}
if (reverse_available)
m_option->setState(RaceManager::get()->getReverseTrack());
else if (random_item)
m_option->setState(UserConfigParams::m_random_arena_item);
else
m_option->setState(false);
// Record race or not
// -------------
const bool record_available = (RaceManager::get()->isTimeTrialMode() || RaceManager::get()->isEggHuntMode());
m_record_race->setVisible(record_available);
getWidget<LabelWidget>("record-race-text")->setVisible(record_available);
if (RaceManager::get()->isRecordingRace())
{
// isRecordingRace() is true when it's pre-set by ghost replay selection
// which force record this race
m_record_this_race = true;
m_record_race->setState(true);
m_record_race->setActive(false);
m_ai_kart_spinner->setValue(0);
m_ai_kart_spinner->setActive(false);
RaceManager::get()->setNumKarts(RaceManager::get()->getNumLocalPlayers());
UserConfigParams::m_num_karts_per_gamemode[RaceManager::get()->getMinorMode()] = RaceManager::get()->getNumLocalPlayers();
}
else if (record_available)
{
m_record_race->setActive(true);
m_record_race->setState(false);
m_ai_kart_spinner->setActive(true);
}
else
m_ai_kart_spinner->setActive(true);
// ---- High Scores
m_highscore_label->setText(_("High Scores"), false);
m_highscore_label->setVisible(has_highscores);
if (has_highscores)
{
m_icon_bank->setScale(1.0f / 128.0f);
m_icon_bank->setTargetIconSize(128, 128);
m_highscore_entries->setIcons(m_icon_bank, 1.2f);
m_highscore_entries->setVisible(has_highscores);
updateHighScores();
} //has_highscores
RibbonWidget* bt_start = getWidget<GUIEngine::RibbonWidget>("buttons");
bt_start->setFocusForPlayer(PLAYER_ID_GAME_MASTER);
} // init
void TrackInfoScreen::setSoccerWidgets(bool has_AI)
{
// Set up the spinner for the choice between time or goal limit
m_target_type_label->setText(_("Soccer game type"), false);
m_target_value_spinner->setVisible(true);
m_target_value_label->setVisible(true);
if (UserConfigParams::m_num_goals <= 0)
UserConfigParams::m_num_goals = UserConfigParams::m_num_goals.getDefaultValue();
if (UserConfigParams::m_soccer_time_limit <= 0)
UserConfigParams::m_soccer_time_limit = UserConfigParams::m_soccer_time_limit.getDefaultValue();
m_target_type_spinner->clearLabels();
m_target_type_spinner->addLabel(_("Time limit"));
m_target_type_spinner->addLabel(_("Goals limit"));
m_target_type_spinner->setValue(UserConfigParams::m_soccer_use_time_limit ? 0 : 1);
setSoccerTarget(UserConfigParams::m_soccer_use_time_limit /* true if time limit */);
if (has_AI)
{
const int max_arena_players = m_track->getMaxArenaPlayers();
const int local_players = RaceManager::get()->getNumLocalPlayers();
// Set up the spinners for the number of red and blue AIs
m_ai_blue_spinner->setVisible(true);
m_ai_blue_label->setVisible(true);
m_ai_blue_spinner->setActive(true);
m_ai_kart_label->setText(_("Number of red team AI karts"), false);
m_ai_kart_spinner->setValue(UserConfigParams::m_soccer_red_ai_num);
m_ai_blue_spinner->setValue(UserConfigParams::m_soccer_blue_ai_num);
// Check if there's any local players in both team
int num_blue_players = 0, num_red_players = 0;
for (int i = 0; i < local_players; i++)
{
KartTeam team = RaceManager::get()->getKartInfo(i).getKartTeam();
// Happens in profiling mode
if (team == KART_TEAM_NONE)
num_blue_players++; // No team will be set to blue
else
team == KART_TEAM_BLUE ? num_blue_players++ : num_red_players++;
}
const int max_num_ai = max_arena_players - local_players;
// Make sure each team has at least 1 (ai + player)
bool reuse_ai = ((num_blue_players + UserConfigParams::m_soccer_blue_ai_num) > 0) &&
((num_red_players + UserConfigParams::m_soccer_red_ai_num ) > 0) &&
((UserConfigParams::m_soccer_red_ai_num + UserConfigParams::m_soccer_blue_ai_num) <= max_num_ai);
// Try the saved values.
// If they can't be used, use default balanced values
if (!reuse_ai)
{
const int additional_blue = num_red_players - num_blue_players;
int num_blue_ai = (max_num_ai - additional_blue) / 2 + additional_blue;
int num_red_ai = (max_num_ai - additional_blue) / 2;
if ((max_num_ai + additional_blue)%2 == 1)
additional_blue < 0 ? num_red_ai++ : num_blue_ai++;
UserConfigParams::m_soccer_red_ai_num = num_red_ai;
UserConfigParams::m_soccer_blue_ai_num = num_blue_ai;
m_ai_kart_spinner->setValue(UserConfigParams::m_soccer_red_ai_num);
m_ai_blue_spinner->setValue(UserConfigParams::m_soccer_blue_ai_num);
}
if( local_players == 1
&& !UserConfigParams::m_artist_debug_mode)
{
if(num_blue_players == 0)
m_ai_blue_spinner->setMin(1);
if(num_red_players == 0)
m_ai_kart_spinner->setMin(1);
}
if(num_blue_players == 0 && !UserConfigParams::m_artist_debug_mode)
m_ai_kart_spinner->setMax(max_arena_players - local_players - 1);
if(num_red_players == 0 && !UserConfigParams::m_artist_debug_mode)
m_ai_blue_spinner->setMax(max_arena_players - local_players - 1);
}
} // setSoccerWidgets
void TrackInfoScreen::setSoccerTarget(bool time_limit)
{
if (time_limit)
{
m_target_value_label->setText(_("Maximum time (min.)"), false);
m_target_value_spinner->setValue(UserConfigParams::m_soccer_time_limit);
}
else
{
m_target_value_label->setText(_("Number of goals to win"), false);
m_target_value_spinner->setValue(UserConfigParams::m_num_goals);
}
} // setSoccerTarget
// ----------------------------------------------------------------------------
TrackInfoScreen::~TrackInfoScreen()
{
} // ~TrackInfoScreen
// ----------------------------------------------------------------------------
void TrackInfoScreen::tearDown()
{
// This will save number of ai karts per game mode
user_config->saveConfig();
m_highscore_entries->setIcons(NULL);
}
// ----------------------------------------------------------------------------
void TrackInfoScreen::unloaded()
{
delete m_icon_bank;
m_icon_bank = NULL;
} // unloaded
// ----------------------------------------------------------------------------
void TrackInfoScreen::updateHighScores()
{
if (!RaceManager::get()->modeHasHighscores())
return;
std::string game_mode_ident = RaceManager::getIdentOf( RaceManager::get()->getMinorMode() );
const Highscores::HighscoreType type = "HST_" + game_mode_ident;
Highscores* highscores =
highscore_manager->getHighscores(type,
RaceManager::get()->getNumberOfKarts(),
RaceManager::get()->getDifficulty(),
m_track->getIdent(),
RaceManager::get()->isLapTrialMode() ? m_target_value_spinner->getValue() * 60 : RaceManager::get()->getNumLaps(),
RaceManager::get()->getReverseTrack() );
const int amount = highscores->getNumberEntries();
std::string kart_name;
core::stringw name;
float time;
int time_precision = RaceManager::get()->currentModeTimePrecision();
m_highscore_entries->clear();
// fill highscore entries
for (int n=0; n<HIGHSCORE_COUNT; n++)
{
irr::core::stringw line;
int icon = -1;
// Check if this entry is filled or still empty
if (n < amount)
{
highscores->getEntry(n, kart_name, name, &time);
std::string highscore_string;
if (RaceManager::get()->isLapTrialMode())
highscore_string = std::to_string(static_cast<int>(time));
else
highscore_string = StringUtils::timeToString(time, time_precision);
for(unsigned int i=0; i<kart_properties_manager->getNumberOfKarts(); i++)
{
const KartProperties* prop = kart_properties_manager->getKartById(i);
if (kart_name == prop->getIdent())
{
icon = i;
break;
}
}
line = name + " " + core::stringw(highscore_string.c_str());
}
else
{
//I18N: for empty highscores entries
line = _("(Empty)");
}
if (icon == -1)
{
icon = m_icon_unknown_kart;
}
std::vector<GUIEngine::ListWidget::ListCell> row;
row.push_back(GUIEngine::ListWidget::ListCell(line.c_str(), icon, 5, false));
m_highscore_entries->addItem(StringUtils::toString(n), row);
}
} // updateHighScores
// ----------------------------------------------------------------------------
void TrackInfoScreen::onEnterPressedInternal()
{
RaceManager::get()->setRecordRace(m_record_this_race);
// Create a copy of member variables we still need, since they will
// not be accessible after dismiss:
const int num_laps = RaceManager::get()->modeHasLaps() ? m_target_value_spinner->getValue()
: -1;
const bool option_state = m_option == NULL ? false
: m_option->getState();
// Avoid negative lap numbers (after e.g. easter egg mode).
if(num_laps>=0)
m_track->setActualNumberOfLaps(num_laps);
if(m_track->hasNavMesh())
UserConfigParams::m_random_arena_item = option_state;
else
RaceManager::get()->setReverseTrack(option_state);
// Avoid invaild Ai karts number during switching game modes
const int max_arena_players = m_track->getMaxArenaPlayers();
const int local_players = RaceManager::get()->getNumLocalPlayers();
const bool has_AI =
(RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_3_STRIKES ||
RaceManager::get()->getMinorMode() == RaceManager::MINOR_MODE_FREE_FOR_ALL ||
RaceManager::get()->isSoccerMode() ?
m_track->hasNavMesh() && (max_arena_players - local_players) > 0 :
RaceManager::get()->hasAI());
int num_ai = 0;
if (has_AI)
{
num_ai = m_ai_kart_spinner->getValue();
if (m_is_soccer) // Soccer mode
num_ai += m_ai_blue_spinner->getValue();
}
const int selected_target_type = m_target_type_spinner->getValue();
const int selected_target_value = m_target_value_spinner->getValue();
const bool enable_ffa = m_show_ffa_spinner && selected_target_type != 0;
if (enable_ffa)
{
RaceManager::get()->setMinorMode(RaceManager::MINOR_MODE_FREE_FOR_ALL);
RaceManager::get()->setHitCaptureTime(0, static_cast<float>(selected_target_value) * 60);
}
if (m_is_soccer)
{
if (selected_target_type == 0)
RaceManager::get()->setTimeTarget(static_cast<float>(selected_target_value) * 60);
else
RaceManager::get()->setMaxGoal(selected_target_value);
}
if (m_is_lap_trial)
{
RaceManager::get()->setMinorMode(RaceManager::MINOR_MODE_LAP_TRIAL);
RaceManager::get()->setTimeTarget(static_cast<float>(selected_target_value) * 60);
}
if (UserConfigParams::m_num_karts_per_gamemode
[RaceManager::get()->getMinorMode()] != unsigned(local_players + num_ai))
{
UserConfigParams::m_num_karts_per_gamemode[RaceManager::get()->getMinorMode()] = local_players + num_ai;
}
// Disable accidentally unlocking of a challenge
PlayerManager::getCurrentPlayer()->setCurrentChallenge("");
RaceManager::get()->setNumKarts(num_ai + local_players);
if (m_is_soccer)
{
RaceManager::get()->setNumRedAI(m_ai_kart_spinner->getValue());
RaceManager::get()->setNumBlueAI(m_ai_blue_spinner->getValue());
}
RaceManager::get()->startSingleRace(m_track->getIdent(), num_laps, false);
} // onEnterPressedInternal
// ----------------------------------------------------------------------------
void TrackInfoScreen::eventCallback(Widget* widget, const std::string& name,
const int playerID)
{
if (name == "buttons")
{
const std::string &button = getWidget<GUIEngine::RibbonWidget>("buttons")
->getSelectionIDString(PLAYER_ID_GAME_MASTER);
if(button=="start")
onEnterPressedInternal();
else if(button=="back")
StateManager::get()->escapePressed();
}
else if (name == "back")
{
StateManager::get()->escapePressed();
}
else if (name == "target-type-spinner")
{
const bool target_value = m_target_type_spinner->getValue();
if (m_is_soccer)
{
const bool timed = target_value == 0;
UserConfigParams::m_soccer_use_time_limit = timed;
setSoccerTarget(timed /* true if time limit */);
}
else if (m_show_ffa_spinner)
{
const bool enable_ffa = target_value != 0;
UserConfigParams::m_use_ffa_mode = enable_ffa;
m_target_value_label->setVisible(enable_ffa);
m_target_value_spinner->setVisible(enable_ffa);
}
}
else if (name == "target-value-spinner")
{
if (m_is_soccer)
{
const bool timed = m_target_type_spinner->getValue() == 0;
if (timed)
UserConfigParams::m_soccer_time_limit = m_target_value_spinner->getValue();
else
UserConfigParams::m_num_goals = m_target_value_spinner->getValue();
}
else if (m_show_ffa_spinner)
{
const bool enable_ffa = m_target_type_spinner->getValue() != 0;
if (enable_ffa)
UserConfigParams::m_ffa_time_limit = m_target_value_spinner->getValue();
}
else if (m_is_lap_trial)
{
UserConfigParams::m_lap_trial_time_limit = m_target_value_spinner->getValue();
updateHighScores();
}
else
{
assert(RaceManager::get()->modeHasLaps());
const int num_laps = m_target_value_spinner->getValue();
RaceManager::get()->setNumLaps(num_laps);
UserConfigParams::m_num_laps = num_laps;
updateHighScores();
}
}
else if (name == "option")
{
if (m_track->hasNavMesh())
{
UserConfigParams::m_random_arena_item = m_option->getState();
}
else
{
RaceManager::get()->setReverseTrack(m_option->getState());
// Makes sure the highscores get swapped when clicking the 'reverse'
// checkbox.
updateHighScores();
}
}
else if (name == "record")
{
const bool record = m_record_race->getState();
m_record_this_race = record;
// Disable the AI when recording ghost race
m_ai_kart_spinner->setValue(0);
m_ai_kart_spinner->setActive(!record);
if (record)
{
RaceManager::get()->setNumKarts(RaceManager::get()->getNumLocalPlayers());
UserConfigParams::m_num_karts_per_gamemode[RaceManager::get()->getMinorMode()] = RaceManager::get()->getNumLocalPlayers();
}
}
else if (name=="ai-spinner")
{
if (m_is_soccer) // Soccer mode
{
soccerSpinnerUpdate(false /* blue spinner */);
}
else // Other modes
{
const int num_ai = m_ai_kart_spinner->getValue();
RaceManager::get()->setNumKarts( RaceManager::get()->getNumLocalPlayers() + num_ai );
UserConfigParams::m_num_karts_per_gamemode[RaceManager::get()->getMinorMode()] = RaceManager::get()->getNumLocalPlayers() + num_ai;
updateHighScores();
}
}
else if (name == "ai-blue-spinner" && m_is_soccer)
{
soccerSpinnerUpdate(true /* blue spinner */);
}
} // eventCallback
void TrackInfoScreen::soccerSpinnerUpdate(bool blue_spinner)
{
const int max_arena_players = m_track->getMaxArenaPlayers();
const int local_players = RaceManager::get()->getNumLocalPlayers();
const int num_ai = max_arena_players - local_players;
// Reduce the value of the other spinner if going over the max total num of AI
if(m_ai_kart_spinner->getValue() + m_ai_blue_spinner->getValue() > num_ai)
{
if (blue_spinner)
m_ai_kart_spinner->setValue(num_ai - m_ai_blue_spinner->getValue());
else
m_ai_blue_spinner->setValue(num_ai - m_ai_kart_spinner->getValue());
}
KartTeam all_team = KART_TEAM_NONE;
unsigned num_red = 0;
unsigned num_blue = 0;
for (unsigned i = 0; i < RaceManager::get()->getNumLocalPlayers(); i++)
{
RaceManager::get()->getKartInfo(i).getKartTeam() == KART_TEAM_RED ?
num_red++ : num_blue++;
}
if (num_red == 0)
all_team = KART_TEAM_BLUE;
else if (num_blue == 0)
all_team = KART_TEAM_RED;
// Need at least 1 kart at each side to avoid soccer ai crash,
// if there is any ai
if (blue_spinner)
{
if (all_team == KART_TEAM_BLUE &&
(m_ai_blue_spinner->getValue() == 0 || m_ai_kart_spinner->getValue() == 0))
{
if (m_ai_kart_spinner->getValue() == 0)
m_ai_kart_spinner->setValue(1);
}
if (all_team == KART_TEAM_RED && m_ai_blue_spinner->getValue() == 0 &&
m_ai_kart_spinner->getValue() != 0)
m_ai_blue_spinner->setValue(1);
}
else
{
if (all_team == KART_TEAM_RED &&
(m_ai_kart_spinner->getValue() == 0 || m_ai_blue_spinner->getValue() == 0))
{
if (m_ai_blue_spinner->getValue() == 0)
m_ai_blue_spinner->setValue(1);
}
if (all_team == KART_TEAM_BLUE && m_ai_kart_spinner->getValue() == 0 &&
m_ai_blue_spinner->getValue() != 0)
m_ai_kart_spinner->setValue(1);
}
UserConfigParams::m_soccer_red_ai_num = m_ai_kart_spinner->getValue();
UserConfigParams::m_soccer_blue_ai_num = m_ai_blue_spinner->getValue();
} // soccerSpinnerUpdate
// ----------------------------------------------------------------------------