479 lines
12 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(0), you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation(0), 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(0), 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(0), if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "titanic/pet_control/pet_text.h"
namespace Titanic {
CPetText::CPetText(uint count) :
_stringsMerged(false), _maxCharsPerLine(-1), _lineCount(0),
_linesStart(-1), _field3C(0), _field40(0), _field44(0),
_backR(0xff), _backG(0xff), _backB(0xff),
_textR(0), _textG(0), _textB(200),
_fontNumber(0), _field64(0), _field68(0), _field6C(0),
_hasBorder(true), _scrollTop(0), _textCursor(nullptr), _field7C(0) {
setupArrays(count);
}
void CPetText::setupArrays(int count) {
freeArrays();
if (count < 10 || count > 60)
count = 10;
_array.resize(count);
}
void CPetText::freeArrays() {
_array.clear();
}
void CPetText::setup() {
for (int idx = 0; idx < (int)_array.size(); ++idx) {
_array[idx]._line.clear();
setLineColor(idx, _textR, _textG, _textB);
_array[idx]._string3.clear();
}
_lineCount = 0;
_stringsMerged = false;
}
void CPetText::setLineColor(uint lineNum, uint col) {
setLineColor(lineNum, col & 0xff, (col >> 16) & 0xff, (col >> 8) & 0xff);
}
void CPetText::setLineColor(uint lineNum, byte r, byte g, byte b) {
char buffer[6];
if (!r)
r = 1;
if (!g)
g = 1;
if (!b)
b = 1;
buffer[0] = TEXTCMD_SET_COLOR;
buffer[1] = r;
buffer[2] = g;
buffer[3] = b;
buffer[4] = TEXTCMD_SET_COLOR;
buffer[5] = '\0';
_array[lineNum]._rgb = buffer;
_stringsMerged = false;
}
void CPetText::load(SimpleFile *file, int param) {
if (!param) {
uint numLines = file->readNumber();
uint charsPerLine = file->readNumber();
uint count = file->readNumber();
_bounds = file->readRect();
_field3C = file->readNumber();
_field40 = file->readNumber();
_field44 = file->readNumber();
_backR = file->readNumber();
_backG = file->readNumber();
_backB = file->readNumber();
_textR = file->readNumber();
_textG = file->readNumber();
_textB = file->readNumber();
_hasBorder = file->readNumber() != 0;
_scrollTop = file->readNumber();
resize(numLines);
setMaxCharsPerLine(charsPerLine);
assert(_array.size() >= count);
for (uint idx = 0; idx < count; ++idx) {
_array[idx]._line = file->readString();
_array[idx]._rgb = file->readString();
_array[idx]._string3 = file->readString();
}
}
}
void CPetText::save(SimpleFile *file, int indent) {
int numLines = _lineCount + 1;
file->writeNumberLine(_array.size(), indent);
file->writeNumberLine(_maxCharsPerLine, indent);
file->writeNumberLine(numLines, indent);
file->writeRect(_bounds, indent);
file->writeNumberLine(_field3C, indent);
file->writeNumberLine(_field40, indent);
file->writeNumberLine(_field44, indent);
file->writeNumberLine(_backR, indent);
file->writeNumberLine(_backG, indent);
file->writeNumberLine(_backB, indent);
file->writeNumberLine(_textR, indent);
file->writeNumberLine(_textG, indent);
file->writeNumberLine(_textB, indent);
file->writeNumberLine(_hasBorder, indent);
file->writeNumberLine(_scrollTop, indent);
for (int idx = 0; idx < numLines; ++idx) {
file->writeQuotedLine(_array[idx]._line, indent);
file->writeQuotedLine(_array[idx]._rgb, indent);
file->writeQuotedLine(_array[idx]._string3, indent);
}
}
void CPetText::draw(CScreenManager *screenManager) {
Rect tempRect = _bounds;
if (_hasBorder) {
// Create border effect
// Top edge
tempRect.bottom = tempRect.top + 1;
screenManager->fillRect(SURFACE_BACKBUFFER, &tempRect, _backR, _backG, _backB);
// Bottom edge
tempRect.top = _bounds.bottom - 1;
tempRect.bottom = _bounds.bottom;
screenManager->fillRect(SURFACE_BACKBUFFER, &tempRect, _backR, _backG, _backB);
// Left edge
tempRect = _bounds;
tempRect.right = tempRect.left + 1;
screenManager->fillRect(SURFACE_BACKBUFFER, &tempRect, _backR, _backG, _backB);
// Right edge
tempRect = _bounds;
tempRect.left = tempRect.right - 1;
screenManager->fillRect(SURFACE_BACKBUFFER, &tempRect, _backR, _backG, _backB);
}
getTextHeight(screenManager);
tempRect = _bounds;
tempRect.grow(-2);
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
screenManager->writeString(SURFACE_BACKBUFFER, tempRect, _scrollTop, _lines, _textCursor);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::mergeStrings() {
if (!_stringsMerged) {
_lines.clear();
for (int idx = 0; idx <= _lineCount; ++idx) {
CString line = _array[idx]._rgb + _array[idx]._string3 +
_array[idx]._line + "\n";
_lines += line;
}
_stringsMerged = true;
}
}
void CPetText::resize(uint count) {
if (!count || _array.size() == count)
return;
_array.clear();
_array.resize(count);
}
CString CPetText::getText() const {
CString result = "";
for (int idx = 0; idx <= _lineCount; ++idx)
result += _array[idx]._line;
return result;
}
void CPetText::setText(const CString &str) {
setup();
appendText(str);
}
void CPetText::appendText(const CString &str) {
int lineSize = _array[_lineCount]._line.size();
int strSize = str.size();
if (_maxCharsPerLine == -1) {
// No limit on horizontal characters, so append string to current line
_array[_lineCount]._line += str;
} else if ((lineSize + strSize) <= _maxCharsPerLine) {
// New string fits into line, so add it on
_array[_lineCount]._line += str;
} else {
// Only add part of the str up to the maximum allowed limit for line
_array[_lineCount]._line += str.left(_maxCharsPerLine - lineSize);
}
updateStr3(_lineCount);
_stringsMerged = false;
}
void CPetText::setColor(uint col) {
_textR = col & 0xff;
_textG = (col >> 8) & 0xff;
_textB = (col >> 16) & 0xff;
}
void CPetText::setColor(byte r, byte g, byte b) {
_textR = r;
_textG = g;
_textB = b;
}
void CPetText::remapColors(uint count, uint *srcColors, uint *destColors) {
if (_lineCount >= 0) {
for (int lineNum = 0; lineNum <= _lineCount; ++lineNum) {
// Get the rgb values
uint r = _array[lineNum]._rgb[1];
uint g = _array[lineNum]._rgb[2];
uint b = _array[lineNum]._rgb[3];
uint color = r | (g << 8) | (b << 16);
for (uint index = 0; index < count; ++index) {
if (color == srcColors[index]) {
// Found a match, so replace the color
setLineColor(lineNum, destColors[lineNum]);
break;
}
}
}
}
_stringsMerged = false;
}
void CPetText::setMaxCharsPerLine(int maxChars) {
if (maxChars >= -1 && maxChars < 257)
_maxCharsPerLine = maxChars;
}
void CPetText::updateStr3(int lineNum) {
if (_field64 > 0 && _field68 > 0) {
char line[5];
line[0] = line[3] = TEXTCMD_NPC;
line[1] = _field64;
line[2] = _field68;
line[4] = '\0';
_array[lineNum]._string3 = CString(line);
_stringsMerged = false;
_field64 = _field68 = 0;
}
}
int CPetText::getTextWidth(CScreenManager *screenManager) {
mergeStrings();
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
int textWidth = screenManager->stringWidth(_lines);
screenManager->setFontNumber(oldFontNumber);
return textWidth;
}
int CPetText::getTextHeight(CScreenManager *screenManager) {
mergeStrings();
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
int textHeight = screenManager->getTextBounds(_lines, _bounds.width());
screenManager->setFontNumber(oldFontNumber);
return textHeight;
}
void CPetText::deleteLastChar() {
if (!_array[_lineCount]._line.empty()) {
_array[_lineCount]._line.deleteLastChar();
_stringsMerged = false;
}
}
void CPetText::setNPC(int val1, int npcId) {
_field64 = val1;
_field68 = npcId;
}
void CPetText::scrollUp(CScreenManager *screenManager) {
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
_scrollTop -= screenManager->getFontHeight();
constrainScrollUp(screenManager);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::scrollDown(CScreenManager *screenManager) {
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
_scrollTop += screenManager->getFontHeight();
constrainScrollDown(screenManager);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::scrollUpPage(CScreenManager *screenManager) {
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
_scrollTop -= getPageHeight(screenManager);
constrainScrollUp(screenManager);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::scrollDownPage(CScreenManager *screenManager) {
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
_scrollTop += getPageHeight(screenManager);
constrainScrollDown(screenManager);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::scrollToTop(CScreenManager *screenManager) {
_scrollTop = 0;
}
void CPetText::scrollToBottom(CScreenManager *screenManager) {
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
_scrollTop = getTextHeight(screenManager);
constrainScrollDown(screenManager);
screenManager->setFontNumber(oldFontNumber);
}
void CPetText::constrainScrollUp(CScreenManager *screenManager) {
if (_scrollTop < 0)
_scrollTop = 0;
}
void CPetText::constrainScrollDown(CScreenManager *screenManager) {
// Figure out the maximum scroll amount allowed
int maxScroll = getTextHeight(screenManager) - _bounds.height() - 4;
if (maxScroll < 0)
maxScroll = 0;
if (_scrollTop > maxScroll)
_scrollTop = maxScroll;
}
int CPetText::getPageHeight(CScreenManager *screenManager) {
int textHeight = _bounds.height();
int oldFontNumber = screenManager->setFontNumber(_fontNumber);
int fontHeight = screenManager->getFontHeight();
screenManager->setFontNumber(oldFontNumber);
if (fontHeight) {
int lines = textHeight / fontHeight;
if (lines > 1)
--lines;
return lines * fontHeight;
} else {
return 0;
}
}
void CPetText::addLine(const CString &str) {
addLine(str, _textR, _textG, _textB);
}
void CPetText::addLine(const CString &str, uint color) {
addLine(str, color & 0xff, (color >> 8) & 0xff,
(color >> 16) & 0xff);
}
void CPetText::addLine(const CString &str, byte r, byte g, byte b) {
if (_lineCount == ((int)_array.size() - 1)) {
// Lines array is full
if (_array.size() > 1) {
// Delete the oldest line, and add a new entry at the end
_array.remove_at(0);
_array.resize(_array.size() + 1);
}
--_lineCount;
}
setLineColor(_lineCount, r, g, b);
appendText(str);
++_lineCount;
}
bool CPetText::handleKey(char c) {
switch (c) {
case (char)Common::KEYCODE_BACKSPACE:
deleteLastChar();
break;
case (char)Common::KEYCODE_RETURN:
return true;
default:
if ((byte)c >= 32 && (byte)c <= 127)
appendText(CString(c, 1));
break;
}
return false;
}
void CPetText::showCursor(int mode) {
CScreenManager *screenManager = CScreenManager::setCurrent();
_textCursor = screenManager->_textCursor;
if (_textCursor) {
_textCursor->setPos(Point(0, 0));
_textCursor->setSize(Point(2, 10));
_textCursor->setColor(0, 0, 0);
_textCursor->setBlinkRate(300);
_textCursor->setMode(mode);
_textCursor->setBounds(_bounds);
_textCursor->show();
}
}
void CPetText::hideCursor() {
if (_textCursor) {
_textCursor->setMode(-1);
_textCursor->hide();
_textCursor = nullptr;
}
}
int CPetText::getNPCNum(uint npcId, uint startIndex) {
if (!_stringsMerged) {
mergeStrings();
if (!_stringsMerged)
return -1;
}
uint size = _lines.size();
if (startIndex < 5 || startIndex >= size)
return -1;
// Loop through string
for (const char *strP = _lines.c_str(); size >= 5; ++strP, --size) {
if (*strP == 26) {
byte id = *(strP - 2);
if (id == npcId)
return *(strP - 1);
} else if (*strP == 27) {
strP += 4;
}
}
return - 1;
}
void CPetText::setFontNumber(int fontNumber) {
if (fontNumber >= 0 && fontNumber <= 2)
_fontNumber = fontNumber;
}
} // End of namespace Titanic