scummvm/engines/nancy/font.cpp
Kaloyan Chehlarski 6abc61edf7 NANCY: Render text in The Vampire Diaries
Text in TVD is now rendered in the textbox, and should be
pixel-accurate to the original engine.
2021-04-29 15:59:00 +03:00

239 lines
6.3 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 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.
*
*/
#include "engines/nancy/font.h"
#include "engines/nancy/nancy.h"
#include "engines/nancy/resource.h"
#include "engines/nancy/graphics.h"
#include "engines/nancy/util.h"
namespace Nancy {
void Font::read(Common::SeekableReadStream &stream) {
_transColor = g_nancy->_graphicsManager->getTransColor();
_maxCharWidth = 0;
_fontHeight = 0;
Common::String imageName;
readFilename(stream, imageName);
g_nancy->_resource->loadImage(imageName, _image);
char desc[0x20];
stream.read(desc, 0x20);
desc[0x1F] = '\0';
_description = desc;
stream.skip(8);
_colorCoordsOffset.x = stream.readUint16LE();
_colorCoordsOffset.y = stream.readUint16LE();
stream.skip(2);
_spaceWidth = stream.readUint16LE();
stream.skip(2);
_uppercaseOffset = stream.readUint16LE();
_lowercaseOffset = stream.readUint16LE();
_digitOffset = stream.readUint16LE();
_periodOffset = stream.readUint16LE();
_commaOffset = stream.readUint16LE();
_equalitySignOffset = stream.readUint16LE();
_colonOffset = stream.readUint16LE();
_dashOffset = stream.readUint16LE();
_questionMarkOffset = stream.readUint16LE();
_exclamationMarkOffset = stream.readUint16LE();
_percentOffset = stream.readUint16LE();
_ampersandOffset = stream.readUint16LE();
_asteriskOffset = stream.readUint16LE();
_leftBracketOffset = stream.readUint16LE();
_rightBracketOffset = stream.readUint16LE();
_plusOffset = stream.readUint16LE();
_apostropheOffset = stream.readUint16LE();
_semicolonOffset = stream.readUint16LE();
_slashOffset = stream.readUint16LE();
_symbolRects.reserve(78);
for (uint i = 0; i < 78; ++i) {
_symbolRects.push_back(Common::Rect());
Common::Rect &cur = _symbolRects[i];
readRect(stream, cur);
if (g_nancy->getGameType() == kGameTypeVampire) {
++cur.bottom;
++cur.right;
}
_maxCharWidth = MAX<int>(cur.width(), _maxCharWidth);
_fontHeight = MAX<int>(cur.height(), _maxCharWidth);
}
}
int Font::getCharWidth(uint32 chr) const {
return getCharacterSourceRect(chr).width();
}
void Font::drawChar(Graphics::Surface *dst, uint32 chr, int x, int y, uint32 color) const {
Common::Rect srcRect = getCharacterSourceRect(chr);
if (color != 0) {
srcRect.translate(_colorCoordsOffset.x, _colorCoordsOffset.y);
}
uint vampireAdjust = g_nancy->getGameType() == kGameTypeVampire ? 1 : 0;
uint width = MAX<int>(srcRect.width() - vampireAdjust, 0);
uint height = srcRect.height();
uint yOffset = getFontHeight() - height;
height = MAX<int>(height - vampireAdjust, 0);
for (uint curY = 0; curY < height; ++curY) {
for (uint curX = 0; curX < width; ++curX) {
switch (g_nancy->_graphicsManager->getInputPixelFormat().bytesPerPixel) {
case 1: {
byte colorID = *(const byte *)_image.getBasePtr(srcRect.left + curX, srcRect.top + curY);
if (colorID != _transColor) {
uint8 r, g, b;
uint curColor = _image.getPalette()[colorID];
r = curColor & 0xFF;
g = (curColor & 0xFF00) >> 8;
b = (curColor & 0xFF0000) >> 16;
*(uint16 *)dst->getBasePtr(x + curX, y + yOffset + curY) = dst->format.RGBToColor(r, g, b);
}
break;
}
case 2: {
uint16 curColor = *(const uint16 *)_image.getBasePtr(srcRect.left + curX, srcRect.top + curY);
if (curColor != _transColor) {
uint8 r, g, b;
_image.format.colorToRGB(curColor, r, g, b);
*(uint16 *)dst->getBasePtr(x + curX, y + yOffset + curY) = dst->format.RGBToColor(r, g, b);
}
break;
}
default:
break;
}
}
}
}
void Font::wordWrap(const Common::String &str, int maxWidth, Common::Array<Common::String> &lines, int initWidth) const {
Common::String temp;
for (const char *c = str.begin(); c != str.end(); ++c) {
if (*c == '\n') {
lines.push_back(temp);
temp.clear();
continue;
}
temp += *c;
int size = getStringWidth(temp) + (lines.size() == 0 ? initWidth : 0);
if (size >= maxWidth) {
do {
temp.deleteLastChar();
--c;
} while (temp.size() && temp.lastChar() != ' ');
lines.push_back(temp);
temp.clear();
}
}
if (temp.size()) {
lines.push_back(temp);
}
}
Common::Rect Font::getCharacterSourceRect(char chr) const {
using namespace Common;
uint offset = 0;
Common::Rect ret;
if (isUpper(chr)) {
offset = chr + _uppercaseOffset - 0x41;
} else if (isLower(chr)) {
offset = chr + _lowercaseOffset - 0x61;
} else if (isDigit(chr)) {
offset = chr + _digitOffset - 0x30;
} else if (isSpace(chr)) {
ret.setWidth(_spaceWidth - 1); // Not sure why we sutract 1
return ret;
} else if (isPunct(chr)) {
switch (chr) {
case '.':
offset = _periodOffset;
break;
case ',':
offset = _commaOffset;
break;
case '=':
offset = _equalitySignOffset;
break;
case ':':
offset = _colonOffset;
break;
case '-':
offset = _dashOffset;
break;
case '?':
offset = _questionMarkOffset;
break;
case '!':
offset = _exclamationMarkOffset;
break;
case '%':
offset = _percentOffset;
break;
case '&':
offset = _ampersandOffset;
break;
case '*':
offset = _asteriskOffset;
break;
case '(':
offset = _leftBracketOffset;
break;
case ')':
offset = _rightBracketOffset;
break;
case '+':
offset = _plusOffset;
break;
case '\'':
offset = _apostropheOffset;
break;
case ';':
offset = _semicolonOffset;
break;
case '/':
offset = _slashOffset;
break;
default:
error("Unsupported FONT character: %c", chr);
}
}
ret = _symbolRects[offset];
return ret;
}
} // End of namespace Nancy