Improved new race result gui - it now consists of two parts: first
the results of the just finished race, then an animated display of the GP standings (to test this please change the #undef USE_NEW_RACE_RESULT in race_result_gui.cpp). git-svn-id: svn+ssh://svn.code.sf.net/p/supertuxkart/code/main/trunk@5710 178a84e3-b1eb-0310-8ba1-8eac791a3b58
This commit is contained in:
parent
57b5c788ea
commit
0b35249b43
src
@ -99,7 +99,18 @@ void RaceManager::setLocalKartInfo(unsigned int player_id, const std::string& ka
|
||||
} // setLocalKartInfo
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
/** Returns the kart with a given GP rank (or NULL if no such kart exists).
|
||||
* \param n Rank (0<=n<num_karts) to look for.
|
||||
*/
|
||||
const Kart *RaceManager::getKartWithGPRank(unsigned int n)
|
||||
{
|
||||
for(unsigned int i=0; i<m_kart_status.size(); i++)
|
||||
if(m_kart_status[i].m_gp_rank == n)
|
||||
return World::getWorld()->getKart(i);
|
||||
return NULL;
|
||||
} // getKLartWithGPRank
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
int RaceManager::getLocalPlayerGPRank(const int playerID) const
|
||||
{
|
||||
const int amount = m_kart_status.size();
|
||||
@ -199,8 +210,13 @@ void RaceManager::startNew()
|
||||
|
||||
// First add the AI karts (randomly chosen)
|
||||
// ----------------------------------------
|
||||
int init_gp_rank = 0;
|
||||
for(unsigned int i=0; i<m_random_kart_list.size(); i++)
|
||||
m_kart_status.push_back(KartStatus(m_random_kart_list[i], i, -1, -1, KT_AI));
|
||||
{
|
||||
m_kart_status.push_back(KartStatus(m_random_kart_list[i], i, -1, -1,
|
||||
init_gp_rank, KT_AI));
|
||||
init_gp_rank ++;
|
||||
}
|
||||
|
||||
// Then the players, which start behind the AI karts
|
||||
// -------------------------------------------------
|
||||
@ -211,8 +227,9 @@ void RaceManager::startNew()
|
||||
m_kart_status.push_back(KartStatus(m_player_karts[i].getKartName(), i,
|
||||
m_player_karts[i].getLocalPlayerId(),
|
||||
m_player_karts[i].getGlobalPlayerId(),
|
||||
kt
|
||||
init_gp_rank, kt
|
||||
) );
|
||||
init_gp_rank ++;
|
||||
}
|
||||
|
||||
// Then start the race with the first track
|
||||
@ -322,12 +339,11 @@ void RaceManager::computeGPRanks()
|
||||
{
|
||||
// calculate the rank of each kart
|
||||
const unsigned int NUM_KARTS = getNumberOfKarts();
|
||||
|
||||
int *scores = new int[NUM_KARTS];
|
||||
int *position = new int[NUM_KARTS];
|
||||
double *race_time = new double[NUM_KARTS];
|
||||
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;
|
||||
int start=(race_manager->getMinorMode()==RaceManager::MINOR_MODE_FOLLOW_LEADER);
|
||||
for (unsigned int kart_id = start; kart_id < NUM_KARTS; ++kart_id)
|
||||
{
|
||||
position[kart_id] = kart_id;
|
||||
@ -394,7 +410,6 @@ void RaceManager::exitRace()
|
||||
// were finished, and not when a race is aborted.
|
||||
if (m_major_mode==MAJOR_MODE_GRAND_PRIX && m_track_number==(int)m_tracks.size())
|
||||
{
|
||||
computeGPRanks();
|
||||
unlock_manager->grandPrixFinished();
|
||||
|
||||
StateManager::get()->resetAndGoToScreen( MainMenuScreen::getInstance() );
|
||||
|
@ -188,20 +188,23 @@ private:
|
||||
std::string m_player_name; // for networked karts
|
||||
int m_score; // score for this kart
|
||||
int m_last_score; // needed for restart race, and for race results GUI.
|
||||
double m_overall_time; // sum of times of all races
|
||||
double m_last_time; // needed for restart
|
||||
float m_overall_time; // sum of times of all races
|
||||
float m_last_time; // needed for restart
|
||||
int m_prev_finish_pos; // previous finished position
|
||||
KartType m_kart_type; // Kart type: AI, player, network player etc.
|
||||
int m_local_player_id; // player controling the kart, for AI: -1
|
||||
int m_global_player_id; // global ID of player
|
||||
int m_gp_rank; // In GPs, at the end, will hold the overall rank of this kart.
|
||||
int m_gp_rank; // In GPs, at the end, will hold the overall
|
||||
// rank of this kart (0<=m_gp_rank < num_karts-1)
|
||||
|
||||
KartStatus(const std::string& ident, const int& prev_finish_pos,
|
||||
int local_player_id, int global_player_id, KartType kt) :
|
||||
int local_player_id, int global_player_id,
|
||||
int init_gp_rank, KartType kt) :
|
||||
m_ident(ident), m_score(0), m_last_score(0),
|
||||
m_overall_time(0.0f), m_last_time(0.0f),
|
||||
m_prev_finish_pos(prev_finish_pos), m_kart_type(kt),
|
||||
m_local_player_id(local_player_id),
|
||||
m_gp_rank(init_gp_rank),
|
||||
m_global_player_id(global_player_id)
|
||||
{}
|
||||
|
||||
@ -283,7 +286,7 @@ public:
|
||||
void computeGPRanks();
|
||||
int getKartGPRank(const int kart_id)
|
||||
const { return m_kart_status[kart_id].m_gp_rank; }
|
||||
|
||||
const Kart* getKartWithGPRank(unsigned int n);
|
||||
/** \return the GP rank of a local player, or -1 if the given player ID doesn't exist */
|
||||
int getLocalPlayerGPRank(const int playerID) const;
|
||||
|
||||
@ -295,7 +298,7 @@ public:
|
||||
const { return m_kart_status[k].m_local_player_id; }
|
||||
int getKartGlobalPlayerId(int k)
|
||||
const { return m_kart_status[k].m_global_player_id; }
|
||||
double getOverallTime(int kart) const { return m_kart_status[kart].m_overall_time;}
|
||||
float getOverallTime(int kart) const { return m_kart_status[kart].m_overall_time;}
|
||||
KartType getKartType(int kart) const { return m_kart_status[kart].m_kart_type;}
|
||||
int getCoinTarget() const { return m_coin_target; }
|
||||
int getPositionScore(int p) const { return m_score_for_position[p-1]; }
|
||||
|
@ -29,7 +29,7 @@
|
||||
*/
|
||||
RaceResultGUI::RaceResultGUI()
|
||||
{
|
||||
#define USE_NEW_RACE_RESULT
|
||||
#undef USE_NEW_RACE_RESULT
|
||||
|
||||
#ifndef USE_NEW_RACE_RESULT
|
||||
// FIXME: for now disable the new race result display
|
||||
@ -38,9 +38,9 @@ RaceResultGUI::RaceResultGUI()
|
||||
new RaceOverDialog(0.6f, 0.9f);
|
||||
return;
|
||||
#else
|
||||
determineLayout();
|
||||
determineTableLayout();
|
||||
m_timer = 0;
|
||||
m_animation_state = RR_BEGIN_FIRST_TABLE;
|
||||
m_animation_state = RR_INIT;
|
||||
#endif
|
||||
} // RaceResultGUI
|
||||
|
||||
@ -63,20 +63,21 @@ RaceResultGUI::~RaceResultGUI()
|
||||
*/
|
||||
void RaceResultGUI::nextPhase()
|
||||
{
|
||||
// FIXME: make sure that computeGPRanks is called here!!
|
||||
new RaceOverDialog(0.6f, 0.9f);
|
||||
} // nextPhase
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
/** This determines the layout, i.e. the size of all columns, font size etc.
|
||||
*/
|
||||
void RaceResultGUI::determineLayout()
|
||||
void RaceResultGUI::determineTableLayout()
|
||||
{
|
||||
m_font = GUIEngine::getFont();
|
||||
assert(m_font);
|
||||
m_was_monospace = m_font->getMonospaceDigits();
|
||||
m_font->setMonospaceDigits(true);
|
||||
|
||||
assert(m_font);
|
||||
World *world = World::getWorld();
|
||||
|
||||
std::vector<int> order;
|
||||
world->raceResultOrder(&order);
|
||||
|
||||
@ -103,10 +104,6 @@ void RaceResultGUI::determineLayout()
|
||||
core::dimension2d<u32> rect = m_font->getDimension(kart->getName().c_str());
|
||||
if(rect.Width > m_width_kart_name)
|
||||
m_width_kart_name = rect.Width;
|
||||
m_new_points.push_back(race_manager->getPositionScore(i+1));
|
||||
int p = race_manager->getKartPrevScore(order[i]);
|
||||
m_new_overall_points.push_back(p+m_new_points[i]);
|
||||
m_current_displayed_points.push_back((float)p);
|
||||
} // for i < order.size()
|
||||
|
||||
std::string max_time = StringUtils::timeToString(max_finish_time);
|
||||
@ -119,7 +116,7 @@ void RaceResultGUI::determineLayout()
|
||||
unsigned int num_karts = m_finish_time_string.size();
|
||||
|
||||
// Top pixel where to display text
|
||||
unsigned int top = (int)(0.15f*UserConfigParams::m_height);
|
||||
m_top = (int)(0.15f*UserConfigParams::m_height);
|
||||
|
||||
// Height of the result display
|
||||
unsigned int height = (int)(0.7f *UserConfigParams::m_height);
|
||||
@ -133,7 +130,7 @@ void RaceResultGUI::determineLayout()
|
||||
m_time_single_scroll = 0.2f;
|
||||
|
||||
// Time to rotate the entries to the proper GP position.
|
||||
m_time_rotation = 2.0f;
|
||||
m_time_rotation = 1.0f;
|
||||
|
||||
// The time the first phase is being displayed: add the start time
|
||||
// of the last kart to the duration of the scroll plus some time
|
||||
@ -162,10 +159,10 @@ void RaceResultGUI::determineLayout()
|
||||
|
||||
// Determine width of new points column
|
||||
core::dimension2du r_new_p = m_font->getDimension(L"+99");
|
||||
m_width_new_points = r_new_p.Width;
|
||||
m_width_new_points = r_new_p.Width;
|
||||
|
||||
// Determine width of overall points column
|
||||
core::dimension2du r_all_p = m_font->getDimension(L"9999");
|
||||
core::dimension2du r_all_p = m_font->getDimension(L"9999");
|
||||
unsigned int width_all_points = r_all_p.Width;
|
||||
|
||||
unsigned int table_width = m_width_icon + m_width_kart_name
|
||||
@ -177,27 +174,7 @@ void RaceResultGUI::determineLayout()
|
||||
+ 2 * m_width_column_space;
|
||||
|
||||
m_leftmost_column = (UserConfigParams::m_width - table_width)/2;
|
||||
if(race_manager->getMajorMode()==RaceManager::MAJOR_MODE_GRAND_PRIX)
|
||||
race_manager->computeGPRanks();
|
||||
m_start_at.clear();
|
||||
for(unsigned int i=0; i<num_karts; i++)
|
||||
{
|
||||
m_start_at.push_back(m_time_between_rows * i);
|
||||
m_x_pos.push_back((float)UserConfigParams::m_width);
|
||||
m_y_pos.push_back(top+i*m_distance_between_rows);
|
||||
|
||||
if(race_manager->getMajorMode()==RaceManager::MAJOR_MODE_GRAND_PRIX)
|
||||
{
|
||||
int gp_position = race_manager->getKartGPRank(order[i]);
|
||||
if(gp_position<(int)i)
|
||||
printf("X");
|
||||
m_radius.push_back( (gp_position-(int)i)*(int)m_distance_between_rows*0.5f);
|
||||
m_centre_point.push_back(top+(gp_position+i)*m_distance_between_rows*0.5f);
|
||||
}
|
||||
} // i < num_karts
|
||||
|
||||
|
||||
} // determineLayout
|
||||
} // determineTableLayout
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
/** Render all global parts of the race gui, i.e. things that are only
|
||||
@ -210,31 +187,57 @@ void RaceResultGUI::renderGlobal(float dt)
|
||||
World *world = World::getWorld();
|
||||
assert(world->getPhase()==WorldStatus::RESULT_DISPLAY_PHASE);
|
||||
unsigned int num_karts = world->getNumKarts();
|
||||
|
||||
|
||||
|
||||
// First: Update the finite state machine, and set
|
||||
// the current X and Y positions.
|
||||
// ===============================================
|
||||
switch(m_animation_state)
|
||||
{
|
||||
case RR_BEGIN_FIRST_TABLE: if(m_timer > m_time_overall_scroll)
|
||||
{
|
||||
m_animation_state = RR_INCREASE_POINTS;
|
||||
m_timer = 0;
|
||||
}
|
||||
break;
|
||||
case RR_INCREASE_POINTS: if(m_timer > 5)
|
||||
{
|
||||
m_animation_state = RR_RESORT_TABLE;
|
||||
m_timer = 0;
|
||||
}
|
||||
break;
|
||||
case RR_RESORT_TABLE: if(m_timer > m_time_rotation)
|
||||
{
|
||||
m_animation_state = RR_WAIT_TILL_END;
|
||||
// Make the new row permanent.
|
||||
for(unsigned int i=0; i<num_karts; i++)
|
||||
m_y_pos[i] = m_centre_point[i]
|
||||
+ m_radius[i];
|
||||
}
|
||||
break;
|
||||
case RR_INIT:
|
||||
for(unsigned int i=0; i<num_karts; i++)
|
||||
{
|
||||
m_start_at.push_back(m_time_between_rows * i);
|
||||
m_x_pos.push_back((float)UserConfigParams::m_width);
|
||||
m_y_pos.push_back((float)(m_top+i*m_distance_between_rows));
|
||||
}
|
||||
m_animation_state = RR_RACE_RESULT;
|
||||
break;
|
||||
case RR_RACE_RESULT:
|
||||
if(m_timer > m_time_overall_scroll)
|
||||
{
|
||||
if(race_manager->getMajorMode()!=RaceManager::MAJOR_MODE_GRAND_PRIX)
|
||||
{
|
||||
m_animation_state = RR_WAIT_TILL_END;
|
||||
break;
|
||||
}
|
||||
|
||||
determineGPLayout();
|
||||
m_animation_state = RR_OLD_GP_RESULTS;
|
||||
m_timer = 0;
|
||||
}
|
||||
break;
|
||||
case RR_OLD_GP_RESULTS:
|
||||
if(m_timer > m_time_overall_scroll)
|
||||
{
|
||||
m_animation_state = RR_INCREASE_POINTS;
|
||||
m_timer = 0;
|
||||
}
|
||||
case RR_INCREASE_POINTS:
|
||||
if(m_timer > 5)
|
||||
{
|
||||
m_animation_state = RR_RESORT_TABLE;
|
||||
m_timer = 0;
|
||||
}
|
||||
break;
|
||||
case RR_RESORT_TABLE:
|
||||
if(m_timer > m_time_rotation)
|
||||
{
|
||||
m_animation_state = RR_WAIT_TILL_END;
|
||||
// Make the new row permanent.
|
||||
for(unsigned int i=0; i<num_karts; i++)
|
||||
m_y_pos[i] = m_centre_point[i] - m_radius[i];
|
||||
}
|
||||
break;
|
||||
case RR_WAIT_TILL_END: break;
|
||||
} // switch
|
||||
|
||||
@ -245,7 +248,9 @@ void RaceResultGUI::renderGlobal(float dt)
|
||||
float y = (float)m_y_pos[i];
|
||||
switch(m_animation_state)
|
||||
{
|
||||
case RR_BEGIN_FIRST_TABLE:
|
||||
// Both states use the same scrolling:
|
||||
case RR_RACE_RESULT:
|
||||
case RR_OLD_GP_RESULTS:
|
||||
if(m_timer > m_start_at[i])
|
||||
{ // if active
|
||||
m_x_pos[i] -= dt*v;
|
||||
@ -272,6 +277,52 @@ void RaceResultGUI::renderGlobal(float dt)
|
||||
}
|
||||
} // renderGlobal
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
/** Determine the layout and fields for the GP table based on the previous
|
||||
* GP results.
|
||||
*/
|
||||
void RaceResultGUI::determineGPLayout()
|
||||
{
|
||||
unsigned int num_karts = m_kart_icons.size();
|
||||
m_current_displayed_points.resize(num_karts);
|
||||
m_new_points.resize(num_karts);
|
||||
std::vector<int> old_rank(num_karts, 0);
|
||||
for(unsigned int kart_id=0; kart_id<num_karts; kart_id++)
|
||||
{
|
||||
int rank = race_manager->getKartGPRank(kart_id);
|
||||
old_rank[kart_id] = rank;
|
||||
const Kart *kart = World::getWorld()->getKart(kart_id);
|
||||
m_kart_icons[rank] =
|
||||
kart->getKartProperties()->getIconMaterial()->getTexture();
|
||||
m_kart_names[rank] = kart->getName();
|
||||
float time = race_manager->getOverallTime(kart_id);
|
||||
m_finish_time_string[rank]
|
||||
= StringUtils::timeToString(time).c_str();
|
||||
m_start_at[rank] = m_time_between_rows * rank;
|
||||
m_x_pos[rank] = (float)UserConfigParams::m_width;
|
||||
m_y_pos[rank] = (float)(m_top+rank*m_distance_between_rows);
|
||||
int p = race_manager->getKartPrevScore(kart_id);
|
||||
m_current_displayed_points[rank] = (float)p;
|
||||
m_new_points[rank] = race_manager->getPositionScore(kart->getPosition());
|
||||
}
|
||||
|
||||
// Now update the GP ranks, and determine the new position
|
||||
// -------------------------------------------------------
|
||||
m_radius.resize(num_karts);
|
||||
m_centre_point.resize(num_karts);
|
||||
m_new_overall_points.resize(num_karts);
|
||||
race_manager->computeGPRanks();
|
||||
for(unsigned int i=0; i<num_karts; i++)
|
||||
{
|
||||
int j = old_rank[i];
|
||||
int gp_position = race_manager->getKartGPRank(i);
|
||||
m_radius[j] = (j-gp_position)*(int)m_distance_between_rows*0.5f;
|
||||
m_centre_point[j] = m_top+(gp_position+j)*m_distance_between_rows*0.5f;
|
||||
int p = race_manager->getKartScore(i);
|
||||
m_new_overall_points[j] = p;
|
||||
} // i < num_karts
|
||||
} // determineGPLayout
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
/** Displays the race results for a single kart.
|
||||
* \param n Index of the kart to be displayed.
|
||||
@ -312,9 +363,11 @@ void RaceResultGUI::displayOneEntry(unsigned int x, unsigned int y,
|
||||
m_font->draw(m_finish_time_string[n], dest_rect, color);
|
||||
m_font->setMonospaceDigits(mono);
|
||||
|
||||
// Only display points in GP mode.
|
||||
// ===============================
|
||||
if(race_manager->getMajorMode()!=RaceManager::MAJOR_MODE_GRAND_PRIX)
|
||||
|
||||
// Only display points in GP mode and when the GP results are displayed.
|
||||
// =====================================================================
|
||||
if(race_manager->getMajorMode()!=RaceManager::MAJOR_MODE_GRAND_PRIX ||
|
||||
m_animation_state == RR_RACE_RESULT)
|
||||
return;
|
||||
|
||||
// Draw the new points
|
||||
|
@ -44,13 +44,18 @@ private:
|
||||
float m_timer;
|
||||
|
||||
/** Finite state machine for the animations:
|
||||
BEGIN_FIRST_TABLE: The rows scroll into place.
|
||||
INCREASE_POINTS: The overall points are added up
|
||||
RESORT_TABLE: Resort the table so that it is now sorted by
|
||||
GP points.
|
||||
WAIT_TILL_END Some delay to wait for end, after a period it
|
||||
wii automatically end. */
|
||||
enum {RR_BEGIN_FIRST_TABLE,
|
||||
INIT: Set up data structures.
|
||||
RACE_RESULT: The rows scroll into place.
|
||||
OLD_GP_TABLE: Scroll new table into place, sorted by previous
|
||||
GP ranks
|
||||
INCREASE_POINTS: The overall points are added up
|
||||
RESORT_TABLE: Resort the table so that it is now sorted by
|
||||
GP points.
|
||||
WAIT_TILL_END Some delay to wait for end, after a period it
|
||||
wii automatically end. */
|
||||
enum {RR_INIT,
|
||||
RR_RACE_RESULT,
|
||||
RR_OLD_GP_RESULTS,
|
||||
RR_INCREASE_POINTS,
|
||||
RR_RESORT_TABLE,
|
||||
RR_WAIT_TILL_END}
|
||||
@ -63,7 +68,7 @@ private:
|
||||
std::vector<float> m_x_pos;
|
||||
|
||||
/** Currenct Y position. */
|
||||
std::vector<int> m_y_pos;
|
||||
std::vector<float> m_y_pos;
|
||||
|
||||
/** The center point when sorting the entries. */
|
||||
std::vector<float> m_centre_point;
|
||||
@ -131,6 +136,9 @@ private:
|
||||
table is aligned. */
|
||||
unsigned int m_leftmost_column;
|
||||
|
||||
/** Top-most pixel for first row. */
|
||||
unsigned int m_top;
|
||||
|
||||
/** Size of space between columns. */
|
||||
unsigned int m_width_column_space;
|
||||
|
||||
@ -142,7 +150,8 @@ private:
|
||||
|
||||
void displayOneEntry(unsigned int x, unsigned int y,
|
||||
unsigned int n, bool display_points);
|
||||
void determineLayout();
|
||||
void determineTableLayout();
|
||||
void determineGPLayout();
|
||||
|
||||
public:
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user