/* 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 "hdb/hdb.h" namespace HDB { static const char *cineTypeStr[] = { "C_NO_COMMAND", "C_STOPCINE", "C_LOCKPLAYER", "C_UNLOCKPLAYER", "C_SETCAMERA", "C_MOVECAMERA", "C_WAIT", "C_WAITUNTILDONE", "C_MOVEENTITY", "C_DIALOG", "C_ANIMENTITY", "C_RESETCAMERA", "C_SETENTITY", "C_STARTMAP", "C_MOVEPIC", "C_MOVEMASKEDPIC", "C_DRAWPIC", "C_DRAWMASKEDPIC", "C_FADEIN", "C_FADEOUT", "C_SPAWNENTITY", "C_PLAYSOUND", "C_CLEAR_FG", "C_SET_FG", "C_SET_BG", "C_FUNCTION", "C_ENTITYFACE", "C_USEENTITY", "C_REMOVEENTITY", "C_SETANIMFRAME", "C_TEXTOUT", "C_CENTERTEXTOUT", "C_PLAYVOICE", "C_ENDLIST" }; void AI::processCines() { AIEntity *e; bool complete, bailOut; if (!_cineActive) { return; } bailOut = complete = false; // Make sure Dialogs are timing out g_hdb->_window->checkDialogClose(0, 0); // Make sure Cine Pics are drawing for (int i = 0; i < _numCineBlitList; i++) { if (_cineBlitList[i]->masked == false) _cineBlitList[i]->pic->draw(_cine[i]->x, _cine[i]->y); else _cineBlitList[i]->pic->drawMasked(_cine[i]->x, _cine[i]->y); } // TODO: Check for Game Pause for (uint i = 0; i < _cine.size(); i++) { debug(3, "processCines: [%d] %s now: %d start: %d delay: %d", i, cineTypeStr[_cine[i]->cmdType], g_system->getMillis(), _cine[i]->start, _cine[i]->delay); switch (_cine[i]->cmdType) { case C_STOPCINE: char func[64]; memset(func, 0, 64); if (_cine[i]->title) strcpy(func, _cine[i]->title); cineCleanup(); if (func[0]) g_hdb->_lua->callFunction(func, 0); break; case C_LOCKPLAYER: _playerLock = true; complete = true; if (_player) { stopEntity(_player); } clearWaypoints(); break; case C_UNLOCKPLAYER: _playerLock = false; complete = true; break; case C_SETCAMERA: _cameraX = _cine[i]->x; _cameraY = _cine[i]->y; g_hdb->_map->centerMapXY((int)_cameraX + 16, (int)_cameraY + 16); _cameraLock = true; complete = true; break; case C_RESETCAMERA: int px, py; _cameraLock = false; g_hdb->_ai->getPlayerXY(&px, &py); g_hdb->_map->centerMapXY(px + 16, py + 16); complete = true; break; case C_MOVECAMERA: _cameraLock = true; if (!(_cine[i]->start)) { debug(3, "C_MOVECAMERA: [%d] now: x: %f, y: %f, speed: %d", i, _cine[i]->x, _cine[i]->y, _cine[i]->speed); _cine[i]->xv = (((double)_cine[i]->x) - _cameraX) / (double)_cine[i]->speed; _cine[i]->yv = (((double)_cine[i]->y) - _cameraY) / (double)_cine[i]->speed; _cine[i]->start = 1; } _cameraX += _cine[i]->xv; _cameraY += _cine[i]->yv; debug(3, "C_MOVECAMERA: _cine[%d]->xv: %f, _cine[%d]->yv: %f", i, _cine[i]->xv, i, _cine[i]->yv); debug(3, "C_MOVECAMERA: abs(_cameraX - _cine[i]->x): %f, abs(_cameraY - _cine[i]->y): %f", abs(_cameraX - _cine[i]->x), abs(_cameraY - _cine[i]->y)); if (abs(_cameraX - _cine[i]->x) <= 1 && abs(_cameraY - _cine[i]->y) <= 1) { _cameraX = _cine[i]->x; _cameraY = _cine[i]->y; complete = true; } g_hdb->_map->centerMapXY((int)_cameraX + 16, (int)_cameraY + 16); break; case C_WAIT: if (!(_cine[i]->start)) { _cine[i]->start = 1; _cine[i]->delay = g_system->getMillis() + _cine[i]->delay * 1000; } else { if (_cine[i]->delay < g_system->getMillis()) { complete = true; } else { bailOut = true; } } break; case C_WAITUNTILDONE: if (!i) { complete = true; } else { bailOut = true; } break; case C_SETENTITY: _cine[i]->e = locateEntity(_cine[i]->string); if (_cine[i]->e) { _cine[i]->e->tileX = (int)_cine[i]->x / kTileWidth; _cine[i]->e->x = (int)_cine[i]->x; _cine[i]->e->tileY = (int)_cine[i]->y / kTileHeight; _cine[i]->e->y = (int)_cine[i]->y; _cine[i]->e->level = (int)_cine[i]->x2; debug(2, "Found '%s' in setEntity", _cine[i]->string); } else { warning("Can't locate '%s' in setEntity", _cine[i]->string); } complete = true; break; case C_MOVEENTITY: if (!_cine[i]->start) { e = locateEntity(_cine[i]->title); if (e) { _cine[i]->e = e; _cine[i]->e->moveSpeed = _cine[i]->speed; _cine[i]->e->level = (int)_cine[i]->x2; setEntityGoal(_cine[i]->e, (int)_cine[i]->x, (int)_cine[i]->y); _cine[i]->start = 1; } else { warning("Can't locate '%s' in moveEntity", _cine[i]->title); } } else { debug(3, "C_MOVEENTITY: _cine[%d]->e->tileX: %d, _cine[%d]->e->goalX: %d", i, _cine[i]->e->tileX, i, _cine[i]->e->goalX); debug(3, "C_MOVEENTITY: _cine[%d]->e->tileY: %d, _cine[%d]->e->goalY: %d", i, _cine[i]->e->tileY, i, _cine[i]->e->goalY); if (!_cine[i]->e->goalX) { complete = true; } } break; case C_ANIMENTITY: if (!_cine[i]->start) { e = locateEntity(_cine[i]->title); if (e) { _cine[i]->e = e; e->state = (AIState)_cine[i]->speed; _cine[i]->start = 1; if (_cine[i]->end) // Loop ? complete = true; e->animFrame = 0; e->animDelay = e->animCycle; animEntFrames(e); } else { warning("Can't locate '%s' in animEntity", _cine[i]->title); complete = true; } } else { e = _cine[i]->e; if (!e->animFrame && e->animDelay == e->animCycle) { e->state = STATE_STANDDOWN; e->animFrame = 0; e->animDelay = e->animCycle; complete = true; } } break; case C_SETANIMFRAME: e = locateEntity(_cine[i]->title); if (e) { e->state = (AIState)_cine[i]->start; e->animFrame = _cine[i]->end; e->animDelay = e->animCycle; animEntFrames(e); e->state = STATE_NONE; complete = true; } break; case C_ENTITYFACE: { e = locateEntity(_cine[i]->title); if (e) { int d = (int)_cine[i]->x; e->dir = (AIDir)d; switch (e->dir) { case DIR_UP: e->state = STATE_STANDUP; break; case DIR_DOWN: e->state = STATE_STANDDOWN; break; case DIR_LEFT: e->state = STATE_STANDLEFT; break; case DIR_RIGHT: e->state = STATE_STANDRIGHT; break; default: warning("AI-CINEMATIC: processCines: DIR_NONE"); } } else { warning("Can't find %s to ENTITYFACE", _cine[i]->title); } complete = true; break; } case C_DIALOG: debug("In processCines: C_DIALOG: _cine[i]->start: %d", _cine[i]->start); if (_cine[i]->start) { g_hdb->_window->openDialog(_cine[i]->title, -1, _cine[i]->string, 0, NULL); g_hdb->_window->setDialogDelay(_cine[i]->delay); _cine[i]->start = 0; } else { if (g_hdb->_window->getDialogDelay() < g_hdb->getTimeSlice()) complete = true; } break; case C_MOVEMASKEDPIC: if (!_cine[i]->start) { Picture *pic = cineFindInBlitList(_cine[i]->id); if (!pic) { pic = g_hdb->_drawMan->loadPic(_cine[i]->string); cineAddToFreeList(pic); } else cineRemoveFromBlitList(_cine[i]->id); _cine[i]->pic = pic; _cine[i]->start = 1; } cineRemoveFromBlitList(_cine[i]->id); _cine[i]->x += _cine[i]->xv; _cine[i]->y += _cine[i]->yv; cineAddToBlitList(_cine[i]->id, _cine[i]->pic, (int)_cine[i]->x, (int)_cine[i]->y, true); if (abs((int)(_cine[i]->x - _cine[i]->x2)) <= 1 && abs((int)(_cine[i]->y - _cine[i]->y2)) <= 1) complete = true; break; case C_USEENTITY: for (Common::Array::iterator it = _ents->begin(); it != _ents->end(); it++) { if ((*it)->entityName && Common::matchString((*it)->entityName, _cine[i]->string)) { g_hdb->useEntity((*it)); } } for (int k = 0; k < kMaxActions; k++) { if (_actions[k].entityName && Common::matchString(_actions[k].entityName, _cine[i]->string)) { checkActionList(&_dummyPlayer, _actions[k].x1, _actions[k].y1, false); checkActionList(&_dummyPlayer, _actions[k].x2, _actions[k].y2, false); } } for (int j = 0; j < kMaxAutoActions; j++) { if (_autoActions[j].entityName && Common::matchString(_autoActions[j].entityName, _cine[i]->string) && !_autoActions[j].activated) checkAutoList(&_dummyPlayer, _autoActions[j].x, _autoActions[j].y); } complete = true; break; case C_FADEIN: if (!_cine[i]->start) { g_hdb->_drawMan->setFade(true, (bool)_cine[i]->end, _cine[i]->speed); _cine[i]->start = 1; } else if (!g_hdb->_drawMan->isFadeActive()) { complete = true; } break; case C_FADEOUT: if (!_cine[i]->start) { g_hdb->_drawMan->setFade(false, (bool)_cine[i]->end, _cine[i]->speed); _cine[i]->start = 1; } else if (!g_hdb->_drawMan->isFadeActive()) { complete = true; } break; case C_CLEAR_FG: g_hdb->_map->setMapFGTileIndex((int)_cine[i]->x, (int)_cine[i]->y, -1); g_hdb->_map->removeFGTileAnimation((int)_cine[i]->x, (int)_cine[i]->y); complete = true; break; case C_SET_BG: g_hdb->_map->setMapBGTileIndex((int)_cine[i]->x, (int)_cine[i]->y, (int)_cine[i]->start); g_hdb->_map->addBGTileAnimation((int)_cine[i]->x, (int)_cine[i]->y); complete = true; break; case C_SET_FG: g_hdb->_map->setMapFGTileIndex((int)_cine[i]->x, (int)_cine[i]->y, (int)_cine[i]->start); g_hdb->_map->addFGTileAnimation((int)_cine[i]->x, (int)_cine[i]->y); complete = true; break; default: warning("STUB: AI::PROCESSCINES incomplete for %d", _cine[i]->cmdType); break; } if (bailOut) { return; } if (complete) { if (_cine.size()) { _cine.remove_at(i); i--; complete = false; } } } } void AI::cineCleanup() { cineFreeGfx(); _cineActive = false; // If aborted and abort function specified, call it if (_cineAborted && _cineAbortFunc) g_hdb->_lua->callFunction(_cineAbortFunc, 0); _cameraLock = false; _playerLock = false; g_hdb->_window->setInfobarDark(0); warning("STUB: DrawMan:: Set Pointer State"); int px, py; getPlayerXY(&px, &py); g_hdb->_map->centerMapXY(px + 16, py + 16); } void AI::cineAbort() { for (Common::Array::iterator it = _cine.begin(); it != _cine.end(); it++) { if ((*it)->cmdType == C_STARTMAP || (*it)->cmdType == C_STOPCINE) _cine[0] = *it; if (it != _cine.begin()) _cine.erase(it); } warning("STUB: Window: closeAll() required"); if (_player) stopEntity(_player); _cineAborted = true; } void AI::cineAddToBlitList(const char *id, Picture *pic, int x, int y, bool masked) { _cineBlitList[_numCineBlitList] = new CineBlit; _cineBlitList[_numCineBlitList]->id = id; _cineBlitList[_numCineBlitList]->pic = pic; _cineBlitList[_numCineBlitList]->x = x; _cineBlitList[_numCineBlitList]->y = y; _cineBlitList[_numCineBlitList]->masked = masked; _numCineBlitList++; } Picture *AI::cineFindInBlitList(const char *name) { for (int i = 0; i < _numCineBlitList; i++) { if (Common::matchString(_cineBlitList[i]->id, name)) return _cineBlitList[i]->pic; } return NULL; } void AI::cineRemoveFromBlitList(const char *name) { for (int i = 0; i < _numCineBlitList; i++) { if (Common::matchString(_cineBlitList[i]->id, name)) delete _cineBlitList[i]; for (; i < _numCineBlitList - 1; i++) _cineBlitList[i] = _cineBlitList[i + 1]; _numCineBlitList--; _cineBlitList[_numCineBlitList] = NULL; return; } } void AI::cineAddToFreeList(Picture *pic) { if (_numCineFreeList >= kMaxCineGfx) { warning("cineAddToFreeList: Too many gfx in Cinematic!"); return; } _cineFreeList[_numCineFreeList] = pic; _numCineFreeList++; } void AI::cineFreeGfx() { for (int i = 0; i < _numCineFreeList; i++) { delete _cineFreeList[i]; } _numCineFreeList = 0; } void AI::cineStart(bool abortable, const char *abortFunc) { _cineAbortable = abortable; _cineAborted = false; _cineAbortFunc = abortFunc; _numCineBlitList = 0; _numCineFreeList = 0; _cineActive = true; _playerLock = false; _cameraLock = false; } void AI::cineStop(const char *funcNext) { CineCommand *cmd = new CineCommand; cmd->cmdType = C_STOPCINE; if (funcNext) strcpy(cmd->title, funcNext); _cine.push_back(cmd); } void AI::cineLockPlayer() { CineCommand *cmd = new CineCommand; cmd->cmdType = C_LOCKPLAYER; _cine.push_back(cmd); } void AI::cineUnlockPlayer() { CineCommand *cmd = new CineCommand; cmd->cmdType = C_UNLOCKPLAYER; _cine.push_back(cmd); } void AI::cineSetCamera(int x, int y) { CineCommand *cmd = new CineCommand; cmd->x = x * kTileWidth; cmd->y = y * kTileHeight; cmd->cmdType = C_SETCAMERA; _cine.push_back(cmd); } void AI::cineResetCamera() { CineCommand *cmd = new CineCommand; cmd->cmdType = C_RESETCAMERA; _cine.push_back(cmd); } void AI::cineMoveCamera(int x, int y, int speed) { CineCommand *cmd = new CineCommand; cmd->start = 0; cmd->x = x * kTileWidth; cmd->y = y * kTileHeight; cmd->speed = speed; debug(2, "Setting up C_MOVECAMERA: x: %f, y: %f", cmd->x, cmd->y); cmd->cmdType = C_MOVECAMERA; _cine.push_back(cmd); } void AI::cineWait(int seconds) { CineCommand *cmd = new CineCommand; cmd->start = 0; cmd->cmdType = C_WAIT; cmd->delay = seconds; _cine.push_back(cmd); } void AI::cineWaitUntilDone() { CineCommand *cmd = new CineCommand; cmd->cmdType = C_WAITUNTILDONE; _cine.push_back(cmd); } void AI::cineSetEntity(const char *entName, int x, int y, int level) { CineCommand *cmd = new CineCommand; strcpy(cmd->string, entName); cmd->x = x * kTileWidth; cmd->y = y * kTileHeight; cmd->x2 = level; cmd->cmdType = C_SETENTITY; _cine.push_back(cmd); } void AI::cineMoveEntity(const char *entName, int x, int y, int level, int speed) { CineCommand *cmd = new CineCommand; cmd->x = x; cmd->y = y; cmd->x2 = level; cmd->start = 0; cmd->speed = speed; strcpy(cmd->title, entName); cmd->cmdType = C_MOVEENTITY; _cine.push_back(cmd); } void AI::cineAnimEntity(const char *entName, AIState state, int loop) { CineCommand *cmd = new CineCommand; cmd->start = 0; strcpy(cmd->title, entName); cmd->speed = state; cmd->end = loop; cmd->cmdType = C_ANIMENTITY; _cine.push_back(cmd); } void AI::cineSetAnimFrame(const char *entName, AIState state, int frame) { CineCommand *cmd = new CineCommand; cmd->start = state; strcpy(cmd->title, entName); cmd->end = frame; cmd->cmdType = C_SETANIMFRAME; _cine.push_back(cmd); } void AI::cineEntityFace(const char *luaName, double dir) { CineCommand *cmd = new CineCommand; strcpy(cmd->title, luaName); cmd->x = dir; cmd->cmdType = C_ENTITYFACE; _cine.push_back(cmd); } void AI::cineDialog(const char *title, const char *string, int seconds) { CineCommand *cmd = new CineCommand; strcpy(cmd->title, title); strcpy(cmd->string, string); cmd->delay = seconds; cmd->start = 1; if (!title || !string) warning("cineDialog: Missing Title or Text"); cmd->cmdType = C_DIALOG; debug("In cineDialog: C_DIALOG created. cmd->start: %d", cmd->start); _cine.push_back(cmd); } void AI::cineMoveMaskedPic(const char *id, const char *pic, int x1, int y1, int x2, int y2, int speed) { if (!pic || !id) { warning("cineMoveMaskedPic: Missing ID or PIC"); return; } CineCommand *cmd = new CineCommand; cmd->x = x1; cmd->y = y1; cmd->x2 = x2; cmd->y2 = y2; cmd->speed = speed; cmd->xv = ((double)(x2-x1)) / (double)speed; cmd->yv = ((double)(y2-y1)) / (double)speed; cmd->start = 0; strcpy(cmd->string, pic); strcpy(cmd->id, id); cmd->cmdType = C_MOVEMASKEDPIC; _cine.push_back(cmd); } void AI::cineUse(const char *entName) { CineCommand *cmd = new CineCommand; strcpy(cmd->string, entName); cmd->cmdType = C_USEENTITY; _cine.push_back(cmd); } void AI::cineFadeIn(bool isBlack, int steps) { CineCommand *cmd = new CineCommand; cmd->speed = steps; cmd->end = (int) isBlack; cmd->start = 0; cmd->cmdType = C_FADEIN; _cine.push_back(cmd); } void AI::cineFadeOut(bool isBlack, int steps) { CineCommand *cmd = new CineCommand; cmd->speed = steps; cmd->end = (int) isBlack; cmd->start = 0; cmd->cmdType = C_FADEOUT; _cine.push_back(cmd); } void AI::cineClearForeground(int x, int y) { CineCommand *cmd = new CineCommand; cmd->x = x; cmd->y = y; cmd->cmdType = C_CLEAR_FG; _cine.push_back(cmd); } void AI::cineSetBackground(int x, int y, int index) { CineCommand *cmd = new CineCommand; cmd->x = x; cmd->y = y; cmd->start = index; cmd->cmdType = C_SET_BG; _cine.push_back(cmd); } void AI::cineSetForeground(int x, int y, int index) { CineCommand *cmd = new CineCommand; cmd->x = x; cmd->y = y; cmd->start = index; cmd->cmdType = C_SET_FG; _cine.push_back(cmd); } } // End of Namespace