scummvm/engines/sci/engine/message.cpp

259 lines
6.1 KiB
C++
Raw Normal View History

2009-02-17 15:02:16 +00:00
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* $URL$
* $Id$
*
*/
#include "sci/engine/message.h"
#include "sci/tools.h"
namespace Sci {
MessageTuple MessageState::getTuple() {
MessageTuple t;
t.noun = *(_engineCursor.index_record + 0);
t.verb = *(_engineCursor.index_record + 1);
if (_version == 2101) {
t.cond = 0;
t.seq = 1;
} else {
t.cond = *(_engineCursor.index_record + 2);
t.seq = *(_engineCursor.index_record + 3);
}
return t;
}
MessageTuple MessageState::getRefTuple() {
MessageTuple t;
if (_version == 2101) {
t.noun = 0;
t.verb = 0;
t.cond = 0;
} else {
t.noun = *(_engineCursor.index_record + 7);
t.verb = *(_engineCursor.index_record + 8);
t.cond = *(_engineCursor.index_record + 9);
}
t.seq = 1;
return t;
}
void MessageState::initCursor() {
_engineCursor.index_record = _indexRecords;
_engineCursor.index = 0;
_engineCursor.nextSeq = 0;
}
void MessageState::advanceCursor(bool increaseSeq) {
_engineCursor.index_record += ((_version == 2101) ? 4 : 11);
_engineCursor.index++;
if (increaseSeq)
_engineCursor.nextSeq++;
}
int MessageState::findTuple(MessageTuple &t) {
if (_module == -1)
return 0;
// Reset the cursor
initCursor();
_engineCursor.nextSeq = t.seq;
// Do a linear search for the message
while (1) {
MessageTuple looking_at = getTuple();
if (t.noun == looking_at.noun &&
t.verb == looking_at.verb &&
t.cond == looking_at.cond &&
t.seq == looking_at.seq)
break;
advanceCursor(false);
// Message tuple is not present
if (_engineCursor.index == _recordCount)
return 0;
}
return 1;
}
int MessageState::getMessage() {
if (_module == -1)
return 0;
if (_engineCursor.index != _recordCount) {
MessageTuple mesg = getTuple();
MessageTuple ref = getRefTuple();
if (_engineCursor.nextSeq == mesg.seq) {
// We found the right sequence number, check for recursion
if (ref.noun != 0) {
// Recursion, advance the current cursor and load the reference
advanceCursor(true);
if (findTuple(ref))
return getMessage();
else {
// Reference not found
return 0;
}
} else {
// No recursion, we are done
return 1;
}
}
}
// We either ran out of records, or found an incorrect sequence number. Go to previous stack frame.
if (!_cursorStack.empty()) {
_engineCursor = _cursorStack.pop();
return getMessage();
}
// Stack is empty, no message available
return 0;
}
int MessageState::getTalker() {
2009-05-10 14:57:27 +00:00
return (_version == 2101) ? -1 : *(_engineCursor.index_record + 4);
}
MessageTuple &MessageState::getLastTuple() {
return _lastReturned;
}
int MessageState::getLastModule() {
return _lastReturnedModule;
}
Common::String MessageState::getText() {
char *str = (char *)_currentResource->data + READ_LE_UINT16(_engineCursor.index_record + ((_version == 2101) ? 2 : 5));
Common::String strippedStr;
Common::String skippedSubstr;
bool skipping = false;
for (uint i = 0; i < strlen(str); i++) {
if (skipping) {
// Skip stage direction
skippedSubstr += str[i];
// Hopefully these locale-dependant functions are good enough
if (islower(str[i]) || isdigit(str[i])) {
// Lowercase or digit found, this is not a stage direction
strippedStr += skippedSubstr;
skipping = false;
} else if (str[i] == ')') {
// End of stage direction, skip trailing white space
while ((i + 1 < strlen(str)) && isspace(str[i + 1]))
i++;
skipping = false;
}
} else {
if (str[i] == '(') {
// Start skipping stage direction
skippedSubstr = str[i];
skipping = true;
} else if (str[i] == '\\') {
// Escape sequence
if ((i + 2 < strlen(str)) && isdigit(str[i + 1]) && isdigit(str[i + 2])) {
// Hex escape sequence
char hexStr[3];
hexStr[0] = str[++i];
hexStr[1] = str[++i];
hexStr[2] = 0;
char *endptr;
int hexNr = strtol(hexStr, &endptr, 16);
if (*endptr == 0)
strippedStr += hexNr;
} else if (i + 1 < strlen(str)) {
// Literal escape sequence
strippedStr += str[++i];
}
} else {
strippedStr += str[i];
}
}
}
return strippedStr;
}
void MessageState::gotoNext() {
_lastReturned = getTuple();
_lastReturnedModule = _module;
advanceCursor(true);
}
int MessageState::getLength() {
int offset = READ_LE_UINT16(_engineCursor.index_record + ((_version == 2101) ? 2 : 5));
char *stringptr = (char *)_currentResource->data + offset;
return strlen(stringptr);
}
int MessageState::loadRes(ResourceManager *resmgr, int module, bool lock) {
if (_locked) {
// We already have a locked resource
if (_module == module) {
// If it's the same resource, we are done
return 1;
}
// Otherwise, free the old resource
resmgr->unlockResource(_currentResource);
_locked = false;
}
_currentResource = resmgr->findResource(ResourceId(kResourceTypeMessage, module), lock);
if (_currentResource == NULL || _currentResource->data == NULL) {
warning("Message: failed to load %d.msg", module);
return 0;
}
_module = module;
_locked = lock;
_version = READ_LE_UINT16(_currentResource->data);
2009-05-10 14:57:27 +00:00
int offs = (_version == 2101) ? 0 : 4;
_recordCount = READ_LE_UINT16(_currentResource->data + 4 + offs);
_indexRecords = _currentResource->data + 6 + offs;
_cursorStack.clear();
initCursor();
return 1;
}
} // End of namespace Sci