scummvm/engines/agos/window.cpp
2021-12-26 18:48:43 +01:00

367 lines
8.8 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/system.h"
#include "common/textconsole.h"
#include "graphics/surface.h"
#include "agos/agos.h"
#include "agos/intern.h"
namespace AGOS {
uint AGOSEngine::getWindowNum(WindowBlock *window) {
uint i;
for (i = 0; i != ARRAYSIZE(_windowArray); i++)
if (_windowArray[i] == window)
return i;
error("getWindowNum: not found");
return 0; // for compilers that don't support NORETURN
}
WindowBlock *AGOSEngine::openWindow(uint x, uint y, uint w, uint h, uint flags, uint fillColor, uint textColor) {
WindowBlock *window;
window = _windowList;
while (window->mode != 0)
window++;
if (getGameType() == GType_ELVIRA1 && y >= 133)
textColor += 16;
window->mode = 2;
window->x = x;
window->y = y;
window->width = w;
window->height = h;
window->flags = flags;
window->fillColor = fillColor;
window->textColor = textColor;
window->textColumn = 0;
window->textColumnOffset = 0;
window->textRow = 0;
window->scrollY = 0;
// Characters are 6 pixels (except Japanese: when downscaled, 1-byte characters are 4 pixels, 2-byte characters are 8 pixels)
if (getGameType() == GType_ELVIRA2)
window->textMaxLength = (window->width * 8 - 4) / 6;
else if (getGameType() == GType_PN)
window->textMaxLength = window->width * 8 / 6 + 1;
else if (getGameType() == GType_ELVIRA1 && getPlatform() == Common::kPlatformPC98)
window->textMaxLength = window->width << 1;
else
window->textMaxLength = window->width * 8 / 6;
if (getGameType() == GType_PN || getGameType() == GType_ELVIRA1 || getGameType() == GType_ELVIRA2 || getGameType() == GType_WW)
clearWindow(window);
if (getGameType() == GType_SIMON1 && getPlatform() == Common::kPlatformAmiga && window->fillColor == 225)
window->fillColor = (getFeatures() & GF_32COLOR) ? 17 : 241;
return window;
}
void AGOSEngine::changeWindow(uint a) {
a &= 7;
if (_windowArray[a] == nullptr || _curWindow == a)
return;
_curWindow = a;
justifyOutPut(0);
_textWindow = _windowArray[a];
justifyStart();
}
void AGOSEngine::closeWindow(uint a) {
if (_windowArray[a] == nullptr)
return;
removeIconArray(a);
resetWindow(_windowArray[a]);
_windowArray[a] = nullptr;
if (_curWindow == a) {
_textWindow = nullptr;
changeWindow(0);
}
}
void AGOSEngine::clearWindow(WindowBlock *window) {
clearHiResTextLayer();
if (window->flags & 0x10)
restoreWindow(window);
else
colorWindow(window);
window->textColumn = 0;
window->textRow = 0;
window->textColumnOffset = (getGameType() == GType_ELVIRA2) ? 4 : 0;
window->textLength = 0;
window->scrollY = 0;
}
#ifdef ENABLE_AGOS2
void AGOSEngine_Feeble::colorWindow(WindowBlock *window) {
byte *dst;
uint16 h, w;
_videoLockOut |= 0x8000;
dst = getBackGround() + _backGroundBuf->pitch * window->y + window->x;
for (h = 0; h < window->height; h++) {
for (w = 0; w < window->width; w++) {
if (dst[w] == 113 || dst[w] == 116 || dst[w] == 252)
dst[w] = window->fillColor;
}
dst += _backGroundBuf->pitch;
}
_videoLockOut &= ~0x8000;
}
#endif
void AGOSEngine::colorWindow(WindowBlock *window) {
uint16 y, h;
y = window->y;
h = window->height * 8;
if (getGameType() == GType_ELVIRA2 && window->y == 146) {
if (window->fillColor == 1) {
_displayPalette[33 * 3 + 0] = 48 * 4;
_displayPalette[33 * 3 + 1] = 40 * 4;
_displayPalette[33 * 3 + 2] = 32 * 4;
} else {
_displayPalette[33 * 3 + 0] = 56 * 4;
_displayPalette[33 * 3 + 1] = 56 * 4;
_displayPalette[33 * 3 + 2] = 40 * 4;
}
y--;
h += 2;
_paletteFlag = 1;
}
colorBlock(window, window->x * 8, y, window->width * 8, h);
}
void AGOSEngine::colorBlock(WindowBlock *window, uint16 x, uint16 y, uint16 w, uint16 h) {
_videoLockOut |= 0x8000;
Graphics::Surface *screen = getBackendSurface();
byte *dst = (byte *)screen->getBasePtr(x, y);
uint8 color = window->fillColor;
if (getGameType() == GType_ELVIRA2 || getGameType() == GType_WW)
color += dst[0] & 0xF0;
uint16 h2 = h;
do {
memset(dst, color, w);
dst += screen->pitch;
} while (--h);
Common::Rect dirtyRect(x, y, x + w, y + h2);
updateBackendSurface(&dirtyRect);
_videoLockOut &= ~0x8000;
}
void AGOSEngine::resetWindow(WindowBlock *window) {
if (window->flags & 8)
restoreWindow(window);
window->mode = 0;
}
void AGOSEngine::restoreWindow(WindowBlock *window) {
_videoLockOut |= 0x8000;
if (getGameType() == GType_FF || getGameType() == GType_PP) {
restoreBlock(window->x, window->y, window->x + window->width, window->y + window->height);
} else if (getGameType() == GType_SIMON2) {
if (_restoreWindow6 && _windowArray[2] == window) {
window = _windowArray[6];
_restoreWindow6 = 0;
}
restoreBlock(window->x * 8, window->y, (window->x + window->width) * 8, window->y + window->height * 8);
} else if (getGameType() == GType_SIMON1) {
restoreBlock(window->x * 8, window->y, (window->x + window->width) * 8, window->y + window->height * 8 + ((window == _windowArray[2]) ? 1 : 0));
} else {
uint16 x = window->x;
uint16 w = window->width;
if (getGameType() == GType_ELVIRA1) {
// Adjustments to remove inventory arrows
if (x & 1) {
x--;
w++;
}
if (w & 1) {
w++;
}
}
restoreBlock(x * 8, window->y, (x + w) * 8, window->y + window->height * 8);
}
_videoLockOut &= ~0x8000;
}
void AGOSEngine::restoreBlock(uint16 left, uint16 top, uint16 right, uint16 bottom) {
byte *dst, *src;
uint i;
Graphics::Surface *screen = getBackendSurface();
dst = (byte *)screen->getPixels();
src = getBackGround();
dst += top * screen->pitch;
src += top * _backGroundBuf->pitch;
uint8 paletteMod = 0;
Common::Rect dirtyRect(left, top, right, bottom);
if (getGameType() == GType_ELVIRA1 && !(getFeatures() & GF_DEMO) && top >= 133)
paletteMod = 16;
while (top < bottom) {
for (i = left; i < right; i++)
dst[i] = src[i] + paletteMod;
top++;
dst += screen->pitch;
src += _backGroundBuf->pitch;
}
updateBackendSurface(&dirtyRect);
}
void AGOSEngine::setTextColor(uint color) {
WindowBlock *window = _windowArray[_curWindow];
if ((getFeatures() & GF_32COLOR) && color != 0) {
if (window->fillColor == 17)
color = 25;
else
color = 220;
}
window->textColor = color;
}
void AGOSEngine::sendWindow(uint a) {
if (getGameType() == GType_PN || _textWindow != _windowArray[0]) {
if (getGameType() == GType_ELVIRA1 || getGameType() == GType_ELVIRA2 || getGameType() == GType_WW) {
if (!(_textWindow->flags & 1)) {
haltAnimation();
}
}
windowPutChar(_textWindow, a);
}
}
void AGOSEngine::waitWindow(WindowBlock *window) {
HitArea *ha;
const char *message;
window->textColumn = (window->width / 2) - 3;
window->textRow = window->height - 1;
window->textLength = 0;
_forceAscii = true;
message = "[ OK ]";
for (; *message; message++)
windowPutChar(window, *message);
_forceAscii = false;
ha = findEmptyHitArea();
ha->x = (window->width / 2 + window->x - 3) * 8;
ha->y = window->height * 8 + window->y - 8;
ha->width = 48;
ha->height = 8;
ha->flags = kBFBoxInUse;
ha->id = 0x7FFF;
ha->priority = 999;
while (!shouldQuit()) {
_lastHitArea = nullptr;
_lastHitArea3 = nullptr;
while (!shouldQuit()) {
if (_lastHitArea3 != nullptr)
break;
delay(1);
}
ha = _lastHitArea;
if (ha == nullptr) {
} else if (ha->id == 0x7FFF) {
break;
}
}
undefineBox(0x7FFF);
}
void AGOSEngine::writeChar(WindowBlock *window, int x, int y, int offs, int val) {
int chr;
_forceAscii = true;
// Clear background of first digit
window->textColumnOffset = offs;
window->textColor = 0;
windowDrawChar(window, x * 8, y, 129);
if (val != -1) {
// Print first digit
chr = val / 10 + 48;
window->textColor = 15;
windowDrawChar(window, x * 8, y, chr);
}
offs += 6;
if (offs >= 7) {
offs -= 8;
x++;
}
// Clear background of second digit
window->textColumnOffset = offs;
window->textColor = 0;
windowDrawChar(window, x * 8, y, 129);
if (val != -1) {
// Print second digit
chr = val % 10 + 48;
window->textColor = 15;
windowDrawChar(window, x * 8, y, chr);
}
_forceAscii = false;
}
} // End of namespace AGOS