tron@2186: /* $Id$ */ tron@2186: rubidium@9111: /** @file terraform_gui.cpp GUI related to terraforming the map. */ belugas@6422: truelight@0: #include "stdafx.h" Darkvater@1891: #include "openttd.h" celestar@5385: #include "bridge_map.h" tron@3144: #include "clear_map.h" rubidium@8254: #include "player_func.h" rubidium@8254: #include "player_base.h" truelight@0: #include "gui.h" rubidium@8107: #include "window_gui.h" rubidium@8224: #include "viewport_func.h" rubidium@8224: #include "gfx_func.h" rubidium@8116: #include "command_func.h" rubidium@8790: #include "signs_func.h" tron@2159: #include "variables.h" rubidium@8131: #include "functions.h" rubidium@8157: #include "sound_func.h" rubidium@8786: #include "station_base.h" rubidium@8205: #include "unmovable_map.h" rubidium@8205: #include "textbuf_gui.h" rubidium@8236: #include "genworld.h" rubidium@8270: #include "settings_type.h" frosch@8459: #include "tree_map.h" rubidium@9126: #include "landscape_type.h" rubidium@9127: #include "tilehighlight_func.h" truelight@0: rubidium@8264: #include "table/sprites.h" rubidium@8264: #include "table/strings.h" rubidium@8264: tron@1977: void CcTerraform(bool success, TileIndex tile, uint32 p1, uint32 p2) truelight@0: { truelight@0: if (success) { tron@541: SndPlayTileFx(SND_1F_SPLAT, tile); truelight@0: } else { rubidium@8210: extern TileIndex _terraform_err_tile; truelight@0: SetRedErrorSquare(_terraform_err_tile); truelight@0: } truelight@0: } truelight@0: truelight@0: Darkvater@1632: /** Scenario editor command that generates desert areas */ Darkvater@1632: static void GenerateDesertArea(TileIndex end, TileIndex start) Darkvater@1632: { Darkvater@1632: int size_x, size_y; Darkvater@1632: int sx = TileX(start); Darkvater@1632: int sy = TileY(start); Darkvater@1632: int ex = TileX(end); Darkvater@1632: int ey = TileY(end); Darkvater@1632: Darkvater@1632: if (_game_mode != GM_EDITOR) return; Darkvater@1632: tron@6106: if (ex < sx) Swap(ex, sx); tron@6106: if (ey < sy) Swap(ey, sy); Darkvater@1632: size_x = (ex - sx) + 1; Darkvater@1632: size_y = (ey - sy) + 1; Darkvater@1632: Darkvater@1710: _generating_world = true; tron@1981: BEGIN_TILE_LOOP(tile, size_x, size_y, TileXY(sx, sy)) { Darkvater@1710: if (GetTileType(tile) != MP_WATER) { frosch@8450: SetTropicZone(tile, (_ctrl_pressed) ? TROPICZONE_NORMAL : TROPICZONE_DESERT); Darkvater@1780: DoCommandP(tile, 0, 0, NULL, CMD_LANDSCAPE_CLEAR); Darkvater@1710: MarkTileDirtyByTile(tile); Darkvater@1710: } Darkvater@1632: } END_TILE_LOOP(tile, size_x, size_y, 0); Darkvater@1710: _generating_world = false; Darkvater@1632: } Darkvater@1632: tron@3017: /** Scenario editor command that generates rocky areas */ Darkvater@2615: static void GenerateRockyArea(TileIndex end, TileIndex start) Darkvater@2615: { Darkvater@2615: int size_x, size_y; Darkvater@2615: bool success = false; Darkvater@2615: int sx = TileX(start); Darkvater@2615: int sy = TileY(start); Darkvater@2615: int ex = TileX(end); Darkvater@2615: int ey = TileY(end); Darkvater@2615: Darkvater@2615: if (_game_mode != GM_EDITOR) return; Darkvater@2615: tron@6106: if (ex < sx) Swap(ex, sx); tron@6106: if (ey < sy) Swap(ey, sy); Darkvater@2615: size_x = (ex - sx) + 1; Darkvater@2615: size_y = (ey - sy) + 1; Darkvater@2615: Darkvater@2615: BEGIN_TILE_LOOP(tile, size_x, size_y, TileXY(sx, sy)) { celestar@5385: switch (GetTileType(tile)) { frosch@8459: case MP_TREES: frosch@8459: if (GetTreeGround(tile) == TREE_GROUND_SHORE) continue; frosch@8459: /* FALL THROUGH */ celestar@5385: case MP_CLEAR: celestar@5385: MakeClear(tile, CLEAR_ROCKS, 3); celestar@5385: break; celestar@5385: celestar@5385: default: continue; Darkvater@2615: } celestar@5385: MarkTileDirtyByTile(tile); celestar@5385: success = true; Darkvater@2615: } END_TILE_LOOP(tile, size_x, size_y, 0); Darkvater@2615: Darkvater@2615: if (success) SndPlayTileFx(SND_1F_SPLAT, end); Darkvater@2615: } Darkvater@2615: Darkvater@1632: /** Darkvater@1632: * A central place to handle all X_AND_Y dragged GUI functions. belugas@6422: * @param e WindowEvent variable holding in its higher bits (excluding the lower Darkvater@1632: * 4, since that defined the X_Y drag) the type of action to be performed Darkvater@1632: * @return Returns true if the action was found and handled, and false otherwise. This Darkvater@1632: * allows for additional implements that are more local. For example X_Y drag belugas@6422: * of convertrail which belongs in rail_gui.cpp and not terraform_gui.cpp Darkvater@1632: **/ rubidium@9293: bool GUIPlaceProcDragXY(ViewportDragDropSelectionProcess proc, TileIndex start_tile, TileIndex end_tile) Darkvater@1632: { rubidium@9293: switch (proc) { maedhros@6670: case DDSP_DEMOLISH_AREA: maedhros@6669: DoCommandP(end_tile, start_tile, 0, CcPlaySound10, CMD_CLEAR_AREA | CMD_MSG(STR_00B5_CAN_T_CLEAR_THIS_AREA)); maedhros@6669: break; peter1138@8196: case DDSP_RAISE_AND_LEVEL_AREA: smatz@8204: DoCommandP(end_tile, start_tile, 1, CcTerraform, CMD_LEVEL_LAND | CMD_MSG(STR_0808_CAN_T_RAISE_LAND_HERE)); peter1138@8196: break; peter1138@8196: case DDSP_LOWER_AND_LEVEL_AREA: rubidium@8202: DoCommandP(end_tile, start_tile, (uint32)-1, CcTerraform, CMD_LEVEL_LAND | CMD_MSG(STR_0809_CAN_T_LOWER_LAND_HERE)); peter1138@8196: break; maedhros@6670: case DDSP_LEVEL_AREA: rubidium@7521: DoCommandP(end_tile, start_tile, 0, CcPlaySound10, CMD_LEVEL_LAND); maedhros@6669: break; maedhros@6670: case DDSP_CREATE_ROCKS: maedhros@6669: GenerateRockyArea(end_tile, start_tile); maedhros@6669: break; maedhros@6670: case DDSP_CREATE_DESERT: maedhros@6669: GenerateDesertArea(end_tile, start_tile); maedhros@6669: break; maedhros@6670: case DDSP_CREATE_WATER: rubidium@7521: DoCommandP(end_tile, start_tile, _ctrl_pressed, CcBuildCanal, CMD_BUILD_CANAL | CMD_MSG(STR_CANT_BUILD_CANALS)); maedhros@6669: break; peter1138@8360: case DDSP_CREATE_RIVER: peter1138@8360: DoCommandP(end_tile, start_tile, 2, CcBuildCanal, CMD_BUILD_CANAL | CMD_MSG(STR_CANT_BUILD_CANALS)); peter1138@8360: break; maedhros@6669: default: maedhros@6669: return false; Darkvater@1632: } Darkvater@1632: Darkvater@1632: return true; Darkvater@1632: } truelight@0: truelight@0: typedef void OnButtonClick(Window *w); truelight@0: truelight@0: static const uint16 _terraform_keycodes[] = { dominik@606: 'Q', dominik@606: 'W', dominik@606: 'E', dominik@936: 'D', dominik@606: 'U', dominik@606: 'I', dominik@613: 'O', truelight@0: }; truelight@0: rubidium@9150: void CcPlaySound1E(bool success, TileIndex tile, uint32 p1, uint32 p2); rubidium@9150: rubidium@9149: static void PlaceProc_BuyLand(TileIndex tile) rubidium@9149: { rubidium@9149: DoCommandP(tile, 0, 0, CcPlaySound1E, CMD_PURCHASE_LAND_AREA | CMD_NO_WATER | CMD_MSG(STR_5806_CAN_T_PURCHASE_THIS_LAND)); rubidium@9149: } rubidium@9149: tron@1977: void PlaceProc_DemolishArea(TileIndex tile) truelight@0: { maedhros@6670: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_DEMOLISH_AREA); truelight@0: } truelight@0: tron@2817: static void PlaceProc_RaiseLand(TileIndex tile) truelight@0: { peter1138@8196: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_RAISE_AND_LEVEL_AREA); truelight@0: } truelight@0: tron@2817: static void PlaceProc_LowerLand(TileIndex tile) truelight@0: { peter1138@8196: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_LOWER_AND_LEVEL_AREA); truelight@0: } truelight@0: rubidium@9149: static void PlaceProc_LevelLand(TileIndex tile) truelight@0: { maedhros@6670: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_LEVEL_AREA); truelight@0: } truelight@0: smatz@9328: /** Enum referring to the widgets of the terraform toolbar */ smatz@9328: enum TerraformToolbarWidgets { smatz@9328: TTW_CLOSEBOX = 0, ///< Close window button smatz@9328: TTW_CAPTION, ///< Window caption smatz@9328: TTW_STICKY, ///< Sticky window button smatz@9328: TTW_SEPERATOR, ///< Thin seperator line between level land button and demolish button smatz@9328: TTW_BUTTONS_START, ///< Start of pushable buttons smatz@9328: TTW_LOWER_LAND = TTW_BUTTONS_START, ///< Lower land button smatz@9328: TTW_RAISE_LAND, ///< Raise land button smatz@9328: TTW_LEVEL_LAND, ///< Level land button smatz@9328: TTW_DEMOLISH, ///< Demolish aka dynamite button smatz@9328: TTW_BUY_LAND, ///< Buy land button smatz@9328: TTW_PLANT_TREES, ///< Plant trees button (note: opens seperate window, no place-push-button) smatz@9328: TTW_PLACE_SIGN, ///< Place sign button smatz@9328: }; smatz@9328: truelight@0: static void TerraformClick_Lower(Window *w) truelight@0: { smatz@9328: HandlePlacePushButton(w, TTW_LOWER_LAND, ANIMCURSOR_LOWERLAND, VHM_POINT, PlaceProc_LowerLand); truelight@0: } truelight@0: truelight@0: static void TerraformClick_Raise(Window *w) truelight@0: { smatz@9328: HandlePlacePushButton(w, TTW_RAISE_LAND, ANIMCURSOR_RAISELAND, VHM_POINT, PlaceProc_RaiseLand); truelight@0: } truelight@0: truelight@0: static void TerraformClick_Level(Window *w) truelight@0: { smatz@9328: HandlePlacePushButton(w, TTW_LEVEL_LAND, SPR_CURSOR_LEVEL_LAND, VHM_POINT, PlaceProc_LevelLand); truelight@0: } truelight@0: dominik@936: static void TerraformClick_Dynamite(Window *w) dominik@936: { smatz@9328: HandlePlacePushButton(w, TTW_DEMOLISH, ANIMCURSOR_DEMOLISH , VHM_RECT, PlaceProc_DemolishArea); dominik@936: } dominik@936: truelight@0: static void TerraformClick_BuyLand(Window *w) truelight@0: { smatz@9328: HandlePlacePushButton(w, TTW_BUY_LAND, SPR_CURSOR_BUY_LAND, VHM_RECT, PlaceProc_BuyLand); truelight@0: } truelight@0: truelight@0: static void TerraformClick_Trees(Window *w) truelight@0: { celestar@3817: /* This button is NOT a place-push-button, so don't treat it as such */ celestar@3817: ShowBuildTreesToolbar(); truelight@0: } truelight@0: truelight@0: static void TerraformClick_PlaceSign(Window *w) truelight@0: { smatz@9328: HandlePlacePushButton(w, TTW_PLACE_SIGN, SPR_CURSOR_SIGN, VHM_RECT, PlaceProc_Sign); truelight@0: } truelight@0: truelight@0: static OnButtonClick * const _terraform_button_proc[] = { truelight@0: TerraformClick_Lower, truelight@0: TerraformClick_Raise, truelight@0: TerraformClick_Level, dominik@936: TerraformClick_Dynamite, truelight@0: TerraformClick_BuyLand, truelight@0: TerraformClick_Trees, truelight@0: TerraformClick_PlaceSign, truelight@0: }; truelight@0: rubidium@9304: struct TerraformToolbarWindow : Window { rubidium@9304: TerraformToolbarWindow(const WindowDesc *desc, WindowNumber window_number) : Window(desc, window_number) rubidium@9304: { rubidium@9304: this->FindWindowPlacementAndResize(desc); rubidium@9304: } tron@2639: rubidium@9304: ~TerraformToolbarWindow() rubidium@9304: { rubidium@9304: } truelight@0: rubidium@9304: virtual void OnPaint() rubidium@9304: { rubidium@9304: this->DrawWidgets(); rubidium@9304: } dominik@616: rubidium@9304: virtual void OnClick(Point pt, int widget) rubidium@9304: { smatz@9328: if (widget >= TTW_BUTTONS_START) _terraform_button_proc[widget - TTW_BUTTONS_START](this); rubidium@9304: } rubidium@9304: rubidium@9304: virtual EventState OnKeyPress(uint16 key, uint16 keycode) rubidium@9304: { rubidium@9304: for (uint i = 0; i != lengthof(_terraform_keycodes); i++) { rubidium@9304: if (keycode == _terraform_keycodes[i]) { rubidium@9304: _terraform_button_proc[i](this); rubidium@9304: return ES_HANDLED; truelight@0: } tron@2639: } rubidium@9304: return ES_NOT_HANDLED; truelight@0: } truelight@0: rubidium@9304: virtual void OnPlaceObject(Point pt, TileIndex tile) rubidium@9304: { rubidium@9304: _place_proc(tile); rubidium@9304: } truelight@0: rubidium@9304: virtual void OnPlaceDrag(ViewportPlaceMethod select_method, ViewportDragDropSelectionProcess select_proc, Point pt) rubidium@9304: { rubidium@9304: VpSelectTilesWithMethod(pt.x, pt.y, select_method); rubidium@9304: } truelight@0: rubidium@9304: virtual void OnPlaceMouseUp(ViewportPlaceMethod select_method, ViewportDragDropSelectionProcess select_proc, Point pt, TileIndex start_tile, TileIndex end_tile) rubidium@9304: { rubidium@9304: if (pt.x != -1) { rubidium@9304: switch (select_proc) { rubidium@9293: default: NOT_REACHED(); maedhros@6670: case DDSP_DEMOLISH_AREA: peter1138@8196: case DDSP_RAISE_AND_LEVEL_AREA: peter1138@8196: case DDSP_LOWER_AND_LEVEL_AREA: maedhros@6670: case DDSP_LEVEL_AREA: rubidium@9304: GUIPlaceProcDragXY(select_proc, start_tile, end_tile); maedhros@6670: break; maedhros@6670: } truelight@0: } rubidium@9304: } truelight@0: rubidium@9304: virtual void OnPlaceObjectAbort() rubidium@9304: { rubidium@9304: this->RaiseButtons(); truelight@0: } rubidium@9304: }; truelight@0: truelight@0: static const Widget _terraform_widgets[] = { smatz@9328: { WWT_CLOSEBOX, RESIZE_NONE, 7, 0, 10, 0, 13, STR_00C5, STR_018B_CLOSE_WINDOW}, // TTW_CLOSEBOX smatz@9328: { WWT_CAPTION, RESIZE_NONE, 7, 11, 145, 0, 13, STR_LANDSCAPING_TOOLBAR, STR_018C_WINDOW_TITLE_DRAG_THIS}, // TTW_CAPTION smatz@9328: {WWT_STICKYBOX, RESIZE_NONE, 7, 146, 157, 0, 13, STR_NULL, STR_STICKY_BUTTON}, // TTW_STICKY truelight@867: smatz@9328: { WWT_PANEL, RESIZE_NONE, 7, 66, 69, 14, 35, 0x0, STR_NULL}, // TTW_SEPERATOR smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 0, 21, 14, 35, SPR_IMG_TERRAFORM_DOWN, STR_018E_LOWER_A_CORNER_OF_LAND}, // TTW_LOWER_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 22, 43, 14, 35, SPR_IMG_TERRAFORM_UP, STR_018F_RAISE_A_CORNER_OF_LAND}, // TTW_RAISE_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 44, 65, 14, 35, SPR_IMG_LEVEL_LAND, STR_LEVEL_LAND_TOOLTIP}, // TTW_LEVEL_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 70, 91, 14, 35, SPR_IMG_DYNAMITE, STR_018D_DEMOLISH_BUILDINGS_ETC}, // TTW_DEMOLISH smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 92, 113, 14, 35, SPR_IMG_BUY_LAND, STR_0329_PURCHASE_LAND_FOR_FUTURE}, // TTW_BUY_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 114, 135, 14, 35, SPR_IMG_PLANTTREES, STR_0185_PLANT_TREES_PLACE_SIGNS}, // TTW_PLANT_TREES smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 7, 136, 157, 14, 35, SPR_IMG_SIGN, STR_0289_PLACE_SIGN}, // TTW_PLACE_SIGN truelight@0: darkvater@176: { WIDGETS_END}, truelight@0: }; truelight@0: truelight@0: static const WindowDesc _terraform_desc = { rubidium@7341: WDP_ALIGN_TBR, 22 + 36, 158, 36, 158, 36, rubidium@5893: WC_SCEN_LAND_GEN, WC_NONE, darkvater@756: WDF_STD_TOOLTIPS | WDF_STD_BTN | WDF_DEF_WIDGET | WDF_STICKY_BUTTON, truelight@0: _terraform_widgets, truelight@0: }; truelight@0: tron@5860: void ShowTerraformToolbar(Window *link) truelight@0: { Darkvater@5005: if (!IsValidPlayer(_current_player)) return; rubidium@9304: Window *w = AllocateWindowDescFront(&_terraform_desc, 0); tron@5860: if (w != NULL && link != NULL) { tron@5860: /* Align the terraform toolbar under the main toolbar and put the linked tron@5860: * toolbar to left of it tron@5860: */ tron@5860: w->top = 22; tron@5860: link->left = w->left - link->width; rubidium@7322: rubidium@9116: link->SetDirty(); tron@5860: } truelight@0: } rubidium@8205: rubidium@8205: static byte _terraform_size = 1; rubidium@8205: rubidium@8205: /** rubidium@8205: * Raise/Lower a bigger chunk of land at the same time in the editor. When rubidium@8205: * raising get the lowest point, when lowering the highest point, and set all rubidium@8205: * tiles in the selection to that height. rubidium@8205: * @todo : Incorporate into game itself to allow for ingame raising/lowering of rubidium@8205: * larger chunks at the same time OR remove altogether, as we have 'level land' ? rubidium@8205: * @param tile The top-left tile where the terraforming will start rubidium@8205: * @param mode 1 for raising, 0 for lowering land rubidium@8205: */ rubidium@8205: static void CommonRaiseLowerBigLand(TileIndex tile, int mode) rubidium@8205: { rubidium@8205: int sizex, sizey; rubidium@8205: uint h; rubidium@8205: rubidium@8205: if (_terraform_size == 1) { rubidium@8205: StringID msg = rubidium@8205: mode ? STR_0808_CAN_T_RAISE_LAND_HERE : STR_0809_CAN_T_LOWER_LAND_HERE; rubidium@8205: rubidium@8205: DoCommandP(tile, SLOPE_N, (uint32)mode, CcTerraform, CMD_TERRAFORM_LAND | CMD_MSG(msg)); rubidium@8205: } else { rubidium@8205: SndPlayTileFx(SND_1F_SPLAT, tile); rubidium@8205: rubidium@8205: assert(_terraform_size != 0); rubidium@8205: /* check out for map overflows */ rubidium@8205: sizex = min(MapSizeX() - TileX(tile) - 1, _terraform_size); rubidium@8205: sizey = min(MapSizeY() - TileY(tile) - 1, _terraform_size); rubidium@8205: rubidium@8205: if (sizex == 0 || sizey == 0) return; rubidium@8205: rubidium@8205: if (mode != 0) { rubidium@8205: /* Raise land */ rubidium@8205: h = 15; // XXX - max height rubidium@8205: BEGIN_TILE_LOOP(tile2, sizex, sizey, tile) { rubidium@8205: h = min(h, TileHeight(tile2)); rubidium@8205: } END_TILE_LOOP(tile2, sizex, sizey, tile) rubidium@8205: } else { rubidium@8205: /* Lower land */ rubidium@8205: h = 0; rubidium@8205: BEGIN_TILE_LOOP(tile2, sizex, sizey, tile) { rubidium@8205: h = max(h, TileHeight(tile2)); rubidium@8205: } END_TILE_LOOP(tile2, sizex, sizey, tile) rubidium@8205: } rubidium@8205: rubidium@8205: BEGIN_TILE_LOOP(tile2, sizex, sizey, tile) { rubidium@8205: if (TileHeight(tile2) == h) { rubidium@8205: DoCommandP(tile2, SLOPE_N, (uint32)mode, NULL, CMD_TERRAFORM_LAND); rubidium@8205: } rubidium@8205: } END_TILE_LOOP(tile2, sizex, sizey, tile) rubidium@8205: } rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_RaiseBigLand(TileIndex tile) rubidium@8205: { rubidium@8205: CommonRaiseLowerBigLand(tile, 1); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_LowerBigLand(TileIndex tile) rubidium@8205: { rubidium@8205: CommonRaiseLowerBigLand(tile, 0); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_RockyArea(TileIndex tile) rubidium@8205: { rubidium@8205: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_CREATE_ROCKS); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_LightHouse(TileIndex tile) rubidium@8205: { smatz@8472: /* not flat || not(trees || clear without bridge above) */ smatz@8472: if (GetTileSlope(tile, NULL) != SLOPE_FLAT || !(IsTileType(tile, MP_TREES) || (IsTileType(tile, MP_CLEAR) && !IsBridgeAbove(tile)))) { rubidium@8205: return; rubidium@8205: } rubidium@8205: rubidium@8205: MakeLighthouse(tile); rubidium@8205: MarkTileDirtyByTile(tile); rubidium@8205: SndPlayTileFx(SND_1F_SPLAT, tile); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_Transmitter(TileIndex tile) rubidium@8205: { smatz@8472: /* not flat || not(trees || clear without bridge above) */ smatz@8472: if (GetTileSlope(tile, NULL) != SLOPE_FLAT || !(IsTileType(tile, MP_TREES) || (IsTileType(tile, MP_CLEAR) && !IsBridgeAbove(tile)))) { rubidium@8205: return; rubidium@8205: } rubidium@8205: rubidium@8205: MakeTransmitter(tile); rubidium@8205: MarkTileDirtyByTile(tile); rubidium@8205: SndPlayTileFx(SND_1F_SPLAT, tile); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_DesertArea(TileIndex tile) rubidium@8205: { rubidium@8205: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_CREATE_DESERT); rubidium@8205: } rubidium@8205: rubidium@8205: static void PlaceProc_WaterArea(TileIndex tile) rubidium@8205: { rubidium@8205: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_CREATE_WATER); rubidium@8205: } rubidium@8205: peter1138@8360: static void PlaceProc_RiverArea(TileIndex tile) peter1138@8360: { peter1138@8360: VpStartPlaceSizing(tile, VPM_X_AND_Y, DDSP_CREATE_RIVER); peter1138@8360: } peter1138@8360: rubidium@8205: static const Widget _scen_edit_land_gen_widgets[] = { smatz@9328: { WWT_CLOSEBOX, RESIZE_NONE, 7, 0, 10, 0, 13, STR_00C5, STR_018B_CLOSE_WINDOW}, // ETTW_CLOSEBOX smatz@9328: { WWT_CAPTION, RESIZE_NONE, 7, 11, 191, 0, 13, STR_0223_LAND_GENERATION, STR_018C_WINDOW_TITLE_DRAG_THIS}, // ETTW_CAPTION smatz@9328: { WWT_STICKYBOX, RESIZE_NONE, 7, 192, 203, 0, 13, STR_NULL, STR_STICKY_BUTTON}, // ETTW_STICKY smatz@9328: { WWT_PANEL, RESIZE_NONE, 7, 0, 203, 14, 102, 0x0, STR_NULL}, // ETTW_BACKGROUND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 2, 23, 16, 37, SPR_IMG_DYNAMITE, STR_018D_DEMOLISH_BUILDINGS_ETC}, // ETTW_DEMOLISH smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 24, 45, 16, 37, SPR_IMG_TERRAFORM_DOWN, STR_018E_LOWER_A_CORNER_OF_LAND}, // ETTW_LOWER_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 46, 67, 16, 37, SPR_IMG_TERRAFORM_UP, STR_018F_RAISE_A_CORNER_OF_LAND}, // ETTW_RAISE_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 68, 89, 16, 37, SPR_IMG_LEVEL_LAND, STR_LEVEL_LAND_TOOLTIP}, // ETTW_LEVEL_LAND smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 90, 111, 16, 37, SPR_IMG_BUILD_CANAL, STR_CREATE_LAKE}, // ETTW_BUILD_CANAL smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 112, 133, 16, 37, SPR_IMG_BUILD_RIVER, STR_CREATE_RIVER}, // ETTW_BUILD_RIVER smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 134, 156, 16, 37, SPR_IMG_ROCKS, STR_028C_PLACE_ROCKY_AREAS_ON_LANDSCAPE}, // ETTW_PLACE_ROCKS smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 157, 179, 16, 37, SPR_IMG_LIGHTHOUSE_DESERT, STR_NULL}, // ETTW_PLACE_DESERT_LIGHTHOUSE XXX - dynamic smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 180, 201, 16, 37, SPR_IMG_TRANSMITTER, STR_028E_PLACE_TRANSMITTER}, // ETTW_PLACE_TRANSMITTER smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 150, 161, 45, 56, SPR_ARROW_UP, STR_0228_INCREASE_SIZE_OF_LAND_AREA}, // ETTW_INCREASE_SIZE smatz@9328: { WWT_IMGBTN, RESIZE_NONE, 14, 150, 161, 58, 69, SPR_ARROW_DOWN, STR_0229_DECREASE_SIZE_OF_LAND_AREA}, // ETTW_DECREASE_SIZE smatz@9328: { WWT_TEXTBTN, RESIZE_NONE, 14, 24, 179, 76, 87, STR_SE_NEW_WORLD, STR_022A_GENERATE_RANDOM_LAND}, // ETTW_NEW_SCENARIO smatz@9328: { WWT_TEXTBTN, RESIZE_NONE, 14, 24, 179, 89, 100, STR_022B_RESET_LANDSCAPE, STR_RESET_LANDSCAPE_TOOLTIP}, // ETTW_RESET_LANDSCAPE rubidium@8205: { WIDGETS_END}, rubidium@8205: }; rubidium@8205: rubidium@8205: static const int8 _multi_terraform_coords[][2] = { rubidium@8205: { 0, -2}, rubidium@8205: { 4, 0}, { -4, 0}, { 0, 2}, rubidium@8205: { -8, 2}, { -4, 4}, { 0, 6}, { 4, 4}, { 8, 2}, rubidium@8205: {-12, 0}, { -8, -2}, { -4, -4}, { 0, -6}, { 4, -4}, { 8, -2}, { 12, 0}, rubidium@8205: {-16, 2}, {-12, 4}, { -8, 6}, { -4, 8}, { 0, 10}, { 4, 8}, { 8, 6}, { 12, 4}, { 16, 2}, rubidium@8205: {-20, 0}, {-16, -2}, {-12, -4}, { -8, -6}, { -4, -8}, { 0,-10}, { 4, -8}, { 8, -6}, { 12, -4}, { 16, -2}, { 20, 0}, rubidium@8205: {-24, 2}, {-20, 4}, {-16, 6}, {-12, 8}, { -8, 10}, { -4, 12}, { 0, 14}, { 4, 12}, { 8, 10}, { 12, 8}, { 16, 6}, { 20, 4}, { 24, 2}, rubidium@8205: {-28, 0}, {-24, -2}, {-20, -4}, {-16, -6}, {-12, -8}, { -8,-10}, { -4,-12}, { 0,-14}, { 4,-12}, { 8,-10}, { 12, -8}, { 16, -6}, { 20, -4}, { 24, -2}, { 28, 0}, rubidium@8205: }; rubidium@8205: smatz@9328: /** Enum referring to the widgets of the editor terraform toolbar */ smatz@9328: enum EditorTerraformToolbarWidgets { smatz@9328: ETTW_START = 0, ///< Used for iterations smatz@9328: ETTW_CLOSEBOX = ETTW_START, ///< Close window button smatz@9328: ETTW_CAPTION, ///< Window caption smatz@9328: ETTW_STICKY, ///< Sticky window button smatz@9328: ETTW_BACKGROUND, ///< Background of the lower part of the window smatz@9328: ETTW_BUTTONS_START, ///< Start of pushable buttons smatz@9328: ETTW_DEMOLISH = ETTW_BUTTONS_START, ///< Demolish aka dynamite button smatz@9328: ETTW_LOWER_LAND, ///< Lower land button smatz@9328: ETTW_RAISE_LAND, ///< Raise land button smatz@9328: ETTW_LEVEL_LAND, ///< Level land button smatz@9328: ETTW_BUILD_CANAL, ///< Build canal button smatz@9328: ETTW_BUILD_RIVER, ///< Build river button smatz@9328: ETTW_PLACE_ROCKS, ///< Place rocks button smatz@9328: ETTW_PLACE_DESERT_LIGHTHOUSE, ///< Place desert button (in tropical climate) / place lighthouse button (else) smatz@9328: ETTW_PLACE_TRANSMITTER, ///< Place transmitter button smatz@9328: ETTW_BUTTONS_END, ///< End of pushable buttons smatz@9328: ETTW_INCREASE_SIZE = ETTW_BUTTONS_END, ///< Upwards arrow button to increase terraforming size smatz@9328: ETTW_DECREASE_SIZE, ///< Downwards arrow button to decrease terraforming size smatz@9328: ETTW_NEW_SCENARIO, ///< Button for generating a new scenario smatz@9328: ETTW_RESET_LANDSCAPE, ///< Button for removing all player-owned property smatz@9328: }; smatz@9328: rubidium@8205: /** rubidium@8205: * @todo Merge with terraform_gui.cpp (move there) after I have cooled down at its braindeadness rubidium@8205: * and changed OnButtonClick to include the widget as well in the function declaration. Post 0.4.0 - Darkvater rubidium@8205: */ rubidium@8205: static void EditorTerraformClick_Dynamite(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_DEMOLISH, ANIMCURSOR_DEMOLISH, VHM_RECT, PlaceProc_DemolishArea); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_LowerBigLand(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_LOWER_LAND, ANIMCURSOR_LOWERLAND, VHM_POINT, PlaceProc_LowerBigLand); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_RaiseBigLand(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_RAISE_LAND, ANIMCURSOR_RAISELAND, VHM_POINT, PlaceProc_RaiseBigLand); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_LevelLand(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_LEVEL_LAND, SPR_CURSOR_LEVEL_LAND, VHM_POINT, PlaceProc_LevelLand); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_WaterArea(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_BUILD_CANAL, SPR_CURSOR_CANAL, VHM_RECT, PlaceProc_WaterArea); rubidium@8205: } rubidium@8205: peter1138@8360: static void EditorTerraformClick_RiverArea(Window *w) peter1138@8360: { smatz@9328: HandlePlacePushButton(w, ETTW_BUILD_RIVER, SPR_CURSOR_RIVER, VHM_RECT, PlaceProc_RiverArea); peter1138@8360: } peter1138@8360: rubidium@8205: static void EditorTerraformClick_RockyArea(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_PLACE_ROCKS, SPR_CURSOR_ROCKY_AREA, VHM_RECT, PlaceProc_RockyArea); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_DesertLightHouse(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_PLACE_DESERT_LIGHTHOUSE, SPR_CURSOR_LIGHTHOUSE, VHM_RECT, (_opt.landscape == LT_TROPIC) ? PlaceProc_DesertArea : PlaceProc_LightHouse); rubidium@8205: } rubidium@8205: rubidium@8205: static void EditorTerraformClick_Transmitter(Window *w) rubidium@8205: { smatz@9328: HandlePlacePushButton(w, ETTW_PLACE_TRANSMITTER, SPR_CURSOR_TRANSMITTER, VHM_RECT, PlaceProc_Transmitter); rubidium@8205: } rubidium@8205: rubidium@8205: static const uint16 _editor_terraform_keycodes[] = { rubidium@8205: 'D', rubidium@8205: 'Q', rubidium@8205: 'W', rubidium@8205: 'E', rubidium@8205: 'R', rubidium@8205: 'T', rubidium@8205: 'Y', peter1138@8360: 'U', peter1138@8360: 'I' rubidium@8205: }; rubidium@8205: rubidium@8205: typedef void OnButtonClick(Window *w); rubidium@8205: static OnButtonClick * const _editor_terraform_button_proc[] = { rubidium@8205: EditorTerraformClick_Dynamite, rubidium@8205: EditorTerraformClick_LowerBigLand, rubidium@8205: EditorTerraformClick_RaiseBigLand, rubidium@8205: EditorTerraformClick_LevelLand, rubidium@8205: EditorTerraformClick_WaterArea, peter1138@8360: EditorTerraformClick_RiverArea, rubidium@8205: EditorTerraformClick_RockyArea, rubidium@8205: EditorTerraformClick_DesertLightHouse, rubidium@8205: EditorTerraformClick_Transmitter rubidium@8205: }; rubidium@8205: rubidium@8205: rubidium@8205: /** Callback function for the scenario editor 'reset landscape' confirmation window rubidium@8205: * @param w Window unused rubidium@8205: * @param confirmed boolean value, true when yes was clicked, false otherwise */ rubidium@8205: static void ResetLandscapeConfirmationCallback(Window *w, bool confirmed) rubidium@8205: { rubidium@8205: if (confirmed) { rubidium@8205: Player *p; rubidium@8205: rubidium@8205: /* Set generating_world to true to get instant-green grass after removing rubidium@8205: * player property. */ rubidium@8205: _generating_world = true; rubidium@8205: /* Delete all players */ rubidium@8205: FOR_ALL_PLAYERS(p) { rubidium@8205: if (p->is_active) { rubidium@8205: ChangeOwnershipOfPlayerItems(p->index, PLAYER_SPECTATOR); rubidium@8205: p->is_active = false; rubidium@8205: } rubidium@8205: } rubidium@8205: _generating_world = false; rubidium@8205: rubidium@8205: /* Delete all stations owned by a player */ rubidium@8205: Station *st; rubidium@8205: FOR_ALL_STATIONS(st) { rubidium@8205: if (IsValidPlayer(st->owner)) delete st; rubidium@8205: } rubidium@8205: } rubidium@8205: } rubidium@8205: rubidium@9304: struct ScenarioEditorLandscapeGenerationWindow : Window { rubidium@9304: ScenarioEditorLandscapeGenerationWindow(const WindowDesc *desc, WindowNumber window_number) : Window(desc, window_number) rubidium@9304: { smatz@9328: this->widget[ETTW_PLACE_DESERT_LIGHTHOUSE].tooltips = (_opt.landscape == LT_TROPIC) ? STR_028F_DEFINE_DESERT_AREA : STR_028D_PLACE_LIGHTHOUSE; rubidium@9304: this->FindWindowPlacementAndResize(desc); rubidium@9304: } rubidium@8205: rubidium@9304: virtual void OnPaint() { rubidium@9304: this->DrawWidgets(); rubidium@8205: rubidium@9304: int n = _terraform_size * _terraform_size; rubidium@9304: const int8 *coords = &_multi_terraform_coords[0][0]; rubidium@8205: rubidium@9304: assert(n != 0); rubidium@9304: do { rubidium@9304: DrawSprite(SPR_WHITE_POINT, PAL_NONE, 88 + coords[0], 55 + coords[1]); rubidium@9304: coords += 2; rubidium@9304: } while (--n); rubidium@9304: smatz@9328: if (this->IsWidgetLowered(ETTW_LOWER_LAND) || this->IsWidgetLowered(ETTW_RAISE_LAND)) { // change area-size if raise/lower corner is selected rubidium@9304: SetTileSelectSize(_terraform_size, _terraform_size); rubidium@9304: } rubidium@9304: } rubidium@9304: rubidium@9304: virtual EventState OnKeyPress(uint16 key, uint16 keycode) rubidium@9304: { rubidium@9304: for (uint i = 0; i != lengthof(_editor_terraform_keycodes); i++) { rubidium@9304: if (keycode == _editor_terraform_keycodes[i]) { rubidium@9304: _editor_terraform_button_proc[i](this); rubidium@9304: return ES_HANDLED; rubidium@9304: } rubidium@9304: } rubidium@9304: return ES_NOT_HANDLED; rubidium@9304: } rubidium@9304: rubidium@9304: virtual void OnClick(Point pt, int widget) rubidium@9304: { smatz@9328: if (IsInsideMM(widget, ETTW_BUTTONS_START, ETTW_BUTTONS_END)) { smatz@9328: _editor_terraform_button_proc[widget - ETTW_BUTTONS_START](this); smatz@9328: } else { smatz@9328: switch (widget) { smatz@9328: case ETTW_INCREASE_SIZE: smatz@9328: case ETTW_DECREASE_SIZE: { // Increase/Decrease terraform size smatz@9328: int size = (widget == ETTW_INCREASE_SIZE) ? 1 : -1; smatz@9328: this->HandleButtonClick(widget); smatz@9328: size += _terraform_size; rubidium@9304: smatz@9328: if (!IsInsideMM(size, 1, 8 + 1)) return; smatz@9328: _terraform_size = size; smatz@9328: smatz@9328: SndPlayFx(SND_15_BEEP); smatz@9328: this->SetDirty(); smatz@9328: } break; smatz@9328: case ETTW_NEW_SCENARIO: // gen random land smatz@9328: this->HandleButtonClick(widget); smatz@9328: ShowCreateScenario(); smatz@9328: break; smatz@9328: case ETTW_RESET_LANDSCAPE: // Reset landscape smatz@9328: ShowQuery( smatz@9328: STR_022C_RESET_LANDSCAPE, smatz@9328: STR_RESET_LANDSCAPE_CONFIRMATION_TEXT, smatz@9328: NULL, smatz@9328: ResetLandscapeConfirmationCallback); smatz@9328: break; smatz@9328: } rubidium@9304: } rubidium@9304: } rubidium@9304: rubidium@9304: virtual void OnTimeout() rubidium@9304: { smatz@9328: for (uint i = ETTW_START; i < this->widget_count; i++) { smatz@9328: if (i == ETTW_BUTTONS_START) i = ETTW_BUTTONS_END; // skip the buttons rubidium@9304: if (this->IsWidgetLowered(i)) { rubidium@9304: this->RaiseWidget(i); rubidium@9304: this->InvalidateWidget(i); rubidium@9304: } rubidium@9304: } rubidium@9304: } rubidium@9304: rubidium@9304: virtual void OnPlaceObject(Point pt, TileIndex tile) rubidium@9304: { rubidium@9304: _place_proc(tile); rubidium@9304: } rubidium@9304: rubidium@9304: virtual void OnPlaceDrag(ViewportPlaceMethod select_method, ViewportDragDropSelectionProcess select_proc, Point pt) rubidium@9304: { rubidium@9304: VpSelectTilesWithMethod(pt.x, pt.y, select_method); rubidium@9304: } rubidium@9304: rubidium@9304: virtual void OnPlaceMouseUp(ViewportPlaceMethod select_method, ViewportDragDropSelectionProcess select_proc, Point pt, TileIndex start_tile, TileIndex end_tile) rubidium@9304: { rubidium@9304: if (pt.x != -1) { rubidium@9304: switch (select_proc) { rubidium@9304: default: NOT_REACHED(); rubidium@9304: case DDSP_CREATE_ROCKS: rubidium@9304: case DDSP_CREATE_DESERT: rubidium@9304: case DDSP_CREATE_WATER: rubidium@9304: case DDSP_CREATE_RIVER: rubidium@9304: case DDSP_RAISE_AND_LEVEL_AREA: rubidium@9304: case DDSP_LOWER_AND_LEVEL_AREA: rubidium@9304: case DDSP_LEVEL_AREA: rubidium@9304: case DDSP_DEMOLISH_AREA: rubidium@9304: GUIPlaceProcDragXY(select_proc, start_tile, end_tile); rubidium@8205: break; rubidium@8205: } rubidium@9304: } rubidium@9304: } rubidium@8205: rubidium@9304: virtual void OnPlaceObjectAbort() rubidium@9304: { rubidium@9304: this->RaiseButtons(); rubidium@9304: this->SetDirty(); rubidium@8205: } rubidium@9304: }; rubidium@8205: rubidium@8205: static const WindowDesc _scen_edit_land_gen_desc = { peter1138@8360: WDP_AUTO, WDP_AUTO, 204, 103, 204, 103, rubidium@8205: WC_SCEN_LAND_GEN, WC_NONE, rubidium@8205: WDF_STD_TOOLTIPS | WDF_STD_BTN | WDF_DEF_WIDGET | WDF_STICKY_BUTTON, rubidium@8205: _scen_edit_land_gen_widgets, rubidium@8205: }; rubidium@8205: rubidium@8205: void ShowEditorTerraformToolbar() rubidium@8205: { rubidium@9304: AllocateWindowDescFront(&_scen_edit_land_gen_desc, 0); rubidium@8205: }