2007-05-30 21:56:52 +00:00
|
|
|
/* 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.
|
2002-12-25 21:04:47 +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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2014-02-18 01:34:24 +00:00
|
|
|
*
|
2002-12-25 21:04:47 +00:00
|
|
|
*/
|
|
|
|
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2003-10-03 18:33:57 +00:00
|
|
|
#include "scumm/charset.h"
|
|
|
|
#include "scumm/scumm.h"
|
|
|
|
#include "scumm/nut_renderer.h"
|
2005-04-10 12:59:17 +00:00
|
|
|
#include "scumm/util.h"
|
2006-09-17 23:35:09 +00:00
|
|
|
#include "scumm/he/intern_he.h"
|
2006-02-15 00:57:50 +00:00
|
|
|
#include "scumm/he/wiz_he.h"
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2003-10-03 18:33:57 +00:00
|
|
|
namespace Scumm {
|
|
|
|
|
2007-01-28 20:11:31 +00:00
|
|
|
/*
|
|
|
|
TODO:
|
|
|
|
Right now our charset renderers directly access _textSurface, as well as the
|
|
|
|
virtual screens of ScummEngine. Ideally, this would not be the case. Instead,
|
2007-09-19 08:40:12 +00:00
|
|
|
ScummVM would simply pass the appropriate Surface to the resp. methods.
|
2007-01-28 20:11:31 +00:00
|
|
|
Of course it is not quite as simple, various flags and offsets have to
|
|
|
|
be taken into account for that.
|
|
|
|
|
2007-09-19 08:40:12 +00:00
|
|
|
The advantage will be cleaner coder (easier to debug, in particular), and a
|
2007-01-28 20:11:31 +00:00
|
|
|
better separation of the various modules.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2004-09-20 22:04:06 +00:00
|
|
|
void ScummEngine::loadCJKFont() {
|
2005-05-10 22:56:25 +00:00
|
|
|
Common::File fp;
|
2004-09-20 22:04:06 +00:00
|
|
|
_useCJKMode = false;
|
2007-07-10 00:39:12 +00:00
|
|
|
_textSurfaceMultiplier = 1;
|
2008-08-02 21:36:08 +00:00
|
|
|
_newLineCharacter = 0;
|
2007-07-10 00:39:12 +00:00
|
|
|
|
2007-08-14 13:42:05 +00:00
|
|
|
if (_game.version <= 5 && _game.platform == Common::kPlatformFMTowns && _language == Common::JA_JPN) { // FM-TOWNS v3 / v5 Kanji
|
2011-07-09 21:57:25 +00:00
|
|
|
#if defined(DISABLE_TOWNS_DUAL_LAYER_MODE) || !defined(USE_RGB_COLOR)
|
2011-06-16 18:51:09 +00:00
|
|
|
GUIErrorMessage("FM-Towns Kanji font drawing requires dual graphics layer support which is disabled in this build");
|
2010-10-05 19:04:52 +00:00
|
|
|
error("FM-Towns Kanji font drawing requires dual graphics layer support which is disabled in this build");
|
2011-02-20 11:03:44 +00:00
|
|
|
#else
|
2011-01-16 20:24:58 +00:00
|
|
|
// use FM-TOWNS font rom, since game files don't have kanji font resources
|
2011-07-09 18:06:18 +00:00
|
|
|
_cjkFont = Graphics::FontSJIS::createFont(_game.platform);
|
2010-10-17 13:08:00 +00:00
|
|
|
if (!_cjkFont)
|
2011-01-16 20:24:58 +00:00
|
|
|
error("SCUMM::Font: Could not open file 'FMT_FNT.ROM'");
|
2007-07-10 00:39:12 +00:00
|
|
|
_textSurfaceMultiplier = 2;
|
2010-10-17 13:08:00 +00:00
|
|
|
_useCJKMode = true;
|
2011-02-20 11:03:44 +00:00
|
|
|
#endif
|
2009-11-22 08:20:20 +00:00
|
|
|
} else if (_game.id == GID_LOOM && _game.platform == Common::kPlatformPCEngine && _language == Common::JA_JPN) {
|
2011-07-09 21:57:25 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2009-11-22 08:20:20 +00:00
|
|
|
// use PC-Engine System Card, since game files don't have kanji font resources
|
2011-07-09 18:06:18 +00:00
|
|
|
_cjkFont = Graphics::FontSJIS::createFont(_game.platform);
|
|
|
|
if (!_cjkFont)
|
|
|
|
error("SCUMM::Font: Could not open file 'pce.cdbios'");
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2011-07-09 18:06:18 +00:00
|
|
|
_cjkFont->setDrawingMode(Graphics::FontSJIS::kShadowMode);
|
2011-07-09 22:44:18 +00:00
|
|
|
_2byteWidth = _2byteHeight = 12;
|
2012-09-26 02:17:31 +00:00
|
|
|
_useCJKMode = true;
|
2011-07-09 21:57:25 +00:00
|
|
|
#endif
|
2009-12-30 20:51:49 +00:00
|
|
|
} else if (_game.id == GID_MONKEY && _game.platform == Common::kPlatformSegaCD && _language == Common::JA_JPN) {
|
|
|
|
int numChar = 1413;
|
|
|
|
_2byteWidth = 16;
|
|
|
|
_2byteHeight = 16;
|
|
|
|
_useCJKMode = true;
|
|
|
|
_newLineCharacter = 0x5F;
|
|
|
|
// charset resources are not inited yet, load charset later
|
|
|
|
_2byteFontPtr = new byte[_2byteWidth * _2byteHeight * numChar / 8];
|
|
|
|
// set byte 0 to 0xFF (0x00 when loaded) to indicate that the font was not loaded
|
|
|
|
_2byteFontPtr[0] = 0xFF;
|
2009-12-31 18:52:42 +00:00
|
|
|
} else if ((_game.version >= 7 && (_language == Common::KO_KOR || _language == Common::JA_JPN || _language == Common::ZH_TWN)) ||
|
|
|
|
(_game.version >= 3 && _language == Common::ZH_CNA)) {
|
2004-10-23 23:08:53 +00:00
|
|
|
int numChar = 0;
|
2004-09-20 22:04:06 +00:00
|
|
|
const char *fontFile = NULL;
|
2004-10-23 23:08:53 +00:00
|
|
|
|
2004-10-22 10:25:56 +00:00
|
|
|
switch (_language) {
|
2004-09-20 22:04:06 +00:00
|
|
|
case Common::KO_KOR:
|
|
|
|
fontFile = "korean.fnt";
|
2004-10-23 23:08:53 +00:00
|
|
|
numChar = 2350;
|
2004-09-20 22:04:06 +00:00
|
|
|
break;
|
|
|
|
case Common::JA_JPN:
|
2006-02-20 16:51:30 +00:00
|
|
|
fontFile = (_game.id == GID_DIG) ? "kanji16.fnt" : "japanese.fnt";
|
2008-04-26 20:33:59 +00:00
|
|
|
numChar = 8192;
|
2004-09-20 22:04:06 +00:00
|
|
|
break;
|
|
|
|
case Common::ZH_TWN:
|
2010-06-27 13:06:22 +00:00
|
|
|
// Both The DIG and COMI use same font
|
|
|
|
fontFile = "chinese.fnt";
|
|
|
|
numChar = 13630;
|
2004-09-20 22:04:06 +00:00
|
|
|
break;
|
2009-12-31 18:52:42 +00:00
|
|
|
case Common::ZH_CNA:
|
|
|
|
if (_game.id == GID_FT || _game.id == GID_LOOM || _game.id == GID_INDY3 ||
|
|
|
|
_game.id == GID_INDY4 || _game.id == GID_MONKEY || _game.id == GID_MONKEY2 ||
|
|
|
|
_game.id == GID_TENTACLE) {
|
|
|
|
fontFile = "chinese_gb16x12.fnt";
|
|
|
|
numChar = 8178;
|
|
|
|
}
|
|
|
|
break;
|
2004-09-20 22:04:06 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (fontFile && fp.open(fontFile)) {
|
|
|
|
debug(2, "Loading CJK Font");
|
|
|
|
_useCJKMode = true;
|
2007-07-10 00:39:12 +00:00
|
|
|
_textSurfaceMultiplier = 1; // No multiplication here
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-07-21 22:40:02 +00:00
|
|
|
switch (_language) {
|
|
|
|
case Common::KO_KOR:
|
|
|
|
fp.seek(2, SEEK_CUR);
|
|
|
|
_2byteWidth = fp.readByte();
|
|
|
|
_2byteHeight = fp.readByte();
|
2008-04-28 15:21:36 +00:00
|
|
|
_newLineCharacter = 0xff;
|
2007-07-21 22:40:02 +00:00
|
|
|
break;
|
|
|
|
case Common::JA_JPN:
|
|
|
|
_2byteWidth = 16;
|
|
|
|
_2byteHeight = 16;
|
2008-04-28 15:21:36 +00:00
|
|
|
_newLineCharacter = 0xfe;
|
2007-07-21 22:40:02 +00:00
|
|
|
break;
|
|
|
|
case Common::ZH_TWN:
|
|
|
|
_2byteWidth = 16;
|
2007-07-21 22:57:21 +00:00
|
|
|
_2byteHeight = 15;
|
2008-04-28 15:21:36 +00:00
|
|
|
_newLineCharacter = 0x21;
|
2007-07-21 22:40:02 +00:00
|
|
|
break;
|
2009-12-31 18:52:42 +00:00
|
|
|
case Common::ZH_CNA:
|
|
|
|
_2byteWidth = 12;
|
|
|
|
_2byteHeight = 12;
|
|
|
|
_newLineCharacter = 0x21;
|
|
|
|
break;
|
2007-07-21 22:40:02 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2004-09-20 22:04:06 +00:00
|
|
|
|
|
|
|
_2byteFontPtr = new byte[((_2byteWidth + 7) / 8) * _2byteHeight * numChar];
|
|
|
|
fp.read(_2byteFontPtr, ((_2byteWidth + 7) / 8) * _2byteHeight * numChar);
|
|
|
|
fp.close();
|
2004-10-23 23:08:53 +00:00
|
|
|
} else {
|
2009-09-25 11:39:22 +00:00
|
|
|
if (fontFile)
|
2011-01-16 20:24:58 +00:00
|
|
|
error("SCUMM::Font: Could not open %s",fontFile);
|
2009-09-25 11:39:22 +00:00
|
|
|
else
|
2011-01-16 20:24:58 +00:00
|
|
|
error("SCUMM::Font: Could not load any font");
|
2004-09-20 22:04:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
byte *ScummEngine::get2byteCharPtr(int idx) {
|
2011-07-09 18:06:18 +00:00
|
|
|
if (_game.platform == Common::kPlatformFMTowns || _game.platform == Common::kPlatformPCEngine)
|
2011-07-07 14:33:14 +00:00
|
|
|
return 0;
|
|
|
|
|
2007-09-19 08:40:12 +00:00
|
|
|
switch (_language) {
|
2004-09-20 22:04:06 +00:00
|
|
|
case Common::KO_KOR:
|
|
|
|
idx = ((idx % 256) - 0xb0) * 94 + (idx / 256) - 0xa1;
|
|
|
|
break;
|
|
|
|
case Common::JA_JPN:
|
2011-07-09 18:06:18 +00:00
|
|
|
if (_game.id == GID_MONKEY && _game.platform == Common::kPlatformSegaCD && _language == Common::JA_JPN) {
|
2009-12-30 20:51:49 +00:00
|
|
|
// init pointer to charset resource
|
|
|
|
if (_2byteFontPtr[0] == 0xFF) {
|
|
|
|
int charsetId = 5;
|
|
|
|
int numChar = 1413;
|
|
|
|
byte *charsetPtr = getResourceAddress(rtCharset, charsetId);
|
|
|
|
if (charsetPtr == 0)
|
|
|
|
error("ScummEngine::get2byteCharPtr: charset %d not found", charsetId);
|
|
|
|
memcpy(_2byteFontPtr, charsetPtr + 46, _2byteWidth * _2byteHeight * numChar / 8);
|
|
|
|
}
|
|
|
|
|
2009-12-31 10:49:16 +00:00
|
|
|
idx = (SWAP_CONSTANT_16(idx) & 0x7fff) - 1;
|
2009-11-22 08:20:20 +00:00
|
|
|
}
|
2010-10-17 13:08:00 +00:00
|
|
|
|
2004-09-20 22:04:06 +00:00
|
|
|
break;
|
|
|
|
case Common::ZH_TWN:
|
2007-07-21 22:40:02 +00:00
|
|
|
{
|
|
|
|
int base = 0;
|
|
|
|
byte low = idx % 256;
|
|
|
|
int high = 0;
|
|
|
|
|
|
|
|
if (low >= 0x20 && low <= 0x7e) {
|
2008-04-26 20:33:59 +00:00
|
|
|
base = (3 * low + 81012) * 5;
|
2007-07-21 22:40:02 +00:00
|
|
|
} else {
|
|
|
|
if (low >= 0xa1 && low <= 0xa3) {
|
|
|
|
base = 392820;
|
|
|
|
low += 0x5f;
|
|
|
|
} else if (low >= 0xa4 && low <= 0xc6) {
|
|
|
|
base = 0;
|
|
|
|
low += 0x5c;
|
|
|
|
} else if (low >= 0xc9 && low <= 0xf9) {
|
|
|
|
base = 162030;
|
|
|
|
low += 0x37;
|
|
|
|
} else {
|
|
|
|
base = 392820;
|
|
|
|
low = 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (low != 0xff) {
|
|
|
|
high = idx / 256;
|
|
|
|
if (high >= 0x40 && high <= 0x7e) {
|
|
|
|
high -= 0x40;
|
|
|
|
} else {
|
|
|
|
high -= 0x62;
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:57:21 +00:00
|
|
|
base += (low * 0x9d + high) * 30;
|
2007-07-21 22:40:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:57:21 +00:00
|
|
|
return _2byteFontPtr + base;
|
2007-07-21 22:40:02 +00:00
|
|
|
}
|
2009-12-31 18:52:42 +00:00
|
|
|
case Common::ZH_CNA:
|
|
|
|
idx = ((idx % 256) - 0xa1)* 94 + ((idx / 256) - 0xa1);
|
|
|
|
break;
|
2004-09-20 22:04:06 +00:00
|
|
|
default:
|
|
|
|
idx = 0;
|
|
|
|
}
|
2008-01-28 00:14:17 +00:00
|
|
|
return _2byteFontPtr + ((_2byteWidth + 7) / 8) * _2byteHeight * idx;
|
2004-09-20 22:04:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
|
2003-10-02 22:42:03 +00:00
|
|
|
CharsetRenderer::CharsetRenderer(ScummEngine *vm) {
|
2003-05-20 15:12:33 +00:00
|
|
|
_top = 0;
|
|
|
|
_left = 0;
|
|
|
|
_startLeft = 0;
|
|
|
|
_right = 0;
|
|
|
|
|
|
|
|
_color = 0;
|
|
|
|
|
|
|
|
_center = false;
|
|
|
|
_hasMask = false;
|
2004-04-08 23:41:10 +00:00
|
|
|
_textScreenID = kMainVirtScreen;
|
2003-05-20 15:12:33 +00:00
|
|
|
_blitAlso = false;
|
|
|
|
_firstChar = false;
|
|
|
|
_disableOffsX = false;
|
|
|
|
|
|
|
|
_vm = vm;
|
2008-08-02 22:51:53 +00:00
|
|
|
_curId = -1;
|
2005-03-25 01:52:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CharsetRenderer::~CharsetRenderer() {
|
2003-05-20 15:12:33 +00:00
|
|
|
}
|
2004-08-30 11:52:33 +00:00
|
|
|
|
2005-06-04 16:10:39 +00:00
|
|
|
CharsetRendererCommon::CharsetRendererCommon(ScummEngine *vm)
|
2009-09-25 09:13:33 +00:00
|
|
|
: CharsetRenderer(vm), _bytesPerPixel(0), _fontHeight(0), _numChars(0) {
|
2014-07-23 07:16:06 +00:00
|
|
|
_enableShadow = false;
|
2005-06-04 16:10:39 +00:00
|
|
|
_shadowColor = 0;
|
|
|
|
}
|
|
|
|
|
2008-08-02 22:51:53 +00:00
|
|
|
void CharsetRendererCommon::setCurID(int32 id) {
|
|
|
|
if (id == -1)
|
|
|
|
return;
|
|
|
|
|
2006-09-16 13:38:43 +00:00
|
|
|
assertRange(0, id, _vm->_numCharsets - 1, "charset");
|
2002-12-26 00:21:19 +00:00
|
|
|
|
2002-12-25 21:04:47 +00:00
|
|
|
_curId = id;
|
|
|
|
|
2002-12-26 00:21:19 +00:00
|
|
|
_fontPtr = _vm->getResourceAddress(rtCharset, id);
|
2003-05-01 11:03:41 +00:00
|
|
|
if (_fontPtr == 0)
|
2009-05-31 10:02:16 +00:00
|
|
|
error("CharsetRendererCommon::setCurID: charset %d not found", id);
|
2003-05-01 11:03:41 +00:00
|
|
|
|
2006-02-20 16:51:30 +00:00
|
|
|
if (_vm->_game.version == 4)
|
2002-12-26 00:21:19 +00:00
|
|
|
_fontPtr += 17;
|
2002-12-25 21:04:47 +00:00
|
|
|
else
|
2002-12-26 00:21:19 +00:00
|
|
|
_fontPtr += 29;
|
2005-05-26 16:38:02 +00:00
|
|
|
|
2009-09-25 09:13:33 +00:00
|
|
|
_bytesPerPixel = _fontPtr[0];
|
2005-05-26 16:38:02 +00:00
|
|
|
_fontHeight = _fontPtr[1];
|
|
|
|
_numChars = READ_LE_UINT16(_fontPtr + 2);
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2008-08-02 22:51:53 +00:00
|
|
|
void CharsetRendererV3::setCurID(int32 id) {
|
|
|
|
if (id == -1)
|
|
|
|
return;
|
|
|
|
|
2006-09-16 13:38:43 +00:00
|
|
|
assertRange(0, id, _vm->_numCharsets - 1, "charset");
|
2003-05-08 22:44:46 +00:00
|
|
|
|
|
|
|
_curId = id;
|
|
|
|
|
|
|
|
_fontPtr = _vm->getResourceAddress(rtCharset, id);
|
|
|
|
if (_fontPtr == 0)
|
2009-05-31 10:02:16 +00:00
|
|
|
error("CharsetRendererCommon::setCurID: charset %d not found", id);
|
2003-05-08 22:44:46 +00:00
|
|
|
|
2009-09-25 09:13:33 +00:00
|
|
|
_bytesPerPixel = 1;
|
2004-08-23 08:37:55 +00:00
|
|
|
_numChars = _fontPtr[4];
|
2005-05-26 16:38:02 +00:00
|
|
|
_fontHeight = _fontPtr[5];
|
|
|
|
|
2003-05-08 22:44:46 +00:00
|
|
|
_fontPtr += 6;
|
|
|
|
_widthTable = _fontPtr;
|
2004-08-23 08:37:55 +00:00
|
|
|
_fontPtr += _numChars;
|
2003-05-08 22:44:46 +00:00
|
|
|
}
|
|
|
|
|
2004-10-23 23:08:53 +00:00
|
|
|
int CharsetRendererCommon::getFontHeight() {
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_useCJKMode)
|
|
|
|
return MAX(_vm->_2byteHeight + 1, _fontHeight);
|
|
|
|
else
|
2005-05-26 16:38:02 +00:00
|
|
|
return _fontHeight;
|
2004-10-23 23:08:53 +00:00
|
|
|
}
|
|
|
|
|
2002-12-25 21:04:47 +00:00
|
|
|
// do spacing for variable width old-style font
|
2010-10-12 22:17:00 +00:00
|
|
|
int CharsetRendererClassic::getCharWidth(uint16 chr) {
|
2002-12-25 21:04:47 +00:00
|
|
|
int spacing = 0;
|
2002-12-25 21:57:01 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_useCJKMode && chr >= 0x80)
|
|
|
|
return _vm->_2byteWidth / 2;
|
2010-10-12 22:17:00 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
int offs = READ_LE_UINT32(_fontPtr + chr * 4 + 4);
|
|
|
|
if (offs)
|
|
|
|
spacing = _fontPtr[offs] + (signed char)_fontPtr[offs + 2];
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2002-12-25 21:57:01 +00:00
|
|
|
return spacing;
|
|
|
|
}
|
|
|
|
|
2003-05-21 16:28:02 +00:00
|
|
|
int CharsetRenderer::getStringWidth(int arg, const byte *text) {
|
2002-12-25 21:42:22 +00:00
|
|
|
int pos = 0;
|
2002-12-26 00:21:19 +00:00
|
|
|
int width = 1;
|
2010-10-23 00:30:21 +00:00
|
|
|
int chr;
|
2003-06-19 11:13:11 +00:00
|
|
|
int oldID = getCurID();
|
2006-02-20 16:51:30 +00:00
|
|
|
int code = (_vm->_game.heversion >= 80) ? 127 : 64;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
|
|
|
while ((chr = text[pos++]) != 0) {
|
2008-04-28 15:21:36 +00:00
|
|
|
if (chr == '\n' || chr == '\r' || chr == _vm->_newLineCharacter)
|
2005-11-04 23:52:01 +00:00
|
|
|
break;
|
2006-02-20 16:51:30 +00:00
|
|
|
if (_vm->_game.heversion >= 72) {
|
2005-11-04 13:09:57 +00:00
|
|
|
if (chr == code) {
|
|
|
|
chr = text[pos++];
|
2005-11-04 23:52:01 +00:00
|
|
|
if (chr == 84 || chr == 116) { // Strings of speech offset/size
|
2005-11-04 13:09:57 +00:00
|
|
|
while (chr != code)
|
|
|
|
chr = text[pos++];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 119) // 'Wait'
|
|
|
|
break;
|
|
|
|
if (chr == 104|| chr == 110) // 'Newline'
|
2002-12-25 21:04:47 +00:00
|
|
|
break;
|
|
|
|
}
|
2005-11-04 13:09:57 +00:00
|
|
|
} else {
|
2008-05-19 17:34:29 +00:00
|
|
|
if (chr == '@' && !(_vm->_game.id == GID_CMI && _vm->_language == Common::ZH_TWN))
|
2002-12-25 21:04:47 +00:00
|
|
|
continue;
|
2006-02-20 16:51:30 +00:00
|
|
|
if (chr == 255 || (_vm->_game.version <= 6 && chr == 254)) {
|
2005-11-04 13:09:57 +00:00
|
|
|
chr = text[pos++];
|
|
|
|
if (chr == 3) // 'WAIT'
|
|
|
|
break;
|
|
|
|
if (chr == 8) { // 'Verb on next line'
|
|
|
|
if (arg == 1)
|
|
|
|
break;
|
|
|
|
while (text[pos++] == ' ')
|
|
|
|
;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 10 || chr == 21 || chr == 12 || chr == 13) {
|
|
|
|
pos += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 9 || chr == 1 || chr == 2) // 'Newline'
|
|
|
|
break;
|
|
|
|
if (chr == 14) {
|
|
|
|
int set = text[pos] | (text[pos + 1] << 8);
|
|
|
|
pos += 2;
|
|
|
|
setCurID(set);
|
|
|
|
continue;
|
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-07-22 22:59:08 +00:00
|
|
|
// Some localizations may override colors
|
|
|
|
// See credits in Chinese COMI
|
2008-05-19 17:34:29 +00:00
|
|
|
if (_vm->_game.id == GID_CMI && _vm->_language == Common::ZH_TWN &&
|
|
|
|
chr == '^' && pos == 1) {
|
2007-07-22 22:59:08 +00:00
|
|
|
if (text[pos] == 'c') {
|
|
|
|
pos += 4;
|
|
|
|
chr = text[pos++];
|
|
|
|
}
|
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
2010-10-12 22:17:00 +00:00
|
|
|
|
|
|
|
if (_vm->_useCJKMode) {
|
|
|
|
if (_vm->_game.platform == Common::kPlatformFMTowns) {
|
2010-10-17 23:52:33 +00:00
|
|
|
if (checkSJISCode(chr))
|
2010-10-23 00:30:21 +00:00
|
|
|
// This strange character conversion is the exact way the original does it here.
|
|
|
|
// This is the only way to get an accurate text formatting in the MI1 intro.
|
|
|
|
chr = (int8)text[pos++] | (chr << 8);
|
2010-10-12 22:17:00 +00:00
|
|
|
} else if (chr & 0x80) {
|
|
|
|
pos++;
|
|
|
|
width += _vm->_2byteWidth;
|
2014-07-23 08:59:07 +00:00
|
|
|
// Original keeps glyph width and character dimensions separately
|
|
|
|
if (_vm->_language == Common::KO_KOR || _vm->_language == Common::ZH_TWN) {
|
|
|
|
width++;
|
|
|
|
}
|
2010-10-12 22:17:00 +00:00
|
|
|
continue;
|
|
|
|
}
|
2004-10-23 23:08:53 +00:00
|
|
|
}
|
2010-10-12 22:17:00 +00:00
|
|
|
width += getCharWidth(chr);
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2002-12-26 00:21:19 +00:00
|
|
|
setCurID(oldID);
|
|
|
|
|
2002-12-25 21:04:47 +00:00
|
|
|
return width;
|
|
|
|
}
|
|
|
|
|
2003-03-06 17:58:13 +00:00
|
|
|
void CharsetRenderer::addLinebreaks(int a, byte *str, int pos, int maxwidth) {
|
2002-12-25 21:04:47 +00:00
|
|
|
int lastspace = -1;
|
|
|
|
int curw = 1;
|
2010-10-23 00:30:21 +00:00
|
|
|
int chr;
|
2003-06-19 11:13:11 +00:00
|
|
|
int oldID = getCurID();
|
2006-02-20 16:51:30 +00:00
|
|
|
int code = (_vm->_game.heversion >= 80) ? 127 : 64;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
|
|
|
while ((chr = str[pos++]) != 0) {
|
2006-02-20 16:51:30 +00:00
|
|
|
if (_vm->_game.heversion >= 72) {
|
2005-11-04 13:09:57 +00:00
|
|
|
if (chr == code) {
|
|
|
|
chr = str[pos++];
|
2005-11-04 23:52:01 +00:00
|
|
|
if (chr == 84 || chr == 116) { // Strings of speech offset/size
|
2005-11-04 13:09:57 +00:00
|
|
|
while (chr != code)
|
|
|
|
chr = str[pos++];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 119) // 'Wait'
|
|
|
|
break;
|
|
|
|
if (chr == 110) { // 'Newline'
|
2002-12-25 21:04:47 +00:00
|
|
|
curw = 1;
|
2005-11-04 13:09:57 +00:00
|
|
|
continue;
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
2005-11-04 13:09:57 +00:00
|
|
|
if (chr == 104) // 'Don't terminate with \n'
|
|
|
|
break;
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
2005-11-04 13:09:57 +00:00
|
|
|
} else {
|
|
|
|
if (chr == '@')
|
2002-12-25 21:04:47 +00:00
|
|
|
continue;
|
2006-02-20 16:51:30 +00:00
|
|
|
if (chr == 255 || (_vm->_game.version <= 6 && chr == 254)) {
|
2005-11-04 13:09:57 +00:00
|
|
|
chr = str[pos++];
|
|
|
|
if (chr == 3) // 'Wait'
|
|
|
|
break;
|
|
|
|
if (chr == 8) { // 'Verb on next line'
|
|
|
|
if (a == 1) {
|
|
|
|
curw = 1;
|
|
|
|
} else {
|
|
|
|
while (str[pos] == ' ')
|
|
|
|
str[pos++] = '@';
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 10 || chr == 21 || chr == 12 || chr == 13) {
|
|
|
|
pos += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 1) { // 'Newline'
|
|
|
|
curw = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (chr == 2) // 'Don't terminate with \n'
|
|
|
|
break;
|
|
|
|
if (chr == 14) {
|
|
|
|
int set = str[pos] | (str[pos + 1] << 8);
|
|
|
|
pos += 2;
|
|
|
|
setCurID(set);
|
|
|
|
continue;
|
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (chr == ' ')
|
|
|
|
lastspace = pos - 1;
|
|
|
|
|
2008-04-28 15:21:36 +00:00
|
|
|
if (chr == _vm->_newLineCharacter)
|
|
|
|
lastspace = pos - 1;
|
|
|
|
|
2010-10-15 19:10:18 +00:00
|
|
|
if (_vm->_useCJKMode) {
|
|
|
|
if (_vm->_game.platform == Common::kPlatformFMTowns) {
|
2010-10-17 23:52:33 +00:00
|
|
|
if (checkSJISCode(chr))
|
2010-10-23 00:30:21 +00:00
|
|
|
// This strange character conversion is the exact way the original does it here.
|
|
|
|
// This is the only way to get an accurate text formatting in the MI1 intro.
|
|
|
|
chr = (int8)str[pos++] | (chr << 8);
|
2010-10-15 19:10:18 +00:00
|
|
|
curw += getCharWidth(chr);
|
|
|
|
} else if (chr & 0x80) {
|
|
|
|
pos++;
|
|
|
|
curw += _vm->_2byteWidth;
|
2014-07-23 08:59:07 +00:00
|
|
|
// Original keeps glyph width and character dimensions separately
|
|
|
|
if (_vm->_language == Common::KO_KOR || _vm->_language == Common::ZH_TWN) {
|
|
|
|
curw++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
curw += getCharWidth(chr);
|
2010-10-15 19:10:18 +00:00
|
|
|
}
|
2004-10-23 23:08:53 +00:00
|
|
|
} else {
|
|
|
|
curw += getCharWidth(chr);
|
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
if (lastspace == -1)
|
|
|
|
continue;
|
|
|
|
if (curw > maxwidth) {
|
|
|
|
str[lastspace] = 0xD;
|
|
|
|
curw = 1;
|
|
|
|
pos = lastspace + 1;
|
|
|
|
lastspace = -1;
|
|
|
|
}
|
|
|
|
}
|
2002-12-26 00:21:19 +00:00
|
|
|
|
|
|
|
setCurID(oldID);
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2010-10-12 22:17:00 +00:00
|
|
|
int CharsetRendererV3::getCharWidth(uint16 chr) {
|
2003-05-08 22:44:46 +00:00
|
|
|
int spacing = 0;
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_useCJKMode && (chr & 0x80))
|
|
|
|
spacing = _vm->_2byteWidth / 2;
|
2012-09-26 02:17:31 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (!spacing)
|
|
|
|
spacing = *(_widthTable + chr);
|
|
|
|
|
|
|
|
return spacing;
|
|
|
|
}
|
|
|
|
|
2014-07-23 04:43:20 +00:00
|
|
|
void CharsetRendererPC::enableShadow(bool enable) {
|
2011-07-07 14:33:14 +00:00
|
|
|
_shadowColor = 0;
|
2014-07-23 07:16:06 +00:00
|
|
|
_enableShadow = enable;
|
|
|
|
|
|
|
|
if (_vm->_game.version >= 7 && _vm->_language == Common::KO_KOR)
|
|
|
|
_shadowType = kHorizontalShadowType;
|
|
|
|
else
|
|
|
|
_shadowType = kNormalShadowType;
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
|
|
|
|
2014-07-23 04:43:20 +00:00
|
|
|
void CharsetRendererPC::drawBits1(Graphics::Surface &dest, int x, int y, const byte *src, int drawTop, int width, int height) {
|
2011-09-14 17:19:40 +00:00
|
|
|
byte *dst = (byte *)dest.getBasePtr(x, y);
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
byte bits = 0;
|
|
|
|
uint8 col = _color;
|
2011-09-14 17:19:40 +00:00
|
|
|
int pitch = dest.pitch - width * dest.format.bytesPerPixel;
|
|
|
|
byte *dst2 = dst + dest.pitch;
|
2011-07-07 14:33:14 +00:00
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
for (y = 0; y < height && y + drawTop < dest.h; y++) {
|
2011-07-07 14:33:14 +00:00
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
if ((x % 8) == 0)
|
|
|
|
bits = *src++;
|
|
|
|
if ((bits & revBitMask(x % 8)) && y + drawTop >= 0) {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
|
|
|
if (_shadowType == kNormalShadowType)
|
|
|
|
dst[1] = dst2[0] = dst2[1] = _shadowColor;
|
|
|
|
else if (_shadowType == kHorizontalShadowType)
|
|
|
|
dst[1] = _shadowColor;
|
|
|
|
}
|
2011-07-07 14:33:14 +00:00
|
|
|
dst[0] = col;
|
|
|
|
}
|
2011-09-14 17:19:40 +00:00
|
|
|
dst += dest.format.bytesPerPixel;
|
|
|
|
dst2 += dest.format.bytesPerPixel;
|
2011-05-02 12:32:49 +00:00
|
|
|
}
|
2010-10-15 19:10:18 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
dst += pitch;
|
|
|
|
dst2 += pitch;
|
2011-05-02 12:32:49 +00:00
|
|
|
}
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
2003-05-08 22:44:46 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
int CharsetRendererV3::getDrawWidthIntern(uint16 chr) {
|
|
|
|
return getCharWidth(chr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererV3::getDrawHeightIntern(uint16) {
|
|
|
|
return 8;
|
2003-05-08 22:44:46 +00:00
|
|
|
}
|
|
|
|
|
2007-09-10 13:17:20 +00:00
|
|
|
void CharsetRendererV3::setColor(byte color) {
|
2005-06-04 16:10:39 +00:00
|
|
|
bool useShadow = false;
|
2003-04-27 18:49:27 +00:00
|
|
|
_color = color;
|
2005-06-04 16:10:39 +00:00
|
|
|
|
2006-01-23 20:09:25 +00:00
|
|
|
// FM-TOWNS version of Loom uses old color method as well
|
2008-11-06 14:03:38 +00:00
|
|
|
if ((_vm->_game.version >= 2) && ((_vm->_game.features & GF_16COLOR) || (_vm->_game.id == GID_LOOM && _vm->_game.version == 3))) {
|
2005-06-04 16:10:39 +00:00
|
|
|
useShadow = ((_color & 0xF0) != 0);
|
2003-04-27 18:49:27 +00:00
|
|
|
_color &= 0x0f;
|
2006-02-20 16:51:30 +00:00
|
|
|
} else if (_vm->_game.features & GF_OLD256) {
|
2005-06-04 16:10:39 +00:00
|
|
|
useShadow = ((_color & 0x80) != 0);
|
2003-09-14 00:37:47 +00:00
|
|
|
_color &= 0x7f;
|
2003-04-27 18:49:27 +00:00
|
|
|
} else
|
2005-06-04 16:10:39 +00:00
|
|
|
useShadow = false;
|
|
|
|
|
2010-10-05 19:04:52 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
SCUMM/FM-TOWNS: fix palette and other graphics issues
This commit should fix at least the following bugs/feature requests: #1032859, #1252088, #1055391, #1315968, #1315938, #1742106, #812891.
The FM-Towns version of Scumm games use a mixed graphics mode with 2 layers (one with 32767 colors and one with 16 colors). Among other things I have added a screen output class which emulates this dual layer approach which allows specific hardware effects like enabling and disabling layers (e.g. in the voodoo priestess scene in MI1).
Old savegames (saved before this update) will load, but you’ll encounter palette glitches in the verb/inventory screen, since the 16 color palette for layer 2 is not contained in your savegame. This will be true at least for version 5 games. Certain scene change actions (which require the verb/inventory part to be redrawn) might correct this (e.g. try looking at the treasure map in MI1 and closing it). Version 3 games should be okay, since they use a static text palette which is never changed and which will be reset after loading a savegame.
This update requires a USE_RGB_COLORS setting for proper operation. 8 bit users will get a warning that they’ll have to expect palette glitches . Apart from that the engine in 8 bit mode should not only still work okay, but also benefit from some of the other (non palette related) improvements (e.g. bug #1032859 should be fixed even in 8 bit mode).
Japanese font drawing hasn’t been improved much yet. This will be a separate task.
svn-id: r52966
2010-10-01 19:24:52 +00:00
|
|
|
if (_vm->_game.platform == Common::kPlatformFMTowns) {
|
|
|
|
_color = (_color & 0x0f) | ((_color & 0x0f) << 4);
|
|
|
|
if (_color == 0)
|
|
|
|
_color = 0x88;
|
|
|
|
}
|
2010-10-05 19:04:52 +00:00
|
|
|
#endif
|
SCUMM/FM-TOWNS: fix palette and other graphics issues
This commit should fix at least the following bugs/feature requests: #1032859, #1252088, #1055391, #1315968, #1315938, #1742106, #812891.
The FM-Towns version of Scumm games use a mixed graphics mode with 2 layers (one with 32767 colors and one with 16 colors). Among other things I have added a screen output class which emulates this dual layer approach which allows specific hardware effects like enabling and disabling layers (e.g. in the voodoo priestess scene in MI1).
Old savegames (saved before this update) will load, but you’ll encounter palette glitches in the verb/inventory screen, since the 16 color palette for layer 2 is not contained in your savegame. This will be true at least for version 5 games. Certain scene change actions (which require the verb/inventory part to be redrawn) might correct this (e.g. try looking at the treasure map in MI1 and closing it). Version 3 games should be okay, since they use a static text palette which is never changed and which will be reset after loading a savegame.
This update requires a USE_RGB_COLORS setting for proper operation. 8 bit users will get a warning that they’ll have to expect palette glitches . Apart from that the engine in 8 bit mode should not only still work okay, but also benefit from some of the other (non palette related) improvements (e.g. bug #1032859 should be fixed even in 8 bit mode).
Japanese font drawing hasn’t been improved much yet. This will be a separate task.
svn-id: r52966
2010-10-01 19:24:52 +00:00
|
|
|
|
2005-06-04 16:10:39 +00:00
|
|
|
enableShadow(useShadow);
|
2005-02-20 00:17:22 +00:00
|
|
|
|
|
|
|
translateColor();
|
2003-04-27 18:49:27 +00:00
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2009-11-22 11:43:12 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2009-11-22 08:36:14 +00:00
|
|
|
void CharsetRendererPCE::setColor(byte color) {
|
|
|
|
_vm->setPCETextPalette(color);
|
|
|
|
_color = 15;
|
|
|
|
|
|
|
|
enableShadow(true);
|
|
|
|
}
|
2009-11-22 11:43:12 +00:00
|
|
|
#endif
|
2009-11-22 08:36:14 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
void CharsetRendererV3::printChar(int chr, bool ignoreCharsetMask) {
|
2007-12-16 21:46:34 +00:00
|
|
|
// WORKAROUND for bug #1509509: Indy3 Mac does not show black
|
|
|
|
// characters (such as in the grail diary) if ignoreCharsetMask
|
|
|
|
// is true. See also patch #1851568.
|
|
|
|
if (_vm->_game.id == GID_INDY3 && _vm->_game.platform == Common::kPlatformMacintosh && _color == 0)
|
|
|
|
ignoreCharsetMask = false;
|
|
|
|
|
2003-04-28 17:17:39 +00:00
|
|
|
// Indy3 / Zak256 / Loom
|
2006-05-09 14:16:43 +00:00
|
|
|
int width, height, origWidth = 0, origHeight;
|
2002-12-25 21:04:47 +00:00
|
|
|
VirtScreen *vs;
|
2006-10-15 02:00:48 +00:00
|
|
|
const byte *charPtr;
|
2010-10-15 19:10:18 +00:00
|
|
|
int is2byte = (chr >= 256 && _vm->_useCJKMode) ? 1 : 0;
|
2003-04-27 18:30:35 +00:00
|
|
|
|
2006-09-16 13:38:43 +00:00
|
|
|
assertRange(0, _curId, _vm->_numCharsets - 1, "charset");
|
2002-12-25 21:04:47 +00:00
|
|
|
|
|
|
|
if ((vs = _vm->findVirtScreen(_top)) == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (chr == '@')
|
|
|
|
return;
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
charPtr = (_vm->_useCJKMode && chr > 127) ? _vm->get2byteCharPtr(chr) : _fontPtr + chr * 8;
|
|
|
|
width = getDrawWidthIntern(chr);
|
|
|
|
height = getDrawHeightIntern(chr);
|
|
|
|
setDrawCharIntern(chr);
|
2004-10-23 23:08:53 +00:00
|
|
|
|
2011-07-09 18:06:18 +00:00
|
|
|
origWidth = width;
|
|
|
|
origHeight = height;
|
|
|
|
|
2006-05-07 11:45:55 +00:00
|
|
|
// Clip at the right side (to avoid drawing "outside" the screen bounds).
|
|
|
|
if (_left + origWidth > _right + 1)
|
|
|
|
return;
|
|
|
|
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
2010-10-17 13:08:00 +00:00
|
|
|
width++;
|
|
|
|
height++;
|
2004-10-23 23:08:53 +00:00
|
|
|
}
|
|
|
|
|
2002-12-25 21:04:47 +00:00
|
|
|
if (_firstChar) {
|
2003-05-15 22:30:32 +00:00
|
|
|
_str.left = _left;
|
|
|
|
_str.top = _top;
|
|
|
|
_str.right = _left;
|
|
|
|
_str.bottom = _top;
|
2002-12-25 21:04:47 +00:00
|
|
|
_firstChar = false;
|
|
|
|
}
|
2003-04-07 14:38:26 +00:00
|
|
|
|
2004-10-23 23:08:53 +00:00
|
|
|
int drawTop = _top - vs->topline;
|
2003-04-28 17:17:39 +00:00
|
|
|
|
2004-04-06 22:02:02 +00:00
|
|
|
_vm->markRectAsDirty(vs->number, _left, _left + width, drawTop, drawTop + height);
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
if (!ignoreCharsetMask) {
|
2003-04-28 17:17:39 +00:00
|
|
|
_hasMask = true;
|
2004-04-08 23:41:10 +00:00
|
|
|
_textScreenID = vs->number;
|
|
|
|
}
|
SCUMM/FM-TOWNS: fix palette and other graphics issues
This commit should fix at least the following bugs/feature requests: #1032859, #1252088, #1055391, #1315968, #1315938, #1742106, #812891.
The FM-Towns version of Scumm games use a mixed graphics mode with 2 layers (one with 32767 colors and one with 16 colors). Among other things I have added a screen output class which emulates this dual layer approach which allows specific hardware effects like enabling and disabling layers (e.g. in the voodoo priestess scene in MI1).
Old savegames (saved before this update) will load, but you’ll encounter palette glitches in the verb/inventory screen, since the 16 color palette for layer 2 is not contained in your savegame. This will be true at least for version 5 games. Certain scene change actions (which require the verb/inventory part to be redrawn) might correct this (e.g. try looking at the treasure map in MI1 and closing it). Version 3 games should be okay, since they use a static text palette which is never changed and which will be reset after loading a savegame.
This update requires a USE_RGB_COLORS setting for proper operation. 8 bit users will get a warning that they’ll have to expect palette glitches . Apart from that the engine in 8 bit mode should not only still work okay, but also benefit from some of the other (non palette related) improvements (e.g. bug #1032859 should be fixed even in 8 bit mode).
Japanese font drawing hasn’t been improved much yet. This will be a separate task.
svn-id: r52966
2010-10-01 19:24:52 +00:00
|
|
|
|
2011-09-17 20:15:29 +00:00
|
|
|
if ((ignoreCharsetMask || !vs->hasTwoBuffers)
|
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
&& (_vm->_game.platform != Common::kPlatformFMTowns)
|
|
|
|
#endif
|
|
|
|
)
|
2011-09-14 17:19:40 +00:00
|
|
|
drawBits1(*vs, _left + vs->xstart, drawTop, charPtr, drawTop, origWidth, origHeight);
|
|
|
|
else
|
|
|
|
drawBits1(_vm->_textSurface, _left * _vm->_textSurfaceMultiplier, _top * _vm->_textSurfaceMultiplier, charPtr, drawTop, origWidth, origHeight);
|
2012-09-26 02:17:31 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (is2byte) {
|
|
|
|
origWidth /= _vm->_textSurfaceMultiplier;
|
|
|
|
height /= _vm->_textSurfaceMultiplier;
|
2004-08-08 22:10:38 +00:00
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2003-05-15 22:30:32 +00:00
|
|
|
if (_str.left > _left)
|
|
|
|
_str.left = _left;
|
2003-04-27 18:30:35 +00:00
|
|
|
|
2010-10-15 19:10:18 +00:00
|
|
|
_left += origWidth;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2003-05-15 22:30:32 +00:00
|
|
|
if (_str.right < _left) {
|
|
|
|
_str.right = _left;
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow)
|
2003-05-15 22:30:32 +00:00
|
|
|
_str.right++;
|
2003-04-27 18:30:35 +00:00
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2011-06-19 12:46:02 +00:00
|
|
|
if (_str.bottom < _top + height)
|
|
|
|
_str.bottom = _top + height;
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2010-11-05 00:36:23 +00:00
|
|
|
void CharsetRendererV3::drawChar(int chr, Graphics::Surface &s, int x, int y) {
|
2011-07-07 14:33:14 +00:00
|
|
|
const byte *charPtr = (_vm->_useCJKMode && chr > 127) ? _vm->get2byteCharPtr(chr) : _fontPtr + chr * 8;
|
|
|
|
int width = getDrawWidthIntern(chr);
|
|
|
|
int height = getDrawHeightIntern(chr);
|
|
|
|
setDrawCharIntern(chr);
|
2011-09-14 17:19:40 +00:00
|
|
|
drawBits1(s, x, y, charPtr, y, width, height);
|
2004-08-23 08:37:55 +00:00
|
|
|
}
|
|
|
|
|
2005-02-20 00:17:22 +00:00
|
|
|
void CharsetRenderer::translateColor() {
|
|
|
|
// Based on disassembly
|
|
|
|
if (_vm->_renderMode == Common::kRenderCGA) {
|
2006-10-15 02:00:48 +00:00
|
|
|
static const byte CGAtextColorMap[16] = {0, 3, 3, 3, 5, 5, 5, 15,
|
2005-02-20 00:17:22 +00:00
|
|
|
15, 3, 3, 3, 5, 5, 15, 15};
|
|
|
|
_color = CGAtextColorMap[_color & 0x0f];
|
|
|
|
}
|
|
|
|
|
2005-03-02 21:46:51 +00:00
|
|
|
if (_vm->_renderMode == Common::kRenderHercA || _vm->_renderMode == Common::kRenderHercG) {
|
2006-10-15 02:00:48 +00:00
|
|
|
static const byte HercTextColorMap[16] = {0, 15, 2, 15, 15, 5, 15, 15,
|
2005-02-20 00:17:22 +00:00
|
|
|
8, 15, 15, 15, 15, 15, 15, 15};
|
|
|
|
_color = HercTextColorMap[_color & 0x0f];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-29 06:06:12 +00:00
|
|
|
void CharsetRenderer::saveLoadWithSerializer(Common::Serializer &ser) {
|
|
|
|
ser.syncAsByte(_curId, VER(73), VER(73));
|
|
|
|
ser.syncAsSint32LE(_curId, VER(74));
|
|
|
|
ser.syncAsByte(_color, VER(73));
|
2007-05-28 08:02:10 +00:00
|
|
|
|
2017-11-29 06:06:12 +00:00
|
|
|
if (ser.isLoading()) {
|
2007-05-28 08:02:10 +00:00
|
|
|
setCurID(_curId);
|
|
|
|
setColor(_color);
|
|
|
|
}
|
|
|
|
}
|
2004-08-23 08:37:55 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
void CharsetRendererClassic::printChar(int chr, bool ignoreCharsetMask) {
|
2002-12-25 21:04:47 +00:00
|
|
|
VirtScreen *vs;
|
2010-10-15 19:10:18 +00:00
|
|
|
bool is2byte = (chr >= 256 && _vm->_useCJKMode);
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2006-09-16 13:38:43 +00:00
|
|
|
assertRange(1, _curId, _vm->_numCharsets - 1, "charset");
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2003-06-08 15:17:14 +00:00
|
|
|
if ((vs = _vm->findVirtScreen(_top)) == NULL && (vs = _vm->findVirtScreen(_top + getFontHeight())) == NULL)
|
2002-12-25 21:04:47 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (chr == '@')
|
|
|
|
return;
|
|
|
|
|
2005-02-20 00:17:22 +00:00
|
|
|
translateColor();
|
|
|
|
|
2002-12-26 00:21:19 +00:00
|
|
|
_vm->_charsetColorMap[1] = _color;
|
2003-11-08 21:59:32 +00:00
|
|
|
|
2011-07-13 22:50:14 +00:00
|
|
|
if (!prepareDraw(chr))
|
|
|
|
return;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2002-12-25 21:04:47 +00:00
|
|
|
if (_firstChar) {
|
2003-05-15 22:30:32 +00:00
|
|
|
_str.left = 0;
|
|
|
|
_str.top = 0;
|
|
|
|
_str.right = 0;
|
|
|
|
_str.bottom = 0;
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_top += _offsY;
|
|
|
|
_left += _offsX;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_left + _origWidth > _right + 1 || _left < 0) {
|
|
|
|
_left += _origWidth;
|
|
|
|
_top -= _offsY;
|
2002-12-25 21:04:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_disableOffsX = false;
|
|
|
|
|
|
|
|
if (_firstChar) {
|
2003-05-15 22:30:32 +00:00
|
|
|
_str.left = _left;
|
|
|
|
_str.top = _top;
|
|
|
|
_str.right = _left;
|
|
|
|
_str.bottom = _top;
|
2002-12-25 21:04:47 +00:00
|
|
|
_firstChar = false;
|
|
|
|
}
|
|
|
|
|
2003-05-15 22:30:32 +00:00
|
|
|
if (_left < _str.left)
|
|
|
|
_str.left = _left;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
2003-05-15 22:30:32 +00:00
|
|
|
if (_top < _str.top)
|
|
|
|
_str.top = _top;
|
2002-12-25 21:04:47 +00:00
|
|
|
|
|
|
|
int drawTop = _top - vs->topline;
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_vm->markRectAsDirty(vs->number, _left, _left + _width, drawTop, drawTop + _height);
|
2002-12-25 21:04:47 +00:00
|
|
|
|
SCUMM/FM-TOWNS: fix palette and other graphics issues
This commit should fix at least the following bugs/feature requests: #1032859, #1252088, #1055391, #1315968, #1315938, #1742106, #812891.
The FM-Towns version of Scumm games use a mixed graphics mode with 2 layers (one with 32767 colors and one with 16 colors). Among other things I have added a screen output class which emulates this dual layer approach which allows specific hardware effects like enabling and disabling layers (e.g. in the voodoo priestess scene in MI1).
Old savegames (saved before this update) will load, but you’ll encounter palette glitches in the verb/inventory screen, since the 16 color palette for layer 2 is not contained in your savegame. This will be true at least for version 5 games. Certain scene change actions (which require the verb/inventory part to be redrawn) might correct this (e.g. try looking at the treasure map in MI1 and closing it). Version 3 games should be okay, since they use a static text palette which is never changed and which will be reset after loading a savegame.
This update requires a USE_RGB_COLORS setting for proper operation. 8 bit users will get a warning that they’ll have to expect palette glitches . Apart from that the engine in 8 bit mode should not only still work okay, but also benefit from some of the other (non palette related) improvements (e.g. bug #1032859 should be fixed even in 8 bit mode).
Japanese font drawing hasn’t been improved much yet. This will be a separate task.
svn-id: r52966
2010-10-01 19:24:52 +00:00
|
|
|
// This check for kPlatformFMTowns and kMainVirtScreen is at least required for the chat with
|
|
|
|
// the navigator's head in front of the ghost ship in Monkey Island 1
|
2011-07-07 14:33:14 +00:00
|
|
|
if (!ignoreCharsetMask || (_vm->_game.platform == Common::kPlatformFMTowns && vs->number == kMainVirtScreen)) {
|
2002-12-25 21:04:47 +00:00
|
|
|
_hasMask = true;
|
2004-04-08 23:41:10 +00:00
|
|
|
_textScreenID = vs->number;
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2011-08-26 03:51:08 +00:00
|
|
|
// We need to know the virtual screen we draw on for Indy 4 Amiga, since
|
|
|
|
// it selects the palette map according to this. We furthermore can not
|
|
|
|
// use _textScreenID here, since that will cause inventory graphics
|
|
|
|
// glitches.
|
|
|
|
if (_vm->_game.platform == Common::kPlatformAmiga && _vm->_game.id == GID_INDY4)
|
|
|
|
_drawScreen = vs->number;
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
printCharIntern(is2byte, _charPtr, _origWidth, _origHeight, _width, _height, vs, ignoreCharsetMask);
|
2007-01-28 20:11:31 +00:00
|
|
|
|
2014-07-23 08:59:07 +00:00
|
|
|
// Original keeps glyph width and character dimensions separately
|
|
|
|
if ((_vm->_language == Common::ZH_TWN || _vm->_language == Common::KO_KOR) && is2byte)
|
|
|
|
_origWidth++;
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_left += _origWidth;
|
2007-01-28 20:11:31 +00:00
|
|
|
|
|
|
|
if (_str.right < _left) {
|
|
|
|
_str.right = _left;
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_vm->_game.platform != Common::kPlatformFMTowns && _enableShadow)
|
2007-01-28 20:11:31 +00:00
|
|
|
_str.right++;
|
|
|
|
}
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_str.bottom < _top + _origHeight)
|
|
|
|
_str.bottom = _top + _origHeight;
|
2007-01-28 20:11:31 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_top -= _offsY;
|
2007-01-28 20:11:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererClassic::printCharIntern(bool is2byte, const byte *charPtr, int origWidth, int origHeight, int width, int height, VirtScreen *vs, bool ignoreCharsetMask) {
|
|
|
|
byte *dstPtr;
|
|
|
|
byte *back = NULL;
|
|
|
|
int drawTop = _top - vs->topline;
|
|
|
|
|
2009-09-25 09:13:33 +00:00
|
|
|
if ((_vm->_game.heversion >= 71 && _bytesPerPixel >= 8) || (_vm->_game.heversion >= 90 && _bytesPerPixel == 0)) {
|
2008-05-06 03:00:26 +00:00
|
|
|
#ifdef ENABLE_HE
|
2006-01-10 00:28:10 +00:00
|
|
|
if (ignoreCharsetMask || !vs->hasTwoBuffers) {
|
2005-04-23 14:32:40 +00:00
|
|
|
dstPtr = vs->getPixels(0, 0);
|
|
|
|
} else {
|
2013-08-03 00:36:43 +00:00
|
|
|
dstPtr = (byte *)_vm->_textSurface.getPixels();
|
2004-10-06 01:01:44 +00:00
|
|
|
}
|
2004-10-05 23:37:59 +00:00
|
|
|
|
2005-04-23 14:32:40 +00:00
|
|
|
if (_blitAlso && vs->hasTwoBuffers) {
|
|
|
|
dstPtr = vs->getBackPixels(0, 0);
|
2004-10-06 01:01:44 +00:00
|
|
|
}
|
2004-10-05 23:37:59 +00:00
|
|
|
|
2005-04-23 14:32:40 +00:00
|
|
|
Common::Rect rScreen(vs->w, vs->h);
|
2009-09-25 09:13:33 +00:00
|
|
|
if (_bytesPerPixel >= 8) {
|
2005-04-23 14:32:40 +00:00
|
|
|
byte imagePalette[256];
|
|
|
|
memset(imagePalette, 0, sizeof(imagePalette));
|
2005-11-06 11:30:17 +00:00
|
|
|
memcpy(imagePalette, _vm->_charsetColorMap, 4);
|
2009-09-25 09:13:33 +00:00
|
|
|
Wiz::copyWizImage(dstPtr, charPtr, vs->pitch, kDstScreen, vs->w, vs->h, _left, _top, origWidth, origHeight, &rScreen, 0, imagePalette, NULL, _vm->_bytesPerPixel);
|
2005-04-23 14:32:40 +00:00
|
|
|
} else {
|
2009-09-25 09:13:33 +00:00
|
|
|
Wiz::copyWizImage(dstPtr, charPtr, vs->pitch, kDstScreen, vs->w, vs->h, _left, _top, origWidth, origHeight, &rScreen, 0, NULL, NULL, _vm->_bytesPerPixel);
|
2005-04-23 14:32:40 +00:00
|
|
|
}
|
2004-10-05 05:52:48 +00:00
|
|
|
|
2005-04-23 14:32:40 +00:00
|
|
|
if (_blitAlso && vs->hasTwoBuffers) {
|
|
|
|
Common::Rect dst(_left, _top, _left + origWidth, _top + origHeight);
|
2006-09-17 23:35:09 +00:00
|
|
|
((ScummEngine_v71he *)_vm)->restoreBackgroundHE(dst);
|
2005-04-23 14:32:40 +00:00
|
|
|
}
|
2005-05-14 14:06:37 +00:00
|
|
|
#endif
|
2005-04-23 14:32:40 +00:00
|
|
|
} else {
|
|
|
|
Graphics::Surface dstSurface;
|
|
|
|
Graphics::Surface backSurface;
|
2011-07-07 14:33:14 +00:00
|
|
|
if ((ignoreCharsetMask || !vs->hasTwoBuffers)) {
|
2005-04-23 14:32:40 +00:00
|
|
|
dstSurface = *vs;
|
|
|
|
dstPtr = vs->getPixels(_left, drawTop);
|
|
|
|
} else {
|
2007-01-28 20:11:31 +00:00
|
|
|
dstSurface = _vm->_textSurface;
|
2013-08-02 20:23:00 +00:00
|
|
|
dstPtr = (byte *)_vm->_textSurface.getBasePtr(_left * _vm->_textSurfaceMultiplier, (_top - _vm->_screenTop) * _vm->_textSurfaceMultiplier);
|
2005-04-23 14:32:40 +00:00
|
|
|
}
|
2004-10-05 05:52:48 +00:00
|
|
|
|
2005-04-23 14:32:40 +00:00
|
|
|
if (_blitAlso && vs->hasTwoBuffers) {
|
|
|
|
backSurface = dstSurface;
|
|
|
|
back = dstPtr;
|
|
|
|
dstSurface = *vs;
|
|
|
|
dstPtr = vs->getBackPixels(_left, drawTop);
|
|
|
|
}
|
2004-10-05 05:52:48 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
if (!ignoreCharsetMask && vs->hasTwoBuffers) {
|
2005-04-23 14:32:40 +00:00
|
|
|
drawTop = _top - _vm->_screenTop;
|
|
|
|
}
|
2004-10-05 05:52:48 +00:00
|
|
|
|
2014-07-23 04:43:20 +00:00
|
|
|
if (is2byte && _vm->_game.platform != Common::kPlatformFMTowns)
|
|
|
|
drawBits1(dstSurface, _left, drawTop, charPtr, drawTop, origWidth, origHeight);
|
|
|
|
else
|
|
|
|
drawBitsN(dstSurface, dstPtr, charPtr, *_fontPtr, drawTop, origWidth, origHeight);
|
2004-10-05 05:52:48 +00:00
|
|
|
|
|
|
|
if (_blitAlso && vs->hasTwoBuffers) {
|
|
|
|
// FIXME: Revisiting this code, I think the _blitAlso mode is likely broken
|
|
|
|
// right now -- we are copying stuff from "dstPtr" to "back", but "dstPtr" really
|
2005-07-30 21:11:48 +00:00
|
|
|
// only conatains charset data...
|
2004-10-05 05:52:48 +00:00
|
|
|
// One way to fix this: don't copy etc.; rather simply render the char twice,
|
|
|
|
// once to each of the two buffers. That should hypothetically yield
|
|
|
|
// identical results, though I didn't try it and right now I don't know
|
|
|
|
// any spots where I can test this...
|
2006-01-10 00:28:10 +00:00
|
|
|
if (!ignoreCharsetMask)
|
2006-05-07 11:58:01 +00:00
|
|
|
error("This might be broken -- please report where you encountered this to Fingolfin");
|
2004-10-05 05:52:48 +00:00
|
|
|
|
|
|
|
// Perform some clipping
|
|
|
|
int w = MIN(width, dstSurface.w - _left);
|
|
|
|
int h = MIN(height, dstSurface.h - drawTop);
|
|
|
|
if (_left < 0) {
|
|
|
|
w += _left;
|
|
|
|
back -= _left;
|
|
|
|
dstPtr -= _left;
|
|
|
|
}
|
|
|
|
if (drawTop < 0) {
|
|
|
|
h += drawTop;
|
|
|
|
back -= drawTop * backSurface.pitch;
|
|
|
|
dstPtr -= drawTop * dstSurface.pitch;
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-10-05 05:52:48 +00:00
|
|
|
// Blit the image data
|
|
|
|
if (w > 0) {
|
|
|
|
while (h-- > 0) {
|
|
|
|
memcpy(back, dstPtr, w);
|
|
|
|
back += backSurface.pitch;
|
|
|
|
dstPtr += dstSurface.pitch;
|
|
|
|
}
|
2004-09-29 00:04:38 +00:00
|
|
|
}
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
|
|
|
|
2011-07-13 22:50:14 +00:00
|
|
|
bool CharsetRendererClassic::prepareDraw(uint16 chr) {
|
2014-07-23 04:43:20 +00:00
|
|
|
bool is2byte = (chr >= 256 && _vm->_useCJKMode);
|
|
|
|
if (is2byte) {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_vm->_language == Common::KO_KOR)
|
|
|
|
enableShadow(true);
|
2014-07-23 04:43:20 +00:00
|
|
|
|
|
|
|
_charPtr = _vm->get2byteCharPtr(chr);
|
|
|
|
_width = _origWidth = _vm->_2byteWidth;
|
|
|
|
_height = _origHeight = _vm->_2byteHeight;
|
|
|
|
_offsX = _offsY = 0;
|
|
|
|
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
2014-07-23 04:43:20 +00:00
|
|
|
_width++;
|
|
|
|
_height++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
uint32 charOffs = READ_LE_UINT32(_fontPtr + chr * 4 + 4);
|
|
|
|
assert(charOffs < 0x14000);
|
|
|
|
if (!charOffs)
|
2011-07-13 22:50:14 +00:00
|
|
|
return false;
|
2011-07-07 14:33:14 +00:00
|
|
|
_charPtr = _fontPtr + charOffs;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_width = _origWidth = _charPtr[0];
|
|
|
|
_height = _origHeight = _charPtr[1];
|
2004-08-23 08:37:55 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_disableOffsX) {
|
|
|
|
_offsX = 0;
|
|
|
|
} else {
|
|
|
|
_offsX = (signed char)_charPtr[2];
|
2004-10-23 23:08:53 +00:00
|
|
|
}
|
2004-08-23 08:37:55 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_offsY = (signed char)_charPtr[3];
|
2004-10-23 23:08:53 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
_charPtr += 4; // Skip over char header
|
2011-07-13 22:50:14 +00:00
|
|
|
return true;
|
2004-08-23 08:37:55 +00:00
|
|
|
}
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
void CharsetRendererClassic::drawChar(int chr, Graphics::Surface &s, int x, int y) {
|
2011-07-13 22:50:14 +00:00
|
|
|
if (!prepareDraw(chr))
|
|
|
|
return;
|
2014-07-23 04:43:20 +00:00
|
|
|
|
2013-08-02 20:23:00 +00:00
|
|
|
byte *dst = (byte *)s.getBasePtr(x, y);
|
2014-07-23 04:43:20 +00:00
|
|
|
|
|
|
|
bool is2byte = (_vm->_useCJKMode && chr >= 256);
|
|
|
|
if (is2byte)
|
|
|
|
drawBits1(s, x, y, _charPtr, y, _width, _height);
|
|
|
|
else
|
|
|
|
drawBitsN(s, dst, _charPtr, *_fontPtr, y, _width, _height);
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
2010-10-15 21:26:05 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
void CharsetRendererClassic::drawBitsN(const Graphics::Surface &s, byte *dst, const byte *src, byte bpp, int drawTop, int width, int height) {
|
2002-12-25 21:04:47 +00:00
|
|
|
int y, x;
|
|
|
|
int color;
|
|
|
|
byte numbits, bits;
|
|
|
|
|
2010-10-15 21:26:05 +00:00
|
|
|
int pitch = s.pitch - width;
|
|
|
|
|
2003-06-19 11:13:11 +00:00
|
|
|
assert(bpp == 1 || bpp == 2 || bpp == 4 || bpp == 8);
|
2003-06-04 14:37:43 +00:00
|
|
|
bits = *src++;
|
2002-12-25 21:04:47 +00:00
|
|
|
numbits = 8;
|
2010-10-15 21:26:05 +00:00
|
|
|
byte *cmap = _vm->_charsetColorMap;
|
|
|
|
|
2011-08-26 03:51:08 +00:00
|
|
|
// Indy4 Amiga always uses the room or verb palette map to match colors to
|
|
|
|
// the currently setup palette, thus we need to select it over here too.
|
|
|
|
// Done like the original interpreter.
|
|
|
|
byte *amigaMap = 0;
|
|
|
|
if (_vm->_game.platform == Common::kPlatformAmiga && _vm->_game.id == GID_INDY4) {
|
|
|
|
if (_drawScreen == kVerbVirtScreen)
|
|
|
|
amigaMap = _vm->_verbPalette;
|
|
|
|
else
|
|
|
|
amigaMap = _vm->_roomPalette;
|
|
|
|
}
|
|
|
|
|
2004-08-23 08:37:55 +00:00
|
|
|
for (y = 0; y < height && y + drawTop < s.h; y++) {
|
2002-12-25 21:04:47 +00:00
|
|
|
for (x = 0; x < width; x++) {
|
2003-06-04 14:37:43 +00:00
|
|
|
color = (bits >> (8 - bpp)) & 0xFF;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2011-08-26 03:51:08 +00:00
|
|
|
if (color && y + drawTop >= 0) {
|
|
|
|
if (amigaMap)
|
|
|
|
*dst = amigaMap[cmap[color]];
|
|
|
|
else
|
|
|
|
*dst = cmap[color];
|
|
|
|
}
|
2002-12-25 21:04:47 +00:00
|
|
|
dst++;
|
2003-06-04 14:37:43 +00:00
|
|
|
bits <<= bpp;
|
|
|
|
numbits -= bpp;
|
2002-12-25 21:04:47 +00:00
|
|
|
if (numbits == 0) {
|
2003-06-04 14:37:43 +00:00
|
|
|
bits = *src++;
|
2002-12-25 21:04:47 +00:00
|
|
|
numbits = 8;
|
|
|
|
}
|
|
|
|
}
|
2010-10-15 21:26:05 +00:00
|
|
|
dst += pitch;
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CharsetRendererTownsV3::CharsetRendererTownsV3(ScummEngine *vm) : CharsetRendererV3(vm), _sjisCurChar(0) {
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererTownsV3::getCharWidth(uint16 chr) {
|
|
|
|
int spacing = 0;
|
|
|
|
|
|
|
|
if (_vm->_useCJKMode) {
|
|
|
|
if (chr >= 256)
|
|
|
|
spacing = 8;
|
|
|
|
else if (chr >= 128)
|
|
|
|
spacing = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!spacing)
|
|
|
|
spacing = *(_widthTable + chr);
|
|
|
|
|
|
|
|
return spacing;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererTownsV3::getFontHeight() {
|
|
|
|
return _vm->_useCJKMode ? 8 : _fontHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererTownsV3::enableShadow(bool enable) {
|
|
|
|
_shadowColor = 8;
|
2014-07-23 07:16:06 +00:00
|
|
|
_enableShadow = enable;
|
2011-07-07 14:33:14 +00:00
|
|
|
|
2010-10-15 21:26:05 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
2011-07-07 14:33:14 +00:00
|
|
|
_shadowColor = 0x88;
|
2011-07-09 21:57:25 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_cjkFont)
|
|
|
|
_vm->_cjkFont->setDrawingMode(enable ? Graphics::FontSJIS::kFMTownsShadowMode : Graphics::FontSJIS::kDefaultMode);
|
2010-10-15 21:26:05 +00:00
|
|
|
#endif
|
2011-07-09 21:57:25 +00:00
|
|
|
#endif
|
2002-12-25 21:04:47 +00:00
|
|
|
}
|
2002-12-26 01:47:40 +00:00
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
void CharsetRendererTownsV3::drawBits1(Graphics::Surface &dest, int x, int y, const byte *src, int drawTop, int width, int height) {
|
2010-10-15 19:10:18 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
2011-07-09 21:57:25 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_sjisCurChar) {
|
|
|
|
assert(_vm->_cjkFont);
|
2011-09-14 17:19:40 +00:00
|
|
|
_vm->_cjkFont->drawChar(dest, _sjisCurChar, x, y, _color, _shadowColor);
|
2011-07-07 14:33:14 +00:00
|
|
|
return;
|
|
|
|
}
|
2011-07-09 21:57:25 +00:00
|
|
|
#endif
|
2011-09-14 17:19:40 +00:00
|
|
|
bool scale2x = ((&dest == &_vm->_textSurface) && (_vm->_textSurfaceMultiplier == 2) && !(_sjisCurChar >= 256 && _vm->_useCJKMode));
|
2010-10-15 19:10:18 +00:00
|
|
|
#endif
|
|
|
|
|
2003-06-04 14:37:43 +00:00
|
|
|
byte bits = 0;
|
2010-10-15 19:10:18 +00:00
|
|
|
uint8 col = _color;
|
2011-09-14 17:19:40 +00:00
|
|
|
int pitch = dest.pitch - width * dest.format.bytesPerPixel;
|
2011-09-17 20:15:29 +00:00
|
|
|
byte *dst = (byte *)dest.getBasePtr(x, y);
|
2011-09-14 17:19:40 +00:00
|
|
|
byte *dst2 = dst + dest.pitch;
|
2010-10-15 19:10:18 +00:00
|
|
|
|
2010-10-05 19:04:52 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
2010-10-15 19:10:18 +00:00
|
|
|
byte *dst3 = dst2;
|
|
|
|
byte *dst4 = dst2;
|
2011-05-02 12:32:49 +00:00
|
|
|
if (scale2x) {
|
2011-09-14 17:19:40 +00:00
|
|
|
dst3 = dst2 + dest.pitch;
|
|
|
|
dst4 = dst3 + dest.pitch;
|
2010-10-15 19:10:18 +00:00
|
|
|
pitch <<= 1;
|
|
|
|
}
|
2011-05-02 12:32:49 +00:00
|
|
|
#endif
|
2003-06-04 14:37:43 +00:00
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
for (y = 0; y < height && y + drawTop < dest.h; y++) {
|
2003-06-04 14:37:43 +00:00
|
|
|
for (x = 0; x < width; x++) {
|
2003-06-07 00:49:36 +00:00
|
|
|
if ((x % 8) == 0)
|
2003-06-04 14:37:43 +00:00
|
|
|
bits = *src++;
|
2005-05-15 10:40:28 +00:00
|
|
|
if ((bits & revBitMask(x % 8)) && y + drawTop >= 0) {
|
2011-09-14 17:19:40 +00:00
|
|
|
if (dest.format.bytesPerPixel == 2) {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
2009-10-26 14:49:01 +00:00
|
|
|
WRITE_UINT16(dst + 2, _vm->_16BitPalette[_shadowColor]);
|
2011-09-14 17:19:40 +00:00
|
|
|
WRITE_UINT16(dst + dest.pitch, _vm->_16BitPalette[_shadowColor]);
|
2009-10-26 13:22:05 +00:00
|
|
|
}
|
2009-10-26 14:49:01 +00:00
|
|
|
WRITE_UINT16(dst, _vm->_16BitPalette[_color]);
|
2009-10-26 13:22:05 +00:00
|
|
|
} else {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
2010-10-15 19:10:18 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
if (scale2x) {
|
|
|
|
dst[2] = dst[3] = dst2[2] = dst2[3] = _shadowColor;
|
|
|
|
dst3[0] = dst4[0] = dst3[1] = dst4[1] = _shadowColor;
|
2011-05-02 12:32:49 +00:00
|
|
|
} else
|
2010-10-15 19:10:18 +00:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
dst[1] = dst2[0] = _shadowColor;
|
2011-05-02 12:32:49 +00:00
|
|
|
}
|
2009-10-26 13:22:05 +00:00
|
|
|
}
|
2010-10-15 19:10:18 +00:00
|
|
|
dst[0] = col;
|
|
|
|
|
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
if (scale2x)
|
2011-05-02 12:32:49 +00:00
|
|
|
dst[1] = dst2[0] = dst2[1] = col;
|
2010-10-15 19:10:18 +00:00
|
|
|
#endif
|
2005-07-30 21:11:48 +00:00
|
|
|
}
|
2003-06-04 14:37:43 +00:00
|
|
|
}
|
2011-09-14 17:19:40 +00:00
|
|
|
dst += dest.format.bytesPerPixel;
|
|
|
|
dst2 += dest.format.bytesPerPixel;
|
2010-10-15 19:10:18 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
if (scale2x) {
|
|
|
|
dst++;
|
|
|
|
dst2++;
|
|
|
|
dst3 += 2;
|
|
|
|
dst4 += 2;
|
|
|
|
}
|
|
|
|
#endif
|
2003-06-04 14:37:43 +00:00
|
|
|
}
|
|
|
|
|
2010-10-15 19:10:18 +00:00
|
|
|
dst += pitch;
|
|
|
|
dst2 += pitch;
|
2011-05-02 12:32:49 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
2010-10-15 19:10:18 +00:00
|
|
|
dst3 += pitch;
|
|
|
|
dst4 += pitch;
|
|
|
|
#endif
|
2003-06-04 14:37:43 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-07 14:33:14 +00:00
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
int CharsetRendererTownsV3::getDrawWidthIntern(uint16 chr) {
|
2011-07-09 21:57:25 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_useCJKMode && chr > 127) {
|
|
|
|
assert(_vm->_cjkFont);
|
|
|
|
return _vm->_cjkFont->getCharWidth(chr);
|
|
|
|
}
|
2011-07-09 21:57:25 +00:00
|
|
|
#endif
|
2011-07-07 14:33:14 +00:00
|
|
|
return CharsetRendererV3::getDrawWidthIntern(chr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererTownsV3::getDrawHeightIntern(uint16 chr) {
|
2011-07-09 21:57:25 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2011-07-07 14:33:14 +00:00
|
|
|
if (_vm->_useCJKMode && chr > 127) {
|
|
|
|
assert(_vm->_cjkFont);
|
|
|
|
return _vm->_cjkFont->getFontHeight();
|
|
|
|
}
|
2011-07-09 21:57:25 +00:00
|
|
|
#endif
|
2011-07-07 14:33:14 +00:00
|
|
|
return CharsetRendererV3::getDrawHeightIntern(chr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererTownsV3::setDrawCharIntern(uint16 chr) {
|
|
|
|
_sjisCurChar = (_vm->_useCJKMode && chr > 127) ? chr : 0;
|
|
|
|
}
|
|
|
|
#endif
|
2003-06-04 14:37:43 +00:00
|
|
|
|
2009-11-22 11:43:12 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2011-09-14 17:19:40 +00:00
|
|
|
void CharsetRendererPCE::drawBits1(Graphics::Surface &dest, int x, int y, const byte *src, int drawTop, int width, int height) {
|
|
|
|
byte *dst = (byte *)dest.getBasePtr(x, y);
|
2011-07-09 18:06:18 +00:00
|
|
|
if (_sjisCurChar) {
|
|
|
|
assert(_vm->_cjkFont);
|
|
|
|
uint16 col1 = _color;
|
|
|
|
uint16 col2 = _shadowColor;
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
if (dest.format.bytesPerPixel == 2) {
|
2011-07-09 18:06:18 +00:00
|
|
|
col1 = _vm->_16BitPalette[col1];
|
|
|
|
col2 = _vm->_16BitPalette[col2];
|
|
|
|
}
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
_vm->_cjkFont->drawChar(dst, _sjisCurChar, dest.pitch, dest.format.bytesPerPixel, col1, col2, -1, -1);
|
2011-07-09 18:06:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-11-22 08:20:20 +00:00
|
|
|
byte bits = 0;
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
for (y = 0; y < height && y + drawTop < dest.h; y++) {
|
2011-07-09 18:06:18 +00:00
|
|
|
int bitCount = 0;
|
2009-11-22 08:20:20 +00:00
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
if ((bitCount % 8) == 0)
|
|
|
|
bits = *src++;
|
|
|
|
if ((bits & revBitMask(bitCount % 8)) && y + drawTop >= 0) {
|
2011-09-14 17:19:40 +00:00
|
|
|
if (dest.format.bytesPerPixel == 2) {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow)
|
2011-09-14 17:19:40 +00:00
|
|
|
WRITE_UINT16(dst + dest.pitch + 2, _vm->_16BitPalette[_shadowColor]);
|
2009-11-22 08:20:20 +00:00
|
|
|
WRITE_UINT16(dst, _vm->_16BitPalette[_color]);
|
|
|
|
} else {
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow)
|
2011-09-14 17:19:40 +00:00
|
|
|
*(dst + dest.pitch + 1) = _shadowColor;
|
2009-11-22 08:20:20 +00:00
|
|
|
*dst = _color;
|
|
|
|
}
|
|
|
|
}
|
2011-09-14 17:19:40 +00:00
|
|
|
dst += dest.format.bytesPerPixel;
|
2009-11-22 08:20:20 +00:00
|
|
|
bitCount++;
|
|
|
|
}
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
dst += dest.pitch - width * dest.format.bytesPerPixel;
|
2009-11-22 08:20:20 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-07 14:33:14 +00:00
|
|
|
|
|
|
|
int CharsetRendererPCE::getDrawWidthIntern(uint16 chr) {
|
2011-07-09 22:44:18 +00:00
|
|
|
if (_vm->_useCJKMode && chr > 127)
|
|
|
|
return _vm->_2byteWidth;
|
2011-07-07 14:33:14 +00:00
|
|
|
return CharsetRendererV3::getDrawWidthIntern(chr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererPCE::getDrawHeightIntern(uint16 chr) {
|
2011-07-09 22:44:18 +00:00
|
|
|
if (_vm->_useCJKMode && chr > 127)
|
|
|
|
return _vm->_2byteHeight;
|
2011-07-07 14:33:14 +00:00
|
|
|
return CharsetRendererV3::getDrawHeightIntern(chr);
|
|
|
|
}
|
2011-07-09 18:06:18 +00:00
|
|
|
|
|
|
|
void CharsetRendererPCE::setDrawCharIntern(uint16 chr) {
|
|
|
|
_sjisCurChar = (_vm->_useCJKMode && chr > 127) ? chr : 0;
|
|
|
|
}
|
2009-11-22 11:43:12 +00:00
|
|
|
#endif
|
2009-11-22 08:20:20 +00:00
|
|
|
|
2008-05-06 03:00:26 +00:00
|
|
|
#ifdef ENABLE_SCUMM_7_8
|
2003-10-02 22:42:03 +00:00
|
|
|
CharsetRendererNut::CharsetRendererNut(ScummEngine *vm)
|
2003-03-06 17:58:13 +00:00
|
|
|
: CharsetRenderer(vm) {
|
2002-12-26 01:47:40 +00:00
|
|
|
_current = 0;
|
2003-03-04 02:43:43 +00:00
|
|
|
|
2002-12-31 21:41:24 +00:00
|
|
|
for (int i = 0; i < 5; i++) {
|
2007-01-27 01:50:41 +00:00
|
|
|
_fr[i] = NULL;
|
2002-12-26 01:47:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 17:58:13 +00:00
|
|
|
CharsetRendererNut::~CharsetRendererNut() {
|
2003-11-18 05:14:18 +00:00
|
|
|
for (int i = 0; i < 5; i++) {
|
2002-12-26 01:47:40 +00:00
|
|
|
delete _fr[i];
|
2003-11-18 05:14:18 +00:00
|
|
|
}
|
2002-12-26 01:47:40 +00:00
|
|
|
}
|
|
|
|
|
2008-08-02 22:51:53 +00:00
|
|
|
void CharsetRendererNut::setCurID(int32 id) {
|
|
|
|
if (id == -1)
|
|
|
|
return;
|
|
|
|
|
2007-01-27 01:50:41 +00:00
|
|
|
int numFonts = ((_vm->_game.id == GID_CMI) && (_vm->_game.features & GF_DEMO)) ? 4 : 5;
|
|
|
|
assert(id < numFonts);
|
2002-12-26 01:47:40 +00:00
|
|
|
_curId = id;
|
2007-01-27 01:50:41 +00:00
|
|
|
if (!_fr[id]) {
|
|
|
|
char fontname[11];
|
|
|
|
sprintf(fontname, "font%d.nut", id);
|
2007-02-03 05:50:35 +00:00
|
|
|
_fr[id] = new NutRenderer(_vm, fontname);
|
2007-01-27 01:50:41 +00:00
|
|
|
}
|
2002-12-26 01:47:40 +00:00
|
|
|
_current = _fr[id];
|
2003-03-04 02:43:43 +00:00
|
|
|
assert(_current);
|
2002-12-26 01:47:40 +00:00
|
|
|
}
|
|
|
|
|
2005-05-26 12:26:03 +00:00
|
|
|
int CharsetRendererNut::getCharHeight(byte chr) {
|
|
|
|
assert(_current);
|
|
|
|
return _current->getCharHeight(chr);
|
|
|
|
}
|
|
|
|
|
2010-10-12 22:17:00 +00:00
|
|
|
int CharsetRendererNut::getCharWidth(uint16 chr) {
|
2002-12-26 01:47:40 +00:00
|
|
|
assert(_current);
|
|
|
|
return _current->getCharWidth(chr);
|
|
|
|
}
|
|
|
|
|
2003-03-06 17:58:13 +00:00
|
|
|
int CharsetRendererNut::getFontHeight() {
|
2002-12-26 01:47:40 +00:00
|
|
|
// FIXME / TODO: how to implement this properly???
|
|
|
|
assert(_current);
|
|
|
|
return _current->getCharHeight('|');
|
|
|
|
}
|
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
void CharsetRendererNut::printChar(int chr, bool ignoreCharsetMask) {
|
2004-01-08 03:10:16 +00:00
|
|
|
Common::Rect shadow;
|
2003-06-15 15:19:22 +00:00
|
|
|
|
2002-12-26 01:47:40 +00:00
|
|
|
assert(_current);
|
|
|
|
if (chr == '@')
|
|
|
|
return;
|
|
|
|
|
2007-02-03 05:50:35 +00:00
|
|
|
shadow.left = _left;
|
|
|
|
shadow.top = _top;
|
2003-06-15 15:19:22 +00:00
|
|
|
|
2002-12-26 01:47:40 +00:00
|
|
|
if (_firstChar) {
|
2004-04-03 23:44:06 +00:00
|
|
|
_str.left = (shadow.left >= 0) ? shadow.left : 0;
|
|
|
|
_str.top = (shadow.top >= 0) ? shadow.top : 0;
|
|
|
|
_str.right = _str.left;
|
|
|
|
_str.bottom = _str.top;
|
2002-12-26 01:47:40 +00:00
|
|
|
_firstChar = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int width = _current->getCharWidth(chr);
|
|
|
|
int height = _current->getCharHeight(chr);
|
|
|
|
|
2014-07-23 08:59:07 +00:00
|
|
|
bool is2byte = chr >= 256 && _vm->_useCJKMode;
|
|
|
|
if (is2byte)
|
2004-10-23 23:08:53 +00:00
|
|
|
width = _vm->_2byteWidth;
|
2003-06-04 14:37:43 +00:00
|
|
|
|
2007-02-03 05:50:35 +00:00
|
|
|
shadow.right = _left + width;
|
|
|
|
shadow.bottom = _top + height;
|
2003-06-15 15:19:22 +00:00
|
|
|
|
2004-08-08 22:10:38 +00:00
|
|
|
Graphics::Surface s;
|
2006-01-10 00:28:10 +00:00
|
|
|
if (!ignoreCharsetMask) {
|
2004-04-08 01:38:07 +00:00
|
|
|
_hasMask = true;
|
2004-04-08 23:41:10 +00:00
|
|
|
_textScreenID = kMainVirtScreen;
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2004-10-01 09:06:12 +00:00
|
|
|
int drawTop = _top;
|
2006-01-10 00:28:10 +00:00
|
|
|
if (ignoreCharsetMask) {
|
2007-09-08 11:15:27 +00:00
|
|
|
VirtScreen *vs = &_vm->_virtscr[kMainVirtScreen];
|
2004-08-14 19:42:00 +00:00
|
|
|
s = *vs;
|
2013-08-04 00:29:13 +00:00
|
|
|
s.setPixels(vs->getPixels(0, 0));
|
2004-08-08 22:10:38 +00:00
|
|
|
} else {
|
2007-01-28 20:11:31 +00:00
|
|
|
s = _vm->_textSurface;
|
2004-10-01 09:06:12 +00:00
|
|
|
drawTop -= _vm->_screenTop;
|
2004-08-08 22:10:38 +00:00
|
|
|
}
|
2004-04-08 01:38:07 +00:00
|
|
|
|
2007-02-03 05:50:35 +00:00
|
|
|
if (chr >= 256 && _vm->_useCJKMode)
|
|
|
|
_current->draw2byte(s, chr, _left, drawTop, _color);
|
|
|
|
else
|
|
|
|
_current->drawChar(s, (byte)chr, _left, drawTop, _color);
|
2004-01-08 03:24:41 +00:00
|
|
|
_vm->markRectAsDirty(kMainVirtScreen, shadow);
|
2002-12-26 01:47:40 +00:00
|
|
|
|
2004-01-08 02:16:27 +00:00
|
|
|
if (_str.left > _left)
|
|
|
|
_str.left = _left;
|
|
|
|
|
2007-07-21 22:40:02 +00:00
|
|
|
// Original keeps glyph width and character dimensions separately
|
2014-07-23 08:59:07 +00:00
|
|
|
if ((_vm->_language == Common::ZH_TWN || _vm->_language == Common::KO_KOR) && is2byte)
|
|
|
|
width++;
|
2007-07-21 22:40:02 +00:00
|
|
|
|
2002-12-26 01:47:40 +00:00
|
|
|
_left += width;
|
|
|
|
|
2004-02-22 21:02:06 +00:00
|
|
|
if (_str.right < shadow.right)
|
2004-01-08 03:10:16 +00:00
|
|
|
_str.right = shadow.right;
|
2003-06-15 15:19:22 +00:00
|
|
|
|
2004-01-08 03:10:16 +00:00
|
|
|
if (_str.bottom < shadow.bottom)
|
|
|
|
_str.bottom = shadow.bottom;
|
2002-12-26 01:47:40 +00:00
|
|
|
}
|
2005-06-18 15:44:40 +00:00
|
|
|
#endif
|
2002-12-26 01:47:40 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
void CharsetRendererNES::printChar(int chr, bool ignoreCharsetMask) {
|
2005-03-16 03:20:32 +00:00
|
|
|
int width, height, origWidth, origHeight;
|
|
|
|
VirtScreen *vs;
|
2011-09-14 17:19:40 +00:00
|
|
|
byte *charPtr;
|
2005-03-16 03:20:32 +00:00
|
|
|
|
2005-05-20 22:49:09 +00:00
|
|
|
// Init it here each time since it is cheap and fixes bug with
|
|
|
|
// charset after game load
|
|
|
|
_trTable = _vm->getResourceAddress(rtCostume, 77) + 2;
|
2005-04-10 09:58:37 +00:00
|
|
|
|
2005-03-24 04:53:28 +00:00
|
|
|
// HACK: how to set it properly?
|
|
|
|
if (_top == 0)
|
|
|
|
_top = 16;
|
|
|
|
|
2005-03-16 03:20:32 +00:00
|
|
|
if ((vs = _vm->findVirtScreen(_top)) == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (chr == '@')
|
|
|
|
return;
|
|
|
|
|
2005-03-24 03:22:32 +00:00
|
|
|
charPtr = _vm->_NESPatTable[1] + _trTable[chr - 32] * 16;
|
2005-03-16 03:20:32 +00:00
|
|
|
width = getCharWidth(chr);
|
|
|
|
height = 8;
|
|
|
|
|
|
|
|
origWidth = width;
|
|
|
|
origHeight = height;
|
|
|
|
|
|
|
|
if (_firstChar) {
|
|
|
|
_str.left = _left;
|
|
|
|
_str.top = _top;
|
|
|
|
_str.right = _left;
|
|
|
|
_str.bottom = _top;
|
|
|
|
_firstChar = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int drawTop = _top - vs->topline;
|
|
|
|
|
|
|
|
_vm->markRectAsDirty(vs->number, _left, _left + width, drawTop, drawTop + height);
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2006-01-10 00:28:10 +00:00
|
|
|
if (!ignoreCharsetMask) {
|
2005-03-16 03:20:32 +00:00
|
|
|
_hasMask = true;
|
|
|
|
_textScreenID = vs->number;
|
|
|
|
}
|
2005-03-23 04:22:11 +00:00
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
if (ignoreCharsetMask || !vs->hasTwoBuffers)
|
|
|
|
drawBits1(*vs, _left + vs->xstart, drawTop, charPtr, drawTop, origWidth, origHeight);
|
|
|
|
else
|
|
|
|
drawBits1(_vm->_textSurface, _left, _top, charPtr, drawTop, origWidth, origHeight);
|
2005-03-16 03:20:32 +00:00
|
|
|
|
|
|
|
if (_str.left > _left)
|
|
|
|
_str.left = _left;
|
|
|
|
|
|
|
|
_left += origWidth;
|
|
|
|
|
|
|
|
if (_str.right < _left) {
|
|
|
|
_str.right = _left;
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow)
|
2005-03-16 03:20:32 +00:00
|
|
|
_str.right++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_str.bottom < _top + height)
|
|
|
|
_str.bottom = _top + height;
|
|
|
|
}
|
|
|
|
|
2010-11-05 00:36:23 +00:00
|
|
|
void CharsetRendererNES::drawChar(int chr, Graphics::Surface &s, int x, int y) {
|
2011-09-14 17:19:40 +00:00
|
|
|
byte *charPtr;
|
2005-03-16 03:20:32 +00:00
|
|
|
int width, height;
|
|
|
|
|
2005-04-10 09:58:37 +00:00
|
|
|
if (!_trTable)
|
|
|
|
_trTable = _vm->getResourceAddress(rtCostume, 77) + 2;
|
|
|
|
|
2005-03-24 03:22:32 +00:00
|
|
|
charPtr = _vm->_NESPatTable[1] + _trTable[chr - 32] * 16;
|
2005-03-16 03:20:32 +00:00
|
|
|
width = getCharWidth(chr);
|
|
|
|
height = 8;
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
drawBits1(s, x, y, charPtr, y, width, height);
|
2005-03-16 03:20:32 +00:00
|
|
|
}
|
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
|
|
|
#ifndef DISABLE_TOWNS_DUAL_LAYER_MODE
|
|
|
|
CharsetRendererTownsClassic::CharsetRendererTownsClassic(ScummEngine *vm) : CharsetRendererClassic(vm), _sjisCurChar(0) {
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererTownsClassic::getCharWidth(uint16 chr) {
|
|
|
|
int spacing = 0;
|
|
|
|
|
|
|
|
if (_vm->_useCJKMode) {
|
|
|
|
if ((chr & 0xff00) == 0xfd00) {
|
|
|
|
chr &= 0xff;
|
|
|
|
} else if (chr >= 256) {
|
|
|
|
spacing = 8;
|
|
|
|
} else if (useFontRomCharacter(chr)) {
|
|
|
|
spacing = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (spacing) {
|
|
|
|
if (_vm->_game.id == GID_MONKEY) {
|
|
|
|
spacing++;
|
|
|
|
if (_curId == 2)
|
|
|
|
spacing++;
|
|
|
|
} else if (_vm->_game.id != GID_INDY4 && _curId == 1) {
|
|
|
|
spacing++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!spacing) {
|
|
|
|
int offs = READ_LE_UINT32(_fontPtr + chr * 4 + 4);
|
|
|
|
if (offs)
|
|
|
|
spacing = _fontPtr[offs] + (signed char)_fontPtr[offs + 2];
|
|
|
|
}
|
|
|
|
|
|
|
|
return spacing;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CharsetRendererTownsClassic::getFontHeight() {
|
|
|
|
static const uint8 sjisFontHeightM1[] = { 0, 8, 9, 8, 9, 8, 9, 0, 0, 0 };
|
|
|
|
static const uint8 sjisFontHeightM2[] = { 0, 8, 9, 9, 9, 8, 9, 9, 9, 8 };
|
|
|
|
static const uint8 sjisFontHeightI4[] = { 0, 8, 9, 9, 9, 8, 8, 8, 8, 8 };
|
|
|
|
const uint8 *htbl = (_vm->_game.id == GID_MONKEY) ? sjisFontHeightM1 : ((_vm->_game.id == GID_INDY4) ? sjisFontHeightI4 : sjisFontHeightM2);
|
|
|
|
return _vm->_useCJKMode ? htbl[_curId] : _fontHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererTownsClassic::drawBitsN(const Graphics::Surface&, byte *dst, const byte *src, byte bpp, int drawTop, int width, int height) {
|
|
|
|
if (_sjisCurChar) {
|
|
|
|
assert(_vm->_cjkFont);
|
|
|
|
_vm->_cjkFont->drawChar(_vm->_textSurface, _sjisCurChar, _left * _vm->_textSurfaceMultiplier, (_top - _vm->_screenTop) * _vm->_textSurfaceMultiplier, _vm->_townsCharsetColorMap[1], _shadowColor);
|
|
|
|
return;
|
|
|
|
}
|
2012-09-26 02:17:31 +00:00
|
|
|
|
2011-07-07 14:33:14 +00:00
|
|
|
bool scale2x = (_vm->_textSurfaceMultiplier == 2);
|
2013-08-02 20:23:00 +00:00
|
|
|
dst = (byte *)_vm->_textSurface.getBasePtr(_left * _vm->_textSurfaceMultiplier, (_top - _vm->_screenTop) * _vm->_textSurfaceMultiplier);
|
2011-07-07 14:33:14 +00:00
|
|
|
|
|
|
|
int y, x;
|
|
|
|
int color;
|
|
|
|
byte numbits, bits;
|
|
|
|
|
|
|
|
int pitch = _vm->_textSurface.pitch - width;
|
|
|
|
|
|
|
|
assert(bpp == 1 || bpp == 2 || bpp == 4 || bpp == 8);
|
|
|
|
bits = *src++;
|
|
|
|
numbits = 8;
|
|
|
|
byte *cmap = _vm->_charsetColorMap;
|
|
|
|
byte *dst2 = dst;
|
|
|
|
|
|
|
|
if (_vm->_game.platform == Common::kPlatformFMTowns)
|
|
|
|
cmap = _vm->_townsCharsetColorMap;
|
|
|
|
if (scale2x) {
|
|
|
|
dst2 += _vm->_textSurface.pitch;
|
|
|
|
pitch <<= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (y = 0; y < height && y + drawTop < _vm->_textSurface.h; y++) {
|
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
color = (bits >> (8 - bpp)) & 0xFF;
|
|
|
|
|
|
|
|
if (color && y + drawTop >= 0) {
|
|
|
|
*dst = cmap[color];
|
|
|
|
if (scale2x)
|
|
|
|
dst[1] = dst2[0] = dst2[1] = dst[0];
|
|
|
|
}
|
|
|
|
dst++;
|
|
|
|
|
|
|
|
if (scale2x) {
|
|
|
|
dst++;
|
|
|
|
dst2 += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
bits <<= bpp;
|
|
|
|
numbits -= bpp;
|
|
|
|
if (numbits == 0) {
|
|
|
|
bits = *src++;
|
|
|
|
numbits = 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dst += pitch;
|
|
|
|
dst2 += pitch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-13 22:50:14 +00:00
|
|
|
bool CharsetRendererTownsClassic::prepareDraw(uint16 chr) {
|
2011-07-07 14:33:14 +00:00
|
|
|
processCharsetColors();
|
|
|
|
bool noSjis = false;
|
|
|
|
|
|
|
|
if (_vm->_game.platform == Common::kPlatformFMTowns && _vm->_useCJKMode) {
|
|
|
|
if ((chr & 0x00ff) == 0x00fd) {
|
|
|
|
chr >>= 8;
|
|
|
|
noSjis = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (useFontRomCharacter(chr) && !noSjis) {
|
|
|
|
setupShadowMode();
|
|
|
|
_charPtr = 0;
|
|
|
|
_sjisCurChar = chr;
|
|
|
|
|
|
|
|
_width = getCharWidth(chr);
|
|
|
|
// For whatever reason MI1 uses a different font width
|
|
|
|
// for alignment calculation and for drawing when
|
|
|
|
// charset 2 is active. This fixes some subtle glitches.
|
|
|
|
if (_vm->_game.id == GID_MONKEY && _curId == 2)
|
|
|
|
_width--;
|
|
|
|
_origWidth = _width;
|
|
|
|
|
|
|
|
_origHeight = _height = getFontHeight();
|
|
|
|
_offsX = _offsY = 0;
|
|
|
|
} else if (_vm->_useCJKMode && (chr >= 128) && !noSjis) {
|
|
|
|
setupShadowMode();
|
|
|
|
_origWidth = _width = _vm->_2byteWidth;
|
|
|
|
_origHeight = _height = _vm->_2byteHeight;
|
|
|
|
_charPtr = _vm->get2byteCharPtr(chr);
|
|
|
|
_offsX = _offsY = 0;
|
2014-07-23 07:16:06 +00:00
|
|
|
if (_enableShadow) {
|
2011-07-07 14:33:14 +00:00
|
|
|
_width++;
|
|
|
|
_height++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
_sjisCurChar = 0;
|
2011-07-13 22:50:14 +00:00
|
|
|
return CharsetRendererClassic::prepareDraw(chr);
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
2011-07-13 22:50:14 +00:00
|
|
|
return true;
|
2011-07-07 14:33:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererTownsClassic::setupShadowMode() {
|
2014-07-23 07:16:06 +00:00
|
|
|
_enableShadow = true;
|
2011-07-07 14:33:14 +00:00
|
|
|
_shadowColor = _vm->_townsCharsetColorMap[0];
|
|
|
|
assert(_vm->_cjkFont);
|
|
|
|
|
|
|
|
if (((_vm->_game.id == GID_MONKEY) && (_curId == 2 || _curId == 4 || _curId == 6)) ||
|
|
|
|
((_vm->_game.id == GID_MONKEY2) && (_curId != 1 && _curId != 5 && _curId != 9)) ||
|
|
|
|
((_vm->_game.id == GID_INDY4) && (_curId == 2 || _curId == 3 || _curId == 4))) {
|
|
|
|
_vm->_cjkFont->setDrawingMode(Graphics::FontSJIS::kOutlineMode);
|
|
|
|
} else {
|
|
|
|
_vm->_cjkFont->setDrawingMode(Graphics::FontSJIS::kDefaultMode);
|
|
|
|
}
|
|
|
|
|
|
|
|
_vm->_cjkFont->toggleFlippedMode((_vm->_game.id == GID_MONKEY || _vm->_game.id == GID_MONKEY2) && _curId == 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CharsetRendererTownsClassic::useFontRomCharacter(uint16 chr) {
|
|
|
|
if (!_vm->_useCJKMode)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Some SCUMM 5 games contain hard coded logic to determine whether to use
|
|
|
|
// the SCUMM fonts or the FM-Towns font rom to draw a character. For the other
|
|
|
|
// games we will simply check for a character greater 127.
|
|
|
|
if (chr < 128) {
|
|
|
|
if (((_vm->_game.id == GID_MONKEY2 && _curId != 0) || (_vm->_game.id == GID_INDY4 && _curId != 3)) && (chr > 31 && chr != 94 && chr != 95 && chr != 126 && chr != 127))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CharsetRendererTownsClassic::processCharsetColors() {
|
|
|
|
for (int i = 0; i < (1 << _bytesPerPixel); i++) {
|
|
|
|
uint8 c = _vm->_charsetColorMap[i];
|
|
|
|
|
|
|
|
if (c > 16) {
|
|
|
|
uint8 t = (_vm->_currentPalette[c * 3] < 32) ? 4 : 12;
|
|
|
|
t |= ((_vm->_currentPalette[c * 3 + 1] < 32) ? 2 : 10);
|
|
|
|
t |= ((_vm->_currentPalette[c * 3 + 1] < 32) ? 1 : 9);
|
|
|
|
c = t;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == 0)
|
|
|
|
c = _vm->_townsOverrideShadowColor;
|
|
|
|
|
|
|
|
c = ((c & 0x0f) << 4) | (c & 0x0f);
|
|
|
|
_vm->_townsCharsetColorMap[i] = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2011-09-14 17:19:40 +00:00
|
|
|
void CharsetRendererNES::drawBits1(Graphics::Surface &dest, int x, int y, const byte *src, int drawTop, int width, int height) {
|
|
|
|
byte *dst = (byte *)dest.getBasePtr(x, y);
|
2005-03-16 03:20:32 +00:00
|
|
|
for (int i = 0; i < 8; i++) {
|
|
|
|
byte c0 = src[i];
|
|
|
|
byte c1 = src[i + 8];
|
|
|
|
for (int j = 0; j < 8; j++)
|
2005-03-24 03:22:32 +00:00
|
|
|
dst[j] = _vm->_NESPalette[0][((c0 >> (7 - j)) & 1) | (((c1 >> (7 - j)) & 1) << 1) |
|
2005-03-21 23:51:48 +00:00
|
|
|
(_color ? 12 : 8)];
|
2011-09-14 17:19:40 +00:00
|
|
|
dst += dest.pitch;
|
2005-03-16 03:20:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-09 14:17:06 +00:00
|
|
|
} // End of namespace Scumm
|