scummvm/graphics/macgui/macwindowmanager.cpp

608 lines
16 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.
*/
#include "common/array.h"
#include "common/events.h"
#include "common/list.h"
#include "common/system.h"
#include "common/timer.h"
#include "graphics/cursorman.h"
#include "graphics/managed_surface.h"
#include "graphics/palette.h"
#include "graphics/primitives.h"
#include "graphics/macgui/macwindowmanager.h"
#include "graphics/macgui/macfontmanager.h"
#include "graphics/macgui/macwindow.h"
#include "graphics/macgui/mactextwindow.h"
#include "graphics/macgui/macmenu.h"
namespace Graphics {
static const byte palette[] = {
0, 0, 0, // Black
0x80, 0x80, 0x80, // Gray
0xff, 0xff, 0xff, // White
0x00, 0xff, 0x00, // Green
0x00, 0xcf, 0x00 // Green2
};
static byte fillPatterns[][8] = { { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }, // kPatternSolid
{ 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55 }, // kPatternStripes
{ 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55 }, // kPatternCheckers
{ 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa }, // kPatternCheckers2
{ 0x88, 0x22, 0x88, 0x22, 0x88, 0x22, 0x88, 0x22 }, // kPatternLightGray
{ 0x77, 0xdd, 0x77, 0xdd, 0x77, 0xdd, 0x77, 0xdd } // kPatternDarkGray
};
static const byte cursorPalette[] = {
0, 0, 0,
0xff, 0xff, 0xff
};
static const byte macCursorArrow[] = {
1, 1, 3, 3, 3, 3, 3, 3, 3, 3, 3,
1, 0, 1, 3, 3, 3, 3, 3, 3, 3, 3,
1, 0, 0, 1, 3, 3, 3, 3, 3, 3, 3,
1, 0, 0, 0, 1, 3, 3, 3, 3, 3, 3,
1, 0, 0, 0, 0, 1, 3, 3, 3, 3, 3,
1, 0, 0, 0, 0, 0, 1, 3, 3, 3, 3,
1, 0, 0, 0, 0, 0, 0, 1, 3, 3, 3,
1, 0, 0, 0, 0, 0, 0, 0, 1, 3, 3,
1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 3,
1, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1,
1, 0, 0, 1, 0, 0, 1, 3, 3, 3, 3,
1, 0, 1, 3, 1, 0, 0, 1, 3, 3, 3,
1, 1, 3, 3, 1, 0, 0, 1, 3, 3, 3,
1, 3, 3, 3, 3, 1, 0, 0, 1, 3, 3,
3, 3, 3, 3, 3, 1, 0, 0, 1, 3, 3,
3, 3, 3, 3, 3, 3, 1, 1, 1, 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,
};
static const byte macCursorCrossHair[] = {
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,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
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, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
};
static const byte macCursorWatch[] = {
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 0, 1, 1, 1, 1, 1, 1, 0, 1, 3,
0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 3,
0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 3,
0, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0,
0, 1, 1, 0, 0, 0, 1, 1, 1, 0, 0,
0, 1, 1, 1, 1, 0, 1, 1, 1, 0, 3,
0, 1, 1, 1, 1, 0, 1, 1, 1, 0, 3,
3, 0, 1, 1, 1, 1, 1, 1, 0, 1, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 0, 0, 0, 0, 0, 0, 3, 3, 3,
};
static const byte macCursorCrossBar[] = {
3, 3, 3, 0, 0, 0, 0, 3, 3, 3, 3,
3, 3, 3, 0, 1, 1, 0, 0, 3, 3, 3,
3, 3, 3, 0, 1, 1, 0, 0, 3, 3, 3,
3, 3, 3, 0, 1, 1, 0, 0, 3, 3, 3,
0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 3,
0, 1, 1, 1, 0, 0, 1, 1, 1, 0, 0,
0, 1, 1, 1, 0, 0, 1, 1, 1, 0, 0,
0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0,
3, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0,
3, 3, 3, 0, 1, 1, 0, 0, 3, 3, 3,
3, 3, 3, 0, 1, 1, 0, 0, 3, 3, 3,
3, 3, 3, 0, 0, 0, 0, 0, 3, 3, 3,
3, 3, 3, 3, 0, 0, 0, 0, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
};
static void menuTimerHandler(void *refCon);
MacWindowManager::MacWindowManager(uint32 mode) {
_screen = 0;
_screenCopy = nullptr;
_lastId = 0;
_activeWindow = -1;
_needsRemoval = false;
_mode = mode;
_menu = 0;
_menuDelay = 0;
_menuTimerActive = false;
_engineP = nullptr;
_engineR = nullptr;
_pauseEngineCallback = nullptr;
_redrawEngineCallback = nullptr;
_colorBlack = 0;
_colorWhite = 2;
_fullRefresh = true;
_palette = nullptr;
_paletteSize = 0;
for (int i = 0; i < ARRAYSIZE(fillPatterns); i++)
_patterns.push_back(fillPatterns[i]);
g_system->getPaletteManager()->setPalette(palette, 0, ARRAYSIZE(palette) / 3);
_fontMan = new MacFontManager(mode);
CursorMan.replaceCursorPalette(palette, 0, ARRAYSIZE(palette) / 3);
CursorMan.replaceCursor(macCursorArrow, 11, 16, 1, 1, 3);
_cursorIsArrow = true;
CursorMan.showMouse(true);
}
MacWindowManager::~MacWindowManager() {
for (Common::HashMap<uint, BaseMacWindow *>::iterator it = _windows.begin(); it != _windows.end(); it++)
delete it->_value;
delete _fontMan;
delete _screenCopy;
g_system->getTimerManager()->removeTimerProc(&menuTimerHandler);
}
void MacWindowManager::setMode(uint32 mode) {
_mode = mode;
if (mode & kWMModeForceBuiltinFonts)
_fontMan->forceBuiltinFonts();
}
MacWindow *MacWindowManager::addWindow(bool scrollable, bool resizable, bool editable) {
MacWindow *w = new MacWindow(_lastId, scrollable, resizable, editable, this);
addWindowInitialized(w);
setActive(getNextId());
return w;
}
MacTextWindow *MacWindowManager::addTextWindow(const MacFont *font, int fgcolor, int bgcolor, int maxWidth, TextAlign textAlignment, MacMenu *menu, bool cursorHandler) {
MacTextWindow *w = new MacTextWindow(this, font, fgcolor, bgcolor, maxWidth, textAlignment, menu, cursorHandler);
addWindowInitialized(w);
setActive(getNextId());
return w;
}
void MacWindowManager::addWindowInitialized(MacWindow *macwindow) {
_windows[macwindow->getId()] = macwindow;
_windowStack.push_back(macwindow);
}
MacMenu *MacWindowManager::addMenu() {
delete _menu;
_menu = new MacMenu(getNextId(), _screen->getBounds(), this);
_windows[_menu->getId()] = _menu;
return _menu;
}
void MacWindowManager::activateMenu() {
if (!_menu)
return;
if (_mode & kWMModalMenuMode) {
if (!_screenCopy)
_screenCopy = new ManagedSurface(*_screen); // Create a copy
else
*_screenCopy = *_screen;
pauseEngine(true);
}
_menu->setVisible(true);
}
bool MacWindowManager::isMenuActive() {
if (!_menu)
return false;
return _menu->isVisible();
}
void MacWindowManager::setActive(int id) {
if (_activeWindow == id)
return;
if (_activeWindow != -1)
_windows[_activeWindow]->setActive(false);
_activeWindow = id;
_windows[id]->setActive(true);
_windowStack.remove(_windows[id]);
_windowStack.push_back(_windows[id]);
_fullRefresh = true;
}
void MacWindowManager::removeWindow(MacWindow *target) {
_windowsToRemove.push_back(target);
_needsRemoval = true;
}
void macDrawPixel(int x, int y, int color, void *data) {
MacPlotData *p = (MacPlotData *)data;
if (p->fillType > p->patterns->size() || !p->fillType)
return;
byte *pat = p->patterns->operator[](p->fillType - 1);
if (p->thickness == 1) {
if (x >= 0 && x < p->surface->w && y >= 0 && y < p->surface->h) {
uint xu = (uint)x; // for letting compiler optimize it
uint yu = (uint)y;
*((byte *)p->surface->getBasePtr(xu, yu)) =
(pat[(yu - p->fillOriginY) % 8] & (1 << (7 - (xu - p->fillOriginX) % 8))) ?
color : p->bgColor;
}
} else {
int x1 = x;
int x2 = x1 + p->thickness;
int y1 = y;
int y2 = y1 + p->thickness;
for (y = y1; y < y2; y++)
for (x = x1; x < x2; x++)
if (x >= 0 && x < p->surface->w && y >= 0 && y < p->surface->h) {
uint xu = (uint)x; // for letting compiler optimize it
uint yu = (uint)y;
*((byte *)p->surface->getBasePtr(xu, yu)) =
(pat[(yu - p->fillOriginY) % 8] & (1 << (7 - (xu - p->fillOriginX) % 8))) ?
color : p->bgColor;
}
}
}
void MacWindowManager::drawDesktop() {
Common::Rect r(_screen->getBounds());
MacPlotData pd(_screen, &_patterns, kPatternCheckers, 0, 0, 1, _colorWhite);
Graphics::drawRoundRect(r, kDesktopArc, _colorBlack, true, macDrawPixel, &pd);
g_system->copyRectToScreen(_screen->getPixels(), _screen->pitch, 0, 0, _screen->w, _screen->h);
}
void MacWindowManager::draw() {
assert(_screen);
removeMarked();
if (_fullRefresh) {
if (!(_mode & kWMModeNoDesktop))
drawDesktop();
if (_redrawEngineCallback != nullptr)
_redrawEngineCallback(_engineR);
}
for (Common::List<BaseMacWindow *>::const_iterator it = _windowStack.begin(); it != _windowStack.end(); it++) {
BaseMacWindow *w = *it;
if (w->draw(_screen, _fullRefresh)) {
w->setDirty(false);
Common::Rect clip(w->getDimensions().left - 2, w->getDimensions().top - 2, w->getDimensions().right - 2, w->getDimensions().bottom - 2);
clip.clip(_screen->getBounds());
clip.clip(Common::Rect(0, 0, g_system->getWidth() - 1, g_system->getHeight() - 1));
if (!clip.isEmpty())
g_system->copyRectToScreen(_screen->getBasePtr(clip.left, clip.top), _screen->pitch, clip.left, clip.top, clip.width(), clip.height());
}
}
// Menu is drawn on top of everything and always
if (_menu)
_menu->draw(_screen, _fullRefresh);
_fullRefresh = false;
}
static void menuTimerHandler(void *refCon) {
MacWindowManager *wm = (MacWindowManager *)refCon;
if (wm->_menuHotzone.contains(wm->_lastMousePos)) {
wm->activateMenu();
}
wm->_menuTimerActive = false;
g_system->getTimerManager()->removeTimerProc(&menuTimerHandler);
}
bool MacWindowManager::processEvent(Common::Event &event) {
if (event.type == Common::EVENT_MOUSEMOVE)
_lastMousePos = event.mouse;
if (_menu && !_menu->isVisible()) {
if ((_mode & kWMModeAutohideMenu) && event.type == Common::EVENT_MOUSEMOVE) {
if (!_menuTimerActive && _menuHotzone.contains(event.mouse)) {
_menuTimerActive = true;
g_system->getTimerManager()->installTimerProc(&menuTimerHandler, _menuDelay, this, "menuWindowCursor");
}
}
}
// Menu gets events first for shortcuts and menu bar
if (_menu && _menu->processEvent(event))
return true;
if (_activeWindow != -1) {
if (_windows[_activeWindow]->isEditable() && _windows[_activeWindow]->getType() == kWindowWindow &&
((MacWindow *)_windows[_activeWindow])->getInnerDimensions().contains(event.mouse.x, event.mouse.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;
}
}
}
for (Common::List<BaseMacWindow *>::const_iterator it = _windowStack.end(); it != _windowStack.begin();) {
it--;
BaseMacWindow *w = *it;
if (w->hasAllFocus() || (w->isEditable() && event.type == Common::EVENT_KEYDOWN) ||
w->getDimensions().contains(event.mouse.x, event.mouse.y)) {
if (event.type == Common::EVENT_LBUTTONDOWN || event.type == Common::EVENT_LBUTTONUP)
setActive(w->getId());
return w->processEvent(event);
}
}
return false;
}
void MacWindowManager::removeMarked() {
if (!_needsRemoval) return;
Common::List<BaseMacWindow *>::const_iterator it;
for (it = _windowsToRemove.begin(); it != _windowsToRemove.end(); it++) {
removeFromStack(*it);
removeFromWindowList(*it);
delete *it;
_activeWindow = 0;
_fullRefresh = true;
}
_windowsToRemove.clear();
_needsRemoval = false;
// Do we need compact lastid?
_lastId = 0;
for (Common::HashMap<uint, BaseMacWindow *>::iterator lit = _windows.begin(); lit != _windows.end(); lit++) {
if (lit->_key >= (uint)_lastId)
_lastId = lit->_key + 1;
}
}
void MacWindowManager::removeFromStack(BaseMacWindow *target) {
Common::List<BaseMacWindow *>::iterator stackIt;
for (stackIt = _windowStack.begin(); stackIt != _windowStack.end(); stackIt++) {
if (*stackIt == target) {
stackIt = _windowStack.erase(stackIt);
stackIt--;
}
}
}
void MacWindowManager::removeFromWindowList(BaseMacWindow *target) {
// _windows.erase(target->getId()); // Is applicable?
for (Common::HashMap<uint, BaseMacWindow *>::iterator it = _windows.begin(); it != _windows.end(); it++) {
if (it->_value == target) {
_windows.erase(it);
break;
}
}
}
/////////////////
// Cursor stuff
/////////////////
void MacWindowManager::pushArrowCursor() {
CursorMan.pushCursor(macCursorArrow, 11, 16, 1, 1, 3);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::pushBeamCursor() {
CursorMan.pushCursor(macCursorBeam, 11, 16, 1, 1, 3);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::pushCrossHairCursor() {
CursorMan.pushCursor(macCursorCrossHair, 11, 16, 1, 1, 3);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::pushCrossBarCursor() {
CursorMan.pushCursor(macCursorCrossBar, 11, 16, 1, 1, 3);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::pushWatchCursor() {
CursorMan.pushCursor(macCursorWatch, 11, 16, 1, 1, 3);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::pushCustomCursor(byte *data, int w, int h, int transcolor) {
CursorMan.pushCursor(data, w, h, 1, 1, transcolor);
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
void MacWindowManager::popCursor() {
CursorMan.popCursor();
CursorMan.pushCursorPalette(cursorPalette, 0, 2);
}
///////////////////
// Palette stuff
///////////////////
void MacWindowManager::passPalette(const byte *pal, uint size) {
const byte *p = pal;
if (_palette)
free(_palette);
_palette = (byte *)malloc(size * 3);
_paletteSize = size;
_colorWhite = -1;
_colorBlack = -1;
// Search pure white and black colors
for (uint i = 0; i < size; i++) {
if (_colorWhite == -1 && p[0] == 0xff && p[1] == 0xff && p[2] == 0xff)
_colorWhite = i;
if (_colorBlack == -1 && p[0] == 0x00 && p[1] == 0x00 && p[2] == 0x00)
_colorBlack = i;
_palette[i * 3 + 0] = *p++;
_palette[i * 3 + 1] = *p++;
_palette[i * 3 + 2] = *p++;
}
if (_colorWhite != -1 && _colorBlack != -1)
return;
// We did not find some color. Let's find closest approximations
float darkest = 1000.0f, brightest = -1.0f;
int di = -1, bi = -1;
p = pal;
for (uint i = 0; i < size; i++) {
float gray = p[0] * 0.3f + p[1] * 0.59f + p[2] * 0.11f;
if (darkest > gray) {
darkest = gray;
di = i;
}
if (brightest < gray) {
brightest = gray;
bi = i;
}
p += 3;
}
_colorWhite = bi;
_colorBlack = di;
}
uint MacWindowManager::findBestColor(byte cr, byte cg, byte cb) {
uint bestColor = 0;
double min = 0xFFFFFFFF;
for (uint i = 0; i < _paletteSize; ++i) {
int rmean = (*(_palette + 3 * i + 0) + cr) / 2;
int r = *(_palette + 3 * i + 0) - cr;
int g = *(_palette + 3 * i + 1) - cg;
int b = *(_palette + 3 * i + 2) - cb;
double dist = sqrt((((512 + rmean) * r * r) >> 8) + 4 * g * g + (((767 - rmean) * b * b) >> 8));
if (min > dist) {
bestColor = i;
min = dist;
}
}
return bestColor;
}
void MacWindowManager::pauseEngine(bool pause) {
if (_engineP && _pauseEngineCallback) {
_pauseEngineCallback(_engineP, pause);
} else {
warning("MacWindowManager::pauseEngine(): no pauseEngineCallback is set");
}
}
void MacWindowManager::setEnginePauseCallback(void *engine, void (*pauseCallback)(void *, bool)) {
_engineP = engine;
_pauseEngineCallback = pauseCallback;
}
void MacWindowManager::setEngineRedrawCallback(void *engine, void (*redrawCallback)(void *)) {
_engineR = engine;
_redrawEngineCallback = redrawCallback;
}
} // End of namespace Graphics