/* 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 "ngi/ngi.h" #include "ngi/objects.h" #include "ngi/messages.h" #include "ngi/modal.h" #include "ngi/statics.h" #include "ngi/gameloader.h" namespace NGI { ExCommand::ExCommand() { _field_3C = 1; _messageNum = 0; _excFlags = 0; _parId = 0; } ExCommand::ExCommand(ExCommand *src) : Message(*src) { _field_3C = 1; _messageNum = src->_messageNum; _excFlags = src->_excFlags; _parId = src->_parId; } ExCommand *ExCommand::createClone() { return new ExCommand(this); } ExCommand::ExCommand(int16 parentId, int messageKind, int messageNum, int x, int y, int a7, int a8, int sceneClickX, int sceneClickY, int a11) : Message(parentId, messageKind, x, y, a7, a8, sceneClickX, sceneClickY, a11) { _field_3C = 1; _messageNum = messageNum; _excFlags = 0; _parId = 0; } struct exDesc { byte num; const char *name; } static const exTypes[] = { { 1, "START_MOVEMENT" }, { 5, "SHOW" }, { 17, "MESSAGE" }, { 63, "USER" }, { 0, "" } }; static const char *exCommandType2str(int type) { static char buf[10]; for (int i = 0; exTypes[i].num; i++) if (exTypes[i].num == type) return exTypes[i].name; snprintf(buf, 10, "%d", type); return buf; } bool ExCommand::load(MfcArchive &file) { debugC(5, kDebugLoading, "ExCommand::load()"); _parentId = file.readUint16LE(); _messageKind = file.readUint32LE(); _x = file.readSint32LE(); _y = file.readSint32LE(); _z = file.readUint32LE(); _sceneClickX = file.readUint32LE(); _sceneClickY = file.readUint32LE(); _invId = file.readUint32LE(); _field_24 = file.readUint32LE(); _param = file.readUint32LE(); _field_2C = file.readUint32LE(); _field_30 = file.readUint32LE(); _field_34 = file.readUint32LE(); _messageNum = file.readUint32LE(); _field_3C = 0; if (g_nmi->_gameProjectVersion >= 12) { _excFlags = file.readUint32LE(); _parId = file.readUint32LE(); } _objtype = kObjTypeExCommand; debugC(6, kDebugXML, "%% ", _parentId, exCommandType2str(_messageKind), _x, _y, _z, _sceneClickX, _sceneClickY, _invId, _field_24, _param, _field_2C, _field_30, _field_34, _messageNum, _excFlags, _parId); return true; } bool ExCommand::handleMessage() { int cnt = 0; for (MessageHandler *m = g_nmi->_messageHandlers; m; m = m->nextItem) cnt += m->callback(this); if (_messageKind == 17 || (_excFlags & 1)) { if (_parId) { MessageQueue *mq = g_nmi->_globalMessageQueueList->getMessageQueueById(_parId); if (mq) mq->update(); } } if (_excFlags & 2) delete this; return (cnt > 0); } void ExCommand::sendMessage() { g_nmi->_exCommandList.push_back(this); processMessages(); } void ExCommand::postMessage() { g_nmi->_exCommandList.push_back(this); } void ExCommand::handle() { if (g_nmi->_modalObject) { g_nmi->_modalObject->handleMessage(this); delete this; } else { postMessage(); } } void ExCommand::setf3c(int val) { if (val != -1) _field_3C = val; _field_34 = 1; } void ExCommand::firef34() { if (_field_34) { if (_field_3C >= _param) { _field_34 = 0; sendMessage(); if (!_field_30 ) setf3c(_field_2C); } } } ExCommand2::ExCommand2(int messageKind, int parentId, const PointList &points) : ExCommand(parentId, messageKind, 0, 0, 0, 0, 1, 0, 0, 0) { _objtype = kObjTypeExCommand2; _points = points; } ExCommand2::ExCommand2(ExCommand2 *src) : ExCommand(src), _points(src->_points) {} ExCommand2 *ExCommand2::createClone() { return new ExCommand2(this); } Message::Message() { _messageKind = 0; _parentId = 0; _x = 0; _y = 0; _z = 0; _sceneClickX = 0; _sceneClickY = 0; _invId = 0; _field_24 = 0; _param = 0; _field_2C = 0; _field_30 = 0; _field_34 = 0; } Message::Message(int16 parentId, int messageKind, int x, int y, int z, int a7, int sceneClickX, int sceneClickY, int a10) { _messageKind = messageKind; _parentId = parentId; _x = x; _y = y; _z = z; _sceneClickX = sceneClickX; _sceneClickY = sceneClickY; _field_24 = a7; _invId = a10; _param = 0; _field_2C = 0; _field_30 = 0; _field_34 = 0; } ObjstateCommand::ObjstateCommand() { _value = 0; _objtype = kObjTypeObjstateCommand; } ObjstateCommand::ObjstateCommand(ObjstateCommand *src) : ExCommand(src) { _value = src->_value; _objtype = kObjTypeObjstateCommand; _objCommandName = src->_objCommandName; } bool ObjstateCommand::load(MfcArchive &file) { debugC(5, kDebugLoading, "ObjStateCommand::load()"); _objtype = kObjTypeObjstateCommand; ExCommand::load(file); _value = file.readUint32LE(); _objCommandName = file.readPascalString(); debugC(6, kDebugXML, "%% ", transCyrillic(_objCommandName.c_str()), _value); return true; } ObjstateCommand *ObjstateCommand::createClone() { return new ObjstateCommand(this); } MessageQueue::MessageQueue() { _field_14 = 0; _parId = 0; _dataId = 0; _id = 0; _isFinished = 0; _flags = 0; _counter = 0; _field_38 = 0; _flag1 = 0; } MessageQueue::MessageQueue(int dataId) { _field_14 = 0; _parId = 0; _dataId = dataId; _id = g_nmi->_globalMessageQueueList->compact(); _isFinished = 0; _flags = 0; _counter = 0; _field_38 = 0; _flag1 = 0; } MessageQueue::MessageQueue(MessageQueue *src, int parId, int field_38) { _counter = 0; _field_38 = (field_38 == 0); for (Common::List::iterator it = src->_exCommands.begin(); it != src->_exCommands.end(); ++it) { ExCommand *ex = (*it)->createClone(); ex->_excFlags |= 2; _exCommands.push_back(ex); } _field_14 = src->_field_14; if (parId) _parId = parId; else _parId = src->_parId; _id = g_nmi->_globalMessageQueueList->compact(); _dataId = src->_dataId; _flags = src->_flags & ~kInGlobalQueue; _queueName = ""; g_nmi->_globalMessageQueueList->addMessageQueue(this); _isFinished = 0; _flag1 = 0; } MessageQueue::~MessageQueue() { for (Common::List::iterator it = _exCommands.begin(); it != _exCommands.end(); ++it) { ExCommand *ex = *it; if (ex && ex->_excFlags & 2) delete ex; } if (_field_14) delete _field_14; if (_flags & kInGlobalQueue) { g_nmi->_globalMessageQueueList->removeQueueById(_id); } finish(); } bool MessageQueue::load(MfcArchive &file) { debugC(5, kDebugLoading, "MessageQueue::load()"); _dataId = file.readUint16LE(); int count = file.readUint16LE(); assert(g_nmi->_gameProjectVersion >= 12); _queueName = file.readPascalString(); for (int i = 0; i < count; i++) { ExCommand *tmp = file.readClass(); tmp->_excFlags |= 2; _exCommands.push_back(tmp); } _id = -1; _field_14 = 0; _parId = 0; _isFinished = 0; return true; } bool MessageQueue::chain(StaticANIObject *ani) { if (checkGlobalExCommandList1() && checkGlobalExCommandList2()) { if (!(getFlags() & kInGlobalQueue)) { g_nmi->_globalMessageQueueList->addMessageQueue(this); } if (ani) { ani->queueMessageQueue(this); } else { sendNextCommand(); } return true; } return false; } void MessageQueue::update() { if (_counter > 0) _counter--; if (getCount()) { sendNextCommand(); } else if (_counter == 0) { _isFinished = 1; finish(); } } void MessageQueue::messageQueueCallback1(int par) { if (g_nmi->_isSaveAllowed && par == 16) { if (g_nmi->_globalMessageQueueList->size() && (*g_nmi->_globalMessageQueueList)[0] != 0) { for (uint i = 0; i < g_nmi->_globalMessageQueueList->size(); i++) { if ((*g_nmi->_globalMessageQueueList)[i]->_flags & 1) if ((*g_nmi->_globalMessageQueueList)[i] != this && !(*g_nmi->_globalMessageQueueList)[i]->_isFinished) return; } } if (g_nmi->_currentScene) g_nmi->_gameLoader->writeSavegame(g_nmi->_currentScene, "savetmp.sav", ""); } } void MessageQueue::addExCommand(ExCommand *ex) { _exCommands.push_front(ex); } void MessageQueue::addExCommandToEnd(ExCommand *ex) { _exCommands.push_back(ex); } void MessageQueue::insertExCommandAt(int pos, ExCommand *ex) { Common::List::iterator it = _exCommands.begin(); for (int i = pos; i > 0; i--) ++it; _exCommands.insert(it, ex); } ExCommand *MessageQueue::getExCommandByIndex(uint idx) { if (idx >= getCount()) return 0; Common::List::iterator it = _exCommands.begin(); while (idx) { ++it; idx--; } return *it; } void MessageQueue::deleteExCommandByIndex(uint idx, bool doFree) { if (idx >= getCount()) return; Common::List::iterator it = _exCommands.begin(); while (idx) { ++it; idx--; } if (doFree) delete *it; _exCommands.erase(it); } void MessageQueue::mergeQueue(MessageQueue *mq) { // Original belongs to AniHandler while (mq->_exCommands.size()) { _exCommands.push_back(mq->_exCommands.front()); mq->_exCommands.pop_front(); } } void MessageQueue::sendNextCommand() { if (getCount()) { if (!(_flags & 4) && (_flags & 1)) { messageQueueCallback1(16); } ExCommand *ex = _exCommands.front(); _exCommands.pop_front(); _counter++; ex->_parId = _id; ex->_excFlags |= (ex->_field_24 == 0 ? 1 : 0) | (ex->_field_3C != 0 ? 2 : 0); _flags |= 4; ex->sendMessage(); } else if (_counter <= 0) { _isFinished = 1; finish(); } } bool MessageQueue::checkGlobalExCommandList1() { ExCommand *ex, *ex1; for (uint i = 0; i < getCount(); i++) { ex = getExCommandByIndex(i); if (ex->_messageKind != 1 && ex->_messageKind != 20 && ex->_messageKind != 5 && ex->_messageKind != 27) continue; for (Common::List::iterator it = g_nmi->_exCommandList.begin(); it != g_nmi->_exCommandList.end(); it++) { ex1 = *it; if (ex1->_messageKind != 1 && ex1->_messageKind != 20 && ex1->_messageKind != 5 && ex1->_messageKind != 27) continue; if (ex1->_parentId != ex->_parentId) continue; if (ex1->_param != ex->_param && ex1->_param != -1 && ex->_param != -1) continue; MessageQueue *mq = g_nmi->_globalMessageQueueList->getMessageQueueById(ex1->_parId); if (mq) { if (mq->getFlags() & 1) return false; } } } return true; } bool MessageQueue::checkGlobalExCommandList2() { ExCommand *ex, *ex1; for (uint i = 0; i < getCount(); i++) { ex = getExCommandByIndex(i); if (ex->_messageKind != 1 && ex->_messageKind != 20 && ex->_messageKind != 5 && ex->_messageKind != 27) continue; for (Common::List::iterator it = g_nmi->_exCommandList.begin(); it != g_nmi->_exCommandList.end();) { ex1 = *it; if (ex1->_messageKind != 1 && ex1->_messageKind != 20 && ex1->_messageKind != 5 && ex1->_messageKind != 27) { it++; continue; } if (ex1->_parentId != ex->_parentId) { it++; continue; } if (ex1->_param != ex->_param && ex1->_param != -1 && ex->_param != -1) { it++; continue; } MessageQueue *mq = g_nmi->_globalMessageQueueList->getMessageQueueById(ex1->_parId); if (mq) { if (mq->getFlags() & 1) return false; delete mq; } it = g_nmi->_exCommandList.erase(it); if (ex1->_excFlags & 2) { delete ex1; } } } return true; } void MessageQueue::finish() { if (!_parId) return; MessageQueue *mq = g_nmi->_globalMessageQueueList->getMessageQueueById(_parId); _parId = 0; if (!mq) return; if (!_flag1) { mq->update(); return; } mq->_counter--; if (!mq->_counter && !mq->_exCommands.size()) mq->update(); } void MessageQueue::setParamInt(int key1, int key2) { for (uint i = 0; i < getCount(); i++) { ExCommand *ex = getExCommandByIndex(i); int k = ex->_messageKind; if ((k == 1 || k == 20 || k == 5 || k == 6 || k == 2 || k == 18 || k == 19 || k == 22 || k == 55) && ex->_param == key1) ex->_param = key2; } } int MessageQueue::calcDuration(StaticANIObject *obj) { int res = 0; ExCommand *ex; Movement *mov; for (uint i = 0; i < getCount(); i++) { ex = getExCommandByIndex(i); if (ex->_parentId == obj->_id) { if (ex->_messageKind == 1 || ex->_messageKind == 20) { if ((mov = obj->getMovementById(ex->_messageNum)) != 0) { if (ex->_z >= 1) res += ex->_z; else res += mov->calcDuration(); } } } } return res; } void MessageQueue::changeParam28ForObjectId(int objId, int oldParam28, int newParam28) { for (uint i = 0; i < getCount(); i++) { ExCommand *ex = getExCommandByIndex(i); int k = ex->_messageKind; if ((k == 1 || k == 20 || k == 5 || k == 6 || k == 2 || k == 18 || k == 19 || k == 22 || k == 55) && ex->_param == oldParam28 && ex->_parentId == objId) ex->_param = newParam28; } } int MessageQueue::activateExCommandsByKind(int kind) { int res = 0; for (uint i = 0; i < getCount(); i++) { ExCommand *ex = getExCommandByIndex(i); if (ex->_messageKind == kind) { ex->_messageKind = 0; ex->_excFlags |= 1; res++; } } return res; } MessageQueue *GlobalMessageQueueList::getMessageQueueById(int id) { for (Common::Array::iterator s = begin(); s != end(); ++s) { if ((*s)->_id == id) return *s; } return 0; } void GlobalMessageQueueList::deleteQueueById(int id) { for (uint i = 0; i < size(); i++) if (_storage[i]->_id == id) { delete remove_at(i); disableQueueById(id); return; } } void GlobalMessageQueueList::removeQueueById(int id) { for (uint i = 0; i < size(); i++) if (_storage[i]->_id == id) { _storage[i]->_flags &= ~kInGlobalQueue; remove_at(i); disableQueueById(id); return; } } void GlobalMessageQueueList::disableQueueById(int id) { for (Common::Array::iterator s = begin(); s != end(); ++s) { if ((*s)->_parId == id) (*s)->_parId = 0; } } int GlobalMessageQueueList::compact() { Common::Array useList(size() + 2); for (uint i = 0; i < size();) { if (_storage[i]->_isFinished) { disableQueueById(_storage[i]->_id); delete remove_at(i); } else { if ((uint)_storage[i]->_id < size() + 2) useList[_storage[i]->_id] = true; i++; } } uint i; for (i = 1; i < size() + 2; i++) { if (!useList[i]) break; } return i; } void GlobalMessageQueueList::addMessageQueue(MessageQueue *msg) { if ((msg->getFlags() & kInGlobalQueue) == 0) { msg->setFlags(msg->getFlags() | kInGlobalQueue); push_back(msg); } else { warning("Trying to add a MessageQueue already in the queue"); } } void GlobalMessageQueueList::clear() { for (iterator it = begin(); it != end(); ++it) { // The MessageQueue destructor will try to remove itself from the global // queue if it thinks it is in the global queue, which will break the // iteration over the list (*it)->_flags &= ~kInGlobalQueue; delete *it; } Common::Array::clear(); } void clearGlobalMessageQueueList() { g_nmi->_globalMessageQueueList->clear(); } void clearGlobalMessageQueueList1() { clearMessages(); g_nmi->_globalMessageQueueList->clear(); } void clearMessages() { while (g_nmi->_exCommandList.size()) { ExCommand *ex = g_nmi->_exCommandList.front(); g_nmi->_exCommandList.pop_front(); if (ex->_excFlags & 2) delete ex; } } bool removeMessageHandler(int16 id, int pos) { if (g_nmi->_messageHandlers) { MessageHandler *curItem = g_nmi->_messageHandlers; MessageHandler *prevItem = 0; int curPos = 0; while (id != curItem->id) { prevItem = curItem; curItem = curItem->nextItem; curPos++; if (!curItem) return false; } if (pos == -1 || curPos == pos) { prevItem->nextItem = curItem->nextItem; delete curItem; updateMessageHandlerIndex(prevItem->nextItem, -1); return true; } } return false; } void updateMessageHandlerIndex(MessageHandler *msg, int offset) { for (; msg; msg = msg->nextItem) msg->index += offset; } void addMessageHandler(int (*callback)(ExCommand *), int16 id) { if (getMessageHandlerById(id)) return; MessageHandler *curItem = g_nmi->_messageHandlers; if (!curItem) return; int index = 0; for (MessageHandler *i = g_nmi->_messageHandlers->nextItem; i; i = i->nextItem) { curItem = i; index++; } allocMessageHandler(curItem, id, callback, index); if (curItem) updateMessageHandlerIndex(curItem->nextItem->nextItem, 1); } MessageHandler *getMessageHandlerById(int16 id) { MessageHandler *curItem = g_nmi->_messageHandlers; if (!curItem) return 0; while (id != curItem->id) { curItem = curItem->nextItem; if (!curItem) return 0; } return curItem; } bool allocMessageHandler(MessageHandler *where, int16 id, int (*callback)(ExCommand *), int index) { MessageHandler *msg = new MessageHandler; if (where) { msg->nextItem = where->nextItem; where->nextItem = msg; msg->id = id; msg->callback = callback; msg->index = index; } else { msg->nextItem = 0; msg->id = id; msg->callback = callback; msg->index = 0; g_nmi->_messageHandlers = msg; } return true; } int getMessageHandlersCount() { int result; MessageHandler *curItem = g_nmi->_messageHandlers; for (result = 0; curItem; result++) curItem = curItem->nextItem; return result; } bool addMessageHandlerByIndex(int (*callback)(ExCommand *), int index, int16 id) { if (getMessageHandlerById(id)) return false; if (index) { MessageHandler *curItem = g_nmi->_messageHandlers; for (int i = index - 1; i > 0; i--) if (curItem) curItem = curItem->nextItem; if (!curItem) return false; bool res = allocMessageHandler(curItem, id, callback, index); if (res) updateMessageHandlerIndex(curItem->nextItem->nextItem, 1); return res; } else { MessageHandler *newItem = new MessageHandler; newItem->nextItem = g_nmi->_messageHandlers; newItem->id = id; newItem->callback = callback; newItem->index = 0; updateMessageHandlerIndex(g_nmi->_messageHandlers, 1); g_nmi->_messageHandlers = newItem; return true; } } bool insertMessageHandler(int (*callback)(ExCommand *), int index, int16 id) { if (getMessageHandlerById(id)) return false; MessageHandler *curItem = g_nmi->_messageHandlers; for (int i = index; i > 0; i--) if (curItem) curItem = curItem->nextItem; bool res = allocMessageHandler(curItem, id, callback, index + 1); if (curItem) updateMessageHandlerIndex(curItem->nextItem->nextItem, 1); return res; } void clearMessageHandlers() { MessageHandler *curItem; MessageHandler *nextItem; curItem = g_nmi->_messageHandlers; if (curItem) { do { nextItem = curItem->nextItem; delete curItem; curItem = nextItem; } while (nextItem); g_nmi->_messageHandlers = 0; } } void processMessages() { if (!g_nmi->_isProcessingMessages) { g_nmi->_isProcessingMessages = true; while (g_nmi->_exCommandList.size()) { ExCommand *ex = g_nmi->_exCommandList.front(); g_nmi->_exCommandList.pop_front(); ex->handleMessage(); } g_nmi->_isProcessingMessages = false; } } void updateGlobalMessageQueue(int id, int objid) { MessageQueue *m = g_nmi->_globalMessageQueueList->getMessageQueueById(id); if (m) { m->update(); } } bool chainQueue(int queueId, int flags) { MessageQueue *mq = g_nmi->_currentScene->getMessageQueueById(queueId); if (!mq) return false; MessageQueue *nmq = new MessageQueue(mq, 0, 0); nmq->_flags |= flags; if (!nmq->chain(nullptr)) { g_nmi->_globalMessageQueueList->deleteQueueById(nmq->_id); return false; } return true; } bool chainObjQueue(StaticANIObject *obj, int queueId, int flags) { MessageQueue *mq = g_nmi->_currentScene->getMessageQueueById(queueId); if (!mq) return false; MessageQueue *nmq = new MessageQueue(mq, 0, 0); nmq->_flags |= flags; if (!nmq->chain(obj)) { g_nmi->_globalMessageQueueList->deleteQueueById(nmq->_id); return false; } return true; } void postExCommand(int parentId, int keyCode, int x, int y, int f20, int f14) { ExCommand *ex = new ExCommand(parentId, 17, 64, 0, 0, 0, 1, 0, 0, 0); ex->_param = keyCode; ex->_excFlags |= 3; ex->_x = x; ex->_y = y; ex->_invId = f20; ex->_z = f14; ex->postMessage(); } } // End of namespace NGI