2016-01-19 11:07:54 +01:00
|
|
|
/* 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.
|
2016-09-03 12:46:38 +02:00
|
|
|
*
|
2016-01-19 11:07:54 +01:00
|
|
|
* 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.
|
2016-09-03 12:46:38 +02:00
|
|
|
*
|
2016-01-19 11:07:54 +01:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* MIT License:
|
|
|
|
*
|
|
|
|
* Copyright (c) 2009 Alexei Svitkine, Eugene Sandulenko
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person
|
|
|
|
* obtaining a copy of this software and associated documentation
|
|
|
|
* files (the "Software"), to deal in the Software without
|
|
|
|
* restriction, including without limitation the rights to use,
|
|
|
|
* copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the
|
|
|
|
* Software is furnished to do so, subject to the following
|
|
|
|
* conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be
|
|
|
|
* included in all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/system.h"
|
2016-01-19 18:41:35 +01:00
|
|
|
#include "common/events.h"
|
2016-01-19 11:07:54 +01:00
|
|
|
|
2016-10-06 23:48:50 +02:00
|
|
|
#include "graphics/macgui/macfontmanager.h"
|
2016-07-29 11:43:28 +02:00
|
|
|
#include "graphics/macgui/macwindowmanager.h"
|
|
|
|
|
2016-01-19 11:07:54 +01:00
|
|
|
#include "wage/wage.h"
|
2016-01-19 18:16:36 +01:00
|
|
|
#include "wage/design.h"
|
2016-01-19 11:07:54 +01:00
|
|
|
#include "wage/gui.h"
|
|
|
|
#include "wage/dialog.h"
|
|
|
|
|
|
|
|
namespace Wage {
|
|
|
|
|
2016-01-19 18:16:36 +01:00
|
|
|
enum {
|
2016-01-20 22:36:36 +01:00
|
|
|
kDialogHeight = 113
|
2016-01-19 18:16:36 +01:00
|
|
|
};
|
|
|
|
|
2016-02-14 15:25:44 +01:00
|
|
|
Dialog::Dialog(Gui *gui, int width, const char *text, DialogButtonArray *buttons, uint defaultButton) :
|
2016-01-21 11:09:29 +01:00
|
|
|
_gui(gui), _text(text), _buttons(buttons), _defaultButton(defaultButton) {
|
2016-01-19 11:07:54 +01:00
|
|
|
assert(_gui->_engine);
|
|
|
|
assert(_gui->_engine->_world);
|
|
|
|
|
|
|
|
_font = getDialogFont();
|
|
|
|
|
2016-01-21 11:47:58 +01:00
|
|
|
_tempSurface.create(width + 1, kDialogHeight + 1, Graphics::PixelFormat::createFormatCLUT8());
|
2016-01-19 18:16:36 +01:00
|
|
|
|
2016-01-21 11:09:29 +01:00
|
|
|
_bbox.left = (_gui->_screen.w - width) / 2;
|
2016-01-19 18:16:36 +01:00
|
|
|
_bbox.top = (_gui->_screen.h - kDialogHeight) / 2;
|
2016-01-21 11:09:29 +01:00
|
|
|
_bbox.right = (_gui->_screen.w + width) / 2;
|
2016-01-19 18:16:36 +01:00
|
|
|
_bbox.bottom = (_gui->_screen.h + kDialogHeight) / 2;
|
|
|
|
|
2016-01-20 22:36:36 +01:00
|
|
|
_pressedButton = -1;
|
2016-01-19 18:16:36 +01:00
|
|
|
|
|
|
|
_mouseOverPressedButton = false;
|
2016-01-20 22:36:36 +01:00
|
|
|
|
|
|
|
// Adjust button positions
|
2016-02-14 15:25:44 +01:00
|
|
|
for (uint i = 0; i < _buttons->size(); i++)
|
2016-01-20 22:36:36 +01:00
|
|
|
_buttons->operator[](i)->bounds.translate(_bbox.left, _bbox.top);
|
2016-01-21 11:47:58 +01:00
|
|
|
|
|
|
|
_needsRedraw = true;
|
2016-01-19 11:07:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Dialog::~Dialog() {
|
2016-07-15 11:24:05 +02:00
|
|
|
for (uint i = 0; i < _buttons->size(); i++)
|
|
|
|
delete _buttons->operator[](i);
|
2016-01-19 11:07:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const Graphics::Font *Dialog::getDialogFont() {
|
2016-10-06 23:48:50 +02:00
|
|
|
return _gui->_wm._fontMan->getFont(Graphics::MacFont(Graphics::kMacFontChicago, 12));
|
2016-01-19 11:07:54 +01:00
|
|
|
}
|
|
|
|
|
2016-01-19 18:16:36 +01:00
|
|
|
void Dialog::paint() {
|
2016-04-28 11:28:55 +02:00
|
|
|
Design::drawFilledRect(&_gui->_screen, _bbox, kColorWhite, _gui->_wm.getPatterns(), kPatternSolid);
|
2016-01-20 22:36:36 +01:00
|
|
|
_font->drawString(&_gui->_screen, _text, _bbox.left + 24, _bbox.top + 16, _bbox.width(), kColorBlack);
|
2016-01-19 18:16:36 +01:00
|
|
|
|
|
|
|
static int boxOutline[] = { 1, 0, 0, 1, 1 };
|
|
|
|
drawOutline(_bbox, boxOutline, ARRAYSIZE(boxOutline));
|
|
|
|
|
2016-02-14 15:25:44 +01:00
|
|
|
for (uint i = 0; i < _buttons->size(); i++) {
|
2016-01-19 18:16:36 +01:00
|
|
|
DialogButton *button = _buttons->operator[](i);
|
|
|
|
static int buttonOutline[] = { 0, 0, 0, 0, 1 };
|
|
|
|
|
2016-01-20 22:36:36 +01:00
|
|
|
if (i == _defaultButton) {
|
2016-01-19 18:16:36 +01:00
|
|
|
buttonOutline[0] = buttonOutline[1] = 1;
|
|
|
|
} else {
|
|
|
|
buttonOutline[0] = buttonOutline[1] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int color = kColorBlack;
|
|
|
|
|
2016-02-14 15:25:44 +01:00
|
|
|
if ((int)i == _pressedButton && _mouseOverPressedButton) {
|
2016-01-19 18:16:36 +01:00
|
|
|
Common::Rect bb(button->bounds.left + 5, button->bounds.top + 5,
|
|
|
|
button->bounds.right - 5, button->bounds.bottom - 5);
|
|
|
|
|
2016-04-28 11:28:55 +02:00
|
|
|
Design::drawFilledRect(&_gui->_screen, bb, kColorBlack, _gui->_wm.getPatterns(), kPatternSolid);
|
2016-01-19 18:16:36 +01:00
|
|
|
|
|
|
|
color = kColorWhite;
|
|
|
|
}
|
|
|
|
int w = _font->getStringWidth(button->text);
|
|
|
|
int x = button->bounds.left + (button->bounds.width() - w) / 2;
|
2016-01-20 22:36:36 +01:00
|
|
|
int y = button->bounds.top + 6;
|
2016-01-19 18:16:36 +01:00
|
|
|
|
2016-01-20 22:36:36 +01:00
|
|
|
_font->drawString(&_gui->_screen, button->text, x, y, _bbox.width(), color);
|
2016-01-19 18:16:36 +01:00
|
|
|
|
|
|
|
drawOutline(button->bounds, buttonOutline, ARRAYSIZE(buttonOutline));
|
|
|
|
}
|
|
|
|
|
|
|
|
g_system->copyRectToScreen(_gui->_screen.getBasePtr(_bbox.left, _bbox.top), _gui->_screen.pitch,
|
2016-01-20 22:36:36 +01:00
|
|
|
_bbox.left, _bbox.top, _bbox.width() + 1, _bbox.height() + 1);
|
2016-01-21 11:47:58 +01:00
|
|
|
|
|
|
|
_needsRedraw = false;
|
2016-01-19 18:16:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Dialog::drawOutline(Common::Rect &bounds, int *spec, int speclen) {
|
|
|
|
for (int i = 0; i < speclen; i++)
|
|
|
|
if (spec[i] != 0)
|
2016-01-20 22:36:36 +01:00
|
|
|
Design::drawRect(&_gui->_screen, bounds.left + i, bounds.top + i, bounds.right - i, bounds.bottom - i,
|
2016-04-28 11:28:55 +02:00
|
|
|
1, kColorBlack, _gui->_wm.getPatterns(), kPatternSolid);
|
2016-01-19 18:16:36 +01:00
|
|
|
}
|
|
|
|
|
2016-01-21 11:47:58 +01:00
|
|
|
int Dialog::run() {
|
2016-01-19 18:41:35 +01:00
|
|
|
bool shouldQuit = false;
|
2016-01-21 11:47:58 +01:00
|
|
|
Common::Rect r(_bbox);
|
2016-01-19 18:41:35 +01:00
|
|
|
|
2016-01-21 12:09:37 +01:00
|
|
|
_tempSurface.copyRectToSurface(_gui->_screen.getBasePtr(_bbox.left, _bbox.top), _gui->_screen.pitch, 0, 0, _bbox.width() + 1, _bbox.height() + 1);
|
2016-04-28 12:09:08 +02:00
|
|
|
_gui->_wm.pushArrowCursor();
|
2016-01-19 18:41:35 +01:00
|
|
|
|
|
|
|
while (!shouldQuit) {
|
|
|
|
Common::Event event;
|
|
|
|
|
|
|
|
while (_gui->_engine->_eventMan->pollEvent(event)) {
|
|
|
|
switch (event.type) {
|
|
|
|
case Common::EVENT_QUIT:
|
2016-01-21 11:09:29 +01:00
|
|
|
_gui->_engine->_shouldQuit = true;
|
2016-01-19 18:41:35 +01:00
|
|
|
shouldQuit = true;
|
|
|
|
break;
|
2016-01-21 11:09:29 +01:00
|
|
|
case Common::EVENT_MOUSEMOVE:
|
2016-01-21 11:47:58 +01:00
|
|
|
mouseMove(event.mouse.x, event.mouse.y);
|
|
|
|
break;
|
|
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
|
|
mouseClick(event.mouse.x, event.mouse.y);
|
|
|
|
break;
|
|
|
|
case Common::EVENT_LBUTTONUP:
|
|
|
|
shouldQuit = mouseRaise(event.mouse.x, event.mouse.y);
|
|
|
|
break;
|
|
|
|
case Common::EVENT_KEYDOWN:
|
|
|
|
switch (event.kbd.keycode) {
|
|
|
|
case Common::KEYCODE_ESCAPE:
|
|
|
|
_pressedButton = -1;
|
|
|
|
shouldQuit = true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2016-01-19 18:41:35 +01:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-21 11:47:58 +01:00
|
|
|
if (_needsRedraw)
|
|
|
|
paint();
|
|
|
|
|
2016-01-19 18:41:35 +01:00
|
|
|
g_system->updateScreen();
|
|
|
|
g_system->delayMillis(50);
|
|
|
|
}
|
2016-01-21 11:47:58 +01:00
|
|
|
|
2016-01-21 12:09:37 +01:00
|
|
|
_gui->_screen.copyRectToSurface(_tempSurface.getBasePtr(0, 0), _tempSurface.pitch, _bbox.left, _bbox.top, _bbox.width() + 1, _bbox.height() + 1);
|
|
|
|
g_system->copyRectToScreen(_gui->_screen.getBasePtr(r.left, r.top), _gui->_screen.pitch, r.left, r.top, r.width() + 1, r.height() + 1);
|
2016-01-21 11:47:58 +01:00
|
|
|
|
2016-04-28 12:09:08 +02:00
|
|
|
_gui->_wm.popCursor();
|
2016-02-04 18:59:36 +01:00
|
|
|
|
2016-01-21 11:47:58 +01:00
|
|
|
return _pressedButton;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Dialog::matchButton(int x, int y) {
|
2016-02-14 15:25:44 +01:00
|
|
|
for (uint i = 0; i < _buttons->size(); i++)
|
2016-01-21 11:47:58 +01:00
|
|
|
if (_buttons->operator[](i)->bounds.contains(x, y))
|
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dialog::mouseMove(int x, int y) {
|
|
|
|
if (_pressedButton != -1) {
|
|
|
|
int match = matchButton(x, y);
|
|
|
|
|
|
|
|
if (_mouseOverPressedButton && match != _pressedButton) {
|
|
|
|
_mouseOverPressedButton = false;
|
|
|
|
_needsRedraw = true;
|
|
|
|
} else if (!_mouseOverPressedButton && match == _pressedButton) {
|
|
|
|
_mouseOverPressedButton = true;
|
|
|
|
_needsRedraw = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Dialog::mouseClick(int x, int y) {
|
|
|
|
int match = matchButton(x, y);
|
|
|
|
|
2016-01-21 12:09:37 +01:00
|
|
|
if (match != -1) {
|
2016-01-21 11:47:58 +01:00
|
|
|
_pressedButton = match;
|
|
|
|
_mouseOverPressedButton = true;
|
|
|
|
|
|
|
|
_needsRedraw = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int Dialog::mouseRaise(int x, int y) {
|
|
|
|
bool res = false;
|
|
|
|
|
|
|
|
if (_pressedButton != -1) {
|
|
|
|
if (matchButton(x, y) == _pressedButton)
|
|
|
|
res = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2016-01-19 18:41:35 +01:00
|
|
|
}
|
|
|
|
|
2016-01-19 11:07:54 +01:00
|
|
|
} // End of namespace Wage
|