/* 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/lol.h" #include "kyra/screen_lol.h" #include "kyra/gui_lol.h" #include "kyra/resource.h" #include "kyra/util.h" #include "common/savefile.h" #include "common/config-manager.h" #include "graphics/scaler.h" #include "base/version.h" namespace Kyra { void LoLEngine::gui_drawPlayField() { _screen->loadBitmap("PLAYFLD.CPS", 3, 3, 0); if (_flagsTable[31] & 0x40) { // copy compass shape static const int cx[] = { 112, 152, 224 }; _screen->copyRegion(cx[_flags.isTalkie ? _lang : 0], 32, 288, 0, 32, 32, 2, 2, Screen::CR_NO_P_CHECK); _compassDirection = -1; } if (_flagsTable[31] & 0x10) // draw automap book _screen->drawShape(2, _gameShapes[_flags.isTalkie ? 78 : 76], 290, 32, 0, 0); int cp = _screen->setCurPage(2); if (_flagsTable[31] & 0x20) { gui_drawScroll(); } else { _selectedSpell = 0; } if (_flagsTable[31] & 0x08) resetLampStatus(); updateDrawPage2(); gui_drawScene(2); gui_drawAllCharPortraitsWithStats(); gui_drawInventory(); gui_drawMoneyBox(_screen->_curPage); _screen->setCurPage(cp); _screen->copyPage(2, 0); updateDrawPage2(); } void LoLEngine::gui_drawScene(int pageNum) { if (!(_updateFlags & 1) && _weaponsDisabled == false && _partyAwake && _vcnBlocks) drawScene(pageNum); } void LoLEngine::gui_drawInventory() { if (!_currentControlMode || !_needSceneRestore) { for (int i = 0; i < 9; i++) gui_drawInventoryItem(i); } } void LoLEngine::gui_drawInventoryItem(int index) { static const uint16 inventoryXpos[] = { 0x6A, 0x7F, 0x94, 0xA9, 0xBE, 0xD3, 0xE8, 0xFD, 0x112 }; int x = inventoryXpos[index]; int item = _inventoryCurItem + index; if (item > 47) item -= 48; int flag = item & 1 ? 0 : 1; _screen->drawShape(_screen->_curPage, _gameShapes[4], x, 179, 0, flag); if (_inventory[item]) _screen->drawShape(_screen->_curPage, getItemIconShapePtr(_inventory[item]), x + 1, 180, 0, 0); } void LoLEngine::gui_drawScroll() { _screen->copyRegion(112, 0, 12, 0, 87, 15, 2, 2, Screen::CR_NO_P_CHECK); Screen::FontId of = _screen->setFont(Screen::FID_9_FNT); int h = 0; for (int i = 0; i < 7; i++) { if (_availableSpells[i] != -1) h += 9; } if (h == 18) h = 27; if (h) { _screen->copyRegion(201, 1, 17, 15, 6, h, 2, 2, Screen::CR_NO_P_CHECK); _screen->copyRegion(208, 1, 89, 15, 6, h, 2, 2, Screen::CR_NO_P_CHECK); _screen->fillRect(21, 15, 89, h + 15, _flags.use16ColorMode ? 0xbb : 206); } _screen->copyRegion(112, 16, 12, h + 15, 87, 14, 2, 2, Screen::CR_NO_P_CHECK); int y = 15; for (int i = 0; i < 7; i++) { if (_availableSpells[i] == -1) continue; uint8 col = (i == _selectedSpell) ? (_flags.use16ColorMode ? 0x88 : 132) : (_flags.use16ColorMode ? 0x44 : 1); _screen->fprintString("%s", 24, y, col, 0, 0, getLangString(_spellProperties[_availableSpells[i]].spellNameCode)); y += 9; } _screen->setFont(of); } void LoLEngine::gui_highlightSelectedSpell(bool mode) { int y = 15; Screen::FontId of = _screen->setFont(Screen::FID_9_FNT); for (int i = 0; i < 7; i++) { if (_availableSpells[i] == -1) continue; uint8 col = (mode && (i == _selectedSpell)) ? (_flags.use16ColorMode ? 0x88 : 132) : (_flags.use16ColorMode ? 0x44 : 1); _screen->fprintString("%s", 24, y, col, 0, 0, getLangString(_spellProperties[_availableSpells[i]].spellNameCode)); y += 9; } _screen->setFont(of); } void LoLEngine::gui_displayCharInventory(int charNum) { static const uint8 inventoryTypes[] = { 0, 1, 2, 6, 3, 1, 1, 3, 5, 4 }; int cp = _screen->setCurPage(2); LoLCharacter *l = &_characters[charNum]; int id = l->id; if (id < 0) id = -id; if (id != _lastCharInventory) { char file[13]; sprintf(file, "invent%d.cps", inventoryTypes[id]); _screen->loadBitmap(file, 3, 3, 0); _screen->copyRegion(0, 0, 112, 0, 208, 120, 2, 6); } else { _screen->copyRegion(112, 0, 0, 0, 208, 120, 6, 2); } _screen->copyRegion(80, 143, 80, 143, 232, 35, 0, 2); gui_drawAllCharPortraitsWithStats(); if (_flags.use16ColorMode) _screen->fprintString("%s", 156, 8, 0xe1, 0, 1, l->name); else _screen->fprintString("%s", 157, 9, 254, 0, 5, l->name); gui_printCharInventoryStats(charNum); for (int i = 0; i < 11; i++) gui_drawCharInventoryItem(i); Screen::FontId of = _screen->setFont(Screen::FID_9_FNT); _screen->fprintString("%s", 182, 103, _flags.use16ColorMode ? 0xbb : 172, 0, 5, getLangString(0x4033)); _screen->setFont(of); static const uint16 statusFlags[] = { 0x0080, 0x0000, 0x1000, 0x0002, 0x100, 0x0001, 0x0000, 0x0000 }; memset(_charStatusFlags, 0xffff, sizeof(_charStatusFlags)); int x = 0; int32 c = 0; for (int i = 0; i < 3; i++) { if (!(l->flags & statusFlags[i << 1])) continue; uint8 *shp = _gameShapes[statusFlags[(i << 1) + 1]]; _screen->drawShape(_screen->_curPage, shp, 108 + x, 98, 0, 0); x += (shp[3] + 2); _charStatusFlags[c] = statusFlags[(i << 1) + 1]; c++; } for (int i = 0; i < 3; i++) { int32 b = l->experiencePts[i] - _expRequirements[l->skillLevels[i] - 1]; int32 e = _expRequirements[l->skillLevels[i]] - _expRequirements[l->skillLevels[i] - 1]; while (e & 0xffff8000) { e >>= 1; c = b; b >>= 1; if (c && !b) b = 1; } if (_flags.use16ColorMode) gui_drawBarGraph(154, 66 + i * 8, 34, 5, b, e, 0x88, 0); else gui_drawBarGraph(154, 64 + i * 10, 34, 5, b, e, 132, 0); } _screen->drawClippedLine(14, 120, 194, 120, 1); _screen->copyRegion(0, 0, 112, 0, 208, 121, 2, 0); _screen->copyRegion(80, 143, 80, 143, 232, 35, 2, 0); _screen->setCurPage(cp); } void LoLEngine::gui_printCharInventoryStats(int charNum) { for (int i = 0; i < 5; i++) gui_printCharacterStats(i, 1, calculateCharacterStats(charNum, i)); _charInventoryUnk |= (1 << charNum); } void LoLEngine::gui_printCharacterStats(int index, int redraw, int value) { uint32 offs = _screen->_curPage ? 0 : 112; int y = 0; int col = 0; if (index < 2) { // might // protection if (_flags.use16ColorMode) { y = (index + 2) << 3; col = 0xa1; if (redraw) _screen->fprintString("%s", offs + 108, y, col, 0, 0, getLangString(0x4014 + index)); } else { y = index * 10 + 22; col = 158; if (redraw) _screen->fprintString("%s", offs + 108, y, col, 0, 4, getLangString(0x4014 + index)); } } else { //skills int s = index - 2; y = s * 10 + 62; if (_flags.use16ColorMode) { y = (s + 8) << 3; col = _characters[_selectedCharacter].flags & (0x200 << s) ? 0xe1 : 0x81; if (redraw) _screen->fprintString("%s", offs + 108, y, col, 0, 0, getLangString(0x4014 + index)); } else { y = s * 10 + 62; col = _characters[_selectedCharacter].flags & (0x200 << s) ? 254 : 180; if (redraw) _screen->fprintString("%s", offs + 108, y, col, 0, 4, getLangString(0x4014 + index)); } } if (offs) _screen->copyRegion(294, y, 182 + offs, y, 18, 8, 6, _screen->_curPage, Screen::CR_NO_P_CHECK); Screen::FontId of = _flags.use16ColorMode ? _screen->setFont(Screen::FID_SJIS_FNT) : _screen->_currentFont; _screen->fprintString("%d", 200 + offs, y, col, 0, _flags.use16ColorMode ? 2 : 6, value); _screen->setFont(of); } void LoLEngine::gui_changeCharacterStats(int charNum) { int tmp[5]; int inc[5]; bool prc = false; for (int i = 0; i < 5; i++) { tmp[i] = calculateCharacterStats(charNum, i); int diff = tmp[i] - _charStatsTemp[i]; inc[i] = diff / 15; if (diff) { prc = true; if (!inc[i]) inc[i] = (diff < 0) ? -1 : 1; } } if (!prc) return; do { prc = false; for (int i = 0; i < 5; i++) { if (tmp[i] == _charStatsTemp[i]) continue; _charStatsTemp[i] += inc[i]; if ((inc[i] > 0 && tmp[i] < _charStatsTemp[i]) || (inc[i] < 0 && tmp[i] > _charStatsTemp[i])) _charStatsTemp[i] = tmp[i]; gui_printCharacterStats(i, 0, _charStatsTemp[i]); prc = true; } delay(_tickLength, true); } while (prc); } void LoLEngine::gui_drawCharInventoryItem(int itemIndex) { static const uint8 slotShapes[] = { 0x30, 0x34, 0x30, 0x34, 0x2E, 0x2F, 0x32, 0x33, 0x31, 0x35, 0x35 }; const uint8 *coords = &_charInvDefs[_charInvIndex[_characters[_selectedCharacter].raceClassSex] * 22 + itemIndex * 2]; uint8 x = *coords++; uint8 y = *coords; if (y == 0xff) return; if (!_screen->_curPage) x += 112; int i = _characters[_selectedCharacter].items[itemIndex]; int shapeNum = i ? ((itemIndex < 9) ? 4 : 5) : (_flags.isTalkie ? slotShapes[itemIndex] : slotShapes[itemIndex] - 2); _screen->drawShape(_screen->_curPage, _gameShapes[shapeNum], x, y, 0, 0); if (itemIndex > 8) { x -= 5; y -= 5; } if (i) _screen->drawShape(_screen->_curPage, getItemIconShapePtr(i), x + 1, y + 1, 0, 0); } void LoLEngine::gui_drawBarGraph(int x, int y, int w, int h, int32 cur, int32 max, int col1, int col2) { if (max < 1) return; if (cur < 0) cur = 0; int32 e = MIN(cur, max); if (!--w) return; if (!--h) return; int32 t = (e * w) / max; if (!t && e) t++; if (t) _screen->fillRect(x, y, x + t - 1, y + h, col1); if (t < w && col2) _screen->fillRect(x + t, y, x + w, y + h, col2); } void LoLEngine::gui_drawAllCharPortraitsWithStats() { int numChars = countActiveCharacters(); if (!numChars) return; for (int i = 0; i < numChars; i++) gui_drawCharPortraitWithStats(i); } void LoLEngine::gui_drawCharPortraitWithStats(int charNum) { if (!(_characters[charNum].flags & 1) || _updateFlags & 2) return; Screen::FontId tmpFid = _screen->setFont(Screen::FID_6_FNT); int cp = _screen->setCurPage(6); gui_drawBox(0, 0, 66, 34, 1, 1, -1); gui_drawCharFaceShape(charNum, 0, 1, _screen->_curPage); if (_flags.use16ColorMode) { gui_drawLiveMagicBar(33, 32, _characters[charNum].magicPointsCur, 0, _characters[charNum].magicPointsMax, 5, 32, 0xaa, 0x44, 0); gui_drawLiveMagicBar(39, 32, _characters[charNum].hitPointsCur, 0, _characters[charNum].hitPointsMax, 5, 32, 0x66, 0x44, 1); _screen->printText(getLangString(0x4253), 33, 1, 0x99, 0); _screen->printText(getLangString(0x4254), 39, 1, 0x55, 0); } else { gui_drawLiveMagicBar(33, 32, _characters[charNum].magicPointsCur, 0, _characters[charNum].magicPointsMax, 5, 32, 162, 1, 0); gui_drawLiveMagicBar(39, 32, _characters[charNum].hitPointsCur, 0, _characters[charNum].hitPointsMax, 5, 32, 154, 1, 1); _screen->printText(getLangString(0x4253), 33, 1, 160, 0); _screen->printText(getLangString(0x4254), 39, 1, 152, 0); } int spellLevels = 0; if (_availableSpells[_selectedSpell] != -1) { for (int i = 0; i < 4; i++) { if (_spellProperties[_availableSpells[_selectedSpell]].mpRequired[i] <= _characters[charNum].magicPointsCur && _spellProperties[_availableSpells[_selectedSpell]].hpRequired[i] <= _characters[charNum].hitPointsCur) spellLevels++; } } if (_characters[charNum].flags & 0x10) { // magic submenu open _screen->drawShape(_screen->_curPage, _gameShapes[_flags.isTalkie ? 73 : 71], 44, 0, 0, 0); if (spellLevels < 4) _screen->drawGridBox(44, (spellLevels << 3) + 1, 22, 32 - (spellLevels << 3), 1); } else { // magic submenu closed int handIndex = 0; if (_characters[charNum].items[0]) { if (_itemProperties[_itemsInPlay[_characters[charNum].items[0]].itemPropertyIndex].might != -1) handIndex = _itemsInPlay[_characters[charNum].items[0]].itemPropertyIndex; } handIndex = _gameShapeMap[(_itemProperties[handIndex].shpIndex << 1) + 1]; if (handIndex == _gameShapeMap[1]) { // draw raceClassSex specific hand shape handIndex = _characters[charNum].raceClassSex - 1; if (handIndex < 0) handIndex = 0; handIndex += (_flags.isTalkie ? 68 : 66); } // draw hand/weapon _screen->drawShape(_screen->_curPage, _gameShapes[handIndex], 44, 0, 0, 0); // draw magic symbol _screen->drawShape(_screen->_curPage, _gameShapes[(_flags.isTalkie ? 72 : 70) + _characters[charNum].field_41], 44, 17, 0, 0); if (spellLevels == 0) _screen->drawGridBox(44, 17, 22, 16, 1); } uint16 f = _characters[charNum].flags & 0x314C; if ((f == 0 && _weaponsDisabled) || (f && (f != 4 || _characters[charNum].weaponHit == 0 || (_characters[charNum].weaponHit && _weaponsDisabled)))) _screen->drawGridBox(44, 0, 22, 34, 1); if (_characters[charNum].weaponHit) { _screen->drawShape(_screen->_curPage, _gameShapes[_flags.isTalkie ? 34 : 32], 44, 0, 0, 0); _screen->fprintString("%d", 57, 7, _flags.use16ColorMode ? 0x33 : 254, 0, 1, _characters[charNum].weaponHit); } if (_characters[charNum].damageSuffered) _screen->fprintString("%d", 17, 28, _flags.use16ColorMode ? 0x33 : 254, 0, 1, _characters[charNum].damageSuffered); uint8 col = (charNum != _selectedCharacter || countActiveCharacters() == 1) ? 1 : 212; if (_flags.use16ColorMode) col = (charNum != _selectedCharacter || countActiveCharacters() == 1) ? 0x44 : 0x22; _screen->drawBox(0, 0, 65, 33, col); _screen->copyRegion(0, 0, _activeCharsXpos[charNum], 143, 66, 34, _screen->_curPage, cp, Screen::CR_NO_P_CHECK); _screen->setCurPage(cp); _screen->setFont(tmpFid); } void LoLEngine::gui_drawBox(int x, int y, int w, int h, int frameColor1, int frameColor2, int fillColor) { w--; h--; if (fillColor != -1) _screen->fillRect(x + 1, y + 1, x + w - 1, y + h - 1, fillColor); _screen->drawClippedLine(x + 1, y, x + w, y, frameColor2); _screen->drawClippedLine(x + w, y, x + w, y + h - 1, frameColor2); _screen->drawClippedLine(x, y, x, y + h, frameColor1); _screen->drawClippedLine(x, y + h, x + w, y + h, frameColor1); } void LoLEngine::gui_drawCharFaceShape(int charNum, int x, int y, int pageNum) { if (_characters[charNum].curFaceFrame < 7 && _characters[charNum].tempFaceFrame) _characters[charNum].curFaceFrame = _characters[charNum].tempFaceFrame; if (_characters[charNum].tempFaceFrame == 0 && _characters[charNum].curFaceFrame > 1 && _characters[charNum].curFaceFrame < 7) _characters[charNum].curFaceFrame = _characters[charNum].tempFaceFrame; int frm = (_characters[charNum].flags & 0x1108 && _characters[charNum].curFaceFrame < 7) ? 1 : _characters[charNum].curFaceFrame; if (_characters[charNum].hitPointsCur <= (_characters[charNum].hitPointsMax >> 1)) frm += 14; if (!pageNum) _screen->hideMouse(); _screen->drawShape(pageNum, _characterFaceShapes[frm][charNum], x, y, 0, 0x100, _screen->_paletteOverlay2, (_characters[charNum].flags & 0x80 ? 1 : 0)); if (_characters[charNum].flags & 0x40) // draw spider web _screen->drawShape(pageNum, _gameShapes[21], x, y, 0, 0); if (!pageNum) _screen->showMouse(); } void LoLEngine::gui_highlightPortraitFrame(int charNum) { if (charNum != _selectedCharacter) { int o = _selectedCharacter; _selectedCharacter = charNum; gui_drawCharPortraitWithStats(o); } gui_drawCharPortraitWithStats(charNum); } void LoLEngine::gui_drawLiveMagicBar(int x, int y, int curPoints, int unk, int maxPoints, int w, int h, int col1, int col2, int flag) { w--; h--; if (maxPoints < 1) return; int t = (curPoints < 1) ? 0 : curPoints; curPoints = (maxPoints < t) ? maxPoints : t; int barHeight = (curPoints * h) / maxPoints; if (barHeight < 1 && curPoints > 0) barHeight = 1; _screen->drawClippedLine(x - 1, y - h, x - 1, y, _flags.use16ColorMode ? 0x44 : 1); if (flag) { t = maxPoints >> 1; if (t > curPoints) col1 = _flags.use16ColorMode ? 0xbb : 144; t = maxPoints >> 2; if (t > curPoints) col1 = _flags.use16ColorMode ? 0x88 : 132; } if (barHeight > 0) _screen->fillRect(x, y - barHeight, x + w, y, col1); if (barHeight < h) _screen->fillRect(x, y - h, x + w, y - barHeight, col2); if (unk > 0 && unk < maxPoints) _screen->drawBox(x, y - barHeight, x + w, y, col1 - 2); } void LoLEngine::calcCharPortraitXpos() { int nc = countActiveCharacters(); if (_currentControlMode && !textEnabled()) { int t = (280 - (nc * 33)) / (nc + 1); for (int i = 0; i < nc; i++) _activeCharsXpos[i] = i * 33 + t * (i + 1) + 10; } else { int t = (235 - (nc * 66)) / (nc + 1); for (int i = 0; i < nc; i++) _activeCharsXpos[i] = i * 66 + t * (i + 1) + 83; } } void LoLEngine::gui_drawMoneyBox(int pageNum) { static const uint16 moneyX256[] = { 0x128, 0x134, 0x12b, 0x131, 0x12e}; static const uint16 moneyY256[] = { 0x73, 0x73, 0x74, 0x74, 0x75}; static const uint16 moneyX16[] = { 0x127, 0x133, 0x12a, 0x130, 0x12d}; static const uint16 moneyY16[] = { 0x74, 0x74, 0x75, 0x75, 0x76}; int backupPage = _screen->_curPage; _screen->_curPage = pageNum; const uint16 *moneyX; const uint16 *moneyY; if (_flags.use16ColorMode) { moneyX = moneyX16; moneyY = moneyY16; _screen->fillRect(291, 98, 315, 118, 0x11, pageNum); } else { moneyX = moneyX256; moneyY = moneyY256; _screen->fillRect(292, 97, 316, 118, 252, pageNum); } for (int i = 0; i < 5; i++) { if (!_moneyColumnHeight[i]) continue; uint8 h = _moneyColumnHeight[i] - 1; _screen->drawClippedLine(moneyX[i], moneyY[i], moneyX[i], moneyY[i] - h, _flags.use16ColorMode ? 1 : 0xd2); _screen->drawClippedLine(moneyX[i] + 1, moneyY[i], moneyX[i] + 1, moneyY[i] - h, _flags.use16ColorMode ? 2 : 0xd1); _screen->drawClippedLine(moneyX[i] + 2, moneyY[i], moneyX[i] + 2, moneyY[i] - h, _flags.use16ColorMode ? 3 : 0xd0); _screen->drawClippedLine(moneyX[i] + 3, moneyY[i], moneyX[i] + 3, moneyY[i] - h, _flags.use16ColorMode ? 2 : 0xd1); _screen->drawClippedLine(moneyX[i] + 4, moneyY[i], moneyX[i] + 4, moneyY[i] - h, _flags.use16ColorMode ? 1 : 0xd2); } Screen::FontId backupFont = _screen->setFont(Screen::FID_6_FNT); if (_flags.use16ColorMode) _screen->fprintString("%d", 304, 99, 0x33, 0, 1, _credits); else _screen->fprintString("%d", 305, 98, 254, 0, 1, _credits); _screen->setFont(backupFont); _screen->_curPage = backupPage; if (pageNum == 6) { if (_flags.use16ColorMode) _screen->copyRegion(291, 98, 291, 98, 24, 20, 6, 0); else _screen->copyRegion(292, 97, 292, 97, 25, 22, 6, 0); } } void LoLEngine::gui_drawCompass() { if (!(_flagsTable[31] & 0x40)) return; if (_compassDirection == -1) { _compassDirectionIndex = -1; _compassDirection = _currentDirection << 6; } int t = ((_compassDirection + 4) >> 3) & 0x1f; if (t == _compassDirectionIndex) return; _compassDirectionIndex = t; if (!_screen->_curPage) _screen->hideMouse(); const CompassDef *c = &_compassDefs[t]; int compassShp = 22; int compassPtr = 23; if (_flags.isTalkie) { compassShp += _lang; compassPtr = 25; } _screen->drawShape(_screen->_curPage, _gameShapes[compassShp], 294, 3, 0, 0); _screen->drawShape(_screen->_curPage, _gameShapes[compassPtr + c->shapeIndex], 298 + c->x, c->y + 9, 0, c->flags | 0x300, _screen->_paletteOverlay1, 1); _screen->drawShape(_screen->_curPage, _gameShapes[compassPtr + c->shapeIndex], 299 + c->x, c->y + 8, 0, c->flags); if (!_screen->_curPage) _screen->showMouse(); } int LoLEngine::gui_enableControls() { _floatingCursorControl = 0; int start = 74; int end = 83; if (_flags.isTalkie) { start = 76; end = 85; } if (!_currentControlMode) { for (int i = start; i < end; i++) gui_toggleButtonDisplayMode(i, 2); } gui_toggleFightButtons(false); return 1; } int LoLEngine::gui_disableControls(int controlMode) { if (_currentControlMode) return 0; _floatingCursorControl = (controlMode & 2) ? 2 : 1; gui_toggleFightButtons(true); int start = 74; int end = 83; int swtch = 76; if (_flags.isTalkie) { start = 76; end = 85; swtch = 78; } for (int i = start; i < end; i++) gui_toggleButtonDisplayMode(i, ((controlMode & 2) && (i > swtch)) ? 2 : 3); return 1; } void LoLEngine::gui_toggleButtonDisplayMode(int shapeIndex, int mode) { static const int16 buttonX[] = { 0x0056, 0x0128, 0x000C, 0x0021, 0x0122, 0x000C, 0x0021, 0x0036, 0x000C, 0x0021, 0x0036 }; static const int16 buttonY[] = { 0x00B4, 0x00B4, 0x00B4, 0x00B4, 0x0020, 0x0084, 0x0084, 0x0084, 0x0096, 0x0096, 0x0096 }; int swtch = 76; int subst = 72; if (_flags.isTalkie) { swtch = 78; subst = 74; } if (shapeIndex == swtch && !(_flagsTable[31] & 0x10)) return; if (_currentControlMode && _needSceneRestore) return; if (mode == 0) shapeIndex = _lastButtonShape; int pageNum = 0; int16 x1 = buttonX[shapeIndex - subst]; int16 y1 = buttonY[shapeIndex - subst]; int16 x2 = 0; int16 y2 = 0; uint32 t = 0; switch (mode) { case 1: mode = 0x100; _lastButtonShape = shapeIndex; break; case 0: if (!_lastButtonShape) return; t = _system->getMillis(); if (_buttonPressTimer > t) delay(_buttonPressTimer - t); case 2: mode = 0; _lastButtonShape = 0; break; case 3: mode = 0; _lastButtonShape = 0; pageNum = 6; x2 = x1; y2 = y1; x1 = 0; y1 = 0; break; default: break; } _screen->drawShape(pageNum, _gameShapes[shapeIndex], x1, y1, 0, mode, _screen->_paletteOverlay1, 1); if (!pageNum) _screen->updateScreen(); if (pageNum == 6) { int cp = _screen->setCurPage(6); _screen->drawGridBox(x1, y1, _gameShapes[shapeIndex][3], _gameShapes[shapeIndex][2], 1); _screen->copyRegion(x1, y1, x2, y2, _gameShapes[shapeIndex][3], _gameShapes[shapeIndex][2], pageNum, 0, Screen::CR_NO_P_CHECK); _screen->updateScreen(); _screen->setCurPage(cp); } _buttonPressTimer = _system->getMillis() + 6 * _tickLength; } void LoLEngine::gui_toggleFightButtons(bool disable) { for (int i = 0; i < 3; i++) { if (!(_characters[i].flags & 1)) continue; if (disable) _characters[i].flags |= 0x2000; else _characters[i].flags &= 0xdfff; if (disable && !textEnabled()) { int u = _selectedCharacter; _selectedCharacter = 99; int f = _updateFlags; _updateFlags &= 0xfffd; gui_drawCharPortraitWithStats(i); _updateFlags = f; _selectedCharacter = u; } else { gui_drawCharPortraitWithStats(i); } } } void LoLEngine::gui_updateInput() { // TODO: We need to catch all cases where loading is not possible and // set the "mainLoop" parameter to false for them. int inputFlag = checkInput(_activeButtons, !((_updateFlags & 3) || _weaponsDisabled), 0); if (_preserveEvents) _preserveEvents = false; else removeInputTop(); if (inputFlag && _activeMagicMenu != -1 && !(inputFlag & 0x8800)) { gui_enableDefaultPlayfieldButtons(); _characters[_activeMagicMenu].flags &= 0xffef; gui_drawCharPortraitWithStats(_activeMagicMenu); gui_triggerEvent(inputFlag); _preserveEvents = false; _activeMagicMenu = -1; inputFlag = 0; } if (inputFlag == _keyMap[Common::KEYCODE_SPACE] || inputFlag == _keyMap[Common::KEYCODE_RETURN]) { snd_stopSpeech(true); } else if (inputFlag == _keyMap[Common::KEYCODE_SLASH]) { if (_weaponsDisabled || _availableSpells[1] == -1) return; gui_highlightSelectedSpell(false); if (_availableSpells[++_selectedSpell] == -1) _selectedSpell = 0; gui_highlightSelectedSpell(true); gui_drawAllCharPortraitsWithStats(); } } void LoLEngine::gui_triggerEvent(int eventType) { Common::Event evt; memset(&evt, 0, sizeof(Common::Event)); evt.mouse.x = _mouseX; evt.mouse.y = _mouseY; if (eventType == 65 || eventType == 199) { evt.type = Common::EVENT_LBUTTONDOWN; } else if (eventType == 66 || eventType == 201) { evt.type = Common::EVENT_RBUTTONDOWN; } else { evt.type = Common::EVENT_KEYDOWN; for (Common::HashMap::const_iterator c = _keyMap.begin(); c != _keyMap.end(); ++c) { if (c->_value == eventType) evt.kbd.keycode = (Common::KeyCode) c->_key; } } removeInputTop(); _eventList.push_back(Event(evt, true)); _preserveEvents = true; } void LoLEngine::removeInputTop() { if (!_eventList.empty()) { if (_eventList.begin()->event.type == Common::EVENT_LBUTTONDOWN) _gui->_mouseClick = 1; else if (_eventList.begin()->event.type == Common::EVENT_RBUTTONDOWN) _gui->_mouseClick = 2; else _gui->_mouseClick = 0; _eventList.erase(_eventList.begin()); } } void LoLEngine::gui_enableDefaultPlayfieldButtons() { gui_resetButtonList(); gui_initButtonsFromList(_buttonList1); gui_setFaceFramesControlButtons(7, 44); gui_setFaceFramesControlButtons(11, 44); gui_setFaceFramesControlButtons(17, 0); gui_setFaceFramesControlButtons(29, 0); gui_setFaceFramesControlButtons(25, 33); if (_flagsTable[31] & 0x20) gui_initMagicScrollButtons(); } void LoLEngine::gui_enableSequenceButtons(int x, int y, int w, int h, int enableFlags) { gui_resetButtonList(); _sceneWindowButton.x = x; _sceneWindowButton.y = y; _sceneWindowButton.w = w; _sceneWindowButton.h = h; gui_initButtonsFromList(_buttonList3); if (enableFlags & 1) gui_initButtonsFromList(_buttonList4); if (enableFlags & 2) gui_initButtonsFromList(_buttonList5); } void LoLEngine::gui_specialSceneRestoreButtons() { if (!_spsWindowW && !_spsWindowH) return; gui_enableDefaultPlayfieldButtons(); _spsWindowX = _spsWindowY = _spsWindowW = _spsWindowH = _seqTrigger = 0; } void LoLEngine::gui_enableCharInventoryButtons(int charNum) { gui_resetButtonList(); gui_initButtonsFromList(_buttonList2); gui_initCharInventorySpecialButtons(charNum); gui_setFaceFramesControlButtons(21, 0); } void LoLEngine::gui_resetButtonList() { for (uint i = 0; i < ARRAYSIZE(_activeButtonData); ++i) _activeButtonData[i].nextButton = 0; gui_notifyButtonListChanged(); _activeButtons = 0; } void LoLEngine::gui_initButtonsFromList(const int16 *list) { while (*list != -1) gui_initButton(*list++); } void LoLEngine::gui_setFaceFramesControlButtons(int index, int xOffs) { int c = countActiveCharacters(); for (int i = 0; i < c; i++) gui_initButton(index + i, _activeCharsXpos[i] + xOffs); } void LoLEngine::gui_initCharInventorySpecialButtons(int charNum) { const uint8 *s = &_charInvDefs[_charInvIndex[_characters[charNum].raceClassSex] * 22]; for (int i = 0; i < 11; i++) { if (*s != 0xff) gui_initButton(33 + i, s[0], s[1], i); s += 2; } } void LoLEngine::gui_initMagicScrollButtons() { for (int i = 0; i < 7; i++) { if (_availableSpells[i] == -1) continue; gui_initButton(71 + i, -1, -1, i); } } void LoLEngine::gui_initMagicSubmenu(int charNum) { gui_resetButtonList(); _subMenuIndex = charNum; gui_initButtonsFromList(_buttonList7); } void LoLEngine::gui_initButton(int index, int x, int y, int val) { Button *b = 0; int cnt = 1; if (_activeButtons) { Button *n = _activeButtons; while (n->nextButton) { ++cnt; n = n->nextButton; } ++cnt; b = n->nextButton = &_activeButtonData[cnt]; } else { b = &_activeButtonData[0]; _activeButtons = b; } *b = Button(); b->nextButton = 0; b->data0Val2 = b->data1Val2 = b->data2Val2 = 0xfe; b->data0Val3 = b->data1Val3 = b->data2Val3 = 0x01; b->index = cnt; b->keyCode = _buttonData[index].keyCode; b->keyCode2 = _buttonData[index].keyCode2; b->dimTableIndex = _buttonData[index].screenDim; b->flags = _buttonData[index].buttonflags; b->arg = (val != -1) ? (uint8)(val & 0xff) : _buttonData[index].index; if (index == 15) { // magic sub menu b->x = _activeCharsXpos[_subMenuIndex] + 44; b->arg = _subMenuIndex; b->y = _buttonData[index].y; b->width = _buttonData[index].w - 1; b->height = _buttonData[index].h - 1; } else if (index == 64) { // scene window button b->x = _sceneWindowButton.x; b->y = _sceneWindowButton.y; b->width = _sceneWindowButton.w - 1; b->height = _sceneWindowButton.h - 1; } else { b->x = x != -1 ? x : _buttonData[index].x; b->y = y != -1 ? y : _buttonData[index].y; b->width = _buttonData[index].w - 1; b->height = _buttonData[index].h - 1; } b->buttonCallback = _buttonCallbacks[index]; } void LoLEngine::gui_notifyButtonListChanged() { if (_gui) { if (!_gui->_buttonListChanged && !_preserveEvents) removeInputTop(); _gui->_buttonListChanged = true; } } int LoLEngine::clickedUpArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; moveParty(_currentDirection, ((button->flags2 & 0x1080) == 0x1080) ? 1 : 0, 0, _flags.isTalkie ? 80 : 78); return 1; } int LoLEngine::clickedDownArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; moveParty(_currentDirection ^ 2, 0, 1, _flags.isTalkie ? 83 : 81); return 1; } int LoLEngine::clickedLeftArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; moveParty((_currentDirection - 1) & 3, ((button->flags2 & 0x1080) == 0x1080) ? 1 : 0, 2, _flags.isTalkie ? 82 : 80); return 1; } int LoLEngine::clickedRightArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; moveParty((_currentDirection + 1) & 3, ((button->flags2 & 0x1080) == 0x1080) ? 1 : 0, 3, _flags.isTalkie ? 84 : 82); return 1; } int LoLEngine::clickedTurnLeftArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; gui_toggleButtonDisplayMode(_flags.isTalkie ? 79 : 77, 1); _currentDirection = (_currentDirection - 1) & 3; _sceneDefaultUpdate = 1; runLevelScript(_currentBlock, 0x4000); initTextFading(2, 0); if (!_sceneDefaultUpdate) gui_drawScene(0); else movePartySmoothScrollTurnLeft(1); gui_toggleButtonDisplayMode(_flags.isTalkie ? 79 : 77, 0); runLevelScript(_currentBlock, 0x10); return 1; } int LoLEngine::clickedTurnRightArrow(Button *button) { if (button->arg && !_floatingCursorsEnabled) return 0; gui_toggleButtonDisplayMode(_flags.isTalkie ? 81 : 79, 1); _currentDirection = (_currentDirection + 1) & 3; _sceneDefaultUpdate = 1; runLevelScript(_currentBlock, 0x4000); initTextFading(2, 0); if (!_sceneDefaultUpdate) gui_drawScene(0); else movePartySmoothScrollTurnRight(1); gui_toggleButtonDisplayMode(_flags.isTalkie ? 81 : 79, 0); runLevelScript(_currentBlock, 0x10); return 1; } int LoLEngine::clickedAttackButton(Button *button) { int c = button->arg; if (_characters[c].flags & 0x314C) return 1; int bl = calcNewBlockPosition(_currentBlock, _currentDirection); if (_levelBlockProperties[bl].flags & 0x10) { breakIceWall(0, 0); return 1; } uint16 target = getNearestMonsterFromCharacter(c); int s = 0; for (int i = 0; i < 4; i++) { if (!_characters[c].items[i]) continue; runItemScript(c, _characters[c].items[i], 0x400, target, s); runLevelScriptCustom(_currentBlock, 0x400, c, _characters[c].items[i], target, s); s -= 10; } if (!s) { runItemScript(c, 0, 0x400, target, s); runLevelScriptCustom(_currentBlock, 0x400, c, 0, target, s); } s = _characters[c].weaponHit ? 4 : calcMonsterSkillLevel(c, 8) + 4; // check for Zephyr ring if (itemEquipped(c, 230)) s >>= 1; _characters[c].flags |= 4; gui_highlightPortraitFrame(c); setCharacterUpdateEvent(c, 1, s, 1); return 1; } int LoLEngine::clickedMagicButton(Button *button) { int c = button->arg; if (_characters[c].flags & 0x314C) return 1; if (checkMagic(c, _availableSpells[_selectedSpell], 0)) return 1; _characters[c].flags ^= 0x10; gui_drawCharPortraitWithStats(c); gui_initMagicSubmenu(c); _activeMagicMenu = c; return 1; } int LoLEngine::clickedMagicSubmenu(Button *button) { int spellLevel = (_mouseY - 144) >> 3; int c = button->arg; gui_enableDefaultPlayfieldButtons(); if (checkMagic(c, _availableSpells[_selectedSpell], spellLevel)) { _characters[c].flags &= 0xffef; gui_drawCharPortraitWithStats(c); } else { _characters[c].flags |= 4; _characters[c].flags &= 0xffef; if (castSpell(c, _availableSpells[_selectedSpell], spellLevel)) { setCharacterUpdateEvent(c, 1, 8, 1); increaseExperience(c, 2, spellLevel * spellLevel); } else { _characters[c].flags &= 0xfffb; gui_drawCharPortraitWithStats(c); } } _activeMagicMenu = -1; return 1; } int LoLEngine::clickedScreen(Button *button) { _characters[_activeMagicMenu].flags &= 0xffef; gui_drawCharPortraitWithStats(_activeMagicMenu); _activeMagicMenu = -1; if (!(button->flags2 & 0x80)) { if (button->flags2 & 0x100) gui_triggerEvent(65); else gui_triggerEvent(66); } gui_enableDefaultPlayfieldButtons(); return 1; } int LoLEngine::clickedPortraitLeft(Button *button) { disableSysTimer(2); if (!_weaponsDisabled) { _screen->copyRegionToBuffer(2, 0, 0, 320, 200, _pageBuffer2); _screen->copyPage(0, 2); _screen->copyRegionToBuffer(2, 0, 0, 320, 200, _pageBuffer1); _updateFlags |= 0x0C; gui_disableControls(1); } _selectedCharacter = button->arg; _weaponsDisabled = true; if (_flags.use16ColorMode) _screen->fillRect(112, 0, 288, 120, 0, 2); gui_displayCharInventory(_selectedCharacter); gui_enableCharInventoryButtons(_selectedCharacter); return 1; } int LoLEngine::clickedLiveMagicBarsLeft(Button *button) { gui_highlightPortraitFrame(button->arg); _txt->printMessage(0, getLangString(0x4047), _characters[button->arg].name, _characters[button->arg].hitPointsCur, _characters[button->arg].hitPointsMax, _characters[button->arg].magicPointsCur, _characters[button->arg].magicPointsMax); return 1; } int LoLEngine::clickedPortraitEtcRight(Button *button) { if (!_itemInHand) return 1; int flg = _itemProperties[_itemsInPlay[_itemInHand].itemPropertyIndex].flags; int c = button->arg; if (flg & 1) { if (!(_characters[c].flags & 8) || (flg & 0x20)) { runItemScript(c, _itemInHand, 0x400, 0, 0); runLevelScriptCustom(_currentBlock, 0x400, c, _itemInHand, 0, 0); } else { _txt->printMessage(2, getLangString(0x402c), _characters[c].name); } return 1; } _txt->printMessage(2, "%s", getLangString((flg & 8) ? 0x4029 : ((flg & 0x10) ? 0x402a : 0x402b))); return 1; } int LoLEngine::clickedCharInventorySlot(Button *button) { if (_itemInHand) { uint16 sl = 1 << button->arg; int type = _itemProperties[_itemsInPlay[_itemInHand].itemPropertyIndex].type; if (!(sl & type)) { bool f = false; for (int i = 0; i < 11; i++) { if (!(type & (1 << i))) continue; _txt->printMessage(0, getLangString(i > 3 ? 0x418A : 0x418B), getLangString(_itemProperties[_itemsInPlay[_itemInHand].itemPropertyIndex].nameStringId), getLangString(_inventorySlotDesc[i])); f = true; } if (!f) _txt->printMessage(_itemsInPlay[_itemInHand].itemPropertyIndex == 231 ? 2 : 0, "%s", getLangString(0x418C)); return 1; } } else { if (!_characters[_selectedCharacter].items[button->arg]) { _txt->printMessage(0, "%s", getLangString(_inventorySlotDesc[button->arg] + 8)); return 1; } } int ih = _itemInHand; setHandItem(_characters[_selectedCharacter].items[button->arg]); _characters[_selectedCharacter].items[button->arg] = ih; gui_drawCharInventoryItem(button->arg); recalcCharacterStats(_selectedCharacter); if (_itemInHand) runItemScript(_selectedCharacter, _itemInHand, 0x100, 0, 0); if (ih) runItemScript(_selectedCharacter, ih, 0x80, 0, 0); gui_drawCharInventoryItem(button->arg); gui_drawCharPortraitWithStats(_selectedCharacter); gui_changeCharacterStats(_selectedCharacter); return 1; } int LoLEngine::clickedExitCharInventory(Button *button) { _updateFlags &= 0xfff3; gui_enableDefaultPlayfieldButtons(); _weaponsDisabled = false; for (int i = 0; i < 4; i++) { if (_charInventoryUnk & (1 << i)) _characters[i].flags &= 0xf1ff; } _screen->copyBlockToPage(2, 0, 0, 320, 200, _pageBuffer1); int cp = _screen->setCurPage(2); gui_drawAllCharPortraitsWithStats(); gui_drawInventory(); _screen->setCurPage(cp); _screen->copyPage(2, 0); _screen->updateScreen(); gui_enableControls(); _screen->copyBlockToPage(2, 0, 0, 320, 200, _pageBuffer2); _lastCharInventory = -1; updateDrawPage2(); enableSysTimer(2); return 1; } int LoLEngine::clickedSceneDropItem(Button *button) { static const uint8 offsX[] = { 0x40, 0xC0, 0x40, 0xC0 }; static const uint8 offsY[] = { 0x40, 0x40, 0xC0, 0xC0 }; static const uint8 dirIndex[] = { 0, 1, 2, 3, 1, 3, 0, 2, 3, 2, 1, 0, 2, 0, 3, 1 }; if ((_updateFlags & 1) || !_itemInHand) return 0; uint16 block = _currentBlock; if (button->arg > 1) { block = calcNewBlockPosition(_currentBlock, _currentDirection); int f = _wllWallFlags[_levelBlockProperties[block].walls[_currentDirection ^ 2]]; if (!(f & 0x80) || (f & 2)) return 1; } uint16 x = 0; uint16 y = 0; int i = dirIndex[(_currentDirection << 2) + button->arg]; calcCoordinates(x, y, block, offsX[i], offsY[i]); setItemPosition(_itemInHand, x, y, 0, 1); setHandItem(0); return 1; } int LoLEngine::clickedScenePickupItem(Button *button) { static const int8 checkX[] = { 0, 0, 1, 0, -1, -1, 1, 1, -1, 0, 2, 0, -2, -1, 1, 2, 2, 1, -1, -2, -2 }; static const int8 checkY[] = { 0, -1, 0, 1, 0, -1, -1, 1, 1, -2, 0, 2, 0, -2, -2, -1, 1, 2, 2, 1, -1 }; static const int len = ARRAYSIZE(checkX); if ((_updateFlags & 1) || _itemInHand) return 0; int cp = _screen->setCurPage(_sceneDrawPage1); redrawSceneItem(); int p = 0; for (int i = 0; i < len; i++) { p = _screen->getPagePixel(_screen->_curPage, CLIP(_mouseX + checkX[i], 0, 320), CLIP(_mouseY + checkY[i], 0, 200)); if (p) break; } _screen->setCurPage(cp); if (!p) return 0; uint16 block = (p <= 128) ? calcNewBlockPosition(_currentBlock, _currentDirection) : _currentBlock; int found = checkSceneForItems(&_levelBlockProperties[block].drawObjects, p & 0x7f); if (found != -1) { removeLevelItem(found, block); setHandItem(found); } _sceneUpdateRequired = true; return 1; } int LoLEngine::clickedInventorySlot(Button *button) { int slot = _inventoryCurItem + button->arg; if (slot > 47) slot -= 48; uint16 slotItem = _inventory[slot]; int hItem = _itemInHand; if ((_itemsInPlay[hItem].itemPropertyIndex == 281 || _itemsInPlay[slotItem].itemPropertyIndex == 281) && (_itemsInPlay[hItem].itemPropertyIndex == 220 || _itemsInPlay[slotItem].itemPropertyIndex == 220)) { // merge ruby of truth WSAMovie_v2 *wsa = new WSAMovie_v2(this); wsa->open("truth.wsa", 0, 0); _screen->hideMouse(); _inventory[slot] = 0; gui_drawInventoryItem(button->arg); _screen->copyRegion(button->x, button->y - 3, button->x, button->y - 3, 25, 27, 0, 2); KyraEngine_v1::snd_playSoundEffect(99); for (int i = 0; i < 25; i++) { uint32 delayTimer = _system->getMillis() + 7 * _tickLength; _screen->copyRegion(button->x, button->y - 3, 0, 0, 25, 27, 2, 2); wsa->displayFrame(i, 2, 0, 0, 0x4000, 0, 0); _screen->copyRegion(0, 0, button->x, button->y - 3, 25, 27, 2, 0); _screen->updateScreen(); delayUntil(delayTimer); } _screen->showMouse(); wsa->close(); delete wsa; deleteItem(slotItem); deleteItem(hItem); setHandItem(0); _inventory[slot] = makeItem(280, 0, 0); } else { setHandItem(slotItem); _inventory[slot] = hItem; } gui_drawInventoryItem(button->arg); return 1; } int LoLEngine::clickedInventoryScroll(Button *button) { int8 inc = (int8)button->arg; int shp = (inc == 1) ? 75 : 74; if (!_flags.isTalkie) shp -= 2; if (button->flags2 & 0x1000) inc *= 9; _inventoryCurItem += inc; gui_toggleButtonDisplayMode(shp, 1); if (_inventoryCurItem < 0) _inventoryCurItem += 48; if (_inventoryCurItem > 47) _inventoryCurItem -= 48; gui_drawInventory(); gui_toggleButtonDisplayMode(shp, 0); return 1; } int LoLEngine::clickedWall(Button *button) { int block = calcNewBlockPosition(_currentBlock, _currentDirection); int dir = _currentDirection ^ 2; uint8 type = _wllBuffer3[_levelBlockProperties[block].walls[dir]]; int res = 0; switch (type) { case 1: res = clickedWallShape(block, dir); break; case 2: res = clickedLeverOn(block, dir); break; case 3: res = clickedLeverOff(block, dir); break; case 4: res = clickedWallOnlyScript(block); break; case 5: res = clickedDoorSwitch(block, dir); break; case 6: res = clickedNiche(block, dir); break; default: break; } return res; } int LoLEngine::clickedSequenceWindow(Button *button) { runLevelScript(calcNewBlockPosition(_currentBlock, _currentDirection), 0x40); if (!_seqTrigger || !posWithinRect(_mouseX, _mouseY, _seqWindowX1, _seqWindowY1, _seqWindowX2, _seqWindowY2)) { _seqTrigger = 0; removeInputTop(); } return 1; } int LoLEngine::clickedScroll(Button *button) { if (_selectedSpell == button->arg) return 1; gui_highlightSelectedSpell(false); _selectedSpell = button->arg; gui_highlightSelectedSpell(true); gui_drawAllCharPortraitsWithStats(); return 1; } int LoLEngine::clickedSpellTargetCharacter(Button *button) { int t = button->arg; _txt->printMessage(0, "%s.\r", _characters[t].name); if ((_spellProperties[_activeSpell.spell].flags & 0xff) == 1) { _activeSpell.target = t; castHealOnSingleCharacter(&_activeSpell); } gui_enableDefaultPlayfieldButtons(); return 1; } int LoLEngine::clickedSpellTargetScene(Button *button) { LoLCharacter *c = &_characters[_activeSpell.charNum]; _txt->printMessage(0, "%s", getLangString(0x4041)); c->magicPointsCur += _activeSpell.p->mpRequired[_activeSpell.level]; if (c->magicPointsCur > c->magicPointsMax) c->magicPointsCur = c->magicPointsMax; c->hitPointsCur += _activeSpell.p->hpRequired[_activeSpell.level]; if (c->hitPointsCur > c->hitPointsMax) c->hitPointsCur = c->hitPointsMax; gui_drawCharPortraitWithStats(_activeSpell.charNum); gui_enableDefaultPlayfieldButtons(); return 1; } int LoLEngine::clickedSceneThrowItem(Button *button) { if (_updateFlags & 1) return 0; uint16 block = calcNewBlockPosition(_currentBlock, _currentDirection); if ((_wllWallFlags[_levelBlockProperties[block].walls[_currentDirection ^ 2]] & 2) || !_itemInHand) return 0; uint16 x = 0; uint16 y = 0; calcCoordinates(x, y, _currentBlock, 0x80, 0x80); if (launchObject(0, _itemInHand, x, y, 12, _currentDirection << 1, 6, _selectedCharacter, 0x3f)) { snd_playSoundEffect(18, -1); setHandItem(0); } _sceneUpdateRequired = true; return 1; } int LoLEngine::clickedOptions(Button *button) { removeInputTop(); gui_toggleButtonDisplayMode(_flags.isTalkie ? 76 : 74, 1); _updateFlags |= 4; Button b; b.data0Val2 = b.data1Val2 = b.data2Val2 = 0xfe; b.data0Val3 = b.data1Val3 = b.data2Val3 = 0x01; if (_weaponsDisabled) clickedExitCharInventory(&b); initTextFading(0, 1); stopPortraitSpeechAnim(); setLampMode(true); setMouseCursorToIcon(0); disableSysTimer(2); gui_toggleButtonDisplayMode(_flags.isTalkie ? 76 : 74, 0); bool speechWasEnabled = speechEnabled(); if (_flags.isTalkie && getVolume(kVolumeSpeech) == 2) _configVoice |= (textEnabled() ? 2 : 1); _gui->runMenu(_gui->_mainMenu); _updateFlags &= 0xfffb; setMouseCursorToItemInHand(); resetLampStatus(); gui_enableDefaultPlayfieldButtons(); enableSysTimer(2); updateDrawPage2(); gui_drawPlayField(); if (getVolume(kVolumeSpeech) == 2) _configVoice &= (textEnabled() ? ~2 : ~1); if (speechWasEnabled && !textEnabled() && !speechEnabled()) _configVoice = 0; writeSettings(); return 1; } int LoLEngine::clickedRestParty(Button *button) { gui_toggleButtonDisplayMode(_flags.isTalkie ? 77 : 75, 1); Button b; b.data0Val2 = b.data1Val2 = b.data2Val2 = 0xfe; b.data0Val3 = b.data1Val3 = b.data2Val3 = 0x01; if (_weaponsDisabled) clickedExitCharInventory(&b); int tHp = -1; int tMp = -1; int tHa = -1; int needPoisoningFlags = 0; int needHealingFlags = 0; int needMagicGainFlags = 0; for (int i = 0; i < 4; i++) { LoLCharacter *c = &_characters[i]; if (!(c->flags & 1) || (c->flags & 8)) continue; if (c->hitPointsMax > tHp) tHp = c->hitPointsMax; if (c->magicPointsMax > tMp) tMp = c->magicPointsMax; if (c->flags & 0x80) { needPoisoningFlags |= (1 << i); if (c->hitPointsCur > tHa) tHa = c->hitPointsCur; } else { if (c->hitPointsCur < c->hitPointsMax) needHealingFlags |= (1 << i); } if (c->magicPointsCur < c->magicPointsMax) needMagicGainFlags |= (1 << i); c->flags |= 0x1000; } removeInputTop(); if (needHealingFlags || needMagicGainFlags) { _screen->fillRect(112, 0, 288, 120, _flags.use16ColorMode ? 0x44 : 1); gui_drawAllCharPortraitsWithStats(); _txt->printMessage(0x8000, "%s", getLangString(0x4057)); gui_toggleButtonDisplayMode(_flags.isTalkie ? 77 : 75, 0); int h = 600 / tHp; if (h > 30) h = 30; int m = 600 / tMp; if (m > 30) m = 30; int a = 600 / tHa; if (a > 15) a = 15; uint32 delay1 = _system->getMillis() + h * _tickLength; uint32 delay2 = _system->getMillis() + m * _tickLength; uint32 delay3 = _system->getMillis() + a * _tickLength; _partyAwake = false; _updateFlags |= 1; for (int i = 0, im = _smoothScrollModeNormal ? 32 : 16; i < im; i++) { timerProcessMonsters(0); timerProcessMonsters(1); timerProcessDoors(0); timerProcessFlyingObjects(0); if (_partyAwake) break; } resetBlockProperties(); do { for (int i = 0, im = _smoothScrollModeNormal ? 8 : 4; i < im; i++) { timerProcessMonsters(0); timerProcessMonsters(1); timerProcessDoors(0); timerProcessFlyingObjects(0); if (_partyAwake) break; } int f = checkInput(0); removeInputTop(); if (f & 0x800) { gui_notifyButtonListChanged(); } else if (f) { gui_triggerEvent(f); break; } if (!_partyAwake) { if (_system->getMillis() > delay3) { for (int i = 0; i < 4; i++) { if (!(needPoisoningFlags & (1 << i))) continue; inflictDamage(i, 1, 0x8000, 1, 0x80); if (_characters[i].flags & 8) needPoisoningFlags &= ~(1 << i); } delay3 = _system->getMillis() + a * _tickLength; } if (_system->getMillis() > delay1) { for (int i = 0; i < 4; i++) { if (!(needHealingFlags & (1 << i))) continue; increaseCharacterHitpoints(i, 1, false); gui_drawCharPortraitWithStats(i); if (_characters[i].hitPointsCur == _characters[i].hitPointsMax) needHealingFlags &= ~(1 << i); } delay1 = _system->getMillis() + h * _tickLength; } if (_system->getMillis() > delay2) { for (int i = 0; i < 4; i++) { if (!(needMagicGainFlags & (1 << i))) continue; _characters[i].magicPointsCur++; gui_drawCharPortraitWithStats(i); if (_characters[i].magicPointsCur == _characters[i].magicPointsMax) needMagicGainFlags &= ~(1 << i); } delay2 = _system->getMillis() + m * _tickLength; } _screen->updateScreen(); } } while (!_partyAwake && (needHealingFlags || needMagicGainFlags)); for (int i = 0; i < 4; i++) { int frm = 0; int upd = 0; bool setframe = true; if (_characters[i].flags & 0x1000) { _characters[i].flags &= 0xefff; if (_partyAwake) { if (_characters[i].damageSuffered) { frm = 5; snd_playSoundEffect(_characters[i].screamSfx, -1); } else { frm = 4; } upd = 6; } } else { if (_characters[i].damageSuffered) setframe = false; else frm = 4; } if (setframe) setTemporaryFaceFrame(i, frm, upd, 1); } _updateFlags &= 0xfffe; _partyAwake = true; updateDrawPage2(); gui_drawScene(0); _txt->printMessage(0x8000, "%s", getLangString(0x4059)); _screen->fadeToPalette1(40); } else { for (int i = 0; i < 4; i++) _characters[i].flags &= 0xefff; if (needPoisoningFlags) { setTemporaryFaceFrameForAllCharacters(0, 0, 0); for (int i = 0; i < 4; i++) { if (needPoisoningFlags & (1 << i)) setTemporaryFaceFrame(i, 3, 8, 0); } _txt->printMessage(0x8000, "%s", getLangString(0x405a)); gui_drawAllCharPortraitsWithStats(); } else { setTemporaryFaceFrameForAllCharacters(2, 4, 1); _txt->printMessage(0x8000, "%s", getLangString(0x4058)); } gui_toggleButtonDisplayMode(_flags.isTalkie ? 77 : 75, 0); } return 1; } int LoLEngine::clickedMoneyBox(Button *button) { _txt->printMessage(0, getLangString(_credits == 1 ? 0x402D : 0x402E), _credits); return 1; } int LoLEngine::clickedCompass(Button *button) { if (!(_flagsTable[31] & 0x40)) return 0; if (_compassBroken) { if (characterSays(0x425b, -1, true)) _txt->printMessage(4, "%s", getLangString(0x425b)); } else { _txt->printMessage(0, "%s", getLangString(0x402f + _currentDirection)); } return 1; } int LoLEngine::clickedAutomap(Button *button) { if (!(_flagsTable[31] & 0x10)) return 0; removeInputTop(); displayAutomap(); gui_drawPlayField(); setPaletteBrightness(_screen->getPalette(0), _brightness, _lampEffect); return 1; } int LoLEngine::clickedLamp(Button *button) { if (!(_flagsTable[31] & 0x08)) return 0; if (_itemsInPlay[_itemInHand].itemPropertyIndex == 248) { if (_lampOilStatus >= 100) { _txt->printMessage(0, "%s", getLangString(0x4061)); return 1; } _txt->printMessage(0, "%s", getLangString(0x4062)); deleteItem(_itemInHand); snd_playSoundEffect(181, -1); setHandItem(0); _lampOilStatus += 100; } else { uint16 s = (_lampOilStatus >= 100) ? 0x4060 : ((!_lampOilStatus) ? 0x405c : (_lampOilStatus / 33) + 0x405d); _txt->printMessage(0, getLangString(0x405b), getLangString(s)); } if (_brightness) setPaletteBrightness(_screen->getPalette(0), _brightness, _lampEffect); return 1; } int LoLEngine::clickedStatusIcon(Button *button) { int t = _mouseX - 220; if (t < 0) t = 0; t /= 14; if (t > 2) t = 2; uint16 str = _charStatusFlags[t] + 1; if (str == 0 || str > 3) return 1; _txt->printMessage(0x8002, "%s", getLangString(str == 1 ? 0x424c : (str == 2 ? 0x424e : 0x424d))); return 1; } GUI_LoL::GUI_LoL(LoLEngine *vm) : GUI(vm), _vm(vm), _screen(vm->_screen) { _scrollUpFunctor = BUTTON_FUNCTOR(GUI_LoL, this, &GUI_LoL::scrollUp); _scrollDownFunctor = BUTTON_FUNCTOR(GUI_LoL, this, &GUI_LoL::scrollDown); _redrawButtonFunctor = BUTTON_FUNCTOR(GUI, this, &GUI::redrawButtonCallback); _redrawShadedButtonFunctor = BUTTON_FUNCTOR(GUI, this, &GUI::redrawShadedButtonCallback); _specialProcessButton = _backUpButtonList = 0; _flagsModifier = 0; _mouseClick = 0; _sliderSfx = 11; _buttonListChanged = false; _savegameList = 0; _savegameListSize = 0; } void GUI_LoL::processButton(Button *button) { if (!button) return; if (button->flags & 8) { if (button->flags & 0x10) { // XXX } return; } int entry = button->flags2 & 5; byte val1 = 0, val2 = 0, val3 = 0; const uint8 *dataPtr = 0; Button::Callback callback; if (entry == 1) { val1 = button->data1Val1; dataPtr = button->data1ShapePtr; callback = button->data1Callback; val2 = button->data1Val2; val3 = button->data1Val3; } else if (entry == 4 || entry == 5) { val1 = button->data2Val1; dataPtr = button->data2ShapePtr; callback = button->data2Callback; val2 = button->arg; val3 = button->data2Val3; } else { val1 = button->data0Val1; dataPtr = button->data0ShapePtr; callback = button->data0Callback; val2 = button->data0Val2; val3 = button->data0Val3; } int x = 0, y = 0, x2 = 0, y2 = 0; x = button->x; if (x < 0) x += _screen->getScreenDim(button->dimTableIndex)->w << 3; x += _screen->getScreenDim(button->dimTableIndex)->sx << 3; x2 = x + button->width - 1; y = button->y; if (y < 0) y += _screen->getScreenDim(button->dimTableIndex)->h << 3; y += _screen->getScreenDim(button->dimTableIndex)->sy << 3; y2 = y + button->height - 1; switch (val1 - 1) { case 0: _screen->hideMouse(); _screen->drawShape(_screen->_curPage, dataPtr, x, y, button->dimTableIndex, 0x10); _screen->showMouse(); break; case 1: _screen->hideMouse(); _screen->printText((const char *)dataPtr, x, y, val2, val3); _screen->showMouse(); break; case 3: if (callback) (*callback)(button); break; case 4: _screen->hideMouse(); _screen->drawBox(x, y, x2, y2, val2); _screen->showMouse(); break; case 5: _screen->hideMouse(); _screen->fillRect(x, y, x2, y2, val2, -1, true); _screen->showMouse(); break; default: break; } _screen->updateScreen(); } int GUI_LoL::processButtonList(Button *buttonList, uint16 inputFlag, int8 mouseWheel) { if (!buttonList) return inputFlag & 0x7FFF; if (_backUpButtonList != buttonList || _buttonListChanged) { _specialProcessButton = 0; _flagsModifier = 0; if (_mouseClick == 1) _flagsModifier |= 0x200; if (_mouseClick == 2) _flagsModifier |= 0x2000; _mouseClick = 0; _backUpButtonList = buttonList; _buttonListChanged = false; while (buttonList) { processButton(buttonList); buttonList = buttonList->nextButton; } } int mouseX = _vm->_mouseX; int mouseY = _vm->_mouseY; uint16 flags = 0; if (1/*!_screen_cursorDisable*/) { uint16 inFlags = inputFlag & 0xFF; uint16 temp = 0; // HACK: inFlags == 200 is our left button (up) if (inFlags == 199 || inFlags == 200) temp = 0x100; if (inFlags == 201 || inFlags == 202) temp = 0x1000; if (inputFlag & 0x800) temp <<= 2; flags |= temp; _flagsModifier &= ~((temp & 0x4400) >> 1); _flagsModifier |= (temp & 0x1100) * 2; flags |= _flagsModifier; flags |= (_flagsModifier << 2) ^ 0x8800; } buttonList = _backUpButtonList; if (_specialProcessButton) { buttonList = _specialProcessButton; if (_specialProcessButton->flags & 8) _specialProcessButton = 0; } int returnValue = 0; while (buttonList) { if (buttonList->flags & 8) { buttonList = buttonList->nextButton; continue; } buttonList->flags2 &= ~0x18; buttonList->flags2 |= (buttonList->flags2 & 3) << 3; int x = buttonList->x; if (x < 0) x += _screen->getScreenDim(buttonList->dimTableIndex)->w << 3; x += _screen->getScreenDim(buttonList->dimTableIndex)->sx << 3; int y = buttonList->y; if (y < 0) y += _screen->getScreenDim(buttonList->dimTableIndex)->h; y += _screen->getScreenDim(buttonList->dimTableIndex)->sy; bool progress = false; if (mouseX >= x && mouseY >= y && mouseX <= x+buttonList->width && mouseY <= y+buttonList->height) progress = true; buttonList->flags2 &= ~0x80; uint16 inFlags = inputFlag & 0x7FFF; if (inFlags) { if (buttonList->keyCode == inFlags) { progress = true; flags = buttonList->flags & 0x0F00; buttonList->flags2 |= 0x80; inputFlag = 0; _specialProcessButton = buttonList; } else if (buttonList->keyCode2 == inFlags) { flags = buttonList->flags & 0xF000; if (!flags) flags = buttonList->flags & 0x0F00; progress = true; buttonList->flags2 |= 0x80; inputFlag = 0; _specialProcessButton = buttonList; } } bool unk1 = false; if (mouseWheel && buttonList->mouseWheel == mouseWheel) { progress = true; unk1 = true; } if (!progress) buttonList->flags2 &= ~6; if ((flags & 0x3300) && (buttonList->flags & 4) && progress && (buttonList == _specialProcessButton || !_specialProcessButton)) { buttonList->flags |= 6; if (!_specialProcessButton) _specialProcessButton = buttonList; } else if ((flags & 0x8800) && !(buttonList->flags & 4) && progress) { buttonList->flags2 |= 6; } else { buttonList->flags2 &= ~6; } bool progressSwitch = false; if (!_specialProcessButton) { progressSwitch = progress; } else { if (_specialProcessButton->flags & 0x40) progressSwitch = (_specialProcessButton == buttonList); else progressSwitch = progress; } if (progressSwitch) { if ((flags & 0x1100) && progress && !_specialProcessButton) { inputFlag = 0; _specialProcessButton = buttonList; } if ((buttonList->flags & flags) && (progress || !(buttonList->flags & 1))) { uint16 combinedFlags = (buttonList->flags & flags); combinedFlags = ((combinedFlags & 0xF000) >> 4) | (combinedFlags & 0x0F00); combinedFlags >>= 8; static const uint16 flagTable[] = { 0x000, 0x100, 0x200, 0x100, 0x400, 0x100, 0x400, 0x100, 0x800, 0x100, 0x200, 0x100, 0x400, 0x100, 0x400, 0x100 }; assert(combinedFlags < ARRAYSIZE(flagTable)); switch (flagTable[combinedFlags]) { case 0x400: if (!(buttonList->flags & 1) || ((buttonList->flags & 1) && _specialProcessButton == buttonList)) { buttonList->flags2 ^= 1; returnValue = buttonList->index | 0x8000; unk1 = true; } if (!(buttonList->flags & 4)) { buttonList->flags2 &= ~4; buttonList->flags2 &= ~2; } break; case 0x800: if (!(buttonList->flags & 4)) { buttonList->flags2 |= 4; buttonList->flags2 |= 2; } if (!(buttonList->flags & 1)) unk1 = true; break; case 0x200: if (buttonList->flags & 4) { buttonList->flags2 |= 4; buttonList->flags2 |= 2; } if (!(buttonList->flags & 1)) unk1 = true; break; case 0x100: default: buttonList->flags2 ^= 1; returnValue = buttonList->index | 0x8000; unk1 = true; if (buttonList->flags & 4) { buttonList->flags2 |= 4; buttonList->flags2 |= 2; } _specialProcessButton = buttonList; } } } bool unk2 = false; if ((flags & 0x2200) && progress) { buttonList->flags2 |= 6; if (!(buttonList->flags & 4) && !(buttonList->flags2 & 1)) { unk2 = true; buttonList->flags2 |= 1; } } if ((flags & 0x8800) == 0x8800) { _specialProcessButton = 0; if (!progress || (buttonList->flags & 4)) buttonList->flags2 &= ~6; } if (!progress && buttonList == _specialProcessButton && !(buttonList->flags & 0x40)) _specialProcessButton = 0; if ((buttonList->flags2 & 0x18) != ((buttonList->flags2 & 3) << 3)) processButton(buttonList); if (unk2) buttonList->flags2 &= ~1; if (unk1) { buttonList->flags2 &= 0xFF; buttonList->flags2 |= flags; if (buttonList->buttonCallback) { //_vm->removeInputTop(); if ((*buttonList->buttonCallback.get())(buttonList)) break; } if (buttonList->flags & 0x20) break; } if (_specialProcessButton == buttonList && (buttonList->flags & 0x40)) break; buttonList = buttonList->nextButton; } if (!returnValue) returnValue = inputFlag & 0x7FFF; return returnValue; } int GUI_LoL::redrawButtonCallback(Button *button) { if (!_displayMenu || _vm->gameFlags().use16ColorMode) return 0; _screen->drawBox(button->x + 1, button->y + 1, button->x + button->width - 1, button->y + button->height - 1, 225); return 0; } int GUI_LoL::redrawShadedButtonCallback(Button *button) { if (!_displayMenu || _vm->gameFlags().use16ColorMode) return 0; _screen->drawShadedBox(button->x, button->y, button->x + button->width, button->y + button->height, 223, 227); return 0; } int GUI_LoL::runMenu(Menu &menu) { _currentMenu = &menu; _lastMenu = _currentMenu; _newMenu = 0; _displayMenu = true; _menuResult = 1; _savegameOffset = 0; backupPage0(); const ScreenDim *d = _screen->getScreenDim(8); uint32 textCursorTimer = 0; uint8 textCursorStatus = 1; Screen::FontId of = _screen->setFont(Screen::FID_9_FNT); int wW = _screen->getCharWidth('W'); _screen->setFont(of); int fW = (d->w << 3) - wW; int fC = 0; // LoL doesnt't have default higlighted items. No item should be // highlighted when entering a new menu. // Instead, the respevtive struct entry is used to determine whether // a menu has scroll buttons or slider bars. uint8 hasSpecialButtons = 0; _savegameListUpdateNeeded = true; while (_displayMenu) { _vm->_mouseX = _vm->_mouseY = 0; if (_currentMenu == &_loadMenu || _currentMenu == &_saveMenu || _currentMenu == &_deleteMenu) { updateSavegameList(); setupSaveMenuSlots(*_currentMenu, 4); } hasSpecialButtons = _currentMenu->highlightedItem; _currentMenu->highlightedItem = 255; if (_currentMenu == &_gameOptions) { char *s = (char *)_vm->_tempBuffer5120; strncpy(s, _vm->getLangString(0x406f + _vm->_monsterDifficulty), 30); s[29] = 0; _currentMenu->item[_vm->gameFlags().isTalkie ? 0 : 2].itemString = s; s += (strlen(s) + 1); strncpy(s, _vm->getLangString(_vm->_smoothScrollingEnabled ? 0x4068 : 0x4069), 30); s[29] = 0; _currentMenu->item[_vm->gameFlags().isTalkie ? 1 : 3].itemString = s; s += (strlen(s) + 1); strncpy(s, _vm->getLangString(_vm->_floatingCursorsEnabled ? 0x4068 : 0x4069), 30); s[29] = 0; _currentMenu->item[_vm->gameFlags().isTalkie ? 2 : 4].itemString = s; s += (strlen(s) + 1); if (_vm->gameFlags().isTalkie) { strncpy(s, _vm->getLangString(0x42d6 + _vm->_lang), 30); s[29] = 0; _currentMenu->item[3].itemString = s; s += (strlen(s) + 1); strncpy(s, _vm->getLangString(_vm->textEnabled() ? 0x4068 : 0x4069), 30); s[29] = 0; _currentMenu->item[4].itemString = s; s += (strlen(s) + 1); } else { strncpy(s, _vm->getLangString(_vm->_configMusic ? 0x4068 : 0x4069), 30); s[29] = 0; _currentMenu->item[0].itemString = s; s += (strlen(s) + 1); strncpy(s, _vm->getLangString(_vm->_configSounds ? 0x4068 : 0x4069), 30); s[29] = 0; _currentMenu->item[1].itemString = s; s += (strlen(s) + 1); } } if (hasSpecialButtons == 1) { if (_savegameOffset == 0) { _scrollUpButton.data0ShapePtr = _scrollUpButton.data1ShapePtr = _scrollUpButton.data2ShapePtr = 0; } else { _scrollUpButton.data0ShapePtr = _vm->_gameShapes[17]; _scrollUpButton.data1ShapePtr = _scrollUpButton.data2ShapePtr = _vm->_gameShapes[19]; } int slotOffs = (_currentMenu == &_saveMenu) ? 1 : 0; if (((uint)_savegameOffset == _saveSlots.size() - (4 - slotOffs)) || _saveSlots.size() < (uint)(5 - slotOffs)) { _scrollDownButton.data0ShapePtr = _scrollDownButton.data1ShapePtr = _scrollDownButton.data2ShapePtr = 0; } else { _scrollDownButton.data0ShapePtr = _vm->_gameShapes[18]; _scrollDownButton.data1ShapePtr = _scrollDownButton.data2ShapePtr = _vm->_gameShapes[20]; } } for (uint i = 0; i < _currentMenu->numberOfItems; ++i) { _menuButtons[i].data0Val1 = _menuButtons[i].data1Val1 = _menuButtons[i].data2Val1 = 4; _menuButtons[i].data0Callback = _redrawShadedButtonFunctor; _menuButtons[i].data1Callback = _menuButtons[i].data2Callback = _redrawButtonFunctor; _menuButtons[i].flags = 0x4487; _menuButtons[i].flags2 = 0; } initMenu(*_currentMenu); if (_currentMenu == &_loadMenu || _currentMenu == &_deleteMenu) { if (_saveSlots.begin() == _saveSlots.end()) // "no savegames to load" message _screen->fprintString("%s", _currentMenu->x + _currentMenu->width / 2, _currentMenu->y + 42, 204, 0, 9, _vm->getLangString(0x4009)); } if (hasSpecialButtons == 2) { static const uint8 oX[] = { 0, 10, 124 }; static const uint8 oW[] = { 10, 114, 10 }; for (int i = 1; i < 4; ++i) { int tX = _currentMenu->x + _currentMenu->item[i].x; int tY = _currentMenu->y + _currentMenu->item[i].y; for (int ii = 0; ii < 3; ++ii) { Button *b = getButtonListData() + 1 + (i - 1) * 3 + ii; b->nextButton = 0; b->data0Val2 = b->data1Val2 = b->data2Val2 = 0xfe; b->data0Val3 = b->data1Val3 = b->data2Val3 = 0x01; b->index = ii; b->keyCode = b->keyCode2 = 0; b->x = tX + oX[ii]; b->y = tY; b->width = oW[ii]; b->height = _currentMenu->item[i].height; b->data0Val1 = b->data1Val1 = b->data2Val1 = 0; b->flags = (ii == 1) ? 0x6606 : 0x4406; b->dimTableIndex = 0; b->buttonCallback = _currentMenu->item[i].callback; b->arg = _currentMenu->item[i].itemId; _menuButtonList = addButtonToList(_menuButtonList, b); processButton(b); updateButton(b); } _currentMenu->item[i].labelX = _currentMenu->item[i].x - 5; _currentMenu->item[i].labelY = _currentMenu->item[i].y + 3; printMenuText(getMenuItemLabel(_currentMenu->item[i]), _currentMenu->x + _currentMenu->item[i].labelX, _currentMenu->y + _currentMenu->item[i].labelY, _currentMenu->item[i].textColor, 0, 10); int volume = _vm->getVolume((KyraEngine_v1::kVolumeEntry)(i - 1)); _screen->drawShape(_screen->_curPage, _vm->_gameShapes[85], tX , tY, 0, 0x10); _screen->drawShape(_screen->_curPage, _vm->_gameShapes[87], tX + 2 + oX[1], tY, 0, 0x10); _screen->drawShape(_screen->_curPage, _vm->_gameShapes[86], tX + oX[1] + volume, tY, 0, 0x10); } _screen->updateScreen(); } if (_currentMenu == &_mainMenu && !_vm->gameFlags().use16ColorMode) { Screen::FontId f = _screen->setFont(Screen::FID_6_FNT); _screen->fprintString("%s", menu.x + 8, menu.y + menu.height - 12, 204, 0, 8, gScummVMVersion); _screen->setFont(f); _screen->updateScreen(); } if (_currentMenu == &_savenameMenu) { int mx = (d->sx << 3) - 1; int my = d->sy - 1; int mw = (d->w << 3) + 1; int mh = d->h + 1; if (_vm->gameFlags().use16ColorMode) { _screen->drawShadedBox(mx, my, mx + mw + 1, my + mh + 1, 0xdd, 0xff); _screen->drawLine(true, mx + mw + 1, my, mh + 1, 0xcc); _screen->drawLine(false, mx, my + mh + 1, mw + 2, 0xcc); } else { _screen->drawShadedBox(mx, my, mx + mw, my + mh, 227, 223); } int pg = _screen->setCurPage(0); _vm->_txt->clearDim(8); textCursorTimer = 0; textCursorStatus = 0; Screen::FontId f = _screen->setFont(Screen::FID_9_FNT); fC = _screen->getTextWidth(_saveDescription); while (fC >= fW) { _saveDescription[strlen(_saveDescription) - 1] = 0; fC = _screen->getTextWidth(_saveDescription); } _screen->fprintString("%s", (d->sx << 3), d->sy + 2, d->unk8, d->unkA, 0, _saveDescription); f = _screen->setFont(f); _screen->fillRect((d->sx << 3) + fC, d->sy, (d->sx << 3) + fC + wW, d->sy + d->h - (_vm->gameFlags().use16ColorMode ? 2 : 1), d->unk8, 0); _screen->setCurPage(pg); } while (!_newMenu && _displayMenu) { processHighlights(*_currentMenu); if (_currentMenu == &_savenameMenu) { if (textCursorTimer <= _vm->_system->getMillis()) { Screen::FontId f = _screen->setFont(Screen::FID_9_FNT); fC = _screen->getTextWidth(_saveDescription); textCursorStatus ^= 1; textCursorTimer = _vm->_system->getMillis() + 20 * _vm->_tickLength; _screen->fillRect((d->sx << 3) + fC, d->sy, (d->sx << 3) + fC + wW, d->sy + d->h - (_vm->gameFlags().use16ColorMode ? 2 : 1), textCursorStatus ? d->unk8 : d->unkA, 0); _screen->updateScreen(); f = _screen->setFont(f); } } if (getInput()) { if (!_newMenu) { if (_currentMenu == &_savenameMenu) { Screen::FontId f = _screen->setFont(Screen::FID_9_FNT); _screen->fillRect((d->sx << 3) + fC, d->sy, (d->sx << 3) + fC + wW, d->sy + d->h - (_vm->gameFlags().use16ColorMode ? 2 : 1), d->unkA, 0); fC = _screen->getTextWidth(_saveDescription); while (fC >= fW) { _saveDescription[strlen(_saveDescription) - 1] = 0; fC = _screen->getTextWidth(_saveDescription); } _screen->fprintString("%s", (d->sx << 3), d->sy + 2, d->unk8, d->unkA, 0, _saveDescription); _screen->fillRect((d->sx << 3) + fC, d->sy, (d->sx << 3) + fC + wW, d->sy + d->h - (_vm->gameFlags().use16ColorMode ? 2 : 1), textCursorStatus ? d->unk8 : d->unkA, 0); f = _screen->setFont(f); textCursorTimer = 0; textCursorStatus = 0; } else { _newMenu = (_currentMenu != &_audioOptions) ? _currentMenu : 0; } } else { _lastMenu = _menuResult == -1 ? _lastMenu : _currentMenu; } } if (!_menuResult) _displayMenu = false; } if (_newMenu != _currentMenu || !_displayMenu) restorePage0(); _currentMenu->highlightedItem = hasSpecialButtons; if (_newMenu) _currentMenu = _newMenu; _newMenu = 0; } if (_savegameList) { for (int i = 0; i < _savegameListSize; i++) delete[] _savegameList[i]; delete[] _savegameList; _savegameList = 0; } return _menuResult; } void GUI_LoL::createScreenThumbnail(Graphics::Surface &dst) { uint8 *screenPal = new uint8[768]; _screen->getRealPalette(1, screenPal); if (_vm->gameFlags().platform == Common::kPlatformPC98) { uint8 *screen = new uint8[Screen::SCREEN_W * Screen::SCREEN_H]; assert(screen); _screen->copyRegionToBuffer(7, 0, 0, 320, 200, screen); Screen_LoL::convertPC98Gfx(screen, Screen::SCREEN_W, Screen::SCREEN_H, Screen::SCREEN_W); ::createThumbnail(&dst, screen, Screen::SCREEN_W, Screen::SCREEN_H, screenPal); delete[] screen; } else { ::createThumbnail(&dst, _screen->getCPagePtr(7), Screen::SCREEN_W, Screen::SCREEN_H, screenPal); } delete[] screenPal; } void GUI_LoL::backupPage0() { _screen->copyPage(0, 7); } void GUI_LoL::restorePage0() { _screen->copyPage(7, 0); _screen->updateScreen(); } void GUI_LoL::setupSaveMenuSlots(Menu &menu, int num) { char *s = (char *)_vm->_tempBuffer5120; for (int i = 0; i < num; ++i) { menu.item[i].saveSlot = -1; menu.item[i].enabled = false; } int startSlot = 0; int slotOffs = 0; if (&menu == &_saveMenu) { if (_savegameOffset == 0) startSlot = 1; slotOffs = 1; } for (int i = startSlot; i < num && _savegameOffset + i - slotOffs < _savegameListSize; ++i) { if (_savegameList[_saveSlots[i + _savegameOffset - slotOffs]]) { strncpy(s, _savegameList[_saveSlots[i + _savegameOffset - slotOffs]], 80); s[79] = 0; menu.item[i].itemString = s; s += (strlen(s) + 1); menu.item[i].saveSlot = _saveSlots[i + _savegameOffset - slotOffs]; menu.item[i].enabled = true; } } if (_savegameOffset == 0) { if (&menu == &_saveMenu) { strcpy(s, _vm->getLangString(0x4010)); menu.item[0].itemString = s; menu.item[0].saveSlot = -3; menu.item[0].enabled = true; } } } void GUI_LoL::updateSavegameList() { if (!_savegameListUpdateNeeded) return; _savegameListUpdateNeeded = false; if (_savegameList) { for (int i = 0; i < _savegameListSize; i++) delete[] _savegameList[i]; delete[] _savegameList; } updateSaveList(true); _savegameListSize = _saveSlots.size(); if (_savegameListSize) { Common::sort(_saveSlots.begin(), _saveSlots.end(), Common::Greater()); KyraEngine_v1::SaveHeader header; Common::InSaveFile *in; _savegameList = new char*[_savegameListSize]; for (int i = 0; i < _savegameListSize; i++) { in = _vm->openSaveForReading(_vm->getSavegameFilename(i), header); if (in) { _savegameList[i] = new char[header.description.size() + 1]; strncpy(_savegameList[i], header.description.c_str(), header.description.size() + 1); Util::convertISOToDOS(_savegameList[i]); delete in; } else { _savegameList[i] = 0; error("GUI_LoL::updateSavegameList(): Unexpected missing save file for slot: %d.", i); } } } else { _savegameList = 0; } } void GUI_LoL::printMenuText(const char *str, int x, int y, uint8 c0, uint8 c1, uint8 flags) { _screen->fprintString("%s", x, y, c0, c1, _vm->gameFlags().use16ColorMode ? (flags & 3) : flags , str); } int GUI_LoL::getMenuCenterStringX(const char *str, int x1, int x2) { if (!str) return 0; int strWidth = _screen->getTextWidth(str); int w = x2 - x1 + 1; return x1 + (w - strWidth) / 2; } int GUI_LoL::getInput() { if (!_displayMenu) return 0; Common::Point p = _vm->getMousePos(); _vm->_mouseX = p.x; _vm->_mouseY = p.y; if (_currentMenu == &_savenameMenu) { _vm->updateInput(); for (Common::List::const_iterator evt = _vm->_eventList.begin(); evt != _vm->_eventList.end(); ++evt) { if (evt->event.type == Common::EVENT_KEYDOWN) _keyPressed = evt->event.kbd; } } int inputFlag = _vm->checkInput(_menuButtonList); if (_currentMenu == &_savenameMenu && _keyPressed.ascii){ char inputKey = _keyPressed.ascii; Util::convertISOToDOS(inputKey); if ((uint8)inputKey > 31 && (uint8)inputKey < (_vm->gameFlags().lang == Common::JA_JPN ? 128 : 226)) { _saveDescription[strlen(_saveDescription) + 1] = 0; _saveDescription[strlen(_saveDescription)] = inputKey; inputFlag |= 0x8000; } else if (_keyPressed.keycode == Common::KEYCODE_BACKSPACE && strlen(_saveDescription)) { _saveDescription[strlen(_saveDescription) - 1] = 0; inputFlag |= 0x8000; } } _vm->removeInputTop(); _keyPressed.reset(); if (_vm->shouldQuit()) _displayMenu = false; _vm->delay(8); return inputFlag & 0x8000 ? 1 : 0; } int GUI_LoL::clickedMainMenu(Button *button) { updateMenuButton(button); switch (button->arg) { case 0x4001: _savegameOffset = 0; _newMenu = &_loadMenu; break; case 0x4002: _savegameOffset = 0; _newMenu = &_saveMenu; break; case 0x4003: _savegameOffset = 0; _newMenu = &_deleteMenu; break; case 0x4004: _newMenu = &_gameOptions; break; case 0x42D9: _newMenu = &_audioOptions; break; case 0x4006: _choiceMenu.menuNameId = 0x400a; _newMenu = &_choiceMenu; break; case 0x4005: _displayMenu = false; break; } return 1; } int GUI_LoL::clickedLoadMenu(Button *button) { updateMenuButton(button); if (button->arg == 0x4011) { if (_currentMenu != _lastMenu) _newMenu = _lastMenu; else _menuResult = 0; return 1; } int16 s = (int16)button->arg; _vm->_gameToLoad = _loadMenu.item[-s - 2].saveSlot; _displayMenu = false; return 1; } int GUI_LoL::clickedSaveMenu(Button *button) { updateMenuButton(button); if (button->arg == 0x4011) { _newMenu = &_mainMenu; return 1; } _newMenu = &_savenameMenu; int16 s = (int16)button->arg; _menuResult = _saveMenu.item[-s - 2].saveSlot + 1; _saveDescription = (char*)_vm->_tempBuffer5120 + 1000; _saveDescription[0] = 0; if (_saveMenu.item[-s - 2].saveSlot != -3) strcpy(_saveDescription, _saveMenu.item[-s - 2].itemString); return 1; } int GUI_LoL::clickedDeleteMenu(Button *button) { updateMenuButton(button); if (button->arg == 0x4011) { _newMenu = &_mainMenu; return 1; } _choiceMenu.menuNameId = 0x400b; _newMenu = &_choiceMenu; int16 s = (int16)button->arg; _menuResult = _deleteMenu.item[-s - 2].saveSlot + 1; return 1; } int GUI_LoL::clickedOptionsMenu(Button *button) { updateMenuButton(button); switch (button->arg) { case 0xfff9: _vm->_configMusic ^= 1; _vm->sound()->enableMusic(_vm->_configMusic); if (_vm->_configMusic) _vm->snd_playTrack(_vm->_curMusicTheme); else _vm->_sound->beginFadeOut(); break; case 0xfff8: _vm->_configSounds ^= true; _vm->sound()->enableSFX(_vm->_configSounds); break; case 0xfff7: _vm->_monsterDifficulty = (_vm->_monsterDifficulty + 1) % 3; break; case 0xfff6: _vm->_smoothScrollingEnabled ^= true; break; case 0xfff5: _vm->_floatingCursorsEnabled ^= true; break; case 0xfff4: _vm->_lang = (_vm->_lang + 1) % 3; break; case 0xfff3: _vm->_configVoice ^= 3; break; case 0x4072: char filename[13]; snprintf(filename, sizeof(filename), "LEVEL%02d.%s", _vm->_currentLevel, _vm->_languageExt[_vm->_lang]); delete[] _vm->_levelLangFile; _vm->_levelLangFile = _vm->resource()->fileData(filename, 0); snprintf(filename, sizeof(filename), "LANDS.%s", _vm->_languageExt[_vm->_lang]); delete[] _vm->_landsFile; _vm->_landsFile = _vm->resource()->fileData(filename, 0); _newMenu = _lastMenu; break; } return 1; } int GUI_LoL::clickedAudioMenu(Button *button) { updateMenuButton(button); if (button->arg == 0x4072) { _newMenu = _lastMenu; return 1; } int tX = button->x; const int oldVolume = _vm->getVolume((KyraEngine_v1::kVolumeEntry)(button->arg - 3)); int newVolume = oldVolume; if (button->index == 0) { newVolume -= 10; tX += 10; } else if (button->index == 1) { newVolume = _vm->_mouseX - (tX + 7); } else if (button->index == 2) { newVolume += 10; tX -= 114; } newVolume = CLIP(newVolume, 2, 102); if (newVolume == oldVolume) { _screen->updateScreen(); return 0; } _screen->drawShape(0, _vm->_gameShapes[87], tX + oldVolume, button->y, 0, 0x10); // Temporary HACK const int volumeDrawX = _vm->convertVolumeFromMixer(_vm->convertVolumeToMixer(newVolume)); _screen->drawShape(0, _vm->_gameShapes[86], tX + volumeDrawX, button->y, 0, 0x10); _screen->updateScreen(); _vm->snd_stopSpeech(0); _vm->setVolume((KyraEngine_v1::kVolumeEntry)(button->arg - 3), newVolume); if (newVolume) { if (button->arg == 4) { _vm->snd_playSoundEffect(_sliderSfx, -1); int16 vocIndex = (int16)READ_LE_UINT16(&_vm->_ingameSoundIndex[_sliderSfx * 2]); do { ++_sliderSfx; if (_sliderSfx < 47) _sliderSfx++; if (vocIndex == 199) _sliderSfx = 11; vocIndex = (int16)READ_LE_UINT16(&_vm->_ingameSoundIndex[_sliderSfx * 2]); if (vocIndex == -1) continue; if (!scumm_stricmp(_vm->_ingameSoundList[vocIndex], "EMPTY")) continue; break; } while (1); } else if (button->arg == 5) { _vm->_lastSpeechId = -1; _vm->snd_playCharacterSpeech(0x42e0, 0, 0); } } return 1; } int GUI_LoL::clickedDeathMenu(Button *button) { updateMenuButton(button); if (button->arg == _deathMenu.item[0].itemId) { _vm->quitGame(); } else if (button->arg == _deathMenu.item[1].itemId) { _newMenu = &_loadMenu; } return 1; } int GUI_LoL::clickedSavenameMenu(Button *button) { updateMenuButton(button); if (button->arg == _savenameMenu.item[0].itemId) { Util::convertDOSToISO(_saveDescription); int slot = _menuResult == -2 ? getNextSavegameSlot() : _menuResult - 1; Graphics::Surface thumb; createScreenThumbnail(thumb); _vm->saveGameState(slot, _saveDescription, &thumb); thumb.free(); _displayMenu = false; } else if (button->arg == _savenameMenu.item[1].itemId) { _newMenu = &_saveMenu; } return 1; } int GUI_LoL::clickedChoiceMenu(Button *button) { updateMenuButton(button); if (button->arg == _choiceMenu.item[0].itemId) { if (_lastMenu == &_mainMenu) { _vm->quitGame(); } else if (_lastMenu == &_deleteMenu) { _vm->_saveFileMan->removeSavefile(_vm->getSavegameFilename(_menuResult - 1)); Common::Array::iterator i = Common::find(_saveSlots.begin(), _saveSlots.end(), _menuResult - 1); while (i != _saveSlots.begin()) { --i; // not rename quicksave slot filenames if (*i >= 990) break; Common::String oldName = _vm->getSavegameFilename(*i); Common::String newName = _vm->getSavegameFilename(*i-1); _vm->_saveFileMan->renameSavefile(oldName, newName); } _newMenu = &_mainMenu; _savegameListUpdateNeeded = true; } } else if (button->arg == _choiceMenu.item[1].itemId) { _newMenu = &_mainMenu; } return 1; } int GUI_LoL::scrollUp(Button *button) { if (!_scrollUpButton.data0ShapePtr) return 0; updateButton(button); if (_savegameOffset > 0) { _savegameOffset--; _newMenu = _currentMenu; _menuResult = -1; } return 1; } int GUI_LoL::scrollDown(Button *button) { if (!_scrollDownButton.data0ShapePtr) return 0; updateButton(button); if ((uint)_savegameOffset < _saveSlots.size() - (_currentMenu == &_saveMenu ? 3 : 4)) { _savegameOffset++; _newMenu = _currentMenu; _menuResult = -1; } return 1; } const char *GUI_LoL::getMenuTitle(const Menu &menu) { if (!menu.menuNameId) return 0; return _vm->getLangString(menu.menuNameId); } const char *GUI_LoL::getMenuItemTitle(const MenuItem &menuItem) { if (menuItem.itemId & 0x8000 && menuItem.itemString) return menuItem.itemString; else if (menuItem.itemId & 0x8000 || !menuItem.itemId) return 0; return _vm->getLangString(menuItem.itemId); } const char *GUI_LoL::getMenuItemLabel(const MenuItem &menuItem) { if (menuItem.labelId & 0x8000 && menuItem.labelString) return menuItem.labelString; else if (menuItem.labelId & 0x8000 || !menuItem.labelId) return 0; return _vm->getLangString(menuItem.labelId); } } // End of namespace Kyra #endif // ENABLE_LOL