2015-12-26 13:12:30 +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.
|
|
|
|
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2015-12-28 12:38:26 +01:00
|
|
|
#include "common/system.h"
|
2016-01-03 21:59:57 +01:00
|
|
|
#include "common/timer.h"
|
2015-12-31 18:50:34 +01:00
|
|
|
#include "common/unzip.h"
|
2016-01-01 01:43:59 +01:00
|
|
|
#include "graphics/cursorman.h"
|
2015-12-31 18:50:34 +01:00
|
|
|
#include "graphics/fonts/bdf.h"
|
2016-01-01 01:43:59 +01:00
|
|
|
#include "graphics/palette.h"
|
2015-12-26 13:12:30 +01:00
|
|
|
#include "wage/wage.h"
|
2015-12-28 12:38:26 +01:00
|
|
|
#include "wage/design.h"
|
|
|
|
#include "wage/entities.h"
|
2015-12-26 13:12:30 +01:00
|
|
|
#include "wage/gui.h"
|
2016-01-02 02:39:47 +01:00
|
|
|
#include "wage/world.h"
|
2015-12-26 13:12:30 +01:00
|
|
|
|
|
|
|
namespace Wage {
|
|
|
|
|
2015-12-28 12:38:26 +01:00
|
|
|
enum {
|
2016-01-04 20:10:31 +01:00
|
|
|
kMenuHeight = 20,
|
2015-12-28 12:38:26 +01:00
|
|
|
kMenuPadding = 6,
|
2016-01-04 20:10:31 +01:00
|
|
|
kMenuItemHeight = 20,
|
2016-01-01 02:23:43 +01:00
|
|
|
kBorderWidth = 17,
|
2016-01-01 02:36:44 +01:00
|
|
|
kDesktopArc = 7,
|
2016-01-03 22:11:58 +01:00
|
|
|
kComponentsPadding = 10,
|
|
|
|
kCursorHeight = 12
|
2015-12-28 12:38:26 +01:00
|
|
|
};
|
|
|
|
|
2016-01-01 01:43:59 +01:00
|
|
|
static const byte palette[] = {
|
|
|
|
0, 0, 0, // Black
|
|
|
|
0x80, 0x80, 0x80, // Gray
|
|
|
|
0xff, 0xff, 0xff, // White
|
|
|
|
0x00, 0xff, 0x00 // Green
|
|
|
|
};
|
|
|
|
|
2016-01-01 02:23:43 +01:00
|
|
|
static byte checkersPattern[8] = { 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa };
|
|
|
|
static byte fillPattern[8] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
|
2016-01-01 01:43:59 +01:00
|
|
|
|
|
|
|
static const byte macCursorArrow[] = {
|
|
|
|
2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
2, 0, 2, 3, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
2, 0, 0, 2, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
2, 0, 0, 0, 2, 3, 3, 3, 3, 3, 3,
|
|
|
|
2, 0, 0, 0, 0, 2, 3, 3, 3, 3, 3,
|
|
|
|
2, 0, 0, 0, 0, 0, 2, 3, 3, 3, 3,
|
|
|
|
2, 0, 0, 0, 0, 0, 0, 2, 3, 3, 3,
|
|
|
|
2, 0, 0, 0, 0, 0, 0, 0, 2, 3, 3,
|
|
|
|
2, 0, 0, 0, 0, 0, 0, 0, 0, 2, 3,
|
|
|
|
2, 0, 0, 0, 0, 0, 2, 2, 2, 2, 2,
|
|
|
|
2, 0, 0, 2, 0, 0, 2, 3, 3, 3, 3,
|
|
|
|
2, 0, 2, 3, 2, 0, 0, 2, 3, 3, 3,
|
|
|
|
2, 2, 3, 3, 2, 0, 0, 2, 3, 3, 3,
|
|
|
|
2, 3, 3, 3, 3, 2, 0, 0, 2, 3, 3,
|
|
|
|
3, 3, 3, 3, 3, 2, 0, 0, 2, 3, 3,
|
|
|
|
3, 3, 3, 3, 3, 3, 2, 2, 2, 3, 3
|
|
|
|
};
|
|
|
|
|
|
|
|
static const byte macCursorBeam[] = {
|
|
|
|
0, 0, 3, 3, 3, 0, 0, 3, 3, 3, 3,
|
|
|
|
3, 3, 0, 3, 0, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 3, 0, 3, 3, 3, 3, 3, 3, 3,
|
|
|
|
3, 3, 0, 3, 0, 3, 3, 3, 3, 3, 3,
|
|
|
|
0, 0, 3, 3, 3, 0, 0, 3, 3, 3, 3,
|
|
|
|
};
|
2015-12-29 01:46:39 +01:00
|
|
|
|
2016-01-03 21:59:57 +01:00
|
|
|
static void cursor_timer_handler(void *refCon) {
|
|
|
|
Gui *gui = (Gui *)refCon;
|
|
|
|
|
|
|
|
int x = gui->_cursorX;
|
|
|
|
int y = gui->_cursorY;
|
|
|
|
|
|
|
|
if (x == 0 && y == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!gui->_screen.getPixels())
|
|
|
|
return;
|
|
|
|
|
|
|
|
x += gui->_consoleTextArea.left;
|
|
|
|
y += gui->_consoleTextArea.top;
|
|
|
|
|
2016-01-04 00:13:28 +01:00
|
|
|
gui->_screen.vLine(x, y, y + kCursorHeight, gui->_cursorState ? kColorBlack : kColorWhite);
|
2016-01-04 00:09:10 +01:00
|
|
|
|
|
|
|
if (!gui->_cursorOff)
|
|
|
|
gui->_cursorState = !gui->_cursorState;
|
2016-01-03 21:59:57 +01:00
|
|
|
|
2016-01-04 00:13:28 +01:00
|
|
|
g_system->copyRectToScreen(gui->_screen.getBasePtr(x, y), gui->_screen.pitch, x, y, 1, kCursorHeight);
|
2016-01-03 21:59:57 +01:00
|
|
|
g_system->updateScreen();
|
|
|
|
}
|
|
|
|
|
2016-01-02 02:39:47 +01:00
|
|
|
Gui::Gui(WageEngine *engine) {
|
|
|
|
_engine = engine;
|
2015-12-28 12:38:26 +01:00
|
|
|
_scene = NULL;
|
|
|
|
_sceneDirty = true;
|
2016-01-02 02:53:30 +01:00
|
|
|
_consoleDirty = true;
|
2016-01-01 18:48:16 +01:00
|
|
|
_bordersDirty = true;
|
2016-01-02 02:53:30 +01:00
|
|
|
_menuDirty = true;
|
2016-01-03 15:31:24 +01:00
|
|
|
_consoleFullRedraw = true;
|
2015-12-28 12:38:26 +01:00
|
|
|
_screen.create(g_system->getWidth(), g_system->getHeight(), Graphics::PixelFormat::createFormatCLUT8());
|
2015-12-29 01:46:39 +01:00
|
|
|
|
|
|
|
Patterns p;
|
2016-01-01 02:23:43 +01:00
|
|
|
p.push_back(checkersPattern);
|
2015-12-29 01:46:39 +01:00
|
|
|
|
2015-12-31 17:45:36 +01:00
|
|
|
_scrollPos = 0;
|
2016-01-03 15:31:24 +01:00
|
|
|
_consoleLineHeight = 8; // Dummy value which makes sense
|
|
|
|
_consoleNumLines = 24; // Dummy value
|
2015-12-31 18:50:34 +01:00
|
|
|
_builtInFonts = false;
|
2016-01-01 18:48:16 +01:00
|
|
|
_sceneIsActive = false;
|
2015-12-31 17:45:36 +01:00
|
|
|
|
2016-01-03 21:59:57 +01:00
|
|
|
_cursorX = 0;
|
|
|
|
_cursorY = 0;
|
|
|
|
_cursorState = false;
|
2016-01-04 00:09:10 +01:00
|
|
|
_cursorOff = false;
|
2016-01-03 21:59:57 +01:00
|
|
|
|
2016-01-01 01:43:59 +01:00
|
|
|
g_system->getPaletteManager()->setPalette(palette, 0, 4);
|
|
|
|
|
|
|
|
CursorMan.replaceCursorPalette(palette, 0, 4);
|
|
|
|
CursorMan.replaceCursor(macCursorArrow, 11, 16, 1, 1, 3);
|
|
|
|
_cursorIsArrow = true;
|
|
|
|
CursorMan.showMouse(true);
|
|
|
|
|
2016-01-01 02:23:43 +01:00
|
|
|
// Draw desktop
|
2016-01-01 02:16:00 +01:00
|
|
|
Common::Rect r(0, 0, _screen.w - 1, _screen.h - 1);
|
2016-01-01 02:23:43 +01:00
|
|
|
Design::drawFilledRoundRect(&_screen, r, kDesktopArc, kColorBlack, p, 1);
|
2016-01-03 22:11:58 +01:00
|
|
|
g_system->copyRectToScreen(_screen.getPixels(), _screen.pitch, 0, 0, _screen.w, _screen.h);
|
2015-12-31 18:50:34 +01:00
|
|
|
|
|
|
|
loadFonts();
|
2016-01-03 21:59:57 +01:00
|
|
|
|
2016-01-03 22:11:58 +01:00
|
|
|
g_system->getTimerManager()->installTimerProc(&cursor_timer_handler, 200000, this, "wageCursor");
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Gui::~Gui() {
|
2016-01-03 21:59:57 +01:00
|
|
|
g_system->getTimerManager()->removeTimerProc(&cursor_timer_handler);
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
|
|
|
|
2016-01-03 12:39:29 +01:00
|
|
|
const Graphics::Font *Gui::getFont(const char *name, Graphics::FontManager::FontUsage fallback) {
|
|
|
|
const Graphics::Font *font;
|
|
|
|
|
|
|
|
if (!_builtInFonts) {
|
|
|
|
font = FontMan.getFontByName(name);
|
|
|
|
|
|
|
|
if (!font)
|
|
|
|
warning("Cannot load font %s", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_builtInFonts || !font)
|
|
|
|
font = FontMan.getFontByUsage(fallback);
|
|
|
|
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Graphics::Font *Gui::getConsoleFont() {
|
|
|
|
char fontName[128];
|
|
|
|
Scene *scene = _engine->_world->_player->_currentScene;
|
|
|
|
|
|
|
|
snprintf(fontName, 128, "%s-%d", scene->getFontName(), scene->_fontSize);
|
|
|
|
|
|
|
|
return getFont(fontName, Graphics::FontManager::kConsoleFont);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Graphics::Font *Gui::getMenuFont() {
|
|
|
|
return getFont("Chicago-12", Graphics::FontManager::kBigGUIFont);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Graphics::Font *Gui::getTitleFont() {
|
|
|
|
return getFont("Chicago-12", Graphics::FontManager::kBigGUIFont);
|
|
|
|
}
|
|
|
|
|
2016-01-02 00:55:45 +01:00
|
|
|
void Gui::clearOutput() {
|
|
|
|
_out.clear();
|
|
|
|
}
|
|
|
|
|
2015-12-30 18:22:46 +01:00
|
|
|
void Gui::appendText(String &str) {
|
2016-01-02 02:53:30 +01:00
|
|
|
_consoleDirty = true;
|
|
|
|
|
2015-12-31 17:45:36 +01:00
|
|
|
if (!str.contains('\n')) {
|
|
|
|
_out.push_back(str);
|
2016-01-03 12:23:17 +01:00
|
|
|
flowText(str);
|
2015-12-31 17:45:36 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, we got new lines, need to split it
|
|
|
|
// and push substrings individually
|
|
|
|
Common::String tmp = "";
|
|
|
|
|
|
|
|
for (int i = 0; i < str.size(); i++) {
|
|
|
|
if (str[i] == '\n') {
|
|
|
|
_out.push_back(tmp);
|
2016-01-03 12:23:17 +01:00
|
|
|
flowText(tmp);
|
2015-12-31 17:45:36 +01:00
|
|
|
tmp = "";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp += str[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
_out.push_back(tmp);
|
2016-01-03 12:23:17 +01:00
|
|
|
flowText(tmp);
|
2015-12-30 18:22:46 +01:00
|
|
|
}
|
|
|
|
|
2015-12-28 12:38:26 +01:00
|
|
|
void Gui::draw() {
|
2016-01-02 02:39:47 +01:00
|
|
|
if (_scene != _engine->_world->_player->_currentScene || _sceneDirty) {
|
|
|
|
_scene = _engine->_world->_player->_currentScene;
|
2016-01-02 02:53:30 +01:00
|
|
|
_sceneDirty = true;
|
2016-01-02 02:39:47 +01:00
|
|
|
|
2016-01-01 02:36:44 +01:00
|
|
|
_scene->paint(&_screen, 0 + kComponentsPadding, kMenuHeight + kComponentsPadding);
|
2016-01-01 18:48:16 +01:00
|
|
|
|
|
|
|
_sceneArea.left = 0 + kComponentsPadding + kBorderWidth;
|
|
|
|
_sceneArea.top = kMenuHeight + kComponentsPadding + kBorderWidth;
|
|
|
|
_sceneArea.setWidth(_scene->_design->getBounds()->width() - 2 * kBorderWidth);
|
|
|
|
_sceneArea.setHeight(_scene->_design->getBounds()->height() - 2 * kBorderWidth);
|
2016-01-03 12:23:17 +01:00
|
|
|
|
|
|
|
int sceneW = _scene->_design->getBounds()->width();
|
|
|
|
int consoleW = _screen.w - sceneW - 2 * kComponentsPadding - 2 * kBorderWidth;
|
|
|
|
int consoleH = _scene->_design->getBounds()->height() - 2 * kBorderWidth;
|
|
|
|
int consoleX = sceneW + kComponentsPadding + kBorderWidth;
|
|
|
|
int consoleY = kMenuHeight + kComponentsPadding + kBorderWidth;
|
|
|
|
|
|
|
|
_consoleTextArea.left = consoleX;
|
|
|
|
_consoleTextArea.top = consoleY;
|
|
|
|
_consoleTextArea.right = consoleX + consoleW;
|
|
|
|
_consoleTextArea.bottom = consoleY + consoleH;
|
2015-12-28 12:38:26 +01:00
|
|
|
}
|
|
|
|
|
2016-01-02 02:53:30 +01:00
|
|
|
if (_scene && (_bordersDirty || _sceneDirty))
|
2016-01-03 12:23:17 +01:00
|
|
|
paintBorder(&_screen, _sceneArea, kWindowScene);
|
2016-01-02 02:53:30 +01:00
|
|
|
|
2015-12-30 18:22:46 +01:00
|
|
|
// Render console
|
2016-01-02 02:53:30 +01:00
|
|
|
if (_consoleDirty)
|
2016-01-03 12:23:17 +01:00
|
|
|
renderConsole(&_screen, _consoleTextArea);
|
2016-01-01 18:48:16 +01:00
|
|
|
|
2016-01-02 02:53:30 +01:00
|
|
|
if (_bordersDirty || _consoleDirty)
|
2016-01-03 12:23:17 +01:00
|
|
|
paintBorder(&_screen, _consoleTextArea, kWindowConsole);
|
2016-01-01 18:48:16 +01:00
|
|
|
|
2016-01-02 02:53:30 +01:00
|
|
|
if (_menuDirty)
|
|
|
|
renderMenu();
|
2015-12-30 01:58:42 +01:00
|
|
|
|
2016-01-02 02:53:30 +01:00
|
|
|
_sceneDirty = false;
|
|
|
|
_consoleDirty = false;
|
|
|
|
_bordersDirty = false;
|
|
|
|
_menuDirty = false;
|
2016-01-03 15:31:24 +01:00
|
|
|
_consoleFullRedraw = false;
|
2015-12-28 12:38:26 +01:00
|
|
|
}
|
|
|
|
|
2015-12-26 13:12:30 +01:00
|
|
|
void Gui::drawBox(Graphics::Surface *g, int x, int y, int w, int h) {
|
2015-12-26 13:25:33 +01:00
|
|
|
Common::Rect r(x, y, x + w + 1, y + h + 1);
|
2015-12-26 13:12:30 +01:00
|
|
|
|
2015-12-26 13:25:33 +01:00
|
|
|
g->fillRect(r, kColorWhite);
|
2015-12-26 13:12:30 +01:00
|
|
|
g->frameRect(r, kColorBlack);
|
|
|
|
}
|
|
|
|
|
2015-12-26 13:54:20 +01:00
|
|
|
void Gui::fillRect(Graphics::Surface *g, int x, int y, int w, int h) {
|
2015-12-30 01:46:51 +01:00
|
|
|
Common::Rect r(x, y, x + w, y + h);
|
2015-12-26 13:54:20 +01:00
|
|
|
|
|
|
|
g->fillRect(r, kColorBlack);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define ARROW_W 12
|
|
|
|
#define ARROW_H 6
|
|
|
|
const int arrowPixels[ARROW_H][ARROW_W] = {
|
|
|
|
{0,0,0,0,0,1,1,0,0,0,0,0},
|
|
|
|
{0,0,0,0,1,1,1,1,0,0,0,0},
|
|
|
|
{0,0,0,1,1,1,1,1,1,0,0,0},
|
|
|
|
{0,0,1,1,1,1,1,1,1,1,0,0},
|
|
|
|
{0,1,1,1,1,1,1,1,1,1,1,0},
|
|
|
|
{1,1,1,1,1,1,1,1,1,1,1,1}};
|
|
|
|
|
2016-01-03 12:23:17 +01:00
|
|
|
void Gui::paintBorder(Graphics::Surface *g, Common::Rect &r, WindowType windowType) {
|
2015-12-31 18:07:01 +01:00
|
|
|
bool active, scrollable, closeable, closeBoxPressed, drawTitle;
|
2016-01-03 12:23:17 +01:00
|
|
|
const int size = kBorderWidth;
|
|
|
|
int x = r.left - size;
|
|
|
|
int y = r.top - size;
|
|
|
|
int width = r.width() + 2 * size;
|
|
|
|
int height = r.height() + 2 * size;
|
2015-12-31 17:55:33 +01:00
|
|
|
|
|
|
|
switch (windowType) {
|
|
|
|
case kWindowScene:
|
2016-01-01 18:48:16 +01:00
|
|
|
active = _sceneIsActive;
|
2015-12-31 17:55:33 +01:00
|
|
|
scrollable = false;
|
2016-01-01 18:48:16 +01:00
|
|
|
closeable = _sceneIsActive;
|
2015-12-31 17:55:33 +01:00
|
|
|
closeBoxPressed = false;
|
2015-12-31 18:07:01 +01:00
|
|
|
drawTitle = true;
|
2015-12-31 17:55:33 +01:00
|
|
|
break;
|
|
|
|
case kWindowConsole:
|
2016-01-01 18:48:16 +01:00
|
|
|
active = !_sceneIsActive;
|
2015-12-31 17:55:33 +01:00
|
|
|
scrollable = true;
|
2016-01-01 18:48:16 +01:00
|
|
|
closeable = !_sceneIsActive;
|
2015-12-31 17:55:33 +01:00
|
|
|
closeBoxPressed = false;
|
2015-12-31 18:07:01 +01:00
|
|
|
drawTitle = false;
|
2015-12-31 17:55:33 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-12-26 13:12:30 +01:00
|
|
|
drawBox(g, x, y, size, size);
|
|
|
|
drawBox(g, x+width-size-1, y, size, size);
|
|
|
|
drawBox(g, x+width-size-1, y+height-size-1, size, size);
|
|
|
|
drawBox(g, x, y+height-size-1, size, size);
|
|
|
|
drawBox(g, x + size, y + 2, width - 2*size - 1, size - 4);
|
|
|
|
drawBox(g, x + size, y + height - size + 1, width - 2*size - 1, size - 4);
|
|
|
|
drawBox(g, x + 2, y + size, size - 4, height - 2*size - 1);
|
|
|
|
drawBox(g, x + width - size + 1, y + size, size - 4, height - 2*size-1);
|
|
|
|
|
|
|
|
if (active) {
|
2015-12-26 13:54:20 +01:00
|
|
|
fillRect(g, x + size, y + 5, width - 2*size - 1, 8);
|
|
|
|
fillRect(g, x + size, y + height - 13, width - 2*size - 1, 8);
|
|
|
|
fillRect(g, x + 5, y + size, 8, height - 2*size - 1);
|
2015-12-26 13:12:30 +01:00
|
|
|
if (!scrollable) {
|
2015-12-26 13:54:20 +01:00
|
|
|
fillRect(g, x + width - 13, y + size, 8, height - 2*size - 1);
|
2015-12-26 13:12:30 +01:00
|
|
|
} else {
|
|
|
|
int x1 = x + width - 15;
|
|
|
|
int y1 = y + size + 1;
|
2015-12-26 13:54:20 +01:00
|
|
|
for (int yy = 0; yy < ARROW_H; yy++) {
|
|
|
|
for (int xx = 0; xx < ARROW_W; xx++) {
|
|
|
|
if (arrowPixels[yy][xx] != 0) {
|
2015-12-30 01:46:51 +01:00
|
|
|
g->hLine(x1+xx, y1+yy, x1+xx, kColorBlack);
|
2015-12-27 00:51:18 +01:00
|
|
|
} else {
|
2015-12-30 01:46:51 +01:00
|
|
|
g->hLine(x1+xx, y1+yy, x1+xx, kColorWhite);
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-12-26 13:54:20 +01:00
|
|
|
fillRect(g, x + width - 13, y + size + ARROW_H, 8, height - 2*size - 1 - ARROW_H*2);
|
|
|
|
y1 += height - 2*size - ARROW_H - 2;
|
|
|
|
for (int yy = 0; yy < ARROW_H; yy++) {
|
|
|
|
for (int xx = 0; xx < ARROW_W; xx++) {
|
|
|
|
if (arrowPixels[ARROW_H-yy-1][xx] != 0) {
|
2015-12-30 01:46:51 +01:00
|
|
|
g->hLine(x1+xx, y1+yy, x1+xx, kColorBlack);
|
2015-12-27 00:51:18 +01:00
|
|
|
} else {
|
2015-12-30 01:46:51 +01:00
|
|
|
g->hLine(x1+xx, y1+yy, x1+xx, kColorWhite);
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (closeable) {
|
|
|
|
if (closeBoxPressed) {
|
2015-12-26 13:54:20 +01:00
|
|
|
fillRect(g, x + 6, y + 6, 6, 6);
|
2015-12-26 13:12:30 +01:00
|
|
|
} else {
|
|
|
|
drawBox(g, x + 5, y + 5, 7, 7);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-31 18:07:01 +01:00
|
|
|
if (drawTitle) {
|
2016-01-03 12:39:29 +01:00
|
|
|
const Graphics::Font *font = getTitleFont();
|
|
|
|
int yOff = _builtInFonts ? 3 : 1;
|
2015-12-31 18:07:01 +01:00
|
|
|
|
|
|
|
int w = font->getStringWidth(_scene->_name) + 6;
|
2015-12-26 13:12:30 +01:00
|
|
|
int maxWidth = width - size*2 - 7;
|
2015-12-31 18:07:01 +01:00
|
|
|
if (w > maxWidth)
|
2015-12-26 13:12:30 +01:00
|
|
|
w = maxWidth;
|
|
|
|
drawBox(g, x + (width - w) / 2, y, w, size);
|
2015-12-31 19:01:26 +01:00
|
|
|
font->drawString(g, _scene->_name, x + (width - w) / 2 + 3, y + yOff, w, kColorBlack);
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
2016-01-03 22:11:58 +01:00
|
|
|
|
|
|
|
g_system->copyRectToScreen(g->getBasePtr(x, y), g->pitch, x, y, width, height);
|
2015-12-26 13:12:30 +01:00
|
|
|
}
|
|
|
|
|
2015-12-31 17:45:36 +01:00
|
|
|
enum {
|
|
|
|
kConWOverlap = 20,
|
|
|
|
kConHOverlap = 20,
|
2016-01-01 02:58:34 +01:00
|
|
|
kConWPadding = 3,
|
2016-01-01 02:49:23 +01:00
|
|
|
kConHPadding = 4,
|
2016-01-03 15:31:24 +01:00
|
|
|
kConOverscan = 3
|
2015-12-31 17:45:36 +01:00
|
|
|
};
|
|
|
|
|
2016-01-03 12:23:17 +01:00
|
|
|
void Gui::flowText(String &str) {
|
|
|
|
Common::StringArray wrappedLines;
|
|
|
|
int textW = _consoleTextArea.width() - kConWPadding * 2;
|
2016-01-03 12:39:29 +01:00
|
|
|
const Graphics::Font *font = getConsoleFont();
|
2016-01-03 12:23:17 +01:00
|
|
|
|
|
|
|
font->wordWrapText(str, textW, wrappedLines);
|
|
|
|
|
|
|
|
if (wrappedLines.size() == 0) // Sometimes we have empty lines
|
|
|
|
_lines.push_back("");
|
|
|
|
|
|
|
|
for (Common::StringArray::const_iterator j = wrappedLines.begin(); j != wrappedLines.end(); ++j)
|
|
|
|
_lines.push_back(*j);
|
2016-01-03 15:31:24 +01:00
|
|
|
|
|
|
|
int pos = _scrollPos;
|
|
|
|
_scrollPos = MAX<int>(0, (_lines.size() - _consoleNumLines) * _consoleLineHeight);
|
|
|
|
|
2016-01-04 11:17:51 +01:00
|
|
|
_cursorX = kConWPadding;
|
2016-01-03 21:59:57 +01:00
|
|
|
|
|
|
|
if (_scrollPos)
|
2016-01-04 11:17:51 +01:00
|
|
|
_cursorY = (_consoleNumLines) * _consoleLineHeight + kConHPadding;
|
2016-01-03 21:59:57 +01:00
|
|
|
else
|
2016-01-04 11:11:27 +01:00
|
|
|
_cursorY = (_lines.size()) * _consoleLineHeight + kConHPadding;
|
2016-01-03 21:59:57 +01:00
|
|
|
|
2016-01-03 15:31:24 +01:00
|
|
|
if (pos != _scrollPos)
|
|
|
|
_consoleFullRedraw = true;
|
2016-01-03 15:34:26 +01:00
|
|
|
|
|
|
|
if (!_engine->_temporarilyHidden)
|
|
|
|
draw();
|
2016-01-03 12:23:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Gui::renderConsole(Graphics::Surface *g, Common::Rect &r) {
|
2016-01-03 15:31:24 +01:00
|
|
|
bool fullRedraw = _consoleFullRedraw;
|
2015-12-31 17:45:36 +01:00
|
|
|
bool textReflow = false;
|
2016-01-03 12:23:17 +01:00
|
|
|
int surfW = r.width() + kConWOverlap * 2;
|
|
|
|
int surfH = r.height() + kConHOverlap * 2;
|
2015-12-31 17:45:36 +01:00
|
|
|
|
2015-12-31 17:55:33 +01:00
|
|
|
Common::Rect boundsR(kConWOverlap - kConOverscan, kConHOverlap - kConOverscan,
|
2016-01-03 12:23:17 +01:00
|
|
|
r.width() + kConWOverlap + kConOverscan, r.height() + kConHOverlap + kConOverscan);
|
2015-12-31 17:45:36 +01:00
|
|
|
Common::Rect fullR(0, 0, surfW, surfH);
|
|
|
|
|
|
|
|
if (_console.w != surfW || _console.h != surfH) {
|
|
|
|
if (_console.w != surfW)
|
|
|
|
textReflow = true;
|
2015-12-30 18:41:18 +01:00
|
|
|
|
|
|
|
_console.free();
|
|
|
|
|
2015-12-31 17:45:36 +01:00
|
|
|
_console.create(surfW, surfH, Graphics::PixelFormat::createFormatCLUT8());
|
2015-12-30 18:41:18 +01:00
|
|
|
fullRedraw = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fullRedraw)
|
|
|
|
_console.fillRect(fullR, kColorWhite);
|
|
|
|
|
2016-01-03 12:39:29 +01:00
|
|
|
const Graphics::Font *font = getConsoleFont();
|
2015-12-31 19:01:26 +01:00
|
|
|
|
2016-01-03 15:31:24 +01:00
|
|
|
_consoleLineHeight = font->getFontHeight();
|
2016-01-03 12:23:17 +01:00
|
|
|
int textW = r.width() - kConWPadding * 2;
|
|
|
|
int textH = r.height() - kConHPadding * 2;
|
2015-12-31 17:45:36 +01:00
|
|
|
|
|
|
|
if (textReflow) {
|
|
|
|
_lines.clear();
|
|
|
|
|
2016-01-03 12:23:17 +01:00
|
|
|
for (int i = 0; i < _out.size(); i++)
|
|
|
|
flowText(_out[i]);
|
2015-12-31 17:45:36 +01:00
|
|
|
}
|
|
|
|
|
2016-01-03 15:31:24 +01:00
|
|
|
const int firstLine = _scrollPos / _consoleLineHeight;
|
|
|
|
const int lastLine = MIN((_scrollPos + textH) / _consoleLineHeight + 1, _lines.size());
|
2015-12-31 17:45:36 +01:00
|
|
|
const int xOff = kConWOverlap;
|
|
|
|
const int yOff = kConHOverlap;
|
|
|
|
int x1 = xOff + kConWPadding;
|
2016-01-03 15:31:24 +01:00
|
|
|
int y1 = yOff - (_scrollPos % _consoleLineHeight) + kConHPadding;
|
|
|
|
|
|
|
|
if (fullRedraw)
|
2016-01-04 11:11:27 +01:00
|
|
|
_consoleNumLines = (r.height() - 2 * kConWPadding) / _consoleLineHeight - 2;
|
2015-12-31 17:45:36 +01:00
|
|
|
|
|
|
|
for (int line = firstLine; line < lastLine; line++) {
|
|
|
|
const char *str = _lines[line].c_str();
|
|
|
|
|
|
|
|
if (*str)
|
|
|
|
font->drawString(&_console, _lines[line], x1, y1, textW, kColorBlack);
|
|
|
|
|
2016-01-03 15:31:24 +01:00
|
|
|
y1 += _consoleLineHeight;
|
2015-12-31 17:45:36 +01:00
|
|
|
}
|
|
|
|
|
2016-01-03 12:23:17 +01:00
|
|
|
g->copyRectToSurface(_console, r.left - kConOverscan, r.top - kConOverscan, boundsR);
|
2016-01-03 22:11:58 +01:00
|
|
|
g_system->copyRectToScreen(g->getBasePtr(r.left, r.top), g->pitch, r.left, r.top, r.width(), r.height());
|
2015-12-30 18:41:18 +01:00
|
|
|
}
|
|
|
|
|
2016-01-04 00:09:10 +01:00
|
|
|
void Gui::drawInput() {
|
|
|
|
if (!_screen.getPixels())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const Graphics::Font *font = getConsoleFont();
|
|
|
|
|
2016-01-04 11:11:27 +01:00
|
|
|
int x = kConWPadding + _consoleTextArea.left;
|
2016-01-04 00:09:10 +01:00
|
|
|
int y = _cursorY + _consoleTextArea.top;
|
|
|
|
Common::String text(_engine->_inputText);
|
|
|
|
int textW = font->getStringWidth(text);
|
|
|
|
|
|
|
|
// undraw cursor
|
|
|
|
_cursorOff = true;
|
|
|
|
_cursorState = false;
|
|
|
|
cursor_timer_handler(this);
|
|
|
|
_cursorOff = false;
|
|
|
|
|
|
|
|
Common::Rect r(x, y, x + textW + 10, y + font->getFontHeight());
|
|
|
|
|
|
|
|
_screen.fillRect(r, kColorWhite);
|
|
|
|
|
|
|
|
font->drawString(&_screen, text, x, y, _screen.w, kColorBlack);
|
|
|
|
|
|
|
|
g_system->copyRectToScreen(_screen.getBasePtr(x, y), _screen.pitch, x, y, textW + 10, font->getFontHeight());
|
|
|
|
|
2016-01-04 00:13:28 +01:00
|
|
|
_cursorX = font->getStringWidth(_engine->_inputText) + kConHPadding;
|
2016-01-04 00:09:10 +01:00
|
|
|
}
|
|
|
|
|
2015-12-31 18:50:34 +01:00
|
|
|
void Gui::loadFonts() {
|
|
|
|
Common::Archive *dat;
|
|
|
|
|
|
|
|
dat = Common::makeZipArchive("wage.dat");
|
|
|
|
|
|
|
|
if (!dat) {
|
|
|
|
warning("Could not find wage.dat. Falling back to built-in fonts");
|
|
|
|
_builtInFonts = true;
|
2016-01-04 00:47:16 +01:00
|
|
|
|
|
|
|
return;
|
2015-12-31 18:50:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::ArchiveMemberList list;
|
|
|
|
dat->listMembers(list);
|
|
|
|
|
|
|
|
for (Common::ArchiveMemberList::iterator it = list.begin(); it != list.end(); ++it) {
|
|
|
|
Common::SeekableReadStream *stream = dat->createReadStreamForMember((*it)->getName());
|
|
|
|
|
|
|
|
Graphics::BdfFont *font = Graphics::BdfFont::loadFont(*stream);
|
|
|
|
|
|
|
|
delete stream;
|
|
|
|
|
|
|
|
Common::String fontName = (*it)->getName();
|
|
|
|
|
|
|
|
// Trim the .bdf extension
|
|
|
|
for (int i = fontName.size() - 1; i >= 0; --i) {
|
|
|
|
if (fontName[i] == '.') {
|
|
|
|
while ((uint)i < fontName.size()) {
|
|
|
|
fontName.deleteLastChar();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FontMan.assignFontToName(fontName, font);
|
|
|
|
|
|
|
|
debug(2, " %s", fontName.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
_builtInFonts = false;
|
|
|
|
|
|
|
|
delete dat;
|
|
|
|
}
|
2015-12-26 13:12:30 +01:00
|
|
|
|
2016-01-01 01:43:59 +01:00
|
|
|
void Gui::mouseMove(int x, int y) {
|
|
|
|
if (_consoleTextArea.contains(x, y)) {
|
|
|
|
if (_cursorIsArrow) {
|
|
|
|
CursorMan.replaceCursor(macCursorBeam, 11, 16, 3, 8, 3);
|
|
|
|
_cursorIsArrow = false;
|
|
|
|
}
|
|
|
|
} else if (_cursorIsArrow == false) {
|
|
|
|
CursorMan.replaceCursor(macCursorArrow, 11, 16, 1, 1, 3);
|
|
|
|
_cursorIsArrow = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-01 13:09:39 +01:00
|
|
|
static const char *menuItems[] = {
|
|
|
|
"\xf0", "File", "Edit", "Commands", "Weapons", 0
|
|
|
|
};
|
|
|
|
|
2016-01-01 02:23:43 +01:00
|
|
|
void Gui::renderMenu() {
|
|
|
|
Common::Rect r(0, 0, _screen.w - 1, kMenuHeight - 1);
|
|
|
|
Patterns p;
|
|
|
|
p.push_back(fillPattern);
|
|
|
|
|
|
|
|
Design::drawFilledRoundRect(&_screen, r, kDesktopArc, kColorWhite, p, 1);
|
|
|
|
r.top = 7;
|
|
|
|
Design::drawFilledRect(&_screen, r, kColorWhite, p, 1);
|
|
|
|
r.top = kMenuHeight - 1;
|
|
|
|
Design::drawFilledRect(&_screen, r, kColorBlack, p, 1);
|
2016-01-01 13:09:39 +01:00
|
|
|
|
2016-01-03 12:39:29 +01:00
|
|
|
const Graphics::Font *font = getMenuFont();
|
2016-01-04 20:10:31 +01:00
|
|
|
int y = _builtInFonts ? 3 : 2;
|
2016-01-01 13:09:39 +01:00
|
|
|
int x = 18;
|
|
|
|
|
|
|
|
for (int i = 0; menuItems[i]; i++) {
|
|
|
|
const char *s = menuItems[i];
|
|
|
|
|
|
|
|
if (i == 0 && _builtInFonts)
|
|
|
|
s = "\xa9"; // (c) Symbol as the most resembling apple
|
|
|
|
|
|
|
|
int w = font->getStringWidth(s);
|
|
|
|
font->drawString(&_screen, s, x, y, w, kColorBlack);
|
|
|
|
|
|
|
|
x += w + 13;
|
|
|
|
}
|
2016-01-03 22:11:58 +01:00
|
|
|
|
|
|
|
g_system->copyRectToScreen(_screen.getPixels(), _screen.pitch, 0, 0, _screen.w, kMenuHeight);
|
2016-01-01 02:23:43 +01:00
|
|
|
}
|
|
|
|
|
2016-01-01 18:48:16 +01:00
|
|
|
Designed *Gui::getClickTarget(int x, int y) {
|
|
|
|
if (_sceneArea.contains(x, y)) {
|
|
|
|
if (!_sceneIsActive) {
|
|
|
|
_sceneIsActive = true;
|
|
|
|
_bordersDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Common::List<Obj *>::const_iterator it = _scene->_objs.begin(); it != _scene->_objs.end(); ++it) {
|
|
|
|
if ((*it)->_design->isPointOpaque(x - _sceneArea.left + kBorderWidth, y - _sceneArea.top + kBorderWidth))
|
|
|
|
return *it;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Common::List<Chr *>::const_iterator it = _scene->_chrs.begin(); it != _scene->_chrs.end(); ++it) {
|
|
|
|
if ((*it)->_design->isPointOpaque(x - _sceneArea.left + kBorderWidth, y - _sceneArea.top + kBorderWidth))
|
|
|
|
return *it;
|
|
|
|
}
|
|
|
|
} else if (_consoleTextArea.contains(x, y)) {
|
|
|
|
if (_sceneIsActive) {
|
|
|
|
_sceneIsActive = false;
|
|
|
|
_bordersDirty = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-12-26 13:12:30 +01:00
|
|
|
} // End of namespace Wage
|