2009-01-03 18:44:20 +00:00
|
|
|
/* $Id$ */
|
|
|
|
|
2009-08-21 20:21:05 +00:00
|
|
|
/*
|
|
|
|
* This file is part of OpenTTD.
|
|
|
|
* OpenTTD 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, version 2.
|
|
|
|
* OpenTTD 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 OpenTTD. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2009-01-03 18:44:20 +00:00
|
|
|
/** @file highscore_gui.cpp Definition of the HighScore and EndGame windows */
|
|
|
|
|
2010-01-15 16:41:15 +00:00
|
|
|
#include "stdafx.h"
|
2009-01-03 18:44:20 +00:00
|
|
|
#include "highscore.h"
|
|
|
|
#include "table/strings.h"
|
|
|
|
#include "gfx_func.h"
|
|
|
|
#include "table/sprites.h"
|
|
|
|
#include "window_gui.h"
|
|
|
|
#include "window_func.h"
|
|
|
|
#include "network/network.h"
|
|
|
|
#include "command_func.h"
|
|
|
|
#include "company_func.h"
|
|
|
|
#include "company_base.h"
|
|
|
|
#include "strings_func.h"
|
|
|
|
#include "openttd.h"
|
2010-07-04 11:28:16 +00:00
|
|
|
#include "hotkeys.h"
|
2009-01-03 18:44:20 +00:00
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
enum HighscoreWidgets {
|
|
|
|
HSW_BACKGROUND,
|
|
|
|
};
|
|
|
|
|
2009-01-03 18:44:20 +00:00
|
|
|
struct EndGameHighScoreBaseWindow : Window {
|
|
|
|
uint32 background_img;
|
|
|
|
int8 rank;
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
EndGameHighScoreBaseWindow(const WindowDesc *desc) : Window()
|
2009-01-03 18:44:20 +00:00
|
|
|
{
|
2009-10-17 20:38:27 +00:00
|
|
|
this->InitNested(desc);
|
|
|
|
ResizeWindow(this, _screen.width - this->width, _screen.height - this->height);
|
2009-01-03 18:44:20 +00:00
|
|
|
}
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
/* Always draw a maximized window and within it the centered background */
|
|
|
|
void SetupHighScoreEndWindow()
|
2009-01-03 18:44:20 +00:00
|
|
|
{
|
2009-10-17 20:38:27 +00:00
|
|
|
/* Resize window to "full-screen". */
|
|
|
|
if (this->width != _screen.width || this->height != _screen.height) ResizeWindow(this, _screen.width - this->width, _screen.height - this->height);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
this->DrawWidgets();
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
Point pt = this->GetTopLeft640x480();
|
2009-01-03 18:44:20 +00:00
|
|
|
/* Center Highscore/Endscreen background */
|
|
|
|
for (uint i = 0; i < 10; i++) { // the image is split into 10 50px high parts
|
2009-10-17 20:38:27 +00:00
|
|
|
DrawSprite(this->background_img + i, PAL_NONE, pt.x, pt.y + (i * 50));
|
2009-01-03 18:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
/** Return the coordinate of the screen such that a window of 640x480 is centered at the screen. */
|
|
|
|
Point GetTopLeft640x480()
|
|
|
|
{
|
|
|
|
Point pt = {max(0, (_screen.width / 2) - (640 / 2)), max(0, (_screen.height / 2) - (480 / 2))};
|
|
|
|
return pt;
|
|
|
|
}
|
|
|
|
|
2010-01-30 18:34:48 +00:00
|
|
|
virtual void OnClick(Point pt, int widget, int click_count)
|
2009-01-03 18:44:20 +00:00
|
|
|
{
|
|
|
|
delete this;
|
|
|
|
}
|
2010-01-02 16:05:27 +00:00
|
|
|
|
|
|
|
virtual EventState OnKeyPress(uint16 key, uint16 keycode)
|
|
|
|
{
|
2010-07-04 11:28:16 +00:00
|
|
|
/* All keys are 'handled' by this window but we want to make
|
|
|
|
* sure that 'quit' still works correctly. Not handling the
|
|
|
|
* quit key is enough so the main toolbar can handle it. */
|
|
|
|
if (IsQuitKey(keycode)) return ES_NOT_HANDLED;
|
|
|
|
|
2010-01-02 16:05:27 +00:00
|
|
|
switch (keycode) {
|
|
|
|
/* Keys for telling we want to go on */
|
|
|
|
case WKC_RETURN:
|
|
|
|
case WKC_ESC:
|
|
|
|
case WKC_SPACE:
|
|
|
|
delete this;
|
|
|
|
return ES_HANDLED;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* We want to handle all keys; we don't want windows in
|
|
|
|
* the background to open. Especially the ones that do
|
|
|
|
* locate themselves based on the status-/toolbars. */
|
|
|
|
return ES_HANDLED;
|
|
|
|
}
|
|
|
|
}
|
2009-01-03 18:44:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/** End game window shown at the end of the game */
|
|
|
|
struct EndGameWindow : EndGameHighScoreBaseWindow {
|
|
|
|
EndGameWindow(const WindowDesc *desc) : EndGameHighScoreBaseWindow(desc)
|
|
|
|
{
|
|
|
|
/* Pause in single-player to have a look at the highscore at your own leisure */
|
2009-05-06 15:06:57 +00:00
|
|
|
if (!_networking) DoCommandP(0, PM_PAUSED_NORMAL, 1, CMD_PAUSE);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
this->background_img = SPR_TYCOON_IMG1_BEGIN;
|
|
|
|
|
|
|
|
if (_local_company != COMPANY_SPECTATOR) {
|
2009-05-16 23:34:14 +00:00
|
|
|
const Company *c = Company::Get(_local_company);
|
2009-01-03 18:44:20 +00:00
|
|
|
if (c->old_economy[0].performance_history == SCORE_MAX) {
|
|
|
|
this->background_img = SPR_TYCOON_IMG2_BEGIN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* In a network game show the endscores of the custom difficulty 'network' which is the last one
|
|
|
|
* as well as generate a TOP5 of that game, and not an all-time top5. */
|
|
|
|
if (_networking) {
|
|
|
|
this->window_number = lengthof(_highscore_table) - 1;
|
|
|
|
this->rank = SaveHighScoreValueNetwork();
|
|
|
|
} else {
|
|
|
|
/* in single player _local company is always valid */
|
2009-05-16 23:34:14 +00:00
|
|
|
const Company *c = Company::Get(_local_company);
|
2009-01-03 18:44:20 +00:00
|
|
|
this->window_number = _settings_game.difficulty.diff_level;
|
|
|
|
this->rank = SaveHighScoreValue(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
MarkWholeScreenDirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
~EndGameWindow()
|
|
|
|
{
|
2009-05-06 15:06:57 +00:00
|
|
|
if (!_networking) DoCommandP(0, PM_PAUSED_NORMAL, 0, CMD_PAUSE); // unpause
|
2009-01-03 18:44:20 +00:00
|
|
|
ShowHighscoreTable(this->window_number, this->rank);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void OnPaint()
|
|
|
|
{
|
2009-10-17 20:38:27 +00:00
|
|
|
this->SetupHighScoreEndWindow();
|
|
|
|
Point pt = this->GetTopLeft640x480();
|
2009-01-03 18:44:20 +00:00
|
|
|
|
2009-05-18 16:21:28 +00:00
|
|
|
const Company *c = Company::GetIfValid(_local_company);
|
|
|
|
if (c == NULL) return;
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
/* We need to get performance from last year because the image is shown
|
|
|
|
* at the start of the new year when these things have already been copied */
|
|
|
|
if (this->background_img == SPR_TYCOON_IMG2_BEGIN) { // Tycoon of the century \o/
|
|
|
|
SetDParam(0, c->index);
|
|
|
|
SetDParam(1, c->index);
|
|
|
|
SetDParam(2, EndGameGetPerformanceTitleFromValue(c->old_economy[0].performance_history));
|
2009-11-23 13:18:20 +00:00
|
|
|
DrawStringMultiLine(pt.x + 15, pt.x + 640 - 25, pt.y + 90, pt.y + 160, STR_HIGHSCORE_PRESIDENT_OF_COMPANY_ACHIEVES_STATUS, TC_FROMSTRING, SA_CENTER);
|
2009-01-03 18:44:20 +00:00
|
|
|
} else {
|
|
|
|
SetDParam(0, c->index);
|
|
|
|
SetDParam(1, EndGameGetPerformanceTitleFromValue(c->old_economy[0].performance_history));
|
2009-11-23 13:18:20 +00:00
|
|
|
DrawStringMultiLine(pt.x + 36, pt.x + 640, pt.y + 140, pt.y + 206, STR_HIGHSCORE_COMPANY_ACHIEVES_STATUS, TC_FROMSTRING, SA_CENTER);
|
2009-01-03 18:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct HighScoreWindow : EndGameHighScoreBaseWindow {
|
|
|
|
HighScoreWindow(const WindowDesc *desc, int difficulty, int8 ranking) : EndGameHighScoreBaseWindow(desc)
|
|
|
|
{
|
|
|
|
/* pause game to show the chart */
|
2009-05-06 15:06:57 +00:00
|
|
|
if (!_networking) DoCommandP(0, PM_PAUSED_NORMAL, 1, CMD_PAUSE);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
/* Close all always on-top windows to get a clean screen */
|
|
|
|
if (_game_mode != GM_MENU) HideVitalWindows();
|
|
|
|
|
|
|
|
MarkWholeScreenDirty();
|
|
|
|
this->window_number = difficulty; // show highscore chart for difficulty...
|
|
|
|
this->background_img = SPR_HIGHSCORE_CHART_BEGIN; // which background to show
|
|
|
|
this->rank = ranking;
|
|
|
|
}
|
|
|
|
|
|
|
|
~HighScoreWindow()
|
|
|
|
{
|
|
|
|
if (_game_mode != GM_MENU) ShowVitalWindows();
|
|
|
|
|
2009-05-06 15:06:57 +00:00
|
|
|
if (!_networking) DoCommandP(0, PM_PAUSED_NORMAL, 0, CMD_PAUSE); // unpause
|
2009-01-03 18:44:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void OnPaint()
|
|
|
|
{
|
|
|
|
const HighScore *hs = _highscore_table[this->window_number];
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
this->SetupHighScoreEndWindow();
|
|
|
|
Point pt = this->GetTopLeft640x480();
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
SetDParam(0, ORIGINAL_END_YEAR);
|
2009-04-21 23:40:56 +00:00
|
|
|
SetDParam(1, this->window_number + STR_DIFFICULTY_LEVEL_EASY);
|
2009-10-17 20:38:27 +00:00
|
|
|
DrawStringMultiLine(pt.x + 70, pt.x + 570, pt.y, pt.y + 140, !_networking ? STR_HIGHSCORE_TOP_COMPANIES_WHO_REACHED : STR_HIGHSCORE_TOP_COMPANIES_NETWORK_GAME, TC_FROMSTRING, SA_CENTER);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
/* Draw Highscore peepz */
|
|
|
|
for (uint8 i = 0; i < lengthof(_highscore_table[0]); i++) {
|
|
|
|
SetDParam(0, i + 1);
|
2009-10-17 20:38:27 +00:00
|
|
|
DrawString(pt.x + 40, pt.x + 600, pt.y + 140 + (i * 55), STR_HIGHSCORE_POSITION);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
if (hs[i].company[0] != '\0') {
|
|
|
|
TextColour colour = (this->rank == i) ? TC_RED : TC_BLACK; // draw new highscore in red
|
|
|
|
|
2009-10-17 20:38:27 +00:00
|
|
|
DrawString(pt.x + 71, pt.x + 569, pt.y + 140 + (i * 55), hs[i].company, colour);
|
2009-01-03 18:44:20 +00:00
|
|
|
SetDParam(0, hs[i].title);
|
|
|
|
SetDParam(1, hs[i].score);
|
2009-11-23 13:18:20 +00:00
|
|
|
DrawString(pt.x + 71, pt.x + 569, pt.y + 140 + FONT_HEIGHT_LARGE + (i * 55), STR_HIGHSCORE_STATS, colour);
|
2009-01-03 18:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-04-11 18:10:32 +00:00
|
|
|
static const NWidgetPart _nested_highscore_widgets[] = {
|
2009-10-17 20:38:27 +00:00
|
|
|
NWidget(WWT_PANEL, COLOUR_END, HSW_BACKGROUND), SetMinimalSize(641, 481), SetResize(1, 1), EndContainer(),
|
2009-04-11 18:10:32 +00:00
|
|
|
};
|
|
|
|
|
2009-03-15 15:12:06 +00:00
|
|
|
static const WindowDesc _highscore_desc(
|
2009-11-28 15:01:49 +00:00
|
|
|
WDP_MANUAL, 0, 0,
|
2009-01-03 18:44:20 +00:00
|
|
|
WC_HIGHSCORE, WC_NONE,
|
|
|
|
0,
|
2009-11-15 10:26:01 +00:00
|
|
|
_nested_highscore_widgets, lengthof(_nested_highscore_widgets)
|
2009-03-15 15:12:06 +00:00
|
|
|
);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
2009-03-15 15:12:06 +00:00
|
|
|
static const WindowDesc _endgame_desc(
|
2009-11-28 15:01:49 +00:00
|
|
|
WDP_MANUAL, 0, 0,
|
2009-01-03 18:44:20 +00:00
|
|
|
WC_ENDSCREEN, WC_NONE,
|
|
|
|
0,
|
2009-11-15 10:26:01 +00:00
|
|
|
_nested_highscore_widgets, lengthof(_nested_highscore_widgets)
|
2009-03-15 15:12:06 +00:00
|
|
|
);
|
2009-01-03 18:44:20 +00:00
|
|
|
|
2010-08-01 19:22:34 +00:00
|
|
|
/**
|
|
|
|
* Show the highscore table for a given difficulty. When called from
|
2009-01-03 18:44:20 +00:00
|
|
|
* endgame ranking is set to the top5 element that was newly added
|
2010-08-01 19:44:49 +00:00
|
|
|
* and is thus highlighted
|
|
|
|
*/
|
2009-01-03 18:44:20 +00:00
|
|
|
void ShowHighscoreTable(int difficulty, int8 ranking)
|
|
|
|
{
|
|
|
|
DeleteWindowByClass(WC_HIGHSCORE);
|
|
|
|
new HighScoreWindow(&_highscore_desc, difficulty, ranking);
|
|
|
|
}
|
|
|
|
|
2010-08-01 19:22:34 +00:00
|
|
|
/**
|
|
|
|
* Show the endgame victory screen in 2050. Update the new highscore
|
2010-08-01 19:44:49 +00:00
|
|
|
* if it was high enough
|
|
|
|
*/
|
2009-01-03 18:44:20 +00:00
|
|
|
void ShowEndGameChart()
|
|
|
|
{
|
2009-02-23 01:51:19 +00:00
|
|
|
/* Dedicated server doesn't need the highscore window and neither does -v null. */
|
2009-05-17 01:00:56 +00:00
|
|
|
if (_network_dedicated || (!_networking && !Company::IsValidID(_local_company))) return;
|
2009-01-03 18:44:20 +00:00
|
|
|
|
|
|
|
HideVitalWindows();
|
|
|
|
DeleteWindowByClass(WC_ENDSCREEN);
|
|
|
|
new EndGameWindow(&_endgame_desc);
|
|
|
|
}
|