Initial commit
This commit is contained in:
972
engines/chewy/atds.cpp
Normal file
972
engines/chewy/atds.cpp
Normal file
@@ -0,0 +1,972 @@
|
||||
/* 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 3 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, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
#include "common/memstream.h"
|
||||
#include "chewy/atds.h"
|
||||
#include "chewy/defines.h"
|
||||
#include "chewy/events.h"
|
||||
#include "chewy/font.h"
|
||||
#include "chewy/globals.h"
|
||||
#include "chewy/mcga_graphics.h"
|
||||
#include "chewy/sound.h"
|
||||
#include "chewy/text.h"
|
||||
|
||||
namespace Chewy {
|
||||
|
||||
bool AtsTxtHeader::load(Common::SeekableReadStream *src) {
|
||||
_txtNr = src->readUint16LE();
|
||||
_aMov = src->readSint16LE();
|
||||
_curNr = src->readSint16LE();
|
||||
src->skip(2);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
void AadInfo::load(Common::SeekableReadStream *src) {
|
||||
_x = src->readSint16LE();
|
||||
_y = src->readSint16LE();
|
||||
_color = src->readSint16LE();
|
||||
}
|
||||
|
||||
void AadInfoArray::load(const void *data, size_t count) {
|
||||
resize(count);
|
||||
Common::MemoryReadStream src((const byte *)data, count * AadInfo::SIZE());
|
||||
|
||||
for (uint i = 0; i < count; ++i)
|
||||
(*this)[i].load(&src);
|
||||
}
|
||||
|
||||
bool DialogCloseupTxtHeader::load(const void *src) {
|
||||
Common::MemoryReadStream rs((const byte *)src, 8);
|
||||
|
||||
_diaNr = rs.readSint16LE();
|
||||
_perNr = rs.readSint16LE();
|
||||
_aMov = rs.readSint16LE();
|
||||
_curNr = rs.readSint16LE();
|
||||
return true;
|
||||
}
|
||||
|
||||
Atdsys::Atdsys() {
|
||||
SplitStringInit init_ssi = { nullptr, 0, 0 };
|
||||
_aadv._dialog = false;
|
||||
_aadv._strNr = -1;
|
||||
_aadv._silentCount = false;
|
||||
_dialogCloseup._dialog = -1;
|
||||
_dialogCloseup._autoDia = false;
|
||||
_dialogCloseup._strNr = -1;
|
||||
_dialogCloseup._silentCount = false;
|
||||
_atdsv._delay = 1;
|
||||
_atdsv._silent = false;
|
||||
_atdsv._diaNr = -1;
|
||||
_atdsv.aad_str = nullptr;
|
||||
_atdsv._vocNr = -1;
|
||||
_atdsv._eventsEnabled = true;
|
||||
for (int16 i = 0; i < AAD_MAX_PERSON; i++)
|
||||
_ssi[i] = init_ssi;
|
||||
_invBlockNr = -1;
|
||||
|
||||
_dialogResource = new DialogResource(ADS_TXT_STEUER);
|
||||
_text = _G(txt);
|
||||
|
||||
_dialogCloseupNextBlock._blkNr = 0;
|
||||
_dialogCloseupNextBlock._endNr = 0;
|
||||
_dialogCloseupStackPtr = 0;
|
||||
|
||||
init();
|
||||
initItemUseWith();
|
||||
}
|
||||
|
||||
Atdsys::~Atdsys() {
|
||||
for (int16 i = 0; i < MAX_HANDLE; i++) {
|
||||
if (_atdsMem[i])
|
||||
free(_atdsMem[i]);
|
||||
_atdsMem[i] = nullptr;
|
||||
}
|
||||
|
||||
delete _dialogResource;
|
||||
}
|
||||
|
||||
void Atdsys::init() {
|
||||
set_handle(AAD_DATA, AAD_TAP_OFF, AAD_TAP_MAX);
|
||||
set_handle(DIALOG_CLOSEUP_DATA, ADS_TAP_OFF, ADS_TAP_MAX);
|
||||
_G(gameState).AadSilent = 10;
|
||||
_G(gameState).DelaySpeed = 5;
|
||||
_G(moveState)[P_CHEWY].Delay = _G(gameState).DelaySpeed;
|
||||
set_delay(&_G(gameState).DelaySpeed, _G(gameState).AadSilent);
|
||||
set_string_end_func(&atdsStringStart);
|
||||
}
|
||||
|
||||
void Atdsys::initItemUseWith() {
|
||||
int16 objA, objB, txtNum;
|
||||
|
||||
Common::File f;
|
||||
f.open(INV_USE_IDX);
|
||||
|
||||
// The file is 25200 bytes, and contains 25200 / 50 / 6 = 84 blocks
|
||||
int totalEntries = f.size() / 6;
|
||||
|
||||
for (int entry = 0; entry < totalEntries; entry++) {
|
||||
objA = f.readSint16LE();
|
||||
objB = f.readSint16LE();
|
||||
txtNum = f.readSint16LE();
|
||||
|
||||
assert(objA <= 255);
|
||||
|
||||
const uint32 key = (objA & 0xff) << 16 | objB;
|
||||
_itemUseWithDesc[key] = txtNum;
|
||||
}
|
||||
|
||||
f.close();
|
||||
}
|
||||
|
||||
void Atdsys::set_delay(int16 *delay, int16 silent) {
|
||||
_atdsv._delay = *delay;
|
||||
_atdsv._silent = silent;
|
||||
}
|
||||
|
||||
void Atdsys::set_string_end_func
|
||||
(void (*strFunc)(int16 diaNr, int16 strNr, int16 personNr, int16 mode)) {
|
||||
_atdsv.aad_str = strFunc;
|
||||
}
|
||||
|
||||
int16 Atdsys::get_delay(int16 txt_len) {
|
||||
const int16 width = 220;
|
||||
const int16 lines = 4;
|
||||
const int16 w = _G(fontMgr)->getFont()->getDataWidth();
|
||||
int16 z_len = (width / w) + 1;
|
||||
int16 maxLen = z_len * lines;
|
||||
if (txt_len > maxLen)
|
||||
txt_len = maxLen;
|
||||
|
||||
int16 ret = _atdsv._delay * (txt_len + z_len);
|
||||
return ret;
|
||||
}
|
||||
|
||||
void Atdsys::split_string(SplitStringInit *ssi_, SplitStringRet *ret) {
|
||||
const int16 w = _G(fontMgr)->getFont()->getDataWidth();
|
||||
const int16 h = _G(fontMgr)->getFont()->getDataHeight();
|
||||
const int16 width = 220;
|
||||
const int16 lines = 4;
|
||||
|
||||
ret->_nr = 0;
|
||||
ret->_next = false;
|
||||
ret->_strPtr = _splitPtr;
|
||||
ret->_x = _splitX;
|
||||
int16 zeichen_anz = (width / w) + 1;
|
||||
memset(_splitPtr, 0, sizeof(char *) * MAX_STR_SPLIT);
|
||||
calc_txt_win(ssi_);
|
||||
char *str_adr = ssi_->_str;
|
||||
int16 count = 0;
|
||||
int16 tmp_count = 0;
|
||||
bool endLoop = false;
|
||||
char *start_adr = str_adr;
|
||||
|
||||
while (!endLoop) {
|
||||
switch (*str_adr) {
|
||||
case 0:
|
||||
if (str_adr[1] != ATDS_END_TEXT) {
|
||||
str_adr[0] = ' ';
|
||||
}
|
||||
// Fall through
|
||||
case 0x20:
|
||||
if (count < zeichen_anz && *str_adr == 0) {
|
||||
|
||||
tmp_count = count;
|
||||
}
|
||||
if (count < zeichen_anz && *str_adr != 0) {
|
||||
|
||||
tmp_count = count;
|
||||
++str_adr;
|
||||
++count;
|
||||
} else {
|
||||
_splitPtr[ret->_nr] = start_adr;
|
||||
start_adr[tmp_count] = 0;
|
||||
_splitX[ret->_nr] = ssi_->_x + ((width - (strlen(start_adr) * w)) >> 1);
|
||||
++ret->_nr;
|
||||
if (ret->_nr == lines) {
|
||||
endLoop = true;
|
||||
bool endInnerLoop = false;
|
||||
while (!endInnerLoop) {
|
||||
if (*str_adr == ATDS_END_TEXT)
|
||||
endInnerLoop = true;
|
||||
else if (*str_adr != ' ' && *str_adr != 0) {
|
||||
endInnerLoop = true;
|
||||
ret->_next = true;
|
||||
}
|
||||
++str_adr;
|
||||
}
|
||||
} else if (*str_adr == 0 && count < zeichen_anz) {
|
||||
endLoop = true;
|
||||
} else {
|
||||
str_adr = start_adr + tmp_count + 1;
|
||||
start_adr = str_adr;
|
||||
count = 0;
|
||||
tmp_count = 0;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case '!':
|
||||
case '?':
|
||||
case '.':
|
||||
case ',':
|
||||
if (str_adr[1] == 0 || str_adr[1] == ' ') {
|
||||
int16 test_zeilen;
|
||||
if (*str_adr == ',')
|
||||
test_zeilen = 1;
|
||||
else
|
||||
test_zeilen = 2;
|
||||
++count;
|
||||
++str_adr;
|
||||
if (ret->_nr + test_zeilen >= lines) {
|
||||
if (count < zeichen_anz) {
|
||||
tmp_count = count;
|
||||
endLoop = true;
|
||||
}
|
||||
_splitPtr[ret->_nr] = start_adr;
|
||||
start_adr[tmp_count] = 0;
|
||||
_splitX[ret->_nr] = ssi_->_x + ((width - (strlen(start_adr) * w)) >> 1);
|
||||
++ret->_nr;
|
||||
bool ende1 = false;
|
||||
while (!ende1) {
|
||||
if (*str_adr == ATDS_END_TEXT)
|
||||
ende1 = true;
|
||||
else if (*str_adr != ' ' && *str_adr != 0) {
|
||||
ende1 = true;
|
||||
ret->_next = true;
|
||||
}
|
||||
++str_adr;
|
||||
}
|
||||
if (!endLoop) {
|
||||
str_adr = start_adr + tmp_count + 1;
|
||||
start_adr = str_adr;
|
||||
count = 0;
|
||||
tmp_count = 0;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
++count;
|
||||
++str_adr;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
++count;
|
||||
++str_adr;
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
if (ret->_nr <= lines)
|
||||
ret->_y = ssi_->_y + (lines - ret->_nr) * h;
|
||||
else
|
||||
ret->_y = ssi_->_y;
|
||||
}
|
||||
|
||||
void Atdsys::str_null2leer(char *strStart, char *strEnd) {
|
||||
while (strStart < strEnd) {
|
||||
if (*strStart == 0)
|
||||
*strStart = 32;
|
||||
++strStart;
|
||||
}
|
||||
}
|
||||
|
||||
void Atdsys::calc_txt_win(SplitStringInit *ssi_) {
|
||||
const int16 h = _G(fontMgr)->getFont()->getDataHeight();
|
||||
const int16 width = 220;
|
||||
const int16 lines = 4;
|
||||
|
||||
if (ssi_->_x - (width >> 1) < 2)
|
||||
ssi_->_x = 2;
|
||||
else if (ssi_->_x + (width >> 1) > (SCREEN_WIDTH - 2))
|
||||
ssi_->_x = ((SCREEN_WIDTH - 2) - width);
|
||||
else
|
||||
ssi_->_x -= (width >> 1);
|
||||
|
||||
if (ssi_->_y - (lines * h) < 2) {
|
||||
ssi_->_y = 2;
|
||||
} else if (ssi_->_y + (lines * h) > (SCREEN_HEIGHT - 2))
|
||||
ssi_->_y = (SCREEN_HEIGHT - 2) - (lines * h);
|
||||
else {
|
||||
ssi_->_y -= (lines * h);
|
||||
}
|
||||
}
|
||||
|
||||
void Atdsys::set_split_win(int16 nr, int16 x, int16 y) {
|
||||
_ssi[nr]._x = x;
|
||||
_ssi[nr]._y = y;
|
||||
}
|
||||
|
||||
void Atdsys::set_handle(int16 mode, int16 chunkStart, int16 chunkNr) {
|
||||
assert(mode == AAD_DATA || mode == DIALOG_CLOSEUP_DATA);
|
||||
|
||||
uint32 size = _text->findLargestChunk(chunkStart, chunkStart + chunkNr);
|
||||
char *tmp_adr = size ? (char *)MALLOC(size + 3) : nullptr;
|
||||
|
||||
if (_atdsMem[mode])
|
||||
free(_atdsMem[mode]);
|
||||
_atdsMem[mode] = tmp_adr;
|
||||
_atdsPoolOff[mode] = chunkStart;
|
||||
}
|
||||
|
||||
void Atdsys::load_atds(int16 chunkNr, int16 mode) {
|
||||
assert(mode == AAD_DATA || mode == DIALOG_CLOSEUP_DATA);
|
||||
|
||||
char *txt_adr = _atdsMem[mode];
|
||||
|
||||
if (txt_adr) {
|
||||
const uint32 chunkSize = _text->getChunk(chunkNr + _atdsPoolOff[mode])->size;
|
||||
const uint8 *chunkData = _text->getChunkData(chunkNr + _atdsPoolOff[mode]);
|
||||
memcpy(txt_adr, chunkData, chunkSize);
|
||||
delete[] chunkData;
|
||||
txt_adr[chunkSize] = (char)BLOCKENDE;
|
||||
txt_adr[chunkSize + 1] = (char)BLOCKENDE;
|
||||
txt_adr[chunkSize + 2] = (char)BLOCKENDE;
|
||||
}
|
||||
}
|
||||
|
||||
bool Atdsys::start_ats(int16 txtNr, int16 txtMode, int16 color, int16 mode, int16 *vocNr) {
|
||||
assert(mode == ATS_DATA || mode == INV_USE_DATA || mode == INV_USE_DEF);
|
||||
|
||||
EVENTS_CLEAR;
|
||||
g_events->_kbInfo._scanCode = Common::KEYCODE_INVALID;
|
||||
g_events->_kbInfo._keyCode = '\0';
|
||||
_G(minfo).button = 0;
|
||||
|
||||
*vocNr = -1;
|
||||
|
||||
_atsv.shown = false;
|
||||
|
||||
Common::StringArray textArray;
|
||||
|
||||
if (mode != INV_USE_DEF) {
|
||||
const uint8 roomNum = _G(room)->_roomInfo->_roomNr;
|
||||
textArray = getTextArray(roomNum, txtNr, mode, txtMode);
|
||||
} else {
|
||||
textArray = getTextArray(0, txtNr, mode, -1);
|
||||
}
|
||||
|
||||
_atsv.text.clear();
|
||||
for (uint i = 0; i < textArray.size(); i++)
|
||||
_atsv.text += textArray[i] + " ";
|
||||
_atsv.text.deleteLastChar();
|
||||
|
||||
if (_atsv.text.size() > 0) {
|
||||
*vocNr = txtMode != TXT_MARK_NAME ? _text->getLastSpeechId() : -1;
|
||||
_atsv.shown = true;
|
||||
_atsv._txtMode = txtMode;
|
||||
_atsv._delayCount = get_delay(_atsv.text.size());
|
||||
_atsv._color = color;
|
||||
_printDelayCount1 = _atsv._delayCount / 10;
|
||||
_mousePush = true;
|
||||
}
|
||||
|
||||
return _atsv.shown;
|
||||
}
|
||||
|
||||
void Atdsys::stop_ats() {
|
||||
_atsv.shown = false;
|
||||
}
|
||||
|
||||
void Atdsys::print_ats(int16 x, int16 y, int16 scrX, int16 scrY) {
|
||||
if (_atsv.shown) {
|
||||
if (_atdsv._eventsEnabled) {
|
||||
switch (g_events->getSwitchCode()) {
|
||||
case Common::KEYCODE_ESCAPE:
|
||||
case Common::KEYCODE_RETURN:
|
||||
case Common::MOUSE_BUTTON_LEFT:
|
||||
if (!_mousePush) {
|
||||
EVENTS_CLEAR;
|
||||
g_events->_kbInfo._scanCode = Common::KEYCODE_INVALID;
|
||||
g_events->_kbInfo._keyCode = '\0';
|
||||
_G(minfo).button = 0;
|
||||
|
||||
if (_atsv._silentCount <= 0 && _atsv._delayCount > _printDelayCount1) {
|
||||
_mousePush = true;
|
||||
_atsv._delayCount = 0;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
_mousePush = false;
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
_mousePush = false;
|
||||
}
|
||||
|
||||
if (_atsv._silentCount <= 0) {
|
||||
// TODO: Rewrite this
|
||||
SplitStringInit *atsSsi = &_ssi[0];
|
||||
char *txt = new char[_atsv.text.size() + 2];
|
||||
const int16 h = _G(fontMgr)->getFont()->getDataHeight();
|
||||
uint shownLen = 0;
|
||||
SplitStringRet splitString;
|
||||
|
||||
Common::strlcpy(txt, _atsv.text.c_str(), _atsv.text.size() + 1);
|
||||
txt[_atsv.text.size() + 1] = ATDS_END_TEXT;
|
||||
atsSsi->_str = txt;
|
||||
atsSsi->_x = x - scrX;
|
||||
atsSsi->_y = y - scrY;
|
||||
split_string(atsSsi, &splitString);
|
||||
|
||||
for (int16 i = 0; i < splitString._nr; i++) {
|
||||
if (g_engine->_sound->subtitlesEnabled()) {
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h) + 1,
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h) - 1,
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i] + 1,
|
||||
splitString._y + (i * h),
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i] - 1,
|
||||
splitString._y + (i * h),
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h),
|
||||
_atsv._color,
|
||||
300, 0, splitString._strPtr[i]);
|
||||
}
|
||||
|
||||
shownLen += strlen(splitString._strPtr[i]) + 1;
|
||||
}
|
||||
|
||||
delete[] txt;
|
||||
|
||||
if (_atsv._delayCount <= 0) {
|
||||
if (!splitString._next) {
|
||||
_atsv.shown = false;
|
||||
} else {
|
||||
_atsv._delayCount = get_delay(_atsv.text.size() - shownLen);
|
||||
_printDelayCount1 = _atsv._delayCount / 10;
|
||||
_atsv._silentCount = _atdsv._silent;
|
||||
}
|
||||
} else {
|
||||
--_atsv._delayCount;
|
||||
}
|
||||
} else {
|
||||
--_atsv._silentCount;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Atdsys::set_ats_str(int16 txtNr, int16 txtMode, int16 strNr, int16 mode) {
|
||||
_text->setSubtextNum(txtNr, txtMode, strNr, mode);
|
||||
}
|
||||
|
||||
void Atdsys::set_all_ats_str(int16 txtNr, int16 strNr, int16 mode) {
|
||||
for (int16 i = 0; i < 5; i++)
|
||||
set_ats_str(txtNr, i, strNr, mode);
|
||||
}
|
||||
|
||||
int16 Atdsys::getControlBit(int16 txtNr, int16 bitIdx) {
|
||||
return _text->getControlBit(txtNr, bitIdx);
|
||||
}
|
||||
|
||||
void Atdsys::setControlBit(int16 txtNr, int16 bitIdx) {
|
||||
_text->setControlBit(txtNr, bitIdx);
|
||||
}
|
||||
|
||||
void Atdsys::delControlBit(int16 txtNr, int16 bitIdx) {
|
||||
_text->delControlBit(txtNr, bitIdx);
|
||||
}
|
||||
|
||||
int16 Atdsys::start_aad(int16 diaNr, bool continueWhenSpeechEnds) {
|
||||
if (_aadv._dialog)
|
||||
stopAad();
|
||||
|
||||
_continueWhenSpeechEnds = continueWhenSpeechEnds;
|
||||
|
||||
EVENTS_CLEAR;
|
||||
g_events->_kbInfo._scanCode = Common::KEYCODE_INVALID;
|
||||
g_events->_kbInfo._keyCode = '\0';
|
||||
_G(minfo).button = 0;
|
||||
|
||||
if (_atdsMem[AAD_HANDLE]) {
|
||||
_aadv._ptr = _atdsMem[AAD_HANDLE];
|
||||
aad_search_dia(diaNr, &_aadv._ptr);
|
||||
|
||||
if (_aadv._ptr) {
|
||||
_aadv._person.load(_aadv._ptr, _aadv._txtHeader->_perNr);
|
||||
_aadv._ptr += _aadv._txtHeader->_perNr * sizeof(AadInfo);
|
||||
|
||||
_aadv._dialog = true;
|
||||
_aadv._strNr = 0;
|
||||
_aadv._strHeader = (AadStrHeader *)_aadv._ptr;
|
||||
_aadv._ptr += sizeof(AadStrHeader);
|
||||
int16 txtLen = aadGetTxtLen(_aadv._ptr);
|
||||
_aadv._delayCount = get_delay(txtLen);
|
||||
_printDelayCount1 = _aadv._delayCount / 10;
|
||||
|
||||
_atdsv._diaNr = diaNr;
|
||||
if (_atdsv.aad_str != nullptr)
|
||||
_atdsv.aad_str(_atdsv._diaNr, 0, _aadv._strHeader->_akPerson, AAD_STR_START);
|
||||
_mousePush = true;
|
||||
stop_ats();
|
||||
_atdsv._vocNr = -1;
|
||||
}
|
||||
}
|
||||
|
||||
return _aadv._dialog;
|
||||
}
|
||||
|
||||
void Atdsys::stopAad() {
|
||||
_aadv._dialog = false;
|
||||
_aadv._strNr = -1;
|
||||
}
|
||||
|
||||
void Atdsys::print_aad(int16 scrX, int16 scrY) {
|
||||
if (_aadv._dialog) {
|
||||
if (_atdsv._eventsEnabled) {
|
||||
switch (g_events->getSwitchCode()) {
|
||||
case Common::KEYCODE_ESCAPE:
|
||||
case Common::KEYCODE_RETURN:
|
||||
case Common::MOUSE_BUTTON_LEFT:
|
||||
if (!_mousePush) {
|
||||
EVENTS_CLEAR;
|
||||
g_events->_kbInfo._scanCode = Common::KEYCODE_INVALID;
|
||||
g_events->_kbInfo._keyCode = '\0';
|
||||
_G(minfo).button = 0;
|
||||
|
||||
if (_aadv._silentCount <= 0 && _aadv._delayCount > _printDelayCount1) {
|
||||
_mousePush = true;
|
||||
_aadv._delayCount = 0;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
_mousePush = false;
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
_mousePush = false;
|
||||
}
|
||||
|
||||
if (_aadv._silentCount <= 0) {
|
||||
char *tmp_ptr = _aadv._ptr;
|
||||
const int16 personId = _aadv._strHeader->_akPerson;
|
||||
_ssi[personId]._str = tmp_ptr;
|
||||
if (_aadv._person[personId]._x != -1) {
|
||||
_ssi[personId]._x = _aadv._person[personId]._x - scrX;
|
||||
}
|
||||
if (_aadv._person[personId]._y != -1) {
|
||||
_ssi[personId]._y = _aadv._person[personId]._y - scrY;
|
||||
}
|
||||
char *start_ptr = tmp_ptr;
|
||||
int16 txtLen = aadGetTxtLen(start_ptr);
|
||||
str_null2leer(start_ptr, start_ptr + txtLen - 1);
|
||||
SplitStringInit tmp_ssi = _ssi[personId];
|
||||
SplitStringRet splitString;
|
||||
split_string(&tmp_ssi, &splitString);
|
||||
|
||||
const int16 h = _G(fontMgr)->getFont()->getDataHeight();
|
||||
for (int16 i = 0; i < splitString._nr; i++) {
|
||||
if (g_engine->_sound->subtitlesEnabled() ||
|
||||
_aadv._strHeader->_vocNr - ATDS_VOC_OFFSET == -1) {
|
||||
_G(out)->printxy(splitString._x[i] + 1,
|
||||
splitString._y + (i * h),
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i] - 1,
|
||||
splitString._y + (i * h),
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h) + 1,
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h) - 1,
|
||||
0, 300, 0, splitString._strPtr[i]);
|
||||
_G(out)->printxy(splitString._x[i],
|
||||
splitString._y + (i * h),
|
||||
_aadv._person[personId]._color,
|
||||
300, 0, splitString._strPtr[i]);
|
||||
}
|
||||
tmp_ptr += strlen(splitString._strPtr[i]) + 1;
|
||||
}
|
||||
str_null2leer(start_ptr, start_ptr + txtLen - 1);
|
||||
|
||||
if (g_engine->_sound->speechEnabled() &&
|
||||
_aadv._strHeader->_vocNr - ATDS_VOC_OFFSET != -1) {
|
||||
if (_atdsv._vocNr != _aadv._strHeader->_vocNr - ATDS_VOC_OFFSET) {
|
||||
_atdsv._vocNr = _aadv._strHeader->_vocNr - ATDS_VOC_OFFSET;
|
||||
g_engine->_sound->playSpeech(_atdsv._vocNr, false);
|
||||
}
|
||||
|
||||
if (_continueWhenSpeechEnds && _atdsv._vocNr >= 0 && !g_engine->_sound->isSpeechActive())
|
||||
stopAad();
|
||||
}
|
||||
|
||||
if (_aadv._delayCount <= 0) {
|
||||
_aadv._ptr = tmp_ptr;
|
||||
while (*tmp_ptr == ' ' || *tmp_ptr == 0)
|
||||
++tmp_ptr;
|
||||
if (tmp_ptr[1] == ATDS_END ||
|
||||
tmp_ptr[1] == ATDS_END_ENTRY) {
|
||||
if (_atdsv.aad_str != 0)
|
||||
_atdsv.aad_str(_atdsv._diaNr, _aadv._strNr, personId, AAD_STR_END);
|
||||
_aadv._dialog = false;
|
||||
_dialogCloseup._autoDia = false;
|
||||
_aadv._strNr = -1;
|
||||
splitString._next = false;
|
||||
} else {
|
||||
if (!splitString._next) {
|
||||
++_aadv._strNr;
|
||||
while (*_aadv._ptr++ != ATDS_END_TEXT) {}
|
||||
|
||||
const int16 tmp_person = _aadv._strHeader->_akPerson;
|
||||
const int16 tmp_str_nr = _aadv._strNr;
|
||||
_aadv._strHeader = (AadStrHeader *)_aadv._ptr;
|
||||
_aadv._ptr += sizeof(AadStrHeader);
|
||||
if (_atdsv.aad_str != nullptr) {
|
||||
if (tmp_person != _aadv._strHeader->_akPerson) {
|
||||
_atdsv.aad_str(_atdsv._diaNr, tmp_str_nr, tmp_person, AAD_STR_END);
|
||||
_atdsv.aad_str(_atdsv._diaNr, _aadv._strNr, _aadv._strHeader->_akPerson, AAD_STR_START);
|
||||
}
|
||||
}
|
||||
}
|
||||
txtLen = aadGetTxtLen(_aadv._ptr);
|
||||
_aadv._delayCount = get_delay(txtLen);
|
||||
_printDelayCount1 = _aadv._delayCount / 10;
|
||||
_aadv._silentCount = _atdsv._silent;
|
||||
}
|
||||
} else {
|
||||
if (_aadv._strHeader->_vocNr - ATDS_VOC_OFFSET == -1)
|
||||
--_aadv._delayCount;
|
||||
}
|
||||
} else {
|
||||
--_aadv._silentCount;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int16 Atdsys::aadGetStatus() {
|
||||
return _aadv._strNr;
|
||||
}
|
||||
|
||||
int16 Atdsys::aadGetTxtLen(char *str) {
|
||||
char *ptr = str;
|
||||
while (*str != ATDS_END_TEXT)
|
||||
++str;
|
||||
|
||||
int16 txtLen = (str - ptr) - 1;
|
||||
return txtLen;
|
||||
}
|
||||
|
||||
void Atdsys::aad_search_dia(int16 diaNr, char **ptr) {
|
||||
char *start_ptr = *ptr;
|
||||
|
||||
if (start_ptr[0] == (char)BLOCKENDE &&
|
||||
start_ptr[1] == (char)BLOCKENDE &&
|
||||
start_ptr[2] == (char)BLOCKENDE) {
|
||||
*ptr = nullptr;
|
||||
} else {
|
||||
bool ende = false;
|
||||
while (!ende) {
|
||||
uint16 *pos = (uint16 *)start_ptr;
|
||||
if (pos[0] == diaNr) {
|
||||
ende = true;
|
||||
_aadv._txtHeader = (DialogCloseupTxtHeader *)start_ptr;
|
||||
*ptr = start_ptr + sizeof(DialogCloseupTxtHeader);
|
||||
} else {
|
||||
start_ptr += sizeof(DialogCloseupTxtHeader) + pos[1] * sizeof(AadInfo);
|
||||
bool ende1 = false;
|
||||
for (; !ende1; ++start_ptr) {
|
||||
if (*start_ptr != ATDS_END_TEXT)
|
||||
continue;
|
||||
if (start_ptr[1] == ATDS_END) {
|
||||
++start_ptr;
|
||||
|
||||
if (start_ptr[1] == (char)BLOCKENDE &&
|
||||
start_ptr[2] == (char)BLOCKENDE &&
|
||||
start_ptr[3] == (char)BLOCKENDE) {
|
||||
ende = true;
|
||||
ende1 = true;
|
||||
*ptr = nullptr;
|
||||
} else {
|
||||
ende1 = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool Atdsys::startDialogCloseup(int16 diaNr) {
|
||||
bool ret = false;
|
||||
bool end = false;
|
||||
|
||||
load_atds(diaNr, DIALOG_CLOSEUP_DATA);
|
||||
|
||||
if (_atdsMem[ADS_HANDLE][0] == (char)BLOCKENDE &&
|
||||
_atdsMem[ADS_HANDLE][1] == (char)BLOCKENDE &&
|
||||
_atdsMem[ADS_HANDLE][2] == (char)BLOCKENDE)
|
||||
end = true;
|
||||
|
||||
if (!end) {
|
||||
_dialogCloseup._ptr = _atdsMem[ADS_HANDLE];
|
||||
_dialogCloseup._txtHeader.load(_dialogCloseup._ptr);
|
||||
|
||||
if (_dialogCloseup._txtHeader._diaNr == diaNr) {
|
||||
ret = true;
|
||||
_dialogCloseup._ptr += DialogCloseupTxtHeader::SIZE();
|
||||
_dialogCloseup._person.load(_dialogCloseup._ptr, _dialogCloseup._txtHeader._perNr);
|
||||
_dialogCloseup._ptr += _dialogCloseup._txtHeader._perNr * AadInfo::SIZE();
|
||||
_dialogCloseup._dialog = diaNr;
|
||||
_dialogCloseup._strNr = 0;
|
||||
_dialogCloseupStack[0] = 0;
|
||||
_dialogCloseupStackPtr = 1;
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
void Atdsys::stopDialogCloseup() {
|
||||
_dialogCloseup._dialog = -1;
|
||||
_dialogCloseup._autoDia = false;
|
||||
}
|
||||
|
||||
int16 Atdsys::getDialogCloseupStatus() {
|
||||
return _dialogCloseup._dialog;
|
||||
}
|
||||
|
||||
char **Atdsys::dialogCloseupItemPtr(uint16 dialogNum, int16 blockNr, int16 *retNr) {
|
||||
*retNr = 0;
|
||||
memset(_ePtr, 0, sizeof(char *) * DIALOG_CLOSEUP_MAX);
|
||||
if (_dialogCloseup._dialog != -1) {
|
||||
_dialogCloseup._blockPtr = _dialogCloseup._ptr;
|
||||
dialogCloseupSearchBlock(blockNr, &_dialogCloseup._blockPtr);
|
||||
if (_dialogCloseup._blockPtr) {
|
||||
for (int16 i = 0; i < DIALOG_CLOSEUP_MAX; i++) {
|
||||
char *itemPtr = _dialogCloseup._blockPtr;
|
||||
dialogCloseupSearchItem(i, &itemPtr);
|
||||
if (itemPtr) {
|
||||
char nr = itemPtr[-1];
|
||||
itemPtr += sizeof(AadStrHeader);
|
||||
if (_dialogResource->isItemShown(dialogNum, blockNr, (int16)nr)) {
|
||||
_ePtr[*retNr] = itemPtr;
|
||||
_eNr[*retNr] = (int16)nr;
|
||||
++(*retNr);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return _ePtr;
|
||||
}
|
||||
|
||||
DialogCloseupNextBlock *Atdsys::dialogCloseupItemChoice(uint16 dialogNum, int16 blockNr, int16 itemNr) {
|
||||
_dialogCloseupNextBlock._blkNr = blockNr;
|
||||
if (!_aadv._dialog) {
|
||||
if (!_dialogCloseup._autoDia) {
|
||||
dialogCloseupSearchItem(_eNr[itemNr], &_dialogCloseup._blockPtr);
|
||||
if (_dialogCloseup._blockPtr) {
|
||||
if (startAutoDialogCloseup(_dialogCloseup._blockPtr))
|
||||
_dialogCloseup._autoDia = true;
|
||||
if (_dialogResource->hasExitBit(dialogNum, blockNr, _eNr[itemNr])) {
|
||||
stopDialogCloseup();
|
||||
_dialogCloseupNextBlock._endNr = _eNr[itemNr];
|
||||
_dialogCloseupNextBlock._blkNr = -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return &_dialogCloseupNextBlock;
|
||||
}
|
||||
|
||||
DialogCloseupNextBlock *Atdsys::calcNextDialogCloseupBlock(uint16 dialogNum, int16 blockNr, int16 itemNr) {
|
||||
if (!_dialogResource->hasShowBit(dialogNum, blockNr, _eNr[itemNr]))
|
||||
_dialogResource->setItemShown(dialogNum, blockNr, _eNr[itemNr], false);
|
||||
_dialogCloseupNextBlock._endNr = _eNr[itemNr];
|
||||
|
||||
if (_dialogResource->hasRestartBit(dialogNum, blockNr, _eNr[itemNr])) {
|
||||
_dialogCloseupNextBlock._blkNr = 0;
|
||||
|
||||
_dialogCloseupStackPtr = 0;
|
||||
} else {
|
||||
const uint8 nextBlock = _dialogResource->getNextBlock(dialogNum, blockNr, _eNr[itemNr]);
|
||||
if (nextBlock) {
|
||||
_dialogCloseupNextBlock._blkNr = nextBlock;
|
||||
|
||||
int16 option = 0;
|
||||
while (!option && _dialogCloseupNextBlock._blkNr != -1) {
|
||||
|
||||
option = 0;
|
||||
dialogCloseupItemPtr(dialogNum, _dialogCloseupNextBlock._blkNr, &option);
|
||||
if (!option) {
|
||||
_dialogCloseupNextBlock._blkNr = getDialogCloseupBlock(dialogNum);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
_dialogCloseupNextBlock._blkNr = getDialogCloseupBlock(dialogNum);
|
||||
}
|
||||
}
|
||||
_dialogCloseupStack[_dialogCloseupStackPtr] = _dialogCloseupNextBlock._blkNr;
|
||||
++_dialogCloseupStackPtr;
|
||||
|
||||
return &_dialogCloseupNextBlock;
|
||||
}
|
||||
|
||||
int16 Atdsys::getDialogCloseupBlock(uint16 dialogNum) {
|
||||
_dialogCloseupStackPtr -= 1;
|
||||
int16 ret = -1;
|
||||
bool end = false;
|
||||
while (_dialogCloseupStackPtr >= 0 && !end) {
|
||||
short blk_nr = _dialogCloseupStack[_dialogCloseupStackPtr];
|
||||
int16 option;
|
||||
dialogCloseupItemPtr(dialogNum, blk_nr, &option);
|
||||
if (option) {
|
||||
ret = blk_nr;
|
||||
end = true;
|
||||
} else {
|
||||
--_dialogCloseupStackPtr;
|
||||
}
|
||||
}
|
||||
|
||||
++_dialogCloseupStackPtr;
|
||||
return ret;
|
||||
}
|
||||
|
||||
void Atdsys::dialogCloseupSearchBlock(int16 blockNr, char **ptr) {
|
||||
char *start_ptr = *ptr;
|
||||
bool end = false;
|
||||
while (!end) {
|
||||
if (*start_ptr == (char)blockNr) {
|
||||
end = true;
|
||||
*ptr = start_ptr;
|
||||
} else {
|
||||
start_ptr += 2 + sizeof(AadStrHeader);
|
||||
while (*start_ptr++ != ATDS_END_BLOCK) {}
|
||||
if (start_ptr[0] == ATDS_END &&
|
||||
start_ptr[1] == ATDS_END) {
|
||||
end = true;
|
||||
*ptr = nullptr;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Atdsys::dialogCloseupSearchItem(int16 itemNr, char **blkAdr) {
|
||||
char *start_ptr = *blkAdr + 1;
|
||||
bool end = false;
|
||||
while (!end) {
|
||||
if (*start_ptr == itemNr) {
|
||||
end = true;
|
||||
*blkAdr = start_ptr + 1;
|
||||
} else {
|
||||
start_ptr += 1 + sizeof(AadStrHeader);
|
||||
while (*start_ptr++ != ATDS_END_ENTRY) {}
|
||||
if (*start_ptr == ATDS_END_BLOCK) {
|
||||
end = true;
|
||||
*blkAdr = nullptr;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int16 Atdsys::startAutoDialogCloseup(char *itemAdr) {
|
||||
_aadv._dialog = false;
|
||||
if (itemAdr) {
|
||||
_aadv._person = _dialogCloseup._person;
|
||||
_aadv._ptr = itemAdr;
|
||||
_aadv._dialog = true;
|
||||
_aadv._strNr = 0;
|
||||
_aadv._strHeader = (AadStrHeader *)_aadv._ptr;
|
||||
_aadv._ptr += sizeof(AadStrHeader);
|
||||
int16 txtLen = aadGetTxtLen(_aadv._ptr);
|
||||
_aadv._delayCount = get_delay(txtLen);
|
||||
_atdsv._diaNr = _dialogCloseup._txtHeader._diaNr + 10000;
|
||||
|
||||
if (_atdsv.aad_str != nullptr)
|
||||
_atdsv.aad_str(_atdsv._diaNr, 0, _aadv._strHeader->_akPerson, AAD_STR_START);
|
||||
_mousePush = true;
|
||||
stop_ats();
|
||||
} else {
|
||||
_aadv._dialog = false;
|
||||
}
|
||||
|
||||
return _aadv._dialog;
|
||||
}
|
||||
|
||||
void Atdsys::hideDialogCloseupItem(int16 diaNr, int16 blockNr, int16 itemNr) {
|
||||
_dialogResource->setItemShown(diaNr, blockNr, itemNr, false);
|
||||
}
|
||||
|
||||
void Atdsys::showDialogCloseupItem(int16 diaNr, int16 blockNr, int16 itemNr) {
|
||||
_dialogResource->setItemShown(diaNr, blockNr, itemNr, true);
|
||||
}
|
||||
|
||||
int16 Atdsys::calc_inv_no_use(int16 curInv, int16 testNr) {
|
||||
if (curInv != -1)
|
||||
_invBlockNr = curInv + 1;
|
||||
|
||||
assert(curInv <= 255);
|
||||
|
||||
const uint32 key = (curInv & 0xff) << 16 | testNr;
|
||||
return (_itemUseWithDesc.contains(key)) ? _itemUseWithDesc[key] : -1;
|
||||
}
|
||||
|
||||
int8 Atdsys::getStereoPos(int16 x) {
|
||||
return floor(x / 2.5);
|
||||
}
|
||||
|
||||
void Atdsys::saveAtdsStream(Common::WriteStream *stream) {
|
||||
_dialogResource->saveStream(stream);
|
||||
}
|
||||
|
||||
void Atdsys::loadAtdsStream(Common::SeekableReadStream* stream) {
|
||||
_dialogResource->loadStream(stream);
|
||||
}
|
||||
|
||||
uint32 Atdsys::getAtdsStreamSize() const {
|
||||
return _dialogResource->getStreamSize();
|
||||
}
|
||||
|
||||
Common::StringArray Atdsys::getTextArray(uint dialogNum, uint entryNum, int type, int subEntry) {
|
||||
if (dialogNum == 45 && entryNum == 295 && type == 1 && subEntry == -1 &&
|
||||
g_engine->getLanguage() == Common::EN_ANY) {
|
||||
// WORKAROUND: Taxi hotspot in room 45 (Big City)
|
||||
Common::StringArray results;
|
||||
results.push_back("Taxi");
|
||||
return results;
|
||||
} else if (!getControlBit(entryNum, ATS_ACTIVE_BIT))
|
||||
return _text->getTextArray(dialogNum, entryNum, type, subEntry);
|
||||
else
|
||||
return Common::StringArray();
|
||||
}
|
||||
|
||||
Common::String Atdsys::getTextEntry(uint dialogNum, uint entryNum, int type, int subEntry) {
|
||||
if (!getControlBit(entryNum, ATS_ACTIVE_BIT))
|
||||
return _text->getTextEntry(dialogNum, entryNum, type, subEntry);
|
||||
else
|
||||
return Common::String();
|
||||
}
|
||||
|
||||
int16 Atdsys::getLastSpeechId() {
|
||||
return _text->getLastSpeechId();
|
||||
}
|
||||
|
||||
} // namespace Chewy
|
||||
Reference in New Issue
Block a user