SCUMM: Add Macintosh b/w rendering mode

This is currently only (partially) implemented for the 16-color Mac
versions of Loom and Indiana Jones and the Last Crusade. The text is
still drawn in color, since that's rendered separately, but I'm
committing this now while it still works.
This commit is contained in:
Torbjörn Andersson 2021-07-10 15:21:00 +02:00 committed by Eugene Sandulenko
parent e2f1e9b23e
commit 2ba156184c
6 changed files with 123 additions and 31 deletions

View File

@ -168,7 +168,7 @@ static const char HELP_STRING[] =
" (default: enabled)\n"
" --render-mode=MODE Enable additional render modes (hercGreen, hercAmber,\n"
" cga, ega, vga, amiga, fmtowns, pc9821, pc9801, 2gs,\n"
" atari, macintosh)\n"
" atari, macintosh, macintoshbw)\n"
#ifdef ENABLE_EVENTRECORDER
" --record-mode=MODE Specify record mode for event recorder (record, playback,\n"
" passthrough [default])\n"

View File

@ -44,6 +44,7 @@ const RenderModeDescription g_renderModes[] = {
{ "2gs", "Apple IIgs", kRenderApple2GS },
{ "atari", "Atari ST", kRenderAtariST },
{ "macintosh", "Macintosh", kRenderMacintosh },
{ "macintoshbw", _s("Macintosh b/w"), kRenderMacintoshBW },
{nullptr, nullptr, kRenderDefault}
};

View File

@ -57,7 +57,8 @@ enum RenderMode {
kRenderPC9801 = 9,
kRenderApple2GS = 10,
kRenderAtariST = 11,
kRenderMacintosh = 12
kRenderMacintosh = 12,
kRenderMacintoshBW = 13
};
struct RenderModeDescription {

View File

@ -21,6 +21,7 @@
*/
#include "common/system.h"
#include "graphics/macega.h"
#include "scumm/actor.h"
#include "scumm/charset.h"
#include "scumm/usage_bits.h"
@ -29,6 +30,7 @@
namespace Scumm {
void ScummEngine::mac_drawStripToScreen(VirtScreen *vs, int top, int x, int y, int width, int height) {
const byte *pixels = vs->getPixels(x, top);
const byte *ts = (byte *)_textSurface.getBasePtr(x * 2, y * 2);
byte *mac = (byte *)_macScreen->getBasePtr(x * 2, y * 2);
@ -37,23 +39,43 @@ void ScummEngine::mac_drawStripToScreen(VirtScreen *vs, int top, int x, int y, i
int tsPitch = _textSurface.pitch;
int macPitch = _macScreen->pitch;
for (int h = 0; h < height; h++) {
for (int w = 0; w < width; w++) {
if (ts[2 * w] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w] = pixels[w];
if (ts[2 * w + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + 1] = pixels[w];
if (ts[2 * w + tsPitch] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch] = pixels[w];
if (ts[2 * w + tsPitch + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch + 1] = pixels[w];
}
if (_renderMode == Common::kRenderMacintoshBW) {
for (int h = 0; h < height; h++) {
for (int w = 0; w < width; w++) {
if (ts[2 * w] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w] = Graphics::macEGADither[pixels[w]][0];
if (ts[2 * w + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + 1] = Graphics::macEGADither[pixels[w]][1];
if (ts[2 * w + tsPitch] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch] = Graphics::macEGADither[pixels[w]][2];
if (ts[2 * w + tsPitch + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch + 1] = Graphics::macEGADither[pixels[w]][3];
}
pixels += pixelsPitch;
ts += tsPitch * 2;
mac += macPitch * 2;
pixels += pixelsPitch;
ts += tsPitch * 2;
mac += macPitch * 2;
}
} else {
for (int h = 0; h < height; h++) {
for (int w = 0; w < width; w++) {
if (ts[2 * w] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w] = pixels[w];
if (ts[2 * w + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + 1] = pixels[w];
if (ts[2 * w + tsPitch] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch] = pixels[w];
if (ts[2 * w + tsPitch + 1] == CHARSET_MASK_TRANSPARENCY)
mac[2 * w + macPitch + 1] = pixels[w];
}
pixels += pixelsPitch;
ts += tsPitch * 2;
mac += macPitch * 2;
}
}
_system->copyRectToScreen(_macScreen->getBasePtr(x * 2, y * 2), _macScreen->pitch, x * 2, y * 2, width * 2, height * 2);
}

View File

@ -25,6 +25,7 @@
#include "common/textconsole.h"
#include "common/util.h"
#include "graphics/macega.h"
#include "graphics/palette.h"
#include "scumm/resource.h"
@ -137,20 +138,6 @@ void ScummEngine::resetPalette() {
0xFF, 0x99, 0x99, 0xFF, 0x55, 0xFF, 0xFF, 0xFF, 0x77, 0xFF, 0xFF, 0xFF
};
// Theoreticaly, it should be possible to get the palette from the
// game's "clut" reosurce. But when I try that, I still get the wrong
// colours. This table is based on what Basilisk II draws.
// 6 = brown
// 11 = bright cyan
static const byte tableMacPalette[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0xBE, 0x00, 0xBE, 0x00, 0x00, 0xBE, 0xBE,
0xBE, 0x00, 0x00, 0xBE, 0x00, 0xBE, 0xBE, 0x75, 0x00, 0xBE, 0xBE, 0xBE,
0x75, 0x75, 0x75, 0x75, 0x75, 0xFC, 0x75, 0xFC, 0x75, 0x75, 0xFC, 0xFC,
0xFC, 0x75, 0x75, 0xFC, 0x75, 0xFC, 0xFC, 0xFC, 0x75, 0xFC, 0xFC, 0xFC
};
static const byte tableEGAPalette[] = {
0x00, 0x00, 0x00, 0x00, 0x00, 0xAA, 0x00, 0xAA, 0x00, 0x00, 0xAA, 0xAA,
0xAA, 0x00, 0x00, 0xAA, 0x00, 0xAA, 0xAA, 0x55, 0x00, 0xAA, 0xAA, 0xAA,
@ -254,7 +241,7 @@ void ScummEngine::resetPalette() {
if ((_game.platform == Common::kPlatformAmiga) || (_game.platform == Common::kPlatformAtariST))
setPaletteFromTable(tableAmigaPalette, sizeof(tableAmigaPalette) / 3);
else if ((_game.id == GID_LOOM || _game.id == GID_INDY3) && _game.platform == Common::kPlatformMacintosh)
setPaletteFromTable(tableMacPalette, sizeof(tableMacPalette) / 3);
setPaletteFromTable(Graphics::macEGAPalette, sizeof(Graphics::macEGAPalette) / 3);
else
setPaletteFromTable(tableEGAPalette, sizeof(tableEGAPalette) / 3);
}

81
graphics/macega.h Normal file
View File

@ -0,0 +1,81 @@
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#ifndef GRAPHICS_MACEGA_H
#define GRAPHICS_MACEGA_H
namespace Graphics {
// This is the palette that Basilisk II uses by default for 16-color games.
// It's basically the same as EGA, but a little bit brighter. While it does
// seem like games can have their own palette (CLUT) resources, at least Loom
// didn't seem to use it. (I could be wrong about that.)
static const byte macEGAPalette[] = {
0x00, 0x00, 0x00, // Black
0x00, 0x00, 0xBE, // Blue
0x00, 0xBE, 0x00, // Green
0x00, 0xBE, 0xBE, // Cyan
0xBE, 0x00, 0x00, // Red
0xBE, 0x00, 0xBE, // Magenta
0xBE, 0x75, 0x00, // Brown
0xBE, 0xBE, 0xBE, // Light Gray
0x75, 0x75, 0x75, // Dark Gray
0x75, 0x75, 0xFC, // Bright Blue
0x75, 0xFC, 0x75, // Bright Green
0x75, 0xFC, 0xFC, // Bright Cyan
0xFC, 0x75, 0x75, // Bright Red
0xFC, 0x75, 0xFC, // Bright Magenta
0xFC, 0xFC, 0x75, // Bright Yellow
0xFC, 0xFC, 0xFC // White
};
// Each color has its own 2x2 dithering pattern. This array remaps them to the
// color indexes for black and white. The order of the pixels is upper left,
// upper right, lower left, lower right. I don't know if this is the standard
// method, but it does seem to be what LucasArts used so maybe it's useful for
// other games as well.
//
// One obvious candidate would be the Mac AGI games, though looking at
// screenshots makes me suspect that they used 3x2 pixels for each color.
const byte macEGADither[16][4] = {
{ 0, 0, 0, 0 }, // Black
{ 0, 0, 0, 15 }, // Blue
{ 0, 0, 15, 0 }, // Green
{ 0, 0, 15, 15 }, // Cyan
{ 0, 15, 0, 0 }, // Red
{ 0, 15, 0, 15 }, // Magenta
{ 0, 15, 15, 0 }, // Brown
{ 0, 15, 15, 15 }, // Light Gray
{ 15, 0, 0, 0 }, // Dark Gray
{ 15, 0, 0, 15 }, // Bright Blue
{ 15, 0, 15, 0 }, // Bright Green
{ 15, 0, 15, 15 }, // Bright Cyan
{ 15, 15, 0, 0 }, // Bright Red
{ 15, 15, 0, 15 }, // Bright Magenta
{ 15, 15, 15, 0 }, // Bright Yellow
{ 15, 15, 15, 15 } // White
};
} // end of namespace Graphics
#endif // GRAPHICS_MACEGA_H