2002-07-05 16:56:53 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2006-01-18 17:39:49 +00:00
|
|
|
* Copyright (C) 2002-2006 The ScummVM project
|
2002-07-05 16:56:53 +00:00
|
|
|
*
|
|
|
|
* 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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2002-07-05 16:56:53 +00:00
|
|
|
*
|
2006-02-11 10:08:56 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2002-07-05 16:56:53 +00:00
|
|
|
*/
|
|
|
|
|
2005-06-24 15:23:51 +00:00
|
|
|
#include "common/stdafx.h"
|
2005-06-24 16:05:24 +00:00
|
|
|
|
2005-01-10 22:06:49 +00:00
|
|
|
#include "gui/newgui.h"
|
|
|
|
#include "gui/dialog.h"
|
|
|
|
#include "gui/widget.h"
|
2005-05-19 17:03:31 +00:00
|
|
|
#include "gui/PopUpWidget.h"
|
2005-01-10 22:06:49 +00:00
|
|
|
|
|
|
|
#include "common/system.h"
|
2002-07-05 16:56:53 +00:00
|
|
|
|
2003-11-10 23:40:48 +00:00
|
|
|
namespace GUI {
|
|
|
|
|
2002-07-28 20:10:39 +00:00
|
|
|
/*
|
|
|
|
* TODO list
|
2005-01-06 19:09:34 +00:00
|
|
|
* - add some sense of the window being "active" (i.e. in front) or not. If it
|
2002-08-04 01:18:06 +00:00
|
|
|
* was inactive and just became active, reset certain vars (like who is focused).
|
|
|
|
* Maybe we should just add lostFocus and receivedFocus methods to Dialog, just
|
|
|
|
* like we have for class Widget?
|
2002-07-28 20:10:39 +00:00
|
|
|
* ...
|
|
|
|
*/
|
|
|
|
|
2004-12-25 23:20:37 +00:00
|
|
|
Dialog::Dialog(int x, int y, int w, int h)
|
|
|
|
: GuiObject(x, y, w, h),
|
2006-01-28 23:03:39 +00:00
|
|
|
_mouseWidget(0), _focusedWidget(0), _dragWidget(0), _visible(false), _drawingHints(0) {
|
|
|
|
_drawingHints = THEME_HINT_FIRST_DRAW | THEME_HINT_SAVE_BACKGROUND;
|
2004-12-25 23:20:37 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
Dialog::~Dialog() {
|
2003-11-07 14:50:32 +00:00
|
|
|
delete _firstWidget;
|
2002-10-12 00:26:24 +00:00
|
|
|
_firstWidget = 0;
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
int Dialog::runModal() {
|
2002-10-16 17:37:30 +00:00
|
|
|
// Open up
|
|
|
|
open();
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-10-16 17:37:30 +00:00
|
|
|
// Start processing events
|
2003-11-02 02:18:16 +00:00
|
|
|
g_gui.runLoop();
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-11-19 01:36:47 +00:00
|
|
|
// Return the result code
|
|
|
|
return _result;
|
2002-10-16 17:37:30 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::open() {
|
2002-09-08 16:00:13 +00:00
|
|
|
Widget *w = _firstWidget;
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-11-19 01:36:47 +00:00
|
|
|
_result = 0;
|
2002-09-08 16:00:13 +00:00
|
|
|
_visible = true;
|
2003-11-02 02:18:16 +00:00
|
|
|
g_gui.openDialog(this);
|
2003-11-08 23:22:16 +00:00
|
|
|
|
2002-09-08 16:00:13 +00:00
|
|
|
// Search for the first objects that wantsFocus() (if any) and give it the focus
|
|
|
|
while (w && !w->wantsFocus()) {
|
|
|
|
w = w->_next;
|
|
|
|
}
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-09-08 16:00:13 +00:00
|
|
|
if (w) {
|
|
|
|
w->receivedFocus();
|
|
|
|
_focusedWidget = w;
|
|
|
|
}
|
2002-07-26 20:38:55 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::close() {
|
2002-09-08 16:00:13 +00:00
|
|
|
_visible = false;
|
2003-11-02 02:18:16 +00:00
|
|
|
g_gui.closeTopDialog();
|
2002-09-08 16:00:13 +00:00
|
|
|
|
2002-07-26 20:38:55 +00:00
|
|
|
if (_mouseWidget) {
|
|
|
|
_mouseWidget->handleMouseLeft(0);
|
|
|
|
_mouseWidget = 0;
|
|
|
|
}
|
2002-11-21 15:20:52 +00:00
|
|
|
releaseFocus();
|
|
|
|
}
|
|
|
|
|
2006-01-27 15:43:23 +00:00
|
|
|
void Dialog::handleScreenChanged() {
|
|
|
|
// The screen has changed. That means the screen visual may also have
|
|
|
|
// changed, so any cached image may be invalid. The subsequent redraw
|
|
|
|
// should be treated as the very first draw.
|
|
|
|
|
2006-01-28 23:03:39 +00:00
|
|
|
_drawingHints |= THEME_HINT_FIRST_DRAW;
|
2006-01-27 15:43:23 +00:00
|
|
|
Widget *w = _firstWidget;
|
|
|
|
while (w) {
|
|
|
|
w->setHints(THEME_HINT_FIRST_DRAW);
|
|
|
|
w = w->_next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::releaseFocus() {
|
2002-07-26 20:38:55 +00:00
|
|
|
if (_focusedWidget) {
|
|
|
|
_focusedWidget->lostFocus();
|
|
|
|
_focusedWidget = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::draw() {
|
2003-11-02 02:18:16 +00:00
|
|
|
g_gui._needRedraw = true;
|
2002-11-10 19:39:32 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::drawDialog() {
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2002-09-08 16:00:13 +00:00
|
|
|
if (!isVisible())
|
|
|
|
return;
|
2002-07-07 13:14:34 +00:00
|
|
|
|
2006-01-28 23:03:39 +00:00
|
|
|
g_gui.theme()->drawDialogBackground(Common::Rect(_x, _y, _x+_w, _y+_h), _drawingHints);
|
|
|
|
_drawingHints &= ~THEME_HINT_FIRST_DRAW;
|
2002-07-07 13:14:34 +00:00
|
|
|
|
2003-11-02 22:31:20 +00:00
|
|
|
// Draw all children
|
|
|
|
Widget *w = _firstWidget;
|
2002-07-05 16:56:53 +00:00
|
|
|
while (w) {
|
|
|
|
w->draw();
|
|
|
|
w = w->_next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleMouseDown(int x, int y, int button, int clickCount) {
|
2002-07-16 10:52:48 +00:00
|
|
|
Widget *w;
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
w = findWidget(x, y);
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2004-12-28 21:07:34 +00:00
|
|
|
_dragWidget = w;
|
2002-07-13 18:32:09 +00:00
|
|
|
|
2002-09-08 16:00:13 +00:00
|
|
|
// If the click occured inside a widget which is not the currently
|
|
|
|
// focused one, change the focus to that widget.
|
2004-12-25 23:20:37 +00:00
|
|
|
if (w && w != _focusedWidget && w->wantsFocus()) {
|
2002-07-16 10:52:48 +00:00
|
|
|
// The focus will change. Tell the old focused widget (if any)
|
|
|
|
// that it lost the focus.
|
2002-11-21 15:20:52 +00:00
|
|
|
releaseFocus();
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
// Tell the new focused widget (if any) that it just gained the focus.
|
|
|
|
if (w)
|
2002-07-16 22:34:16 +00:00
|
|
|
w->receivedFocus();
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-07-16 10:52:48 +00:00
|
|
|
_focusedWidget = w;
|
2002-07-13 22:41:29 +00:00
|
|
|
}
|
2002-07-16 10:52:48 +00:00
|
|
|
|
2004-12-25 23:20:37 +00:00
|
|
|
if (w)
|
|
|
|
w->handleMouseDown(x - (w->getAbsX() - _x), y - (w->getAbsY() - _y), button, clickCount);
|
2002-07-13 18:32:09 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleMouseUp(int x, int y, int button, int clickCount) {
|
2002-07-13 22:41:29 +00:00
|
|
|
Widget *w;
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-07-13 22:41:29 +00:00
|
|
|
if (_focusedWidget) {
|
2004-12-25 23:20:37 +00:00
|
|
|
//w = _focusedWidget;
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2002-07-13 22:41:29 +00:00
|
|
|
// Lose focus on mouseup unless the widget requested to retain the focus
|
2002-07-16 10:52:48 +00:00
|
|
|
if (! (_focusedWidget->getFlags() & WIDGET_RETAIN_FOCUS )) {
|
2002-11-21 15:20:52 +00:00
|
|
|
releaseFocus();
|
2002-07-16 10:52:48 +00:00
|
|
|
}
|
2002-07-13 22:41:29 +00:00
|
|
|
}
|
|
|
|
|
2004-12-28 21:07:34 +00:00
|
|
|
w = _dragWidget;
|
|
|
|
|
|
|
|
if (w)
|
2003-11-02 17:41:01 +00:00
|
|
|
w->handleMouseUp(x - (w->getAbsX() - _x), y - (w->getAbsY() - _y), button, clickCount);
|
2004-12-25 23:20:37 +00:00
|
|
|
|
2004-12-28 21:07:34 +00:00
|
|
|
_dragWidget = 0;
|
2002-07-05 16:56:53 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleMouseWheel(int x, int y, int direction) {
|
2002-10-16 20:32:12 +00:00
|
|
|
Widget *w;
|
|
|
|
|
|
|
|
// This may look a bit backwards, but I think it makes more sense for
|
|
|
|
// the mouse wheel to primarily affect the widget the mouse is at than
|
|
|
|
// the widget that happens to be focused.
|
|
|
|
|
|
|
|
w = findWidget(x, y);
|
|
|
|
if (!w)
|
|
|
|
w = _focusedWidget;
|
|
|
|
if (w)
|
|
|
|
w->handleMouseWheel(x, y, direction);
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleKeyDown(uint16 ascii, int keycode, int modifiers) {
|
2002-07-13 22:41:29 +00:00
|
|
|
if (_focusedWidget) {
|
2002-11-22 14:02:54 +00:00
|
|
|
if (_focusedWidget->handleKeyDown(ascii, keycode, modifiers))
|
2002-07-27 14:16:14 +00:00
|
|
|
return;
|
2003-07-22 16:30:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Hotkey handling
|
2003-07-23 16:44:15 +00:00
|
|
|
if (ascii != 0) {
|
|
|
|
Widget *w = _firstWidget;
|
|
|
|
ascii = toupper(ascii);
|
|
|
|
while (w) {
|
|
|
|
if (w->_type == kButtonWidget && ascii == toupper(((ButtonWidget *)w)->_hotkey)) {
|
|
|
|
// The hotkey for widget w was pressed. We fake a mouse click into the
|
|
|
|
// button by invoking the appropriate methods.
|
|
|
|
w->handleMouseDown(0, 0, 1, 1);
|
|
|
|
w->handleMouseUp(0, 0, 1, 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
w = w->_next;
|
2002-07-08 00:29:47 +00:00
|
|
|
}
|
|
|
|
}
|
2002-07-27 14:16:14 +00:00
|
|
|
|
|
|
|
// ESC closes all dialogs by default
|
2003-12-06 13:55:50 +00:00
|
|
|
if (keycode == 27) {
|
|
|
|
setResult(-1);
|
2002-07-27 14:16:14 +00:00
|
|
|
close();
|
2003-12-06 13:55:50 +00:00
|
|
|
}
|
2003-03-06 19:52:54 +00:00
|
|
|
|
2002-11-21 15:20:52 +00:00
|
|
|
// TODO: tab/shift-tab should focus the next/previous focusable widget
|
2002-07-08 00:29:47 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleKeyUp(uint16 ascii, int keycode, int modifiers) {
|
2003-11-07 02:31:44 +00:00
|
|
|
// Focused widget receives keyup events
|
2002-07-13 22:41:29 +00:00
|
|
|
if (_focusedWidget)
|
2002-11-22 14:02:54 +00:00
|
|
|
_focusedWidget->handleKeyUp(ascii, keycode, modifiers);
|
2002-07-13 18:32:09 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleMouseMoved(int x, int y, int button) {
|
2002-07-13 22:41:29 +00:00
|
|
|
Widget *w;
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2004-12-25 23:20:37 +00:00
|
|
|
//if (!button)
|
2004-12-28 21:07:34 +00:00
|
|
|
// _dragWidget = 0;
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2004-12-28 21:07:34 +00:00
|
|
|
if (_focusedWidget && !_dragWidget) {
|
2002-07-13 22:41:29 +00:00
|
|
|
w = _focusedWidget;
|
2003-11-02 17:41:01 +00:00
|
|
|
int wx = w->getAbsX() - _x;
|
|
|
|
int wy = w->getAbsY() - _y;
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2002-07-13 22:41:29 +00:00
|
|
|
// We still send mouseEntered/Left messages to the focused item
|
|
|
|
// (but to no other items).
|
2003-11-02 17:41:01 +00:00
|
|
|
bool mouseInFocusedWidget = (x >= wx && x < wx + w->_w && y >= wy && y < wy + w->_h);
|
2002-07-13 22:41:29 +00:00
|
|
|
if (mouseInFocusedWidget && _mouseWidget != w) {
|
2004-12-25 23:20:37 +00:00
|
|
|
if (_mouseWidget)
|
|
|
|
_mouseWidget->handleMouseLeft(button);
|
2002-07-13 22:41:29 +00:00
|
|
|
_mouseWidget = w;
|
|
|
|
w->handleMouseEntered(button);
|
|
|
|
} else if (!mouseInFocusedWidget && _mouseWidget == w) {
|
|
|
|
_mouseWidget = 0;
|
|
|
|
w->handleMouseLeft(button);
|
|
|
|
}
|
|
|
|
|
2003-11-02 17:41:01 +00:00
|
|
|
w->handleMouseMoved(x - wx, y - wy, button);
|
2002-08-04 01:18:06 +00:00
|
|
|
}
|
2005-01-06 19:09:34 +00:00
|
|
|
|
2004-12-25 23:20:37 +00:00
|
|
|
// While a "drag" is in process (i.e. mouse is moved while a button is pressed),
|
|
|
|
// only deal with the widget in which the click originated.
|
2004-12-28 21:07:34 +00:00
|
|
|
if (_dragWidget)
|
|
|
|
w = _dragWidget;
|
|
|
|
else
|
|
|
|
w = findWidget(x, y);
|
2002-08-04 01:18:06 +00:00
|
|
|
|
|
|
|
if (_mouseWidget != w) {
|
|
|
|
if (_mouseWidget)
|
|
|
|
_mouseWidget->handleMouseLeft(button);
|
|
|
|
if (w)
|
|
|
|
w->handleMouseEntered(button);
|
|
|
|
_mouseWidget = w;
|
2005-01-06 19:09:34 +00:00
|
|
|
}
|
2002-08-04 01:18:06 +00:00
|
|
|
|
2004-12-25 23:20:37 +00:00
|
|
|
if (w && (w->getFlags() & WIDGET_TRACK_MOUSE)) {
|
|
|
|
w->handleMouseMoved(x - (w->getAbsX() - _x), y - (w->getAbsY() - _y), button);
|
2002-07-06 12:57:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleTickle() {
|
2003-11-07 02:31:44 +00:00
|
|
|
// Focused widget receives tickle notifications
|
2002-07-13 22:41:29 +00:00
|
|
|
if (_focusedWidget && _focusedWidget->getFlags() & WIDGET_WANT_TICKLE) {
|
|
|
|
_focusedWidget->handleTickle();
|
|
|
|
}
|
|
|
|
}
|
2002-07-06 12:57:51 +00:00
|
|
|
|
2003-03-06 19:52:54 +00:00
|
|
|
void Dialog::handleCommand(CommandSender *sender, uint32 cmd, uint32 data) {
|
2002-07-07 22:44:30 +00:00
|
|
|
switch (cmd) {
|
|
|
|
case kCloseCmd:
|
|
|
|
close();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-05 16:56:53 +00:00
|
|
|
/*
|
|
|
|
* Determine the widget at location (x,y) if any. Assumes the coordinates are
|
|
|
|
* in the local coordinate system, i.e. relative to the top left of the dialog.
|
|
|
|
*/
|
2003-03-06 19:52:54 +00:00
|
|
|
Widget *Dialog::findWidget(int x, int y) {
|
2003-11-02 18:57:20 +00:00
|
|
|
return Widget::findWidgetInChain(_firstWidget, x, y);
|
2002-07-05 16:56:53 +00:00
|
|
|
}
|
|
|
|
|
2005-05-18 14:11:53 +00:00
|
|
|
ButtonWidget *Dialog::addButton(GuiObject *boss, int x, int y, const Common::String &label, uint32 cmd, char hotkey, WidgetSize ws) {
|
2005-05-11 19:30:30 +00:00
|
|
|
int w = kButtonWidth;
|
|
|
|
int h = kButtonHeight;
|
|
|
|
if (ws == kBigWidgetSize) {
|
|
|
|
w = kBigButtonWidth;
|
|
|
|
h = kBigButtonHeight;
|
|
|
|
}
|
2005-05-18 14:11:53 +00:00
|
|
|
return new ButtonWidget(boss, x, y, w, h, label, cmd, hotkey, ws);
|
2002-07-05 16:56:53 +00:00
|
|
|
}
|
2003-11-10 23:40:48 +00:00
|
|
|
|
2005-05-18 10:12:20 +00:00
|
|
|
CheckboxWidget *Dialog::addCheckbox(GuiObject *boss, int x, int y, const Common::String &label, uint32 cmd, char hotkey, WidgetSize ws) {
|
2005-05-18 09:21:03 +00:00
|
|
|
int w, h;
|
|
|
|
|
|
|
|
if (ws == kBigWidgetSize) {
|
|
|
|
h = kBigButtonHeight;
|
|
|
|
} else {
|
|
|
|
h = kButtonHeight;
|
|
|
|
}
|
|
|
|
|
2005-06-03 12:33:03 +00:00
|
|
|
w = g_gui.getFontHeight() + 10 + g_gui.getStringWidth(label);
|
2005-05-18 09:21:03 +00:00
|
|
|
|
2005-05-18 10:12:20 +00:00
|
|
|
return new CheckboxWidget(boss, x, y, w, h, label, cmd, hotkey, ws);
|
2005-05-18 09:21:03 +00:00
|
|
|
}
|
2005-01-10 22:06:49 +00:00
|
|
|
|
2005-05-18 15:58:39 +00:00
|
|
|
SliderWidget *Dialog::addSlider(GuiObject *boss, int x, int y, uint32 cmd, WidgetSize ws) {
|
|
|
|
int w, h;
|
|
|
|
|
|
|
|
if (ws == kBigWidgetSize) {
|
|
|
|
w = kBigSliderWidth;
|
|
|
|
h = kBigSliderHeight;
|
|
|
|
} else {
|
|
|
|
w = kSliderWidth;
|
|
|
|
h = kSliderHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new SliderWidget(boss, x, y, w, h, cmd);
|
|
|
|
}
|
|
|
|
|
2005-05-19 17:03:31 +00:00
|
|
|
PopUpWidget *Dialog::addPopUp(GuiObject *boss, int x, int y, int w, const Common::String &label, uint labelWidth, WidgetSize ws) {
|
2005-06-03 12:33:03 +00:00
|
|
|
return new PopUpWidget(boss, x, y, w, kLineHeight, label, labelWidth, ws);
|
2005-05-19 17:03:31 +00:00
|
|
|
}
|
|
|
|
|
2005-01-10 22:06:49 +00:00
|
|
|
uint32 GuiObject::getMillis() {
|
|
|
|
return g_system->getMillis();
|
|
|
|
}
|
|
|
|
|
2003-11-10 23:40:48 +00:00
|
|
|
} // End of namespace GUI
|