scummvm/engines/tetraedge/te/te_font2.cpp
2023-12-24 13:19:25 +01:00

233 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 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 "tetraedge/te/te_font2.h"
#include "tetraedge/tetraedge.h"
#include "tetraedge/te/te_3d_object2.h"
#include "tetraedge/te/te_core.h"
namespace Tetraedge {
bool operator==(const KernChars &l, const KernChars &r) {
return l._c1 == r._c1 && l._c2 == r._c2;
}
TeFont2::TeFont2() : _numChars(0), _hasKernData(false), _maxHeight(0) {
}
TeFont2::~TeFont2() {
}
bool TeFont2::load(const Common::Path &path) {
if (_loadedPath == path)
return true; // already open
TeCore *core = g_engine->getCore();
Common::FSNode node = core->findFile(path);
return load(node);
}
bool TeFont2::load(const Common::FSNode &node) {
const Common::Path path = node.getPath();
unload();
setAccessName(path);
_loadedPath = path;
if (!node.isReadable()) {
warning("TeFont2::load: Can't read from %s", path.toString(Common::Path::kNativeSeparator).c_str());
return false;
}
Common::File file;
file.open(node);
if (!Te3DObject2::loadAndCheckFourCC(file, "TESF")) {
warning("TeFont2::load: Invalid magic in %s", path.toString(Common::Path::kNativeSeparator).c_str());
return false;
}
_numChars = file.readUint32LE();
if (_numChars > 65535)
error("TeFont2::load: improbable number of char points %d", _numChars);
TeVector2s32::deserialize(file, _somePt);
TeVector3f32::deserialize(file, _someVec);
_hasKernData = (file.readByte() != 0);
if (_hasKernData) {
uint32 numKernData = file.readUint32LE();
if (numKernData > 10000)
error("TeFont2::load: improbable number of kerning points %d", numKernData);
for (uint32 i = 0; i < numKernData; i++) {
KernChars kc;
TeVector3f32 vec;
kc._c1 = file.readUint32LE();
kc._c2 = file.readUint32LE();
vec.x() = file.readFloatLE();
vec.y() = file.readFloatLE();
_kernData[kc] = vec;
//debug("KernChars: '%c'-'%c' (%.2f, %.2f)", (char)kc._c1, (char)kc._c2, vec.x(), vec.y());
}
}
for (uint32 i = 0; i < _numChars; i++) {
GlyphData2 g;
g._xSz = file.readFloatLE();
g._ySz = file.readFloatLE();
_maxHeight = MAX(_maxHeight, g._ySz);
g._xOff = file.readFloatLE();
g._yOff = file.readFloatLE();
g._xAdvance = file.readFloatLE();
// TODO: What are these other floats?
for (uint j = 0; j < 3; j++)
g._floats[j] = file.readFloatLE();
g._vec.x() = file.readFloatLE();
g._vec.y() = file.readFloatLE();
_glyphs.push_back(g);
uint32 charNo = file.readUint32LE();
_uintArray.push_back(charNo);
/*
if (i >= 35 && i <= 127)
debug("Char data %c: sz (%.1f %.1f) off (%.1f %.1f) xadv %.1f %.1f %.1f %.1f texloc (%.1f, %.1f) %d", (char)i,
g._xSz, g._ySz, g._xOff, g._yOff, g._xAdvance, g._floats[0], g._floats[1], g._floats[2],
g._vec.x(), g._vec.y(), charNo);
*/
}
if (!_texture.load(file, "png")) {
warning("Invalid png data in %s", path.toString(Common::Path::kNativeSeparator).c_str());
return false;
}
return true;
}
void TeFont2::unload() {
_texture.free();
_uintArray.clear();
_glyphs.clear();
_kernData.clear();
_numChars = 0;
}
int TeFont2::getFontHeight() const {
return _maxHeight;
}
int TeFont2::getMaxCharWidth() const {
int maxWidth = 0;
for (auto &glyph : _glyphs) {
maxWidth = MAX(maxWidth, (int)glyph._xAdvance);
}
return maxWidth;
}
int TeFont2::getCharWidth(uint32 chr) const {
if (chr >= _glyphs.size())
return 0;
return (int)_glyphs[chr]._xAdvance;
}
void TeFont2::drawChar(Graphics::Surface *dst, uint32 chr, int x, int y, uint32 color) const {
if (chr >= _glyphs.size())
return;
if (_texture.empty())
error("Call to TeFont2::drawChar but no texture loaded");
const GlyphData2 g = _glyphs[chr];
if (!g._xSz || !g._ySz)
return;
Common::Rect srcRect;
srcRect.left = (int)g._vec.x();
srcRect.top = _texture.h - (int)g._vec.y() - g._ySz;
srcRect.right = srcRect.left + g._xSz;
srcRect.bottom = srcRect.top + g._ySz;
int dstX = x + g._xOff;
int dstY = _maxHeight - g._yOff;
if (dstX + srcRect.width() > dst->w)
srcRect.right = srcRect.left + (dst->w - dstX);
if (dstY + srcRect.height() > dst->h)
srcRect.bottom = srcRect.top + (dst->h - dstY);
debug("TeFont2::drawChar %c (%d, %d) from (%d,%d-%d,%d)", chr, x, y,
srcRect.left, srcRect.top, srcRect.right, srcRect.bottom);
dst->copyRectToSurface(_texture, dstX, dstY, srcRect);
}
int TeFont2::getKerningOffset(uint32 left, uint32 right) const {
KernChars kc { left, right };
if (_kernData.contains(kc))
return (int)_kernData[kc].x();
return 0;
}
TeVector3f32 TeFont2::kerning(uint pxSize, uint isocode1, uint isocode2) {
KernChars kc { isocode1, isocode2 };
if (_kernData.contains(kc))
return _kernData[kc];
return TeVector3f32();
}
Common::Rect TeFont2::getBBox(const Common::String &str, int fontSize) {
Common::Rect rect;
for (uint i = 0; i < str.size(); i++) {
uint c = str[i];
if (c >= _glyphs.size())
continue;
const GlyphData2 &g = _glyphs[c];
rect.top = MIN(rect.top, (int16)-g._yOff);
rect.bottom = MAX(rect.bottom, (int16)(-g._yOff + g._ySz));
rect.right += g._xAdvance;
if (i < str.size() - 1) {
rect.right += kerning(fontSize, c, str[i+1]).x();
}
}
return rect;
}
Common::Rect TeFont2::getBoundingBox(uint32 chr) const {
if (chr > _glyphs.size())
return Common::Rect();
Common::Rect rect;
rect.left = (int)_glyphs[chr]._xOff;
rect.right = rect.left + _glyphs[chr]._xSz;
rect.top = _maxHeight - _glyphs[chr]._yOff;
rect.bottom = _maxHeight;
return rect;
}
float TeFont2::height(uint pxSize) {
return _maxHeight;
}
Graphics::Font *TeFont2::getAtSize(uint size) {
return this;
}
} // end namespace Tetraedge