/* $Id$ */ /** @file dropdown.cpp Implementation of the dropdown widget. */ #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 "../core/math_func.hpp" #include "dropdown_type.h" #include "dropdown_func.h" #include "../table/sprites.h" #include "table/strings.h" StringID DropDownListItem::String() const { return STR_NULL; } uint DropDownListItem::Height(uint width) const { return 10; } 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; } void DropDownListItem::Draw(int x, int y, uint width, uint height, bool sel) const { DrawStringTruncated(x + 2, y, this->String(), sel ? TC_WHITE : TC_BLACK, x + width); } /** * Delete all items of a drop down list and the list itself * @param list List to delete. */ static void DeleteDropDownList(DropDownList *list) { for (DropDownList::iterator it = list->begin(); it != list->end(); ++it) { DropDownListItem *item = *it; delete item; } delete list; } struct dropdown_d { WindowClass parent_wnd_class; WindowNumber parent_wnd_num; byte parent_button; DropDownList *list; int selected_index; byte click_delay; bool drag_mode; int scrolling; }; 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; int width = w->widget[0].right - 3; int pos = w->vscroll.pos; const DropDownList *list = WP(w, dropdown_d).list; for (DropDownList::const_iterator it = list->begin(); it != list->end(); ++it) { /* Skip items that are scrolled up */ if (--pos >= 0) continue; const DropDownListItem *item = *it; int item_height = item->Height(width); if (y < item_height) { if (item->masked || item->String() == STR_NULL) return -1; return item->result; } y -= item_height; } return -1; } static void DropDownMenuWndProc(Window *w, WindowEvent *e) { switch (e->event) { case WE_PAINT: { DrawWindowWidgets(w); int x = 1; int y = 2; int sel = WP(w, dropdown_d).selected_index; int width = w->widget[0].right - 3; int height = w->widget[0].bottom; int pos = w->vscroll.pos; DropDownList *list = WP(w, dropdown_d).list; for (DropDownList::const_iterator it = list->begin(); it != list->end(); ++it) { const DropDownListItem *item = *it; int item_height = item->Height(width); /* Skip items that are scrolled up */ if (--pos >= 0) continue; if (y + item_height < height) { if (item->String() != STR_NULL) { if (sel == item->result) GfxFillRect(x + 1, y, x + width, y + item_height - 1, 0); item->Draw(x, y, width, 10, sel == item->result); if (item->masked) { GfxFillRect(x, y, x + width, y + item_height - 1, (1 << PALETTE_MODIFIER_GREYOUT) | _colour_gradient[w->widget[0].color][5] ); } } else { int c1 = _colour_gradient[w->widget[0].color][3]; int c2 = _colour_gradient[w->widget[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 += item_height; } } 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_TICK: if (WP(w, dropdown_d).scrolling == -1) { w->vscroll.pos = max(0, w->vscroll.pos - 1); SetWindowDirty(w); } else if (WP(w, dropdown_d).scrolling == 1) { w->vscroll.pos = min(w->vscroll.count - w->vscroll.cap, w->vscroll.pos + 1); SetWindowDirty(w); } WP(w, dropdown_d).scrolling = 0; 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->HandleWindowEvent(&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 (_cursor.pos.y <= w->top + 2) { /* Cursor is above the list, set scroll up */ WP(w, dropdown_d).scrolling = -1; return; } else if (_cursor.pos.y >= w->top + w->height - 2) { /* Cursor is below list, set scroll down */ WP(w, dropdown_d).scrolling = 1; return; } 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); } DeleteDropDownList(WP(w, dropdown_d).list); } break; } } void ShowDropDownList(Window *w, DropDownList *list, int selected, int button, uint width) { bool is_dropdown_menu_shown = w->IsWidgetLowered(button); DeleteWindowById(WC_DROPDOWN_MENU, 0); if (is_dropdown_menu_shown) { DeleteDropDownList(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 + 1; /* Total length of list */ int list_height = 0; for (DropDownList::const_iterator it = list->begin(); it != list->end(); ++it) { DropDownListItem *item = *it; list_height += item->Height(width); } /* Height of window visible */ int height = list_height; /* 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 + 4 >= 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 > screen_top) { top = w->top + wi->top - height - 4; } else { /* ... and lastly if it won't, enable the scroll bar and fit the * list in below the widget */ int avg_height = list_height / list->size(); int rows = (screen_bottom - 4 - top) / avg_height; height = rows * avg_height; scroll = true; } } if (width == 0) width = wi->right - wi->left + 1; Window *dw = AllocateWindow( w->left + wi->left, top, width, height + 4, DropDownMenuWndProc, WC_DROPDOWN_MENU, _dropdown_menu_widgets); dw->widget[0].color = wi->color; dw->widget[0].right = width - 1; dw->widget[0].bottom = height + 3; 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 = dw->widget[0].bottom; dw->widget[0].right -= 12; /* Capacity is the average number of items visible */ dw->vscroll.cap = height * list->size() / list_height; 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 width) { /* Don't create a new list if we're just closing an existing menu */ if (w->IsWidgetLowered(button)) { DeleteWindowById(WC_DROPDOWN_MENU, 0); return; } 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) { DeleteDropDownList(list); return; } ShowDropDownList(w, list, selected, button, width); } /** * Delete the drop-down menu from window \a pw * @param pw Parent window of the drop-down menu window */ 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; } } }