262 lines
7.6 KiB
C++

/* 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.
*
*/
/*
* Based on
* WebVenture (c) 2010, Sean Kasun
* https://github.com/mrkite/webventure, http://seancode.com/webventure/
*
* Used with explicit permission from the author
*/
#include "common/system.h"
#include "macventure/dialog.h"
namespace MacVenture {
Dialog::Dialog(Gui *gui, Common::Point pos, uint width, uint height) :
_gui(gui), _bounds(Common::Rect(pos.x, pos.y, pos.x + width, pos.y + height)) {}
Dialog::Dialog(Gui *gui, PrebuiltDialogs prebuilt) {
_gui = gui;
const PrebuiltDialog &dialog = g_prebuiltDialogs[prebuilt];
calculateBoundsFromPrebuilt(dialog.bounds);
for (int i = 0; dialog.elements[i].type != kDEEnd; i++) {
addPrebuiltElement(dialog.elements[i]);
}
}
Dialog::~Dialog() {
for (Common::Array<DialogElement*>::iterator it = _elements.begin(); it != _elements.end(); it++) {
delete *it;
}
}
void Dialog::handleDialogAction(DialogElement *trigger, DialogAction action) {
switch(action) {
case kDACloseDialog:
_gui->closeDialog();
break;
case kDASubmit:
_gui->setTextInput(_userInput);
_gui->closeDialog();
break;
case kDASaveAs:
_gui->saveGame();
_gui->closeDialog();
break;
case kDALoadGame:
_gui->loadGame();
_gui->closeDialog();
break;
case kDANewGame:
_gui->newGame();
_gui->closeDialog();
break;
case kDAQuit:
_gui->quitGame();
_gui->closeDialog();
break;
default:
break;
}
}
const Graphics::Font &Dialog::getFont() {
return _gui->getCurrentFont();
}
bool Dialog::processEvent(Common::Event event) {
for (Common::Array<DialogElement*>::iterator it = _elements.begin(); it != _elements.end(); it++) {
if ((*it)->processEvent(this, event)) {
return true;
}
}
return false;
}
void Dialog::addButton(Common::String title, MacVenture::DialogAction action, Common::Point position, uint width, uint height) {
_elements.push_back(new DialogButton(this, title, action, position, width, height));
}
void Dialog::addText(Common::String content, Common::Point position) {
_elements.push_back(new DialogPlainText(this, content, position));
}
void Dialog::addTextInput(Common::Point position, int width, int height) {
_elements.push_back(new DialogTextInput(this, position, width, height));
}
void Dialog::draw() {
Graphics::ManagedSurface compose;
// Compose the surface
compose.create(_bounds.width(), _bounds.height());
Common::Rect base(0, 0, _bounds.width(), _bounds.height());
compose.fillRect(base, kColorWhite);
compose.frameRect(base, kColorBlack);
for (Common::Array<DialogElement*>::iterator it = _elements.begin(); it != _elements.end(); it++) {
(*it)->draw(this, compose);
}
g_system->copyRectToScreen(compose.getPixels(), compose.pitch,
_bounds.left, _bounds.top, _bounds.width(), _bounds.height());
}
void Dialog::localize(Common::Point &point) {
point.x -= _bounds.left;
point.y -= _bounds.top;
}
void Dialog::setUserInput(Common::String content) {
_userInput = content;
}
void Dialog::addPrebuiltElement(const MacVenture::PrebuiltDialogElement &element) {
Common::Point position(element.left, element.top);
switch(element.type) {
case kDEButton:
addButton(element.title, element.action, position, element.width, element.height);
break;
case kDEPlainText:
addText(element.title, position);
break;
case kDETextInput:
addTextInput(position, element.width, element.height);
break;
default:
break;
}
}
// Dialog Element
DialogElement::DialogElement(Dialog *dialog, Common::String title, DialogAction action, Common::Point position, uint width, uint height) :
_text(title), _action(action) {
if (width == 0) {
width = dialog->getFont().getStringWidth(title);
}
if (height == 0) {
height = dialog->getFont().getFontHeight();
}
_bounds = Common::Rect(position.x, position.y, position.x + width, position.y + height);
}
bool DialogElement::processEvent(MacVenture::Dialog *dialog, Common::Event event) {
return doProcessEvent(dialog, event);
}
void DialogElement::draw(MacVenture::Dialog *dialog, Graphics::ManagedSurface &target) {
doDraw(dialog, target);
}
const Common::String &DialogElement::getText() {
return doGetText();
}
const Common::String &DialogElement::doGetText() {
return _text;
}
// CONCRETE DIALOG ELEMENTS
DialogButton::DialogButton(Dialog *dialog, Common::String title, DialogAction action, Common::Point position, uint width, uint height):
DialogElement(dialog, title, action, position, width, height) {}
bool DialogButton::doProcessEvent(MacVenture::Dialog *dialog, Common::Event event) {
Common::Point mouse = event.mouse;
if (event.type == Common::EVENT_LBUTTONDOWN) {
dialog->localize(mouse);
if (_bounds.contains(mouse)) {
debugC(2, kMVDebugGUI, "Click! Button: %s", _text.c_str());
dialog->handleDialogAction(this, _action);
return true;
}
}
return false;
}
void DialogButton::doDraw(MacVenture::Dialog *dialog, Graphics::ManagedSurface &target) {
target.fillRect(_bounds, kColorWhite);
target.frameRect(_bounds, kColorBlack);
// Draw title
dialog->getFont().drawString(
&target, _text, _bounds.left, _bounds.top, _bounds.width(), kColorBlack, Graphics::kTextAlignCenter);
}
DialogPlainText::DialogPlainText(Dialog *dialog, Common::String content, Common::Point position) :
DialogElement(dialog, content, kDANone, position, 0, 0) { }
DialogPlainText::~DialogPlainText() {}
bool DialogPlainText::doProcessEvent(MacVenture::Dialog *dialog, Common::Event event) {
return false;
}
void DialogPlainText::doDraw(MacVenture::Dialog *dialog, Graphics::ManagedSurface &target) {
// Draw contents
dialog->getFont().drawString(
&target, _text, _bounds.left, _bounds.top, _bounds.width(), kColorBlack, Graphics::kTextAlignCenter);
}
DialogTextInput::DialogTextInput(Dialog *dialog, Common::Point position, uint width, uint height) :
DialogElement(dialog, "", kDANone, position, width, height) {}
DialogTextInput::~DialogTextInput() {}
bool DialogTextInput::doProcessEvent(Dialog *dialog, Common::Event event) {
if (event.type == Common::EVENT_KEYDOWN) {
switch (event.kbd.keycode) {
case Common::KEYCODE_BACKSPACE:
if (!_text.empty()) {
_text.deleteLastChar();
dialog->setUserInput(_text);
return true;
}
break;
default:
if (event.kbd.ascii >= 0x20 && event.kbd.ascii <= 0x7f) {
_text += (char)event.kbd.ascii;
dialog->setUserInput(_text);
return true;
}
break;
}
}
return false;
}
void DialogTextInput::doDraw(MacVenture::Dialog *dialog, Graphics::ManagedSurface &target) {
target.fillRect(_bounds, kColorWhite);
target.frameRect(_bounds, kColorBlack);
dialog->getFont().drawString(&target, _text, _bounds.left, _bounds.top, _bounds.width(), kColorBlack);
}
void Dialog::calculateBoundsFromPrebuilt(const PrebuiltDialogBounds &bounds) {
_bounds = Common::Rect(
bounds.left,
bounds.top,
bounds.right,
bounds.bottom);
}
} // End of namespace MacVenture