diff --git a/projects/openrct2.vcxproj b/projects/openrct2.vcxproj
index eff1866b1e..baa3d25704 100644
--- a/projects/openrct2.vcxproj
+++ b/projects/openrct2.vcxproj
@@ -102,6 +102,7 @@
+
@@ -128,6 +129,7 @@
+
diff --git a/projects/openrct2.vcxproj.filters b/projects/openrct2.vcxproj.filters
index 397f06c03d..3cf9846b55 100644
--- a/projects/openrct2.vcxproj.filters
+++ b/projects/openrct2.vcxproj.filters
@@ -381,7 +381,6 @@
-
Source\Drawing
@@ -428,6 +427,13 @@
Source\Windows
+
+ Source\Windows
+
+
+ Source\Windows
+
+
Source\Ride
diff --git a/src/addresses.h b/src/addresses.h
index 7166cddab7..fcd74c5c49 100644
--- a/src/addresses.h
+++ b/src/addresses.h
@@ -372,6 +372,10 @@
#define RCT2_ADDRESS_CURRENT_WINDOW_COLOUR_3 0x0141F742
#define RCT2_ADDRESS_CURRENT_WINDOW_COLOUR_4 0x0141F743
+// size: 500 chars
+#define RCT2_ADDRESS_TOOLTIP_TEXT_BUFFER 0x0141FE44
+#define RCT2_ADDRESS_TOOLTIP_TEXT_HEIGHT 0x01420044
+
#define RCT2_ADDRESS_WINDOW_LIST 0x01420078
#define RCT2_ADDRESS_NEW_WINDOW_PTR 0x014234B8
#define RCT2_ADDRESS_VIEWPORT_LIST 0x014234BC
diff --git a/src/drawing/drawing.c b/src/drawing/drawing.c
index 8688f6bf0c..a6ddf2073d 100644
--- a/src/drawing/drawing.c
+++ b/src/drawing/drawing.c
@@ -486,13 +486,13 @@ void redraw_peep_and_rain()
if (RCT2_GLOBAL(0x009ABDF2, uint32) != 0) {
int sprite = RCT2_GLOBAL(RCT2_ADDRESS_PICKEDUP_PEEP_SPRITE, sint32);
if (sprite != -1) {
- sprite = (sprite & 0x7FFFF) * 16;
+ sprite = sprite & 0x7FFFF;
- rct_g1_element *g1_elements = RCT2_ADDRESS(RCT2_ADDRESS_G1_ELEMENTS, rct_g1_element);
- int left = RCT2_GLOBAL(RCT2_ADDRESS_PICKEDUP_PEEP_X, sint32) + g1_elements[sprite].x_offset;
- int top = RCT2_GLOBAL(RCT2_ADDRESS_PICKEDUP_PEEP_Y, sint32) + g1_elements[sprite].y_offset;
- int right = left + g1_elements[sprite].width;
- int bottom = top + g1_elements[sprite].height;
+ rct_g1_element *g1_elements = &RCT2_ADDRESS(RCT2_ADDRESS_G1_ELEMENTS, rct_g1_element)[sprite];
+ int left = RCT2_GLOBAL(RCT2_ADDRESS_PICKEDUP_PEEP_X, sint16) + g1_elements->x_offset;
+ int top = RCT2_GLOBAL(RCT2_ADDRESS_PICKEDUP_PEEP_Y, sint16) + g1_elements->y_offset;
+ int right = left + g1_elements->width;
+ int bottom = top + g1_elements->height;
gfx_set_dirty_blocks(left, top, right, bottom);
}
@@ -511,6 +511,6 @@ void redraw_peep_and_rain()
}
RCT2_GLOBAL(0x009E2C78, uint32) = 1;
}
- RCT2_GLOBAL(RCT2_ADDRESS_NO_RAIN_PIXELS, uint32) = 0;
}
+ RCT2_GLOBAL(RCT2_ADDRESS_NO_RAIN_PIXELS, uint32) = 0;
}
diff --git a/src/drawing/drawing.h b/src/drawing/drawing.h
index 9a206eb6b5..ab22bc8f7e 100644
--- a/src/drawing/drawing.h
+++ b/src/drawing/drawing.h
@@ -100,6 +100,7 @@ void gfx_load_character_widths();
int clip_text(char *buffer, int width);
int gfx_wrap_string(char* buffer, int width, int* num_lines, int* font_height);
int gfx_get_string_width(char *buffer);
+int gfx_get_string_width_new_lined(char* buffer);
void gfx_draw_string(rct_drawpixelinfo *dpi, char *buffer, int colour, int x, int y);
void gfx_draw_string_left(rct_drawpixelinfo *dpi, int format, void *args, int colour, int x, int y);
void gfx_draw_string_left_clipped(rct_drawpixelinfo *dpi, int format, void *args, int colour, int x, int y, int width);
diff --git a/src/drawing/string.c b/src/drawing/string.c
index 2bfbfc8753..8080ae8afd 100644
--- a/src/drawing/string.c
+++ b/src/drawing/string.c
@@ -85,6 +85,79 @@ void gfx_load_character_widths(){
}
}
+
+/* rct2: 0x006C23B1 */
+int gfx_get_string_width_new_lined(char* buffer){
+ // Current font sprites
+ uint16* current_font_sprite_base;
+ // Width of string
+ int width = 0, max_width = 0, no_lines = 1;
+ rct_g1_element g1_element;
+
+ current_font_sprite_base = RCT2_ADDRESS(RCT2_ADDRESS_CURRENT_FONT_SPRITE_BASE, uint16);
+
+ for (uint8* curr_char = (uint8*)buffer; *curr_char != (uint8)0; curr_char++) {
+
+ if (*curr_char >= 0x20) {
+ width += RCT2_ADDRESS(RCT2_ADDRESS_FONT_CHAR_WIDTH, uint8)[*current_font_sprite_base + (*curr_char - 0x20)];
+ continue;
+ }
+ switch (*curr_char) {
+ case FORMAT_MOVE_X:
+ curr_char++;
+ width = *curr_char;
+ break;
+ case FORMAT_ADJUST_PALETTE:
+ case 3:
+ case 4:
+ curr_char++;
+ break;
+ case FORMAT_NEWLINE:
+ case FORMAT_NEWLINE_SMALLER:
+ no_lines++;
+ max_width = max(max_width, width);
+ width = 0;
+ break;
+ case FORMAT_TINYFONT:
+ *current_font_sprite_base = 0x1C0;
+ break;
+ case FORMAT_BIGFONT:
+ *current_font_sprite_base = 0x2A0;
+ break;
+ case FORMAT_MEDIUMFONT:
+ *current_font_sprite_base = 0x0E0;
+ break;
+ case FORMAT_SMALLFONT:
+ *current_font_sprite_base = 0;
+ break;
+ case FORMAT_OUTLINE:
+ case FORMAT_OUTLINE_OFF:
+ case FORMAT_WINDOW_COLOUR_1:
+ case FORMAT_WINDOW_COLOUR_2:
+ case FORMAT_WINDOW_COLOUR_3:
+ case 0x10:
+ continue;
+ case FORMAT_INLINE_SPRITE:
+ g1_element = RCT2_ADDRESS(RCT2_ADDRESS_G1_ELEMENTS, rct_g1_element)[*((uint32*)(curr_char + 1)) & 0x7FFFF];
+ width += g1_element.width;
+ curr_char += 4;
+ break;
+ default:
+ if (*curr_char <= 0x16) { //case 0x11? FORMAT_NEW_LINE_X_Y
+ curr_char += 2;
+ continue;
+ }
+ curr_char += 4;//never happens?
+ break;
+ }
+ }
+
+ if (width > max_width)
+ return width;
+ return max_width;
+}
+
+
/**
* Return the width of the string in buffer
*
diff --git a/src/game.c b/src/game.c
index d3374c0b87..8421a95786 100644
--- a/src/game.c
+++ b/src/game.c
@@ -276,6 +276,13 @@ void game_logic_update()
if (RCT2_GLOBAL(0x009DEA66, sint16) == 0)
RCT2_GLOBAL(0x009DEA66, sint16)--;
+ // This is a hack for now to force a complete rerender of the screen to
+ // stop viewports from failing. Remove this when real bug cause has been
+ // found.
+ // ***********
+ gfx_invalidate_screen();
+ // ***********
+
sub_68B089();
scenario_update();
climate_update();
diff --git a/src/input.c b/src/input.c
index 4a97219c64..6ec0f334f8 100644
--- a/src/input.c
+++ b/src/input.c
@@ -34,6 +34,7 @@
#include "windows/dropdown.h"
#include "world/map.h"
#include "world/sprite.h"
+#include "world/scenery.h"
POINT _dragPosition;
@@ -814,38 +815,76 @@ static void game_handle_input_mouse(int x, int y, int state)
if (RCT2_GLOBAL(0x009DE540, sint16) < 500) {
// Right click
{
- int eax, ebx, ecx, edx, esi, edi, ebp;
+ int eax, ebx, ecx, esi, edi, ebp;
+ rct_map_element* map_element;
+ rct_scenery_entry* scenery_entry;
eax = RCT2_GLOBAL(RCT2_ADDRESS_CURSOR_DRAG_LAST_X, sint16);
ebx = RCT2_GLOBAL(RCT2_ADDRESS_CURSOR_DRAG_LAST_Y, sint16);
- RCT2_CALLFUNC_X(0x006EDE88, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp);
+ RCT2_CALLFUNC_X(0x006EDE88, &eax, &ebx, &ecx, (int*)&map_element, &esi, &edi, &ebp);
switch (ebx & 0xFF) {
case 2:
- if (*((uint8*)edx) == 0)
- RCT2_CALLPROC_X(0x006B4857, eax, 0, ecx, edx, 0, 0, 0);
+ if (map_element->type == 0)
+ RCT2_CALLPROC_X(0x006B4857, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 3:
- RCT2_CALLPROC_X(0x006CC056, eax, 0, ecx, edx, 0, 0, 0);
+ RCT2_CALLPROC_X(0x006CC056, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 5:
- RCT2_CALLPROC_X(0x006E08D2, eax, 0, ecx, edx, 0, 0, 0);
+ RCT2_CALLPROC_X(0x006E08D2, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 6:
- RCT2_CALLPROC_X(0x006A614A, eax, 0, ecx, edx, 0, 0, 0);
+ RCT2_CALLPROC_X(0x006A614A, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 7:
- RCT2_CALLPROC_X(0x006A61AB, eax, 0, ecx, edx, 0, 0, 0);
+ RCT2_CALLPROC_X(0x006A61AB, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 8:
- RCT2_CALLPROC_X(0x00666C0E, eax, 0, ecx, edx, 0, 0, 0);
+ RCT2_CALLPROC_X(0x00666C0E, eax, 0, ecx, (int)map_element, 0, 0, 0);
break;
case 9:
- RCT2_CALLPROC_X(0x006E57A9, eax, 0, ecx, edx, 0, 0, 0);
+ //0x006e57a9
+ scenery_entry = g_wallSceneryEntries[map_element->properties.fence.slope];
+ if (scenery_entry->wall.var_0D != 0xFF){
+ window_sign_small_open(map_element->properties.fence.item[0]);
+ }
+ else{
+ RCT2_GLOBAL(RCT2_ADDRESS_GAME_COMMAND_ERROR_STRING_ID, rct_string_id) = 1158;
+ game_do_command(
+ eax,
+ 1,
+ ecx,
+ (map_element->type & 0x3) | (map_element->base_height << 8),
+ GAME_COMMAND_42,
+ 0,
+ 0);
+ }
break;
case 10:
- RCT2_CALLPROC_X(0x006B88DC, eax, 0, ecx, edx, 0, 0, 0);
+ //0x006B88DC
+ ebx = map_element->properties.scenerymultiple.type;
+ ebx |= (map_element->properties.scenerymultiple.index & 0x3) << 8;
+ scenery_entry = g_largeSceneryEntries[ebx];
+
+ if (scenery_entry->large_scenery.var_11 != 0xFF){
+ int id = (map_element->type & 0xC0) |
+ ((map_element->properties.scenerymultiple.colour[0] & 0xE0) >> 2) |
+ ((map_element->properties.scenerymultiple.colour[1] & 0xE0) >> 5);
+ window_sign_open(id);
+ }
+ else{
+ RCT2_GLOBAL(RCT2_ADDRESS_GAME_COMMAND_ERROR_STRING_ID, rct_string_id) = 1158;
+ game_do_command(
+ eax,
+ 1 | ((map_element->type & 0x3) << 8),
+ ecx,
+ map_element->base_height | ((map_element->properties.scenerymultiple.index >> 2) << 8),
+ GAME_COMMAND_44,
+ 0,
+ 0);
+ }
break;
case 12:
- RCT2_CALLPROC_X(0x006BA233, eax, 0, ecx, edx, 0, 0, 0);
+ window_banner_open(map_element->properties.banner.index);
break;
default:
break;
diff --git a/src/interface/viewport.c b/src/interface/viewport.c
index 8c09b4e45e..6b7e57cb23 100644
--- a/src/interface/viewport.c
+++ b/src/interface/viewport.c
@@ -289,7 +289,6 @@ void viewport_update_position(rct_window *window)
center_2d_coordinates(sprite->unknown.x, sprite->unknown.y, sprite->unknown.z, ¢er_x, ¢er_y, window->viewport);
RCT2_CALLPROC_X(0x6E7DE1, center_x, center_y, 0, 0, (int)window, (int)viewport, 0);
- window_invalidate(window);//Added to force a redraw.
return;
}
diff --git a/src/interface/window.h b/src/interface/window.h
index 8b9de08ee3..2e504e8aa2 100644
--- a/src/interface/window.h
+++ b/src/interface/window.h
@@ -501,11 +501,14 @@ void window_finances_open();
void window_finances_research_open();
void window_new_campaign_open(sint16 campaignType);
rct_window *window_ride_main_open(int rideIndex);
+void window_ride_demolish_prompt_open(int rideIndex);
void window_ride_construct(rct_window *w);
void window_ride_list_open();
void window_track_place_open();
void window_new_ride_open();
void window_banner_open(rct_windownumber number);
+void window_sign_open(rct_windownumber number);
+void window_sign_small_open(rct_windownumber number);
void window_cheats_open();
void window_research_open();
void window_scenery_open();
diff --git a/src/localisation/string_ids.h b/src/localisation/string_ids.h
index cd288cf13e..00029e9b53 100644
--- a/src/localisation/string_ids.h
+++ b/src/localisation/string_ids.h
@@ -144,6 +144,9 @@ enum {
STR_CONSTRUCTION = 990,
STR_DEMOLISH_RIDE_TIP = 992,
+ STR_DEMOLISH_RIDE = 993,
+ STR_DEMOLISH = 994,
+ STR_DEMOLISH_RIDE_ID = 995,
STR_OVERALL_VIEW = 996,
STR_VIEW_SELECTION = 997,
@@ -679,6 +682,8 @@ enum {
STR_MONTH_NOVEMBER = STR_MONTH_JANUARY + 10,
STR_MONTH_DECEMBER = STR_MONTH_JANUARY + 11,
+ STR_CANT_DEMOLISH_RIDE = 2248,
+
STR_RESEARCH_TRANSPORT_RIDES = 2253,
STR_RESEARCH_GENTLE_RIDES = 2254,
STR_RESEARCH_ROLLER_COASTERS = 2255,
@@ -993,6 +998,10 @@ enum {
STR_DEMOLISH_BANNER_TIP = 2988,
STR_SELECT_MAIN_COLOR_TIP = 2989,
STR_SELECT_TEXT_COLOR_TIP = 2990,
+ STR_SIGN = 2991,
+
+ STR_CHANGE_SIGN_TEXT_TIP = 2994,
+ STR_DEMOLISH_SIGN_TIP = 2995,
STR_LICENCE_AGREEMENT_NOTICE_1 = 2969,
STR_LICENCE_AGREEMENT_NOTICE_2 = 2970,
diff --git a/src/rct2.c b/src/rct2.c
index 776fb73b59..54333779c6 100644
--- a/src/rct2.c
+++ b/src/rct2.c
@@ -229,6 +229,7 @@ void check_file_path(int pathId)
{
case PATH_ID_GAMECFG:
case PATH_ID_SCORES:
+ case PATH_ID_TRACKSIDX:
// Do nothing; these will be created later if they do not exist yet
break;
diff --git a/src/windows/demolish_ride_prompt.c b/src/windows/demolish_ride_prompt.c
new file mode 100644
index 0000000000..fc2d9aad4f
--- /dev/null
+++ b/src/windows/demolish_ride_prompt.c
@@ -0,0 +1,154 @@
+/*****************************************************************************
+* Copyright (c) 2014 Ted John, Duncan Frost
+* OpenRCT2, an open source clone of Roller Coaster Tycoon 2.
+*
+* This file is part of OpenRCT2.
+*
+* OpenRCT2 is free software: you can redistribute it and/or modify
+* it under the terms of the GNU General Public License as published by
+* the Free Software Foundation, either version 3 of the License, or
+* (at your option) any later version.
+
+* This program is distributed in the hope that it will be useful,
+* but WITHOUT ANY WARRANTY; without even the implied warranty of
+* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+* GNU General Public License for more details.
+
+* You should have received a copy of the GNU General Public License
+* along with this program. If not, see .
+*****************************************************************************/
+
+#include "../addresses.h"
+#include "../game.h"
+#include "../interface/widget.h"
+#include "../interface/window.h"
+#include "../localisation/localisation.h"
+#include "../peep/peep.h"
+#include "../peep/staff.h"
+#include "../sprites.h"
+#include "../world/sprite.h"
+
+#define WW 200
+#define WH 100
+
+enum WINDOW_RIDE_DEMOLISH_WIDGET_IDX {
+ WIDX_BACKGROUND,
+ WIDX_TITLE,
+ WIDX_CLOSE,
+ WIDX_DEMOLISH,
+ WIDX_CANCEL
+};
+
+// 0x009AEBA0
+static rct_widget window_ride_demolish_widgets[] = {
+ { WWT_FRAME, 0, 0, WW - 1, 0, WH - 1, STR_NONE, STR_NONE },
+ { WWT_CAPTION, 0, 1, WW - 2, 1, 14, STR_DEMOLISH_RIDE, STR_WINDOW_TITLE_TIP },
+ { WWT_CLOSEBOX, 0, WW - 13, WW - 3, 2, 13, STR_CLOSE_X, STR_CLOSE_WINDOW_TIP },
+ { WWT_DROPDOWN_BUTTON, 0, 10, 94, WH - 20, WH - 9, STR_DEMOLISH, STR_NONE },
+ { WWT_DROPDOWN_BUTTON, 0, WW - 95, WW - 11, WH - 20, WH - 9, STR_SAVE_PROMPT_CANCEL, STR_NONE },
+ { WIDGETS_END }
+};
+
+static void window_ride_demolish_emptysub(){}
+static void window_ride_demolish_mouseup();
+static void window_ride_demolish_paint();
+
+//0x0098E2E4
+static void* window_ride_demolish_events[] = {
+ window_ride_demolish_emptysub,
+ window_ride_demolish_mouseup,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_emptysub,
+ window_ride_demolish_paint,
+ window_ride_demolish_emptysub
+};
+
+/** Based off of rct2: 0x006B486A */
+void window_ride_demolish_prompt_open(int rideIndex){
+ rct_window *w;
+
+ w = window_bring_to_front_by_number(WC_DEMOLISH_RIDE_PROMPT, rideIndex);
+ if (w != NULL)
+ return;
+
+ // Find center of the screen.
+ int screen_height = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, sint16);
+ int screen_width = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, sint16);
+ int x = screen_width / 2 - WW / 2;
+ int y = screen_height / 2 - WH / 2;
+
+ w = window_create(x, y, WW, WH, (uint32*)window_ride_demolish_events, WC_DEMOLISH_RIDE_PROMPT, 0);
+ w->widgets = window_ride_demolish_widgets;
+ w->enabled_widgets = (1 << WIDX_CLOSE) | (1 << WIDX_CANCEL) | (1 << WIDX_DEMOLISH);
+ window_init_scroll_widgets(w);
+ w->flags |= WF_TRANSPARENT;
+ w->number = rideIndex;
+ w->colours[0] = 154;
+}
+
+
+/**
+*
+* rct2: 0x006B4933
+*/
+static void window_ride_demolish_mouseup(){
+ short widgetIndex;
+ rct_window *w;
+
+ window_widget_get_registers(w, widgetIndex);
+
+ switch (widgetIndex){
+ case WIDX_DEMOLISH:
+ RCT2_GLOBAL(RCT2_ADDRESS_GAME_COMMAND_ERROR_STRING_ID, rct_string_id) = STR_CANT_DEMOLISH_RIDE;
+ game_do_command(0, 1, 0, w->number, GAME_COMMAND_7, 0, 0);
+ break;
+ case WIDX_CANCEL:
+ case WIDX_CLOSE:
+ window_close(w);
+ }
+}
+
+/**
+*
+* rct2: 0x006B48E5
+*/
+static void window_ride_demolish_paint(){
+ rct_window *w;
+ rct_drawpixelinfo *dpi;
+
+ window_paint_get_registers(w, dpi);
+
+ window_draw_widgets(w, dpi);
+
+ rct_ride* ride = GET_RIDE(w->number);
+
+ RCT2_GLOBAL(0x13CE952, uint16) = ride->name;
+ RCT2_GLOBAL(0x13CE954, uint32) = ride->name_arguments;
+
+ int x = w->x + WW / 2;
+ int y = w->y + (WH / 2) - 3;
+
+ gfx_draw_string_centred_wrapped(dpi, (void*)0x13CE952, x, y, WW - 4, STR_DEMOLISH_RIDE_ID, 0);
+}
diff --git a/src/windows/error.c b/src/windows/error.c
index 46b5df077e..1fc74c8cd4 100644
--- a/src/windows/error.c
+++ b/src/windows/error.c
@@ -72,18 +72,6 @@ static void* window_error_events[] = {
static char _window_error_text[512];
static uint16 _window_error_num_lines;
-/**
- *
- * rct2: 0x006C23B1
- */
-int sub_6C23B1(char *text)
-{
- int eax, ebx, ecx, edx, esi, edi, ebp;
- esi = (int)text;
- RCT2_CALLFUNC_X(0x006C23B1, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp);
- return *((sint16*)&ecx);
-}
-
/**
*
* rct2: 0x0066792F
@@ -120,7 +108,7 @@ void window_error_open(rct_string_id title, rct_string_id message)
return;
RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_FONT_SPRITE_BASE, uint16) = 224;
- width = sub_6C23B1(_window_error_text);
+ width = gfx_get_string_width_new_lined(_window_error_text);
width = min(196, width);
RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_FONT_SPRITE_BASE, uint16) = 224;
diff --git a/src/windows/ride.c b/src/windows/ride.c
index c9ec2e7f07..1c7176f650 100644
--- a/src/windows/ride.c
+++ b/src/windows/ride.c
@@ -1437,33 +1437,6 @@ static void window_ride_locate(rct_window *w)
window_scroll_to_location(mainWindow, x, y, z);
}
-/**
- *
- * rct2: 0x006B486A
- */
-static void window_ride_demolish(rct_window *w)
-{
- rct_window *demolishWindow;
- int x, y, screenWidth, screenHeight;
-
- demolishWindow = window_bring_to_front_by_number(WC_DEMOLISH_RIDE_PROMPT, w->number);
- if (demolishWindow != NULL)
- return;
-
- screenWidth = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, sint16);
- screenHeight = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, sint16);
- x = screenWidth / 2 - 100;
- y = max(28, screenHeight / 2 - 50);
-
- demolishWindow = window_create(x, y, 200, 100, (uint32*)0x0098E2E4, WC_DEMOLISH_RIDE_PROMPT, 0);
- demolishWindow->widgets = (rct_widget*)0x009AEBA0;
- demolishWindow->enabled_widgets = 4 | 8 | 16;
- window_init_scroll_widgets(demolishWindow);
- demolishWindow->flags |= WF_TRANSPARENT;
- demolishWindow->number = w->number;
- demolishWindow->colours[0] = 154;
-}
-
/**
*
* rct2: 0x006AF17E
@@ -1501,7 +1474,7 @@ static void window_ride_main_mouseup()
window_ride_locate(w);
break;
case WIDX_DEMOLISH:
- window_ride_demolish(w);
+ window_ride_demolish_prompt_open(w->number);
break;
}
}
diff --git a/src/windows/sign.c b/src/windows/sign.c
new file mode 100644
index 0000000000..060d07f378
--- /dev/null
+++ b/src/windows/sign.c
@@ -0,0 +1,678 @@
+/*****************************************************************************
+* Copyright (c) 2014 Ted John, Dennis Devriendt
+* OpenRCT2, an open source clone of Roller Coaster Tycoon 2.
+*
+* This file is part of OpenRCT2.
+*
+* OpenRCT2 is free software: you can redistribute it and/or modify
+* it under the terms of the GNU General Public License as published by
+* the Free Software Foundation, either version 3 of the License, or
+* (at your option) any later version.
+
+* This program is distributed in the hope that it will be useful,
+* but WITHOUT ANY WARRANTY; without even the implied warranty of
+* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+* GNU General Public License for more details.
+
+* You should have received a copy of the GNU General Public License
+* along with this program. If not, see .
+*****************************************************************************/
+
+#include
+#include "../addresses.h"
+#include "../game.h"
+#include "../config.h"
+#include "../localisation/localisation.h"
+#include "../interface/viewport.h"
+#include "../interface/widget.h"
+#include "../interface/window.h"
+#include "../ride/ride.h"
+#include "../world/map.h"
+#include "../world/banner.h"
+#include "../world/scenery.h"
+#include "error.h"
+#include "dropdown.h"
+#include "../drawing/drawing.h"
+
+#define WW 113
+#define WH 96
+
+enum WINDOW_SIGN_WIDGET_IDX {
+ WIDX_BACKGROUND,
+ WIDX_TITLE,
+ WIDX_CLOSE,
+ WIDX_VIEWPORT,
+ WIDX_SIGN_TEXT,
+ WIDX_SIGN_DEMOLISH,
+ WIDX_MAIN_COLOR,
+ WIDX_TEXT_COLOR
+};
+
+// 0x9AEE00
+rct_widget window_sign_widgets[] = {
+ { WWT_FRAME, 0, 0, WW - 1, 0, WH - 1, 0x0FFFFFFFF, 65535 }, // panel / background
+ { WWT_CAPTION, 0, 1, WW - 2, 1, 14, STR_SIGN, STR_WINDOW_TITLE_TIP }, // title bar
+ { WWT_CLOSEBOX, 0, WW - 13, WW - 3, 2, 13, STR_CLOSE_X, STR_CLOSE_WINDOW_TIP }, // close x button
+ { WWT_VIEWPORT, 1, 3, WW - 26, 17, WH - 20, 0x0FFFFFFFE, 65535 }, // Viewport
+ { WWT_FLATBTN, 1, WW - 25, WW - 2, 19, 42, 5168, STR_CHANGE_SIGN_TEXT_TIP }, // change sign button
+ { WWT_FLATBTN, 1, WW - 25, WW - 2, 67, 90, 5165, STR_DEMOLISH_SIGN_TIP }, // demolish button
+ { WWT_COLORBTN, 1, 5, 16, WH - 16, WH - 5, 0x0FFFFFFFF, STR_SELECT_MAIN_COLOR_TIP },// Main colour
+ { WWT_COLORBTN, 1, 17, 28, WH - 16, WH - 5, 0x0FFFFFFFF, STR_SELECT_TEXT_COLOR_TIP },// Text colour
+ { WIDGETS_END },
+};
+
+static void window_sign_emptysub() { }
+static void window_sign_mouseup();
+static void window_sign_mousedown(int widgetIndex, rct_window*w, rct_widget* widget);
+static void window_sign_dropdown();
+static void window_sign_textinput();
+static void window_sign_invalidate();
+static void window_sign_paint();
+static void window_sign_unknown_14();
+
+// 0x98E44C
+static void* window_sign_events[] = {
+ window_sign_emptysub,
+ window_sign_mouseup,
+ window_sign_emptysub,
+ window_sign_mousedown,
+ window_sign_dropdown,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_textinput,
+ window_sign_unknown_14,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_invalidate,
+ window_sign_paint,
+ window_sign_emptysub
+};
+
+static void window_sign_small_mouseup();
+static void window_sign_small_dropdown();
+static void window_sign_small_invalidate();
+
+// 0x9A410C
+static void* window_sign_small_events[] = {
+ window_sign_emptysub,
+ window_sign_small_mouseup,
+ window_sign_emptysub,
+ window_sign_mousedown,
+ window_sign_small_dropdown,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_textinput,
+ window_sign_unknown_14,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_emptysub,
+ window_sign_small_invalidate,
+ window_sign_paint,
+ window_sign_emptysub
+};
+
+/**
+*
+* rct2: 0x006BA305
+*/
+void window_sign_open(rct_windownumber number)
+{
+ rct_window* w;
+ rct_widget *viewportWidget;
+
+
+ // Check if window is already open
+ w = window_bring_to_front_by_number(WC_BANNER, number);
+ if (w != NULL)
+ return;
+
+ w = window_create_auto_pos(WW, WH, (uint32*)window_sign_events, WC_BANNER, 0);
+ w->widgets = window_sign_widgets;
+ w->enabled_widgets =
+ (1 << WIDX_CLOSE) |
+ (1 << WIDX_SIGN_TEXT) |
+ (1 << WIDX_SIGN_DEMOLISH) |
+ (1 << WIDX_MAIN_COLOR) |
+ (1 << WIDX_TEXT_COLOR);
+
+ w->number = number;
+ window_init_scroll_widgets(w);
+ w->colours[0] = 24;
+ w->colours[1] = 24;
+ w->colours[2] = 24;
+
+ int view_x = gBanners[w->number].x << 5;
+ int view_y = gBanners[w->number].y << 5;
+ int ebp = ((view_y << 8) | view_x) >> 5;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(ebp);
+
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_SCENERY_MULTIPLE){
+ int ebx = map_element->properties.scenerymultiple.type;
+ ebx |= (map_element->properties.scenerymultiple.index & 0x3) << 8;
+ rct_scenery_entry* scenery_entry = g_largeSceneryEntries[ebx];
+ if (scenery_entry->large_scenery.var_11 != 0xFF){
+ int id = (map_element->type & 0xC0) |
+ ((map_element->properties.scenerymultiple.colour[0] & 0xE0) >> 2) |
+ ((map_element->properties.scenerymultiple.colour[1] & 0xE0) >> 5);
+ if (id == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+
+ int view_z = map_element->base_height << 3;
+ w->frame_no = view_z;
+
+ w->list_information_type = map_element->properties.scenerymultiple.colour[0] & 0x1F;
+ w->var_492 = map_element->properties.scenerymultiple.colour[1] & 0x1F;
+ w->var_48C = map_element->properties.scenerymultiple.type;
+
+ view_x += 16;
+ view_y += 16;
+
+ // Create viewport
+ viewportWidget = &window_sign_widgets[WIDX_VIEWPORT];
+ viewport_create(
+ w,
+ w->x + viewportWidget->left + 1,
+ w->y + viewportWidget->top + 1,
+ (viewportWidget->right - viewportWidget->left) - 1,
+ (viewportWidget->bottom - viewportWidget->top) - 1,
+ 0,
+ view_x,
+ view_y,
+ view_z,
+ 0,
+ -1
+ );
+
+ w->viewport->flags = (RCT2_GLOBAL(RCT2_ADDRESS_CONFIG_FLAGS, uint8) & CONFIG_FLAG_ALWAYS_SHOW_GRIDLINES) ? VIEWPORT_FLAG_GRIDLINES : 0;
+ w->flags |= WF_2;
+ window_invalidate(w);
+}
+
+/* rct2: 0x6B9765*/
+static void window_sign_mouseup()
+{
+ short widgetIndex;
+ rct_window *w;
+
+ window_widget_get_registers(w, widgetIndex);
+
+ rct_banner* banner = &gBanners[w->number];
+ int x = banner->x << 5;
+ int y = banner->y << 5;
+
+ rct_string_id string_id;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(((y << 8) | x) >> 5);
+
+ switch (widgetIndex) {
+ case WIDX_CLOSE:
+ window_close(w);
+ break;
+ case WIDX_SIGN_DEMOLISH:
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_SCENERY_MULTIPLE){
+ int ebx = map_element->properties.scenerymultiple.type;
+ ebx |= (map_element->properties.scenerymultiple.index & 0x3) << 8;
+ rct_scenery_entry* scenery_entry = g_largeSceneryEntries[ebx];
+ if (scenery_entry->large_scenery.var_11 != 0xFF){
+ int id = (map_element->type & 0xC0) |
+ ((map_element->properties.scenerymultiple.colour[0] & 0xE0) >> 2) |
+ ((map_element->properties.scenerymultiple.colour[1] & 0xE0) >> 5);
+ if (id == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+ game_do_command(
+ x,
+ 1 | ((map_element->type&0x3) << 8),
+ y,
+ map_element->base_height | ((map_element->properties.scenerymultiple.index >> 2) << 8),
+ GAME_COMMAND_44,
+ 0,
+ 0);
+ break;
+ case WIDX_SIGN_TEXT:
+ if (banner->flags&BANNER_FLAG_2){
+ rct_ride* ride = GET_RIDE(banner->colour);
+ RCT2_GLOBAL(0x13CE962, uint32) = ride->name_arguments;
+ string_id = ride->name;
+ }
+ else
+ {
+ string_id = gBanners[w->number].string_idx;
+ }
+ window_text_input_open(w, WIDX_SIGN_TEXT, 2992, 2993, string_id, 0);
+ break;
+ }
+}
+
+/* rct2: 0x6B9784 & 0x6E6164 */
+static void window_sign_mousedown(int widgetIndex, rct_window*w, rct_widget* widget)
+{
+ rct_banner* banner = &gBanners[w->number];
+
+ switch (widgetIndex) {
+ case WIDX_MAIN_COLOR:
+ window_dropdown_show_colour(w, widget, w->colours[1] | 0x80, (uint8)w->list_information_type);
+ break;
+ case WIDX_TEXT_COLOR:
+ window_dropdown_show_colour(w, widget, w->colours[1] | 0x80, (uint8)w->var_492);
+ break;
+ }
+}
+
+/* rct2: 0x6B979C */
+static void window_sign_dropdown()
+{
+ short widgetIndex, dropdownIndex;
+ rct_window* w;
+
+ window_dropdown_get_registers(w, widgetIndex, dropdownIndex);
+
+ switch (widgetIndex){
+ case WIDX_MAIN_COLOR:
+ if (dropdownIndex == -1) return;
+ w->list_information_type = dropdownIndex;
+ break;
+ case WIDX_TEXT_COLOR:
+ if (dropdownIndex == -1) return;
+ w->var_492 = dropdownIndex;
+ break;
+ default:
+ return;
+ }
+
+ rct_banner* banner = &gBanners[w->number];
+ int x = banner->x << 5;
+ int y = banner->y << 5;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(((y << 8) | x) >> 5);
+
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_SCENERY_MULTIPLE){
+ int ebx = map_element->properties.scenerymultiple.type;
+ ebx |= (map_element->properties.scenerymultiple.index & 0x3) << 8;
+ rct_scenery_entry* scenery_entry = g_largeSceneryEntries[ebx];
+ if (scenery_entry->large_scenery.var_11 != 0xFF){
+ int id = (map_element->type & 0xC0) |
+ ((map_element->properties.scenerymultiple.colour[0] & 0xE0) >> 2) |
+ ((map_element->properties.scenerymultiple.colour[1] & 0xE0) >> 5);
+ if (id == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+
+ int edx = map_element->base_height | ((map_element->properties.scenerymultiple.index >> 2) << 8);
+ int ebp = w->list_information_type | (w->var_492 << 8);
+ int ebx = (map_element->type & 0x3) << 8;
+ RCT2_CALLPROC_X(0x6B9B05, x, ebx, y, edx, 0, w->number, ebp);
+ window_invalidate(w);
+}
+
+/* rct2: 0x6B9791 & 0x6E6171*/
+static void window_sign_textinput()
+{
+ short widgetIndex;
+ rct_window *w;
+ uint8 result;
+ uint8* text;
+
+ window_text_input_get_registers(w, widgetIndex, result, text);
+ rct_banner* banner = &gBanners[w->number];
+ int x = banner->x << 5;
+ int y = banner->y << 5;
+
+ if (widgetIndex == WIDX_SIGN_TEXT && result) {
+
+ if (*text != 0){
+ int string_id = 0, ebx = 0, ecx = 128, edx = 0, ebp = 0, esi = 0;
+ RCT2_CALLFUNC_X(0x6C421D, &string_id, &ebx, &ecx, &edx, &esi, (int*)&text, &ebp);
+ if (string_id){
+ rct_string_id prev_string_id = banner->string_idx;
+ banner->string_idx = string_id;
+ // De-allocate previous string id?
+ RCT2_CALLPROC_X(0x6C42AC, prev_string_id, 0, 0, 0, 0, 0, 0);
+
+ banner->flags &= ~(BANNER_FLAG_2);
+ gfx_invalidate_screen();
+ }
+ else{
+ window_error_open(2984, RCT2_GLOBAL(RCT2_ADDRESS_GAME_COMMAND_ERROR_TEXT, rct_string_id));
+ }
+ }
+ else{
+ int eax = x, ebx = 0, ecx = y, edx = 16, ebp = 0, edi = 0, esi = 0;
+ RCT2_CALLFUNC_X(0x6B7D86, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp);
+ if ((eax & 0xFF) == 0xFF)return;
+ banner->colour = eax & 0xFF;
+ banner->flags |= BANNER_FLAG_2;
+
+ rct_string_id prev_string_id = banner->string_idx;
+ banner->string_idx = 778;
+ RCT2_CALLPROC_X(0x6C42AC, prev_string_id, 0, 0, 0, 0, 0, 0);
+ gfx_invalidate_screen();
+ }
+ }
+}
+
+/* rct2: 0x006B96F5 */
+static void window_sign_invalidate()
+{
+ rct_window* w;
+
+ window_get_register(w);
+
+ rct_widget* main_colour_btn = &window_sign_widgets[WIDX_MAIN_COLOR];
+ rct_widget* text_colour_btn = &window_sign_widgets[WIDX_TEXT_COLOR];
+
+ rct_scenery_entry* scenery_entry = g_largeSceneryEntries[w->var_48C];
+
+ main_colour_btn->type = WWT_EMPTY;
+ text_colour_btn->type = WWT_EMPTY;
+
+ if (scenery_entry->large_scenery.flags&(1 << 0)){
+ main_colour_btn->type = WWT_COLORBTN;
+ }
+ if (scenery_entry->large_scenery.flags&(1 << 1)) {
+ text_colour_btn->type = WWT_COLORBTN;
+ }
+
+ main_colour_btn->image = (w->list_information_type << 19) | 0x600013C3;
+ text_colour_btn->image = (w->var_492 << 19) | 0x600013C3;
+}
+
+/* rct2: 0x006B9754 & 0x006E6134 */
+static void window_sign_paint()
+{
+ rct_window *w;
+ rct_drawpixelinfo *dpi;
+
+ window_paint_get_registers(w, dpi);
+
+ window_draw_widgets(w, dpi);
+
+ // Draw viewport
+ if (w->viewport != NULL) {
+ window_draw_viewport(dpi, w);
+ }
+}
+
+/* rct2: 0x6B9A6C & 0x6E6424 */
+static void window_sign_unknown_14()
+{
+ rct_window* w;
+ window_get_register(w);
+
+ rct_viewport* view = w->viewport;
+ w->viewport = 0;
+
+ view->width = 0;
+ viewport_update_pointers();
+
+ rct_banner* banner = &gBanners[w->number];
+
+ int view_x = (banner->x << 5) + 16;
+ int view_y = (banner->y << 5) + 16;
+ int view_z = w->frame_no;
+
+ // Create viewport
+ rct_widget* viewportWidget = &window_sign_widgets[WIDX_VIEWPORT];
+ viewport_create(
+ w,
+ w->x + viewportWidget->left + 1,
+ w->y + viewportWidget->top + 1,
+ (viewportWidget->right - viewportWidget->left) - 1,
+ (viewportWidget->bottom - viewportWidget->top) - 1,
+ 0,
+ view_x,
+ view_y,
+ view_z,
+ 0,
+ -1
+ );
+
+ w->viewport->flags = (RCT2_GLOBAL(RCT2_ADDRESS_CONFIG_FLAGS, uint8) & CONFIG_FLAG_ALWAYS_SHOW_GRIDLINES) ? VIEWPORT_FLAG_GRIDLINES : 0;
+ window_invalidate(w);
+}
+
+
+/* rct2: 0x6E5F52 */
+void window_sign_small_open(rct_windownumber number){
+ rct_window* w;
+ rct_widget *viewportWidget;
+
+
+ // Check if window is already open
+ w = window_bring_to_front_by_number(WC_BANNER, number);
+ if (w != NULL)
+ return;
+
+ w = window_create_auto_pos(WW, WH, (uint32*)window_sign_small_events, WC_BANNER, 0);
+ w->widgets = window_sign_widgets;
+ w->enabled_widgets =
+ (1 << WIDX_CLOSE) |
+ (1 << WIDX_SIGN_TEXT) |
+ (1 << WIDX_SIGN_DEMOLISH) |
+ (1 << WIDX_MAIN_COLOR) |
+ (1 << WIDX_TEXT_COLOR);
+
+ w->number = number;
+ window_init_scroll_widgets(w);
+ w->colours[0] = 24;
+ w->colours[1] = 24;
+ w->colours[2] = 24;
+
+ int view_x = gBanners[w->number].x << 5;
+ int view_y = gBanners[w->number].y << 5;
+ int ebp = ((view_y << 8) | view_x) >> 5;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(ebp);
+
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_FENCE){
+ rct_scenery_entry* scenery_entry = g_wallSceneryEntries[map_element->properties.fence.slope];
+ if (scenery_entry->wall.var_0D != 0xFF){
+ if (map_element->properties.fence.item[0] == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+
+ int view_z = map_element->base_height << 3;
+ w->frame_no = view_z;
+
+ w->list_information_type = map_element->properties.fence.item[1] & 0x1F;
+ w->var_492 = (map_element->properties.fence.item[1] >> 5) | ((map_element->flags&0x60) >> 2);
+ w->var_48C = map_element->properties.fence.slope;
+
+ view_x += 16;
+ view_y += 16;
+
+ // Create viewport
+ viewportWidget = &window_sign_widgets[WIDX_VIEWPORT];
+ viewport_create(
+ w,
+ w->x + viewportWidget->left + 1,
+ w->y + viewportWidget->top + 1,
+ (viewportWidget->right - viewportWidget->left) - 1,
+ (viewportWidget->bottom - viewportWidget->top) - 1,
+ 0,
+ view_x,
+ view_y,
+ view_z,
+ 0,
+ -1
+ );
+
+ w->viewport->flags = (RCT2_GLOBAL(RCT2_ADDRESS_CONFIG_FLAGS, uint8) & CONFIG_FLAG_ALWAYS_SHOW_GRIDLINES) ? VIEWPORT_FLAG_GRIDLINES : 0;
+ w->flags |= WF_2;
+ window_invalidate(w);
+}
+
+/* rct2: 0x6E6145 */
+static void window_sign_small_mouseup()
+{
+ short widgetIndex;
+ rct_window *w;
+
+ window_widget_get_registers(w, widgetIndex);
+
+ rct_banner* banner = &gBanners[w->number];
+ int x = banner->x << 5;
+ int y = banner->y << 5;
+
+ rct_string_id string_id;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(((y << 8) | x) >> 5);
+
+ switch (widgetIndex) {
+ case WIDX_CLOSE:
+ window_close(w);
+ break;
+ case WIDX_SIGN_DEMOLISH:
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_FENCE){
+ rct_scenery_entry* scenery_entry = g_wallSceneryEntries[map_element->properties.fence.slope];
+ if (scenery_entry->wall.var_0D != 0xFF){
+ if (map_element->properties.fence.item[0] == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+ RCT2_GLOBAL(RCT2_ADDRESS_GAME_COMMAND_ERROR_STRING_ID, rct_string_id) = 1158;
+ game_do_command(
+ x,
+ 1 | ((map_element->type & 0x3) << 8),
+ y,
+ (map_element->base_height << 8) | (map_element->type & 0x3),
+ GAME_COMMAND_42,
+ 0,
+ 0);
+ break;
+ case WIDX_SIGN_TEXT:
+ if (banner->flags&BANNER_FLAG_2){
+ rct_ride* ride = GET_RIDE(banner->colour);
+ RCT2_GLOBAL(0x13CE962, uint32) = ride->name_arguments;
+ string_id = ride->name;
+ }
+ else
+ {
+ string_id = gBanners[w->number].string_idx;
+ }
+ window_text_input_open(w, WIDX_SIGN_TEXT, 2992, 2993, string_id, 0);
+ break;
+ }
+}
+
+/* rct2: 0x6E617C */
+static void window_sign_small_dropdown()
+{
+ short widgetIndex, dropdownIndex;
+ rct_window* w;
+
+ window_dropdown_get_registers(w, widgetIndex, dropdownIndex);
+
+ switch (widgetIndex){
+ case WIDX_MAIN_COLOR:
+ if (dropdownIndex == -1) return;
+ w->list_information_type = dropdownIndex;
+ break;
+ case WIDX_TEXT_COLOR:
+ if (dropdownIndex == -1) return;
+ w->var_492 = dropdownIndex;
+ break;
+ default:
+ return;
+ }
+
+ rct_banner* banner = &gBanners[w->number];
+ int x = banner->x << 5;
+ int y = banner->y << 5;
+
+ rct_map_element* map_element = TILE_MAP_ELEMENT_POINTER(((y << 8) | x) >> 5);
+
+ while (1){
+ if ((map_element->type & MAP_ELEMENT_TYPE_MASK) == MAP_ELEMENT_TYPE_FENCE){
+ rct_scenery_entry* scenery_entry = g_wallSceneryEntries[map_element->properties.fence.slope];
+ if (scenery_entry->wall.var_0D != 0xFF){
+ if (map_element->properties.fence.item[0] == w->number)
+ break;
+ }
+ }
+ map_element++;
+ }
+
+ map_element->flags &= 0x9F;
+ map_element->properties.fence.item[1] =
+ w->list_information_type |
+ ((w->var_492 & 0x7) << 5);
+ map_element->flags |= ((w->var_492 & 0x18) << 2);
+
+ RCT2_CALLPROC_X(0x6EC847, x, 0, y, 0, map_element->clearance_height << 3, map_element->base_height << 3, 0);
+ window_invalidate(w);
+}
+
+/* rct2: 0x006E60D5 */
+static void window_sign_small_invalidate()
+{
+ rct_window* w;
+
+ window_get_register(w);
+
+ rct_widget* main_colour_btn = &window_sign_widgets[WIDX_MAIN_COLOR];
+ rct_widget* text_colour_btn = &window_sign_widgets[WIDX_TEXT_COLOR];
+
+ rct_scenery_entry* scenery_entry = g_wallSceneryEntries[w->var_48C];
+
+ main_colour_btn->type = WWT_EMPTY;
+ text_colour_btn->type = WWT_EMPTY;
+
+ if (scenery_entry->wall.flags&(1 << 0)){
+ main_colour_btn->type = WWT_COLORBTN;
+ }
+ if (scenery_entry->wall.flags&(1 << 6)) {
+ text_colour_btn->type = WWT_COLORBTN;
+ }
+
+ main_colour_btn->image = (w->list_information_type << 19) | 0x600013C3;
+ text_colour_btn->image = (w->var_492 << 19) | 0x600013C3;
+}
\ No newline at end of file
diff --git a/src/windows/tooltip.c b/src/windows/tooltip.c
index 695a39b2de..2031f6b2e5 100644
--- a/src/windows/tooltip.c
+++ b/src/windows/tooltip.c
@@ -73,14 +73,15 @@ static void* window_tooltip_events[] = {
void window_tooltip_reset(int x, int y)
{
- RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_CURSOR_X, uint8) = x;
- RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_CURSOR_Y, uint8) = y;
- RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_TIMEOUT, uint8) = 0;
+ RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_CURSOR_X, uint16) = x;
+ RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_CURSOR_Y, uint16) = y;
+ RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_TIMEOUT, uint16) = 0;
RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_WINDOW_CLASS, uint8) = 255;
RCT2_GLOBAL(RCT2_ADDRESS_INPUT_STATE, uint8) = 1;
RCT2_GLOBAL(RCT2_ADDRESS_INPUT_FLAGS, uint32) &= ~(1 << 4);
}
+extern uint8* gTooltip_text_buffer = RCT2_ADDRESS(RCT2_ADDRESS_TOOLTIP_TEXT_BUFFER, uint8);
/**
*
* rct2: 0x006EA10D
@@ -120,9 +121,9 @@ void window_tooltip_open(rct_window *widgetWindow, int widgetIndex, int x, int y
format_string(buffer, widget->tooltip, (void*)0x013CE952);
RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_FONT_SPRITE_BASE, uint16) = 224;
- int tooltip_text_width = ecx, tooltip_text_height = 0;
- //gfx_get_string_width_new_lined
- RCT2_CALLFUNC_X(0x006C23B1, &eax, &ebx, &tooltip_text_width, &edx, (int*)(&buffer), &edi, &ebp);
+ int tooltip_text_width = 0, tooltip_text_height = 0;
+
+ tooltip_text_width = gfx_get_string_width_new_lined(buffer);
buffer = RCT2_ADDRESS(RCT2_ADDRESS_COMMON_STRING_FORMAT_BUFFER, char);
tooltip_text_width &= 0xFFFF;
if (tooltip_text_width > 196)
@@ -131,13 +132,13 @@ void window_tooltip_open(rct_window *widgetWindow, int widgetIndex, int x, int y
RCT2_GLOBAL(RCT2_ADDRESS_CURRENT_FONT_SPRITE_BASE, uint16) = 224;
tooltip_text_width = gfx_wrap_string(buffer, tooltip_text_width + 1, &tooltip_text_height, &ebx);
- RCT2_GLOBAL(0x01420044, sint16) = tooltip_text_height;
+ RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_TEXT_HEIGHT, sint16) = tooltip_text_height;
width = tooltip_text_width + 3;
height = ((tooltip_text_height + 1) * 10) + 4;
window_tooltip_widgets[WIDX_BACKGROUND].right = width;
window_tooltip_widgets[WIDX_BACKGROUND].bottom = height;
- memcpy((void*)0x0141FE44, (void*)buffer, 512);
+ memcpy(gTooltip_text_buffer, buffer, 512);
x = clamp(0, x - (width / 2), RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, uint16) - width);
y = clamp(22, y + 26, RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, uint16) - height - 40);
@@ -215,5 +216,5 @@ static void window_tooltip_paint()
// Text
left = w->x + ((w->width + 1) / 2) - 1;
top = w->y + 1;
- draw_string_centred_raw(dpi, left, top, RCT2_GLOBAL(0x01420044, uint16), (char*)0x0141FE44);
+ draw_string_centred_raw(dpi, left, top, RCT2_GLOBAL(RCT2_ADDRESS_TOOLTIP_TEXT_HEIGHT, uint16), gTooltip_text_buffer);
}
\ No newline at end of file
diff --git a/src/world/banner.h b/src/world/banner.h
index 05eed34e52..61b4832cd3 100644
--- a/src/world/banner.h
+++ b/src/world/banner.h
@@ -27,7 +27,7 @@
typedef struct {
uint8 var_00;
- uint8 flags; //bit 0 is no entry
+ uint8 flags; //0x01 bit 0 is no entry
rct_string_id string_idx; //0x02
uint8 colour; //0x04
uint8 text_colour; //0x05
@@ -36,7 +36,8 @@ typedef struct {
} rct_banner;
enum{
- BANNER_FLAG_NO_ENTRY = (1<<0)
+ BANNER_FLAG_NO_ENTRY = (1<<0),
+ BANNER_FLAG_2 = (1<<2)
} BANNER_FLAGS;
extern rct_banner *gBanners;
diff --git a/src/world/map.h b/src/world/map.h
index b562d48786..a9db6a1d50 100644
--- a/src/world/map.h
+++ b/src/world/map.h
@@ -31,42 +31,42 @@ typedef struct {
} rct_map_element_surface_properties;
typedef struct {
- uint8 type;
- uint8 additions;
- uint8 edges;
- uint8 addition_status;
+ uint8 type; //4
+ uint8 additions; //5
+ uint8 edges; //6
+ uint8 addition_status; //7
} rct_map_element_path_properties;
typedef struct {
- uint8 type;
- uint8 sequence;
- uint8 colour;
- uint8 ride_index;
+ uint8 type; //4
+ uint8 sequence; //5
+ uint8 colour; //6
+ uint8 ride_index; //7
} rct_map_element_track_properties;
typedef struct {
- uint8 type;
- uint8 age;
- uint8 colour;
- uint8 unused;
+ uint8 type; //4
+ uint8 age; //5
+ uint8 colour; //6
+ uint8 unused; //7
} rct_map_element_scenery_properties;
typedef struct {
- uint8 type;
- uint8 index;
- uint8 path_type;
- uint8 ride_index;
+ uint8 type; //4
+ uint8 index; //5
+ uint8 path_type; //6
+ uint8 ride_index; //7
} rct_map_element_entrance_properties;
typedef struct {
- uint8 slope;
- uint8 item[3];
+ uint8 slope; //4
+ uint8 item[3]; //5
} rct_map_element_fence_properties;
typedef struct {
- uint8 type;
- uint8 index;
- uint8 colour[2];
+ uint8 type; //4
+ uint8 index; //5
+ uint8 colour[2]; //6
} rct_map_element_scenerymultiple_properties;
typedef struct {
diff --git a/src/world/scenery.h b/src/world/scenery.h
index 44f7f27081..e87c490634 100644
--- a/src/world/scenery.h
+++ b/src/world/scenery.h
@@ -61,6 +61,7 @@ typedef struct {
uint16 price; // 0x08
uint8 pad_0A[6];
uint8 scenery_tab_id; // 0x10
+ uint8 var_11;
} rct_large_scenery_entry;
@@ -71,6 +72,7 @@ typedef struct {
uint8 flags2; // 0x09
uint16 price; // 0x0A
uint8 scenery_tab_id; // 0x0C
+ uint8 var_0D;
} rct_wall_scenery_entry;
typedef enum {