mirror of
https://github.com/libretro/scummvm.git
synced 2025-04-03 23:31:57 +00:00
TINSEL: Simplify font loading and handling
This introduces a new helper method, GetFont(), which loads font data using a memory stream and handles endianess reads internally. This simplifies font loading and endianess handling considerably, and allows for the usage of a common font struct for all engine versions
This commit is contained in:
parent
4f1676e38b
commit
aa7b3470a0
engines/tinsel
@ -43,22 +43,19 @@ SCNHANDLE Font::GetTalkFontHandle() {
|
||||
}
|
||||
|
||||
void Font::FettleFontPal(SCNHANDLE fontPal) {
|
||||
const FONT *pFont;
|
||||
IMAGE *pImg;
|
||||
|
||||
assert(fontPal);
|
||||
assert(_hTagFont); // Tag font not declared
|
||||
assert(_hTalkFont); // Talk font not declared
|
||||
|
||||
pFont = (const FONT *)_vm->_handle->LockMem(_hTagFont);
|
||||
pImg = (IMAGE *)_vm->_handle->LockMem(FROM_32(pFont->fontInit.hObjImg)); // get image for char 0
|
||||
pImg = (IMAGE *)_vm->_handle->LockMem(_vm->_handle->GetFontImageHandle(_hTagFont)); // get image for char 0
|
||||
if (!TinselV2)
|
||||
pImg->hImgPal = TO_32(fontPal);
|
||||
else
|
||||
pImg->hImgPal = 0;
|
||||
|
||||
pFont = (const FONT *)_vm->_handle->LockMem(_hTalkFont);
|
||||
pImg = (IMAGE *)_vm->_handle->LockMem(FROM_32(pFont->fontInit.hObjImg)); // get image for char 0
|
||||
pImg = (IMAGE *)_vm->_handle->LockMem(_vm->_handle->GetFontImageHandle(_hTalkFont)); // get image for char 0
|
||||
if (!TinselV2)
|
||||
pImg->hImgPal = TO_32(fontPal);
|
||||
else
|
||||
|
@ -24,6 +24,7 @@
|
||||
#define BODGE
|
||||
|
||||
#include "common/file.h"
|
||||
#include "common/memstream.h"
|
||||
#include "common/textconsole.h"
|
||||
|
||||
#include "tinsel/drives.h"
|
||||
@ -297,6 +298,47 @@ void Handle::LoadFile(MEMHANDLE *pH) {
|
||||
error(CANNOT_FIND_FILE, szFilename);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a font specified by a SCHNHANDLE
|
||||
* Handles endianess internally
|
||||
* @param offset Handle and offset to data
|
||||
* @return FONT structure
|
||||
*/
|
||||
FONT *Handle::GetFont(SCNHANDLE offset) {
|
||||
byte *fontData = LockMem(offset);
|
||||
const bool isBE = TinselV1Mac || TinselV1Saturn;
|
||||
const uint32 size = (TinselV3 ? 12 * 4 : 11 * 4) + 300 * 4; // FONT struct size
|
||||
Common::MemoryReadStreamEndian *fontStream = new Common::MemoryReadStreamEndian(fontData, size, isBE);
|
||||
|
||||
FONT *font = new FONT();
|
||||
font->xSpacing = fontStream->readSint32();
|
||||
font->ySpacing = fontStream->readSint32();
|
||||
font->xShadow = fontStream->readSint32();
|
||||
font->yShadow = fontStream->readSint32();
|
||||
font->spaceSize = fontStream->readSint32();
|
||||
font->baseColor = TinselV3 ? fontStream->readSint32() : 0;
|
||||
font->fontInit.hObjImg = fontStream->readUint32();
|
||||
font->fontInit.objFlags = fontStream->readSint32();
|
||||
font->fontInit.objID = fontStream->readSint32();
|
||||
font->fontInit.objX = fontStream->readSint32();
|
||||
font->fontInit.objY = fontStream->readSint32();
|
||||
font->fontInit.objZ = fontStream->readSint32();
|
||||
for (int i = 0; i < 300; i++)
|
||||
font->fontDef[i] = fontStream->readUint32();
|
||||
|
||||
delete fontStream;
|
||||
|
||||
return font;
|
||||
}
|
||||
|
||||
SCNHANDLE Handle::GetFontImageHandle(SCNHANDLE offset) {
|
||||
FONT *font = GetFont(offset);
|
||||
SCNHANDLE handle = font->fontInit.hObjImg;
|
||||
delete font;
|
||||
|
||||
return handle;
|
||||
}
|
||||
|
||||
/**
|
||||
* Compute and return the address specified by a SCNHANDLE.
|
||||
* @param offset Handle and offset to data
|
||||
|
@ -33,6 +33,7 @@ class File;
|
||||
|
||||
namespace Tinsel {
|
||||
|
||||
struct FONT;
|
||||
struct MEMHANDLE;
|
||||
|
||||
class Handle {
|
||||
@ -45,6 +46,8 @@ public:
|
||||
*/
|
||||
void SetupHandleTable();
|
||||
|
||||
FONT *GetFont(SCNHANDLE offset);
|
||||
SCNHANDLE GetFontImageHandle(SCNHANDLE offset);
|
||||
byte *LockMem(SCNHANDLE offset);
|
||||
|
||||
void LockScene(SCNHANDLE offset);
|
||||
|
@ -35,24 +35,6 @@ namespace Tinsel {
|
||||
/** TinselV3, base color for the text color replacement */
|
||||
static uint32 g_t3fontBaseColor;
|
||||
|
||||
/**
|
||||
* Returns the handle for the character image.
|
||||
* @param pFont Which font to use
|
||||
* @param c Index of the character
|
||||
*/
|
||||
SCNHANDLE GetFontDef(const FONT *pFont, int c)
|
||||
{
|
||||
if (TinselV3)
|
||||
{
|
||||
const T3_FONT *pT3Font = (const T3_FONT *)pFont;
|
||||
return FROM_32(pT3Font->fontDef[c]);
|
||||
}
|
||||
else
|
||||
{
|
||||
return FROM_32(pFont->fontDef[c]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the length of one line of a string in pixels.
|
||||
* @param szStr String
|
||||
@ -69,7 +51,7 @@ int StringLengthPix(char *szStr, const FONT *pFont) {
|
||||
if (c & 0x80)
|
||||
c = ((c & ~0x80) << 8) + *++szStr;
|
||||
}
|
||||
hImg = GetFontDef(pFont, c);
|
||||
hImg = pFont->fontDef[c];
|
||||
|
||||
if (hImg) {
|
||||
// there is a IMAGE for this character
|
||||
@ -79,14 +61,14 @@ int StringLengthPix(char *szStr, const FONT *pFont) {
|
||||
strLen += FROM_16(pChar->imgWidth);
|
||||
} else
|
||||
// use width of space character
|
||||
strLen += FROM_32(pFont->spaceSize);
|
||||
strLen += pFont->spaceSize;
|
||||
|
||||
// finally add the inter-character spacing
|
||||
strLen += FROM_32(pFont->xSpacing);
|
||||
strLen += pFont->xSpacing;
|
||||
}
|
||||
|
||||
// return length of line in pixels - minus inter-char spacing for last character
|
||||
strLen -= FROM_32(pFont->xSpacing);
|
||||
strLen -= pFont->xSpacing;
|
||||
return (strLen > 0) ? strLen : 0;
|
||||
}
|
||||
|
||||
@ -141,15 +123,16 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
assert(pList);
|
||||
|
||||
// get font pointer
|
||||
const FONT *pFont = (const FONT *)_vm->_handle->LockMem(hFont);
|
||||
const OBJ_INIT *pFontInit = TinselV3 ? (&((const T3_FONT *)pFont)->fontInit) : (&pFont->fontInit);
|
||||
FONT *pFont = _vm->_handle->GetFont(hFont);
|
||||
const OBJ_INIT *pFontInit = &pFont->fontInit;
|
||||
|
||||
// init head of text list
|
||||
pFirst = nullptr;
|
||||
|
||||
// get image for capital W
|
||||
assert(GetFontDef(pFont, (int)'W'));
|
||||
pImg = (const IMAGE *)_vm->_handle->LockMem(GetFontDef(pFont, (int)'W'));
|
||||
SCNHANDLE imgHandle = pFont->fontDef[(int)'W'];
|
||||
assert(imgHandle);
|
||||
pImg = (const IMAGE *)_vm->_handle->LockMem(imgHandle);
|
||||
|
||||
// get height of capital W for offset to next line
|
||||
yOffset = FROM_16(pImg->imgHeight) & ~C16_FLAG_MASK;
|
||||
@ -164,32 +147,24 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
if (c & 0x80)
|
||||
c = ((c & ~0x80) << 8) + *++szStr;
|
||||
}
|
||||
hImg = GetFontDef(pFont, c);
|
||||
hImg = pFont->fontDef[c];
|
||||
|
||||
if (hImg == 0) {
|
||||
// no image for this character
|
||||
|
||||
// add font spacing for a space character
|
||||
xJustify += FROM_32(pFont->spaceSize);
|
||||
xJustify += pFont->spaceSize;
|
||||
} else { // printable character
|
||||
|
||||
int aniX, aniY; // char image animation offsets
|
||||
|
||||
OBJ_INIT oi;
|
||||
oi.hObjImg = FROM_32(pFontInit->hObjImg);
|
||||
oi.objFlags = FROM_32(pFontInit->objFlags);
|
||||
oi.objID = FROM_32(pFontInit->objID);
|
||||
oi.objX = FROM_32(pFontInit->objX);
|
||||
oi.objY = FROM_32(pFontInit->objY);
|
||||
oi.objZ = FROM_32(pFontInit->objZ);
|
||||
|
||||
// allocate and init a character object
|
||||
if (pFirst == NULL)
|
||||
// first time - init head of list
|
||||
pFirst = pChar = InitObject(&oi); // FIXME: endian issue using fontInit!!!
|
||||
pFirst = pChar = InitObject(pFontInit);
|
||||
else
|
||||
// chain to multi-char list
|
||||
pChar = pChar->pSlave = InitObject(&oi); // FIXME: endian issue using fontInit!!!
|
||||
pChar = pChar->pSlave = InitObject(pFontInit);
|
||||
|
||||
// convert image handle to pointer
|
||||
pImg = (const IMAGE *)_vm->_handle->LockMem(hImg);
|
||||
@ -208,9 +183,7 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
pChar->constant = color;
|
||||
|
||||
// set the base font color to be replaced with supplied color, only for Tinsel V3
|
||||
if (TinselV3) {
|
||||
g_t3fontBaseColor = FROM_32(((const T3_FONT*)pFont)->baseColor);
|
||||
}
|
||||
g_t3fontBaseColor = TinselV3 ? pFont->baseColor : 0;
|
||||
|
||||
// get Y animation offset
|
||||
GetAniOffset(hImg, pChar->flags, &aniX, &aniY);
|
||||
@ -232,8 +205,8 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
CopyObject(pShad, pChar);
|
||||
|
||||
// add shadow offsets to characters position
|
||||
pShad->xPos += intToFrac(FROM_32(pFont->xShadow));
|
||||
pShad->yPos += intToFrac(FROM_32(pFont->yShadow));
|
||||
pShad->xPos += intToFrac(pFont->xShadow);
|
||||
pShad->yPos += intToFrac(pFont->yShadow);
|
||||
|
||||
// shadow is behind the character
|
||||
pShad->zPos--;
|
||||
@ -265,14 +238,14 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
}
|
||||
|
||||
// finally add the inter-character spacing
|
||||
xJustify += FROM_32(pFont->xSpacing);
|
||||
xJustify += pFont->xSpacing;
|
||||
|
||||
// next character in string
|
||||
++szStr;
|
||||
}
|
||||
|
||||
// adjust the text y position and add the inter-line spacing
|
||||
yPos += yOffset + FROM_32(pFont->ySpacing);
|
||||
yPos += yOffset + pFont->ySpacing;
|
||||
|
||||
// check for newline
|
||||
if (c == LF_CHAR)
|
||||
@ -280,6 +253,8 @@ OBJECT *ObjectTextOut(OBJECT **pList, char *szStr, int color,
|
||||
++szStr;
|
||||
}
|
||||
|
||||
delete pFont;
|
||||
|
||||
// return head of list
|
||||
return pFirst;
|
||||
}
|
||||
@ -298,9 +273,11 @@ bool IsCharImage(SCNHANDLE hFont, char c) {
|
||||
return false;
|
||||
|
||||
// get font pointer
|
||||
const FONT *pFont = (const FONT *)_vm->_handle->LockMem(hFont);
|
||||
FONT *pFont = _vm->_handle->GetFont(hFont);
|
||||
bool result = pFont->fontDef[c2] != 0;
|
||||
delete pFont;
|
||||
|
||||
return GetFontDef(pFont, c2) != 0;
|
||||
return result;
|
||||
}
|
||||
|
||||
uint32 t3GetBaseColor()
|
||||
|
@ -45,25 +45,12 @@ enum {
|
||||
#define C16_MAP 0xC000
|
||||
#define C16_FLAG_MASK (C16_240 | C16_224 | C16_MAP)
|
||||
|
||||
#include "common/pack-start.h" // START STRUCT PACKING
|
||||
|
||||
/**
|
||||
* Text font data structure.
|
||||
* @note only the pointer is used so the size of fontDef[] is not important.
|
||||
* It is currently set at 300 because it suited me for debugging.
|
||||
*/
|
||||
struct FONT {
|
||||
int xSpacing; ///< x spacing between characters
|
||||
int ySpacing; ///< y spacing between characters
|
||||
int xShadow; ///< x shadow offset
|
||||
int yShadow; ///< y shadow offset
|
||||
int spaceSize; ///< x spacing to use for a space character
|
||||
OBJ_INIT fontInit; ///< structure used to init text objects
|
||||
SCNHANDLE fontDef[300]; ///< image handle array for all characters in the font
|
||||
} PACKED_STRUCT;
|
||||
|
||||
|
||||
struct T3_FONT {
|
||||
int xSpacing; ///< x spacing between characters
|
||||
int ySpacing; ///< y spacing between characters
|
||||
int xShadow; ///< x shadow offset
|
||||
@ -72,9 +59,7 @@ struct T3_FONT {
|
||||
int baseColor; ///< base color which can be replaced, specific to Tinsel 3
|
||||
OBJ_INIT fontInit; ///< structure used to init text objects
|
||||
SCNHANDLE fontDef[300]; ///< image handle array for all characters in the font
|
||||
} PACKED_STRUCT;
|
||||
|
||||
#include "common/pack-end.h" // END STRUCT PACKING
|
||||
};
|
||||
|
||||
|
||||
/** structure for passing the correct parameters to ObjectTextOut */
|
||||
|
Loading…
x
Reference in New Issue
Block a user