summaryrefslogtreecommitdiff
path: root/src/terraform_gui.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/terraform_gui.cpp')
-rw-r--r--src/terraform_gui.cpp285
1 files changed, 285 insertions, 0 deletions
diff --git a/src/terraform_gui.cpp b/src/terraform_gui.cpp
new file mode 100644
index 000000000..8d6e1276a
--- /dev/null
+++ b/src/terraform_gui.cpp
@@ -0,0 +1,285 @@
+/* $Id$ */
+
+#include "stdafx.h"
+#include "openttd.h"
+#include "bridge_map.h"
+#include "clear_map.h"
+#include "table/sprites.h"
+#include "table/strings.h"
+#include "functions.h"
+#include "player.h"
+#include "tile.h"
+#include "window.h"
+#include "gui.h"
+#include "viewport.h"
+#include "gfx.h"
+#include "sound.h"
+#include "command.h"
+#include "vehicle.h"
+#include "signs.h"
+#include "variables.h"
+
+void CcTerraform(bool success, TileIndex tile, uint32 p1, uint32 p2)
+{
+ if (success) {
+ SndPlayTileFx(SND_1F_SPLAT, tile);
+ } else {
+ SetRedErrorSquare(_terraform_err_tile);
+ }
+}
+
+
+/** Scenario editor command that generates desert areas */
+static void GenerateDesertArea(TileIndex end, TileIndex start)
+{
+ int size_x, size_y;
+ int sx = TileX(start);
+ int sy = TileY(start);
+ int ex = TileX(end);
+ int ey = TileY(end);
+
+ if (_game_mode != GM_EDITOR) return;
+
+ if (ex < sx) intswap(ex, sx);
+ if (ey < sy) intswap(ey, sy);
+ size_x = (ex - sx) + 1;
+ size_y = (ey - sy) + 1;
+
+ _generating_world = true;
+ BEGIN_TILE_LOOP(tile, size_x, size_y, TileXY(sx, sy)) {
+ if (GetTileType(tile) != MP_WATER) {
+ SetTropicZone(tile, (_ctrl_pressed) ? TROPICZONE_INVALID : TROPICZONE_DESERT);
+ DoCommandP(tile, 0, 0, NULL, CMD_LANDSCAPE_CLEAR);
+ MarkTileDirtyByTile(tile);
+ }
+ } END_TILE_LOOP(tile, size_x, size_y, 0);
+ _generating_world = false;
+}
+
+/** Scenario editor command that generates rocky areas */
+static void GenerateRockyArea(TileIndex end, TileIndex start)
+{
+ int size_x, size_y;
+ bool success = false;
+ int sx = TileX(start);
+ int sy = TileY(start);
+ int ex = TileX(end);
+ int ey = TileY(end);
+
+ if (_game_mode != GM_EDITOR) return;
+
+ if (ex < sx) intswap(ex, sx);
+ if (ey < sy) intswap(ey, sy);
+ size_x = (ex - sx) + 1;
+ size_y = (ey - sy) + 1;
+
+ BEGIN_TILE_LOOP(tile, size_x, size_y, TileXY(sx, sy)) {
+ switch (GetTileType(tile)) {
+ case MP_CLEAR:
+ case MP_TREES:
+ MakeClear(tile, CLEAR_ROCKS, 3);
+ break;
+
+ default: continue;
+ }
+ MarkTileDirtyByTile(tile);
+ success = true;
+ } END_TILE_LOOP(tile, size_x, size_y, 0);
+
+ if (success) SndPlayTileFx(SND_1F_SPLAT, end);
+}
+
+/**
+ * A central place to handle all X_AND_Y dragged GUI functions.
+ * @param e @WindowEvent variable holding in its higher bits (excluding the lower
+ * 4, since that defined the X_Y drag) the type of action to be performed
+ * @return Returns true if the action was found and handled, and false otherwise. This
+ * allows for additional implements that are more local. For example X_Y drag
+ * of convertrail which belongs in rail_gui.c and not terraform_gui.c
+ **/
+bool GUIPlaceProcDragXY(const WindowEvent *e)
+{
+ TileIndex start_tile = e->we.place.starttile;
+ TileIndex end_tile = e->we.place.tile;
+
+ switch (e->we.place.userdata >> 4) {
+ case GUI_PlaceProc_DemolishArea >> 4:
+ DoCommandP(end_tile, start_tile, 0, CcPlaySound10, CMD_CLEAR_AREA | CMD_MSG(STR_00B5_CAN_T_CLEAR_THIS_AREA));
+ break;
+ case GUI_PlaceProc_LevelArea >> 4:
+ DoCommandP(end_tile, start_tile, 0, CcPlaySound10, CMD_LEVEL_LAND | CMD_AUTO);
+ break;
+ case GUI_PlaceProc_RockyArea >> 4:
+ GenerateRockyArea(end_tile, start_tile);
+ break;
+ case GUI_PlaceProc_DesertArea >> 4:
+ GenerateDesertArea(end_tile, start_tile);
+ break;
+ case GUI_PlaceProc_WaterArea >> 4:
+ DoCommandP(end_tile, start_tile, _ctrl_pressed, CcBuildCanal, CMD_BUILD_CANAL | CMD_AUTO | CMD_MSG(STR_CANT_BUILD_CANALS));
+ break;
+ default: return false;
+ }
+
+ return true;
+}
+
+typedef void OnButtonClick(Window *w);
+
+static const uint16 _terraform_keycodes[] = {
+ 'Q',
+ 'W',
+ 'E',
+ 'D',
+ 'U',
+ 'I',
+ 'O',
+};
+
+void PlaceProc_DemolishArea(TileIndex tile)
+{
+ VpStartPlaceSizing(tile, VPM_X_AND_Y | GUI_PlaceProc_DemolishArea);
+}
+
+static void PlaceProc_RaiseLand(TileIndex tile)
+{
+ DoCommandP(
+ tile, SLOPE_N, 1, CcTerraform,
+ CMD_TERRAFORM_LAND | CMD_AUTO | CMD_MSG(STR_0808_CAN_T_RAISE_LAND_HERE)
+ );
+}
+
+static void PlaceProc_LowerLand(TileIndex tile)
+{
+ DoCommandP(
+ tile, SLOPE_N, 0, CcTerraform,
+ CMD_TERRAFORM_LAND | CMD_AUTO | CMD_MSG(STR_0809_CAN_T_LOWER_LAND_HERE)
+ );
+}
+
+void PlaceProc_LevelLand(TileIndex tile)
+{
+ VpStartPlaceSizing(tile, VPM_X_AND_Y | GUI_PlaceProc_LevelArea);
+}
+
+static void TerraformClick_Lower(Window *w)
+{
+ HandlePlacePushButton(w, 4, ANIMCURSOR_LOWERLAND, 2, PlaceProc_LowerLand);
+}
+
+static void TerraformClick_Raise(Window *w)
+{
+ HandlePlacePushButton(w, 5, ANIMCURSOR_RAISELAND, 2, PlaceProc_RaiseLand);
+}
+
+static void TerraformClick_Level(Window *w)
+{
+ HandlePlacePushButton(w, 6, SPR_CURSOR_LEVEL_LAND, 2, PlaceProc_LevelLand);
+}
+
+static void TerraformClick_Dynamite(Window *w)
+{
+ HandlePlacePushButton(w, 7, ANIMCURSOR_DEMOLISH , 1, PlaceProc_DemolishArea);
+}
+
+static void TerraformClick_BuyLand(Window *w)
+{
+ HandlePlacePushButton(w, 8, SPR_CURSOR_BUY_LAND, 1, PlaceProc_BuyLand);
+}
+
+static void TerraformClick_Trees(Window *w)
+{
+ /* This button is NOT a place-push-button, so don't treat it as such */
+ ShowBuildTreesToolbar();
+}
+
+static void TerraformClick_PlaceSign(Window *w)
+{
+ HandlePlacePushButton(w, 10, SPR_CURSOR_SIGN, 1, PlaceProc_Sign);
+}
+
+static OnButtonClick * const _terraform_button_proc[] = {
+ TerraformClick_Lower,
+ TerraformClick_Raise,
+ TerraformClick_Level,
+ TerraformClick_Dynamite,
+ TerraformClick_BuyLand,
+ TerraformClick_Trees,
+ TerraformClick_PlaceSign,
+};
+
+static void TerraformToolbWndProc(Window *w, WindowEvent *e)
+{
+ switch (e->event) {
+ case WE_PAINT:
+ DrawWindowWidgets(w);
+ break;
+
+ case WE_CLICK:
+ if (e->we.click.widget >= 4) _terraform_button_proc[e->we.click.widget - 4](w);
+ break;
+
+ case WE_KEYPRESS: {
+ uint i;
+
+ for (i = 0; i != lengthof(_terraform_keycodes); i++) {
+ if (e->we.keypress.keycode == _terraform_keycodes[i]) {
+ e->we.keypress.cont = false;
+ _terraform_button_proc[i](w);
+ break;
+ }
+ }
+ break;
+ }
+
+ case WE_PLACE_OBJ:
+ _place_proc(e->we.place.tile);
+ return;
+
+ case WE_PLACE_DRAG:
+ VpSelectTilesWithMethod(e->we.place.pt.x, e->we.place.pt.y, e->we.place.userdata & 0xF);
+ break;
+
+ case WE_PLACE_MOUSEUP:
+ if (e->we.place.pt.x != -1 &&
+ (e->we.place.userdata & 0xF) == VPM_X_AND_Y) { // dragged actions
+ GUIPlaceProcDragXY(e);
+ }
+ break;
+
+ case WE_ABORT_PLACE_OBJ:
+ RaiseWindowButtons(w);
+ break;
+ }
+}
+
+static const Widget _terraform_widgets[] = {
+{ WWT_CLOSEBOX, RESIZE_NONE, 7, 0, 10, 0, 13, STR_00C5, STR_018B_CLOSE_WINDOW},
+{ WWT_CAPTION, RESIZE_NONE, 7, 11, 145, 0, 13, STR_LANDSCAPING_TOOLBAR, STR_018C_WINDOW_TITLE_DRAG_THIS},
+{WWT_STICKYBOX, RESIZE_NONE, 7, 146, 157, 0, 13, STR_NULL, STR_STICKY_BUTTON},
+
+{ WWT_PANEL, RESIZE_NONE, 7, 66, 69, 14, 35, 0x0, STR_NULL},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 0, 21, 14, 35, SPR_IMG_TERRAFORM_DOWN, STR_018E_LOWER_A_CORNER_OF_LAND},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 22, 43, 14, 35, SPR_IMG_TERRAFORM_UP, STR_018F_RAISE_A_CORNER_OF_LAND},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 44, 65, 14, 35, SPR_IMG_LEVEL_LAND, STR_LEVEL_LAND_TOOLTIP},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 70, 91, 14, 35, SPR_IMG_DYNAMITE, STR_018D_DEMOLISH_BUILDINGS_ETC},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 92, 113, 14, 35, SPR_IMG_BUY_LAND, STR_0329_PURCHASE_LAND_FOR_FUTURE},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 114, 135, 14, 35, SPR_IMG_PLANTTREES, STR_0185_PLANT_TREES_PLACE_SIGNS},
+{ WWT_IMGBTN, RESIZE_NONE, 7, 136, 157, 14, 35, SPR_IMG_PLACE_SIGN, STR_0289_PLACE_SIGN},
+
+{ WIDGETS_END},
+};
+
+static const WindowDesc _terraform_desc = {
+ WDP_ALIGN_TBR, 22+36, 158, 36,
+ WC_SCEN_LAND_GEN, 0,
+ WDF_STD_TOOLTIPS | WDF_STD_BTN | WDF_DEF_WIDGET | WDF_STICKY_BUTTON,
+ _terraform_widgets,
+ TerraformToolbWndProc
+};
+
+void ShowTerraformToolbar(void)
+{
+ if (!IsValidPlayer(_current_player)) return;
+ AllocateWindowDescFront(&_terraform_desc, 0);
+}