/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program 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 2 * 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, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "common/scummsys.h" #include "common/system.h" #include "common/rect.h" #include "common/textconsole.h" #include "common/translation.h" #include "graphics/pixelformat.h" #include "gui/widget.h" #include "gui/gui-manager.h" #include "gui/ThemeEval.h" #include "gui/dialog.h" #include "gui/widgets/popup.h" #include "gui/widgets/scrollcontainer.h" namespace GUI { Widget::Widget(GuiObject *boss, int x, int y, int w, int h, Common::U32String tooltip) : GuiObject(x, y, w, h), _type(0), _boss(boss), _tooltip(tooltip), _flags(0), _hasFocus(false), _state(ThemeEngine::kStateEnabled) { init(); } Widget::Widget(GuiObject *boss, const Common::String &name, Common::U32String tooltip) : GuiObject(name), _type(0), _boss(boss), _tooltip(tooltip), _flags(0), _hasFocus(false), _state(ThemeEngine::kStateDisabled) { init(); } void Widget::init() { // Insert into the widget list of the boss _next = _boss->_firstWidget; _boss->_firstWidget = this; _needsRedraw = true; } Widget::~Widget() { delete _next; _next = nullptr; } void Widget::resize(int x, int y, int w, int h) { _x = x; _y = y; _w = w; _h = h; } void Widget::setFlags(int flags) { updateState(_flags, _flags | flags); _flags |= flags; } void Widget::clearFlags(int flags) { updateState(_flags, _flags & ~flags); _flags &= ~flags; } void Widget::updateState(int oldFlags, int newFlags) { if (newFlags & WIDGET_ENABLED) { _state = ThemeEngine::kStateEnabled; if (newFlags & WIDGET_HILITED) _state = ThemeEngine::kStateHighlight; if (newFlags & WIDGET_PRESSED) _state = ThemeEngine::kStatePressed; } else { _state = ThemeEngine::kStateDisabled; } } void Widget::markAsDirty() { _needsRedraw = true; Widget *w = _firstWidget; while (w) { w->markAsDirty(); w = w->next(); } } void Widget::draw() { if (!isVisible() || !_boss->isVisible()) return; if (_needsRedraw) { int oldX = _x, oldY = _y; // Account for our relative position in the dialog _x = getAbsX(); _y = getAbsY(); Common::Rect oldClip = g_gui.theme()->swapClipRect(_boss->getClipRect()); if (g_gui.useRTL()) { _x = g_system->getOverlayWidth() - _x - _w; if (this->_name.contains("GameOptions") || this->_name.contains("GlobalOptions") || this->_name.contains("Browser") || this->_name.empty()) { /** The dialogs named above are the stacked dialogs for which the left+right paddings need to be adjusted for RTL. The _name is empty for some special widgets - like RemapWidgets, NavBars, ScrollBars and they need to be adjusted too. */ _x = _x + g_gui.getOverlayOffset(); } Common::Rect r = _boss->getClipRect(); r.moveTo(_x, r.top); g_gui.theme()->swapClipRect(r); } // Draw border if (_flags & WIDGET_BORDER) { g_gui.theme()->drawWidgetBackground(Common::Rect(_x, _y, _x + _w, _y + _h), ThemeEngine::kWidgetBackgroundBorder); _x += 4; _y += 4; _w -= 8; _h -= 8; } // Now perform the actual widget draw drawWidget(); g_gui.theme()->swapClipRect(oldClip); // Restore x/y if (_flags & WIDGET_BORDER) { _x -= 4; _y -= 4; _w += 8; _h += 8; } _x = oldX; _y = oldY; _needsRedraw = false; } // Draw all children Widget *w = _firstWidget; while (w) { w->draw(); w = w->_next; } } Widget *Widget::findWidgetInChain(Widget *w, int x, int y) { while (w) { // Stop as soon as we find a widget that contains the point (x,y) if (x >= w->_x && x < w->_x + w->_w && y >= w->_y && y < w->_y + w->getHeight()) break; w = w->_next; } if (w) w = w->findWidget(x - w->_x, y - w->_y); return w; } Widget *Widget::findWidgetInChain(Widget *w, const char *name) { while (w) { if (w->_name == name) { return w; } w = w->_next; } return nullptr; } bool Widget::containsWidgetInChain(Widget *w, Widget *search) { while (w) { if (w == search || w->containsWidget(search)) return true; w = w->_next; } return false; } void Widget::setEnabled(bool e) { if ((_flags & WIDGET_ENABLED) != e) { if (e) setFlags(WIDGET_ENABLED); else clearFlags(WIDGET_ENABLED); g_gui.scheduleTopDialogRedraw(); } } bool Widget::isEnabled() const { return ((_flags & WIDGET_ENABLED) != 0); } void Widget::setVisible(bool e) { if (e) clearFlags(WIDGET_INVISIBLE); else setFlags(WIDGET_INVISIBLE); } bool Widget::isVisible() const { if (g_gui.xmlEval()->getVar("Dialog." + _name + ".Visible", 1) == 0) return false; return !(_flags & WIDGET_INVISIBLE); } bool Widget::useRTL() const { return _useRTL; } uint8 Widget::parseHotkey(const Common::U32String &label) { if (!label.contains('~')) return 0; int state = 0; uint8 hotkey = 0; for (uint i = 0; i < label.size() && state != 3; i++) { switch (state) { case 0: if (label[i] == '~') state = 1; break; case 1: if (label[i] != '~') { state = 2; hotkey = label[i]; } else state = 0; break; case 2: if (label[i] == '~') state = 3; else state = 0; break; default: break; } } if (state == 3) return hotkey; return 0; } Common::U32String Widget::cleanupHotkey(const Common::U32String &label) { Common::U32String res = ""; for (Common::U32String::const_iterator itr = label.begin(); itr != label.end(); itr++) { if (*itr != '~') { res += *itr; } } return res; } void Widget::read(const Common::U32String &str) { #ifdef USE_TTS if (ConfMan.hasKey("tts_enabled", "scummvm") && ConfMan.getBool("tts_enabled", "scummvm")) { Common::TextToSpeechManager *ttsMan = g_system->getTextToSpeechManager(); if (ttsMan == nullptr) return; ttsMan->say(str); } #endif } #pragma mark - StaticTextWidget::StaticTextWidget(GuiObject *boss, int x, int y, int w, int h, const Common::U32String &text, Graphics::TextAlign align, Common::U32String tooltip, ThemeEngine::FontStyle font) : Widget(boss, x, y, w, h, tooltip), _align(align) { setFlags(WIDGET_ENABLED); _type = kStaticTextWidget; _label = text; _font = font; _align = Graphics::convertTextAlignH(align, g_gui.useRTL() && _useRTL); } StaticTextWidget::StaticTextWidget(GuiObject *boss, const Common::String &name, const Common::U32String &text, Common::U32String tooltip, ThemeEngine::FontStyle font) : Widget(boss, name, tooltip) { setFlags(WIDGET_ENABLED | WIDGET_CLEARBG); _type = kStaticTextWidget; _label = text; _align = Graphics::convertTextAlignH(g_gui.xmlEval()->getWidgetTextHAlign(name), g_gui.useRTL() && _useRTL); _font = font; } void StaticTextWidget::setValue(int value) { _label = Common::String::format("%d", value); } void StaticTextWidget::setLabel(const Common::U32String &label) { if (_label != label) { _label = label; markAsDirty(); } } void StaticTextWidget::setAlign(Graphics::TextAlign align) { align = Graphics::convertTextAlignH(align, g_gui.useRTL() && _useRTL); if (_align != align){ _align = align; markAsDirty(); } } void StaticTextWidget::drawWidget() { g_gui.theme()->drawText( Common::Rect(_x, _y, _x + _w, _y + _h), _label, _state, _align, ThemeEngine::kTextInversionNone, 0, true, _font ); } #pragma mark - ButtonWidget::ButtonWidget(GuiObject *boss, int x, int y, int w, int h, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : StaticTextWidget(boss, x, y, w, h, cleanupHotkey(label), Graphics::kTextAlignCenter, tooltip), CommandSender(boss), _cmd(cmd), _hotkey(hotkey), _duringPress(false) { if (hotkey == 0) _hotkey = parseHotkey(label); setFlags(WIDGET_ENABLED/* | WIDGET_BORDER*/ | WIDGET_CLEARBG); _type = kButtonWidget; } ButtonWidget::ButtonWidget(GuiObject *boss, const Common::String &name, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : StaticTextWidget(boss, name, cleanupHotkey(label), tooltip), CommandSender(boss), _cmd(cmd), _hotkey(hotkey), _duringPress(false) { if (hotkey == 0) _hotkey = parseHotkey(label); setFlags(WIDGET_ENABLED/* | WIDGET_BORDER*/ | WIDGET_CLEARBG); _type = kButtonWidget; } void ButtonWidget::getMinSize(int &minWidth, int &minHeight) { const Graphics::Font &font = g_gui.getFont(_font); minWidth = font.getStringWidth(_label); minHeight = font.getFontHeight(); } void ButtonWidget::handleMouseUp(int x, int y, int button, int clickCount) { if (isEnabled() && _duringPress && x >= 0 && x < _w && y >= 0 && y < _h) { setUnpressedState(); sendCommand(_cmd, 0); } _duringPress = false; } void ButtonWidget::handleMouseDown(int x, int y, int button, int clickCount) { _duringPress = true; setPressedState(); } void ButtonWidget::drawWidget() { g_gui.theme()->drawButton(Common::Rect(_x, _y, _x + _w, _y + _h), _label, _state, getFlags()); } void ButtonWidget::setLabel(const Common::U32String &label) { StaticTextWidget::setLabel(cleanupHotkey(label)); } ButtonWidget *addClearButton(GuiObject *boss, const Common::String &name, uint32 cmd, int x, int y, int w, int h) { ButtonWidget *button; #ifndef DISABLE_FANCY_THEMES if (g_gui.xmlEval()->getVar("Globals.ShowSearchPic") == 1 && g_gui.theme()->supportsImages()) { if (!name.empty()) button = new PicButtonWidget(boss, name, (_("Clear value")), cmd); else button = new PicButtonWidget(boss, x, y, w, h, (_("Clear value")), cmd); ((PicButtonWidget *)button)->useThemeTransparency(true); ((PicButtonWidget *)button)->setGfx(g_gui.theme()->getImageSurface(ThemeEngine::kImageEraser)); } else #endif if (!name.empty()) button = new ButtonWidget(boss, name, Common::U32String("C"), (_("Clear value")), cmd); else button = new ButtonWidget(boss, x, y, w, h, Common::U32String("C"), (_("Clear value")), cmd); return button; } void ButtonWidget::setHighLighted(bool enable) { (enable) ? setFlags(WIDGET_HILITED) : clearFlags(WIDGET_HILITED); markAsDirty(); } void ButtonWidget::setPressedState() { setFlags(WIDGET_PRESSED); clearFlags(WIDGET_HILITED); markAsDirty(); } void ButtonWidget::setUnpressedState() { clearFlags(WIDGET_PRESSED); markAsDirty(); } #pragma mark - DropdownButtonWidget::DropdownButtonWidget(GuiObject *boss, int x, int y, int w, int h, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, x, y, w, h, label, tooltip, cmd, hotkey) { setFlags(getFlags() | WIDGET_TRACK_MOUSE); reset(); } DropdownButtonWidget::DropdownButtonWidget(GuiObject *boss, const Common::String &name, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, name, label, tooltip, cmd, hotkey) { setFlags(getFlags() | WIDGET_TRACK_MOUSE); reset(); } void DropdownButtonWidget::reset() { _inDropdown = false; _inButton = false; _dropdownWidth = g_gui.xmlEval()->getVar("Globals.DropdownButton.Width", 13); } bool DropdownButtonWidget::isInDropDown(int x, int y) const { Common::Rect dropdownRect(_w - _dropdownWidth, 0, _w, _h); return dropdownRect.contains(x, y); } void DropdownButtonWidget::handleMouseMoved(int x, int y, int button) { if (_entries.empty()) { return; } // Detect which part of the button the cursor is over bool inDropdown = isInDropDown(x, y); bool inButton = Common::Rect(_w, _h).contains(x, y) && !inDropdown; if (inDropdown != _inDropdown) { _inDropdown = inDropdown; markAsDirty(); } if (inButton != _inButton) { _inButton = inButton; markAsDirty(); } } void DropdownButtonWidget::handleMouseUp(int x, int y, int button, int clickCount) { if (isEnabled() && !_entries.empty() && _duringPress && isInDropDown(x, y)) { PopUpDialog popupDialog(this, "DropdownDialog", x + getAbsX(), y + getAbsY()); popupDialog.setPosition(getAbsX(), getAbsY() + _h); popupDialog.setLineHeight(_h); popupDialog.setPadding(_dropdownWidth, _dropdownWidth); for (uint i = 0; i < _entries.size(); i++) { popupDialog.appendEntry(_entries[i].label); } int newSel = popupDialog.runModal(); if (newSel != -1) { sendCommand(_entries[newSel].cmd, 0); } setUnpressedState(); _duringPress = false; } else { ButtonWidget::handleMouseUp(x, y, button, clickCount); } } void DropdownButtonWidget::reflowLayout() { ButtonWidget::reflowLayout(); reset(); } void DropdownButtonWidget::getMinSize(int &minWidth, int &minHeight) { ButtonWidget::getMinSize(minWidth, minHeight); if (minWidth >= 0) { minWidth += _dropdownWidth * 2; } } void DropdownButtonWidget::appendEntry(const Common::U32String &label, uint32 cmd) { Entry e; e.label = label; e.cmd = cmd; _entries.push_back(e); } void DropdownButtonWidget::clearEntries() { _entries.clear(); } void DropdownButtonWidget::drawWidget() { if (_entries.empty()) { // Degrade to a regular button g_gui.theme()->drawButton(Common::Rect(_x, _y, _x + _w, _y + _h), _label, _state); } else { g_gui.theme()->drawDropDownButton(Common::Rect(_x, _y, _x + _w, _y + _h), _dropdownWidth, _label, _state, _inButton, _inDropdown, (g_gui.useRTL() && _useRTL)); } } #pragma mark - PicButtonWidget::PicButtonWidget(GuiObject *boss, int x, int y, int w, int h, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, x, y, w, h, Common::U32String(""), tooltip, cmd, hotkey), _alpha(255), _transparency(false), _showButton(true) { setFlags(WIDGET_ENABLED/* | WIDGET_BORDER*/ | WIDGET_CLEARBG); _type = kButtonWidget; } PicButtonWidget::PicButtonWidget(GuiObject *boss, const Common::String &name, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, name, Common::U32String(""), tooltip, cmd, hotkey), _alpha(255), _transparency(false), _showButton(true) { setFlags(WIDGET_ENABLED/* | WIDGET_BORDER*/ | WIDGET_CLEARBG); _type = kButtonWidget; } PicButtonWidget::~PicButtonWidget() { for (int i = 0; i < kPicButtonStateMax + 1; i++) _gfx[i].free(); } void PicButtonWidget::setGfx(const Graphics::Surface *gfx, int statenum) { _gfx[statenum].free(); if (!gfx || !gfx->getPixels()) return; if (gfx->format.bytesPerPixel == 1) { warning("PicButtonWidget::setGfx got paletted surface passed"); return; } _gfx[statenum].copyFrom(*gfx); } void PicButtonWidget::setGfx(int w, int h, int r, int g, int b, int statenum) { if (w == -1) w = _w; if (h == -1) h = _h; const Graphics::PixelFormat &requiredFormat = g_gui.theme()->getPixelFormat(); _gfx[statenum].free(); _gfx[statenum].create(w, h, requiredFormat); _gfx[statenum].fillRect(Common::Rect(0, 0, w, h), _gfx[statenum].format.RGBToColor(r, g, b)); } void PicButtonWidget::drawWidget() { if (_showButton) g_gui.theme()->drawButton(Common::Rect(_x, _y, _x + _w, _y + _h), Common::U32String(""), _state, getFlags()); Graphics::Surface *gfx; if (_state == ThemeEngine::kStateHighlight) gfx = &_gfx[kPicButtonHighlight]; else if (_state == ThemeEngine::kStateDisabled) gfx = &_gfx[kPicButtonStateDisabled]; else if (_state == ThemeEngine::kStatePressed) gfx = &_gfx[kPicButtonStatePressed]; else gfx = &_gfx[kPicButtonStateEnabled]; if (!gfx->getPixels()) gfx = &_gfx[kPicButtonStateEnabled]; if (gfx->getPixels()) { // Check whether the set up surface needs to be converted to the GUI // color format. const Graphics::PixelFormat &requiredFormat = g_gui.theme()->getPixelFormat(); if (gfx->format != requiredFormat) { gfx->convertToInPlace(requiredFormat); } const int x = _x + (_w - gfx->w) / 2; const int y = _y + (_h - gfx->h) / 2; g_gui.theme()->drawSurface(Common::Point(x, y), *gfx, _transparency); } } #pragma mark - CheckboxWidget::CheckboxWidget(GuiObject *boss, int x, int y, int w, int h, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, x, y, w, h, label, tooltip, cmd, hotkey), _state(false) { setFlags(WIDGET_ENABLED); _type = kCheckboxWidget; } CheckboxWidget::CheckboxWidget(GuiObject *boss, const Common::String &name, const Common::U32String &label, Common::U32String tooltip, uint32 cmd, uint8 hotkey) : ButtonWidget(boss, name, label, tooltip, cmd, hotkey), _state(false) { setFlags(WIDGET_ENABLED); _type = kCheckboxWidget; } void CheckboxWidget::handleMouseUp(int x, int y, int button, int clickCount) { if (isEnabled() && _duringPress && x >= 0 && x < _w && y >= 0 && y < _h) { toggleState(); } _duringPress = false; } void CheckboxWidget::setState(bool state) { if (_state != state) { _state = state; //_flags ^= WIDGET_INV_BORDER; markAsDirty(); } sendCommand(_cmd, _state); } void CheckboxWidget::drawWidget() { g_gui.theme()->drawCheckbox(Common::Rect(_x, _y, _x + _w, _y + _h), _label, _state, Widget::_state, (g_gui.useRTL() && _useRTL)); } #pragma mark - RadiobuttonGroup::RadiobuttonGroup(GuiObject *boss, uint32 cmd) : CommandSender(boss) { _value = -1; _cmd = cmd; } void RadiobuttonGroup::setValue(int value) { Common::Array::iterator button = _buttons.begin(); while (button != _buttons.end()) { (*button)->setState((*button)->getValue() == value, false); button++; } _value = value; sendCommand(_cmd, _value); } void RadiobuttonGroup::setEnabled(bool ena) { Common::Array::iterator button = _buttons.begin(); while (button != _buttons.end()) { (*button)->setEnabled(ena); button++; } } #pragma mark - RadiobuttonWidget::RadiobuttonWidget(GuiObject *boss, int x, int y, int w, int h, RadiobuttonGroup *group, int value, const Common::U32String &label, Common::U32String tooltip, uint8 hotkey) : ButtonWidget(boss, x, y, w, h, label, tooltip, 0, hotkey), _state(false), _value(value), _group(group) { setFlags(WIDGET_ENABLED); _type = kRadiobuttonWidget; _group->addButton(this); } RadiobuttonWidget::RadiobuttonWidget(GuiObject *boss, const Common::String &name, RadiobuttonGroup *group, int value, const Common::U32String &label, Common::U32String tooltip, uint8 hotkey) : ButtonWidget(boss, name, label, tooltip, 0, hotkey), _state(false), _value(value), _group(group) { setFlags(WIDGET_ENABLED); _type = kRadiobuttonWidget; _group->addButton(this); } void RadiobuttonWidget::handleMouseUp(int x, int y, int button, int clickCount) { if (isEnabled() && _duringPress && x >= 0 && x < _w && y >= 0 && y < _h) { toggleState(); } _duringPress = false; } void RadiobuttonWidget::setState(bool state, bool setGroup) { if (setGroup) { _group->setValue(_value); return; } if (_state != state) { _state = state; //_flags ^= WIDGET_INV_BORDER; markAsDirty(); } sendCommand(_cmd, _state); } void RadiobuttonWidget::drawWidget() { g_gui.theme()->drawRadiobutton(Common::Rect(_x, _y, _x + _w, _y + _h), _label, _state, Widget::_state, (g_gui.useRTL() && _useRTL)); } #pragma mark - SliderWidget::SliderWidget(GuiObject *boss, int x, int y, int w, int h, Common::U32String tooltip, uint32 cmd) : Widget(boss, x, y, w, h, tooltip), CommandSender(boss), _cmd(cmd), _value(0), _oldValue(0), _valueMin(0), _valueMax(100), _isDragging(false), _labelWidth(0) { setFlags(WIDGET_ENABLED | WIDGET_TRACK_MOUSE | WIDGET_CLEARBG); _type = kSliderWidget; } SliderWidget::SliderWidget(GuiObject *boss, const Common::String &name, Common::U32String tooltip, uint32 cmd) : Widget(boss, name, tooltip), CommandSender(boss), _cmd(cmd), _value(0), _oldValue(0), _valueMin(0), _valueMax(100), _isDragging(false), _labelWidth(0) { setFlags(WIDGET_ENABLED | WIDGET_TRACK_MOUSE | WIDGET_CLEARBG); _type = kSliderWidget; } void SliderWidget::handleMouseMoved(int x, int y, int button) { if (g_gui.useRTL() && _useRTL == false) { x = _w - x; // If internal flipping is off, adjust the mouse to behave as if it were LTR. } if (isEnabled() && _isDragging) { int newValue = posToValue(x); if (newValue < _valueMin) newValue = _valueMin; else if (newValue > _valueMax) newValue = _valueMax; if (newValue != _value) { _value = newValue; markAsDirty(); sendCommand(_cmd, _value); // FIXME - hack to allow for "live update" in sound dialog } } } void SliderWidget::handleMouseDown(int x, int y, int button, int clickCount) { if (isEnabled()) { _isDragging = true; handleMouseMoved(x, y, button); } } void SliderWidget::handleMouseUp(int x, int y, int button, int clickCount) { if (isEnabled() && _isDragging) { sendCommand(_cmd, _value); } _isDragging = false; } void SliderWidget::handleMouseWheel(int x, int y, int direction) { if (isEnabled() && !_isDragging) { // Increment or decrement by one int newValue = _value - direction; if (newValue < _valueMin) newValue = _valueMin; else if (newValue > _valueMax) newValue = _valueMax; if (newValue != _value) { _value = newValue; markAsDirty(); sendCommand(_cmd, _value); // FIXME - hack to allow for "live update" in sound dialog } } } void SliderWidget::drawWidget() { Common::Rect r1(_x, _y, _x + _w, _y + _h); g_gui.theme()->drawSlider(r1, valueToBarWidth(_value), _state, (g_gui.useRTL() && _useRTL)); } int SliderWidget::valueToBarWidth(int value) { value = CLIP(value, _valueMin, _valueMax); return (_w * (value - _valueMin) / (_valueMax - _valueMin)); } int SliderWidget::valueToPos(int value) { value = CLIP(value, _valueMin, _valueMax); return ((_w - 1) * (value - _valueMin + 1) / (_valueMax - _valueMin)); } int SliderWidget::posToValue(int pos) { return (pos) * (_valueMax - _valueMin) / (_w - 1) + _valueMin; } #pragma mark - GraphicsWidget::GraphicsWidget(GuiObject *boss, int x, int y, int w, int h, Common::U32String tooltip) : Widget(boss, x, y, w, h, tooltip), _gfx(), _alpha(255), _transparency(false) { setFlags(WIDGET_ENABLED | WIDGET_CLEARBG); _type = kGraphicsWidget; } GraphicsWidget::GraphicsWidget(GuiObject *boss, const Common::String &name, Common::U32String tooltip) : Widget(boss, name, tooltip), _gfx(), _alpha(255), _transparency(false) { setFlags(WIDGET_ENABLED | WIDGET_CLEARBG); _type = kGraphicsWidget; } GraphicsWidget::~GraphicsWidget() { _gfx.free(); } void GraphicsWidget::setGfx(const Graphics::Surface *gfx) { _gfx.free(); if (!gfx || !gfx->getPixels()) return; if (gfx->format.bytesPerPixel == 1) { warning("GraphicsWidget::setGfx got paletted surface passed"); return; } _gfx.copyFrom(*gfx); } void GraphicsWidget::setGfx(int w, int h, int r, int g, int b) { if (w == -1) w = _w; if (h == -1) h = _h; const Graphics::PixelFormat &requiredFormat = g_gui.theme()->getPixelFormat(); _gfx.free(); _gfx.create(w, h, requiredFormat); _gfx.fillRect(Common::Rect(0, 0, w, h), _gfx.format.RGBToColor(r, g, b)); } void GraphicsWidget::drawWidget() { if (_gfx.getPixels()) { // Check whether the set up surface needs to be converted to the GUI // color format. const Graphics::PixelFormat &requiredFormat = g_gui.theme()->getPixelFormat(); if (_gfx.format != requiredFormat) { _gfx.convertToInPlace(requiredFormat); } const int x = _x + (_w - _gfx.w) / 2; const int y = _y + (_h - _gfx.h) / 2; g_gui.theme()->drawSurface(Common::Point(x, y), _gfx, _transparency); } } #pragma mark - ContainerWidget::ContainerWidget(GuiObject *boss, int x, int y, int w, int h) : Widget(boss, x, y, w, h), _backgroundType(ThemeEngine::kWidgetBackgroundBorder) { setFlags(WIDGET_ENABLED | WIDGET_CLEARBG); _type = kContainerWidget; } ContainerWidget::ContainerWidget(GuiObject *boss, const Common::String &name) : Widget(boss, name), _backgroundType(ThemeEngine::kWidgetBackgroundBorder) { setFlags(WIDGET_ENABLED | WIDGET_CLEARBG); _type = kContainerWidget; } ContainerWidget::~ContainerWidget() { // We also remove the widget from the boss to avoid segfaults, when the // deleted widget is an active widget in the boss. for (Widget *w = _firstWidget; w; w = w->next()) { _boss->removeWidget(w); } } bool ContainerWidget::containsWidget(Widget *w) const { return containsWidgetInChain(_firstWidget, w); } Widget *ContainerWidget::findWidget(int x, int y) { Widget *w = findWidgetInChain(_firstWidget, x, y); if (w) return w; return this; } void ContainerWidget::removeWidget(Widget *widget) { // We also remove the widget from the boss to avoid a reference to a // widget not in the widget chain anymore. _boss->removeWidget(widget); Widget::removeWidget(widget); } void ContainerWidget::setBackgroundType(ThemeEngine::WidgetBackground backgroundType) { _backgroundType = backgroundType; } void ContainerWidget::drawWidget() { g_gui.theme()->drawWidgetBackground(Common::Rect(_x, _y, _x + _w, _y + _h), _backgroundType); } #pragma mark - OptionsContainerWidget::OptionsContainerWidget(GuiObject *boss, const Common::String &name, const Common::String &dialogLayout, bool scrollable, const Common::String &domain) : Widget(boss, name), _domain(domain), _dialogLayout(dialogLayout), _parentDialog(nullptr), _scrollContainer(nullptr) { if (scrollable) { _scrollContainer = new ScrollContainerWidget(this, 0, 0, 0, 0, kReflowCmd); _scrollContainer->setTarget(this); _scrollContainer->setBackgroundType(GUI::ThemeEngine::kWidgetBackgroundNo); } } OptionsContainerWidget::~OptionsContainerWidget() { } void OptionsContainerWidget::reflowLayout() { Widget::reflowLayout(); if (!_dialogLayout.empty()) { if (!g_gui.xmlEval()->hasDialog(_dialogLayout)) { defineLayout(*g_gui.xmlEval(), _dialogLayout, _name); } g_gui.xmlEval()->reflowDialogLayout(_dialogLayout, _firstWidget); } if (_scrollContainer) { _scrollContainer->resize(_x, _y, _w, _h); } Widget *w = _firstWidget; while (w) { w->reflowLayout(); w = w->next(); } } bool OptionsContainerWidget::containsWidget(Widget *widget) const { return containsWidgetInChain(_firstWidget, widget); } Widget *OptionsContainerWidget::findWidget(int x, int y) { // Iterate over all child widgets and find the one which was clicked return Widget::findWidgetInChain(_firstWidget, x, y); } void OptionsContainerWidget::removeWidget(Widget *widget) { _boss->removeWidget(widget); Widget::removeWidget(widget); } GuiObject *OptionsContainerWidget::widgetsBoss() { if (_scrollContainer) { return _scrollContainer; } return this; } } // End of namespace GUI