2005-07-24 14:12:37 +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/>.
|
|
|
|
*/
|
|
|
|
|
2008-05-06 15:11:33 +00:00
|
|
|
/** @file main_gui.cpp Handling of the main viewport. */
|
2007-03-03 04:04:22 +00:00
|
|
|
|
2004-08-09 17:04:08 +00:00
|
|
|
#include "stdafx.h"
|
2005-08-06 15:18:26 +00:00
|
|
|
#include "currency.h"
|
2005-02-10 05:43:30 +00:00
|
|
|
#include "spritecache.h"
|
2007-12-19 20:45:46 +00:00
|
|
|
#include "window_gui.h"
|
|
|
|
#include "window_func.h"
|
|
|
|
#include "textbuf_gui.h"
|
2008-01-09 09:45:45 +00:00
|
|
|
#include "viewport_func.h"
|
2007-12-21 21:50:46 +00:00
|
|
|
#include "command_func.h"
|
2008-05-24 10:15:06 +00:00
|
|
|
#include "console_gui.h"
|
(svn r5946) -Add: merged the TGP branch to mainline. TGP adds:
- New optional landscape generator (TerraGenesis Perlin)
- Load heightmaps (either BMP or PNG)
- Progress dialog while generating worlds (no longer a 'hanging' screen)
- New dialogs for NewGame, Create Scenario and Play Heightmap
- Easier to configure your landscape
- More things to configure (tree-placer, ..)
- Speedup of world generation
- New console command 'restart': restart the map EXACTLY as it was when you
first started it (needs a game made after or with this commit)
- New console command 'getseed': get the seed of your map and share it with
others (of course only works with generated maps)
- Many new, world generation related, things
- Many internal cleanups and rewrites
Many tnx to those people who helped making this:
Belugas, DaleStan, glx, KUDr, RichK67, Rubidium, and TrueLight (alfabetic)
Many tnx to those who helped testing:
Arnau, Bjarni, and tokai (alfabetic)
And to all other people who helped testing and sending comments / bugs
Stats: 673 lines changed, 3534 new lines, 79 new strings
2006-08-19 10:00:30 +00:00
|
|
|
#include "genworld.h"
|
2007-04-05 07:49:04 +00:00
|
|
|
#include "transparency_gui.h"
|
2011-02-07 22:37:22 +00:00
|
|
|
#include "map_func.h"
|
2007-12-29 09:24:26 +00:00
|
|
|
#include "sound_func.h"
|
2007-11-10 01:17:15 +00:00
|
|
|
#include "transparency.h"
|
2007-12-21 19:49:27 +00:00
|
|
|
#include "strings_func.h"
|
2007-12-23 10:56:02 +00:00
|
|
|
#include "zoom_func.h"
|
2008-09-30 20:51:04 +00:00
|
|
|
#include "company_base.h"
|
|
|
|
#include "company_func.h"
|
2008-03-23 07:35:29 +00:00
|
|
|
#include "toolbar_gui.h"
|
2008-05-16 07:08:04 +00:00
|
|
|
#include "statusbar_gui.h"
|
2008-05-07 13:10:15 +00:00
|
|
|
#include "tilehighlight_func.h"
|
2010-07-03 13:42:27 +00:00
|
|
|
#include "hotkeys.h"
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-12-10 23:38:26 +00:00
|
|
|
#include "saveload/saveload.h"
|
|
|
|
|
2008-01-13 01:21:35 +00:00
|
|
|
#include "network/network.h"
|
2008-05-30 18:20:26 +00:00
|
|
|
#include "network/network_func.h"
|
2008-01-13 01:21:35 +00:00
|
|
|
#include "network/network_gui.h"
|
2008-12-23 11:06:52 +00:00
|
|
|
#include "network/network_base.h"
|
2008-01-13 01:21:35 +00:00
|
|
|
|
|
|
|
#include "table/sprites.h"
|
|
|
|
#include "table/strings.h"
|
|
|
|
|
2006-12-29 13:59:48 +00:00
|
|
|
static int _rename_id = 1;
|
|
|
|
static int _rename_what = -1;
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-01-11 18:46:09 +00:00
|
|
|
void CcGiveMoney(const CommandCost &result, TileIndex tile, uint32 p1, uint32 p2)
|
2007-06-10 21:34:45 +00:00
|
|
|
{
|
2007-06-13 17:34:05 +00:00
|
|
|
#ifdef ENABLE_NETWORK
|
2010-01-11 18:46:09 +00:00
|
|
|
if (result.Failed() || !_settings_game.economy.give_money) return;
|
2007-06-10 21:34:45 +00:00
|
|
|
|
2010-04-12 14:12:47 +00:00
|
|
|
/* Inform the company of the action of one of its clients (controllers). */
|
2008-12-29 10:37:53 +00:00
|
|
|
char msg[64];
|
|
|
|
SetDParam(0, p2);
|
|
|
|
GetString(msg, STR_COMPANY_NAME, lastof(msg));
|
2007-06-10 21:34:45 +00:00
|
|
|
|
|
|
|
if (!_network_server) {
|
2008-12-29 10:37:53 +00:00
|
|
|
NetworkClientSendChat(NETWORK_ACTION_GIVE_MONEY, DESTTYPE_TEAM, p2, msg, p1);
|
2007-06-10 21:34:45 +00:00
|
|
|
} else {
|
2008-12-29 10:37:53 +00:00
|
|
|
NetworkServerSendChat(NETWORK_ACTION_GIVE_MONEY, DESTTYPE_TEAM, p2, msg, CLIENT_ID_SERVER, p1);
|
2007-06-10 21:34:45 +00:00
|
|
|
}
|
2007-06-13 17:34:05 +00:00
|
|
|
#endif /* ENABLE_NETWORK */
|
2007-06-10 21:34:45 +00:00
|
|
|
}
|
|
|
|
|
2006-12-30 01:17:53 +00:00
|
|
|
void HandleOnEditText(const char *str)
|
2005-05-13 17:09:05 +00:00
|
|
|
{
|
|
|
|
switch (_rename_what) {
|
2004-12-04 17:54:56 +00:00
|
|
|
#ifdef ENABLE_NETWORK
|
2007-03-03 04:04:22 +00:00
|
|
|
case 3: { // Give money, you can only give money in excess of loan
|
2009-11-02 15:39:03 +00:00
|
|
|
const Company *c = Company::GetIfValid(_local_company);
|
|
|
|
if (c == NULL) break;
|
2008-09-30 20:39:50 +00:00
|
|
|
Money money = min(c->money - c->current_loan, (Money)(atoi(str) / _currency->rate));
|
2004-12-28 09:24:02 +00:00
|
|
|
|
2007-11-19 18:38:10 +00:00
|
|
|
uint32 money_c = Clamp(ClampToI32(money), 0, 20000000); // Clamp between 20 million and 0
|
2004-12-28 09:24:02 +00:00
|
|
|
|
2007-03-03 04:04:22 +00:00
|
|
|
/* Give 'id' the money, and substract it from ourself */
|
2009-08-05 17:59:21 +00:00
|
|
|
DoCommandP(0, money_c, _rename_id, CMD_GIVE_MONEY | CMD_MSG(STR_ERROR_INSUFFICIENT_FUNDS), CcGiveMoney, str);
|
2010-08-01 18:53:30 +00:00
|
|
|
break;
|
|
|
|
}
|
2004-12-04 17:54:56 +00:00
|
|
|
#endif /* ENABLE_NETWORK */
|
2006-12-29 13:59:48 +00:00
|
|
|
default: NOT_REACHED();
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
2006-12-29 13:59:48 +00:00
|
|
|
|
|
|
|
_rename_id = _rename_what = -1;
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
|
|
|
|
2005-01-19 20:55:23 +00:00
|
|
|
/**
|
|
|
|
* This code is shared for the majority of the pushbuttons.
|
|
|
|
* Handles e.g. the pressing of a button (to build things), playing of click sound and sets certain parameters
|
|
|
|
*
|
|
|
|
* @param w Window which called the function
|
|
|
|
* @param widget ID of the widget (=button) that called this function
|
|
|
|
* @param cursor How should the cursor image change? E.g. cursor with depot image in it
|
|
|
|
* @param mode Tile highlighting mode, e.g. drawing a rectangle or a dot on the ground
|
|
|
|
* @return true if the button is clicked, false if it's unclicked
|
|
|
|
*/
|
2010-12-24 15:08:19 +00:00
|
|
|
bool HandlePlacePushButton(Window *w, int widget, CursorID cursor, HighLightStyle mode)
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
2007-12-02 14:29:48 +00:00
|
|
|
if (w->IsWidgetDisabled(widget)) return false;
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2005-11-09 11:32:15 +00:00
|
|
|
SndPlayFx(SND_15_BEEP);
|
2008-05-06 22:08:18 +00:00
|
|
|
w->SetDirty();
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2007-12-02 14:29:48 +00:00
|
|
|
if (w->IsWidgetLowered(widget)) {
|
2004-08-09 17:04:08 +00:00
|
|
|
ResetObjectToPlace();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-01-14 19:57:49 +00:00
|
|
|
SetObjectToPlace(cursor, PAL_NONE, mode, w->window_class, w->window_number);
|
2007-12-02 14:29:48 +00:00
|
|
|
w->LowerWidget(widget);
|
2004-08-09 17:04:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-11 18:46:09 +00:00
|
|
|
void CcPlaySound10(const CommandCost &result, TileIndex tile, uint32 p1, uint32 p2)
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
2010-01-11 18:46:09 +00:00
|
|
|
if (result.Succeeded()) SndPlayTileFx(SND_12_EXPLOSION, tile);
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
|
|
|
|
2004-12-04 17:54:56 +00:00
|
|
|
#ifdef ENABLE_NETWORK
|
2008-09-30 20:39:50 +00:00
|
|
|
void ShowNetworkGiveMoneyWindow(CompanyID company)
|
2004-12-04 17:54:56 +00:00
|
|
|
{
|
2008-09-30 20:39:50 +00:00
|
|
|
_rename_id = company;
|
2004-12-04 17:54:56 +00:00
|
|
|
_rename_what = 3;
|
2011-04-17 18:42:17 +00:00
|
|
|
ShowQueryString(STR_EMPTY, STR_NETWORK_GIVE_MONEY_CAPTION, 30, NULL, CS_NUMERAL, QSF_NONE);
|
2004-12-04 17:54:56 +00:00
|
|
|
}
|
|
|
|
#endif /* ENABLE_NETWORK */
|
|
|
|
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-10-23 18:28:20 +00:00
|
|
|
/**
|
|
|
|
* Zooms a viewport in a window in or out.
|
|
|
|
* @param how Zooming direction.
|
|
|
|
* @param w Window owning the viewport.
|
|
|
|
* @return Returns \c true if zooming step could be done, \c false if further zooming is not possible.
|
|
|
|
* @note No button handling or what so ever is done.
|
|
|
|
*/
|
2010-05-13 08:56:01 +00:00
|
|
|
bool DoZoomInOutWindow(ZoomStateChange how, Window *w)
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
|
|
|
ViewPort *vp;
|
2004-09-10 19:02:27 +00:00
|
|
|
|
2006-11-07 13:06:02 +00:00
|
|
|
assert(w != NULL);
|
2004-08-09 17:04:08 +00:00
|
|
|
vp = w->viewport;
|
|
|
|
|
2006-11-07 13:01:36 +00:00
|
|
|
switch (how) {
|
2010-05-13 10:00:51 +00:00
|
|
|
case ZOOM_NONE:
|
|
|
|
/* On initialisation of the viewport we don't do anything. */
|
|
|
|
break;
|
|
|
|
|
2006-11-07 13:01:36 +00:00
|
|
|
case ZOOM_IN:
|
2007-05-15 16:08:46 +00:00
|
|
|
if (vp->zoom == ZOOM_LVL_MIN) return false;
|
2007-12-17 22:04:07 +00:00
|
|
|
vp->zoom = (ZoomLevel)((int)vp->zoom - 1);
|
2006-11-07 13:01:36 +00:00
|
|
|
vp->virtual_width >>= 1;
|
|
|
|
vp->virtual_height >>= 1;
|
|
|
|
|
2008-05-11 15:08:44 +00:00
|
|
|
w->viewport->scrollpos_x += vp->virtual_width >> 1;
|
|
|
|
w->viewport->scrollpos_y += vp->virtual_height >> 1;
|
|
|
|
w->viewport->dest_scrollpos_x = w->viewport->scrollpos_x;
|
|
|
|
w->viewport->dest_scrollpos_y = w->viewport->scrollpos_y;
|
2010-08-04 08:58:45 +00:00
|
|
|
w->viewport->follow_vehicle = INVALID_VEHICLE;
|
2006-11-07 13:01:36 +00:00
|
|
|
break;
|
|
|
|
case ZOOM_OUT:
|
2007-05-15 16:08:46 +00:00
|
|
|
if (vp->zoom == ZOOM_LVL_MAX) return false;
|
2007-12-17 22:04:07 +00:00
|
|
|
vp->zoom = (ZoomLevel)((int)vp->zoom + 1);
|
2006-11-07 13:01:36 +00:00
|
|
|
|
2008-05-11 15:08:44 +00:00
|
|
|
w->viewport->scrollpos_x -= vp->virtual_width >> 1;
|
|
|
|
w->viewport->scrollpos_y -= vp->virtual_height >> 1;
|
|
|
|
w->viewport->dest_scrollpos_x = w->viewport->scrollpos_x;
|
|
|
|
w->viewport->dest_scrollpos_y = w->viewport->scrollpos_y;
|
2006-11-07 13:01:36 +00:00
|
|
|
|
|
|
|
vp->virtual_width <<= 1;
|
|
|
|
vp->virtual_height <<= 1;
|
2010-08-04 08:58:45 +00:00
|
|
|
w->viewport->follow_vehicle = INVALID_VEHICLE;
|
2006-11-07 13:01:36 +00:00
|
|
|
break;
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
2006-12-03 13:40:16 +00:00
|
|
|
if (vp != NULL) { // the vp can be null when how == ZOOM_NONE
|
2008-05-11 15:08:44 +00:00
|
|
|
vp->virtual_left = w->viewport->scrollpos_x;
|
|
|
|
vp->virtual_top = w->viewport->scrollpos_y;
|
2006-12-03 13:40:16 +00:00
|
|
|
}
|
2006-11-07 13:06:02 +00:00
|
|
|
/* Update the windows that have zoom-buttons to perhaps disable their buttons */
|
2009-09-30 21:00:35 +00:00
|
|
|
w->InvalidateData();
|
2004-08-09 17:04:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2004-09-10 19:02:27 +00:00
|
|
|
void ZoomInOrOutToCursorWindow(bool in, Window *w)
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
2008-03-23 07:35:29 +00:00
|
|
|
assert(w != NULL);
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2006-09-03 22:07:28 +00:00
|
|
|
if (_game_mode != GM_MENU) {
|
2008-03-23 07:35:29 +00:00
|
|
|
ViewPort *vp = w->viewport;
|
2010-07-24 10:14:39 +00:00
|
|
|
if ((in && vp->zoom == ZOOM_LVL_MIN) || (!in && vp->zoom == ZOOM_LVL_MAX)) return;
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2009-01-09 22:56:28 +00:00
|
|
|
Point pt = GetTileZoomCenterWindow(in, w);
|
2004-08-09 17:04:08 +00:00
|
|
|
if (pt.x != -1) {
|
2009-03-15 15:25:18 +00:00
|
|
|
ScrollWindowTo(pt.x, pt.y, -1, w, true);
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2004-09-03 19:59:05 +00:00
|
|
|
DoZoomInOutWindow(in ? ZOOM_IN : ZOOM_OUT, w);
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-01 12:42:27 +00:00
|
|
|
/** Widgets of the main window. */
|
|
|
|
enum MainWindowWidgets {
|
|
|
|
MW_VIEWPORT, ///< Main window viewport.
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct NWidgetPart _nested_main_window_widgets[] = {
|
|
|
|
NWidget(NWID_VIEWPORT, INVALID_COLOUR, MW_VIEWPORT), SetResize(1, 1),
|
|
|
|
};
|
|
|
|
|
|
|
|
static const WindowDesc _main_window_desc(
|
2009-11-28 14:42:35 +00:00
|
|
|
WDP_MANUAL, 0, 0,
|
2009-08-01 12:42:27 +00:00
|
|
|
WC_MAIN_WINDOW, WC_NONE,
|
|
|
|
0,
|
2009-11-15 10:26:01 +00:00
|
|
|
_nested_main_window_widgets, lengthof(_nested_main_window_widgets)
|
2009-08-01 12:42:27 +00:00
|
|
|
);
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
enum {
|
|
|
|
GHK_QUIT,
|
2010-12-10 23:38:26 +00:00
|
|
|
GHK_ABANDON,
|
2010-07-03 13:42:27 +00:00
|
|
|
GHK_CONSOLE,
|
|
|
|
GHK_BOUNDING_BOXES,
|
|
|
|
GHK_CENTER,
|
|
|
|
GHK_CENTER_ZOOM,
|
|
|
|
GHK_RESET_OBJECT_TO_PLACE,
|
|
|
|
GHK_DELETE_WINDOWS,
|
|
|
|
GHK_DELETE_NONVITAL_WINDOWS,
|
|
|
|
GHK_REFRESH_SCREEN,
|
|
|
|
GHK_CRASH,
|
|
|
|
GHK_MONEY,
|
|
|
|
GHK_UPDATE_COORDS,
|
|
|
|
GHK_TOGGLE_TRANSPARENCY,
|
|
|
|
GHK_TOGGLE_INVISIBILITY = GHK_TOGGLE_TRANSPARENCY + 9,
|
|
|
|
GHK_TRANSPARENCY_TOOLBAR = GHK_TOGGLE_INVISIBILITY + 8,
|
|
|
|
GHK_TRANSPARANCY,
|
|
|
|
GHK_CHAT,
|
|
|
|
GHK_CHAT_ALL,
|
|
|
|
GHK_CHAT_COMPANY,
|
2010-10-20 07:30:15 +00:00
|
|
|
GHK_CHAT_SERVER,
|
2010-07-03 13:42:27 +00:00
|
|
|
};
|
|
|
|
|
2008-05-13 14:59:50 +00:00
|
|
|
struct MainWindow : Window
|
2006-06-27 21:25:53 +00:00
|
|
|
{
|
2009-08-01 12:42:27 +00:00
|
|
|
MainWindow() : Window()
|
2008-05-13 14:59:50 +00:00
|
|
|
{
|
2009-08-01 12:42:27 +00:00
|
|
|
this->InitNested(&_main_window_desc, 0);
|
|
|
|
ResizeWindow(this, _screen.width, _screen.height);
|
|
|
|
|
2009-09-19 11:31:12 +00:00
|
|
|
NWidgetViewport *nvp = this->GetWidget<NWidgetViewport>(MW_VIEWPORT);
|
2009-08-01 12:42:27 +00:00
|
|
|
nvp->InitializeViewport(this, TileXY(32, 32), ZOOM_LVL_VIEWPORT);
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
2007-09-26 19:27:29 +00:00
|
|
|
|
2008-05-13 14:59:50 +00:00
|
|
|
virtual void OnPaint()
|
|
|
|
{
|
2009-08-01 12:42:27 +00:00
|
|
|
this->DrawWidgets();
|
2008-05-13 14:59:50 +00:00
|
|
|
if (_game_mode == GM_MENU) {
|
2010-07-04 12:49:51 +00:00
|
|
|
static const SpriteID title_sprites[] = {SPR_OTTD_O, SPR_OTTD_P, SPR_OTTD_E, SPR_OTTD_N, SPR_OTTD_T, SPR_OTTD_T, SPR_OTTD_D};
|
|
|
|
static const uint LETTER_SPACING = 10;
|
2010-08-25 19:02:51 +00:00
|
|
|
int name_width = (lengthof(title_sprites) - 1) * LETTER_SPACING;
|
2010-07-04 12:49:51 +00:00
|
|
|
|
|
|
|
for (uint i = 0; i < lengthof(title_sprites); i++) {
|
|
|
|
name_width += GetSpriteSize(title_sprites[i]).width;
|
|
|
|
}
|
|
|
|
int off_x = (this->width - name_width) / 2;
|
|
|
|
|
|
|
|
for (uint i = 0; i < lengthof(title_sprites); i++) {
|
|
|
|
DrawSprite(title_sprites[i], PAL_NONE, off_x, 50);
|
|
|
|
off_x += GetSpriteSize(title_sprites[i]).width + LETTER_SPACING;
|
|
|
|
}
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
|
|
|
}
|
2005-11-14 19:48:04 +00:00
|
|
|
|
2008-05-17 23:11:06 +00:00
|
|
|
virtual EventState OnKeyPress(uint16 key, uint16 keycode)
|
2008-05-13 14:59:50 +00:00
|
|
|
{
|
2010-07-03 13:42:27 +00:00
|
|
|
int num = CheckHotkeyMatch(global_hotkeys, keycode, this);
|
|
|
|
if (num == GHK_QUIT) {
|
|
|
|
HandleExitGameRequest();
|
|
|
|
return ES_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2008-05-13 14:59:50 +00:00
|
|
|
/* Disable all key shortcuts, except quit shortcuts when
|
2009-03-14 18:16:29 +00:00
|
|
|
* generating the world, otherwise they create threading
|
|
|
|
* problem during the generating, resulting in random
|
|
|
|
* assertions that are hard to trigger and debug */
|
2008-05-17 23:11:06 +00:00
|
|
|
if (IsGeneratingWorld()) return ES_NOT_HANDLED;
|
2005-11-14 19:48:04 +00:00
|
|
|
|
2010-12-10 23:38:26 +00:00
|
|
|
switch (num) {
|
|
|
|
case GHK_ABANDON:
|
|
|
|
/* No point returning from the main menu to itself */
|
|
|
|
if (_game_mode == GM_MENU) return ES_HANDLED;
|
|
|
|
if (_settings_client.gui.autosave_on_exit) {
|
|
|
|
DoExitSave();
|
|
|
|
_switch_mode = SM_MENU;
|
|
|
|
} else {
|
|
|
|
AskExitToGameMenu();
|
|
|
|
}
|
|
|
|
return ES_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
|
2010-12-10 23:38:26 +00:00
|
|
|
case GHK_CONSOLE:
|
|
|
|
IConsoleSwitch();
|
|
|
|
return ES_HANDLED;
|
|
|
|
|
|
|
|
case GHK_BOUNDING_BOXES:
|
2011-07-20 16:19:48 +00:00
|
|
|
ToggleBoundingBoxes();
|
2010-12-10 23:38:26 +00:00
|
|
|
return ES_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
|
|
|
|
2008-05-17 23:11:06 +00:00
|
|
|
if (_game_mode == GM_MENU) return ES_NOT_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
switch (num) {
|
|
|
|
case GHK_CENTER:
|
|
|
|
case GHK_CENTER_ZOOM: {
|
2008-05-13 14:59:50 +00:00
|
|
|
Point pt = GetTileBelowCursor();
|
|
|
|
if (pt.x != -1) {
|
2011-02-05 21:07:25 +00:00
|
|
|
bool instant = (num == GHK_CENTER_ZOOM && this->viewport->zoom != ZOOM_LVL_MIN);
|
2010-07-03 13:42:27 +00:00
|
|
|
if (num == GHK_CENTER_ZOOM) MaxZoomInOut(ZOOM_IN, this);
|
2011-02-05 21:07:25 +00:00
|
|
|
ScrollMainWindowTo(pt.x, pt.y, -1, instant);
|
2008-04-18 15:18:16 +00:00
|
|
|
}
|
2008-05-13 14:59:50 +00:00
|
|
|
break;
|
|
|
|
}
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_RESET_OBJECT_TO_PLACE: ResetObjectToPlace(); break;
|
|
|
|
case GHK_DELETE_WINDOWS: DeleteNonVitalWindows(); break;
|
|
|
|
case GHK_DELETE_NONVITAL_WINDOWS: DeleteAllNonVitalWindows(); break;
|
|
|
|
case GHK_REFRESH_SCREEN: MarkWholeScreenDirty(); break;
|
2008-04-03 19:55:40 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_CRASH: // Crash the game
|
2010-09-24 13:45:02 +00:00
|
|
|
*(volatile byte *)0 = 0;
|
2008-05-13 14:59:50 +00:00
|
|
|
break;
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_MONEY: // Gimme money
|
2011-02-08 20:47:48 +00:00
|
|
|
/* You can only cheat for money in single player. */
|
|
|
|
if (!_networking) DoCommandP(0, 10000000, 0, CMD_MONEY_CHEAT);
|
2008-05-13 14:59:50 +00:00
|
|
|
break;
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_UPDATE_COORDS: // Update the coordinates of all station signs
|
2009-12-13 19:33:07 +00:00
|
|
|
UpdateAllVirtCoords();
|
2008-05-13 14:59:50 +00:00
|
|
|
break;
|
2007-04-05 07:49:04 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_TOGGLE_TRANSPARENCY:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 1:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 2:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 3:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 4:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 5:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 6:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 7:
|
|
|
|
case GHK_TOGGLE_TRANSPARENCY + 8:
|
2008-05-13 14:59:50 +00:00
|
|
|
/* Transparency toggle hot keys */
|
2010-07-03 13:42:27 +00:00
|
|
|
ToggleTransparency((TransparencyOption)(num - GHK_TOGGLE_TRANSPARENCY));
|
2008-05-13 14:59:50 +00:00
|
|
|
MarkWholeScreenDirty();
|
|
|
|
break;
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_TOGGLE_INVISIBILITY:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 1:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 2:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 3:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 4:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 5:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 6:
|
|
|
|
case GHK_TOGGLE_INVISIBILITY + 7:
|
2008-05-13 14:59:50 +00:00
|
|
|
/* Invisibility toggle hot keys */
|
2010-07-03 13:42:27 +00:00
|
|
|
ToggleInvisibilityWithTransparency((TransparencyOption)(num - GHK_TOGGLE_INVISIBILITY));
|
2008-05-13 14:59:50 +00:00
|
|
|
MarkWholeScreenDirty();
|
|
|
|
break;
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_TRANSPARENCY_TOOLBAR:
|
2008-05-13 14:59:50 +00:00
|
|
|
ShowTransparencyToolbar();
|
|
|
|
break;
|
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_TRANSPARANCY:
|
2008-05-13 14:59:50 +00:00
|
|
|
ResetRestoreAllTransparency();
|
|
|
|
break;
|
2007-04-05 07:49:04 +00:00
|
|
|
|
2004-12-04 17:54:56 +00:00
|
|
|
#ifdef ENABLE_NETWORK
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_CHAT: // smart chat; send to team if any, otherwise to all
|
2008-05-13 14:59:50 +00:00
|
|
|
if (_networking) {
|
2011-04-22 15:54:16 +00:00
|
|
|
const NetworkClientInfo *cio = NetworkClientInfo::GetByClientID(_network_own_client_id);
|
2008-05-13 14:59:50 +00:00
|
|
|
if (cio == NULL) break;
|
|
|
|
|
2008-12-22 21:15:02 +00:00
|
|
|
ShowNetworkChatQueryWindow(NetworkClientPreferTeamChat(cio) ? DESTTYPE_TEAM : DESTTYPE_BROADCAST, cio->client_playas);
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
|
|
|
break;
|
2006-10-18 21:07:36 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_CHAT_ALL: // send text message to all clients
|
2008-05-13 14:59:50 +00:00
|
|
|
if (_networking) ShowNetworkChatQueryWindow(DESTTYPE_BROADCAST, 0);
|
|
|
|
break;
|
2007-01-14 21:32:13 +00:00
|
|
|
|
2010-07-03 13:42:27 +00:00
|
|
|
case GHK_CHAT_COMPANY: // send text to all team mates
|
2008-05-13 14:59:50 +00:00
|
|
|
if (_networking) {
|
2011-04-22 15:54:16 +00:00
|
|
|
const NetworkClientInfo *cio = NetworkClientInfo::GetByClientID(_network_own_client_id);
|
2008-05-13 14:59:50 +00:00
|
|
|
if (cio == NULL) break;
|
|
|
|
|
|
|
|
ShowNetworkChatQueryWindow(DESTTYPE_TEAM, cio->client_playas);
|
|
|
|
}
|
|
|
|
break;
|
2010-10-20 07:30:15 +00:00
|
|
|
|
|
|
|
case GHK_CHAT_SERVER: // send text to the server
|
|
|
|
if (_networking && !_network_server) {
|
|
|
|
ShowNetworkChatQueryWindow(DESTTYPE_CLIENT, CLIENT_ID_SERVER);
|
|
|
|
}
|
|
|
|
break;
|
2005-05-06 20:38:18 +00:00
|
|
|
#endif
|
2004-12-04 17:54:56 +00:00
|
|
|
|
2008-05-17 23:11:06 +00:00
|
|
|
default: return ES_NOT_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
2008-05-17 23:11:06 +00:00
|
|
|
return ES_HANDLED;
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
2006-08-21 14:34:59 +00:00
|
|
|
|
2008-05-13 14:59:50 +00:00
|
|
|
virtual void OnScroll(Point delta)
|
|
|
|
{
|
2011-01-22 14:51:36 +00:00
|
|
|
this->viewport->scrollpos_x += ScaleByZoom(delta.x, this->viewport->zoom);
|
|
|
|
this->viewport->scrollpos_y += ScaleByZoom(delta.y, this->viewport->zoom);
|
2008-05-13 14:59:50 +00:00
|
|
|
this->viewport->dest_scrollpos_x = this->viewport->scrollpos_x;
|
|
|
|
this->viewport->dest_scrollpos_y = this->viewport->scrollpos_y;
|
2011-01-15 15:36:58 +00:00
|
|
|
}
|
2006-08-21 14:59:58 +00:00
|
|
|
|
2008-05-13 14:59:50 +00:00
|
|
|
virtual void OnMouseWheel(int wheel)
|
|
|
|
{
|
2011-02-05 16:36:37 +00:00
|
|
|
if (_settings_client.gui.scrollwheel_scrolling == 0) {
|
|
|
|
ZoomInOrOutToCursorWindow(wheel < 0, this);
|
|
|
|
}
|
2008-05-13 14:59:50 +00:00
|
|
|
}
|
2006-11-07 13:06:02 +00:00
|
|
|
|
2009-10-24 14:53:55 +00:00
|
|
|
virtual void OnResize()
|
2009-08-01 12:42:27 +00:00
|
|
|
{
|
|
|
|
if (this->viewport != NULL) {
|
2009-09-19 11:31:12 +00:00
|
|
|
NWidgetViewport *nvp = this->GetWidget<NWidgetViewport>(MW_VIEWPORT);
|
2009-08-01 12:42:27 +00:00
|
|
|
nvp->UpdateViewportCoordinates(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-13 21:31:29 +00:00
|
|
|
/**
|
|
|
|
* Some data on this window has become invalid.
|
|
|
|
* @param data Information about the changed data.
|
|
|
|
* @param gui_scope Whether the call is done from GUI scope. You may not do everything when not in GUI scope. See #InvalidateWindowData() for details.
|
|
|
|
*/
|
|
|
|
virtual void OnInvalidateData(int data = 0, bool gui_scope = true)
|
2008-05-13 14:59:50 +00:00
|
|
|
{
|
2011-03-13 21:31:29 +00:00
|
|
|
if (!gui_scope) return;
|
2008-05-13 14:59:50 +00:00
|
|
|
/* Forward the message to the appropiate toolbar (ingame or scenario editor) */
|
2011-02-23 20:54:55 +00:00
|
|
|
InvalidateWindowData(WC_MAIN_TOOLBAR, 0, data, true);
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
2010-07-03 13:42:27 +00:00
|
|
|
|
|
|
|
static Hotkey<MainWindow> global_hotkeys[];
|
|
|
|
};
|
|
|
|
|
|
|
|
const uint16 _ghk_quit_keys[] = {'Q' | WKC_CTRL, 'Q' | WKC_META, 0};
|
2010-12-10 23:38:26 +00:00
|
|
|
const uint16 _ghk_abandon_keys[] = {'W' | WKC_CTRL, 'W' | WKC_META, 0};
|
2010-07-03 13:42:27 +00:00
|
|
|
const uint16 _ghk_chat_keys[] = {WKC_RETURN, 'T', 0};
|
|
|
|
const uint16 _ghk_chat_all_keys[] = {WKC_SHIFT | WKC_RETURN, WKC_SHIFT | 'T', 0};
|
|
|
|
const uint16 _ghk_chat_company_keys[] = {WKC_CTRL | WKC_RETURN, WKC_CTRL | 'T', 0};
|
2010-10-20 07:30:15 +00:00
|
|
|
const uint16 _ghk_chat_server_keys[] = {WKC_CTRL | WKC_SHIFT | WKC_RETURN, WKC_CTRL | WKC_SHIFT | 'T', 0};
|
2010-07-03 13:42:27 +00:00
|
|
|
|
|
|
|
Hotkey<MainWindow> MainWindow::global_hotkeys[] = {
|
|
|
|
Hotkey<MainWindow>(_ghk_quit_keys, "quit", GHK_QUIT),
|
2010-12-10 23:38:26 +00:00
|
|
|
Hotkey<MainWindow>(_ghk_abandon_keys, "abandon", GHK_ABANDON),
|
2010-07-03 13:42:27 +00:00
|
|
|
Hotkey<MainWindow>(WKC_BACKQUOTE, "console", GHK_CONSOLE),
|
|
|
|
Hotkey<MainWindow>('B' | WKC_CTRL, "bounding_boxes", GHK_BOUNDING_BOXES),
|
|
|
|
Hotkey<MainWindow>('C', "center", GHK_CENTER),
|
|
|
|
Hotkey<MainWindow>('Z', "center_zoom", GHK_CENTER_ZOOM),
|
|
|
|
Hotkey<MainWindow>(WKC_ESC, "reset_object_to_place", GHK_RESET_OBJECT_TO_PLACE),
|
|
|
|
Hotkey<MainWindow>(WKC_DELETE, "delete_windows", GHK_DELETE_WINDOWS),
|
|
|
|
Hotkey<MainWindow>(WKC_DELETE | WKC_SHIFT, "delete_all_windows", GHK_DELETE_NONVITAL_WINDOWS),
|
|
|
|
Hotkey<MainWindow>('R' | WKC_CTRL, "refresh_screen", GHK_REFRESH_SCREEN),
|
|
|
|
#if defined(_DEBUG)
|
|
|
|
Hotkey<MainWindow>('0' | WKC_ALT, "crash_game", GHK_CRASH),
|
|
|
|
Hotkey<MainWindow>('1' | WKC_ALT, "money", GHK_MONEY),
|
|
|
|
Hotkey<MainWindow>('2' | WKC_ALT, "update_coordinates", GHK_UPDATE_COORDS),
|
|
|
|
#endif
|
|
|
|
Hotkey<MainWindow>('1' | WKC_CTRL, "transparency_signs", GHK_TOGGLE_TRANSPARENCY),
|
|
|
|
Hotkey<MainWindow>('2' | WKC_CTRL, "transparency_trees", GHK_TOGGLE_TRANSPARENCY + 1),
|
|
|
|
Hotkey<MainWindow>('3' | WKC_CTRL, "transparency_houses", GHK_TOGGLE_TRANSPARENCY + 2),
|
|
|
|
Hotkey<MainWindow>('4' | WKC_CTRL, "transparency_industries", GHK_TOGGLE_TRANSPARENCY + 3),
|
|
|
|
Hotkey<MainWindow>('5' | WKC_CTRL, "transparency_buildings", GHK_TOGGLE_TRANSPARENCY + 4),
|
|
|
|
Hotkey<MainWindow>('6' | WKC_CTRL, "transparency_bridges", GHK_TOGGLE_TRANSPARENCY + 5),
|
|
|
|
Hotkey<MainWindow>('7' | WKC_CTRL, "transparency_structures", GHK_TOGGLE_TRANSPARENCY + 6),
|
|
|
|
Hotkey<MainWindow>('8' | WKC_CTRL, "transparency_catenary", GHK_TOGGLE_TRANSPARENCY + 7),
|
|
|
|
Hotkey<MainWindow>('9' | WKC_CTRL, "transparency_loading", GHK_TOGGLE_TRANSPARENCY + 8),
|
|
|
|
Hotkey<MainWindow>('1' | WKC_CTRL | WKC_SHIFT, "invisibility_signs", GHK_TOGGLE_INVISIBILITY),
|
|
|
|
Hotkey<MainWindow>('2' | WKC_CTRL | WKC_SHIFT, "invisibility_trees", GHK_TOGGLE_INVISIBILITY + 1),
|
|
|
|
Hotkey<MainWindow>('3' | WKC_CTRL | WKC_SHIFT, "invisibility_houses", GHK_TOGGLE_INVISIBILITY + 2),
|
|
|
|
Hotkey<MainWindow>('4' | WKC_CTRL | WKC_SHIFT, "invisibility_industries", GHK_TOGGLE_INVISIBILITY + 3),
|
|
|
|
Hotkey<MainWindow>('5' | WKC_CTRL | WKC_SHIFT, "invisibility_buildings", GHK_TOGGLE_INVISIBILITY + 4),
|
|
|
|
Hotkey<MainWindow>('6' | WKC_CTRL | WKC_SHIFT, "invisibility_bridges", GHK_TOGGLE_INVISIBILITY + 5),
|
|
|
|
Hotkey<MainWindow>('7' | WKC_CTRL | WKC_SHIFT, "invisibility_structures", GHK_TOGGLE_INVISIBILITY + 6),
|
|
|
|
Hotkey<MainWindow>('8' | WKC_CTRL | WKC_SHIFT, "invisibility_catenary", GHK_TOGGLE_INVISIBILITY + 7),
|
|
|
|
Hotkey<MainWindow>('X' | WKC_CTRL, "transparency_toolbar", GHK_TRANSPARENCY_TOOLBAR),
|
|
|
|
Hotkey<MainWindow>('X', "toggle_transparency", GHK_TRANSPARANCY),
|
|
|
|
#ifdef ENABLE_NETWORK
|
|
|
|
Hotkey<MainWindow>(_ghk_chat_keys, "chat", GHK_CHAT),
|
|
|
|
Hotkey<MainWindow>(_ghk_chat_all_keys, "chat_all", GHK_CHAT_ALL),
|
|
|
|
Hotkey<MainWindow>(_ghk_chat_company_keys, "chat_company", GHK_CHAT_COMPANY),
|
2010-10-20 07:30:15 +00:00
|
|
|
Hotkey<MainWindow>(_ghk_chat_server_keys, "chat_server", GHK_CHAT_SERVER),
|
2010-07-03 13:42:27 +00:00
|
|
|
#endif
|
|
|
|
HOTKEY_LIST_END(MainWindow)
|
2008-05-13 14:59:50 +00:00
|
|
|
};
|
2010-07-03 13:42:27 +00:00
|
|
|
Hotkey<MainWindow> *_global_hotkeys = MainWindow::global_hotkeys;
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2010-07-04 11:28:16 +00:00
|
|
|
/**
|
|
|
|
* Does the given keycode match one of the keycodes bound to 'quit game'?
|
|
|
|
* @param keycode The keycode that was pressed by the user.
|
|
|
|
* @return True iff the keycode matches one of the hotkeys for 'quit'.
|
|
|
|
*/
|
|
|
|
bool IsQuitKey(uint16 keycode)
|
|
|
|
{
|
|
|
|
int num = CheckHotkeyMatch<MainWindow>(_global_hotkeys, keycode, NULL);
|
|
|
|
return num == GHK_QUIT;
|
|
|
|
}
|
|
|
|
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2007-03-07 11:47:46 +00:00
|
|
|
void ShowSelectGameWindow();
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2011-05-01 19:14:12 +00:00
|
|
|
/**
|
|
|
|
* Initialise the default colours (remaps and the likes), and load the main windows.
|
|
|
|
*/
|
2009-02-09 02:57:15 +00:00
|
|
|
void SetupColoursAndInitialWindow()
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
2008-04-18 15:18:16 +00:00
|
|
|
for (uint i = 0; i != 16; i++) {
|
2009-02-09 02:57:15 +00:00
|
|
|
const byte *b = GetNonSprite(PALETTE_RECOLOUR_START + i, ST_RECOLOUR);
|
2005-02-11 17:12:11 +00:00
|
|
|
|
2004-08-09 17:04:08 +00:00
|
|
|
assert(b);
|
2006-08-29 19:26:13 +00:00
|
|
|
memcpy(_colour_gradient[i], b + 0xC6, sizeof(_colour_gradient[i]));
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
|
|
|
|
2009-08-01 12:42:27 +00:00
|
|
|
new MainWindow;
|
2006-12-30 01:52:09 +00:00
|
|
|
|
2007-03-03 04:04:22 +00:00
|
|
|
/* XXX: these are not done */
|
2005-11-14 19:48:04 +00:00
|
|
|
switch (_game_mode) {
|
2006-12-30 01:52:09 +00:00
|
|
|
default: NOT_REACHED();
|
|
|
|
case GM_MENU:
|
|
|
|
ShowSelectGameWindow();
|
|
|
|
break;
|
2004-08-09 17:04:08 +00:00
|
|
|
|
2006-12-30 01:52:09 +00:00
|
|
|
case GM_NORMAL:
|
|
|
|
case GM_EDITOR:
|
|
|
|
ShowVitalWindows();
|
|
|
|
break;
|
2004-08-09 17:04:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-01 19:14:12 +00:00
|
|
|
/**
|
|
|
|
* Show the vital in-game windows.
|
|
|
|
*/
|
2007-03-07 11:47:46 +00:00
|
|
|
void ShowVitalWindows()
|
2005-01-11 00:54:06 +00:00
|
|
|
{
|
2008-05-15 19:00:20 +00:00
|
|
|
AllocateToolbar();
|
2006-11-07 14:41:53 +00:00
|
|
|
|
(svn r5946) -Add: merged the TGP branch to mainline. TGP adds:
- New optional landscape generator (TerraGenesis Perlin)
- Load heightmaps (either BMP or PNG)
- Progress dialog while generating worlds (no longer a 'hanging' screen)
- New dialogs for NewGame, Create Scenario and Play Heightmap
- Easier to configure your landscape
- More things to configure (tree-placer, ..)
- Speedup of world generation
- New console command 'restart': restart the map EXACTLY as it was when you
first started it (needs a game made after or with this commit)
- New console command 'getseed': get the seed of your map and share it with
others (of course only works with generated maps)
- Many new, world generation related, things
- Many internal cleanups and rewrites
Many tnx to those people who helped making this:
Belugas, DaleStan, glx, KUDr, RichK67, Rubidium, and TrueLight (alfabetic)
Many tnx to those who helped testing:
Arnau, Bjarni, and tokai (alfabetic)
And to all other people who helped testing and sending comments / bugs
Stats: 673 lines changed, 3534 new lines, 79 new strings
2006-08-19 10:00:30 +00:00
|
|
|
/* Status bad only for normal games */
|
|
|
|
if (_game_mode == GM_EDITOR) return;
|
2005-01-11 00:54:06 +00:00
|
|
|
|
2008-04-18 15:13:45 +00:00
|
|
|
ShowStatusBar();
|
2005-01-11 00:54:06 +00:00
|
|
|
}
|
|
|
|
|
2008-04-07 20:28:58 +00:00
|
|
|
/**
|
|
|
|
* Size of the application screen changed.
|
|
|
|
* Adapt the game screen-size, re-allocate the open windows, and repaint everything
|
|
|
|
*/
|
2007-03-07 11:47:46 +00:00
|
|
|
void GameSizeChanged()
|
2004-08-09 17:04:08 +00:00
|
|
|
{
|
2008-06-16 19:38:41 +00:00
|
|
|
_cur_resolution.width = _screen.width;
|
|
|
|
_cur_resolution.height = _screen.height;
|
2004-08-09 17:04:08 +00:00
|
|
|
ScreenSizeChanged();
|
2008-04-18 21:49:38 +00:00
|
|
|
RelocateAllWindows(_screen.width, _screen.height);
|
2004-08-09 17:04:08 +00:00
|
|
|
MarkWholeScreenDirty();
|
|
|
|
}
|