/* 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