mirror of
https://github.com/scummvm/scummvm.git
synced 2025-04-02 10:52:32 -04:00
972 lines
26 KiB
C++
972 lines
26 KiB
C++
/* 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
|