/* 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. * * $URL$ * $Id$ * */ #ifdef ENABLE_LOL #include "kyra/screen_lol.h" #include "kyra/lol.h" #include "kyra/resource.h" namespace Kyra { Screen_LoL::Screen_LoL(LoLEngine *vm, OSystem *system) : Screen_v2(vm, system), _vm(vm) { _paletteOverlay1 = new uint8[0x100]; _paletteOverlay2 = new uint8[0x100]; _grayOverlay = new uint8[0x100]; memset(_paletteOverlay1, 0, 0x100); memset(_paletteOverlay2, 0, 0x100); memset(_grayOverlay, 0, 0x100); for (int i = 0; i < 8; i++) _levelOverlays[i] = new uint8[256]; _fadeFlag = 2; _curDimIndex = 0; _internDimX = _internDimY = _internDimW = _internDimH = _internDimDstX = _internBlockWidth = _internDimDstY = _internBlockHeight = _internDimU5 = _internDimU6 = _internBlockWidth2 = _internDimU8 = 0; } Screen_LoL::~Screen_LoL() { for (int i = 0; i < _screenDimTableCount; i++) delete _customDimTable[i]; delete[] _customDimTable; for (int i = 0; i < 8; i++) delete[] _levelOverlays[i]; delete[] _paletteOverlay1; delete[] _paletteOverlay2; delete[] _grayOverlay; } bool Screen_LoL::init() { if (Screen::init()) { _screenDimTable = _use16ColorMode ? _screenDimTable16C : _screenDimTable256C; _customDimTable = new ScreenDim*[_screenDimTableCount]; memset(_customDimTable, 0, sizeof(ScreenDim*) * _screenDimTableCount); return true; } return false; } void Screen_LoL::setScreenDim(int dim) { assert(dim < _screenDimTableCount); _curDim = _customDimTable[dim] ? (const ScreenDim *)_customDimTable[dim] : &_screenDimTable[dim]; _curDimIndex = dim; } const ScreenDim *Screen_LoL::getScreenDim(int dim) { assert(dim < _screenDimTableCount); return _customDimTable[dim] ? (const ScreenDim *)_customDimTable[dim] : &_screenDimTable[dim]; } void Screen_LoL::modifyScreenDim(int dim, int x, int y, int w, int h) { delete _customDimTable[dim]; _customDimTable[dim] = new ScreenDim; memcpy(_customDimTable[dim], &_screenDimTable[dim], sizeof(ScreenDim)); _customDimTable[dim]->sx = x; _customDimTable[dim]->sy = y; _customDimTable[dim]->w = w; _customDimTable[dim]->h = h; setScreenDim(dim); } void Screen_LoL::fprintString(const char *format, int x, int y, uint8 col1, uint8 col2, uint16 flags, ...) { if (!format) return; char string[240]; va_list vaList; va_start(vaList, flags); vsnprintf(string, sizeof(string), format, vaList); va_end(vaList); if (flags & 1) x -= getTextWidth(string) >> 1; if (flags & 2) x -= getTextWidth(string); if (flags & 4) { printText(string, x - 1, y, 1, col2); printText(string, x, y + 1, 1, col2); } if (flags & 8) { printText(string, x - 1, y, 227, col2); printText(string, x, y + 1, 227, col2); } printText(string, x, y, col1, col2); } void Screen_LoL::fprintStringIntro(const char *format, int x, int y, uint8 c1, uint8 c2, uint8 c3, uint16 flags, ...) { char buffer[400]; va_list args; va_start(args, flags); vsnprintf(buffer, sizeof(buffer), format, args); va_end(args); if ((flags & 0x0F00) == 0x100) x -= getTextWidth(buffer) >> 1; if ((flags & 0x0F00) == 0x200) x -= getTextWidth(buffer); if ((flags & 0x00F0) == 0x20) { printText(buffer, x-1, y, c3, c2); printText(buffer, x, y+1, c3, c2); } printText(buffer, x, y, c1, c2); } void Screen_LoL::generateGrayOverlay(const uint8 *srcPal, uint8 *grayOverlay, int factor, int addR, int addG, int addB, int lastColor, bool skipSpecialColors) { uint8 tmpPal[768]; for (int i = 0; i != lastColor; i++) { int v = (((srcPal[3 * i] & 0x3f) * factor) / 0x40) + addR; tmpPal[3 * i] = (v > 0x3f) ? 0x3f : v & 0xff; v = (((srcPal[3 * i + 1] & 0x3f) * factor) / 0x40) + addG; tmpPal[3 * i + 1] = (v > 0x3f) ? 0x3f : v & 0xff; v = (((srcPal[3 * i + 2] & 0x3f) * factor) / 0x40) + addB; tmpPal[3 * i + 2] = (v > 0x3f) ? 0x3f : v & 0xff; } for (int i = 0; i < lastColor; i++) grayOverlay[i] = findLeastDifferentColor(tmpPal + 3 * i, srcPal, lastColor, skipSpecialColors); } uint8 *Screen_LoL::generateLevelOverlay(const uint8 *srcPal, uint8 *ovl, int opColor, int weight) { if (!srcPal || !ovl) return ovl; if (weight > 255) weight = 255; uint16 r = srcPal[opColor * 3]; uint16 g = srcPal[opColor * 3 + 1]; uint16 b = srcPal[opColor * 3 + 2]; uint8 *d = ovl; *d++ = 0; for (int i = 1; i != 255; i++) { uint16 a = srcPal[i * 3]; uint8 dr = a - ((((a - r) * (weight >> 1)) << 1) >> 8); a = srcPal[i * 3 + 1]; uint8 dg = a - ((((a - g) * (weight >> 1)) << 1) >> 8); a = srcPal[i * 3 + 2]; uint8 db = a - ((((a - b) * (weight >> 1)) << 1) >> 8); int l = opColor; int m = 0x7fff; int ii = 127; int x = 1; const uint8 *s = srcPal + 3; do { if (i == x) { s += 3; } else { int t = *s++ - dr; int c = t * t; t = *s++ - dg; c += (t * t); t = *s++ - db; c += (t * t); if (!c) { l = x; break; } if (c <= m) { m = c; l = x; } } x++; } while (--ii); *d++ = l & 0xff; } return ovl; } void Screen_LoL::drawGridBox(int x, int y, int w, int h, int col) { if (w <= 0 || x >= 320 || h <= 0 || y >= 200) return; if (x < 0) { x += w; if (x <= 0) return; w = x; x = 0; } int tmp = x + w; if (tmp >= 320) { w = 320 - x; } int pitch = 320 - w; if (y < 0) { y += h; if (y <= 0) return; h = y; y = 0; } tmp = y + h; if (tmp >= 200) { h = 200 - y; } tmp = (y + x) & 1; uint8 *p = getPagePtr(_curPage) + y * 320 + x; uint8 s = (tmp >> 8) & 1; w >>= 1; int w2 = w; while (h--) { if (w) { while (w--) { *(p + tmp) = col; p += 2; } } else { w = 1; } if (s == 1) { if (tmp == 0) *p = col; p++; } tmp ^= 1; p += pitch; w = w2; } } void Screen_LoL::fadeClearSceneWindow(int delay) { if (_fadeFlag == 1) return; uint8 *tpal = new uint8[768]; memcpy(tpal, _currentPalette, 768); memset(tpal, 0, 384); loadSpecialColors(tpal); fadePalette(tpal, delay); fillRect(112, 0, 288, 120, 0); delete[] tpal; _fadeFlag = 1; } void Screen_LoL::backupSceneWindow(int srcPageNum, int dstPageNum) { uint8 *src = getPagePtr(srcPageNum) + 112; uint8 *dst = getPagePtr(dstPageNum) + 0xa500; for (int h = 0; h < 120; h++) { for (int w = 0; w < 176; w++) *dst++ = *src++; src += 144; } } void Screen_LoL::restoreSceneWindow(int srcPageNum, int dstPageNum) { uint8 *src = getPagePtr(srcPageNum) + 0xa500; uint8 *dst = getPagePtr(dstPageNum) + 112; for (int h = 0; h < 120; h++) { memcpy(dst, src, 176); src += 176; dst += 320; } if (!dstPageNum) addDirtyRect(112, 0, 176, 120); } void Screen_LoL::smoothScrollZoomStepTop(int srcPageNum, int dstPageNum, int x, int y) { uint8 *src = getPagePtr(srcPageNum) + 0xa500 + y * 176 + x; uint8 *dst = getPagePtr(dstPageNum) + 0xa500; x <<= 1; uint16 width = 176 - x; uint16 scaleX = (((x + 1) << 8) / width + 0x100); uint16 cntW = scaleX >> 8; scaleX <<= 8; width--; uint16 widthCnt = width; uint16 height = 46 - y; uint16 scaleY = (((y + 1) << 8) / height + 0x100); scaleY <<= 8; uint32 scaleYc = 0; while (height) { uint32 scaleXc = 0; do { scaleXc += scaleX; int numbytes = cntW + (scaleXc >> 16); scaleXc &= 0xffff; memset(dst, *src++, numbytes); dst += numbytes; } while (--widthCnt); *dst++ = *src++; widthCnt = width; src += x; scaleYc += scaleY; if (scaleYc >> 16) { scaleYc = 0; src -= 176; continue; } height--; } } void Screen_LoL::smoothScrollZoomStepBottom(int srcPageNum, int dstPageNum, int x, int y) { uint8 *src = getPagePtr(srcPageNum) + 0xc4a0 + x; uint8 *dst = getPagePtr(dstPageNum) + 0xc4a0; x <<= 1; uint16 width = 176 - x; uint16 scaleX = (((x + 1) << 8) / width + 0x100); uint16 cntW = scaleX >> 8; scaleX <<= 8; width--; uint16 widthCnt = width; uint16 height = 74 - y; uint16 scaleY = (((y + 1) << 8) / height + 0x100); scaleY <<= 8; uint32 scaleYc = 0; while (height) { uint32 scaleXc = 0; do { scaleXc += scaleX; int numbytes = cntW + (scaleXc >> 16); scaleXc &= 0xffff; memset(dst, *src++, numbytes); dst += numbytes; } while (--widthCnt); *dst++ = *src++; widthCnt = width; src += x; scaleYc += scaleY; if (scaleYc >> 16) { scaleYc = 0; src -= 176; continue; } height--; } } void Screen_LoL::smoothScrollHorizontalStep(int pageNum, int srcX, int dstX, int w) { uint8 *d = getPagePtr(pageNum); uint8 *s = d + 112 + srcX; int w2 = srcX + w - dstX; int pitchS = 320 + w2 - (w << 1); int pitchD = 320 - w; int h = 120; while (h--) { for (int i = 0; i < w; i++) *d++ = *s++; d -= w; s -= w2; for (int i = 0; i < w; i++) *s++ = *d++; s += pitchS; d += pitchD; } } void Screen_LoL::smoothScrollTurnStep1(int srcPage1Num, int srcPage2Num, int dstPageNum) { uint8 *s = getPagePtr(srcPage1Num) + 273; uint8 *d = getPagePtr(dstPageNum) + 0xa500; for (int i = 0; i < 120; i++) { uint8 a = *s++; *d++ = a; *d++ = a; for (int ii = 0; ii < 14; ii++) { a = *s++; *d++ = a; *d++ = a; *d++ = a; } s += 305; d += 132; } s = getPagePtr(srcPage2Num) + 112; d = getPagePtr(dstPageNum) + 0xa52c; for (int i = 0; i < 120; i++) { for (int ii = 0; ii < 33; ii++) { *d++ = *s++; *d++ = *s++; uint8 a = *s++; *d++ = a; *d++ = a; } s += 221; d += 44; } } void Screen_LoL::smoothScrollTurnStep2(int srcPage1Num, int srcPage2Num, int dstPageNum) { uint8 *s = getPagePtr(srcPage1Num) + 244; uint8 *d = getPagePtr(dstPageNum) + 0xa500; for (int k = 0; k < 2; k++) { for (int i = 0; i < 120; i++) { for (int ii = 0; ii < 44; ii++) { uint8 a = *s++; *d++ = a; *d++ = a; } s += 276; d += 88; } s = getPagePtr(srcPage2Num) + 112; d = getPagePtr(dstPageNum) + 0xa558; } } void Screen_LoL::smoothScrollTurnStep3(int srcPage1Num, int srcPage2Num, int dstPageNum) { uint8 *s = getPagePtr(srcPage1Num) + 189; uint8 *d = getPagePtr(dstPageNum) + 0xa500; for (int i = 0; i < 120; i++) { for (int ii = 0; ii < 33; ii++) { *d++ = *s++; *d++ = *s++; uint8 a = *s++; *d++ = a; *d++ = a; } s += 221; d += 44; } s = getPagePtr(srcPage2Num) + 112; d = getPagePtr(dstPageNum) + 0xa584; for (int i = 0; i < 120; i++) { for (int ii = 0; ii < 14; ii++) { uint8 a = *s++; *d++ = a; *d++ = a; *d++ = a; } uint8 a = *s++; *d++ = a; *d++ = a; s += 305; d += 132; } } void Screen_LoL::copyRegionSpecial(int page1, int w1, int h1, int x1, int y1, int page2, int w2, int h2, int x2, int y2, int w3, int h3, int mode, ...) { if (!w3 || !h3) return; uint8 *table1 = 0; uint8 *table2 = 0; if (mode == 2) { va_list args; va_start(args, mode); table1 = va_arg(args, uint8*); table2 = va_arg(args, uint8*); va_end(args); } _internDimX = _internDimY = 0; _internDimW = w1; _internDimH = h1; calcBoundariesIntern(x1, y1, w3, h3); if (_internBlockWidth == -1) return; int iu5_1 = _internDimU5; int iu6_1 = _internDimU6; int ibw_1 = _internBlockWidth; int ibh_1 = _internBlockHeight; int dx_1 = _internDimDstX; int dy_1 = _internDimDstY; _internDimX = _internDimY = 0; _internDimW = w2; _internDimH = h2; calcBoundariesIntern(x2, y2, ibw_1, ibh_1); if (_internBlockWidth == -1) return; int iu5_2 = _internDimU5; int iu6_2 = _internDimU6; int ibw_2 = _internBlockWidth; int ibh_2 = _internBlockHeight; int dx_2 = _internDimDstX; int dy_2 = _internDimDstY; uint8 *src = getPagePtr(page1) + (dy_1 + iu6_2) * w1; uint8 *dst = getPagePtr(page2) + (dy_2 + iu6_1) * w2; for (int i = 0; i < ibh_2; i++) { uint8 *s = src + iu5_2 + dx_1; uint8 *d = dst + iu5_1 + dx_2; if (mode == 0) { memcpy(d, s, ibw_2); } else if (mode == 1) { if (!(i & 1)) { s++; d++; } for (int ii = (i & 1) ^ 1; ii < ibw_2; ii += 2 ) { *d = *s; d += 2; s += 2; } } else if (mode == 2) { for (int ii = 0; ii < ibw_2; ii++) { uint8 cmd = *s++; uint8 offs = table1[cmd]; if (!(offs & 0x80)) cmd = table2[(offs << 8) | *d]; *d++ = cmd; } } else if (mode == 3) { s = s - iu5_2 + ibw_1; s = s - iu5_2 - 1; for (int ii = 0; ii < ibw_2; ii++) *d++ = *s--; } dst += w2; src += w1; } if (!page2) addDirtyRect(_internDimDstX + _internDimX, _internDimDstY + _internDimY, _internBlockWidth, _internBlockHeight); } void Screen_LoL::copyBlockAndApplyOverlay(int page1, int x1, int y1, int page2, int x2, int y2, int w, int h, int dim, uint8 *ovl) { if (!w || !h || !ovl) return; const ScreenDim *cdim = getScreenDim(dim); _internDimX = cdim->sx << 3; _internDimY = cdim->sy; _internDimW = cdim->w << 3; _internDimH = cdim->h; calcBoundariesIntern(x2, y2, w, h); if (_internBlockWidth == -1) return; uint8 *src = getPagePtr(page1) + y1 * 320 + x1; uint8 *dst = getPagePtr(page2) + (_internDimDstY + _internDimY) * 320; for (int i = 0; i < _internBlockHeight; i++) { uint8 *s = src + _internDimU5; uint8 *d = dst + (_internDimDstX + _internDimX); for (int ii = 0; ii < _internBlockWidth; ii++) { uint8 p = ovl[*s++]; if (p) *d = p; d++; } dst += 320; src += 320; } if (!page2) addDirtyRect(_internDimDstX + _internDimX, _internDimDstY + _internDimY, _internBlockWidth, _internBlockHeight); } void Screen_LoL::applyOverlaySpecial(int page1, int x1, int y1, int page2, int x2, int y2, int w, int h, int dim, int flag, uint8 *ovl) { if (!w || !h || !ovl) return; const ScreenDim *cdim = getScreenDim(dim); _internDimX = cdim->sx << 3; _internDimY = cdim->sy; _internDimW = cdim->w << 3; _internDimH = cdim->h; calcBoundariesIntern(x2, y2, w, h); if (_internBlockWidth == -1) return; uint8 *src = getPagePtr(page1) + y1 * 320 + x1; uint8 *dst = getPagePtr(page2) + (_internDimDstY + _internDimY) * 320; for (int i = 0; i < _internBlockHeight; i++) { uint8 *s = src + _internDimU5; uint8 *d = dst + (_internDimDstX + _internDimX); if (flag) d += (i >> 1); for (int ii = 0; ii < _internBlockWidth; ii++) { if (*s++) *d = ovl[*d]; d++; } dst += 320; src += 320; } if (!page2) addDirtyRect(_internDimDstX + _internDimX, _internDimDstY + _internDimY, _internBlockWidth, _internBlockHeight); } void Screen_LoL::calcBoundariesIntern(int dstX, int dstY, int width, int height) { _internBlockWidth = _internBlockWidth2 = width; _internBlockHeight = height; _internDimDstX = dstX; _internDimDstY = dstY; _internDimU5 = _internDimU6 = _internDimU8 = 0; int t = _internDimDstX + _internBlockWidth; if (t <= 0) { _internBlockWidth = _internBlockHeight = -1; return; } if (t <= _internDimDstX) { _internDimU5 = _internBlockWidth - t; _internBlockWidth = t; _internDimDstX = 0; } t = _internDimW - _internDimDstX; if (t <= 0) { _internBlockWidth = _internBlockHeight = -1; return; } if (t <= _internBlockWidth) _internBlockWidth = t; _internBlockWidth2 -= _internBlockWidth; t = _internDimDstY + _internBlockHeight; if (t <= 0) { _internBlockWidth = _internBlockHeight = -1; return; } if (t <= _internDimDstY) { _internDimU6 = _internBlockHeight - t; _internBlockHeight = t; _internDimDstY = 0; } t = _internDimH - _internDimDstY; if (t <= 0) { _internBlockWidth = _internBlockHeight = -1; return; } if (t <= _internBlockHeight) _internBlockHeight = t; } void Screen_LoL::fadeToBlack(int delay, const UpdateFunctor *upFunc) { Screen::fadeToBlack(delay, upFunc); _fadeFlag = 2; } void Screen_LoL::fadeToPalette1(int delay) { loadSpecialColors(_palettes[0]); fadePalette(_palettes[0], delay); _fadeFlag = 0; } void Screen_LoL::loadSpecialColors(uint8 *destPalette) { memcpy(destPalette + 0x240, _screenPalette + 0x240, 12); } void Screen_LoL::copyColor(int dstColorIndex, int srcColorIndex) { uint8 *s = _screenPalette + srcColorIndex * 3; uint8 *d = _screenPalette + dstColorIndex * 3; memcpy(d, s, 3); uint8 ci[4]; ci[0] = (d[0] << 2) | (d[0] & 3); ci[1] = (d[1] << 2) | (d[1] & 3); ci[2] = (d[2] << 2) | (d[2] & 3); ci[3] = 0; _system->setPalette(ci, dstColorIndex, 1); } bool Screen_LoL::fadeColor(int dstColorIndex, int srcColorIndex, uint32 elapsedTime, uint32 targetTime) { uint8 *dst = _screenPalette + 3 * dstColorIndex; uint8 *src = _screenPalette + 3 * srcColorIndex; uint8 *p = getPalette(1) + 3 * dstColorIndex; bool res = false; int16 t1 = 0; int16 t2 = 0; int32 t3 = 0; uint8 tmpPalEntry[3]; for (int i = 0; i < 3; i++) { if (elapsedTime < targetTime) { t1 = *src & 0x3f; t2 = *dst & 0x3f; t3 = t1 - t2; if (t3) res = true; t3 = (((t3 << 8) / (int)targetTime) * (int)elapsedTime) >> 8; t3 = t2 + t3; } else { t1 = *dst & 0x3f; *p = t3 = t1; res = false; } tmpPalEntry[i] = t3 & 0xff; src++; dst++; p++; } uint8 tpal[768]; memcpy(tpal, _screenPalette, 768); memcpy(tpal + dstColorIndex * 3, tmpPalEntry, 3); setScreenPalette(tpal); updateScreen(); return res; } bool Screen_LoL::fadePalSpecial(uint8 *pal1, uint8 *pal2, uint32 elapsedTime, uint32 targetTime) { uint8 tpal[768]; uint8 *p1 = _palettes[1]; bool res = false; for (int i = 0; i < 768; i++) { uint8 out = 0; if (elapsedTime < targetTime) { int d = (pal2[i] & 0x3f) - (pal1[i] & 0x3f); if (d) res = true; int val = ((((d << 8) / targetTime) * elapsedTime) >> 8) & 0xff; out = ((pal1[i] & 0x3f) + val) & 0xff; } else { out = p1[i] = (pal2[i] & 0x3f); res = false; } tpal[i] = out; } setScreenPalette(tpal); updateScreen(); return res; } uint8 Screen_LoL::getShapePaletteSize(const uint8 *shp) { return shp[10]; } } // end of namespace Kyra #endif // ENABLE_LOL