mirror of https://github.com/OpenRCT2/OpenRCT2.git
2411 lines
120 KiB
C++
2411 lines
120 KiB
C++
/*****************************************************************************
|
|
* Copyright (c) 2014-2020 OpenRCT2 developers
|
|
*
|
|
* For a complete list of all authors, please refer to contributors.md
|
|
* Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2
|
|
*
|
|
* OpenRCT2 is licensed under the GNU General Public License version 3.
|
|
*****************************************************************************/
|
|
|
|
#include <algorithm>
|
|
#include <iterator>
|
|
#include <openrct2-ui/interface/Dropdown.h>
|
|
#include <openrct2-ui/interface/Widget.h>
|
|
#include <openrct2-ui/windows/Window.h>
|
|
#include <openrct2/Game.h>
|
|
#include <openrct2/Input.h>
|
|
#include <openrct2/actions/TileModifyAction.h>
|
|
#include <openrct2/common.h>
|
|
#include <openrct2/core/Guard.hpp>
|
|
#include <openrct2/localisation/Formatter.h>
|
|
#include <openrct2/localisation/Localisation.h>
|
|
#include <openrct2/localisation/StringIds.h>
|
|
#include <openrct2/object/FootpathObject.h>
|
|
#include <openrct2/object/FootpathRailingsObject.h>
|
|
#include <openrct2/object/FootpathSurfaceObject.h>
|
|
#include <openrct2/object/TerrainEdgeObject.h>
|
|
#include <openrct2/object/TerrainSurfaceObject.h>
|
|
#include <openrct2/ride/RideData.h>
|
|
#include <openrct2/ride/Track.h>
|
|
#include <openrct2/sprites.h>
|
|
#include <openrct2/windows/tile_inspector.h>
|
|
#include <openrct2/world/Banner.h>
|
|
#include <openrct2/world/Footpath.h>
|
|
#include <openrct2/world/LargeScenery.h>
|
|
#include <openrct2/world/Park.h>
|
|
#include <openrct2/world/Scenery.h>
|
|
#include <openrct2/world/SmallScenery.h>
|
|
#include <openrct2/world/Surface.h>
|
|
#include <openrct2/world/TileInspector.h>
|
|
|
|
static constexpr const rct_string_id EntranceTypeStringIds[] = {
|
|
STR_TILE_INSPECTOR_ENTRANCE_TYPE_RIDE_ENTRANCE,
|
|
STR_TILE_INSPECTOR_ENTRANCE_TYPE_RIDE_EXIT,
|
|
STR_TILE_INSPECTOR_ENTRANCE_TYPE_PARK_ENTRANC,
|
|
};
|
|
|
|
static constexpr const rct_string_id ParkEntrancePartStringIds[] = {
|
|
STR_TILE_INSPECTOR_ENTRANCE_MIDDLE,
|
|
STR_TILE_INSPECTOR_ENTRANCE_LEFT,
|
|
STR_TILE_INSPECTOR_ENTRANCE_RIGHT,
|
|
};
|
|
|
|
static constexpr const rct_string_id WallSlopeStringIds[] = {
|
|
STR_TILE_INSPECTOR_WALL_FLAT,
|
|
STR_TILE_INSPECTOR_WALL_SLOPED_LEFT,
|
|
STR_TILE_INSPECTOR_WALL_SLOPED_RIGHT,
|
|
STR_TILE_INSPECTOR_WALL_ANIMATION_FRAME,
|
|
};
|
|
|
|
enum WindowTileInspectorWidgetIdx
|
|
{
|
|
WIDX_BACKGROUND,
|
|
WIDX_TITLE,
|
|
WIDX_CLOSE,
|
|
WIDX_LIST,
|
|
WIDX_SPINNER_X,
|
|
WIDX_SPINNER_X_INCREASE,
|
|
WIDX_SPINNER_X_DECREASE,
|
|
WIDX_SPINNER_Y,
|
|
WIDX_SPINNER_Y_INCREASE,
|
|
WIDX_SPINNER_Y_DECREASE,
|
|
WIDX_BUTTON_REMOVE,
|
|
WIDX_BUTTON_MOVE_UP,
|
|
WIDX_BUTTON_MOVE_DOWN,
|
|
WIDX_BUTTON_ROTATE,
|
|
WIDX_BUTTON_SORT,
|
|
WIDX_BUTTON_PASTE,
|
|
WIDX_BUTTON_COPY,
|
|
WIDX_COLUMN_INVISIBLE,
|
|
WIDX_COLUMN_TYPE,
|
|
WIDX_COLUMN_BASEHEIGHT,
|
|
WIDX_COLUMN_CLEARANCEHEIGHT,
|
|
WIDX_COLUMN_DIRECTION,
|
|
WIDX_COLUMN_GHOSTFLAG,
|
|
WIDX_COLUMN_LASTFLAG,
|
|
WIDX_GROUPBOX_DETAILS,
|
|
WIDX_GROUPBOX_PROPERTIES,
|
|
|
|
PAGE_WIDGETS,
|
|
|
|
// Surface
|
|
WIDX_SURFACE_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_SURFACE_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_SURFACE_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_SURFACE_BUTTON_REMOVE_FENCES,
|
|
WIDX_SURFACE_BUTTON_RESTORE_FENCES,
|
|
WIDX_SURFACE_CHECK_CORNER_N,
|
|
WIDX_SURFACE_CHECK_CORNER_E,
|
|
WIDX_SURFACE_CHECK_CORNER_S,
|
|
WIDX_SURFACE_CHECK_CORNER_W,
|
|
WIDX_SURFACE_CHECK_DIAGONAL,
|
|
|
|
// Path
|
|
WIDX_PATH_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_PATH_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_PATH_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_PATH_CHECK_BROKEN,
|
|
WIDX_PATH_CHECK_SLOPED,
|
|
WIDX_PATH_CHECK_EDGE_NE, // Note: This is NOT named after the world orientation, but after the way
|
|
WIDX_PATH_CHECK_EDGE_E, // it looks in the window (top corner is north). Their order is important,
|
|
WIDX_PATH_CHECK_EDGE_SE, // as this is the same order paths use for their corners / edges.
|
|
WIDX_PATH_CHECK_EDGE_S, // N
|
|
WIDX_PATH_CHECK_EDGE_SW, // NW-------NE
|
|
WIDX_PATH_CHECK_EDGE_W, // W ------------- E
|
|
WIDX_PATH_CHECK_EDGE_NW, // SW-------SE
|
|
WIDX_PATH_CHECK_EDGE_N, // S
|
|
|
|
// Track
|
|
WIDX_TRACK_CHECK_APPLY_TO_ALL = PAGE_WIDGETS,
|
|
WIDX_TRACK_SPINNER_HEIGHT,
|
|
WIDX_TRACK_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_TRACK_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_TRACK_CHECK_CHAIN_LIFT,
|
|
WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED,
|
|
WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE,
|
|
|
|
// Scenery
|
|
WIDX_SCENERY_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_SCENERY_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_SCENERY_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_SCENERY_CHECK_QUARTER_N,
|
|
WIDX_SCENERY_CHECK_QUARTER_E,
|
|
WIDX_SCENERY_CHECK_QUARTER_S,
|
|
WIDX_SCENERY_CHECK_QUARTER_W,
|
|
WIDX_SCENERY_CHECK_COLLISION_N,
|
|
WIDX_SCENERY_CHECK_COLLISION_E,
|
|
WIDX_SCENERY_CHECK_COLLISION_S,
|
|
WIDX_SCENERY_CHECK_COLLISION_W,
|
|
|
|
// Entrance
|
|
WIDX_ENTRANCE_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_ENTRANCE_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_ENTRANCE_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_ENTRANCE_BUTTON_MAKE_USABLE,
|
|
|
|
// Wall
|
|
WIDX_WALL_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_WALL_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_WALL_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_WALL_DROPDOWN_SLOPE,
|
|
WIDX_WALL_DROPDOWN_SLOPE_BUTTON,
|
|
WIDX_WALL_SPINNER_ANIMATION_FRAME,
|
|
WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE,
|
|
WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE,
|
|
|
|
// Large
|
|
WIDX_LARGE_SCENERY_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_LARGE_SCENERY_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_LARGE_SCENERY_SPINNER_HEIGHT_DECREASE,
|
|
|
|
// Banner
|
|
WIDX_BANNER_SPINNER_HEIGHT = PAGE_WIDGETS,
|
|
WIDX_BANNER_SPINNER_HEIGHT_INCREASE,
|
|
WIDX_BANNER_SPINNER_HEIGHT_DECREASE,
|
|
WIDX_BANNER_CHECK_BLOCK_NE,
|
|
WIDX_BANNER_CHECK_BLOCK_SE,
|
|
WIDX_BANNER_CHECK_BLOCK_SW,
|
|
WIDX_BANNER_CHECK_BLOCK_NW,
|
|
};
|
|
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_ROTATE == WIDX_BUTTON_ROTATE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_COPY == WIDX_BUTTON_COPY);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_PASTE == WIDX_BUTTON_PASTE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_REMOVE == WIDX_BUTTON_REMOVE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_MOVE_UP == WIDX_BUTTON_MOVE_UP);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_BUTTON_MOVE_DOWN == WIDX_BUTTON_MOVE_DOWN);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_SPINNER_X_INCREASE == WIDX_SPINNER_X_INCREASE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_SPINNER_X_DECREASE == WIDX_SPINNER_X_DECREASE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_SPINNER_Y_INCREASE == WIDX_SPINNER_Y_INCREASE);
|
|
static_assert(WC_TILE_INSPECTOR__WIDX_SPINNER_Y_DECREASE == WIDX_SPINNER_Y_DECREASE);
|
|
|
|
static constexpr const rct_string_id WINDOW_TITLE = STR_TILE_INSPECTOR_TITLE;
|
|
|
|
// Window sizes
|
|
static constexpr const int32_t WW = 400;
|
|
static constexpr const int32_t WH = 170;
|
|
constexpr int32_t MIN_WW = WW;
|
|
constexpr int32_t MAX_WW = WW;
|
|
constexpr int32_t MIN_WH = 130;
|
|
constexpr int32_t MAX_WH = 800;
|
|
|
|
// Button space for top buttons
|
|
constexpr auto ToolbarButtonAnchor = ScreenCoordsXY{ WW - 27, 17 };
|
|
constexpr auto ToolbarButtonSize = ScreenSize{ 24, 24 };
|
|
constexpr auto ToolbarButtonHalfSize = ScreenSize{ 24, 12 };
|
|
constexpr auto ToolbarButtonOffsetX = ScreenSize{ -24, 0 };
|
|
|
|
// List's column offsets
|
|
constexpr auto InvisibleFlagColumnXY = ScreenCoordsXY{ 3, 42 };
|
|
constexpr auto InvisibleFlagColumnSize = ScreenSize{ 20, 14 };
|
|
constexpr auto TypeColumnXY = InvisibleFlagColumnXY + ScreenSize{ InvisibleFlagColumnSize.width, 0 };
|
|
constexpr auto TypeColumnSize = ScreenSize{ 252, 14 };
|
|
constexpr auto BaseHeightColumnXY = TypeColumnXY + ScreenSize{ TypeColumnSize.width, 0 };
|
|
constexpr auto BaseHeightColumnSize = ScreenSize{ 30, 14 };
|
|
constexpr auto ClearanceHeightColumnXY = BaseHeightColumnXY + ScreenCoordsXY{ BaseHeightColumnSize.width, 0 };
|
|
constexpr auto ClearanceHeightColumnSize = ScreenSize{ 30, 14 };
|
|
constexpr auto DirectionColumnXY = ClearanceHeightColumnXY + ScreenCoordsXY{ ClearanceHeightColumnSize.width, 0 };
|
|
constexpr auto DirectionColumnSize = ScreenSize{ 15, 14 };
|
|
constexpr auto GhostFlagColumnXY = DirectionColumnXY + ScreenCoordsXY{ DirectionColumnSize.width, 0 };
|
|
constexpr auto GhostFlagColumnSize = ScreenSize{ 15, 14 };
|
|
constexpr auto LastFlagColumnXY = GhostFlagColumnXY + ScreenCoordsXY{ GhostFlagColumnSize.width, 0 };
|
|
constexpr auto LastFlagColumnSize = ScreenSize{ 32, 14 };
|
|
|
|
constexpr int32_t PADDING_BOTTOM = 15;
|
|
constexpr int32_t GROUPBOX_PADDING = 6;
|
|
constexpr int32_t HORIZONTAL_GROUPBOX_PADDING = 5;
|
|
constexpr int32_t VERTICAL_GROUPBOX_PADDING = 4;
|
|
constexpr auto PropertyButtonSize = ScreenSize{ 130, 18 };
|
|
constexpr auto PropertyFullWidth = ScreenSize{ 370, 18 };
|
|
|
|
constexpr ScreenCoordsXY PropertyRowCol(ScreenCoordsXY anchor, int32_t row, int32_t column)
|
|
{
|
|
return anchor
|
|
+ ScreenCoordsXY{ column * (PropertyButtonSize.width + HORIZONTAL_GROUPBOX_PADDING),
|
|
row * (PropertyButtonSize.height + VERTICAL_GROUPBOX_PADDING) };
|
|
}
|
|
|
|
constexpr ScreenCoordsXY CheckboxGroupOffset(
|
|
ScreenCoordsXY anchorPoint, int16_t horizontalMultiplier, int16_t verticalMultiplier)
|
|
{
|
|
return anchorPoint + ScreenCoordsXY{ 14 * horizontalMultiplier, 7 * verticalMultiplier };
|
|
}
|
|
|
|
// clang-format off
|
|
// Macros for easily obtaining the top and bottom of a widget inside a properties group box
|
|
#define GBBT(GROUPTOP, row) ((GROUPTOP) + 14 + row * (PropertyButtonSize.height + VERTICAL_GROUPBOX_PADDING))
|
|
#define GBBB(GROUPTOP, row) (GBBT((GROUPTOP), row) + PropertyButtonSize.height)
|
|
|
|
#define MAIN_TILE_INSPECTOR_WIDGETS \
|
|
WINDOW_SHIM(WINDOW_TITLE, WW, WH), \
|
|
MakeWidget({3, 57}, {WW - 6, WH - PADDING_BOTTOM - 58}, WindowWidgetType::Scroll, WindowColour::Secondary, SCROLL_VERTICAL), /* Element list */ \
|
|
/* X and Y spinners */ \
|
|
MakeSpinnerWidgets({20, 23}, {51, 12}, WindowWidgetType::Spinner, WindowColour::Secondary), /* Spinner X (3 widgets) */ \
|
|
MakeSpinnerWidgets({90, 23}, {51, 12}, WindowWidgetType::Spinner, WindowColour::Secondary), /* Spinner Y (3 widgets) */ \
|
|
/* Top buttons */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 0, ToolbarButtonSize, WindowWidgetType::FlatBtn, WindowColour::Secondary, SPR_DEMOLISH, STR_REMOVE_SELECTED_ELEMENT_TIP ), /* Remove button */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 1, ToolbarButtonHalfSize, WindowWidgetType::Button, WindowColour::Secondary, STR_UP, STR_MOVE_SELECTED_ELEMENT_UP_TIP), /* Move up */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 1 + ScreenSize{0, 12}, ToolbarButtonHalfSize, WindowWidgetType::Button, WindowColour::Secondary, STR_DOWN, STR_MOVE_SELECTED_ELEMENT_DOWN_TIP), /* Move down */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 2, ToolbarButtonSize, WindowWidgetType::FlatBtn, WindowColour::Secondary, SPR_ROTATE_ARROW, STR_ROTATE_SELECTED_ELEMENT_TIP), /* Rotate button */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 3, ToolbarButtonSize, WindowWidgetType::FlatBtn, WindowColour::Secondary, SPR_G2_SORT, STR_TILE_INSPECTOR_SORT_TIP), /* Sort button */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 4, ToolbarButtonSize, WindowWidgetType::FlatBtn, WindowColour::Secondary, SPR_G2_PASTE, STR_TILE_INSPECTOR_PASTE_TIP), /* Paste button */ \
|
|
MakeWidget(ToolbarButtonAnchor + ToolbarButtonOffsetX * 5, ToolbarButtonSize, WindowWidgetType::FlatBtn, WindowColour::Secondary, SPR_G2_COPY, STR_TILE_INSPECTOR_COPY_TIP), /* Copy button */ \
|
|
/* Column headers */ \
|
|
MakeWidget(InvisibleFlagColumnXY, InvisibleFlagColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_INVISIBLE_SHORT, STR_TILE_INSPECTOR_FLAG_INVISIBLE), /* Invisible flag */ \
|
|
MakeWidget(TypeColumnXY, TypeColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_ELEMENT_TYPE), /* Type */ \
|
|
MakeWidget(BaseHeightColumnXY, BaseHeightColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_BASE_HEIGHT_SHORT, STR_TILE_INSPECTOR_BASE_HEIGHT), /* Base height */ \
|
|
MakeWidget(ClearanceHeightColumnXY, ClearanceHeightColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_CLEARANGE_HEIGHT_SHORT, STR_TILE_INSPECTOR_CLEARANCE_HEIGHT), /* Clearance height */ \
|
|
MakeWidget(DirectionColumnXY, DirectionColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_DIRECTION_SHORT, STR_TILE_INSPECTOR_DIRECTION), /* Direction */ \
|
|
MakeWidget(GhostFlagColumnXY, GhostFlagColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_FLAG_GHOST_SHORT, STR_TILE_INSPECTOR_FLAG_GHOST), /* Ghost flag */ \
|
|
MakeWidget(LastFlagColumnXY, LastFlagColumnSize, WindowWidgetType::TableHeader, WindowColour::Secondary, STR_TILE_INSPECTOR_FLAG_LAST_SHORT, STR_TILE_INSPECTOR_FLAG_LAST), /* Last of tile flag */ \
|
|
/* Group boxes */ \
|
|
MakeWidget({6, 0}, {WW - 12, 0}, WindowWidgetType::Groupbox, WindowColour::Secondary, STR_NONE, STR_NONE ), /* Details group box */ \
|
|
MakeWidget({6, 0}, {WW - 12, 0}, WindowWidgetType::Groupbox, WindowColour::Secondary, STR_TILE_INSPECTOR_GROUPBOX_PROPERTIES, STR_NONE ) /* Properties group box */
|
|
|
|
static rct_widget DefaultWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumSurfaceProperties = 4;
|
|
constexpr int32_t NumSurfaceDetails = 4;
|
|
constexpr int32_t SurfacePropertiesHeight = 16 + NumSurfaceProperties * 21;
|
|
constexpr int32_t SurfaceDetailsHeight = 20 + NumSurfaceDetails * 11;
|
|
static rct_widget SurfaceWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_SURFACE_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 1, 0), PropertyButtonSize, WindowWidgetType::Button, WindowColour::Secondary, STR_TILE_INSPECTOR_SURFACE_REMOVE_FENCES), // WIDX_SURFACE_BUTTON_REMOVE_FENCES
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 1, 1), PropertyButtonSize, WindowWidgetType::Button, WindowColour::Secondary, STR_TILE_INSPECTOR_SURFACE_RESTORE_FENCES), // WIDX_SURFACE_BUTTON_RESTORE_FENCES
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 1, 0), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SURFACE_CHECK_CORNER_N
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 2, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SURFACE_CHECK_CORNER_E
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 1, 2), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SURFACE_CHECK_CORNER_S
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 0, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SURFACE_CHECK_CORNER_W
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 4, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_SURFACE_DIAGONAL), // WIDX_SURFACE_CHECK_DIAGONAL
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumPathProperties = 5;
|
|
constexpr int32_t NumPathDetails = 3;
|
|
constexpr int32_t PathPropertiesHeight = 16 + NumPathProperties * 21;
|
|
constexpr int32_t PathDetailsHeight = 20 + NumPathDetails * 11;
|
|
static rct_widget PathWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_PATH_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 1, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_PATH_BROKEN), // WIDX_PATH_CHECK_BROKEN
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 2, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_PATH_SLOPED), // WIDX_PATH_CHECK_SLOPED
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 3, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_NE
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 4, 2), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_E
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 3, 3), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_SE
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 2, 4), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_S
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 1, 3), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_SW
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 0, 2), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_W
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 1, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_NW
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 3, 1), 2, 0), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_PATH_CHECK_EDGE_N
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumTrackProperties = 5;
|
|
constexpr int32_t NumTrackDetails = 7;
|
|
constexpr int32_t TrackPropertiesHeight = 16 + NumTrackProperties * 21;
|
|
constexpr int32_t TrackDetailsHeight = 20 + NumTrackDetails * 11;
|
|
static rct_widget TrackWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeWidget(PropertyRowCol({ 12, 0}, 0, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_TRACK_ENTIRE_TRACK_PIECE), // WIDX_TRACK_CHECK_APPLY_TO_ALL
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 1, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_TRACK_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(PropertyRowCol({ 12, 0}, 2, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_TRACK_CHAIN_LIFT), // WIDX_TRACK_CHECK_CHAIN_LIFT
|
|
MakeWidget(PropertyRowCol({ 12, 0}, 3, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_TRACK_BLOCK_BRAKE), // WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED
|
|
MakeWidget(PropertyRowCol({ 12, 0}, 4, 0), PropertyFullWidth, WindowWidgetType::Checkbox, WindowColour::Secondary, STR_TILE_INSPECTOR_TRACK_IS_INDESTRUCTIBLE), // WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumSceneryProperties = 4; // The checkbox groups both count for 2 rows
|
|
constexpr int32_t NumSceneryDetails = 4;
|
|
constexpr int32_t SceneryPropertiesHeight = 16 + NumSceneryProperties * 21;
|
|
constexpr int32_t SceneryDetailsHeight = 20 + NumSceneryDetails * 11;
|
|
static rct_widget SceneryWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_SCENERY_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 1, 0), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_QUARTER_N
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 2, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_QUARTER_E
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 1, 2), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_QUARTER_S
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 0, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_QUARTER_W
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 2, 1), 1, 0), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_COLLISION_N
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 2, 1), 2, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_COLLISION_E
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 2, 1), 1, 2), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_COLLISION_S
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 2, 1), 0, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_SCENERY_CHECK_COLLISION_W
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumEntranceProperties = 2;
|
|
constexpr int32_t NumEntranceDetails = 4;
|
|
constexpr int32_t EntrancePropertiesHeight = 16 + NumEntranceProperties * 21;
|
|
constexpr int32_t EntranceDetailsHeight = 20 + NumEntranceDetails * 11;
|
|
static rct_widget EntranceWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_ENTRANCE_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 1, 0), PropertyButtonSize, WindowWidgetType::Button, WindowColour::Secondary, STR_TILE_INSPECTOR_ENTRANCE_MAKE_USABLE, STR_TILE_INSPECTOR_ENTRANCE_MAKE_USABLE_TIP), // WIDX_ENTRANCE_BUTTON_MAKE_USABLE
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumWallProperties = 3;
|
|
constexpr int32_t NumWallDetails = 2;
|
|
constexpr int32_t WallPropertiesHeight = 16 + NumWallProperties * 21;
|
|
constexpr int32_t WallDetailsHeight = 20 + NumWallDetails * 11;
|
|
static rct_widget WallWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_WALL_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(PropertyRowCol({ 12, 0 }, 1, 1), PropertyButtonSize, WindowWidgetType::DropdownMenu, WindowColour::Secondary), // WIDX_WALL_DROPDOWN_SLOPE
|
|
MakeWidget(PropertyRowCol({ 12 + PropertyButtonSize.width - 12, 0 }, 1, 1), { 11, 12}, WindowWidgetType::Button, WindowColour::Secondary, STR_DROPDOWN_GLYPH), // WIDX_WALL_DROPDOWN_SLOPE_BUTTON
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 2, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_WALL_SPINNER_ANIMATION_FRAME{,_INCREASE,_DECREASE}
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumLargeSceneryProperties = 1;
|
|
constexpr int32_t NumLargeSceneryDetails = 3;
|
|
constexpr int32_t LargeSceneryPropertiesHeight = 16 + NumLargeSceneryProperties * 21;
|
|
constexpr int32_t LargeSceneryDetailsHeight = 20 + NumLargeSceneryDetails * 11;
|
|
static rct_widget LargeSceneryWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_LARGE_SCENERY_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
WIDGETS_END,
|
|
};
|
|
|
|
constexpr int32_t NumBannerProperties = 3;
|
|
constexpr int32_t NumBannerDetails = 1;
|
|
constexpr int32_t BannerPropertiesHeight = 16 + NumBannerProperties * 21;
|
|
constexpr int32_t BannerDetailsHeight = 20 + NumBannerDetails * 11;
|
|
static rct_widget BannerWidgets[] = {
|
|
MAIN_TILE_INSPECTOR_WIDGETS,
|
|
MakeSpinnerWidgets(PropertyRowCol({ 12, 0 }, 0, 1), PropertyButtonSize, WindowWidgetType::Spinner, WindowColour::Secondary), // WIDX_BANNER_SPINNER_HEIGHT{,_INCREASE,_DECREASE}
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 3, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_BANNER_CHECK_BLOCK_NE
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 3, 3), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_BANNER_CHECK_BLOCK_SE
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 1, 3), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_BANNER_CHECK_BLOCK_SW
|
|
MakeWidget(CheckboxGroupOffset(PropertyRowCol({ 12, 0 }, 1, 1), 1, 1), { 12, 12 }, WindowWidgetType::Checkbox, WindowColour::Secondary), // WIDX_BANNER_CHECK_BLOCK_NW
|
|
|
|
WIDGETS_END,
|
|
};
|
|
|
|
static rct_widget *PageWidgets[] = {
|
|
DefaultWidgets,
|
|
SurfaceWidgets,
|
|
PathWidgets,
|
|
TrackWidgets,
|
|
SceneryWidgets,
|
|
EntranceWidgets,
|
|
WallWidgets,
|
|
LargeSceneryWidgets,
|
|
BannerWidgets,
|
|
};
|
|
// clang-format on
|
|
|
|
struct TileInspectorGroupboxSettings
|
|
{
|
|
// Offsets from the bottom of the window
|
|
int16_t details_top_offset, details_bottom_offset;
|
|
int16_t properties_top_offset, properties_bottom_offset;
|
|
// String to be displayed in the details groupbox
|
|
rct_string_id string_id;
|
|
};
|
|
|
|
static constexpr TileInspectorGroupboxSettings MakeGroupboxSettings(
|
|
int16_t detailsHeight, int16_t propertiesHeight, rct_string_id stringId)
|
|
{
|
|
TileInspectorGroupboxSettings settings{};
|
|
decltype(settings.properties_bottom_offset) offsetSum = 0;
|
|
settings.properties_bottom_offset = (offsetSum += PADDING_BOTTOM);
|
|
settings.properties_top_offset = (offsetSum += propertiesHeight);
|
|
settings.details_bottom_offset = (offsetSum += GROUPBOX_PADDING);
|
|
settings.details_top_offset = (offsetSum += detailsHeight);
|
|
settings.string_id = stringId;
|
|
return settings;
|
|
}
|
|
|
|
static constexpr TileInspectorGroupboxSettings PageGroupBoxSettings[] = {
|
|
MakeGroupboxSettings(SurfaceDetailsHeight, SurfacePropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_SURFACE_INFO),
|
|
MakeGroupboxSettings(PathDetailsHeight, PathPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_PATH_INFO),
|
|
MakeGroupboxSettings(TrackDetailsHeight, TrackPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_TRACK_INFO),
|
|
MakeGroupboxSettings(SceneryDetailsHeight, SceneryPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_SCENERY_INFO),
|
|
MakeGroupboxSettings(EntranceDetailsHeight, EntrancePropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_ENTRANCE_INFO),
|
|
MakeGroupboxSettings(WallDetailsHeight, WallPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_WALL_INFO),
|
|
MakeGroupboxSettings(LargeSceneryDetailsHeight, LargeSceneryPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_BANNER_INFO),
|
|
MakeGroupboxSettings(BannerDetailsHeight, BannerPropertiesHeight, STR_TILE_INSPECTOR_GROUPBOX_BANNER_INFO),
|
|
};
|
|
|
|
static constexpr int32_t ViewportInteractionFlags = EnumsToFlags(
|
|
ViewportInteractionItem::Terrain, ViewportInteractionItem::Entity, ViewportInteractionItem::Ride,
|
|
ViewportInteractionItem::Scenery, ViewportInteractionItem::Footpath, ViewportInteractionItem::FootpathItem,
|
|
ViewportInteractionItem::ParkEntrance, ViewportInteractionItem::Wall, ViewportInteractionItem::LargeScenery,
|
|
ViewportInteractionItem::Banner);
|
|
|
|
static int16_t windowTileInspectorHighlightedIndex = -1;
|
|
static bool windowTileInspectorTileSelected = false;
|
|
static int32_t windowTileInspectorToolMouseX = 0;
|
|
static int32_t windowTileInspectorToolMouseY = 0;
|
|
static bool windowTileInspectorToolCtrlDown = false;
|
|
static CoordsXY windowTileInspectorToolMap = {};
|
|
static bool windowTileInspectorApplyToAll = false;
|
|
static bool windowTileInspectorElementCopied = false;
|
|
static TileElement tileInspectorCopiedElement;
|
|
|
|
static void WindowTileInspectorMouseup(rct_window* w, rct_widgetindex widgetIndex);
|
|
static void WindowTileInspectorResize(rct_window* w);
|
|
static void WindowTileInspectorMousedown(rct_window* w, rct_widgetindex widgetIndex, rct_widget* widget);
|
|
static void WindowTileInspectorUpdate(rct_window* w);
|
|
static void WindowTileInspectorDropdown(rct_window* w, rct_widgetindex widgetIndex, int32_t dropdownIndex);
|
|
static void WindowTileInspectorToolUpdate(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorUpdateSelectedTile(rct_window* w, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorToolDown(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorToolDrag(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorScrollgetsize(rct_window* w, int32_t scrollIndex, int32_t* width, int32_t* height);
|
|
static void WindowTileInspectorScrollmousedown(rct_window* w, int32_t scrollIndex, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorScrollmouseover(rct_window* w, int32_t scrollIndex, const ScreenCoordsXY& screenCoords);
|
|
static void WindowTileInspectorInvalidate(rct_window* w);
|
|
static void WindowTileInspectorPaint(rct_window* w, rct_drawpixelinfo* dpi);
|
|
static void WindowTileInspectorScrollpaint(rct_window* w, rct_drawpixelinfo* dpi, int32_t scrollIndex);
|
|
static void WindowTileInspectorSetPage(rct_window* w, const TileInspectorPage page);
|
|
static void WindowTileInspectorClose(rct_window* w);
|
|
|
|
static rct_window_event_list TileInspectorWindowEvents([](auto& events) {
|
|
events.mouse_up = &WindowTileInspectorMouseup;
|
|
events.resize = &WindowTileInspectorResize;
|
|
events.mouse_down = &WindowTileInspectorMousedown;
|
|
events.dropdown = &WindowTileInspectorDropdown;
|
|
events.update = &WindowTileInspectorUpdate;
|
|
events.tool_update = &WindowTileInspectorToolUpdate;
|
|
events.tool_down = &WindowTileInspectorToolDown;
|
|
events.tool_drag = &WindowTileInspectorToolDrag;
|
|
events.get_scroll_size = &WindowTileInspectorScrollgetsize;
|
|
events.scroll_mousedown = &WindowTileInspectorScrollmousedown;
|
|
events.scroll_mouseover = &WindowTileInspectorScrollmouseover;
|
|
events.invalidate = &WindowTileInspectorInvalidate;
|
|
events.paint = &WindowTileInspectorPaint;
|
|
events.scroll_paint = &WindowTileInspectorScrollpaint;
|
|
events.close = &WindowTileInspectorClose;
|
|
});
|
|
|
|
// clang-format off
|
|
|
|
static uint64_t PageHoldDownWidgets[] = {
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_SURFACE_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_SURFACE_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_PATH_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_PATH_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_TRACK_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_TRACK_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_SCENERY_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_SCENERY_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_ENTRANCE_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_ENTRANCE_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_WALL_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_WALL_SPINNER_HEIGHT_DECREASE) | (1ULL << WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE) | (1ULL << WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_LARGE_SCENERY_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_LARGE_SCENERY_SPINNER_HEIGHT_DECREASE),
|
|
(1ULL << WIDX_SPINNER_X_INCREASE) | (1ULL << WIDX_SPINNER_X_DECREASE) | (1ULL << WIDX_SPINNER_Y_INCREASE) | (1ULL << WIDX_SPINNER_Y_DECREASE) | (1ULL << WIDX_BANNER_SPINNER_HEIGHT_INCREASE) | (1ULL << WIDX_BANNER_SPINNER_HEIGHT_DECREASE),
|
|
};
|
|
|
|
static uint64_t PageDisabledWidgets[] = {
|
|
(1ULL << WIDX_BUTTON_MOVE_UP) | (1ULL << WIDX_BUTTON_MOVE_DOWN) | (1ULL << WIDX_BUTTON_REMOVE) | (1ULL << WIDX_BUTTON_ROTATE) | (1ULL << WIDX_BUTTON_COPY),
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
(1ULL << WIDX_BUTTON_ROTATE),
|
|
0,
|
|
};
|
|
// clang-format on
|
|
|
|
rct_window* WindowTileInspectorOpen()
|
|
{
|
|
rct_window* window;
|
|
|
|
// Check if window is already open
|
|
window = window_bring_to_front_by_class(WC_TILE_INSPECTOR);
|
|
if (window != nullptr)
|
|
return window;
|
|
|
|
window = WindowCreate(ScreenCoordsXY(0, 29), WW, WH, &TileInspectorWindowEvents, WC_TILE_INSPECTOR, WF_RESIZABLE);
|
|
|
|
WindowTileInspectorSetPage(window, TileInspectorPage::Default);
|
|
window->min_width = MIN_WW;
|
|
window->min_height = MIN_WH;
|
|
window->max_width = MAX_WW;
|
|
window->max_height = MAX_WH;
|
|
windowTileInspectorSelectedIndex = -1;
|
|
WindowInitScrollWidgets(*window);
|
|
|
|
windowTileInspectorTileSelected = false;
|
|
|
|
tool_set(*window, WIDX_BACKGROUND, Tool::Crosshair);
|
|
|
|
return window;
|
|
}
|
|
|
|
void WindowTileInspectorClearClipboard()
|
|
{
|
|
windowTileInspectorElementCopied = false;
|
|
}
|
|
|
|
static TileElement* WindowTileInspectorGetSelectedElement(rct_window* w)
|
|
{
|
|
openrct2_assert(
|
|
windowTileInspectorSelectedIndex >= 0 && windowTileInspectorSelectedIndex < windowTileInspectorElementCount,
|
|
"Selected list item out of range");
|
|
return map_get_first_element_at(windowTileInspectorToolMap) + windowTileInspectorSelectedIndex;
|
|
}
|
|
|
|
static void WindowTileInspectorSelectElementFromList(rct_window* w, int32_t index)
|
|
{
|
|
if (index < 0 || index >= windowTileInspectorElementCount)
|
|
{
|
|
windowTileInspectorSelectedIndex = -1;
|
|
OpenRCT2::TileInspector::SetSelectedElement(nullptr);
|
|
}
|
|
else
|
|
{
|
|
windowTileInspectorSelectedIndex = index;
|
|
|
|
const TileElement* const tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
OpenRCT2::TileInspector::SetSelectedElement(tileElement);
|
|
}
|
|
|
|
w->Invalidate();
|
|
}
|
|
|
|
static void WindowTileInspectorLoadTile(rct_window* w, TileElement* elementToSelect)
|
|
{
|
|
windowTileInspectorSelectedIndex = -1;
|
|
w->scrolls[0].v_top = 0;
|
|
|
|
TileElement* element = map_get_first_element_at(windowTileInspectorToolMap);
|
|
int16_t numItems = 0;
|
|
do
|
|
{
|
|
if (element == nullptr)
|
|
break;
|
|
if (element == elementToSelect)
|
|
{
|
|
windowTileInspectorSelectedIndex = numItems;
|
|
}
|
|
|
|
numItems++;
|
|
} while (!(element++)->IsLastForTile());
|
|
|
|
windowTileInspectorElementCount = numItems;
|
|
|
|
w->Invalidate();
|
|
}
|
|
|
|
static void WindowTileInspectorRemoveElement(int32_t elementIndex)
|
|
{
|
|
openrct2_assert(elementIndex >= 0 && elementIndex < windowTileInspectorElementCount, "elementIndex out of range");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnyRemove, elementIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorRotateElement(int32_t elementIndex)
|
|
{
|
|
openrct2_assert(elementIndex >= 0 && elementIndex < windowTileInspectorElementCount, "elementIndex out of range");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnyRotate, elementIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
// Swap element with its parent
|
|
static void WindowTileInspectorSwapElements(int16_t first, int16_t second)
|
|
{
|
|
bool firstInRange = first >= 0 && first < windowTileInspectorElementCount;
|
|
bool secondInRange = second >= 0 && second < windowTileInspectorElementCount;
|
|
// This might happen if two people are modifying the same tile.
|
|
if (!firstInRange || !secondInRange)
|
|
return;
|
|
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnySwap, first, second);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorSortElements()
|
|
{
|
|
openrct2_assert(windowTileInspectorTileSelected, "No tile selected");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnySort);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorCopyElement(rct_window* w)
|
|
{
|
|
// Copy value, in case the element gets moved
|
|
tileInspectorCopiedElement = *WindowTileInspectorGetSelectedElement(w);
|
|
windowTileInspectorElementCopied = true;
|
|
w->Invalidate();
|
|
}
|
|
|
|
static void WindowTileInspectorPasteElement(rct_window* w)
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnyPaste, 0, 0, tileInspectorCopiedElement);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorBaseHeightOffset(int16_t elementIndex, int8_t heightOffset)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::AnyBaseHeightOffset, elementIndex, heightOffset);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorSurfaceShowParkFences(bool showFences)
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::SurfaceShowParkFences, showFences);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorSurfaceToggleCorner(int32_t cornerIndex)
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::SurfaceToggleCorner, cornerIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorSurfaceToggleDiagonal()
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::SurfaceToggleDiagonal);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorPathSetSloped(int32_t elementIndex, bool sloped)
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::PathSetSlope, elementIndex, sloped);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorPathSetBroken(int32_t elementIndex, bool broken)
|
|
{
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::PathSetBroken, elementIndex, broken);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorPathToggleEdge(int32_t elementIndex, int32_t cornerIndex)
|
|
{
|
|
openrct2_assert(elementIndex >= 0 && elementIndex < windowTileInspectorElementCount, "elementIndex out of range");
|
|
openrct2_assert(cornerIndex >= 0 && cornerIndex < 8, "cornerIndex out of range");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::PathToggleEdge, elementIndex, cornerIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorEntranceMakeUsable(int32_t elementIndex)
|
|
{
|
|
Guard::ArgumentInRange(elementIndex, 0, windowTileInspectorElementCount - 1);
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::EntranceMakeUsable, elementIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorWallSetSlope(int32_t elementIndex, int32_t slopeValue)
|
|
{
|
|
// Make sure only the correct bits are set
|
|
openrct2_assert((slopeValue & 3) == slopeValue, "slopeValue doesn't match its mask");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::WallSetSlope, elementIndex, slopeValue);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorWallAnimationFrameOffset(int16_t elementIndex, int8_t animationFrameOffset)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::WallSetAnimationFrame, elementIndex, animationFrameOffset);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorTrackBlockHeightOffset(int32_t elementIndex, int8_t heightOffset)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::TrackBaseHeightOffset, elementIndex, heightOffset);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorTrackBlockSetLift(int32_t elementIndex, bool entireTrackBlock, bool chain)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, entireTrackBlock ? TileModifyType::TrackSetChainBlock : TileModifyType::TrackSetChain,
|
|
elementIndex, chain);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorTrackSetBlockBrake(int32_t elementIndex, bool blockBrake)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::TrackSetBlockBrake, elementIndex, blockBrake);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorTrackSetIndestructible(int32_t elementIndex, bool isIndestructible)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::TrackSetIndestructible, elementIndex, isIndestructible);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorQuarterTileSet(int32_t elementIndex, const int32_t quarterIndex)
|
|
{
|
|
// quarterIndex is widget index relative to WIDX_SCENERY_CHECK_QUARTER_N, so a value from 0-3
|
|
openrct2_assert(quarterIndex >= 0 && quarterIndex < 4, "quarterIndex out of range");
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::ScenerySetQuarterLocation, elementIndex,
|
|
(quarterIndex - get_current_rotation()) & 3);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorToggleQuadrantCollosion(int32_t elementIndex, const int32_t quadrantIndex)
|
|
{
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::ScenerySetQuarterCollision, elementIndex,
|
|
(quadrantIndex + 2 - get_current_rotation()) & 3);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorBannerToggleBlock(int32_t elementIndex, int32_t edgeIndex)
|
|
{
|
|
openrct2_assert(edgeIndex >= 0 && edgeIndex < 4, "edgeIndex out of range");
|
|
|
|
// Make edgeIndex abstract
|
|
edgeIndex = (edgeIndex - get_current_rotation()) & 3;
|
|
auto modifyTile = TileModifyAction(
|
|
windowTileInspectorToolMap, TileModifyType::BannerToggleBlockingEdge, elementIndex, edgeIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorToggleInvisibility(int32_t elementIndex)
|
|
{
|
|
openrct2_assert(elementIndex >= 0 && elementIndex < windowTileInspectorElementCount, "elementIndex out of range");
|
|
auto modifyTile = TileModifyAction(windowTileInspectorToolMap, TileModifyType::AnyToggleInvisilibity, elementIndex);
|
|
GameActions::Execute(&modifyTile);
|
|
}
|
|
|
|
static void WindowTileInspectorClose(rct_window* w)
|
|
{
|
|
OpenRCT2::TileInspector::SetSelectedElement(nullptr);
|
|
}
|
|
|
|
static void WindowTileInspectorMouseup(rct_window* w, rct_widgetindex widgetIndex)
|
|
{
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_CLOSE:
|
|
tool_cancel();
|
|
window_close(w);
|
|
return;
|
|
case WIDX_BUTTON_REMOVE:
|
|
{
|
|
int32_t nextItemToSelect = windowTileInspectorSelectedIndex - 1;
|
|
WindowTileInspectorRemoveElement(windowTileInspectorSelectedIndex);
|
|
WindowTileInspectorSelectElementFromList(w, nextItemToSelect);
|
|
break;
|
|
}
|
|
case WIDX_BUTTON_ROTATE:
|
|
WindowTileInspectorRotateElement(windowTileInspectorSelectedIndex);
|
|
break;
|
|
case WIDX_BUTTON_SORT:
|
|
WindowTileInspectorSortElements();
|
|
break;
|
|
case WIDX_BUTTON_COPY:
|
|
WindowTileInspectorCopyElement(w);
|
|
break;
|
|
case WIDX_BUTTON_PASTE:
|
|
WindowTileInspectorPasteElement(w);
|
|
break;
|
|
case WIDX_BUTTON_MOVE_UP:
|
|
WindowTileInspectorSwapElements(windowTileInspectorSelectedIndex, windowTileInspectorSelectedIndex + 1);
|
|
break;
|
|
case WIDX_BUTTON_MOVE_DOWN:
|
|
WindowTileInspectorSwapElements(windowTileInspectorSelectedIndex - 1, windowTileInspectorSelectedIndex);
|
|
break;
|
|
}
|
|
|
|
// Only element-specific widgets from now on
|
|
if (w->tileInspectorPage == TileInspectorPage::Default || windowTileInspectorSelectedIndex == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
TileElement* const tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
|
|
// Update selection, can be nullptr.
|
|
OpenRCT2::TileInspector::SetSelectedElement(tileElement);
|
|
|
|
if (tileElement == nullptr)
|
|
return;
|
|
|
|
// Page widgets
|
|
switch (tileElement->GetType())
|
|
{
|
|
case TileElementType::Surface:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_SURFACE_BUTTON_REMOVE_FENCES:
|
|
WindowTileInspectorSurfaceShowParkFences(false);
|
|
break;
|
|
case WIDX_SURFACE_BUTTON_RESTORE_FENCES:
|
|
WindowTileInspectorSurfaceShowParkFences(true);
|
|
break;
|
|
case WIDX_SURFACE_CHECK_CORNER_N:
|
|
case WIDX_SURFACE_CHECK_CORNER_E:
|
|
case WIDX_SURFACE_CHECK_CORNER_S:
|
|
case WIDX_SURFACE_CHECK_CORNER_W:
|
|
WindowTileInspectorSurfaceToggleCorner(
|
|
((widgetIndex - WIDX_SURFACE_CHECK_CORNER_N) + 2 - get_current_rotation()) & 3);
|
|
break;
|
|
case WIDX_SURFACE_CHECK_DIAGONAL:
|
|
WindowTileInspectorSurfaceToggleDiagonal();
|
|
break;
|
|
} // switch widgetindex
|
|
break;
|
|
|
|
case TileElementType::Path:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_PATH_CHECK_SLOPED:
|
|
WindowTileInspectorPathSetSloped(windowTileInspectorSelectedIndex, !tileElement->AsPath()->IsSloped());
|
|
break;
|
|
case WIDX_PATH_CHECK_BROKEN:
|
|
WindowTileInspectorPathSetBroken(windowTileInspectorSelectedIndex, !tileElement->AsPath()->IsBroken());
|
|
break;
|
|
case WIDX_PATH_CHECK_EDGE_E:
|
|
case WIDX_PATH_CHECK_EDGE_S:
|
|
case WIDX_PATH_CHECK_EDGE_W:
|
|
case WIDX_PATH_CHECK_EDGE_N:
|
|
{
|
|
// 0 = east/right, 1 = south/bottom, 2 = west/left, 3 = north/top
|
|
const int32_t eswn = (widgetIndex - WIDX_PATH_CHECK_EDGE_E) / 2;
|
|
// Transform to world orientation
|
|
const int32_t index = (eswn - get_current_rotation()) & 3;
|
|
WindowTileInspectorPathToggleEdge(
|
|
windowTileInspectorSelectedIndex,
|
|
index + 4); // The corners are stored in the 4 most significant bits, hence the + 4
|
|
break;
|
|
}
|
|
case WIDX_PATH_CHECK_EDGE_NE:
|
|
case WIDX_PATH_CHECK_EDGE_SE:
|
|
case WIDX_PATH_CHECK_EDGE_SW:
|
|
case WIDX_PATH_CHECK_EDGE_NW:
|
|
{
|
|
// 0 = NE, 1 = SE, 2 = SW, 3 = NW
|
|
const int32_t neseswnw = (widgetIndex - WIDX_PATH_CHECK_EDGE_NE) / 2;
|
|
// Transform to world orientation
|
|
const int32_t index = (neseswnw - get_current_rotation()) & 3;
|
|
WindowTileInspectorPathToggleEdge(windowTileInspectorSelectedIndex, index);
|
|
break;
|
|
}
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Track:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_TRACK_CHECK_APPLY_TO_ALL:
|
|
windowTileInspectorApplyToAll ^= 1;
|
|
widget_invalidate(w, widgetIndex);
|
|
break;
|
|
case WIDX_TRACK_CHECK_CHAIN_LIFT:
|
|
{
|
|
bool entireTrackBlock = WidgetIsPressed(*w, WIDX_TRACK_CHECK_APPLY_TO_ALL);
|
|
bool newLift = !tileElement->AsTrack()->HasChain();
|
|
WindowTileInspectorTrackBlockSetLift(windowTileInspectorSelectedIndex, entireTrackBlock, newLift);
|
|
break;
|
|
}
|
|
case WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED:
|
|
WindowTileInspectorTrackSetBlockBrake(
|
|
windowTileInspectorSelectedIndex, !tileElement->AsTrack()->BlockBrakeClosed());
|
|
break;
|
|
case WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE:
|
|
WindowTileInspectorTrackSetIndestructible(
|
|
windowTileInspectorSelectedIndex, !tileElement->AsTrack()->IsIndestructible());
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::SmallScenery:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_SCENERY_CHECK_QUARTER_N:
|
|
case WIDX_SCENERY_CHECK_QUARTER_E:
|
|
case WIDX_SCENERY_CHECK_QUARTER_S:
|
|
case WIDX_SCENERY_CHECK_QUARTER_W:
|
|
WindowTileInspectorQuarterTileSet(
|
|
windowTileInspectorSelectedIndex, widgetIndex - WIDX_SCENERY_CHECK_QUARTER_N);
|
|
break;
|
|
case WIDX_SCENERY_CHECK_COLLISION_N:
|
|
case WIDX_SCENERY_CHECK_COLLISION_E:
|
|
case WIDX_SCENERY_CHECK_COLLISION_S:
|
|
case WIDX_SCENERY_CHECK_COLLISION_W:
|
|
WindowTileInspectorToggleQuadrantCollosion(
|
|
windowTileInspectorSelectedIndex, widgetIndex - WIDX_SCENERY_CHECK_COLLISION_N);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Entrance:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_ENTRANCE_BUTTON_MAKE_USABLE:
|
|
WindowTileInspectorEntranceMakeUsable(windowTileInspectorSelectedIndex);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Banner:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_BANNER_CHECK_BLOCK_NE:
|
|
case WIDX_BANNER_CHECK_BLOCK_SE:
|
|
case WIDX_BANNER_CHECK_BLOCK_SW:
|
|
case WIDX_BANNER_CHECK_BLOCK_NW:
|
|
WindowTileInspectorBannerToggleBlock(
|
|
windowTileInspectorSelectedIndex, widgetIndex - WIDX_BANNER_CHECK_BLOCK_NE);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::LargeScenery:
|
|
case TileElementType::Wall:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorResize(rct_window* w)
|
|
{
|
|
if (w->width < w->min_width)
|
|
{
|
|
w->Invalidate();
|
|
w->width = w->min_width;
|
|
}
|
|
if (w->height < w->min_height)
|
|
{
|
|
w->Invalidate();
|
|
w->height = w->min_height;
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorMousedown(rct_window* w, rct_widgetindex widgetIndex, rct_widget* widget)
|
|
{
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_SPINNER_X_INCREASE:
|
|
windowTileInspectorTile.x = std::min<int32_t>(windowTileInspectorTile.x + 1, MAXIMUM_MAP_SIZE_TECHNICAL - 1);
|
|
windowTileInspectorToolMap.x = std::min<int32_t>(windowTileInspectorToolMap.x + 32, MAXIMUM_TILE_START_XY);
|
|
WindowTileInspectorLoadTile(w, nullptr);
|
|
break;
|
|
case WIDX_SPINNER_X_DECREASE:
|
|
windowTileInspectorTile.x = std::max<int32_t>(windowTileInspectorTile.x - 1, 0);
|
|
windowTileInspectorToolMap.x = std::max<int32_t>(windowTileInspectorToolMap.x - 32, 0);
|
|
WindowTileInspectorLoadTile(w, nullptr);
|
|
break;
|
|
case WIDX_SPINNER_Y_INCREASE:
|
|
windowTileInspectorTile.y = std::min<int32_t>(windowTileInspectorTile.y + 1, MAXIMUM_MAP_SIZE_TECHNICAL - 1);
|
|
windowTileInspectorToolMap.y = std::min<int32_t>(windowTileInspectorToolMap.y + 32, MAXIMUM_TILE_START_XY);
|
|
WindowTileInspectorLoadTile(w, nullptr);
|
|
break;
|
|
case WIDX_SPINNER_Y_DECREASE:
|
|
windowTileInspectorTile.y = std::max<int32_t>(windowTileInspectorTile.y - 1, 0);
|
|
windowTileInspectorToolMap.y = std::max<int32_t>(windowTileInspectorToolMap.y - 32, 0);
|
|
WindowTileInspectorLoadTile(w, nullptr);
|
|
break;
|
|
} // switch widget index
|
|
|
|
// Only element-specific widgets from now on
|
|
if (w->tileInspectorPage == TileInspectorPage::Default || windowTileInspectorSelectedIndex == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const TileElement* tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
if (tileElement == nullptr)
|
|
return;
|
|
|
|
switch (tileElement->GetType())
|
|
{
|
|
case TileElementType::Surface:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_SURFACE_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_SURFACE_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Path:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_PATH_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_PATH_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Track:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_TRACK_SPINNER_HEIGHT_INCREASE:
|
|
if (WidgetIsPressed(*w, WIDX_TRACK_CHECK_APPLY_TO_ALL))
|
|
{
|
|
WindowTileInspectorTrackBlockHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
}
|
|
else
|
|
{
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
}
|
|
break;
|
|
case WIDX_TRACK_SPINNER_HEIGHT_DECREASE:
|
|
if (WidgetIsPressed(*w, WIDX_TRACK_CHECK_APPLY_TO_ALL))
|
|
{
|
|
WindowTileInspectorTrackBlockHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
}
|
|
else
|
|
{
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
}
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::SmallScenery:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_SCENERY_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_SCENERY_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Entrance:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_ENTRANCE_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_ENTRANCE_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
case WIDX_ENTRANCE_BUTTON_MAKE_USABLE:
|
|
WindowTileInspectorEntranceMakeUsable(windowTileInspectorSelectedIndex);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Wall:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_WALL_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_WALL_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
case WIDX_WALL_DROPDOWN_SLOPE_BUTTON:
|
|
{
|
|
// Use dropdown instead of dropdown button
|
|
widget--;
|
|
|
|
// Fill dropdown list
|
|
gDropdownItems[0].Format = STR_DROPDOWN_MENU_LABEL;
|
|
gDropdownItems[1].Format = STR_DROPDOWN_MENU_LABEL;
|
|
gDropdownItems[2].Format = STR_DROPDOWN_MENU_LABEL;
|
|
gDropdownItems[0].Args = STR_TILE_INSPECTOR_WALL_FLAT;
|
|
gDropdownItems[1].Args = STR_TILE_INSPECTOR_WALL_SLOPED_LEFT;
|
|
gDropdownItems[2].Args = STR_TILE_INSPECTOR_WALL_SLOPED_RIGHT;
|
|
WindowDropdownShowTextCustomWidth(
|
|
{ w->windowPos.x + widget->left, w->windowPos.y + widget->top }, widget->height() + 1, w->colours[1], 0,
|
|
Dropdown::Flag::StayOpen, 3, widget->width() - 3);
|
|
|
|
// Set current value as checked
|
|
Dropdown::SetChecked(tileElement->AsWall()->GetSlope(), true);
|
|
break;
|
|
}
|
|
case WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE:
|
|
WindowTileInspectorWallAnimationFrameOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE:
|
|
WindowTileInspectorWallAnimationFrameOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::LargeScenery:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_LARGE_SCENERY_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_LARGE_SCENERY_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
case TileElementType::Banner:
|
|
switch (widgetIndex)
|
|
{
|
|
case WIDX_BANNER_SPINNER_HEIGHT_INCREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, 1);
|
|
break;
|
|
case WIDX_BANNER_SPINNER_HEIGHT_DECREASE:
|
|
WindowTileInspectorBaseHeightOffset(windowTileInspectorSelectedIndex, -1);
|
|
break;
|
|
} // switch widget index
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorUpdate(rct_window* w)
|
|
{
|
|
// Check if the mouse is hovering over the list
|
|
if (!WidgetIsHighlighted(*w, WIDX_LIST))
|
|
{
|
|
windowTileInspectorHighlightedIndex = -1;
|
|
widget_invalidate(w, WIDX_LIST);
|
|
}
|
|
|
|
if (gCurrentToolWidget.window_classification != WC_TILE_INSPECTOR)
|
|
window_close(w);
|
|
}
|
|
|
|
static void WindowTileInspectorDropdown(rct_window* w, rct_widgetindex widgetIndex, int32_t dropdownIndex)
|
|
{
|
|
if (dropdownIndex == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Get selected element
|
|
TileElement* const tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
|
|
if (w->tileInspectorPage == TileInspectorPage::Wall)
|
|
{
|
|
openrct2_assert(tileElement->GetType() == TileElementType::Wall, "Element is not a wall");
|
|
|
|
if (widgetIndex == WIDX_WALL_DROPDOWN_SLOPE_BUTTON)
|
|
{
|
|
WindowTileInspectorWallSetSlope(windowTileInspectorSelectedIndex, dropdownIndex);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorToolUpdate(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
map_invalidate_selection_rect();
|
|
|
|
gMapSelectFlags |= MAP_SELECT_FLAG_ENABLE;
|
|
|
|
CoordsXY mapCoords;
|
|
TileElement* clickedElement = nullptr;
|
|
bool mouseOnViewport = false;
|
|
if (InputTestPlaceObjectModifier(PLACE_OBJECT_MODIFIER_COPY_Z))
|
|
{
|
|
auto info = get_map_coordinates_from_pos(screenCoords, ViewportInteractionFlags);
|
|
clickedElement = info.Element;
|
|
mapCoords = info.Loc;
|
|
}
|
|
|
|
// Even if Ctrl was pressed, fall back to normal selection when there was nothing under the cursor
|
|
if (clickedElement == nullptr)
|
|
{
|
|
auto mouseCoords = screen_pos_to_map_pos(screenCoords, nullptr);
|
|
if (mouseCoords.has_value())
|
|
{
|
|
mouseOnViewport = true;
|
|
mapCoords = mouseCoords.value();
|
|
}
|
|
}
|
|
|
|
if (mouseOnViewport)
|
|
{
|
|
gMapSelectPositionA = gMapSelectPositionB = mapCoords;
|
|
}
|
|
else if (windowTileInspectorTileSelected)
|
|
{
|
|
gMapSelectPositionA = gMapSelectPositionB = windowTileInspectorToolMap;
|
|
}
|
|
else
|
|
{
|
|
gMapSelectFlags &= ~MAP_SELECT_FLAG_ENABLE;
|
|
}
|
|
|
|
gMapSelectType = MAP_SELECT_TYPE_FULL;
|
|
map_invalidate_selection_rect();
|
|
}
|
|
|
|
static void WindowTileInspectorUpdateSelectedTile(rct_window* w, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
const bool ctrlIsHeldDown = InputTestPlaceObjectModifier(PLACE_OBJECT_MODIFIER_COPY_Z);
|
|
|
|
// Mouse hasn't moved
|
|
if (screenCoords.x == windowTileInspectorToolMouseX && screenCoords.y == windowTileInspectorToolMouseY
|
|
&& windowTileInspectorToolCtrlDown == ctrlIsHeldDown)
|
|
{
|
|
return;
|
|
}
|
|
|
|
windowTileInspectorToolMouseX = screenCoords.x;
|
|
windowTileInspectorToolMouseY = screenCoords.y;
|
|
windowTileInspectorToolCtrlDown = ctrlIsHeldDown;
|
|
|
|
CoordsXY mapCoords{};
|
|
TileElement* clickedElement = nullptr;
|
|
if (ctrlIsHeldDown)
|
|
{
|
|
auto info = get_map_coordinates_from_pos(screenCoords, ViewportInteractionFlags);
|
|
clickedElement = info.Element;
|
|
mapCoords = info.Loc;
|
|
}
|
|
|
|
// Even if Ctrl was pressed, fall back to normal selection when there was nothing under the cursor
|
|
if (clickedElement == nullptr)
|
|
{
|
|
auto mouseCoords = screen_pos_to_map_pos(screenCoords, nullptr);
|
|
|
|
if (!mouseCoords.has_value())
|
|
{
|
|
return;
|
|
}
|
|
|
|
mapCoords = mouseCoords.value();
|
|
// Tile is already selected
|
|
if (windowTileInspectorTileSelected && mapCoords.x == windowTileInspectorToolMap.x
|
|
&& mapCoords.y == windowTileInspectorToolMap.y)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
windowTileInspectorTileSelected = true;
|
|
windowTileInspectorToolMap = mapCoords;
|
|
windowTileInspectorTile = TileCoordsXY(mapCoords);
|
|
|
|
OpenRCT2::TileInspector::SetSelectedElement(clickedElement);
|
|
|
|
WindowTileInspectorLoadTile(w, clickedElement);
|
|
}
|
|
|
|
static void WindowTileInspectorToolDown(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
WindowTileInspectorUpdateSelectedTile(w, screenCoords);
|
|
}
|
|
|
|
static void WindowTileInspectorToolDrag(rct_window* w, rct_widgetindex widgetIndex, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
WindowTileInspectorUpdateSelectedTile(w, screenCoords);
|
|
}
|
|
|
|
static void WindowTileInspectorScrollgetsize(rct_window* w, int32_t scrollIndex, int32_t* width, int32_t* height)
|
|
{
|
|
*width = WW - 30;
|
|
*height = w->widgets[WIDX_LIST].height() - 2;
|
|
*height = std::max(windowTileInspectorElementCount * SCROLLABLE_ROW_HEIGHT, *height);
|
|
}
|
|
|
|
static void WindowTileInspectorSetPage(rct_window* w, const TileInspectorPage page)
|
|
{
|
|
// Invalidate the window already, because the size may change
|
|
w->Invalidate();
|
|
|
|
// subtract current page height, then add new page height
|
|
if (w->tileInspectorPage != TileInspectorPage::Default)
|
|
{
|
|
auto index = EnumValue(w->tileInspectorPage) - 1;
|
|
w->height -= PageGroupBoxSettings[index].details_top_offset - GROUPBOX_PADDING - 3;
|
|
w->min_height -= PageGroupBoxSettings[index].details_top_offset - GROUPBOX_PADDING - 3;
|
|
}
|
|
if (page != TileInspectorPage::Default)
|
|
{
|
|
auto index = EnumValue(page) - 1;
|
|
w->height += PageGroupBoxSettings[index].details_top_offset - GROUPBOX_PADDING - 3;
|
|
w->min_height += PageGroupBoxSettings[index].details_top_offset - GROUPBOX_PADDING - 3;
|
|
}
|
|
w->tileInspectorPage = page;
|
|
auto pageIndex = EnumValue(page);
|
|
w->widgets = PageWidgets[pageIndex];
|
|
w->hold_down_widgets = PageHoldDownWidgets[pageIndex];
|
|
w->disabled_widgets = PageDisabledWidgets[pageIndex];
|
|
w->pressed_widgets = 0;
|
|
}
|
|
|
|
static void WindowTileInspectorScrollmousedown(rct_window* w, int32_t scrollIndex, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
// There is nothing to interact with when no tile is selected
|
|
if (!windowTileInspectorTileSelected)
|
|
return;
|
|
|
|
// Because the list items are displayed in reverse order, subtract the calculated index from the amount of elements
|
|
const int16_t index = windowTileInspectorElementCount - (screenCoords.y - 1) / SCROLLABLE_ROW_HEIGHT - 1;
|
|
const ScreenRect checkboxColumnRect{ { 2, 0 }, { 15, screenCoords.y } };
|
|
if (index >= 0 && checkboxColumnRect.Contains(screenCoords))
|
|
{ // Checkbox was clicked
|
|
WindowTileInspectorToggleInvisibility(index);
|
|
}
|
|
else
|
|
{
|
|
WindowTileInspectorSelectElementFromList(w, index);
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorScrollmouseover(rct_window* w, int32_t scrollIndex, const ScreenCoordsXY& screenCoords)
|
|
{
|
|
int16_t index = windowTileInspectorElementCount - (screenCoords.y - 1) / SCROLLABLE_ROW_HEIGHT - 1;
|
|
if (index < 0 || index >= windowTileInspectorElementCount)
|
|
windowTileInspectorHighlightedIndex = -1;
|
|
else
|
|
windowTileInspectorHighlightedIndex = index;
|
|
|
|
widget_invalidate(w, WIDX_LIST);
|
|
}
|
|
|
|
static void WindowTileInspectorInvalidate(rct_window* w)
|
|
{
|
|
// Set the correct page automatically
|
|
TileInspectorPage page = TileInspectorPage::Default;
|
|
if (windowTileInspectorSelectedIndex != -1)
|
|
{
|
|
const auto element = WindowTileInspectorGetSelectedElement(w);
|
|
switch (element->GetType())
|
|
{
|
|
default:
|
|
case TileElementType::Surface:
|
|
page = TileInspectorPage::Surface;
|
|
break;
|
|
case TileElementType::Path:
|
|
page = TileInspectorPage::Path;
|
|
break;
|
|
case TileElementType::Track:
|
|
page = TileInspectorPage::Track;
|
|
break;
|
|
case TileElementType::SmallScenery:
|
|
page = TileInspectorPage::Scenery;
|
|
break;
|
|
case TileElementType::Entrance:
|
|
page = TileInspectorPage::Entrance;
|
|
break;
|
|
case TileElementType::Wall:
|
|
page = TileInspectorPage::Wall;
|
|
break;
|
|
case TileElementType::LargeScenery:
|
|
page = TileInspectorPage::LargeScenery;
|
|
break;
|
|
case TileElementType::Banner:
|
|
page = TileInspectorPage::Banner;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (w->tileInspectorPage != page)
|
|
{
|
|
WindowTileInspectorSetPage(w, page);
|
|
w->Invalidate();
|
|
}
|
|
|
|
// X and Y spinners
|
|
WidgetSetEnabled(
|
|
*w, WIDX_SPINNER_X_INCREASE,
|
|
(windowTileInspectorTileSelected && ((windowTileInspectorToolMap.x / 32) < MAXIMUM_MAP_SIZE_TECHNICAL - 1)));
|
|
WidgetSetEnabled(
|
|
*w, WIDX_SPINNER_X_DECREASE, (windowTileInspectorTileSelected && ((windowTileInspectorToolMap.x / 32) > 0)));
|
|
WidgetSetEnabled(
|
|
*w, WIDX_SPINNER_Y_INCREASE,
|
|
(windowTileInspectorTileSelected && ((windowTileInspectorToolMap.y / 32) < MAXIMUM_MAP_SIZE_TECHNICAL - 1)));
|
|
WidgetSetEnabled(
|
|
*w, WIDX_SPINNER_Y_DECREASE, (windowTileInspectorTileSelected && ((windowTileInspectorToolMap.y / 32) > 0)));
|
|
|
|
// Sort buttons
|
|
WidgetSetEnabled(*w, WIDX_BUTTON_SORT, (windowTileInspectorTileSelected && windowTileInspectorElementCount > 1));
|
|
|
|
// Move Up button
|
|
WidgetSetEnabled(
|
|
*w, WIDX_BUTTON_MOVE_UP,
|
|
(windowTileInspectorSelectedIndex != -1 && windowTileInspectorSelectedIndex < windowTileInspectorElementCount - 1));
|
|
widget_invalidate(w, WIDX_BUTTON_MOVE_UP);
|
|
|
|
// Move Down button
|
|
WidgetSetEnabled(*w, WIDX_BUTTON_MOVE_DOWN, (windowTileInspectorSelectedIndex > 0));
|
|
widget_invalidate(w, WIDX_BUTTON_MOVE_DOWN);
|
|
|
|
// Copy button
|
|
WidgetSetEnabled(*w, WIDX_BUTTON_COPY, windowTileInspectorSelectedIndex >= 0);
|
|
widget_invalidate(w, WIDX_BUTTON_COPY);
|
|
|
|
// Paste button
|
|
WidgetSetEnabled(*w, WIDX_BUTTON_PASTE, windowTileInspectorTileSelected && windowTileInspectorElementCopied);
|
|
widget_invalidate(w, WIDX_BUTTON_PASTE);
|
|
|
|
w->widgets[WIDX_BACKGROUND].bottom = w->height - 1;
|
|
|
|
if (w->tileInspectorPage == TileInspectorPage::Default)
|
|
{
|
|
w->widgets[WIDX_GROUPBOX_DETAILS].type = WindowWidgetType::Empty;
|
|
w->widgets[WIDX_GROUPBOX_PROPERTIES].type = WindowWidgetType::Empty;
|
|
w->widgets[WIDX_LIST].bottom = w->height - PADDING_BOTTOM;
|
|
}
|
|
else
|
|
{
|
|
w->widgets[WIDX_GROUPBOX_DETAILS].type = WindowWidgetType::Groupbox;
|
|
w->widgets[WIDX_GROUPBOX_PROPERTIES].type = WindowWidgetType::Groupbox;
|
|
auto pageIndex = EnumValue(w->tileInspectorPage) - 1;
|
|
w->widgets[WIDX_GROUPBOX_DETAILS].text = PageGroupBoxSettings[pageIndex].string_id;
|
|
w->widgets[WIDX_GROUPBOX_DETAILS].top = w->height - PageGroupBoxSettings[pageIndex].details_top_offset;
|
|
w->widgets[WIDX_GROUPBOX_DETAILS].bottom = w->height - PageGroupBoxSettings[pageIndex].details_bottom_offset;
|
|
w->widgets[WIDX_GROUPBOX_PROPERTIES].top = w->height - PageGroupBoxSettings[pageIndex].properties_top_offset;
|
|
w->widgets[WIDX_GROUPBOX_PROPERTIES].bottom = w->height - PageGroupBoxSettings[pageIndex].properties_bottom_offset;
|
|
w->widgets[WIDX_LIST].bottom = w->widgets[WIDX_GROUPBOX_DETAILS].top - GROUPBOX_PADDING;
|
|
}
|
|
|
|
// The default page doesn't need further invalidation
|
|
if (w->tileInspectorPage == TileInspectorPage::Default)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Using a switch, because I don't think giving each page their own callbacks is
|
|
// needed here, as only the mouseup and invalidate functions are different.
|
|
const int32_t propertiesAnchor = w->widgets[WIDX_GROUPBOX_PROPERTIES].top;
|
|
const TileElement* const tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
if (tileElement == nullptr)
|
|
return;
|
|
|
|
switch (tileElement->GetType())
|
|
{
|
|
case TileElementType::Surface:
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_SURFACE_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_SURFACE_BUTTON_REMOVE_FENCES].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_SURFACE_BUTTON_REMOVE_FENCES].bottom = GBBB(propertiesAnchor, 1);
|
|
w->widgets[WIDX_SURFACE_BUTTON_RESTORE_FENCES].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_SURFACE_BUTTON_RESTORE_FENCES].bottom = GBBB(propertiesAnchor, 1);
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_N].top = GBBT(propertiesAnchor, 2) + 7 * 0;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_N].bottom = w->widgets[WIDX_SURFACE_CHECK_CORNER_N].top + 13;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_E].top = GBBT(propertiesAnchor, 2) + 7 * 1;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_E].bottom = w->widgets[WIDX_SURFACE_CHECK_CORNER_E].top + 13;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_S].top = GBBT(propertiesAnchor, 2) + 7 * 2;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_S].bottom = w->widgets[WIDX_SURFACE_CHECK_CORNER_S].top + 13;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_W].top = GBBT(propertiesAnchor, 2) + 7 * 1;
|
|
w->widgets[WIDX_SURFACE_CHECK_CORNER_W].bottom = w->widgets[WIDX_SURFACE_CHECK_CORNER_W].top + 13;
|
|
w->widgets[WIDX_SURFACE_CHECK_DIAGONAL].top = GBBT(propertiesAnchor, 3) + 7 * 1;
|
|
w->widgets[WIDX_SURFACE_CHECK_DIAGONAL].bottom = w->widgets[WIDX_SURFACE_CHECK_DIAGONAL].top + 13;
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_SURFACE_CHECK_CORNER_N,
|
|
tileElement->AsSurface()->GetSlope() & (1 << ((2 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_SURFACE_CHECK_CORNER_E,
|
|
tileElement->AsSurface()->GetSlope() & (1 << ((3 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_SURFACE_CHECK_CORNER_S,
|
|
tileElement->AsSurface()->GetSlope() & (1 << ((0 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_SURFACE_CHECK_CORNER_W,
|
|
tileElement->AsSurface()->GetSlope() & (1 << ((1 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_SURFACE_CHECK_DIAGONAL, tileElement->AsSurface()->GetSlope() & TILE_ELEMENT_SLOPE_DOUBLE_HEIGHT);
|
|
break;
|
|
case TileElementType::Path:
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_PATH_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_PATH_CHECK_BROKEN].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_PATH_CHECK_BROKEN].bottom = GBBB(propertiesAnchor, 1);
|
|
w->widgets[WIDX_PATH_CHECK_SLOPED].top = GBBT(propertiesAnchor, 2);
|
|
w->widgets[WIDX_PATH_CHECK_SLOPED].bottom = GBBB(propertiesAnchor, 2);
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_N].top = GBBT(propertiesAnchor, 3) + 7 * 0;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_N].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_N].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_NE].top = GBBT(propertiesAnchor, 3) + 7 * 1;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_NE].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_NE].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_E].top = GBBT(propertiesAnchor, 3) + 7 * 2;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_E].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_E].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_SE].top = GBBT(propertiesAnchor, 3) + 7 * 3;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_SE].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_SE].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_S].top = GBBT(propertiesAnchor, 3) + 7 * 4;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_S].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_S].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_SW].top = GBBT(propertiesAnchor, 3) + 7 * 3;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_SW].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_SW].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_W].top = GBBT(propertiesAnchor, 3) + 7 * 2;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_W].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_W].top + 13;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_NW].top = GBBT(propertiesAnchor, 3) + 7 * 1;
|
|
w->widgets[WIDX_PATH_CHECK_EDGE_NW].bottom = w->widgets[WIDX_PATH_CHECK_EDGE_NW].top + 13;
|
|
WidgetSetCheckboxValue(*w, WIDX_PATH_CHECK_SLOPED, tileElement->AsPath()->IsSloped());
|
|
WidgetSetCheckboxValue(*w, WIDX_PATH_CHECK_BROKEN, tileElement->AsPath()->IsBroken());
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_NE, tileElement->AsPath()->GetEdges() & (1 << ((0 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_SE, tileElement->AsPath()->GetEdges() & (1 << ((1 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_SW, tileElement->AsPath()->GetEdges() & (1 << ((2 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_NW, tileElement->AsPath()->GetEdges() & (1 << ((3 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_E, tileElement->AsPath()->GetCorners() & (1 << ((0 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_S, tileElement->AsPath()->GetCorners() & (1 << ((1 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_W, tileElement->AsPath()->GetCorners() & (1 << ((2 - get_current_rotation()) & 3)));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_PATH_CHECK_EDGE_N, tileElement->AsPath()->GetCorners() & (1 << ((3 - get_current_rotation()) & 3)));
|
|
break;
|
|
case TileElementType::Track:
|
|
w->widgets[WIDX_TRACK_CHECK_APPLY_TO_ALL].top = GBBT(propertiesAnchor, 0);
|
|
w->widgets[WIDX_TRACK_CHECK_APPLY_TO_ALL].bottom = GBBB(propertiesAnchor, 0);
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 1) + 3;
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 1) - 3;
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 1) + 4;
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 1) - 4;
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 1) + 4;
|
|
w->widgets[WIDX_TRACK_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 1) - 4;
|
|
w->widgets[WIDX_TRACK_CHECK_CHAIN_LIFT].top = GBBT(propertiesAnchor, 2);
|
|
w->widgets[WIDX_TRACK_CHECK_CHAIN_LIFT].bottom = GBBB(propertiesAnchor, 2);
|
|
w->widgets[WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED].top = GBBT(propertiesAnchor, 3);
|
|
w->widgets[WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED].bottom = GBBB(propertiesAnchor, 3);
|
|
w->widgets[WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE].top = GBBT(propertiesAnchor, 4);
|
|
w->widgets[WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE].bottom = GBBB(propertiesAnchor, 4);
|
|
WidgetSetCheckboxValue(*w, WIDX_TRACK_CHECK_APPLY_TO_ALL, windowTileInspectorApplyToAll);
|
|
WidgetSetCheckboxValue(*w, WIDX_TRACK_CHECK_CHAIN_LIFT, tileElement->AsTrack()->HasChain());
|
|
WidgetSetCheckboxValue(*w, WIDX_TRACK_CHECK_BLOCK_BRAKE_CLOSED, tileElement->AsTrack()->BlockBrakeClosed());
|
|
WidgetSetCheckboxValue(*w, WIDX_TRACK_CHECK_IS_INDESTRUCTIBLE, tileElement->AsTrack()->IsIndestructible());
|
|
break;
|
|
case TileElementType::SmallScenery:
|
|
{
|
|
// Raise / Lower
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_SCENERY_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
|
|
// Quadrant checkboxes
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_N].top = GBBT(propertiesAnchor, 1) - 5 + 7 * 0;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_N].bottom = w->widgets[WIDX_SCENERY_CHECK_QUARTER_N].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_E].top = GBBT(propertiesAnchor, 1) - 5 + 7 * 1;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_E].bottom = w->widgets[WIDX_SCENERY_CHECK_QUARTER_E].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_S].top = GBBT(propertiesAnchor, 1) - 5 + 7 * 2;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_S].bottom = w->widgets[WIDX_SCENERY_CHECK_QUARTER_S].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_W].top = GBBT(propertiesAnchor, 1) - 5 + 7 * 1;
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_W].bottom = w->widgets[WIDX_SCENERY_CHECK_QUARTER_W].top + 13;
|
|
// This gets the relative rotation, by subtracting the camera's rotation, and wrapping it between 0-3 inclusive
|
|
bool N = tileElement->AsSmallScenery()->GetSceneryQuadrant() == ((0 - get_current_rotation()) & 3);
|
|
bool E = tileElement->AsSmallScenery()->GetSceneryQuadrant() == ((1 - get_current_rotation()) & 3);
|
|
bool S = tileElement->AsSmallScenery()->GetSceneryQuadrant() == ((2 - get_current_rotation()) & 3);
|
|
bool W = tileElement->AsSmallScenery()->GetSceneryQuadrant() == ((3 - get_current_rotation()) & 3);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_QUARTER_N, N);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_QUARTER_E, E);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_QUARTER_S, S);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_QUARTER_W, W);
|
|
|
|
// Collision checkboxes
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_N].top = GBBT(propertiesAnchor, 2) + 5 + 7 * 0;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_N].bottom = w->widgets[WIDX_SCENERY_CHECK_COLLISION_N].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_E].top = GBBT(propertiesAnchor, 2) + 5 + 7 * 1;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_E].bottom = w->widgets[WIDX_SCENERY_CHECK_COLLISION_E].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_S].top = GBBT(propertiesAnchor, 2) + 5 + 7 * 2;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_S].bottom = w->widgets[WIDX_SCENERY_CHECK_COLLISION_S].top + 13;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_W].top = GBBT(propertiesAnchor, 2) + 5 + 7 * 1;
|
|
w->widgets[WIDX_SCENERY_CHECK_COLLISION_W].bottom = w->widgets[WIDX_SCENERY_CHECK_COLLISION_W].top + 13;
|
|
auto occupiedQuadrants = tileElement->GetOccupiedQuadrants();
|
|
N = (occupiedQuadrants & (1 << ((2 - get_current_rotation()) & 3))) != 0;
|
|
E = (occupiedQuadrants & (1 << ((3 - get_current_rotation()) & 3))) != 0;
|
|
S = (occupiedQuadrants & (1 << ((0 - get_current_rotation()) & 3))) != 0;
|
|
W = (occupiedQuadrants & (1 << ((1 - get_current_rotation()) & 3))) != 0;
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_COLLISION_N, N);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_COLLISION_E, E);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_COLLISION_S, S);
|
|
WidgetSetCheckboxValue(*w, WIDX_SCENERY_CHECK_COLLISION_W, W);
|
|
break;
|
|
}
|
|
case TileElementType::Entrance:
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_ENTRANCE_BUTTON_MAKE_USABLE].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_ENTRANCE_BUTTON_MAKE_USABLE].bottom = GBBB(propertiesAnchor, 1);
|
|
WidgetSetEnabled(
|
|
*w, WIDX_ENTRANCE_BUTTON_MAKE_USABLE,
|
|
tileElement->AsEntrance()->GetEntranceType() != ENTRANCE_TYPE_PARK_ENTRANCE);
|
|
break;
|
|
case TileElementType::Wall:
|
|
{
|
|
bool canBeSloped = false;
|
|
bool hasAnimation = false;
|
|
const auto wallEntry = tileElement->AsWall()->GetEntry();
|
|
if (wallEntry != nullptr)
|
|
{
|
|
canBeSloped = !(wallEntry->flags & WALL_SCENERY_CANT_BUILD_ON_SLOPE);
|
|
hasAnimation = wallEntry->flags & WALL_SCENERY_IS_DOOR;
|
|
}
|
|
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_WALL_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_WALL_DROPDOWN_SLOPE].top = GBBT(propertiesAnchor, 1) + 3;
|
|
w->widgets[WIDX_WALL_DROPDOWN_SLOPE].bottom = GBBB(propertiesAnchor, 1) - 3;
|
|
w->widgets[WIDX_WALL_DROPDOWN_SLOPE].text = WallSlopeStringIds[tileElement->AsWall()->GetSlope()];
|
|
w->widgets[WIDX_WALL_DROPDOWN_SLOPE_BUTTON].top = GBBT(propertiesAnchor, 1) + 4;
|
|
w->widgets[WIDX_WALL_DROPDOWN_SLOPE_BUTTON].bottom = GBBB(propertiesAnchor, 1) - 4;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME].top = GBBT(propertiesAnchor, 2) + 3;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME].bottom = GBBB(propertiesAnchor, 2) - 3;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE].top = GBBT(propertiesAnchor, 2) + 4;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE].bottom = GBBB(propertiesAnchor, 2) - 4;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE].top = GBBT(propertiesAnchor, 2) + 4;
|
|
w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE].bottom = GBBB(propertiesAnchor, 2) - 4;
|
|
|
|
// Wall slope dropdown
|
|
WidgetSetEnabled(*w, WIDX_WALL_DROPDOWN_SLOPE, canBeSloped);
|
|
widget_invalidate(w, WIDX_WALL_DROPDOWN_SLOPE);
|
|
WidgetSetEnabled(*w, WIDX_WALL_DROPDOWN_SLOPE_BUTTON, canBeSloped);
|
|
widget_invalidate(w, WIDX_WALL_DROPDOWN_SLOPE_BUTTON);
|
|
// Wall animation frame spinner
|
|
WidgetSetEnabled(*w, WIDX_WALL_SPINNER_ANIMATION_FRAME, hasAnimation);
|
|
WidgetSetEnabled(*w, WIDX_WALL_SPINNER_ANIMATION_FRAME_INCREASE, hasAnimation);
|
|
WidgetSetEnabled(*w, WIDX_WALL_SPINNER_ANIMATION_FRAME_DECREASE, hasAnimation);
|
|
break;
|
|
}
|
|
case TileElementType::LargeScenery:
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
break;
|
|
case TileElementType::Banner:
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT].top = GBBT(propertiesAnchor, 0) + 3;
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT].bottom = GBBB(propertiesAnchor, 0) - 3;
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT_INCREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT_INCREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT_DECREASE].top = GBBT(propertiesAnchor, 0) + 4;
|
|
w->widgets[WIDX_BANNER_SPINNER_HEIGHT_DECREASE].bottom = GBBB(propertiesAnchor, 0) - 4;
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_NE].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_NE].bottom = GBBB(propertiesAnchor, 1);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_SE].top = GBBT(propertiesAnchor, 2);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_SE].bottom = GBBB(propertiesAnchor, 2);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_SW].top = GBBT(propertiesAnchor, 2);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_SW].bottom = GBBB(propertiesAnchor, 2);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_NW].top = GBBT(propertiesAnchor, 1);
|
|
w->widgets[WIDX_BANNER_CHECK_BLOCK_NW].bottom = GBBB(propertiesAnchor, 1);
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_BANNER_CHECK_BLOCK_NE,
|
|
!(tileElement->AsBanner()->GetAllowedEdges() & (1 << ((0 - get_current_rotation()) & 3))));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_BANNER_CHECK_BLOCK_SE,
|
|
!(tileElement->AsBanner()->GetAllowedEdges() & (1 << ((1 - get_current_rotation()) & 3))));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_BANNER_CHECK_BLOCK_SW,
|
|
!(tileElement->AsBanner()->GetAllowedEdges() & (1 << ((2 - get_current_rotation()) & 3))));
|
|
WidgetSetCheckboxValue(
|
|
*w, WIDX_BANNER_CHECK_BLOCK_NW,
|
|
!(tileElement->AsBanner()->GetAllowedEdges() & (1 << ((3 - get_current_rotation()) & 3))));
|
|
break;
|
|
default:
|
|
break; // Nothing.
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorPaint(rct_window* w, rct_drawpixelinfo* dpi)
|
|
{
|
|
WindowDrawWidgets(*w, dpi);
|
|
|
|
ScreenCoordsXY screenCoords(w->windowPos.x, w->windowPos.y);
|
|
|
|
// Draw coordinates
|
|
gfx_draw_string(dpi, screenCoords + ScreenCoordsXY(5, 24), "X:", { w->colours[1] });
|
|
gfx_draw_string(dpi, screenCoords + ScreenCoordsXY(74, 24), "Y:", { w->colours[1] });
|
|
if (windowTileInspectorTileSelected)
|
|
{
|
|
auto tileCoords = TileCoordsXY{ windowTileInspectorToolMap };
|
|
auto ft = Formatter();
|
|
ft.Add<int32_t>(tileCoords.x);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 43, 24 }, STR_FORMAT_INTEGER, ft, { w->colours[1], TextAlignment::RIGHT });
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileCoords.y);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 113, 24 }, STR_FORMAT_INTEGER, ft, { w->colours[1], TextAlignment::RIGHT });
|
|
}
|
|
else
|
|
{
|
|
gfx_draw_string(dpi, screenCoords + ScreenCoordsXY(43 - 7, 24), "-", { w->colours[1] });
|
|
gfx_draw_string(dpi, screenCoords + ScreenCoordsXY(113 - 7, 24), "-", { w->colours[1] });
|
|
}
|
|
|
|
if (windowTileInspectorSelectedIndex != -1)
|
|
{
|
|
// X and Y of first element in detail box
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_GROUPBOX_DETAILS].top + 14 };
|
|
|
|
// Get map element
|
|
TileElement* const tileElement = WindowTileInspectorGetSelectedElement(w);
|
|
if (tileElement == nullptr)
|
|
return;
|
|
|
|
switch (tileElement->GetType())
|
|
{
|
|
case TileElementType::Surface:
|
|
{
|
|
// Details
|
|
// Terrain texture name
|
|
rct_string_id terrainNameId = STR_EMPTY;
|
|
auto surfaceStyle = tileElement->AsSurface()->GetSurfaceStyleObject();
|
|
if (surfaceStyle != nullptr)
|
|
{
|
|
terrainNameId = surfaceStyle->NameStringId;
|
|
}
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(terrainNameId);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_SURFACE_TERAIN, ft, { w->colours[1] });
|
|
|
|
// Edge texture name
|
|
rct_string_id terrainEdgeNameId = STR_EMPTY;
|
|
auto edgeStyle = tileElement->AsSurface()->GetEdgeStyleObject();
|
|
if (edgeStyle != nullptr)
|
|
{
|
|
terrainEdgeNameId = edgeStyle->NameStringId;
|
|
}
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(terrainEdgeNameId);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_SURFACE_EDGE, ft, { w->colours[1] });
|
|
|
|
// Land ownership
|
|
rct_string_id landOwnership;
|
|
if (tileElement->AsSurface()->GetOwnership() & OWNERSHIP_OWNED)
|
|
landOwnership = STR_LAND_OWNED;
|
|
else if (tileElement->AsSurface()->GetOwnership() & OWNERSHIP_AVAILABLE)
|
|
landOwnership = STR_LAND_SALE;
|
|
else if (tileElement->AsSurface()->GetOwnership() & OWNERSHIP_CONSTRUCTION_RIGHTS_OWNED)
|
|
landOwnership = STR_CONSTRUCTION_RIGHTS_OWNED;
|
|
else if (tileElement->AsSurface()->GetOwnership() & OWNERSHIP_CONSTRUCTION_RIGHTS_AVAILABLE)
|
|
landOwnership = STR_CONSTRUCTION_RIGHTS_SALE;
|
|
else
|
|
landOwnership = STR_TILE_INSPECTOR_LAND_NOT_OWNED_AND_NOT_AVAILABLE;
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(landOwnership);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_SURFACE_OWNERSHIP, ft, { w->colours[1] });
|
|
|
|
// Water level
|
|
ft = Formatter();
|
|
ft.Add<uint32_t>(tileElement->AsSurface()->GetWaterHeight());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 33 }, STR_TILE_INSPECTOR_SURFACE_WATER_LEVEL, ft, { w->colours[1] });
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_SURFACE_SPINNER_HEIGHT].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_SURFACE_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
|
|
// Raised corners
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_SURFACE_CHECK_CORNER_E].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_SURFACE_CORNERS, {}, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::Path:
|
|
{
|
|
// Details
|
|
auto pathEl = tileElement->AsPath();
|
|
auto footpathObj = pathEl->GetLegacyPathEntry();
|
|
if (footpathObj == nullptr)
|
|
{
|
|
// Surface name
|
|
auto surfaceObj = pathEl->GetSurfaceEntry();
|
|
if (surfaceObj != nullptr)
|
|
{
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(surfaceObj->NameStringId);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_FOOTPATH_SURFACE_NAME, ft, { COLOUR_WHITE });
|
|
}
|
|
|
|
// Railings name
|
|
auto railingsObj = pathEl->GetRailingsEntry();
|
|
if (railingsObj != nullptr)
|
|
{
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(railingsObj->NameStringId);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_FOOTPATH_RAILINGS_NAME, ft,
|
|
{ COLOUR_WHITE });
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Legacy path name
|
|
auto footpathEntry = reinterpret_cast<const rct_footpath_entry*>(footpathObj->GetLegacyData());
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(footpathEntry->string_idx);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_PATH_NAME, ft, { COLOUR_WHITE });
|
|
}
|
|
|
|
// Path addition
|
|
if (tileElement->AsPath()->HasAddition())
|
|
{
|
|
const auto pathAdditionType = tileElement->AsPath()->GetAdditionEntryIndex();
|
|
const auto* pathBitEntry = get_footpath_item_entry(pathAdditionType);
|
|
rct_string_id additionNameId = pathBitEntry != nullptr
|
|
? pathBitEntry->name
|
|
: static_cast<rct_string_id>(STR_UNKNOWN_OBJECT_TYPE);
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(additionNameId);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 2 * 11 }, STR_TILE_INSPECTOR_PATH_ADDITIONS, ft,
|
|
{ COLOUR_WHITE });
|
|
}
|
|
else
|
|
{
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 2 * 11 }, STR_TILE_INSPECTOR_PATH_ADDITIONS_NONE, {},
|
|
{ COLOUR_WHITE });
|
|
}
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_PATH_SPINNER_HEIGHT].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_PATH_SPINNER_HEIGHT].left + 3;
|
|
auto ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
|
|
// Path connections
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_PATH_CHECK_EDGE_W].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_PATH_CONNECTED_EDGES, {}, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::Track:
|
|
{
|
|
auto trackElement = tileElement->AsTrack();
|
|
RideId rideId = trackElement->GetRideIndex();
|
|
auto ride = get_ride(rideId);
|
|
|
|
// Ride ID
|
|
auto ft = Formatter();
|
|
ft.Add<int16_t>(rideId);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_TRACK_RIDE_ID, ft, { w->colours[1] });
|
|
|
|
// Ride name
|
|
if (ride != nullptr)
|
|
{
|
|
ft = Formatter();
|
|
ride->FormatNameTo(ft);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_TRACK_RIDE_NAME, ft, { w->colours[1] });
|
|
}
|
|
|
|
// Ride type. Individual pieces may be of a different ride type from the ride it belongs to.
|
|
const auto& rtd = GetRideTypeDescriptor(trackElement->GetRideType());
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(rtd.Naming.Name);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_TRACK_RIDE_TYPE, ft, { w->colours[1] });
|
|
|
|
// Track
|
|
ft = Formatter();
|
|
ft.Add<track_type_t>(trackElement->GetTrackType());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 33 }, STR_TILE_INSPECTOR_TRACK_PIECE_ID, ft, { w->colours[1] });
|
|
|
|
ft = Formatter();
|
|
ft.Add<track_type_t>(trackElement->GetSequenceIndex());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 44 }, STR_TILE_INSPECTOR_TRACK_SEQUENCE, ft, { w->colours[1] });
|
|
if (trackElement->IsStation())
|
|
{
|
|
auto stationIndex = trackElement->GetStationIndex();
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_COMMA16);
|
|
ft.Add<int16_t>(stationIndex.ToUnderlying());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 55 }, STR_TILE_INSPECTOR_STATION_INDEX, ft, { w->colours[1] });
|
|
}
|
|
else
|
|
{
|
|
const char* stationNone = "-";
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_STRING);
|
|
ft.Add<char*>(stationNone);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 55 }, STR_TILE_INSPECTOR_STATION_INDEX, ft, { w->colours[1] });
|
|
}
|
|
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(ColourSchemeNames[trackElement->GetColourScheme()]);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 66 }, STR_TILE_INSPECTOR_COLOUR_SCHEME, ft, { w->colours[1] });
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_TRACK_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_TRACK_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::SmallScenery:
|
|
{
|
|
// Details
|
|
// Age
|
|
auto ft = Formatter();
|
|
ft.Add<int16_t>(tileElement->AsSmallScenery()->GetAge());
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_SCENERY_AGE, ft, { w->colours[1] });
|
|
|
|
// Quadrant value
|
|
const auto* sceneryEntry = tileElement->AsSmallScenery()->GetEntry();
|
|
if (sceneryEntry != nullptr && !(sceneryEntry->HasFlag(SMALL_SCENERY_FLAG_FULL_TILE)))
|
|
{
|
|
int16_t quadrant = tileElement->AsSmallScenery()->GetSceneryQuadrant();
|
|
static constexpr rct_string_id _quadrantStringIdx[] = {
|
|
STR_TILE_INSPECTOR_SCENERY_QUADRANT_SW,
|
|
STR_TILE_INSPECTOR_SCENERY_QUADRANT_NW,
|
|
STR_TILE_INSPECTOR_SCENERY_QUADRANT_NE,
|
|
STR_TILE_INSPECTOR_SCENERY_QUADRANT_SE,
|
|
};
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(_quadrantStringIdx[quadrant]);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_SCENERY_QUADRANT, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
|
|
// Scenery ID
|
|
ft = Formatter();
|
|
ft.Add<ObjectEntryIndex>(tileElement->AsSmallScenery()->GetEntryIndex());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_SCENERY_ENTRY_IDX, ft, { w->colours[1] });
|
|
|
|
// Properties
|
|
// Raise / Lower
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_SCENERY_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_SCENERY_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
|
|
// Quarter tile
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7,
|
|
w->widgets[WIDX_SCENERY_CHECK_QUARTER_E].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_SCENERY_QUADRANT_LABEL, {}, { w->colours[1] });
|
|
|
|
// Collision
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_SCENERY_CHECK_COLLISION_E].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_COLLISSION, {}, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::Entrance:
|
|
{
|
|
// Details
|
|
// Entrance type
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(EntranceTypeStringIds[tileElement->AsEntrance()->GetEntranceType()]);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_ENTRANCE_TYPE, ft, { w->colours[1] });
|
|
|
|
if (tileElement->AsEntrance()->GetEntranceType() == ENTRANCE_TYPE_PARK_ENTRANCE)
|
|
{
|
|
// TODO: Make this work with Left/Right park entrance parts
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(park_entrance_get_index({ windowTileInspectorToolMap, tileElement->GetBaseZ() }));
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_ENTRANCE_ENTRANCE_ID, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
else
|
|
{
|
|
ft = Formatter();
|
|
ft.Add<int16_t>(tileElement->AsEntrance()->GetStationIndex().ToUnderlying());
|
|
if (tileElement->AsEntrance()->GetEntranceType() == ENTRANCE_TYPE_RIDE_ENTRANCE)
|
|
{
|
|
// Ride entrance ID
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_ENTRANCE_ENTRANCE_ID, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
else
|
|
{
|
|
// Ride exit ID
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_ENTRANCE_EXIT_ID, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
}
|
|
|
|
if (tileElement->AsEntrance()->GetEntranceType() == ENTRANCE_TYPE_PARK_ENTRANCE)
|
|
{
|
|
// Entrance part
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(ParkEntrancePartStringIds[tileElement->AsEntrance()->GetSequenceIndex()]);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_ENTRANCE_PART, ft, { w->colours[1] });
|
|
}
|
|
else
|
|
{
|
|
// Ride ID
|
|
ft = Formatter();
|
|
ft.Add<RideId>(tileElement->AsEntrance()->GetRideIndex());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_ENTRANCE_RIDE_ID, ft,
|
|
{ w->colours[1] });
|
|
// Station index
|
|
auto stationIndex = tileElement->AsEntrance()->GetStationIndex();
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_COMMA16);
|
|
ft.Add<int16_t>(stationIndex.ToUnderlying());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 33 }, STR_TILE_INSPECTOR_STATION_INDEX, ft, { w->colours[1] });
|
|
}
|
|
|
|
// Properties
|
|
// Raise / Lower
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_ENTRANCE_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::Wall:
|
|
{
|
|
// Details
|
|
// Type
|
|
auto ft = Formatter();
|
|
ft.Add<ObjectEntryIndex>(tileElement->AsWall()->GetEntryIndex());
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_WALL_TYPE, ft, { w->colours[1] });
|
|
|
|
// Banner info
|
|
auto banner = tileElement->AsWall()->GetBanner();
|
|
if (banner != nullptr)
|
|
{
|
|
ft = Formatter();
|
|
banner->FormatTextTo(ft);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_ENTRY_BANNER_TEXT, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
else
|
|
{
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_ENTRY_BANNER_NONE, {},
|
|
{ w->colours[1] });
|
|
}
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_WALL_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_WALL_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
|
|
// Slope label
|
|
screenCoords = w->windowPos
|
|
+ ScreenCoordsXY{ w->widgets[WIDX_GROUPBOX_DETAILS].left + 7, w->widgets[WIDX_WALL_DROPDOWN_SLOPE].top };
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_WALL_SLOPE, {}, { w->colours[1] });
|
|
|
|
// Animation frame label
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_WALL_ANIMATION_FRAME, {}, { w->colours[1] });
|
|
|
|
// Current animation frame
|
|
colour_t colour = w->colours[1];
|
|
if (WidgetIsDisabled(*w, WIDX_WALL_SPINNER_ANIMATION_FRAME))
|
|
{
|
|
colour = w->colours[0] | COLOUR_FLAG_INSET;
|
|
}
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_WALL_SPINNER_ANIMATION_FRAME].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->AsWall()->GetAnimationFrame());
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { colour });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::LargeScenery:
|
|
{
|
|
// Details
|
|
// Type
|
|
auto sceneryElement = tileElement->AsLargeScenery();
|
|
ObjectEntryIndex largeSceneryType = sceneryElement->GetEntryIndex();
|
|
auto ft = Formatter();
|
|
ft.Add<ObjectEntryIndex>(largeSceneryType);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_LARGE_SCENERY_TYPE, ft, { w->colours[1] });
|
|
|
|
// Part ID
|
|
ft = Formatter();
|
|
ft.Add<int16_t>(sceneryElement->GetSequenceIndex());
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 11 }, STR_TILE_INSPECTOR_LARGE_SCENERY_PIECE_ID, ft,
|
|
{ w->colours[1] });
|
|
|
|
// Banner info
|
|
auto* largeSceneryEntry = get_large_scenery_entry(largeSceneryType);
|
|
if (largeSceneryEntry != nullptr && largeSceneryEntry->scrolling_mode != SCROLLING_MODE_NONE)
|
|
{
|
|
auto banner = sceneryElement->GetBanner();
|
|
if (banner != nullptr)
|
|
{
|
|
ft = Formatter();
|
|
banner->FormatTextTo(ft);
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_ENTRY_BANNER_TEXT, ft,
|
|
{ w->colours[1] });
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DrawTextBasic(
|
|
dpi, screenCoords + ScreenCoordsXY{ 0, 22 }, STR_TILE_INSPECTOR_ENTRY_BANNER_NONE, {},
|
|
{ w->colours[1] });
|
|
}
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_LARGE_SCENERY_SPINNER_HEIGHT].left + 3;
|
|
ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
break;
|
|
}
|
|
|
|
case TileElementType::Banner:
|
|
{
|
|
// Details
|
|
// Banner info
|
|
auto banner = tileElement->AsBanner()->GetBanner();
|
|
if (banner != nullptr)
|
|
{
|
|
Formatter ft;
|
|
banner->FormatTextTo(ft);
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_ENTRY_BANNER_TEXT, ft, { w->colours[1] });
|
|
}
|
|
|
|
// Properties
|
|
// Raise / lower label
|
|
screenCoords.y = w->windowPos.y + w->widgets[WIDX_BANNER_SPINNER_HEIGHT].top;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BASE_HEIGHT_FULL, {}, { w->colours[1] });
|
|
|
|
// Current base height
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_BANNER_SPINNER_HEIGHT].left + 3;
|
|
auto ft = Formatter();
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords, STR_FORMAT_INTEGER, ft, { w->colours[1] });
|
|
|
|
// Blocked paths
|
|
screenCoords.y += 28;
|
|
screenCoords.x = w->windowPos.x + w->widgets[WIDX_GROUPBOX_DETAILS].left + 7;
|
|
DrawTextBasic(dpi, screenCoords, STR_TILE_INSPECTOR_BANNER_BLOCKED_PATHS, {}, { w->colours[1] });
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void WindowTileInspectorScrollpaint(rct_window* w, rct_drawpixelinfo* dpi, int32_t scrollIndex)
|
|
{
|
|
const int32_t listWidth = w->widgets[WIDX_LIST].width();
|
|
gfx_fill_rect(
|
|
dpi, { { dpi->x, dpi->y }, { dpi->x + dpi->width - 1, dpi->y + dpi->height - 1 } },
|
|
ColourMapA[w->colours[1]].mid_light);
|
|
|
|
// Show usage hint when nothing is selected
|
|
if (!windowTileInspectorTileSelected)
|
|
{
|
|
auto& listWidget = w->widgets[WIDX_LIST];
|
|
auto centrePos = ScreenCoordsXY{ listWidget.width() / 2,
|
|
(listWidget.height() - font_get_line_height(FontSpriteBase::MEDIUM)) / 2 };
|
|
auto ft = Formatter{};
|
|
auto textPaint = TextPaint{ w->colours[1], TextAlignment::CENTRE };
|
|
DrawTextWrapped(dpi, centrePos, listWidth, STR_TILE_INSPECTOR_SELECT_TILE_HINT, ft, textPaint);
|
|
return;
|
|
}
|
|
|
|
ScreenCoordsXY screenCoords{};
|
|
screenCoords.y = SCROLLABLE_ROW_HEIGHT * (windowTileInspectorElementCount - 1);
|
|
int32_t i = 0;
|
|
char buffer[256];
|
|
|
|
const TileElement* tileElement = map_get_first_element_at(windowTileInspectorToolMap);
|
|
|
|
do
|
|
{
|
|
if (tileElement == nullptr)
|
|
break;
|
|
|
|
const bool selectedRow = i == windowTileInspectorSelectedIndex;
|
|
const bool hoveredRow = i == windowTileInspectorHighlightedIndex;
|
|
const char* typeName = "";
|
|
|
|
// Draw row background colour
|
|
auto fillRectangle = ScreenRect{ { 0, screenCoords.y }, { listWidth, screenCoords.y + SCROLLABLE_ROW_HEIGHT - 1 } };
|
|
if (selectedRow)
|
|
{
|
|
gfx_fill_rect(dpi, fillRectangle, ColourMapA[w->colours[1]].mid_dark);
|
|
}
|
|
else if (hoveredRow)
|
|
{
|
|
gfx_fill_rect(dpi, fillRectangle, ColourMapA[w->colours[1]].mid_dark | 0x1000000);
|
|
}
|
|
else if (((windowTileInspectorElementCount - i) & 1) == 0)
|
|
{
|
|
// Zebra stripes
|
|
gfx_fill_rect(dpi, fillRectangle, ColourMapA[w->colours[1]].light | 0x1000000);
|
|
}
|
|
|
|
const rct_string_id stringFormat = (selectedRow || hoveredRow) ? STR_WHITE_STRING : STR_WINDOW_COLOUR_2_STRINGID;
|
|
auto checkboxFormatter = Formatter();
|
|
checkboxFormatter.Add<rct_string_id>(STR_STRING);
|
|
checkboxFormatter.Add<char*>(CheckBoxMarkString);
|
|
|
|
// Draw checkbox and check if visible
|
|
gfx_fill_rect_inset(dpi, { { 2, screenCoords.y }, { 15, screenCoords.y + 11 } }, w->colours[1], INSET_RECT_F_E0);
|
|
if (!tileElement->IsInvisible())
|
|
{
|
|
auto eyeFormatter = Formatter();
|
|
eyeFormatter.Add<rct_string_id>(STR_STRING);
|
|
eyeFormatter.Add<char*>(EyeString);
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ 2, 1 }, stringFormat, eyeFormatter);
|
|
}
|
|
|
|
const auto type = tileElement->GetType();
|
|
switch (type)
|
|
{
|
|
case TileElementType::Surface:
|
|
typeName = language_get_string(STR_TILE_INSPECTOR_SURFACE);
|
|
break;
|
|
case TileElementType::Path:
|
|
typeName = tileElement->AsPath()->IsQueue() ? language_get_string(STR_QUEUE_LINE_MAP_TIP)
|
|
: language_get_string(STR_FOOTPATH_MAP_TIP);
|
|
break;
|
|
case TileElementType::Track:
|
|
typeName = language_get_string(STR_RIDE_COMPONENT_TRACK_CAPITALISED);
|
|
break;
|
|
case TileElementType::SmallScenery:
|
|
{
|
|
const auto* sceneryEntry = tileElement->AsSmallScenery()->GetEntry();
|
|
snprintf(
|
|
buffer, sizeof(buffer), "%s (%s)", language_get_string(STR_OBJECT_SELECTION_SMALL_SCENERY),
|
|
sceneryEntry != nullptr ? language_get_string(sceneryEntry->name) : "");
|
|
typeName = buffer;
|
|
break;
|
|
}
|
|
case TileElementType::Entrance:
|
|
typeName = language_get_string(STR_RIDE_CONSTRUCTION_ENTRANCE);
|
|
break;
|
|
case TileElementType::Wall:
|
|
{
|
|
const auto* entry = tileElement->AsWall()->GetEntry();
|
|
snprintf(
|
|
buffer, sizeof(buffer), "%s (%s)", language_get_string(STR_TILE_INSPECTOR_WALL),
|
|
entry != nullptr ? language_get_string(entry->name) : "");
|
|
typeName = buffer;
|
|
break;
|
|
}
|
|
case TileElementType::LargeScenery:
|
|
typeName = language_get_string(STR_OBJECT_SELECTION_LARGE_SCENERY);
|
|
break;
|
|
case TileElementType::Banner:
|
|
snprintf(
|
|
buffer, sizeof(buffer), "%s (%u)", language_get_string(STR_BANNER_WINDOW_TITLE),
|
|
tileElement->AsBanner()->GetIndex().ToUnderlying());
|
|
typeName = buffer;
|
|
break;
|
|
default:
|
|
snprintf(buffer, sizeof(buffer), "%s (%d)", language_get_string(STR_UNKNOWN_OBJECT_TYPE), EnumValue(type));
|
|
typeName = buffer;
|
|
}
|
|
|
|
const int32_t clearanceHeight = tileElement->clearance_height;
|
|
const bool ghost = tileElement->IsGhost();
|
|
const bool last = tileElement->IsLastForTile();
|
|
|
|
// Element name
|
|
auto ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_STRING);
|
|
ft.Add<char*>(typeName);
|
|
DrawTextEllipsised(dpi, screenCoords + ScreenCoordsXY{ TypeColumnXY.x, 0 }, TypeColumnSize.width, stringFormat, ft);
|
|
|
|
// Base height
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_FORMAT_INTEGER);
|
|
ft.Add<int32_t>(tileElement->base_height);
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ BaseHeightColumnXY.x, 0 }, stringFormat, ft);
|
|
|
|
// Clearance height
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_FORMAT_INTEGER);
|
|
ft.Add<int32_t>(clearanceHeight);
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ ClearanceHeightColumnXY.x, 0 }, stringFormat, ft);
|
|
|
|
// Direction
|
|
ft = Formatter();
|
|
ft.Add<rct_string_id>(STR_FORMAT_INTEGER);
|
|
ft.Add<int32_t>(tileElement->GetDirection());
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ DirectionColumnXY.x, 0 }, stringFormat, ft);
|
|
|
|
// Checkmarks for ghost and last for tile
|
|
if (ghost)
|
|
{
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ GhostFlagColumnXY.x, 0 }, stringFormat, checkboxFormatter);
|
|
}
|
|
if (last)
|
|
{
|
|
DrawTextBasic(dpi, screenCoords + ScreenCoordsXY{ LastFlagColumnXY.x, 0 }, stringFormat, checkboxFormatter);
|
|
}
|
|
|
|
screenCoords.y -= SCROLLABLE_ROW_HEIGHT;
|
|
i++;
|
|
} while (!(tileElement++)->IsLastForTile());
|
|
}
|