scummvm/string.cpp

999 lines
20 KiB
C++
Raw Normal View History

2001-10-09 14:30:12 +00:00
/* ScummVM - Scumm Interpreter
* Copyright (C) 2001 Ludvig Strigeus
* Copyright (C) 2001/2002 The ScummVM project
2001-10-09 14:30:12 +00:00
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Header$
2001-10-09 14:30:12 +00:00
*
*/
#include "stdafx.h"
#include "scumm.h"
int CharsetRenderer::getStringWidth(int arg, byte *text, int pos)
{
2001-10-09 14:30:12 +00:00
byte *ptr;
int width, offs, w;
2001-10-09 14:30:12 +00:00
byte chr;
width = 1;
ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
2001-10-09 14:30:12 +00:00
while ((chr = text[pos++]) != 0) {
if (chr == 0xD)
2001-10-09 14:30:12 +00:00
break;
if (chr == '@')
2001-10-09 14:30:12 +00:00
continue;
if (chr == 254)
chr = 255;
if (chr == 255) {
chr = text[pos++];
if (chr == 3)
2001-10-09 14:30:12 +00:00
break;
if (chr == 8) {
if (arg == 1)
2001-10-09 14:30:12 +00:00
break;
while (text[pos] == ' ')
2001-10-09 14:30:12 +00:00
text[pos++] = '@';
continue;
}
if (chr == 10 || chr == 21 || chr == 12 || chr == 13) {
2001-10-16 10:01:48 +00:00
pos += 2;
continue;
}
if (chr == 9 || chr == 1 || chr == 2)
2001-10-09 14:30:12 +00:00
break;
if (chr == 14) {
int set = text[pos] | (text[pos + 1] << 8);
pos += 2;
ptr = _vm->getResourceAddress(rtCharset, set) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
2001-10-16 10:01:48 +00:00
continue;
}
2001-10-09 14:30:12 +00:00
}
if (_vm->_features & GF_OLD256) {
width += 8;
} else {
offs = READ_LE_UINT32(ptr + chr * 4 + 4);
if (offs) {
if (ptr[offs + 2] >= 0x80) {
w = ptr[offs + 2] - 0x100;
} else {
w = ptr[offs + 2];
}
width += ptr[offs] + w;
2001-10-09 14:30:12 +00:00
}
}
}
return width;
}
void CharsetRenderer::addLinebreaks(int a, byte *str, int pos, int maxwidth)
{
2001-10-09 14:30:12 +00:00
int lastspace = -1;
int curw = 1;
int offs, w;
2001-10-09 14:30:12 +00:00
byte *ptr;
byte chr;
ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
2001-10-09 14:30:12 +00:00
while ((chr = str[pos++]) != 0) {
if (chr == '@')
2001-10-09 14:30:12 +00:00
continue;
if (chr == 254)
chr = 255;
if (chr == 255) {
2001-10-09 14:30:12 +00:00
chr = str[pos++];
if (chr == 3)
2001-10-09 14:30:12 +00:00
break;
if (chr == 8) {
if (a == 1) {
2001-10-09 14:30:12 +00:00
curw = 1;
} else {
while (str[pos] == ' ')
2001-10-09 14:30:12 +00:00
str[pos++] = '@';
}
continue;
}
if (chr == 10 || chr == 21 || chr == 12 || chr == 13) {
2001-10-16 10:01:48 +00:00
pos += 2;
continue;
}
if (chr == 1) {
2001-10-09 14:30:12 +00:00
curw = 1;
continue;
}
if (chr == 2)
2001-10-09 14:30:12 +00:00
break;
if (chr == 14) {
int set = str[pos] | (str[pos + 1] << 8);
pos += 2;
ptr = _vm->getResourceAddress(rtCharset, set) + 29;
if (_vm->_features & GF_SMALL_HEADER)
ptr -= 12;
2001-10-16 10:01:48 +00:00
continue;
}
2001-10-09 14:30:12 +00:00
}
if (chr == ' ')
2001-10-09 14:30:12 +00:00
lastspace = pos - 1;
if (_vm->_features & GF_OLD256) {
curw += getSpacing(chr);
} else {
offs = READ_LE_UINT32(ptr + chr * 4 + 4);
if (offs) {
if (ptr[offs + 2] >= 0x80) {
w = ptr[offs + 2] - 0x100;
} else {
w = ptr[offs + 2];
}
curw += w + ptr[offs];
2001-10-09 14:30:12 +00:00
}
}
if (lastspace == -1)
continue;
2001-10-09 14:30:12 +00:00
if (curw > maxwidth) {
str[lastspace] = 0xD;
curw = 1;
pos = lastspace + 1;
lastspace = -1;
}
}
}
void Scumm::unkMessage1()
{
byte buffer[100];
_msgPtrToAdd = buffer;
2001-10-09 14:30:12 +00:00
_messagePtr = addMessageToStack(_messagePtr);
2001-10-26 17:34:50 +00:00
if (buffer[0] == 0xFF && buffer[1] == 10) {
uint32 a, b;
a = buffer[2] | (buffer[3] << 8) | (buffer[6] << 16) | (buffer[7] << 24);
b =
buffer[10] | (buffer[11] << 8) | (buffer[14] << 16) | (buffer[15] <<
24);
// if (_saveSound != 1)
talkSound(a, b, 1);
}
// warning("unkMessage1(\"%s\")", buffer);
2001-10-09 14:30:12 +00:00
}
void Scumm::unkMessage2()
{
2001-10-09 14:30:12 +00:00
byte buf[100], *tmp;
_msgPtrToAdd = buf;
tmp = _messagePtr = addMessageToStack(_messagePtr);
if (string[3].color == 0)
2001-10-16 10:01:48 +00:00
string[3].color = 4;
2001-10-09 14:30:12 +00:00
2001-10-26 17:34:50 +00:00
warning("unkMessage2(\"%s\")", buf);
2001-10-09 14:30:12 +00:00
_messagePtr = tmp;
}
void Scumm::CHARSET_1()
{
2001-10-09 14:30:12 +00:00
int s, i, t, c;
int frme;
Actor *a;
byte *buffer;
2002-02-13 17:33:52 +00:00
if (!(_features & GF_AFTER_V7)) {
if (!_haveMsg || (camera._dest.x >> 3) != (camera._cur.x >> 3) ||
camera._cur.x != camera._last.x)
return;
} else {
if (!_haveMsg)
return;
}
2001-10-09 14:30:12 +00:00
a = NULL;
2001-10-16 10:01:48 +00:00
if (_vars[VAR_TALK_ACTOR] != 0xFF)
a = derefActorSafe(_vars[VAR_TALK_ACTOR], "CHARSET_1");
2001-10-09 14:30:12 +00:00
if (a && string[0].overhead != 0) {
if (!(_features & GF_AFTER_V6)) {
string[0].xpos = a->x - camera._cur.x + 160;
2001-10-09 14:30:12 +00:00
2001-10-26 17:34:50 +00:00
if (_vars[VAR_V5_TALK_STRING_Y] < 0) {
s = (a->scaley * (int)_vars[VAR_V5_TALK_STRING_Y]) / 0xFF;
string[0].ypos =
((_vars[VAR_V5_TALK_STRING_Y] - s) >> 1) + s - a->elevation + a->y;
} else {
2001-10-26 17:34:50 +00:00
string[0].ypos = _vars[VAR_V5_TALK_STRING_Y];
}
if (string[0].ypos < 1)
string[0].ypos = 1;
if (string[0].xpos < 80)
string[0].xpos = 80;
if (string[0].xpos > 240)
string[0].xpos = 240;
2001-10-09 14:30:12 +00:00
} else {
s = a->scaley * a->new_1 / 0xFF;
string[0].ypos = ((a->new_1 - s) >> 1) + s - a->elevation + a->y;
if (string[0].ypos < 1)
string[0].ypos = 1;
s = a->scalex * a->new_2 / 0xFF;
string[0].xpos = ((a->new_2 - s) >> 1) + s + a->x - camera._cur.x + 160;
if (string[0].xpos < 80)
string[0].xpos = 80;
if (string[0].xpos > 240)
string[0].xpos = 240;
2001-10-09 14:30:12 +00:00
}
}
2001-10-16 10:01:48 +00:00
charset._top = string[0].ypos;
charset._left = string[0].xpos;
charset._left2 = string[0].xpos;
charset._curId = string[0].charset;
2001-10-09 14:30:12 +00:00
if (a && a->charset)
charset._curId = a->charset;
2001-10-16 10:01:48 +00:00
charset._center = string[0].center;
charset._right = string[0].right;
2001-10-09 14:30:12 +00:00
charset._color = _charsetColor;
2001-10-26 17:34:50 +00:00
_bkColor = 0;
if (!(_features & GF_OLD256)) // FIXME
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12];
else
charset._colorMap[i] = _charsetData[charset._curId][i];
2001-10-09 14:30:12 +00:00
if (_keepText) {
charset._strLeft = gdi._mask_left;
charset._strRight = gdi._mask_right;
charset._strTop = gdi._mask_top;
charset._strBottom = gdi._mask_bottom;
2001-10-09 14:30:12 +00:00
}
if (!_haveMsg || _talkDelay)
return;
if (_haveMsg != 0xFF) {
if (_sfxMode == 0)
stopTalk();
2001-10-09 14:30:12 +00:00
return;
}
if (a && !string[0].no_talk_anim) {
startAnimActor(a, a->talkFrame1);
_useTalkAnims = true;
2001-10-09 14:30:12 +00:00
}
_talkDelay = _defaultTalkDelay;
if (!_keepText) {
if (_features & GF_OLD256) {
gdi._mask_left = string[0].xpos;
gdi._mask_top = string[0].ypos;
gdi._mask_bottom = string[0].ypos + 8;
gdi._mask_right = 320;
if (string[0].ypos <= 16) // If we are cleaning the text line, clean 2 lines.
gdi._mask_bottom = 16;
}
2001-10-09 14:30:12 +00:00
restoreCharsetBg();
charset._xpos2 = string[0].xpos;
charset._ypos2 = string[0].ypos;
2001-10-09 14:30:12 +00:00
}
2001-10-16 10:01:48 +00:00
t = charset._right - string[0].xpos - 1;
2001-10-09 14:30:12 +00:00
if (charset._center) {
if (t > charset._xpos2)
t = charset._xpos2;
2001-10-09 14:30:12 +00:00
t <<= 1;
}
buffer = charset._buffer + charset._bufPos;
charset.addLinebreaks(0, buffer, 0, t);
2001-10-09 14:30:12 +00:00
_lastXstart = virtscr[0].xstart;
if (charset._center) {
charset._xpos2 -= charset.getStringWidth(0, buffer, 0) >> 1;
if (charset._xpos2 < 0)
charset._xpos2 = 0;
2001-10-09 14:30:12 +00:00
}
charset._disableOffsX = charset._unk12 = !_keepText;
do {
c = *buffer++;
if (c == 0) {
2001-10-09 14:30:12 +00:00
_haveMsg = 1;
_keepText = false;
break;
}
2001-10-16 10:01:48 +00:00
if (c == 13) {
newLine:;
if (_features & GF_OLD256) {
charset._ypos2 = 8;
charset._xpos2 = 0;
continue;
} else {
charset._xpos2 = string[0].xpos;
if (charset._center) {
charset._xpos2 -= charset.getStringWidth(0, buffer, 0) >> 1;
}
if (_features & GF_SMALL_HEADER)
charset._ypos2 += getResourceAddress(rtCharset, charset._curId)[18];
else
charset._ypos2 += getResourceAddress(rtCharset, charset._curId)[30];
charset._disableOffsX = 1;
continue;
2001-10-09 14:30:12 +00:00
}
2001-10-16 10:01:48 +00:00
}
2001-10-09 14:30:12 +00:00
if (c == 0xFE)
c = 0xFF;
2001-10-16 10:01:48 +00:00
if (c != 0xFF) {
charset._left = charset._xpos2;
charset._top = charset._ypos2;
if (_features & GF_OLD256)
charset.printCharOld(c);
else if (!(_features & GF_AFTER_V6)) {
// if (!_vars[VAR_V5_CHARFLAG]) { /* FIXME */
if (!(a && _noSubtitles))
2001-10-26 17:34:50 +00:00
charset.printChar(c);
// }
2001-10-26 17:34:50 +00:00
} else {
if (!(a && _noSubtitles))
charset.printChar(c);
2001-10-09 14:30:12 +00:00
}
2001-10-26 17:34:50 +00:00
charset._xpos2 = charset._left;
charset._ypos2 = charset._top;
2001-10-16 10:01:48 +00:00
_talkDelay += _vars[VAR_CHARINC];
continue;
2001-10-09 14:30:12 +00:00
}
2001-10-16 10:01:48 +00:00
c = *buffer++;
if (c == 3) {
2001-10-16 10:01:48 +00:00
_haveMsg = 0xFF;
_keepText = false;
break;
} else if (c == 1) {
2001-10-16 10:01:48 +00:00
goto newLine;
} else if (c == 2) {
2001-10-16 10:01:48 +00:00
_haveMsg = 0;
_keepText = true;
break;
} else if (c == 9) {
frme = *buffer++;
frme |= *buffer++ << 8;
2001-10-16 10:01:48 +00:00
if (a)
startAnimActor(a, frme);
} else if (c == 10) {
uint32 a, b;
a =
buffer[0] | (buffer[1] << 8) | (buffer[4] << 16) | (buffer[5] << 24);
b =
buffer[8] | (buffer[9] << 8) | (buffer[12] << 16) | (buffer[13] <<
24);
talkSound(a, b, 2);
buffer += 14;
} else if (c == 14) {
int oldy = getResourceAddress(rtCharset, charset._curId)[30];
2001-10-16 10:01:48 +00:00
charset._curId = *buffer++;
buffer += 2;
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12];
else
charset._colorMap[i] = _charsetData[charset._curId][i];
charset._ypos2 -=
getResourceAddress(rtCharset, charset._curId)[30] - oldy;
} else if (c == 12) {
2001-10-16 10:01:48 +00:00
int color;
color = *buffer++;
color |= *buffer++ << 8;
if (color == 0xFF)
2001-10-16 10:01:48 +00:00
charset._color = _charsetColor;
else
charset._color = color;
} else if (c == 13) {
buffer += 2;
2001-10-16 10:01:48 +00:00
} else {
warning("CHARSET_1: invalid code %d", c);
2001-10-09 14:30:12 +00:00
}
} while (1);
charset._bufPos = buffer - charset._buffer;
gdi._mask_left = charset._strLeft;
gdi._mask_right = charset._strRight;
gdi._mask_top = charset._strTop;
gdi._mask_bottom = charset._strBottom;
2001-10-09 14:30:12 +00:00
}
void Scumm::drawString(int a)
{
2001-10-09 14:30:12 +00:00
byte buf[256];
byte *charsetptr, *space;
2001-10-09 14:30:12 +00:00
int i;
byte byte1 = 0, chr;
2001-10-26 17:34:50 +00:00
uint color;
2001-10-09 14:30:12 +00:00
_msgPtrToAdd = buf;
_messagePtr = addMessageToStack(_messagePtr);
2001-10-16 10:01:48 +00:00
charset._left2 = charset._left = string[a].xpos;
charset._top = string[a].ypos;
charset._curId = string[a].charset;
charset._center = string[a].center;
charset._right = string[a].right;
charset._color = string[a].color;
2001-10-26 17:34:50 +00:00
_bkColor = 0;
2001-10-09 14:30:12 +00:00
charset._unk12 = 1;
charset._disableOffsX = 1;
if (!(_features & GF_OLD256)) {
charsetptr = getResourceAddress(rtCharset, charset._curId);
assert(charsetptr);
charsetptr += 29;
if (_features & GF_SMALL_HEADER)
charsetptr -= 12;
2001-10-09 14:30:12 +00:00
for (i = 0; i < 4; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12];
else
charset._colorMap[i] = _charsetData[charset._curId][i];
2001-10-09 14:30:12 +00:00
byte1 = charsetptr[1];
}
2001-10-09 14:30:12 +00:00
_msgPtrToAdd = buf;
/* trim from the right */
space = NULL;
while (*_msgPtrToAdd) {
if (*_msgPtrToAdd == ' ') {
if (!space)
space = _msgPtrToAdd;
2001-10-09 14:30:12 +00:00
} else {
space = NULL;
}
_msgPtrToAdd++;
}
if (space)
*space = '\0';
2001-10-09 14:30:12 +00:00
if (charset._center) {
charset._left -= charset.getStringWidth(a, buf, 0) >> 1;
}
charset._ignoreCharsetMask = 1;
if (!buf[0]) {
buf[0] = ' ';
buf[1] = 0;
}
for (i = 0; (chr = buf[i++]) != 0;) {
if (chr == 254)
chr = 255;
if (chr == 255) {
2001-10-09 14:30:12 +00:00
chr = buf[i++];
switch (chr) {
2001-10-09 14:30:12 +00:00
case 9:
case 10:
case 13:
case 14:
2001-10-09 14:30:12 +00:00
i += 2;
break;
case 1:
case 8:
2001-10-09 14:30:12 +00:00
if (charset._center) {
charset._left = charset._left2 - charset.getStringWidth(a, buf, i);
} else {
charset._left = charset._left2;
}
2001-10-26 17:34:50 +00:00
charset._top += byte1;
break;
case 12:
color = buf[i] + (buf[i + 1] << 8);
i += 2;
if (color == 0xFF)
2001-10-26 17:34:50 +00:00
charset._color = string[a].color;
else
charset._color = color;
break;
2001-10-09 14:30:12 +00:00
}
} else {
if (a == 1 && (_features & GF_AFTER_V6))
2001-10-26 17:34:50 +00:00
charset._blitAlso = true;
if (_features & GF_OLD256)
charset.printCharOld(chr);
else
charset.printChar(chr);
2001-10-26 17:34:50 +00:00
charset._blitAlso = false;
2001-10-09 14:30:12 +00:00
}
}
charset._ignoreCharsetMask = 0;
if (a == 0) {
charset._xpos2 = charset._left;
charset._ypos2 = charset._top;
}
2001-10-09 14:30:12 +00:00
}
byte *Scumm::addMessageToStack(byte *msg)
{
2001-10-09 14:30:12 +00:00
int num, numorg;
byte *ptr, chr;
numorg = num = _numInMsgStack;
ptr = getResourceAddress(rtTemp, 6);
2001-10-09 14:30:12 +00:00
if (ptr == NULL)
2001-10-09 14:30:12 +00:00
error("Message stack not allocated");
if (msg == NULL)
error("Bad message in addMessageToStack");
while ((chr = *msg++) != 0) {
2001-10-09 14:30:12 +00:00
if (num > 500)
error("Message stack overflow");
ptr[num++] = chr;
if (chr == 255) {
2001-10-09 14:30:12 +00:00
ptr[num++] = chr = *msg++;
if (chr != 1 && chr != 2 && chr != 3 && chr != 8) {
2001-10-09 14:30:12 +00:00
ptr[num++] = chr = *msg++;
ptr[num++] = chr = *msg++;
}
}
}
ptr[num++] = 0;
2001-10-09 14:30:12 +00:00
_numInMsgStack = num;
num = numorg;
while (1) {
ptr = getResourceAddress(rtTemp, 6);
2001-10-09 14:30:12 +00:00
chr = ptr[num++];
if (chr == 0)
2001-10-09 14:30:12 +00:00
break;
if (chr == 0xFF) {
chr = ptr[num++];
switch (chr) {
2001-10-09 14:30:12 +00:00
case 4:
2001-10-16 10:01:48 +00:00
unkAddMsgToStack2(READ_LE_UINT16(ptr + num));
num += 2;
2001-10-09 14:30:12 +00:00
break;
case 5:
2001-10-16 10:01:48 +00:00
unkAddMsgToStack3(READ_LE_UINT16(ptr + num));
num += 2;
2001-10-09 14:30:12 +00:00
break;
case 6:
2001-10-16 10:01:48 +00:00
unkAddMsgToStack4(READ_LE_UINT16(ptr + num));
num += 2;
2001-10-09 14:30:12 +00:00
break;
case 7:
2001-10-16 10:01:48 +00:00
unkAddMsgToStack5(READ_LE_UINT16(ptr + num));
num += 2;
2001-10-09 14:30:12 +00:00
break;
case 9:
//#if defined(DOTT)
case 10:
case 12:
case 13:
case 14:
//#endif
2001-10-09 14:30:12 +00:00
*_msgPtrToAdd++ = 0xFF;
*_msgPtrToAdd++ = chr;
2001-10-16 10:01:48 +00:00
*_msgPtrToAdd++ = ptr[num++];
*_msgPtrToAdd++ = ptr[num++];
2001-10-09 14:30:12 +00:00
break;
default:
*_msgPtrToAdd++ = 0xFF;
*_msgPtrToAdd++ = chr;
}
} else {
if (chr != '@') {
2001-10-09 14:30:12 +00:00
*_msgPtrToAdd++ = chr;
}
}
}
*_msgPtrToAdd = 0;
_numInMsgStack = numorg;
2001-10-09 14:30:12 +00:00
return msg;
}
void Scumm::unkAddMsgToStack2(int var)
{
int num, max;
2001-10-09 14:30:12 +00:00
byte flag;
2001-10-09 14:30:12 +00:00
num = readVar(var);
if (num < 0) {
*_msgPtrToAdd++ = '-';
num = -num;
}
flag = 0;
max = 10000;
do {
if (num >= max || flag) {
*_msgPtrToAdd++ = num / max + '0';
num -= (num / max) * max;
flag = 1;
2001-10-09 14:30:12 +00:00
}
max /= 10;
if (max == 1)
flag = 1;
2001-10-09 14:30:12 +00:00
} while (max);
}
void Scumm::unkAddMsgToStack3(int var)
{
int num, i;
2001-10-09 14:30:12 +00:00
num = readVar(var);
if (num) {
for (i = 1; i < _maxVerbs; i++) {
if (num == _verbs[i].verbid && !_verbs[i].type && !_verbs[i].saveid) {
addMessageToStack(getResourceAddress(rtVerb, i));
2001-10-09 14:30:12 +00:00
break;
}
}
} else {
addMessageToStack((byte *)"");
2001-10-09 14:30:12 +00:00
}
}
void Scumm::unkAddMsgToStack4(int var)
{
2001-10-09 14:30:12 +00:00
int num;
num = readVar(var);
if (num) {
addMessageToStack(getObjOrActorName(num));
2001-10-09 14:30:12 +00:00
} else {
addMessageToStack((byte *)"");
2001-10-09 14:30:12 +00:00
}
}
void Scumm::unkAddMsgToStack5(int var)
{
2001-10-09 14:30:12 +00:00
byte *ptr;
if (_features & GF_AFTER_V6)
var = readVar(var);
2001-10-09 14:30:12 +00:00
if (var) {
ptr = getStringAddress(var);
2001-10-09 14:30:12 +00:00
if (ptr) {
addMessageToStack(ptr);
return;
}
}
addMessageToStack((byte *)"");
2001-10-09 14:30:12 +00:00
}
void Scumm::initCharset(int charsetno)
{
2001-10-09 14:30:12 +00:00
int i;
if (_features & GF_OLD256)
charsetno = !charsetno;
if (_features & GF_SMALL_HEADER)
loadCharset(charsetno);
else if (!getResourceAddress(rtCharset, charsetno))
loadCharset(charsetno);
2001-10-09 14:30:12 +00:00
2001-10-16 10:01:48 +00:00
string[0].t_charset = charsetno;
string[1].t_charset = charsetno;
2001-10-09 14:30:12 +00:00
for (i = 0; i < 0x10; i++)
if (_features & GF_SMALL_HEADER)
charset._colorMap[i] = _charsetData[charset._curId][i - 12];
else
charset._colorMap[i] = _charsetData[charset._curId][i];
2001-10-09 14:30:12 +00:00
}
void CharsetRenderer::printCharOld(int chr)
{ // Loom3 / Zak256
VirtScreen *vs;
byte *char_ptr, *dest_ptr;
unsigned int buffer = 0, mask = 0, x = 0, y = 0;
unsigned char color;
_vm->checkRange(_vm->_maxCharsets - 1, 0, _curId,
"Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
return;
if (chr == '@')
return;
if (_unk12) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_unk12 = 0;
}
char_ptr = _vm->getResourceAddress(rtCharset, _curId) + 224 + (chr + 1) * 8;
dest_ptr = vs->screenPtr + vs->xstart + (_top - vs->topline) * 320 + _left;
_vm->updateDirtyRect(vs->number, _left, _left + 8, _top - vs->topline,
_top - vs->topline + 8, 0);
for (y = 0; y < 8; y++) {
for (x = 0; x < 8; x++) {
if ((mask >>= 1) == 0) {
buffer = *char_ptr++;
mask = 0x80;
}
color = ((buffer & mask) != 0);
if (color)
*(dest_ptr + y * 320 + x) = _color;
}
}
_left += getSpacing(chr);
if (_left > _strRight)
_strRight = _left;
if (_top + 8 > _strBottom)
_strBottom = _top + 8;
}
void CharsetRenderer::printChar(int chr)
{
int d, right;
2001-10-09 14:30:12 +00:00
VirtScreen *vs;
_vm->checkRange(_vm->_maxCharsets - 1, 1, _curId,
"Printing with bad charset %d");
if ((vs = _vm->findVirtScreen(_top)) == NULL)
2001-10-09 14:30:12 +00:00
return;
if (chr == '@')
2001-10-09 14:30:12 +00:00
return;
_ptr = _vm->getResourceAddress(rtCharset, _curId) + 29;
if (_vm->_features & GF_SMALL_HEADER)
_ptr -= 12;
2001-10-09 14:30:12 +00:00
_bpp = _unk2 = *_ptr;
_invNumBits = 8 - _bpp;
_bitMask = 0xFF << _invNumBits;
_colorMap[1] = _color;
_charOffs = READ_LE_UINT32(_ptr + chr * 4 + 4);
2001-10-09 14:30:12 +00:00
if (!_charOffs)
return;
assert(_charOffs < 0x10000);
_charPtr = _ptr + _charOffs;
_width = _charPtr[0];
_height = _charPtr[1];
if (_unk12) {
_strLeft = 0;
_strTop = 0;
_strRight = 0;
_strBottom = 0;
}
if (_disableOffsX) {
_offsX = 0;
} else {
d = _charPtr[2];
if (d >= 0x80)
2001-10-09 14:30:12 +00:00
d -= 0x100;
_offsX = d;
}
d = _charPtr[3];
if (d >= 0x80)
2001-10-09 14:30:12 +00:00
d -= 0x100;
_offsY = d;
_top += _offsY;
_left += _offsX;
right = _left + _width;
if (right > _right + 1 || _left < 0) {
2001-10-09 14:30:12 +00:00
_left = right;
_top -= _offsY;
return;
}
_disableOffsX = 0;
if (_unk12) {
_strLeft = _left;
_strTop = _top;
_strRight = _left;
_strBottom = _top;
_unk12 = 0;
}
if (_left < _strLeft)
_strLeft = _left;
if (_top < _strTop)
_strTop = _top;
_drawTop = _top - vs->topline;
if (_drawTop < 0)
_drawTop = 0;
2001-10-16 10:01:48 +00:00
2001-10-09 14:30:12 +00:00
_bottom = _drawTop + _height + _offsY;
2001-10-26 17:34:50 +00:00
_vm->updateDirtyRect(vs->number, _left, right, _drawTop, _bottom, 0);
2001-10-09 14:30:12 +00:00
2001-10-26 17:34:50 +00:00
#if defined(OLD)
if (vs->number == 0)
2001-10-09 14:30:12 +00:00
_hasMask = true;
2001-10-26 17:34:50 +00:00
#else
if (vs->number != 0)
2001-10-26 17:34:50 +00:00
_blitAlso = false;
if (vs->number == 0 && _blitAlso == 0)
2001-10-26 17:34:50 +00:00
_hasMask = true;
#endif
2001-10-09 14:30:12 +00:00
_dest_ptr = _backbuff_ptr = vs->screenPtr
2001-10-09 14:30:12 +00:00
+ vs->xstart + _drawTop * 320 + _left;
2001-10-26 17:34:50 +00:00
#if !defined(OLD)
if (_blitAlso) {
#else
if (1) {
#endif
_dest_ptr = _bgbak_ptr = _vm->getResourceAddress(rtBuffer, vs->number + 5)
2001-10-26 17:34:50 +00:00
+ vs->xstart + _drawTop * 320 + _left;
}
_mask_ptr = _vm->getResourceAddress(rtBuffer, 9)
+ _drawTop * 40 + _left / 8 + _vm->_screenStartStrip;
2001-10-09 14:30:12 +00:00
_revBitMask = revBitMask[_left & 7];
2001-10-09 14:30:12 +00:00
_virtScreenHeight = vs->height;
_charPtr += 4;
drawBits();
2001-10-26 17:34:50 +00:00
#if !defined(OLD)
if (_blitAlso)
blit(_backbuff_ptr, _bgbak_ptr, _width, _height);
#endif
2001-10-09 14:30:12 +00:00
_left += _width;
if (_left > _strRight)
2001-10-09 14:30:12 +00:00
_strRight = _left;
if (_top + _height > _strBottom)
_strBottom = _top + _height;
_top -= _offsY;
}
void CharsetRenderer::drawBits()
{
2001-10-09 14:30:12 +00:00
bool usemask;
byte *dst, *mask, maskmask;
int y, x;
2001-10-09 14:30:12 +00:00
int maskpos;
int color;
byte numbits, bits;
2001-10-09 14:30:12 +00:00
usemask = (_vm->_curVirtScreen->number == 0 && _ignoreCharsetMask == 0);
2001-10-09 14:30:12 +00:00
bits = *_charPtr++;
numbits = 8;
dst = _dest_ptr;
2001-10-09 14:30:12 +00:00
mask = _mask_ptr;
y = 0;
for (y = 0; y < _height && y + _drawTop < _virtScreenHeight;) {
2001-10-09 14:30:12 +00:00
maskmask = _revBitMask;
maskpos = 0;
for (x = 0; x < _width; x++) {
color = (bits & _bitMask) >> _invNumBits;
2001-10-09 14:30:12 +00:00
if (color) {
if (usemask) {
mask[maskpos] |= maskmask;
}
*dst = _colorMap[color];
}
dst++;
bits <<= _bpp;
if ((numbits -= _bpp) == 0) {
2001-10-09 14:30:12 +00:00
bits = *_charPtr++;
numbits = 8;
}
if ((maskmask >>= 1) == 0) {
2001-10-09 14:30:12 +00:00
maskmask = 0x80;
maskpos++;
}
}
dst = (_dest_ptr += 320);
2001-10-09 14:30:12 +00:00
mask += 40;
y++;
}
}
int CharsetRenderer::getSpacing(char chr)
{
int space;
if (_curId == 1) { // do spacing for variable width old-style font
switch (chr) {
case '.':
space = 1;
break;
case 'i':
case '\'':
case 'I':
case '!':
space = 2;
break;
case 'l':
space = 3;
break;
case ' ':
space = 4;
break;
case 'W':
case 'w':
case 'N':
case 'M':
case 'm':
space = 8;
break;
default:
space = 6;
}
} else
space = 7;
return space;
}