summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorpeter1138 <peter1138@openttd.org>2008-01-14 16:10:58 +0000
committerpeter1138 <peter1138@openttd.org>2008-01-14 16:10:58 +0000
commite4c05f8d789b8a83159313d3ce745745038a7b67 (patch)
tree3a4930e55ee49d4a3da2e6070bc4f6751fd11cda /src
parent7beb63a93bf48b38cbd1e6a258968acf50099a2c (diff)
downloadopenttd-e4c05f8d789b8a83159313d3ce745745038a7b67.tar.xz
(svn r11848) -Codechange: New class-based drop down list functionality. Lists are now dynamically generated, and can include parameters, or be extended however needed.
Diffstat (limited to 'src')
-rw-r--r--src/autoreplace_gui.cpp1
-rw-r--r--src/build_vehicle_gui.cpp1
-rw-r--r--src/genworld_gui.cpp1
-rw-r--r--src/group_gui.cpp10
-rw-r--r--src/network/network_gui.cpp1
-rw-r--r--src/news_gui.cpp1
-rw-r--r--src/player_gui.cpp1
-rw-r--r--src/rail_gui.cpp1
-rw-r--r--src/settings_gui.cpp1
-rw-r--r--src/station_gui.cpp1
-rw-r--r--src/vehicle_gui.cpp1
-rw-r--r--src/widget.cpp230
-rw-r--r--src/widgets/dropdown.cpp287
-rw-r--r--src/widgets/dropdown_func.h12
-rw-r--r--src/widgets/dropdown_type.h65
-rw-r--r--src/window_gui.h15
16 files changed, 377 insertions, 252 deletions
diff --git a/src/autoreplace_gui.cpp b/src/autoreplace_gui.cpp
index f2c368bea..1160c074d 100644
--- a/src/autoreplace_gui.cpp
+++ b/src/autoreplace_gui.cpp
@@ -18,6 +18,7 @@
#include "autoreplace_func.h"
#include "gfx_func.h"
#include "player_func.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/build_vehicle_gui.cpp b/src/build_vehicle_gui.cpp
index b2bf30ca9..0e1d8836f 100644
--- a/src/build_vehicle_gui.cpp
+++ b/src/build_vehicle_gui.cpp
@@ -29,6 +29,7 @@
#include "vehicle_func.h"
#include "settings_type.h"
#include "gfx_func.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/genworld_gui.cpp b/src/genworld_gui.cpp
index d0c1f9f7e..9a0670184 100644
--- a/src/genworld_gui.cpp
+++ b/src/genworld_gui.cpp
@@ -25,6 +25,7 @@
#include "string_func.h"
#include "gfx_func.h"
#include "settings_type.h"
+#include "widgets/dropdown_func.h"
#include "table/strings.h"
#include "table/sprites.h"
diff --git a/src/group_gui.cpp b/src/group_gui.cpp
index 7175e555f..9e97db1be 100644
--- a/src/group_gui.cpp
+++ b/src/group_gui.cpp
@@ -23,6 +23,7 @@
#include "viewport_func.h"
#include "gfx_func.h"
#include "player_func.h"
+#include "widgets/dropdown_func.h"
#include "table/strings.h"
#include "table/sprites.h"
@@ -328,14 +329,7 @@ static void GroupWndProc(Window *w, WindowEvent *e)
/* The drop down menu is out, *but* it may not be used, retract it. */
if (gv->l.list_length == 0 && w->IsWidgetLowered(GRP_WIDGET_MANAGE_VEHICLES_DROPDOWN)) {
w->RaiseWidget(GRP_WIDGET_MANAGE_VEHICLES_DROPDOWN);
- Window **w2;
- FOR_ALL_WINDOWS(w2) {
- if (w->window_class == WP(*w2, dropdown_d).parent_wnd_class &&
- w->window_number == WP(*w2, dropdown_d).parent_wnd_num) {
- DeleteWindow(*w2);
- break;
- }
- }
+ HideDropDownMenu(w);
}
/* Disable all lists management button when the list is empty */
diff --git a/src/network/network_gui.cpp b/src/network/network_gui.cpp
index 1f6929670..bab0b905c 100644
--- a/src/network/network_gui.cpp
+++ b/src/network/network_gui.cpp
@@ -26,6 +26,7 @@
#include "../gfx_func.h"
#include "../player_func.h"
#include "../settings_type.h"
+#include "../widgets/dropdown_func.h"
#include "table/strings.h"
#include "../table/sprites.h"
diff --git a/src/news_gui.cpp b/src/news_gui.cpp
index c81484062..65abc7fe7 100644
--- a/src/news_gui.cpp
+++ b/src/news_gui.cpp
@@ -15,6 +15,7 @@
#include "vehicle_base.h"
#include "sound_func.h"
#include "string_func.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/player_gui.cpp b/src/player_gui.cpp
index 8da3b7f25..df158a4f8 100644
--- a/src/player_gui.cpp
+++ b/src/player_gui.cpp
@@ -28,6 +28,7 @@
#include "date_func.h"
#include "string_func.h"
#include "settings_type.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/rail_gui.cpp b/src/rail_gui.cpp
index af4572c6c..207c9e1a2 100644
--- a/src/rail_gui.cpp
+++ b/src/rail_gui.cpp
@@ -27,6 +27,7 @@
#include "sound_func.h"
#include "player_func.h"
#include "settings_type.h"
+#include "widgets/dropdown_func.h"
#include "bridge_map.h"
#include "rail_map.h"
diff --git a/src/settings_gui.cpp b/src/settings_gui.cpp
index ec7c1d4c9..7ced5a6ac 100644
--- a/src/settings_gui.cpp
+++ b/src/settings_gui.cpp
@@ -24,6 +24,7 @@
#include "core/alloc_func.hpp"
#include "string_func.h"
#include "gfx_func.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/station_gui.cpp b/src/station_gui.cpp
index 28b5955ab..dc3ebc397 100644
--- a/src/station_gui.cpp
+++ b/src/station_gui.cpp
@@ -23,6 +23,7 @@
#include "window_func.h"
#include "viewport_func.h"
#include "gfx_func.h"
+#include "widgets/dropdown_func.h"
#include "table/strings.h"
#include "table/sprites.h"
diff --git a/src/vehicle_gui.cpp b/src/vehicle_gui.cpp
index 042582709..e7a458278 100644
--- a/src/vehicle_gui.cpp
+++ b/src/vehicle_gui.cpp
@@ -35,6 +35,7 @@
#include "core/alloc_func.hpp"
#include "string_func.h"
#include "settings_type.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
diff --git a/src/widget.cpp b/src/widget.cpp
index 943da370b..d9d664b72 100644
--- a/src/widget.cpp
+++ b/src/widget.cpp
@@ -9,6 +9,7 @@
#include "gfx_func.h"
#include "window_gui.h"
#include "window_func.h"
+#include "widgets/dropdown_func.h"
#include "table/sprites.h"
#include "table/strings.h"
@@ -484,235 +485,6 @@ draw_default:;
}
-static const Widget _dropdown_menu_widgets[] = {
-{ WWT_PANEL, RESIZE_NONE, 0, 0, 0, 0, 0, 0x0, STR_NULL},
-{ WWT_SCROLLBAR, RESIZE_NONE, 0, 0, 0, 0, 0, 0x0, STR_0190_SCROLL_BAR_SCROLLS_LIST},
-{ WIDGETS_END},
-};
-
-static int GetDropdownItem(const Window *w)
-{
- byte item, counter;
- int y;
-
- if (GetWidgetFromPos(w, _cursor.pos.x - w->left, _cursor.pos.y - w->top) < 0)
- return -1;
-
- y = _cursor.pos.y - w->top - 2 + w->vscroll.pos * 10;
-
- if (y < 0)
- return - 1;
-
- item = y / 10;
- if (item >= WP(w, dropdown_d).num_items || (HasBit(WP(w,dropdown_d).disabled_state, item) && !HasBit(WP(w,dropdown_d).hidden_state, item)) || WP(w,dropdown_d).items[item] == 0)
- return - 1;
-
- /* Skip hidden items -- +1 for each hidden item before the clicked item. */
- for (counter = 0; item >= counter; ++counter)
- if (HasBit(WP(w, dropdown_d).hidden_state, counter)) item++;
-
- return item;
-}
-
-static void DropdownMenuWndProc(Window *w, WindowEvent *e)
-{
- int item;
-
- switch (e->event) {
- case WE_PAINT: {
- int x,y,i,sel;
- int width, height;
-
- DrawWindowWidgets(w);
-
- x = 1;
- y = 2 - w->vscroll.pos * 10;
-
- sel = WP(w, dropdown_d).selected_index;
- width = w->widget[0].right - 3;
- height = w->widget[0].bottom - 3;
-
- for (i = 0; WP(w, dropdown_d).items[i] != INVALID_STRING_ID; i++, sel--) {
- if (HasBit(WP(w, dropdown_d).hidden_state, i)) continue;
-
- if (y >= 0 && y <= height) {
- if (WP(w, dropdown_d).items[i] != STR_NULL) {
- if (sel == 0) GfxFillRect(x + 1, y, x + width, y + 9, 0);
- DrawStringTruncated(x + 2, y, WP(w, dropdown_d).items[i], sel == 0 ? TC_WHITE : TC_BLACK, x + width);
-
- if (HasBit(WP(w, dropdown_d).disabled_state, i)) {
- GfxFillRect(x, y, x + width, y + 9,
- (1 << PALETTE_MODIFIER_GREYOUT) | _colour_gradient[_dropdown_menu_widgets[0].color][5]
- );
- }
- } else {
- int c1 = _colour_gradient[_dropdown_menu_widgets[0].color][3];
- int c2 = _colour_gradient[_dropdown_menu_widgets[0].color][7];
-
- GfxFillRect(x + 1, y + 3, x + w->width - 5, y + 3, c1);
- GfxFillRect(x + 1, y + 4, x + w->width - 5, y + 4, c2);
- }
- }
- y += 10;
- }
- } break;
-
- case WE_CLICK: {
- if (e->we.click.widget != 0) break;
- item = GetDropdownItem(w);
- if (item >= 0) {
- WP(w, dropdown_d).click_delay = 4;
- WP(w, dropdown_d).selected_index = item;
- SetWindowDirty(w);
- }
- } break;
-
- case WE_MOUSELOOP: {
- Window *w2 = FindWindowById(WP(w, dropdown_d).parent_wnd_class, WP(w,dropdown_d).parent_wnd_num);
- if (w2 == NULL) {
- DeleteWindow(w);
- return;
- }
-
- if (WP(w, dropdown_d).click_delay != 0 && --WP(w,dropdown_d).click_delay == 0) {
- WindowEvent e;
- e.event = WE_DROPDOWN_SELECT;
- e.we.dropdown.button = WP(w, dropdown_d).parent_button;
- e.we.dropdown.index = WP(w, dropdown_d).selected_index;
- w2->wndproc(w2, &e);
- DeleteWindow(w);
- return;
- }
-
- if (WP(w, dropdown_d).drag_mode) {
- item = GetDropdownItem(w);
-
- if (!_left_button_clicked) {
- WP(w, dropdown_d).drag_mode = false;
- if (item < 0) return;
- WP(w, dropdown_d).click_delay = 2;
- } else {
- if (item < 0) return;
- }
-
- WP(w, dropdown_d).selected_index = item;
- SetWindowDirty(w);
- }
- } break;
-
- case WE_DESTROY: {
- Window *w2 = FindWindowById(WP(w, dropdown_d).parent_wnd_class, WP(w,dropdown_d).parent_wnd_num);
- if (w2 != NULL) {
- w2->RaiseWidget(WP(w, dropdown_d).parent_button);
- w2->InvalidateWidget(WP(w, dropdown_d).parent_button);
- }
- } break;
- }
-}
-
-void ShowDropDownMenu(Window *w, const StringID *strings, int selected, int button, uint32 disabled_mask, uint32 hidden_mask)
-{
- int i;
- const Widget *wi;
- Window *w2;
- const Window *w3;
- bool is_dropdown_menu_shown = w->IsWidgetLowered(button);
- int top, height;
- int screen_top, screen_bottom;
- bool scroll = false;
-
- DeleteWindowById(WC_DROPDOWN_MENU, 0);
-
- if (is_dropdown_menu_shown) return;
-
- w->LowerWidget(button);
-
- w->InvalidateWidget(button);
-
- for (i = 0; strings[i] != INVALID_STRING_ID; i++) {}
- if (i == 0) return;
-
- wi = &w->widget[button];
-
- if (hidden_mask != 0) {
- uint j;
-
- for (j = 0; strings[j] != INVALID_STRING_ID; j++) {
- if (HasBit(hidden_mask, j)) i--;
- }
- }
-
- /* The preferred position is just below the dropdown calling widget */
- top = w->top + wi->bottom + 2;
- height = i * 10 + 4;
-
- w3 = FindWindowById(WC_STATUS_BAR, 0);
- screen_bottom = w3 == NULL ? _screen.height : w3->top;
-
- /* Check if the dropdown will fully fit below the widget */
- if (top + height >= screen_bottom) {
- w3 = FindWindowById(WC_MAIN_TOOLBAR, 0);
- screen_top = w3 == NULL ? 0 : w3->top + w3->height;
-
- /* If not, check if it will fit above the widget */
- if (w->top + wi->top - height - 1 > screen_top) {
- top = w->top + wi->top - height - 1;
- } else {
- /* ... and lastly if it won't, enable the scroll bar and fit the
- * list in below the widget */
- int rows = (screen_bottom - 4 - top) / 10;
- height = rows * 10 + 4;
- scroll = true;
- }
- }
-
- w2 = AllocateWindow(
- w->left + wi[-1].left + 1,
- top,
- wi->right - wi[-1].left + 1,
- height,
- DropdownMenuWndProc,
- WC_DROPDOWN_MENU,
- _dropdown_menu_widgets);
-
- w2->widget[0].color = wi->color;
- w2->widget[0].right = wi->right - wi[-1].left;
- w2->widget[0].bottom = height - 1;
-
- w2->SetWidgetHiddenState(1, !scroll);
-
- if (scroll) {
- /* We're scrolling, so enable the scroll bar and shrink the list by
- * the scrollbar's width */
- w2->widget[1].color = wi->color;
- w2->widget[1].right = w2->widget[0].right;
- w2->widget[1].left = w2->widget[1].right - 11;
- w2->widget[1].bottom = height - 1;
- w2->widget[0].right -= 12;
-
- w2->vscroll.cap = (height - 4) / 10;
- w2->vscroll.count = i;
- }
-
- w2->desc_flags = WDF_DEF_WIDGET;
- w2->flags4 &= ~WF_WHITE_BORDER_MASK;
-
- WP(w2, dropdown_d).disabled_state = disabled_mask;
- WP(w2, dropdown_d).hidden_state = hidden_mask;
-
- WP(w2, dropdown_d).parent_wnd_class = w->window_class;
- WP(w2, dropdown_d).parent_wnd_num = w->window_number;
- WP(w2, dropdown_d).parent_button = button;
-
- WP(w2, dropdown_d).num_items = i;
- WP(w2, dropdown_d).selected_index = selected;
- WP(w2, dropdown_d).items = strings;
-
- WP(w2, dropdown_d).click_delay = 0;
- WP(w2, dropdown_d).drag_mode = true;
-}
-
-
static void ResizeWidgets(Window *w, byte a, byte b)
{
int16 offset = w->widget[a].left;
diff --git a/src/widgets/dropdown.cpp b/src/widgets/dropdown.cpp
new file mode 100644
index 000000000..81821b9cb
--- /dev/null
+++ b/src/widgets/dropdown.cpp
@@ -0,0 +1,287 @@
+/* $Id$ */
+
+#include "../stdafx.h"
+#include "../openttd.h"
+#include "../strings_type.h"
+#include "../window_gui.h"
+#include "../strings_func.h"
+#include "../strings_type.h"
+#include "../gfx_func.h"
+#include "../window_func.h"
+#include "dropdown_type.h"
+#include "dropdown_func.h"
+
+#include "../table/sprites.h"
+#include "table/strings.h"
+
+StringID DropDownListItem::String() const
+{
+ return STR_NULL;
+}
+
+StringID DropDownListStringItem::String() const
+{
+ return this->string;
+}
+
+StringID DropDownListParamStringItem::String() const
+{
+ for (uint i = 0; i < lengthof(this->decode_params); i++) SetDParam(i, this->decode_params[i]);
+ return this->string;
+}
+
+struct dropdown_d {
+ WindowClass parent_wnd_class;
+ WindowNumber parent_wnd_num;
+ byte parent_button;
+ DropDownList *list;
+ byte selected_index;
+ byte click_delay;
+ bool drag_mode;
+};
+assert_compile(WINDOW_CUSTOM_SIZE >= sizeof(dropdown_d));
+
+static const Widget _dropdown_menu_widgets[] = {
+{ WWT_PANEL, RESIZE_NONE, 0, 0, 0, 0, 0, 0x0, STR_NULL},
+{ WWT_SCROLLBAR, RESIZE_NONE, 0, 0, 0, 0, 0, 0x0, STR_0190_SCROLL_BAR_SCROLLS_LIST},
+{ WIDGETS_END},
+};
+
+static int GetDropDownItem(const Window *w)
+{
+ if (GetWidgetFromPos(w, _cursor.pos.x - w->left, _cursor.pos.y - w->top) < 0) return -1;
+
+ int y = _cursor.pos.y - w->top - 2 + w->vscroll.pos * 10;
+ if (y < 0) return -1;
+
+ uint selected_row = y / 10;
+ const DropDownList *list = WP(w, dropdown_d).list;
+
+ if (selected_row >= list->size()) return -1;
+
+ for (DropDownList::const_iterator it = list->begin(); it != list->end(); ++it, selected_row--) {
+ if (selected_row == 0) {
+ const DropDownListItem *item = *it;
+ if (item->masked || item->String() == STR_NULL) return -1;
+ return item->result;
+ }
+ }
+
+ return -1;
+}
+
+static void DropDownMenuWndProc(Window *w, WindowEvent *e)
+{
+ switch (e->event) {
+ case WE_PAINT: {
+ DrawWindowWidgets(w);
+
+ int x = 1;
+ int y = 2 - w->vscroll.pos * 10;
+
+ int sel = WP(w, dropdown_d).selected_index;
+ int width = w->widget[0].right - 3;
+ int height = w->widget[0].bottom - 3;
+
+ DropDownList *list = WP(w, dropdown_d).list;
+
+ for (DropDownList::const_iterator it = list->begin(); it != list->end(); ++it) {
+ if (y >= 0 && y <= height) {
+ const DropDownListItem *item = *it;
+ if (item->String() != STR_NULL) {
+ if (sel == item->result) GfxFillRect(x + 1, y, x + width, y + 9, 0);
+
+ DrawStringTruncated(x + 2, y, item->String(), sel == item->result ? TC_WHITE : TC_BLACK, x + width);
+
+ if (item->masked) {
+ GfxFillRect(x, y, x + width, y + 9,
+ (1 << PALETTE_MODIFIER_GREYOUT) | _colour_gradient[_dropdown_menu_widgets[0].color][5]
+ );
+ }
+ } else {
+ int c1 = _colour_gradient[_dropdown_menu_widgets[0].color][3];
+ int c2 = _colour_gradient[_dropdown_menu_widgets[0].color][7];
+
+ GfxFillRect(x + 1, y + 3, x + w->width - 5, y + 3, c1);
+ GfxFillRect(x + 1, y + 4, x + w->width - 5, y + 4, c2);
+ }
+ }
+ y += 10;
+ }
+ } break;
+
+ case WE_CLICK: {
+ if (e->we.click.widget != 0) break;
+ int item = GetDropDownItem(w);
+ if (item >= 0) {
+ WP(w, dropdown_d).click_delay = 4;
+ WP(w, dropdown_d).selected_index = item;
+ SetWindowDirty(w);
+ }
+ } break;
+
+ case WE_MOUSELOOP: {
+ Window *w2 = FindWindowById(WP(w, dropdown_d).parent_wnd_class, WP(w,dropdown_d).parent_wnd_num);
+ if (w2 == NULL) {
+ DeleteWindow(w);
+ return;
+ }
+
+ if (WP(w, dropdown_d).click_delay != 0 && --WP(w,dropdown_d).click_delay == 0) {
+ WindowEvent e;
+ e.event = WE_DROPDOWN_SELECT;
+ e.we.dropdown.button = WP(w, dropdown_d).parent_button;
+ e.we.dropdown.index = WP(w, dropdown_d).selected_index;
+ w2->wndproc(w2, &e);
+ DeleteWindow(w);
+ return;
+ }
+
+ if (WP(w, dropdown_d).drag_mode) {
+ int item = GetDropDownItem(w);
+
+ if (!_left_button_clicked) {
+ WP(w, dropdown_d).drag_mode = false;
+ if (item < 0) return;
+ WP(w, dropdown_d).click_delay = 2;
+ } else {
+ if (item < 0) return;
+ }
+
+ WP(w, dropdown_d).selected_index = item;
+ SetWindowDirty(w);
+ }
+ } break;
+
+ case WE_DESTROY: {
+ Window *w2 = FindWindowById(WP(w, dropdown_d).parent_wnd_class, WP(w,dropdown_d).parent_wnd_num);
+ if (w2 != NULL) {
+ w2->RaiseWidget(WP(w, dropdown_d).parent_button);
+ w2->InvalidateWidget(WP(w, dropdown_d).parent_button);
+ }
+
+ delete WP(w, dropdown_d).list;
+ } break;
+ }
+}
+
+void ShowDropDownList(Window *w, DropDownList *list, int selected, int button)
+{
+ bool is_dropdown_menu_shown = w->IsWidgetLowered(button);
+
+ DeleteWindowById(WC_DROPDOWN_MENU, 0);
+
+ if (is_dropdown_menu_shown) {
+ delete list;
+ return;
+ }
+
+ w->LowerWidget(button);
+ w->InvalidateWidget(button);
+
+ /* Our parent's button widget is used to determine where to place the drop
+ * down list window. */
+ const Widget *wi = &w->widget[button];
+
+ /* The preferred position is just below the dropdown calling widget */
+ int top = w->top + wi->bottom + 2;
+ int height = list->size() * 10 + 4;
+
+ /* Check if the status bar is visible, as we don't want to draw over it */
+ Window *w3 = FindWindowById(WC_STATUS_BAR, 0);
+ int screen_bottom = w3 == NULL ? _screen.height : w3->top;
+
+ bool scroll = false;
+
+ /* Check if the dropdown will fully fit below the widget */
+ if (top + height >= screen_bottom) {
+ w3 = FindWindowById(WC_MAIN_TOOLBAR, 0);
+ int screen_top = w3 == NULL ? 0 : w3->top + w3->height;
+
+ /* If not, check if it will fit above the widget */
+ if (w->top + wi->top - height - 1 > screen_top) {
+ top = w->top + wi->top - height - 1;
+ } else {
+ /* ... and lastly if it won't, enable the scroll bar and fit the
+ * list in below the widget */
+ int rows = (screen_bottom - 4 - top) / 10;
+ height = rows * 10 + 4;
+ scroll = true;
+ }
+ }
+
+ Window *dw = AllocateWindow(
+ w->left + wi[-1].left + 1,
+ top,
+ wi->right - wi[-1].left + 1,
+ height,
+ DropDownMenuWndProc,
+ WC_DROPDOWN_MENU,
+ _dropdown_menu_widgets);
+
+ dw->widget[0].color = wi->color;
+ dw->widget[0].right = wi->right - wi[-1].left;
+ dw->widget[0].bottom = height - 1;
+
+ dw->SetWidgetHiddenState(1, !scroll);
+
+ if (scroll) {
+ /* We're scrolling, so enable the scroll bar and shrink the list by
+ * the scrollbar's width */
+ dw->widget[1].color = wi->color;
+ dw->widget[1].right = dw->widget[0].right;
+ dw->widget[1].left = dw->widget[1].right - 11;
+ dw->widget[1].bottom = height - 1;
+ dw->widget[0].right -= 12;
+
+ dw->vscroll.cap = (height - 4) / 10;
+ dw->vscroll.count = list->size();
+ }
+
+ dw->desc_flags = WDF_DEF_WIDGET;
+ dw->flags4 &= ~WF_WHITE_BORDER_MASK;
+
+ WP(dw, dropdown_d).parent_wnd_class = w->window_class;
+ WP(dw, dropdown_d).parent_wnd_num = w->window_number;
+ WP(dw, dropdown_d).parent_button = button;
+ WP(dw, dropdown_d).list = list;
+ WP(dw, dropdown_d).selected_index = selected;
+ WP(dw, dropdown_d).click_delay = 0;
+ WP(dw, dropdown_d).drag_mode = true;
+}
+
+void ShowDropDownMenu(Window *w, const StringID *strings, int selected, int button, uint32 disabled_mask, uint32 hidden_mask)
+{
+ uint result = 0;
+ DropDownList *list = new DropDownList();
+
+ for (uint i = 0; strings[i] != INVALID_STRING_ID; i++) {
+ if (!HasBit(hidden_mask, i)) {
+ list->push_back(new DropDownListStringItem(strings[i], result, HasBit(disabled_mask, i)));
+ }
+ result++;
+ }
+
+ /* No entries in the list? */
+ if (list->size() == 0) {
+ delete list;
+ return;
+ }
+
+ ShowDropDownList(w, list, selected, button);
+}
+
+void HideDropDownMenu(Window *pw)
+{
+ Window **wz;
+ FOR_ALL_WINDOWS(wz) {
+ if ((*wz)->window_class != WC_DROPDOWN_MENU) continue;
+
+ if (pw->window_class == WP(*wz, dropdown_d).parent_wnd_class &&
+ pw->window_number == WP(*wz, dropdown_d).parent_wnd_num) {
+ DeleteWindow(*wz);
+ break;
+ }
+ }
+}
+
diff --git a/src/widgets/dropdown_func.h b/src/widgets/dropdown_func.h
new file mode 100644
index 000000000..dd6b430bc
--- /dev/null
+++ b/src/widgets/dropdown_func.h
@@ -0,0 +1,12 @@
+/* $Id$ */
+
+#ifndef WIDGETS_DROPDOWN_FUNC_H
+#define WIDGETS_DROPDOWN_FUNC_H
+
+/* Show drop down menu containing a fixed list of strings */
+void ShowDropDownMenu(Window *w, const StringID *strings, int selected, int button, uint32 disabled_mask, uint32 hidden_mask);
+
+/* Hide drop down menu of a parent window */
+void HideDropDownMenu(Window *pw);
+
+#endif /* WIDGETS_DROPDOWN_FUNC_H */
diff --git a/src/widgets/dropdown_type.h b/src/widgets/dropdown_type.h
new file mode 100644
index 000000000..650edc5b1
--- /dev/null
+++ b/src/widgets/dropdown_type.h
@@ -0,0 +1,65 @@
+/* $Id$ */
+
+#ifndef WIDGETS_DROPDOWN_TYPE_H
+#define WIDGETS_DROPDOWN_TYPE_H
+
+#include "../window_type.h"
+#include <list>
+
+/**
+ * Base list item class from which others are derived. If placed in a list it
+ * will appear as a horizontal line in the menu.
+ */
+class DropDownListItem {
+public:
+ int result; ///< Result code to return to window on selection
+ bool masked; ///< Masked and unselectable item
+
+ virtual StringID String() const;
+};
+
+/**
+ * Common string list item.
+ */
+class DropDownListStringItem : public DropDownListItem {
+public:
+ StringID string; ///< String ID of item
+
+ DropDownListStringItem(StringID string, uint result, bool masked)
+ {
+ this->string = string;
+ this->result = result;
+ this->masked = masked;
+ }
+
+ StringID String() const;
+};
+
+/**
+ * String list item with parameters.
+ */
+class DropDownListParamStringItem : public DropDownListStringItem {
+public:
+ uint64 decode_params[10]; ///< Parameters of the string
+
+ StringID String() const;
+ void SetParam(uint index, uint64 value) { decode_params[index] = value; }
+};
+
+/**
+ * A drop down list is a collection of drop down list items.
+ */
+typedef std::list<DropDownListItem *> DropDownList;
+
+/**
+ * Show a drop down list.
+ * @param w Parent window for the list.
+ * @param list Prepopulated DropDownList. Will be deleted when the list is
+ * closed.
+ * @param selected The initially selected list item.
+ * @param button The widget within the parent window that is used to determine
+ * the list's location.
+ */
+void ShowDropDownList(Window *w, DropDownList *list, int selected, int button);
+
+#endif /* WIDGETS_DROPDOWN_TYPE_H */
diff --git a/src/window_gui.h b/src/window_gui.h
index bb7c23199..df825e6d1 100644
--- a/src/window_gui.h
+++ b/src/window_gui.h
@@ -439,20 +439,6 @@ struct message_d {
};
assert_compile(WINDOW_CUSTOM_SIZE >= sizeof(message_d));
-struct dropdown_d {
- uint32 disabled_state;
- uint32 hidden_state;
- WindowClass parent_wnd_class;
- WindowNumber parent_wnd_num;
- byte parent_button;
- byte num_items;
- byte selected_index;
- const StringID *items;
- byte click_delay;
- bool drag_mode;
-};
-assert_compile(WINDOW_CUSTOM_SIZE >= sizeof(dropdown_d));
-
struct vehiclelist_d {
const Vehicle** sort_list; // List of vehicles (sorted)
Listing *_sorting; // pointer to the appropiate subcategory of _sorting
@@ -592,7 +578,6 @@ static inline void GuiShowTooltips(StringID str)
/* widget.cpp */
int GetWidgetFromPos(const Window *w, int x, int y);
void DrawWindowWidgets(const Window *w);
-void ShowDropDownMenu(Window *w, const StringID *strings, int selected, int button, uint32 disabled_mask, uint32 hidden_mask);
Window *GetCallbackWnd();