scummvm/engines/cge/text.cpp
2022-10-23 22:46:19 +02:00

217 lines
5.0 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/>.
*
*/
/*
* This code is based on original Soltys source code
* Copyright (c) 1994-1995 Janusz B. Wisniewski and L.K. Avalon
*/
#include "cge/general.h"
#include "cge/text.h"
#include "cge/talk.h"
#include "cge/game.h"
#include "cge/snail.h"
#include "cge/cge_main.h"
#include "common/str.h"
namespace CGE {
Text::Text(CGEEngine *vm, const char *fname) : _vm(vm) {
_vm->mergeExt(_fileName, fname, kSayExt);
if (!_vm->_resman->exist(_fileName))
error("No talk (%s)", _fileName);
int16 txtCount = count() + 1;
if (!txtCount)
error("Unable to read dialog file %s", _fileName);
_cache = new Handler[txtCount];
for (_size = 0; _size < txtCount; _size++) {
_cache[_size]._ref = 0;
_cache[_size]._text = nullptr;
}
load();
}
Text::~Text() {
clear();
delete[] _cache;
}
int16 Text::count() {
EncryptedStream tf(_vm->_resman, _fileName);
if (tf.err())
return -1;
Common::String line;
char tmpStr[kLineMax + 1];
int counter = 0;
for (line = tf.readLine(); !tf.eos(); line = tf.readLine()) {
char *s;
assert(line.size() <= 513);
Common::strlcpy(tmpStr, line.c_str(), sizeof(tmpStr));
if ((s = strtok(tmpStr, " =,;/\t\n")) == nullptr)
continue;
if (!Common::isDigit(*s))
continue;
counter++;
}
return counter;
}
void Text::clear() {
for (Handler *p = _cache, *q = p + _size; p < q; p++) {
if (p->_ref) {
p->_ref = 0;
delete[] p->_text;
p->_text = nullptr;
}
}
}
void Text::load() {
EncryptedStream tf(_vm->_resman, _fileName);
assert(!tf.err());
Common::String line;
char tmpStr[kLineMax + 1];
int idx;
for (idx = 0, line = tf.readLine(); !tf.eos(); line = tf.readLine()) {
int n = line.size();
char *s;
Common::strlcpy(tmpStr, line.c_str(), sizeof(tmpStr));
if ((s = strtok(tmpStr, " =,;/\t\n")) == nullptr)
continue;
if (!Common::isDigit(*s))
continue;
int r = atoi(s);
s += strlen(s);
if (s < tmpStr + n)
++s;
_cache[idx]._ref = r;
size_t ln = strlen(s) + 1;
_cache[idx]._text = new char[ln];
Common::strcpy_s(_cache[idx]._text, ln, s);
idx++;
}
}
char *Text::getText(int ref) {
int i;
for (i = 0; (i < _size) && (_cache[i]._ref != ref); i++)
;
if (i < _size)
return _cache[i]._text;
warning("getText: Unable to find ref %d", ref);
return nullptr;
}
void Text::say(const char *text, Sprite *spr) {
_vm->killText();
if (!text)
return;
if (*text == 0)
return;
_vm->_talk = new Talk(_vm, text, kTBRound);
if (!_vm->_talk)
return;
bool east = spr->_flags._east;
int x = (east) ? (spr->_x + spr->_w - 2) : (spr->_x + 2);
int y = spr->_y + 2;
Speaker *speaker = new Speaker(_vm);
uint16 sw = speaker->_w;
if (east) {
if (x + sw + kTextRoundCorner + 5 >= kScrWidth)
east = false;
} else {
if (x <= 5 + kTextRoundCorner + sw)
east = true;
}
x = (east) ? (spr->_x + spr->_w - 2) : (spr->_x + 2 - sw);
if (spr->_ref == 1)
x += ((east) ? -10 : 10); // Hero
_vm->_talk->_flags._kill = true;
_vm->_talk->_flags._bDel = true;
_vm->_talk->setName(_vm->_text->getText(kSayName));
_vm->_talk->gotoxy(x - (_vm->_talk->_w - sw) / 2 - 3 + 6 * east, y - speaker->_h - _vm->_talk->_h + 1);
_vm->_talk->_z = 125;
_vm->_talk->_ref = kSayRef;
speaker->gotoxy(x, _vm->_talk->_y + _vm->_talk->_h - 1);
speaker->_z = 126;
speaker->_flags._slav = true;
speaker->_flags._kill = true;
speaker->setName(_vm->_text->getText(kSayName));
speaker->step(east);
speaker->_ref = kSayRef;
_vm->_vga->_showQ->insert(_vm->_talk, _vm->_vga->_showQ->last());
_vm->_vga->_showQ->insert(speaker, _vm->_vga->_showQ->last());
}
void CGEEngine::inf(const char *text, bool wideSpace) {
debugC(1, kCGEDebugEngine, "CGEEngine::inf(%s)", text);
if (!text)
return;
if (*text == 0)
return;
killText();
_talk = new Talk(this, text, kTBRect, wideSpace);
if (!_talk)
return;
_talk->_flags._kill = true;
_talk->_flags._bDel = true;
_talk->setName(_text->getText(kInfName));
_talk->center();
_talk->gotoxy(_talk->_x, _talk->_y - 20);
_talk->_z = 126;
_talk->_ref = kInfRef;
_vga->_showQ->insert(_talk, _vga->_showQ->last());
}
void Text::sayTime(Sprite *spr) {
TimeDate curTime;
_vm->_system->getTimeAndDate(curTime);
char t[6];
Common::sprintf_s(t, "%d:%02d", curTime.tm_hour, curTime.tm_min);
say(t, spr);
}
} // End of namespace CGE