/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "common/system.h" #include "graphics/cursorman.h" #include "graphics/palette.h" #include "gui/message.h" #include "supernova/screen.h" #include "supernova/supernova.h" #include "supernova/state.h" namespace Supernova { bool GameManager::serialize(Common::WriteStream *out) { if (out->err()) return false; // GameState out->writeSint32LE(_state._time); out->writeSint32LE(_state._timeSleep); out->writeSint32LE(_state._timeAlarm); out->writeSint32LE(_state._eventTime); out->writeSint32LE(_state._eventCallback); out->writeSint32LE(_state._arrivalDaysLeft); out->writeSint32LE(_state._shipEnergyDaysLeft); out->writeSint32LE(_state._landingModuleEnergyDaysLeft); out->writeUint16LE(_state._greatFlag); out->writeSint16LE(_state._timeRobot); out->writeSint16LE(_state._money); out->writeByte(_state._coins); out->writeByte(_state._shoes); out->writeByte(_state._origin); out->writeByte(_state._destination); out->writeByte(_state._language); out->writeByte(_state._corridorSearch); out->writeByte(_state._alarmOn); out->writeByte(_state._terminalStripConnected); out->writeByte(_state._terminalStripWire); out->writeByte(_state._cableConnected); out->writeByte(_state._powerOff); out->writeByte(_state._dream); for (int i = 0; i < 4; i++) out->writeByte(_state._nameSeen[i]); out->writeByte(_state._playerHidden); // Inventory out->writeSint32LE(_inventory.getSize()); out->writeSint32LE(_inventoryScroll); for (int i = 0; i < _inventory.getSize(); ++i) { Object *objectStateBegin = _rooms[_inventory.get(i)->_roomId]->getObject(0); byte objectIndex = _inventory.get(i) - objectStateBegin; out->writeSint32LE(_inventory.get(i)->_roomId); out->writeSint32LE(objectIndex); } // Rooms out->writeByte(_currentRoom->getId()); for (int i = 0; i < NUMROOMS; ++i) { _rooms[i]->serialize(out); } return !out->err(); } bool GameManager::deserialize(Common::ReadStream *in, int version) { if (in->err()) return false; // GameState _state._time = in->readSint32LE(); _state._timeSleep = in->readSint32LE(); _state._timeAlarm = in->readSint32LE(); _state._eventTime = in->readSint32LE(); if (version >= 4) _state._eventCallback = (EventFunction)in->readSint32LE(); else _state._eventCallback = kNoFn; _state._arrivalDaysLeft = in->readSint32LE(); _state._shipEnergyDaysLeft = in->readSint32LE(); _state._landingModuleEnergyDaysLeft = in->readSint32LE(); _state._greatFlag = in->readUint16LE(); _state._timeRobot = in->readSint16LE(); _state._money = in->readSint16LE(); _vm->setGameString(kStringInventoryMoney, Common::String::format("%d Xa", _state._money)); _state._coins = in->readByte(); _state._shoes = in->readByte(); if (version >= 6) _state._origin = in->readByte(); else _state._origin = 0; _state._destination = in->readByte(); _state._language = in->readByte(); _state._corridorSearch = in->readByte(); _state._alarmOn = in->readByte(); _state._terminalStripConnected = in->readByte(); _state._terminalStripWire = in->readByte(); _state._cableConnected = in->readByte(); _state._powerOff = in->readByte(); _state._dream = in->readByte(); for (int i = 0; i < 4; i++) { if (version >= 7) _state._nameSeen[i] = in->readByte(); else _state._nameSeen[i] = false; } if (version >= 8) _state._playerHidden = in->readByte(); else _state._playerHidden = false; _oldTime = g_system->getMillis(); // Inventory int inventorySize = in->readSint32LE(); _inventoryScroll = in->readSint32LE(); _inventory.clear(); for (int i = 0; i < inventorySize; ++i) { RoomId objectRoom = static_cast(in->readSint32LE()); int objectIndex = in->readSint32LE(); _inventory.add(*_rooms[objectRoom]->getObject(objectIndex)); } // Rooms RoomId curRoomId = static_cast(in->readByte()); for (int i = 0; i < NUMROOMS; ++i) { _rooms[i]->deserialize(in, version); } changeRoom(curRoomId); // Some additional variables _guiEnabled = true; _animationEnabled = true; return !in->err(); } void Inventory::add(Object &obj) { if (_numObjects < kMaxCarry) { _inventory[_numObjects++] = &obj; obj.setProperty(CARRIED); } if (getSize() > _inventoryScroll + 8) { _inventoryScroll = getSize() - 8; _inventoryScroll += _inventoryScroll % 2; } } void Inventory::remove(Object &obj) { for (int i = 0; i < _numObjects; ++i) { if (_inventory[i] == &obj) { if (_inventoryScroll >= 2 && getSize() % 2) _inventoryScroll -= 2; --_numObjects; while (i < _numObjects) { _inventory[i] = _inventory[i + 1]; ++i; } obj.disableProperty(CARRIED); } } } void Inventory::clear() { for (int i = 0; i < _numObjects; ++i) _inventory[i]->disableProperty(CARRIED); _numObjects = 0; _inventoryScroll = 0; } Object *Inventory::get(int index) const { if (index < _numObjects) return _inventory[index]; return _nullObject; } Object *Inventory::get(ObjectId id) const { for (int i = 0; i < _numObjects; ++i) { if (_inventory[i]->_id == id) return _inventory[i]; } return _nullObject; } GuiElement::GuiElement() : _isHighlighted(false) , _bgColorNormal(kColorWhite25) , _bgColorHighlighted(kColorWhite44) , _bgColor(kColorWhite25) , _textColorNormal(kColorGreen) , _textColorHighlighted(kColorLightGreen) , _textColor(kColorGreen) { memset(_text, 0, sizeof(_text)); } void GuiElement::setText(const char *text) { strncpy(_text, text, sizeof(_text) - 1); } void GuiElement::setTextPosition(int x, int y) { _textPosition = Common::Point(x, y); } void GuiElement::setSize(int x1, int y1, int x2, int y2) { this->left = x1; this->top = y1; this->right = x2; this->bottom = y2; _textPosition = Common::Point(x1 + 1, y1 + 1); } void GuiElement::setColor(int bgColor, int textColor, int bgColorHighlighted, int textColorHightlighted) { _bgColor = bgColor; _textColor = textColor; _bgColorNormal = bgColor; _textColorNormal = textColor; _bgColorHighlighted = bgColorHighlighted; _textColorHighlighted = textColorHightlighted; } void GuiElement::setHighlight(bool isHighlighted_) { if (isHighlighted_) { _bgColor = _bgColorHighlighted; _textColor = _textColorHighlighted; } else { _bgColor = _bgColorNormal; _textColor = _textColorNormal; } } // Used by Look Watch (when it's fixed). Do not remove. static Common::String timeToString(int msec) { char s[9] = " 0:00:00"; msec /= 1000; s[7] = msec % 10 + '0'; msec /= 10; s[6] = msec % 6 + '0'; msec /= 6; s[4] = msec % 10 + '0'; msec /= 10; s[3] = msec % 6 + '0'; msec /= 6; s[1] = msec % 10 + '0'; msec /= 10; if (msec) s[0] = msec + '0'; return Common::String(s); } StringId GameManager::guiCommands[] = { kStringCommandGo, kStringCommandLook, kStringCommandTake, kStringCommandOpen, kStringCommandClose, kStringCommandPress, kStringCommandPull, kStringCommandUse, kStringCommandTalk, kStringCommandGive }; StringId GameManager::guiStatusCommands[] = { kStringStatusCommandGo, kStringStatusCommandLook, kStringStatusCommandTake, kStringStatusCommandOpen, kStringStatusCommandClose, kStringStatusCommandPress, kStringStatusCommandPull, kStringStatusCommandUse, kStringStatusCommandTalk, kStringStatusCommandGive }; GameManager::GameManager(SupernovaEngine *vm, Sound *sound) : _inventory(&_nullObject, _inventoryScroll) , _vm(vm) , _sound(sound) { initRooms(); changeRoom(INTRO); initState(); initGui(); } GameManager::~GameManager() { destroyRooms(); } void GameManager::destroyRooms() { delete _rooms[INTRO]; delete _rooms[CORRIDOR]; delete _rooms[HALL]; delete _rooms[SLEEP]; delete _rooms[COCKPIT]; delete _rooms[AIRLOCK]; delete _rooms[HOLD]; delete _rooms[LANDINGMODULE]; delete _rooms[GENERATOR]; delete _rooms[OUTSIDE]; delete _rooms[CABIN_R1]; delete _rooms[CABIN_R2]; delete _rooms[CABIN_R3]; delete _rooms[CABIN_L1]; delete _rooms[CABIN_L2]; delete _rooms[CABIN_L3]; delete _rooms[BATHROOM]; delete _rooms[ROCKS]; delete _rooms[CAVE]; delete _rooms[MEETUP]; delete _rooms[ENTRANCE]; delete _rooms[REST]; delete _rooms[ROGER]; delete _rooms[GLIDER]; delete _rooms[MEETUP2]; delete _rooms[MEETUP3]; delete _rooms[CELL]; delete _rooms[CORRIDOR1]; delete _rooms[CORRIDOR2]; delete _rooms[CORRIDOR3]; delete _rooms[CORRIDOR4]; delete _rooms[CORRIDOR5]; delete _rooms[CORRIDOR6]; delete _rooms[CORRIDOR7]; delete _rooms[CORRIDOR8]; delete _rooms[CORRIDOR9]; delete _rooms[BCORRIDOR]; delete _rooms[GUARD]; delete _rooms[GUARD3]; delete _rooms[OFFICE_L1]; delete _rooms[OFFICE_L2]; delete _rooms[OFFICE_R1]; delete _rooms[OFFICE_R2]; delete _rooms[OFFICE_L]; delete _rooms[ELEVATOR]; delete _rooms[STATION]; delete _rooms[SIGN]; delete _rooms[OUTRO]; } void GameManager::initState() { _currentInputObject = &_nullObject; _inputObject[0] = &_nullObject; _inputObject[1] = &_nullObject; _inputVerb = ACTION_WALK; _processInput = false; _guiEnabled = true; _animationEnabled = true; _roomBrightness = 255; _mouseClicked = false; _keyPressed = false; _mouseX = -1; _mouseY = -1; _mouseField = -1; _inventoryScroll = 0; _oldTime = g_system->getMillis(); _timerPaused = 0; _timePaused = false; _messageDuration = 0; _animationTimer = 0; _currentSentence = -1; for (int i = 0 ; i < 6 ; ++i) { _sentenceNumber[i] = -1; _texts[i] = kNoString; _rows[i] = 0; _rowsStart[i] = 0; } _state._time = ticksToMsec(916364); // 2 pm _state._timeSleep = 0; _state._timeAlarm = ticksToMsec(458182); // 7 am _state._eventTime = kMaxTimerValue; _state._eventCallback = kNoFn; _state._arrivalDaysLeft = 2840; _state._shipEnergyDaysLeft = 2135; _state._landingModuleEnergyDaysLeft = 923; _state._greatFlag = 0; _state._timeRobot = 0; _state._money = 0; _state._coins = 0; _state._shoes = 0; _state._origin = 0; _state._destination = 255; _state._language = 0; _state._corridorSearch = false; _state._alarmOn = false; _state._terminalStripConnected = false; _state._terminalStripWire = false; _state._cableConnected = false; _state._powerOff = false; _state._dream = false; _prevImgId = 0; } void GameManager::initRooms() { _rooms[INTRO] = new Intro(_vm, this); _rooms[CORRIDOR] = new ShipCorridor(_vm, this); _rooms[HALL] = new ShipHall(_vm, this); _rooms[SLEEP] = new ShipSleepCabin(_vm, this); _rooms[COCKPIT] = new ShipCockpit(_vm, this); _rooms[AIRLOCK] = new ShipAirlock(_vm, this); _rooms[HOLD] = new ShipHold(_vm, this); _rooms[LANDINGMODULE] = new ShipLandingModule(_vm, this); _rooms[GENERATOR] = new ShipGenerator(_vm, this); _rooms[OUTSIDE] = new ShipOuterSpace(_vm, this); _rooms[CABIN_R1] = new ShipCabinR1(_vm, this); _rooms[CABIN_R2] = new ShipCabinR2(_vm, this); _rooms[CABIN_R3] = new ShipCabinR3(_vm, this); _rooms[CABIN_L1] = new ShipCabinL1(_vm, this); _rooms[CABIN_L2] = new ShipCabinL2(_vm, this); _rooms[CABIN_L3] = new ShipCabinL3(_vm, this); _rooms[BATHROOM] = new ShipCabinBathroom(_vm, this); _rooms[ROCKS] = new ArsanoRocks(_vm, this); _rooms[CAVE] = new ArsanoCave(_vm, this); _rooms[MEETUP] = new ArsanoMeetup(_vm, this); _rooms[ENTRANCE] = new ArsanoEntrance(_vm, this); _rooms[REST] = new ArsanoRemaining(_vm, this); _rooms[ROGER] = new ArsanoRoger(_vm, this); _rooms[GLIDER] = new ArsanoGlider(_vm, this); _rooms[MEETUP2] = new ArsanoMeetup2(_vm, this); _rooms[MEETUP3] = new ArsanoMeetup3(_vm, this); _rooms[CELL] = new AxacussCell(_vm, this); _rooms[CORRIDOR1] = new AxacussCorridor1(_vm, this); _rooms[CORRIDOR2] = new AxacussCorridor2(_vm, this); _rooms[CORRIDOR3] = new AxacussCorridor3(_vm, this); _rooms[CORRIDOR4] = new AxacussCorridor4(_vm, this); _rooms[CORRIDOR5] = new AxacussCorridor5(_vm, this); _rooms[CORRIDOR6] = new AxacussCorridor6(_vm, this); _rooms[CORRIDOR7] = new AxacussCorridor7(_vm, this); _rooms[CORRIDOR8] = new AxacussCorridor8(_vm, this); _rooms[CORRIDOR9] = new AxacussCorridor9(_vm, this); _rooms[BCORRIDOR] = new AxacussBcorridor(_vm, this); _rooms[GUARD] = new AxacussIntersection(_vm, this); _rooms[GUARD3] = new AxacussExit(_vm, this); _rooms[OFFICE_L1] = new AxacussOffice1(_vm, this); _rooms[OFFICE_L2] = new AxacussOffice2(_vm, this); _rooms[OFFICE_R1] = new AxacussOffice3(_vm, this); _rooms[OFFICE_R2] = new AxacussOffice4(_vm, this); _rooms[OFFICE_L] = new AxacussOffice5(_vm, this); _rooms[ELEVATOR] = new AxacussElevator(_vm, this); _rooms[STATION] = new AxacussStation(_vm, this); _rooms[SIGN] = new AxacussSign(_vm, this); _rooms[OUTRO] = new Outro(_vm, this); } void GameManager::initGui() { int cmdCount = ARRAYSIZE(_guiCommandButton); int cmdAvailableSpace = 320 - (cmdCount - 1) * 2; for (int i = 0; i < cmdCount; ++i) { const Common::String &text = _vm->getGameString(guiCommands[i]); cmdAvailableSpace -= Screen::textWidth(text); } int commandButtonX = 0; for (int i = 0; i < ARRAYSIZE(_guiCommandButton); ++i) { const Common::String &text = _vm->getGameString(guiCommands[i]); int width; if (i < cmdCount - 1) { int space = cmdAvailableSpace / (cmdCount - i); cmdAvailableSpace -= space; width = Screen::textWidth(text) + space; } else width = 320 - commandButtonX; _guiCommandButton[i].setSize(commandButtonX, 150, commandButtonX + width, 159); _guiCommandButton[i].setText(text.c_str()); _guiCommandButton[i].setColor(kColorWhite25, kColorDarkGreen, kColorWhite44, kColorGreen); commandButtonX += width + 2; } for (int i = 0; i < ARRAYSIZE(_guiInventory); ++i) { int inventoryX = 136 * (i % 2); int inventoryY = 161 + 10 * (i / 2); _guiInventory[i].setSize(inventoryX, inventoryY, inventoryX + 135, inventoryY + 9); _guiInventory[i].setColor(kColorWhite25, kColorDarkRed, kColorWhite35, kColorRed); } _guiInventoryArrow[0].setSize(272, 161, 279, 180); _guiInventoryArrow[0].setColor(kColorWhite25, kColorDarkRed, kColorWhite35, kColorRed); _guiInventoryArrow[0].setText("\x82"); _guiInventoryArrow[0].setTextPosition(273, 166); _guiInventoryArrow[1].setSize(272, 181, 279, 200); _guiInventoryArrow[1].setColor(kColorWhite25, kColorDarkRed, kColorWhite35, kColorRed); _guiInventoryArrow[1].setText("\x83"); _guiInventoryArrow[1].setTextPosition(273, 186); } void GameManager::updateEvents() { handleTime(); if (_animationEnabled && !_vm->_screen->isMessageShown() && _animationTimer == 0) _currentRoom->animation(); if (_state._eventCallback != kNoFn && _state._time >= _state._eventTime) { _vm->_allowLoadGame = false; _vm->_allowSaveGame = false; _state._eventTime = kMaxTimerValue; EventFunction fn = _state._eventCallback; _state._eventCallback = kNoFn; switch (fn) { case kNoFn: break; case kSupernovaFn: supernovaEvent(); break; case kGuardReturnedFn: guardReturnedEvent(); break; case kGuardWalkFn: guardWalkEvent(); break; case kTaxiFn: taxiEvent(); break; case kSearchStartFn: searchStartEvent(); break; } _vm->_allowLoadGame = true; _vm->_allowSaveGame = true; return; } if (_state._alarmOn && _state._timeAlarm <= _state._time) { _state._alarmOn = false; alarm(); return; } _mouseClicked = false; _keyPressed = false; Common::Event event; while (g_system->getEventManager()->pollEvent(event)) { switch (event.type) { case Common::EVENT_KEYDOWN: _keyPressed = true; processInput(event.kbd); break; case Common::EVENT_LBUTTONUP: // fallthrough case Common::EVENT_RBUTTONUP: if (_currentRoom->getId() != INTRO && _sound->isPlaying()) return; _mouseClicked = true; // fallthrough case Common::EVENT_MOUSEMOVE: _mouseClickType = event.type; _mouseX = event.mouse.x; _mouseY = event.mouse.y; if (_guiEnabled) processInput(); break; default: break; } } } void GameManager::processInput(Common::KeyState &state) { _key = state; switch (state.keycode) { case Common::KEYCODE_F1: // help break; case Common::KEYCODE_F2: // show game doc break; case Common::KEYCODE_F3: // show game info break; case Common::KEYCODE_F4: _vm->setTextSpeed(); break; case Common::KEYCODE_F5: // load/save break; case Common::KEYCODE_x: if (state.flags & Common::KBD_ALT) { if (_vm->quitGameDialog()) _vm->quitGame(); } break; case Common::KEYCODE_d: if (state.flags & Common::KBD_CTRL) _vm->_console->attach(); break; default: break; } } void GameManager::resetInputState() { setObjectNull(_inputObject[0]); setObjectNull(_inputObject[1]); _inputVerb = ACTION_WALK; _processInput = false; _mouseClicked = false; _keyPressed = false; _key.reset(); _mouseClickType = Common::EVENT_MOUSEMOVE; processInput(); } void GameManager::processInput() { enum { onNone, onObject, onCmdButton, onInventory, onInventoryArrowUp, onInventoryArrowDown } mouseLocation; if (_mouseField >= 0 && _mouseField < 256) mouseLocation = onObject; else if (_mouseField >= 256 && _mouseField < 512) mouseLocation = onCmdButton; else if (_mouseField >= 512 && _mouseField < 768) mouseLocation = onInventory; else if (_mouseField == 768) mouseLocation = onInventoryArrowUp; else if (_mouseField == 769) mouseLocation = onInventoryArrowDown; else mouseLocation = onNone; if (_mouseClickType == Common::EVENT_LBUTTONUP) { if (_vm->_screen->isMessageShown()) { // Hide the message and consume the event _vm->removeMessage(); if (mouseLocation != onCmdButton) return; } switch(mouseLocation) { case onObject: case onInventory: // Fallthrough if (_inputVerb == ACTION_GIVE || _inputVerb == ACTION_USE) { if (isNullObject(_inputObject[0])) { _inputObject[0] = _currentInputObject; if (!_inputObject[0]->hasProperty(COMBINABLE)) _processInput = true; } else { _inputObject[1] = _currentInputObject; _processInput = true; } } else { _inputObject[0] = _currentInputObject; if (!isNullObject(_currentInputObject)) _processInput = true; } break; case onCmdButton: resetInputState(); _inputVerb = static_cast(_mouseField - 256); break; case onInventoryArrowUp: if (_inventoryScroll >= 2) _inventoryScroll -= 2; break; case onInventoryArrowDown: if (_inventoryScroll < _inventory.getSize() - ARRAYSIZE(_guiInventory)) _inventoryScroll += 2; break; case onNone: break; } } else if (_mouseClickType == Common::EVENT_RBUTTONUP) { if (_vm->_screen->isMessageShown()) { // Hide the message and consume the event _vm->removeMessage(); return; } if (isNullObject(_currentInputObject)) return; if (mouseLocation == onObject || mouseLocation == onInventory) { _inputObject[0] = _currentInputObject; ObjectTypes type = _inputObject[0]->_type; if (type & OPENABLE) _inputVerb = (type & OPENED) ? ACTION_CLOSE : ACTION_OPEN; else if (type & PRESS) _inputVerb = ACTION_PRESS; else if (type & TALK) _inputVerb = ACTION_TALK; else _inputVerb = ACTION_LOOK; _processInput = true; } } else if (_mouseClickType == Common::EVENT_MOUSEMOVE) { int field = -1; int click = -1; if ((_mouseY >= _guiCommandButton[0].top) && (_mouseY <= _guiCommandButton[0].bottom)) { /* command row */ field = 9; while (_mouseX < _guiCommandButton[field].left - 1) field--; field += 256; } else if ((_mouseX >= 283) && (_mouseX <= 317) && (_mouseY >= 163) && (_mouseY <= 197)) { /* exit box */ field = _exitList[(_mouseX - 283) / 7 + 5 * ((_mouseY - 163) / 7)]; } else if ((_mouseY >= 161) && (_mouseX <= 270)) { /* inventory box */ field = (_mouseX + 1) / 136 + ((_mouseY - 161) / 10) * 2; if (field + _inventoryScroll < _inventory.getSize()) field += 512; else field = -1; } else if ((_mouseY >= 161) && (_mouseX >= 271) && (_mouseX < 279)) { /* inventory arrows */ field = (_mouseY > 180) ? 769 : 768; } else { /* normal item */ for (int i = 0; (_currentRoom->getObject(i)->_id != INVALIDOBJECT) && (field == -1) && i < kMaxObject; i++) { click = _currentRoom->getObject(i)->_click; const MSNImage *image = _vm->_screen->getCurrentImage(); if (click != 255 && image) { const MSNImage::ClickField *clickField = image->_clickField; do { if ((_mouseX >= clickField[click].x1) && (_mouseX <= clickField[click].x2) && (_mouseY >= clickField[click].y1) && (_mouseY <= clickField[click].y2)) field = i; click = clickField[click].next; } while ((click != 0) && (field == -1)); } } } if (_mouseField != field) { switch (mouseLocation) { case onInventoryArrowUp: case onInventoryArrowDown: // Fallthrough _guiInventoryArrow[_mouseField - 768].setHighlight(false); break; case onInventory: _guiInventory[_mouseField - 512].setHighlight(false); break; case onCmdButton: _guiCommandButton[_mouseField - 256].setHighlight(false); break; case onObject: case onNone: // Fallthrough break; } setObjectNull(_currentInputObject); _mouseField = field; if (_mouseField >= 0 && _mouseField < 256) mouseLocation = onObject; else if (_mouseField >= 256 && _mouseField < 512) mouseLocation = onCmdButton; else if (_mouseField >= 512 && _mouseField < 768) mouseLocation = onInventory; else if (_mouseField == 768) mouseLocation = onInventoryArrowUp; else if (_mouseField == 769) mouseLocation = onInventoryArrowDown; else mouseLocation = onNone; switch (mouseLocation) { case onInventoryArrowUp: case onInventoryArrowDown: // Fallthrough _guiInventoryArrow[_mouseField - 768].setHighlight(true); break; case onInventory: _guiInventory[_mouseField - 512].setHighlight(true); _currentInputObject = _inventory.get(_mouseField - 512 + _inventoryScroll); break; case onCmdButton: _guiCommandButton[_mouseField - 256].setHighlight(true); break; case onObject: _currentInputObject = _currentRoom->getObject(_mouseField); break; case onNone: break; } } } } void GameManager::setObjectNull(Object *&obj) { obj = &_nullObject; } bool GameManager::isNullObject(Object *obj) { return obj == &_nullObject; } void GameManager::corridorOnEntrance() { if (_state._corridorSearch) busted(0); } void GameManager::telomat(int nr) { static Common::String name[8] = { "DR. ALAB HANSI", "ALAB HANSI", "SAVAL LUN", "x", "PROF. DR. UGNUL TSCHABB", "UGNUL TSCHABB", "ALGA HURZ LI", "x" }; static Common::String name2[4] = { "Alab Hansi", "Saval Lun", "Ugnul Tschabb", "Alga Hurz Li" }; StringId dial1[4]; dial1[0] = kStringTelomat1; dial1[1] = kNoString; dial1[2] = kStringTelomat3; dial1[3] = kStringDialogSeparator; static byte rows1[3] = {1, 2, 1}; StringId dial2[4]; dial2[0] = kStringTelomat4; dial2[1] = kStringTelomat5; dial2[2] = kStringTelomat6; dial2[3] = kStringDialogSeparator; static byte rows2[4] = {1, 1, 1, 1}; _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderText(kStringTelomat7, 100, 70, kColorGreen); _vm->renderText(kStringTelomat8, 100, 81, kColorGreen); _vm->renderText(kStringTelomat9, 100, 92, kColorGreen); _vm->renderText(kStringTelomat10, 100, 103, kColorGreen); _vm->renderText(kStringTelomat11, 100, 120, kColorDarkGreen); Common::String input; do { getInput(); switch (_key.keycode) { case Common::KEYCODE_2: { _vm->renderBox(0, 0, 320, 200, kColorDarkBlue); _vm->renderText(kStringTelomat12, 50, 80, kColorGreen); _vm->renderText(kStringTelomat13, 50, 91, kColorGreen); do { edit(input, 50, 105, 30); } while ((_key.keycode != Common::KEYCODE_RETURN) && (_key.keycode != Common::KEYCODE_ESCAPE)); if (_key.keycode == Common::KEYCODE_ESCAPE) { _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderRoom(*_currentRoom); _vm->paletteBrightness(); _guiEnabled = true; drawMapExits(); return; } input.toUppercase(); int i = 0; while ((i < 8) && (input != name[i])) i++; i >>= 1; if (i == 4) { _vm->renderText(kStringTelomat14, 50, 120, kColorGreen); wait(10); _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderRoom(*_currentRoom); _vm->paletteBrightness(); _guiEnabled = true; drawMapExits(); return; } if ((i == nr) || _rooms[BCORRIDOR]->getObject(4 + i)->hasProperty(CAUGHT)) { _vm->renderText(kStringTelomat15, 50, 120, kColorGreen); wait(10); _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderRoom(*_currentRoom); _vm->paletteBrightness(); _guiEnabled = true; drawMapExits(); return; } _vm->renderText(kStringTelomat16, 50, 120, kColorGreen); wait(10); _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderRoom(*_currentRoom); _vm->paletteBrightness(); _vm->renderMessage(kStringTelomat17, kMessageTop, name2[i]); waitOnInput(_messageDuration); _vm->removeMessage(); if (_state._nameSeen[nr]) { Common::String string = _vm->getGameString(kStringTelomat2); _vm->setGameString(kStringPlaceholder1, Common::String::format(string.c_str(), name2[nr].c_str())); dial1[1] = kStringPlaceholder1; _currentRoom->addSentence(1, 1); } else _currentRoom->removeSentence(1, 1); switch (dialog(3, rows1, dial1, 1)) { case 1: _vm->renderMessage(kStringTelomat18, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); if ((_state._destination == 255) && !_rooms[BCORRIDOR]->isSectionVisible(7)) { _state._eventTime = _state._time + ticksToMsec(150); _state._eventCallback = kGuardWalkFn; _state._origin = i; _state._destination = nr; } break; case 0: _vm->renderMessage(kStringTelomat19, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); if (dialog(4, rows2, dial2, 0) != 3) { wait(10); say(kStringTelomat20); } _rooms[BCORRIDOR]->setSectionVisible(7, true); _rooms[BCORRIDOR]->setSectionVisible(i + 1, true); _state._eventTime = kMaxTimerValue; _currentRoom->addSentence(0, 1); } _guiEnabled = true; drawMapExits(); return; } case Common::KEYCODE_1: case Common::KEYCODE_3: case Common::KEYCODE_4: _vm->renderBox(0, 0, 320, 200, kColorDarkBlue); _vm->renderText(kStringTelomat21, 100, 90, kColorGreen); input = ""; do { edit(input, 100, 105, 30); } while ((_key.keycode != Common::KEYCODE_RETURN) && (_key.keycode != Common::KEYCODE_ESCAPE)); if (_key.keycode == Common::KEYCODE_RETURN) { _vm->renderText(kStringShipSleepCabin9, 100, 120, kColorGreen); wait(10); } // fallthrough case Common::KEYCODE_ESCAPE: _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->renderRoom(*_currentRoom); _vm->paletteBrightness(); _guiEnabled = true; drawMapExits(); return; default: break; } } while (true); } void GameManager::startSearch() { if ((_currentRoom->getId() >= CORRIDOR1) && (_currentRoom->getId() <= BCORRIDOR)) busted(0); _state._corridorSearch = true; } void GameManager::search(int time) { _state._eventTime = _state._time + ticksToMsec(time); _state._eventCallback = kSearchStartFn; } void GameManager::guardNoticed() { _vm->paletteFadeOut(); Room *r = _currentRoom; _currentRoom = _rooms[GUARD]; _vm->setCurrentImage(40); _vm->renderBox(0, 0, 320, 200, 0); _vm->renderImage(0); _vm->paletteFadeIn(); _vm->renderImage(2); reply(kStringGuardNoticed1, 2, 5); wait(2); reply(kStringGuardNoticed2, 2, 5); _vm->paletteFadeOut(); _currentRoom->setSectionVisible(2, false); _currentRoom->setSectionVisible(5, false); _currentRoom = r; _guiEnabled = true; drawMapExits(); } void GameManager::busted(int i) { if (i > 0) _vm->renderImage(i); if (i == 0) { if ((_currentRoom->getId() >= OFFICE_L1) && (_currentRoom->getId() <= OFFICE_R2)) { if (_currentRoom->getId() < OFFICE_R1) i = 10; else i = 5; if (!_currentRoom->getObject(0)->hasProperty(OPENED)) { _vm->renderImage(i - 1); _sound->play(kAudioDoorOpen); wait(2); } _vm->renderImage(i); wait(3); _vm->renderImage(i + 3); _sound->play(kAudioVoiceHalt); _vm->renderImage(i); wait(5); if (_currentRoom->getId() == OFFICE_L2) i = 13; _vm->renderImage(i + 1); wait(3); _vm->renderImage(i + 2); shot(0, 0); } else if (_currentRoom->getId() == BCORRIDOR) _vm->renderImage(21); else if (_currentRoom->isSectionVisible(4)) _vm->renderImage(32); // below else if (_currentRoom->isSectionVisible(2)) _vm->renderImage(30); // right else if (_currentRoom->isSectionVisible(1)) _vm->renderImage(31); // left else _vm->renderImage(33); // above } _sound->play(kAudioVoiceHalt); wait(3); shot(0, 0); } void GameManager::novaScroll() { static byte planet_f[6] = {0xeb,0xec,0xf0,0xed,0xf1,0xf2}; static byte nova_f[13] = {0xea,0xe9,0xf5,0xf3,0xf7,0xf4,0xf6, 0xf9,0xfb,0xfc,0xfd,0xfe,0xfa}; static byte rgb[65][3] = { { 5, 0, 0},{10, 0, 0},{15, 0, 0},{20, 0, 0},{25, 0, 0}, {30, 0, 0},{35, 0, 0},{40, 0, 0},{45, 0, 0},{50, 0, 0}, {55, 0, 0},{60, 0, 0},{63,10, 5},{63,20,10},{63,30,15}, {63,40,20},{63,50,25},{63,60,30},{63,63,33},{63,63,30}, {63,63,25},{63,63,20},{63,63,15},{63,63,10},{60,60,15}, {57,57,20},{53,53,25},{50,50,30},{47,47,35},{43,43,40}, {40,40,45},{37,37,50},{33,33,53},{30,30,56},{27,27,59}, {23,23,61},{20,20,63},{21,25,63},{22,30,63},{25,35,63}, {30,40,63},{35,45,63},{40,50,63},{45,55,63},{50,60,63}, {55,63,63},{59,63,63},{63,63,63},{63,60,63},{60,50,60}, {55,40,55},{50,30,50},{45,20,45},{40,10,40},{42,15,42}, {45,20,45},{47,25,47},{50,30,50},{52,35,52},{55,40,55}, {57,45,57},{60,50,60},{62,55,62},{63,60,63},{63,63,63}}; byte palette[768]; _vm->_system->getPaletteManager()->grabPalette(palette, 0, 255); for (int t = 0; t < 65; ++t) { for (int i = 0; i < 6; ++i) { int idx = 3 * (planet_f[i] - 1); for (int c = 0 ; c < 3 ; ++c) { if (palette[idx+c] < rgb[t][c]) palette[idx+c] = rgb[t][c]; } } for (int cycle = 0; cycle < t && cycle < 13; ++cycle) { int idx = 3 * (nova_f[cycle] - 1); for (int c = 0 ; c < 3 ; ++c) palette[idx + c] = rgb[t - cycle - 1][c]; } _vm->_system->getPaletteManager()->setPalette(palette, 0, 255); _vm->_system->updateScreen(); _vm->_system->delayMillis(_vm->_delay); } } void GameManager::supernovaEvent() { _vm->removeMessage(); CursorMan.showMouse(false); if (_currentRoom->getId() <= CAVE) { _vm->renderMessage(kStringSupernova1); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->paletteFadeOut(); changeRoom(MEETUP); _rooms[AIRLOCK]->getObject(0)->disableProperty(OPENED); _rooms[AIRLOCK]->setSectionVisible(3, true); _rooms[AIRLOCK]->getObject(1)->setProperty(OPENED); _rooms[AIRLOCK]->setSectionVisible(17, true); _rooms[AIRLOCK]->setSectionVisible(6, false); _vm->renderRoom(*_currentRoom); _vm->paletteFadeIn(); } _vm->renderMessage(kStringSupernova2); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->setCurrentImage(26); _vm->renderImage(0); _vm->paletteBrightness(); novaScroll(); _vm->paletteFadeOut(); _vm->renderBox(0, 0, 320, 200, kColorBlack); _vm->_screen->setGuiBrightness(255); _vm->paletteBrightness(); if (_currentRoom->getId() == GLIDER) { _vm->renderMessage(kStringSupernova3); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->_screen->setGuiBrightness(0); _vm->paletteBrightness(); _vm->renderRoom(*_currentRoom); _vm->paletteFadeIn(); _vm->renderMessage(kStringSupernova4, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->renderMessage(kStringSupernova5, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->renderMessage(kStringSupernova6, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->renderMessage(kStringSupernova7, kMessageTop); waitOnInput(_messageDuration); _vm->removeMessage(); changeRoom(MEETUP2); _rooms[MEETUP2]->setSectionVisible(1, true); _rooms[MEETUP2]->removeSentence(0, 1); _inventory.remove(*(_rooms[ROGER]->getObject(3))); _inventory.remove(*(_rooms[ROGER]->getObject(7))); _inventory.remove(*(_rooms[ROGER]->getObject(8))); } else { _vm->renderMessage(kStringSupernova8); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->_screen->setGuiBrightness(0); _vm->paletteBrightness(); changeRoom(MEETUP2); if (_rooms[ROGER]->getObject(3)->hasProperty(CARRIED) && !_rooms[GLIDER]->isSectionVisible(5)) { _rooms[MEETUP2]->setSectionVisible(1, true); _rooms[MEETUP2]->setSectionVisible(12, true); _rooms[MEETUP2]->getObject(1)->_click = 0; _rooms[MEETUP2]->getObject(0)->_click = 1; _rooms[MEETUP2]->removeSentence(0, 1); } _rooms[MEETUP2]->removeSentence(1, 1); } _rooms[AIRLOCK]->getObject(4)->setProperty(WORN); _rooms[AIRLOCK]->getObject(5)->setProperty(WORN); _rooms[AIRLOCK]->getObject(6)->setProperty(WORN); _rooms[CAVE]->getObject(1)->_exitRoom = MEETUP2; _guiEnabled = true; CursorMan.showMouse(true); } void GameManager::guardReturnedEvent() { if (_currentRoom->getId() == GUARD) busted(-1); else if ((_currentRoom->getId() == CORRIDOR9) && (_currentRoom->isSectionVisible(27))) busted(0); _rooms[GUARD]->setSectionVisible(1, false); _rooms[GUARD]->getObject(3)->_click = 0; _rooms[GUARD]->setSectionVisible(6, false); _rooms[GUARD]->getObject(2)->disableProperty(OPENED); _rooms[GUARD]->setSectionVisible(7, false); _rooms[GUARD]->getObject(5)->_click = 255; _rooms[CORRIDOR9]->setSectionVisible(27, false); _rooms[CORRIDOR9]->setSectionVisible(28, true); _rooms[CORRIDOR9]->getObject(1)->disableProperty(OPENED); } void GameManager::walk(int imgId) { if (_prevImgId) _vm->renderImage(_prevImgId + 128); _vm->renderImage(imgId); _prevImgId = imgId; wait(3); } void GameManager::guardWalkEvent() { _prevImgId = 0; bool behind = (!_rooms[BCORRIDOR]->getObject(_state._origin + 4)->hasProperty(OCCUPIED) || _rooms[BCORRIDOR]->getObject(_state._origin + 4)->hasProperty(OPENED)); _rooms[BCORRIDOR]->getObject(_state._origin + 4)->disableProperty(OCCUPIED); if (_currentRoom == _rooms[BCORRIDOR]) { if (_vm->_screen->isMessageShown()) _vm->removeMessage(); if (!behind) { _vm->renderImage(_state._origin + 1); _prevImgId = _state._origin + 1; _sound->play(kAudioDoorOpen); wait(3); } int imgId; switch (_state._origin) { case 0: imgId = 11; break; case 1: imgId = 16; break; case 2: imgId = 15; break; case 3: default: imgId = 20; break; } _vm->renderImage(imgId); if (!behind) { wait(3); _vm->renderImage(_prevImgId + 128); _sound->play(kAudioDoorClose); } _prevImgId = imgId; wait(3); switch (_state._origin) { case 0: walk(12); walk(13); break; case 1: walk(17); walk(18); break; case 2: walk(14); walk(13); break; case 3: walk(19); walk(18); } if (!_state._playerHidden) { if (_state._origin & 1) walk(10); else walk(5); busted(-1); } if ((_state._origin & 1) && !(_state._destination & 1)) { for (int i = 10; i >= 5; i--) walk(i); walk(13); } else if (!(_state._origin & 1) && (_state._destination & 1)) { for (int i = 5; i <= 10; i++) walk(i); walk(18); } switch (_state._destination) { case 0: for (int i = 13; i >= 11; i--) walk(i); break; case 1: for (int i = 18; i >= 16; i--) walk(i); break; case 2: for (int i = 13; i <= 15; i++) walk(i); break; case 3: for (int i = 18; i <= 20; i++) walk(i); } if (behind) { _vm->renderImage(_state._destination + 1); _sound->play(kAudioDoorOpen); wait(3); _vm->renderImage(_prevImgId + 128); wait(3); _vm->renderImage(_state._destination + 1 + 128); _sound->play(kAudioDoorClose); _rooms[BCORRIDOR]->getObject(_state._destination + 4)->setProperty(OCCUPIED); _state._destination = 255; } else if (_rooms[BCORRIDOR]->isSectionVisible(_state._destination + 1)) { _vm->renderImage(_prevImgId + 128); _rooms[BCORRIDOR]->getObject(_state._destination + 4)->setProperty(OCCUPIED); SWAP(_state._origin, _state._destination); _state._eventTime = _state._time + ticksToMsec(60); _state._eventCallback = kGuardWalkFn; } else { wait(18); SWAP(_state._origin, _state._destination); _state._eventCallback = kGuardWalkFn; } } else if (behind) { _rooms[BCORRIDOR]->getObject(_state._destination + 4)->setProperty(OCCUPIED); if (_currentRoom == _rooms[OFFICE_L1 + _state._destination]) busted(0); _state._destination = 255; } else if (_rooms[BCORRIDOR]->isSectionVisible(_state._destination + 1) && _rooms[OFFICE_L1 + _state._destination]->getObject(0)->hasProperty(OPENED)) { _rooms[BCORRIDOR]->getObject(_state._destination + 4)->setProperty(OCCUPIED); if (_currentRoom == _rooms[OFFICE_L1 + _state._destination]) busted(0); SWAP(_state._origin, _state._destination); _state._eventTime = _state._time + ticksToMsec(60); _state._eventCallback = kGuardWalkFn; } else { SWAP(_state._origin, _state._destination); _state._eventCallback = kGuardWalkFn; } } void GameManager::taxiEvent() { if (_currentRoom->getId() == SIGN) { changeRoom(STATION); _vm->renderRoom(*_currentRoom); } _vm->renderImage(1); _vm->renderImage(2); _sound->play(kAudioRocks); screenShake(); _vm->renderImage(9); _currentRoom->getObject(1)->setProperty(OPENED); _vm->renderImage(1); _currentRoom->setSectionVisible(2, false); _vm->renderImage(3); for (int i = 4; i <= 8; i++) { wait(2); _vm->renderImage(invertSection(i - 1)); _vm->renderImage(i); } _rooms[SIGN]->setSectionVisible(2, false); _rooms[SIGN]->setSectionVisible(3, true); } void GameManager::searchStartEvent() { if ((_currentRoom >= _rooms[CORRIDOR1]) && (_currentRoom <= _rooms[BCORRIDOR])) busted(0); _state._corridorSearch = true; } void GameManager::great(uint number) { if (number && (_state._greatFlag & (1 << number))) return; _sound->play(kAudioSuccess); _state._greatFlag |= 1 << number; } bool GameManager::airless() { return (_currentRoom->getId() == HOLD || _currentRoom->getId() == LANDINGMODULE || _currentRoom->getId() == GENERATOR || _currentRoom->getId() == OUTSIDE || _currentRoom->getId() == ROCKS || _currentRoom->getId() == CAVE || _currentRoom->getId() == MEETUP || _currentRoom->getId() == MEETUP2 || _currentRoom->getId() == MEETUP3 || (_currentRoom->getId() == AIRLOCK && _rooms[AIRLOCK]->getObject(1)->hasProperty(OPENED))); } void GameManager::sentence(int number, bool brightness) { if (number < 0) return; _vm->renderBox(0, 141 + _rowsStart[number] * 10, 320, _rows[number] * 10 - 1, brightness ? kColorWhite44 : kColorWhite25); if (_texts[_rowsStart[number]] == kStringDialogSeparator) _vm->renderText(kStringConversationEnd, 1, 142 + _rowsStart[number] * 10, brightness ? kColorRed : kColorDarkRed); else { for (int r = _rowsStart[number]; r < _rowsStart[number] + _rows[number]; ++r) _vm->renderText(_texts[r], 1, 142 + r * 10, brightness ? kColorGreen : kColorDarkGreen); } } void GameManager::say(StringId textId) { Common::String str = _vm->getGameString(textId); if (!str.empty()) say(str.c_str()); } void GameManager::say(const char *text) { Common::String t(text); char *row[6]; Common::String::iterator p = t.begin(); uint numRows = 0; while (*p) { row[numRows++] = p; while ((*p != '\0') && (*p != '|')) { ++p; } if (*p == '|') { *p = 0; ++p; } } _vm->renderBox(0, 138, 320, 62, kColorBlack); _vm->renderBox(0, 141, 320, numRows * 10 - 1, kColorWhite25); for (uint r = 0; r < numRows; ++r) _vm->renderText(row[r], 1, 142 + r * 10, kColorDarkGreen); waitOnInput((t.size() + 20) * _vm->_textSpeed / 10); _vm->renderBox(0, 138, 320, 62, kColorBlack); } void GameManager::reply(StringId textId, int aus1, int aus2) { Common::String str = _vm->getGameString(textId); if (!str.empty()) reply(str.c_str(), aus1, aus2); } void GameManager::reply(const char *text, int aus1, int aus2) { if (*text != '|') _vm->renderMessage(text, kMessageTop); for (int z = (strlen(text) + 20) * _vm->_textSpeed / 40; z > 0; --z) { _vm->renderImage(aus1); waitOnInput(2); if (_keyPressed || _mouseClicked) z = 1; _vm->renderImage(aus2); waitOnInput(2); if (_keyPressed || _mouseClicked) z = 1; } if (*text != '|') _vm->removeMessage(); } int GameManager::dialog(int num, byte rowLength[6], StringId text[6], int number) { _vm->_allowLoadGame = false; _guiEnabled = false; bool remove[6]; for (int i = 0; i < 5; ++i) remove[i] = _currentRoom->sentenceRemoved(i, number); // The original does not initialize remove[5]!!! // Set it to false/0. But maybe the loop above should use 6 instead of 5? remove[5] = false; _vm->renderBox(0, 138, 320, 62, kColorBlack); for (int i = 0; i < 6 ; ++i) _sentenceNumber[i] = -1; int r = 0, rq = 0; for (int i = 0; i < num; ++i) { if (!remove[i]) { _rowsStart[i] = r; _rows[i] = rowLength[i]; for (int j = 0; j < _rows[i]; ++j, ++r, ++rq) { _texts[r] = text[rq]; _sentenceNumber[r] = i; } sentence(i, false); } else rq += rowLength[i]; } _currentSentence = -1; do { do { updateEvents(); mousePosDialog(_mouseX, _mouseY); g_system->updateScreen(); g_system->delayMillis(_vm->_delay); } while (!_mouseClicked && !_vm->shouldQuit()); } while (_currentSentence == -1 && !_vm->shouldQuit()); _vm->renderBox(0, 138, 320, 62, kColorBlack); if (number && _currentSentence != -1 && _texts[_rowsStart[_currentSentence]] != kStringDialogSeparator) _currentRoom->removeSentence(_currentSentence, number); _guiEnabled = true; _vm->_allowLoadGame = true; return _currentSentence; } void GameManager::mousePosDialog(int x, int y) { int a = y < 141 ? -1 : _sentenceNumber[(y - 141) / 10]; if (a != _currentSentence) { sentence(_currentSentence, false); _currentSentence = a; sentence(_currentSentence, true); } } void GameManager::turnOff() { if (_state._powerOff) return; _state._powerOff = true; roomBrightness(); } void GameManager::turnOn() { if (!_state._powerOff) return; _state._powerOff = false; _vm->_screen->setViewportBrightness(255); _rooms[SLEEP]->setSectionVisible(1, false); _rooms[SLEEP]->setSectionVisible(2, false); _rooms[COCKPIT]->setSectionVisible(22, false); } void GameManager::takeObject(Object &obj) { if (obj.hasProperty(CARRIED)) return; if (obj._section != 0) _vm->renderImage(obj._section); obj._click = obj._click2 = 255; _inventory.add(obj); } void GameManager::drawCommandBox() { for (int i = 0; i < ARRAYSIZE(_guiCommandButton); ++i) { _vm->renderBox(_guiCommandButton[i]); int space = (_guiCommandButton[i].width() - Screen::textWidth(_guiCommandButton[i].getText())) / 2; _vm->renderText(_guiCommandButton[i].getText(), _guiCommandButton[i].getTextPos().x + space, _guiCommandButton[i].getTextPos().y, _guiCommandButton[i].getTextColor()); } } void GameManager::drawInventory() { for (int i = 0; i < ARRAYSIZE(_guiInventory); ++i) { _vm->renderBox(_guiInventory[i]); _vm->renderText(_inventory.get(i + _inventoryScroll)->_name, _guiInventory[i].getTextPos().x, _guiInventory[i].getTextPos().y, _guiInventory[i].getTextColor()); } _vm->renderBox(_guiInventoryArrow[0]); _vm->renderBox(_guiInventoryArrow[1]); if (_inventory.getSize() > ARRAYSIZE(_guiInventory)) { if (_inventoryScroll != 0) _vm->renderText(_guiInventoryArrow[0]); if (_inventoryScroll + ARRAYSIZE(_guiInventory) < _inventory.getSize()) _vm->renderText(_guiInventoryArrow[1]); } } uint16 GameManager::getKeyInput(bool blockForPrintChar) { while (!_vm->shouldQuit()) { updateEvents(); if (_keyPressed) { if (blockForPrintChar) { if (Common::isPrint(_key.keycode) || _key.keycode == Common::KEYCODE_BACKSPACE || _key.keycode == Common::KEYCODE_DELETE || _key.keycode == Common::KEYCODE_RETURN || _key.keycode == Common::KEYCODE_SPACE || _key.keycode == Common::KEYCODE_ESCAPE || _key.keycode == Common::KEYCODE_UP || _key.keycode == Common::KEYCODE_DOWN || _key.keycode == Common::KEYCODE_LEFT || _key.keycode == Common::KEYCODE_RIGHT) { if (_key.flags & Common::KBD_SHIFT) return toupper(_key.ascii); else return tolower(_key.ascii); } } else { return _key.ascii; } } g_system->updateScreen(); g_system->delayMillis(_vm->_delay); } return 0; } Common::EventType GameManager::getMouseInput() { while (!_vm->shouldQuit()) { updateEvents(); if (_mouseClicked) return _mouseClickType; g_system->updateScreen(); g_system->delayMillis(_vm->_delay); } return Common::EVENT_INVALID; } void GameManager::getInput() { while (!_vm->shouldQuit()) { updateEvents(); if (_mouseClicked || _keyPressed) break; g_system->updateScreen(); g_system->delayMillis(_vm->_delay); } } void GameManager::roomBrightness() { _roomBrightness = 255; if ((_currentRoom->getId() != OUTSIDE) && (_currentRoom->getId() < ROCKS) && _state._powerOff) _roomBrightness = 153; else if (_currentRoom->getId() == CAVE) _roomBrightness = 0; else if ((_currentRoom->getId() == GUARD3) && _state._powerOff) _roomBrightness = 0; if (_vm->_screen->getViewportBrightness() != 0) _vm->_screen->setViewportBrightness(_roomBrightness); _vm->paletteBrightness(); } void GameManager::changeRoom(RoomId id) { _currentRoom = _rooms[id]; _newRoom = true; } void GameManager::wait(int ticks) { int32 end = _state._time + ticksToMsec(ticks); do { g_system->delayMillis(_vm->_delay); updateEvents(); g_system->updateScreen(); } while (_state._time < end && !_vm->shouldQuit()); } void GameManager::waitOnInput(int ticks) { int32 end = _state._time + ticksToMsec(ticks); do { g_system->delayMillis(_vm->_delay); updateEvents(); g_system->updateScreen(); } while (_state._time < end && !_vm->shouldQuit() && !_keyPressed && !_mouseClicked); } bool GameManager::waitOnInput(int ticks, Common::KeyCode &keycode) { keycode = Common::KEYCODE_INVALID; int32 end = _state._time + ticksToMsec(ticks); do { g_system->delayMillis(_vm->_delay); updateEvents(); g_system->updateScreen(); if (_keyPressed) { keycode = _key.keycode; _key.reset(); return true; } else if (_mouseClicked) return true; } while (_state._time < end && !_vm->shouldQuit()); return false; } void GameManager::setAnimationTimer(int ticks) { _animationTimer = ticksToMsec(ticks); } void GameManager::handleTime() { if (_timerPaused) return; int32 newTime = g_system->getMillis(); int32 delta = newTime - _oldTime; _state._time += delta; if (_state._time > 86400000) { _state._time -= 86400000; // 24h wrap around _state._alarmOn = (_state._timeAlarm > _state._time); } if (_animationTimer > delta) _animationTimer -= delta; else _animationTimer = 0; _oldTime = newTime; } void GameManager::pauseTimer(bool pause) { if (pause == _timerPaused) return; if (pause) { _timerPaused = true; int32 delta = g_system->getMillis() - _oldTime; _timePaused = _state._time + delta; } else { _state._time = _timePaused; _oldTime = g_system->getMillis(); _timerPaused = false; } } void GameManager::loadTime() { pauseTimer(false); } void GameManager::saveTime() { pauseTimer(true); } void GameManager::screenShake() { for (int i = 0; i < 12; ++i) { _vm->_system->setShakePos(8); wait(1); _vm->_system->setShakePos(0); wait(1); } } void GameManager::shock() { _sound->play(kAudioShock); dead(kStringShock); } void GameManager::showMenu() { _vm->renderBox(0, 138, 320, 62, 0); _vm->renderBox(0, 140, 320, 9, kColorWhite25); drawCommandBox(); _vm->renderBox(281, 161, 39, 39, kColorWhite25); drawInventory(); } void GameManager::drawMapExits() { // TODO: Preload _exitList on room entry instead on every call _vm->renderBox(281, 161, 39, 39, kColorWhite25); for (int i = 0; i < 25; i++) _exitList[i] = -1; for (int i = 0; i < kMaxObject; i++) { if (_currentRoom->getObject(i)->hasProperty(EXIT)) { byte r = _currentRoom->getObject(i)->_direction; _exitList[r] = i; int x = 284 + 7 * (r % 5); int y = 164 + 7 * (r / 5); _vm->renderBox(x, y, 5, 5, kColorDarkRed); } } } void GameManager::animationOff() { _animationEnabled = false; } void GameManager::animationOn() { _animationEnabled = true; } void GameManager::edit(Common::String &input, int x, int y, uint length) { bool isEditing = true; uint cursorIndex = input.size(); // NOTE: Pixels for char needed = kFontWidth + 2px left and right side bearing int overdrawWidth = ((int)((length + 1) * (kFontWidth + 2)) > (kScreenWidth - x)) ? kScreenWidth - x : (length + 1) * (kFontWidth + 2); while (isEditing) { _vm->_screen->setTextCursorPos(x, y); _vm->_screen->setTextCursorColor(kColorWhite99); _vm->renderBox(x, y - 1, overdrawWidth, 9, kColorDarkBlue); for (uint i = 0; i < input.size(); ++i) { // Draw char highlight depending on cursor position if (i == cursorIndex) { _vm->renderBox(_vm->_screen->getTextCursorPos().x, y - 1, Screen::textWidth(input[i]), 9, kColorWhite99); _vm->_screen->setTextCursorColor(kColorDarkBlue); _vm->renderText(input[i]); _vm->_screen->setTextCursorColor(kColorWhite99); } else _vm->renderText(input[i]); } if (cursorIndex == input.size()) { _vm->renderBox(_vm->_screen->getTextCursorPos().x + 1, y - 1, 6, 9, kColorDarkBlue); _vm->renderBox(_vm->_screen->getTextCursorPos().x, y - 1, 1, 9, kColorWhite99); } getKeyInput(true); if (_vm->shouldQuit()) break; switch (_key.keycode) { case Common::KEYCODE_RETURN: case Common::KEYCODE_ESCAPE: isEditing = false; break; case Common::KEYCODE_UP: case Common::KEYCODE_DOWN: cursorIndex = input.size(); break; case Common::KEYCODE_LEFT: if (cursorIndex != 0) --cursorIndex; break; case Common::KEYCODE_RIGHT: if (cursorIndex != input.size()) ++cursorIndex; break; case Common::KEYCODE_DELETE: if (cursorIndex != input.size()) input.deleteChar(cursorIndex); break; case Common::KEYCODE_BACKSPACE: if (cursorIndex != 0) { --cursorIndex; input.deleteChar(cursorIndex); } break; default: if (Common::isPrint(_key.ascii) && input.size() < length) { input.insertChar(_key.ascii, cursorIndex); ++cursorIndex; } break; } } } void GameManager::shot(int a, int b) { if (a) _vm->renderImage(a); _sound->play(kAudioGunShot); wait(2); if (b) _vm->renderImage(b); wait(2); if (a) _vm->renderImage(a); _sound->play(kAudioGunShot); wait(2); if (b) _vm->renderImage(b); dead(kStringShot); } void GameManager::takeMoney(int amount) { Object *moneyObject = _rooms[INTRO]->getObject(4); _state._money += amount; _vm->setGameString(kStringInventoryMoney, Common::String::format("%d Xa", _state._money)); if (_state._money > 0) { takeObject(*moneyObject); if (amount > 0) great(0); } else { _inventory.remove(*moneyObject); } } void GameManager::drawStatus() { int index = static_cast(_inputVerb); _vm->renderBox(0, 140, 320, 9, kColorWhite25); _vm->renderText(_vm->getGameString(guiStatusCommands[index]), 1, 141, kColorDarkGreen); if (isNullObject(_inputObject[0])) _vm->renderText(_currentInputObject->_name); else { _vm->renderText(_inputObject[0]->_name); if (_inputVerb == ACTION_GIVE) _vm->renderText(kPhrasalVerbParticleGiveTo); else if (_inputVerb == ACTION_USE) _vm->renderText(kPhrasalVerbParticleUseWith); _vm->renderText(_currentInputObject->_name); } } void GameManager::openLocker(const Room *room, Object *obj, Object *lock, int section) { _vm->renderImage(section); obj->setProperty(OPENED); lock->_click = 255; SWAP(obj->_click, obj->_click2); } void GameManager::closeLocker(const Room *room, Object *obj, Object *lock, int section) { if (!obj->hasProperty(OPENED)) _vm->renderMessage(kStringCloseLocker_1); else { _vm->renderImage(invertSection(section)); obj->disableProperty(OPENED); lock->_click = lock->_click2; SWAP(obj->_click, obj->_click2); } } void GameManager::dead(StringId messageId) { _vm->paletteFadeOut(); _guiEnabled = false; _vm->setCurrentImage(11); _vm->renderImage(0); _vm->renderMessage(messageId); _sound->play(kAudioDeath); _vm->paletteFadeIn(); getInput(); _vm->paletteFadeOut(); _vm->removeMessage(); destroyRooms(); initRooms(); initState(); initGui(); _inventory.clear(); changeRoom(CABIN_R3); g_system->fillScreen(kColorBlack); _vm->paletteFadeIn(); _guiEnabled = true; } int GameManager::invertSection(int section) { if (section < 128) section += 128; else section -= 128; return section; } bool GameManager::isHelmetOff() { Object *helmet = _inventory.get(HELMET); if (helmet && helmet->hasProperty(WORN)) { _vm->renderMessage(kStringIsHelmetOff_1); return false; } return true; } bool GameManager::genericInteract(Action verb, Object &obj1, Object &obj2) { if ((verb == ACTION_USE) && (obj1._id == SCHNUCK)) { if (isHelmetOff()) { takeObject(obj1); _vm->renderMessage(kStringGenericInteract_1); _inventory.remove(obj1); } } else if ((verb == ACTION_USE) && (obj1._id == EGG)) { if (isHelmetOff()) { takeObject(obj1); if (obj1.hasProperty(OPENED)) _vm->renderMessage(kStringGenericInteract_1); else _vm->renderMessage(kStringGenericInteract_2); _inventory.remove(obj1); } } else if ((verb == ACTION_OPEN) && (obj1._id == EGG)) { takeObject(obj1); if (obj1.hasProperty(OPENED)) _vm->renderMessage(kStringGenericInteract_3); else { takeObject(*_rooms[ENTRANCE]->getObject(8)); _vm->renderMessage(kStringGenericInteract_4); obj1.setProperty(OPENED); } } else if ((verb == ACTION_USE) && (obj1._id == PILL)) { if (isHelmetOff()) { _vm->renderMessage(kStringGenericInteract_5); great(0); _inventory.remove(obj1); _state._language = 2; takeObject(*_rooms[ENTRANCE]->getObject(17)); } } else if ((verb == ACTION_LOOK) && (obj1._id == PILL_HULL) && (_state._language == 2)) { _vm->renderMessage(kStringGenericInteract_6); _state._language = 1; } else if ((verb == ACTION_OPEN) && (obj1._id == WALLET)) { if (!_rooms[ROGER]->getObject(3)->hasProperty(CARRIED)) _vm->renderMessage(kStringGenericInteract_7); else if (_rooms[ROGER]->getObject(7)->hasProperty(CARRIED)) _vm->renderMessage(kStringGenericInteract_8); else { _vm->renderMessage(kStringGenericInteract_9); takeObject(*_rooms[ROGER]->getObject(7)); takeObject(*_rooms[ROGER]->getObject(8)); } } else if ((verb == ACTION_LOOK) && (obj1._id == NEWSPAPER)) { _vm->renderMessage(kStringGenericInteract_10); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->renderMessage(kStringGenericInteract_11); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->setCurrentImage(2); _vm->renderImage(0); _vm->setColor63(40); getInput(); _vm->renderRoom(*_currentRoom); roomBrightness(); _vm->renderMessage(kStringGenericInteract_12); } else if ((verb == ACTION_LOOK) && (obj1._id == KEYCARD2)) { _vm->renderMessage(obj1._description); obj1._description = kStringKeycard2Description2; } else if ((verb == ACTION_LOOK) && (obj1._id == WATCH)) _vm->renderMessage(kStringGenericInteract_13, kMessageNormal, timeToString(_state._time), timeToString(_state._timeAlarm)); else if ((verb == ACTION_PRESS) && (obj1._id == WATCH)) { bool validInput = true; int hours = 0; int minutes = 0; animationOff(); _vm->saveScreen(88, 87, 144, 24); _vm->renderBox(88, 87, 144, 24, kColorWhite35); _vm->renderText(kStringGenericInteract_14, 91, 90, kColorWhite99); Common::String input; do { validInput = true; input.clear(); _vm->renderBox(91, 99, 138, 9, kColorDarkBlue); edit(input, 91, 100, 5); int seperator = -1; for (uint i = 0; i < input.size(); ++i) { if (input[i] == ':') { seperator = i; break; } } if ((seperator == -1) || (seperator > 2)) { validInput = false; continue; } int decimalPlace = 1; for (int i = 0; i < seperator; ++i) { if (Common::isDigit(input[i])) { hours = hours * decimalPlace + (input[i] - '0'); decimalPlace *= 10; } else { validInput = false; break; } } decimalPlace = 1; for (uint i = seperator + 1; i < input.size(); ++i) { if (Common::isDigit(input[i])) { minutes = minutes * decimalPlace + (input[i] - '0'); decimalPlace *= 10; } else { validInput = false; break; } } if ((hours > 23) || (minutes > 59)) validInput = false; animationOn(); } while (!validInput && (_key.keycode != Common::KEYCODE_ESCAPE)); _vm->restoreScreen(); if (_key.keycode != Common::KEYCODE_ESCAPE) { _state._timeAlarm = (hours * 60 + minutes) * 60 * 1000; _state._alarmOn = (_state._timeAlarm > _state._time); } } else if ((verb == ACTION_USE) && Object::combine(obj1, obj2, TERMINALSTRIP, WIRE)) { Room *r = _rooms[CABIN_L3]; if (!r->getObject(8)->hasProperty(CARRIED)) { if (r->isSectionVisible(26)) _vm->renderMessage(kStringTakeMessage); else return false; } else { r->getObject(8)->_name = kStringWireAndClip; r = _rooms[HOLD]; _inventory.remove(*r->getObject(2)); _state._terminalStripConnected = true; _state._terminalStripWire = true; _vm->renderMessage(kStringOk); } } else if ((verb == ACTION_USE) && Object::combine(obj1, obj2, TERMINALSTRIP, SPOOL)) { Room *r = _rooms[CABIN_L2]; takeObject(*r->getObject(9)); r->getObject(9)->_name = kSringSpoolAndClip; r = _rooms[HOLD]; _inventory.remove(*r->getObject(2)); _state._terminalStripConnected = true; _vm->renderMessage(kStringOk); } else if ((verb == ACTION_USE) && Object::combine(obj1, obj2, WIRE, SPOOL)) { Room *r = _rooms[CABIN_L3]; if (!_state._terminalStripConnected) { if (r->isSectionVisible(26)) _vm->renderMessage(kStringCable3); else return false; } else { if (!r->getObject(8)->hasProperty(CARRIED)) _vm->renderMessage(kStringTakeMessage); else { r = _rooms[CABIN_L2]; takeObject(*r->getObject(9)); r = _rooms[CABIN_L3]; r->getObject(8)->_name = kStringGeneratorWire; r = _rooms[CABIN_L2]; _inventory.remove(*r->getObject(9)); _state._cableConnected = true; _vm->renderMessage(kStringOk); } } } else if ((verb == ACTION_USE) && (obj1._id == SUIT)) { takeObject(obj1); if ((_currentRoom->getId() >= ENTRANCE) && (_currentRoom->getId() <= ROGER)) { if (obj1.hasProperty(WORN)) { _vm->renderMessage(kStringGenericInteract_15); _rooms[AIRLOCK]->getObject(4)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(5)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(6)->disableProperty(WORN); } else _vm->renderMessage(kStringGenericInteract_16); } else { if (obj1.hasProperty(WORN)) { Room *r = _rooms[AIRLOCK]; if (r->getObject(4)->hasProperty(WORN)) _vm->renderMessage(kStringGenericInteract_17); else if (r->getObject(6)->hasProperty(WORN)) _vm->renderMessage(kStringGenericInteract_18); else { obj1.disableProperty(WORN); _vm->renderMessage(kStringGenericInteract_19); } } else { obj1.setProperty(WORN); _vm->renderMessage(kStringGenericInteract_20); } } } else if ((verb == ACTION_USE) && (obj1._id == HELMET)) { takeObject(obj1); if ((_currentRoom->getId() >= ENTRANCE) && (_currentRoom->getId() <= ROGER)) { if (obj1.hasProperty(WORN)) { _vm->renderMessage(kStringGenericInteract_21); _rooms[AIRLOCK]->getObject(4)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(5)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(6)->disableProperty(WORN); } else _vm->renderMessage(kStringGenericInteract_22); } else if (obj1.hasProperty(WORN)) { obj1.disableProperty(WORN); _vm->renderMessage(kStringGenericInteract_24); getInput(); if (airless()) dead(kStringGenericInteract_23); } else { Room *r = _rooms[AIRLOCK]; if (r->getObject(5)->hasProperty(WORN)) { obj1.setProperty(WORN); _vm->renderMessage(kStringGenericInteract_25); } else _vm->renderMessage(kStringGenericInteract_26); } } else if ((verb == ACTION_USE) && (obj1._id == LIFESUPPORT)) { takeObject(obj1); if ((_currentRoom->getId() >= ENTRANCE) && (_currentRoom->getId() <= ROGER)) { if (obj1.hasProperty(WORN)) { _vm->renderMessage(kStringGenericInteract_21); _rooms[AIRLOCK]->getObject(4)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(5)->disableProperty(WORN); _rooms[AIRLOCK]->getObject(6)->disableProperty(WORN); } else _vm->renderMessage(kStringGenericInteract_22); } else if (obj1.hasProperty(WORN)) { obj1.disableProperty(WORN); _vm->renderMessage(kStringGenericInteract_28); getInput(); if (airless()) dead(kStringGenericInteract_27); } else { Room *r = _rooms[AIRLOCK]; if (r->getObject(5)->hasProperty(WORN)) { obj1.setProperty(WORN); _vm->renderMessage(kStringGenericInteract_29); } else _vm->renderMessage(kStringGenericInteract_26); } } else if ((verb == ACTION_WALK) && (obj1._id == BATHROOM_DOOR)) { _rooms[BATHROOM]->getObject(2)->_exitRoom = _currentRoom->getId(); return false; } else if ((verb == ACTION_USE) && Object::combine(obj1, obj2, WIRE, SOCKET)) _vm->renderMessage(kStringGenericInteract_30); else if ((verb == ACTION_LOOK) && (obj1._id == BOOK2)) { _vm->renderMessage(kStringGenericInteract_31); waitOnInput(_messageDuration); _vm->removeMessage(); _vm->renderMessage(kStringGenericInteract_32); } else return false; return true; } void GameManager::handleInput() { bool validCommand = genericInteract(_inputVerb, *_inputObject[0], *_inputObject[1]); if (!validCommand) validCommand = _currentRoom->interact(_inputVerb, *_inputObject[0], *_inputObject[1]); if (!validCommand) { switch (_inputVerb) { case ACTION_LOOK: _vm->renderMessage(_inputObject[0]->_description); break; case ACTION_WALK: if (_inputObject[0]->hasProperty(CARRIED)) { // You already carry this. _vm->renderMessage(kStringGenericInteract_33); } else if (!_inputObject[0]->hasProperty(EXIT)) { // You're already there. _vm->renderMessage(kStringGenericInteract_34); } else if (_inputObject[0]->hasProperty(OPENABLE) && !_inputObject[0]->hasProperty(OPENED)) { // This is closed _vm->renderMessage(kStringShipHold9); } else changeRoom(_inputObject[0]->_exitRoom); break; case ACTION_TAKE: if (_inputObject[0]->hasProperty(OPENED)) { // You already have that _vm->renderMessage(kStringGenericInteract_35); } else if (_inputObject[0]->hasProperty(UNNECESSARY)) { // You do not need that. _vm->renderMessage(kStringGenericInteract_36); } else if (!_inputObject[0]->hasProperty(TAKE)) { // You can't take that. _vm->renderMessage(kStringGenericInteract_37); } else takeObject(*_inputObject[0]); break; case ACTION_OPEN: if (!_inputObject[0]->hasProperty(OPENABLE)) { // This can't be opened _vm->renderMessage(kStringGenericInteract_38); } else if (_inputObject[0]->hasProperty(OPENED)) { // This is already opened. _vm->renderMessage(kStringGenericInteract_39); } else if (_inputObject[0]->hasProperty(CLOSED)) { // This is locked. _vm->renderMessage(kStringGenericInteract_40); } else { _vm->renderImage(_inputObject[0]->_section); _inputObject[0]->setProperty(OPENED); byte i = _inputObject[0]->_click; _inputObject[0]->_click = _inputObject[0]->_click2; _inputObject[0]->_click2 = i; _sound->play(kAudioDoorOpen); } break; case ACTION_CLOSE: if (!_inputObject[0]->hasProperty(OPENABLE) || (_inputObject[0]->hasProperty(CLOSED) && _inputObject[0]->hasProperty(OPENED))) { // This can't be closed. _vm->renderMessage(kStringGenericInteract_41); } else if (!_inputObject[0]->hasProperty(OPENED)) { // This is already closed. _vm->renderMessage(kStringCloseLocker_1); } else { _vm->renderImage(invertSection(_inputObject[0]->_section)); _inputObject[0]->disableProperty(OPENED); byte i = _inputObject[0]->_click; _inputObject[0]->_click = _inputObject[0]->_click2; _inputObject[0]->_click2 = i; _sound->play(kAudioDoorClose); } break; case ACTION_GIVE: if (_inputObject[0]->hasProperty(CARRIED)) { // Better keep it! _vm->renderMessage(kStringGenericInteract_42); } break; default: // This is not possible. _vm->renderMessage(kStringGenericInteract_43); } } } void GameManager::executeRoom() { if (_processInput && !_vm->_screen->isMessageShown() && _guiEnabled) { handleInput(); if (_mouseClicked) { Common::Event event; event.type = Common::EVENT_MOUSEMOVE; event.mouse = Common::Point(0, 0); _vm->getEventManager()->pushEvent(event); event.type = Common::EVENT_MOUSEMOVE; event.mouse = Common::Point(_mouseX, _mouseY); _vm->getEventManager()->pushEvent(event); } resetInputState(); } if (_guiEnabled) { if (!_vm->_screen->isMessageShown()) { g_system->fillScreen(kColorBlack); _vm->renderRoom(*_currentRoom); } drawMapExits(); drawInventory(); drawStatus(); drawCommandBox(); } roomBrightness(); if (_vm->_screen->getViewportBrightness() == 0) _vm->paletteFadeIn(); if (!_currentRoom->hasSeen() && _newRoom) { _newRoom = false; _currentRoom->onEntrance(); } } void GameManager::guardShot() { _vm->renderImage(2); _vm->renderImage(5); wait(3); _vm->renderImage(2); _sound->play(kAudioVoiceHalt); while (_sound->isPlaying()) wait(1); _vm->renderImage(5); wait(5); _vm->renderImage(3); wait(3); shot(4, 3); } void GameManager::guard3Shot() { _vm->renderImage(1); wait(3); _sound->play(kAudioVoiceHalt); // 46/0 while (_sound->isPlaying()) wait(1); wait(5); _vm->renderImage(2); wait(3); shot(3,2); } void GameManager::alarm() { if (_rooms[INTRO]->getObject(2)->hasProperty(CARRIED)) { alarmSound(); if (_currentRoom->getId() == GUARD) guardShot(); else if (_currentRoom->getId() == CORRIDOR4 || _currentRoom->getId() == CORRIDOR7) { guardNoticed(); _state._corridorSearch = true; } else if (_currentRoom->getId() == GUARD3) guard3Shot(); else if (_currentRoom->getId() == CORRIDOR1) busted(33); } else { if (_currentRoom->getId() == CORRIDOR2 || _currentRoom->getId() == CORRIDOR4 || _currentRoom->getId() == GUARD || _currentRoom->getId() == CORRIDOR7 || _currentRoom->getId() == CELL) { alarmSound(); if (_currentRoom->getId() == GUARD) guardShot(); guardNoticed(); if (_currentRoom->getId() == CORRIDOR4) _state._corridorSearch = true; } _rooms[GUARD]->setSectionVisible(1, true); _rooms[GUARD]->getObject(3)->_click = 255; if (!_rooms[GUARD]->getObject(5)->hasProperty(CARRIED)) { _rooms[GUARD]->setSectionVisible(7, true); _rooms[GUARD]->getObject(5)->_click = 4; } _state._eventTime = _state._time + ticksToMsec(180); _state._eventCallback = kGuardReturnedFn; } } void GameManager::alarmSound() { animationOff(); _vm->removeMessage(); _vm->renderMessage(kStringAlarm); int32 end = _state._time + ticksToMsec(_messageDuration); do { _sound->play(kAudioAlarm); while (_sound->isPlaying()) { g_system->delayMillis(_vm->_delay); updateEvents(); g_system->updateScreen(); } } while (_state._time < end && !_vm->shouldQuit()); _vm->removeMessage(); animationOn(); } }