/* 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