mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-27 05:32:45 +00:00
692 lines
19 KiB
C++
692 lines
19 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* Additional copyright for this file:
|
|
* Copyright (C) 1995 Presto Studios, Inc.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "common/config-manager.h"
|
|
#include "common/fs.h"
|
|
#include "common/str-array.h"
|
|
#include "common/system.h"
|
|
#include "common/compression/unzip.h"
|
|
#include "graphics/cursorman.h"
|
|
#include "graphics/font.h"
|
|
#include "graphics/paletteman.h"
|
|
#include "graphics/surface.h"
|
|
#include "graphics/wincursor.h"
|
|
#include "graphics/fonts/ttf.h"
|
|
#include "image/bmp.h"
|
|
|
|
#include "buried/buried.h"
|
|
#include "buried/graphics.h"
|
|
#include "buried/window.h"
|
|
|
|
namespace Buried {
|
|
|
|
GraphicsManager::GraphicsManager(BuriedEngine *vm) : _vm(vm) {
|
|
_curCursor = kCursorNone;
|
|
_mouseMoved = false;
|
|
_needsErase = false;
|
|
|
|
setCursor(kCursorArrow);
|
|
CursorMan.showMouse(true);
|
|
|
|
_screen = new Graphics::Surface();
|
|
_screen->create(640, 480, g_system->getScreenFormat());
|
|
|
|
if (_vm->isTrueColor()) {
|
|
// No palette to deal with
|
|
_palette = nullptr;
|
|
} else {
|
|
// Grab the palette from our EXE bitmap
|
|
_palette = createDefaultPalette();
|
|
|
|
// Then apply it. The only time we'll use this call even.
|
|
g_system->getPaletteManager()->setPalette(_palette, 0, 256);
|
|
}
|
|
}
|
|
|
|
GraphicsManager::~GraphicsManager() {
|
|
_screen->free();
|
|
delete _screen;
|
|
|
|
delete[] _palette;
|
|
}
|
|
|
|
Graphics::Font *GraphicsManager::createFont(int size, bool bold) const {
|
|
// MS Gothic for the Japanese version
|
|
// Arial or Arial Bold for everything else
|
|
if (_vm->getLanguage() == Common::JA_JPN)
|
|
return createMSGothicFont(size, bold);
|
|
|
|
return createArialFont(size, bold);
|
|
}
|
|
|
|
Graphics::Font *GraphicsManager::createArialFont(int size, bool bold) const {
|
|
const char *defaultBaseName = bold ? "arialbd.ttf" : "arial.ttf";
|
|
|
|
Common::SeekableReadStream *stream = SearchMan.createReadStreamForMember(defaultBaseName);
|
|
|
|
// Map the heights needed to point sizes
|
|
if (bold) {
|
|
if (size != 20)
|
|
error("Unhandled Arial Bold height %d", size);
|
|
|
|
size = 12;
|
|
} else {
|
|
switch (size) {
|
|
case 12:
|
|
case 13:
|
|
size = 7;
|
|
break;
|
|
case 14:
|
|
size = 8;
|
|
break;
|
|
default:
|
|
error("Unhandled Arial height %d", size);
|
|
}
|
|
}
|
|
|
|
// TODO: Make the monochrome mode optional
|
|
// Win3.1 obviously only had raster fonts, but BIT Win3.1 will render
|
|
// with the TrueType font on Win7/Win8 (at least)
|
|
// FIXME: The font is slightly off from the original... need to check. Sizes are right though!
|
|
|
|
Graphics::Font *font;
|
|
|
|
if (stream) {
|
|
font = Graphics::loadTTFFont(*stream, size, Graphics::kTTFSizeModeCharacter, 96, 96, _vm->isTrueColor() ? Graphics::kTTFRenderModeLight : Graphics::kTTFRenderModeMonochrome);
|
|
|
|
delete stream;
|
|
} else {
|
|
const char *fname;
|
|
if (bold)
|
|
fname = "LiberationSans-Bold.ttf";
|
|
else
|
|
fname = "LiberationSans-Regular.ttf";
|
|
|
|
font = Graphics::loadTTFFontFromArchive(fname, size, Graphics::kTTFSizeModeCharacter, 96, 96, _vm->isTrueColor() ? Graphics::kTTFRenderModeLight : Graphics::kTTFRenderModeMonochrome);
|
|
}
|
|
|
|
if (!font)
|
|
error("Failed to load Arial%s font", bold ? " Bold" : "");
|
|
|
|
return font;
|
|
}
|
|
|
|
void GraphicsManager::toggleCursor(bool show) {
|
|
CursorMan.showMouse(show);
|
|
}
|
|
|
|
Cursor GraphicsManager::setCursor(Cursor newCursor) {
|
|
// Don't set the cursor again
|
|
if (newCursor == _curCursor)
|
|
return _curCursor;
|
|
|
|
Cursor oldCursor = _curCursor;
|
|
Graphics::Cursor *cursor = nullptr;
|
|
Graphics::WinCursorGroup *cursorGroup = nullptr;
|
|
|
|
if (newCursor == kCursorArrow) {
|
|
cursor = Graphics::makeDefaultWinCursor();
|
|
} else if (newCursor == kCursorWait) {
|
|
cursor = Graphics::makeBusyWinCursor();
|
|
} else {
|
|
cursorGroup = _vm->getCursorGroup(newCursor);
|
|
|
|
if (!cursorGroup)
|
|
return kCursorNone;
|
|
|
|
cursor = cursorGroup->cursors[0].cursor;
|
|
}
|
|
|
|
if (!cursor)
|
|
error("Failed to find cursor %d", newCursor);
|
|
|
|
CursorMan.replaceCursor(cursor);
|
|
|
|
if (cursorGroup)
|
|
delete cursorGroup;
|
|
else
|
|
delete cursor;
|
|
|
|
_curCursor = newCursor;
|
|
return oldCursor;
|
|
}
|
|
|
|
Graphics::Surface *GraphicsManager::getBitmap(uint32 bitmapID) {
|
|
Common::SeekableReadStream *stream = _vm->getBitmapStream(bitmapID);
|
|
|
|
if (!stream)
|
|
error("Could not find bitmap %d", bitmapID);
|
|
|
|
Graphics::Surface *surface = getBitmap(stream);
|
|
if (!surface)
|
|
error("Failed to decode bitmap %d", bitmapID);
|
|
|
|
return surface;
|
|
}
|
|
|
|
Graphics::Surface *GraphicsManager::getBitmap(const Common::Path &fileName, bool required) {
|
|
Common::SeekableReadStream *stream = SearchMan.createReadStreamForMember(fileName);
|
|
|
|
if (!stream) {
|
|
if (required)
|
|
error("Could not find bitmap '%s'", fileName.toString(Common::Path::kNativeSeparator).c_str());
|
|
return nullptr;
|
|
}
|
|
|
|
Graphics::Surface *surface = getBitmap(stream);
|
|
if (!surface) {
|
|
if (required)
|
|
error("Failed to decode bitmap '%s'", fileName.toString(Common::Path::kNativeSeparator).c_str());
|
|
return nullptr;
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
|
|
Graphics::Surface *GraphicsManager::getBitmap(Common::SeekableReadStream *stream) {
|
|
Image::BitmapDecoder decoder;
|
|
if (!decoder.loadStream(*stream)) {
|
|
delete stream;
|
|
return nullptr;
|
|
}
|
|
|
|
delete stream;
|
|
|
|
// Convert to the screen format, if required
|
|
if (decoder.getSurface()->format != g_system->getScreenFormat()) {
|
|
assert(_vm->isTrueColor());
|
|
return decoder.getSurface()->convertTo(g_system->getScreenFormat(), decoder.getPalette());
|
|
}
|
|
|
|
// Remap the palette, if required
|
|
if (!_vm->isTrueColor() && memcmp(decoder.getPalette() + 3, getDefaultPalette() + 3, 256 - 6) != 0)
|
|
return remapPalettedFrame(decoder.getSurface(), decoder.getPalette());
|
|
|
|
// Just copy the frame
|
|
Graphics::Surface *surface = new Graphics::Surface();
|
|
surface->copyFrom(*decoder.getSurface());
|
|
return surface;
|
|
}
|
|
|
|
uint32 GraphicsManager::getColor(byte r, byte g, byte b) {
|
|
if (_vm->isTrueColor())
|
|
return g_system->getScreenFormat().RGBToColor(r, g, b);
|
|
|
|
// Find the best match color
|
|
int diff = 0x7FFFFFFF;
|
|
byte best = 0;
|
|
|
|
for (uint i = 0; i < 256 && diff > 0; i++) {
|
|
int rDiff = (int)_palette[i * 3] - (int)r;
|
|
int gDiff = (int)_palette[i * 3 + 1] - (int)g;
|
|
int bDiff = (int)_palette[i * 3 + 2] - (int)b;
|
|
|
|
int curDiff = rDiff * rDiff + gDiff * gDiff + bDiff * bDiff;
|
|
|
|
if (curDiff < diff) {
|
|
best = i;
|
|
diff = curDiff;
|
|
}
|
|
}
|
|
|
|
return best;
|
|
}
|
|
|
|
void GraphicsManager::invalidateRect(const Common::Rect &rect, bool erase) {
|
|
if (_dirtyRect.isEmpty())
|
|
_dirtyRect = rect;
|
|
else
|
|
_dirtyRect.extend(rect);
|
|
|
|
_needsErase |= erase;
|
|
}
|
|
|
|
void GraphicsManager::updateScreen(bool drawWindows) {
|
|
bool shouldUpdateScreen = _mouseMoved;
|
|
_mouseMoved = false;
|
|
|
|
if (!_dirtyRect.isEmpty()) {
|
|
// Draw the main window, which will draw its children
|
|
if (drawWindows)
|
|
_vm->_mainWindow->updateWindow();
|
|
|
|
// Copy just that rect
|
|
g_system->copyRectToScreen(_screen->getBasePtr(_dirtyRect.left, _dirtyRect.top), _screen->pitch, _dirtyRect.left, _dirtyRect.top, _dirtyRect.width(), _dirtyRect.height());
|
|
|
|
// Empty out the dirty rect
|
|
_dirtyRect = Common::Rect();
|
|
|
|
// Definitely update
|
|
shouldUpdateScreen = true;
|
|
}
|
|
|
|
if (shouldUpdateScreen)
|
|
g_system->updateScreen();
|
|
|
|
_needsErase = false;
|
|
}
|
|
|
|
void GraphicsManager::blit(const Graphics::Surface *surface, int x, int y) {
|
|
assert(surface->format.bytesPerPixel == _screen->format.bytesPerPixel);
|
|
|
|
for (int i = 0; i < surface->h; i++)
|
|
memcpy(_screen->getBasePtr(x, y + i), surface->getBasePtr(0, i), surface->w * surface->format.bytesPerPixel);
|
|
}
|
|
|
|
void GraphicsManager::blit(const Graphics::Surface *surface, int x, int y, uint width, uint height) {
|
|
assert(surface->format.bytesPerPixel == _screen->format.bytesPerPixel);
|
|
|
|
for (uint i = 0; i < height; i++)
|
|
memcpy(_screen->getBasePtr(x, y + i), surface->getBasePtr(0, i), width * surface->format.bytesPerPixel);
|
|
}
|
|
|
|
void GraphicsManager::blit(const Graphics::Surface *surface, const Common::Rect &srcRect, const Common::Rect &dstRect) {
|
|
assert(surface->format.bytesPerPixel == _screen->format.bytesPerPixel);
|
|
|
|
uint width = MIN(srcRect.width(), dstRect.width());
|
|
uint height = MIN(srcRect.height(), dstRect.height());
|
|
|
|
for (uint i = 0; i < height; i++)
|
|
memcpy(_screen->getBasePtr(dstRect.left, dstRect.top + i), surface->getBasePtr(srcRect.left, srcRect.top + i), width * surface->format.bytesPerPixel);
|
|
}
|
|
|
|
void GraphicsManager::fillRect(const Common::Rect &rect, uint32 color) {
|
|
_screen->fillRect(rect, color);
|
|
}
|
|
|
|
void GraphicsManager::keyBlit(Graphics::Surface *dst, int xDst, int yDst, int w, int h, const Graphics::Surface *src, uint xSrc, uint ySrc, uint32 transColor) {
|
|
assert(dst->format.bytesPerPixel == src->format.bytesPerPixel);
|
|
|
|
w = MIN<int>(src->w, w);
|
|
h = MIN<int>(src->h, h);
|
|
|
|
Common::Rect srcRect(xSrc, ySrc, xSrc + w, ySrc + h);
|
|
Common::Rect dstRect(xDst, yDst, xDst + w, yDst + h);
|
|
|
|
if (dst->clip(srcRect, dstRect))
|
|
dst->copyRectToSurfaceWithKey(*src, dstRect.left, dstRect.top, srcRect, transColor);
|
|
}
|
|
|
|
void GraphicsManager::keyBlit(Graphics::Surface *dst, int xDst, int yDst, int w, int h, const Graphics::Surface *src, uint xSrc, uint ySrc, byte rTrans, byte gTrans, byte bTrans) {
|
|
if (_vm->isTrueColor()) {
|
|
keyBlit(dst, xDst, yDst, w, h, src, xSrc, ySrc, getColor(rTrans, gTrans, bTrans));
|
|
} else {
|
|
// Find the palette index of the color
|
|
int paletteIndex = -1;
|
|
for (int i = 0; i < 256; i++) {
|
|
if (_palette[i * 3] == rTrans && _palette[i * 3 + 1] == gTrans && _palette[i * 3 + 2] == bTrans) {
|
|
paletteIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
assert(paletteIndex >= 0);
|
|
|
|
keyBlit(dst, xDst, yDst, w, h, src, xSrc, ySrc, paletteIndex);
|
|
}
|
|
}
|
|
|
|
void GraphicsManager::opaqueTransparentBlit(Graphics::Surface *dst, int xDst, int yDst, int w, int h, const Graphics::Surface *src, uint xSrc, uint ySrc, int opacityValue, byte rTrans, byte gTrans, byte bTrans) {
|
|
if (_vm->isTrueColor()) {
|
|
uint32 transColor = getColor(rTrans, gTrans, bTrans);
|
|
|
|
for (int y = 0; y < h; y++) {
|
|
if (y + yDst < dst->h && y + yDst >= 0) {
|
|
for (int x = 0; x < w; x++) {
|
|
if (x + xDst < dst->w && x + xDst >= 0) {
|
|
uint32 srcColor;
|
|
|
|
if (src->format.bytesPerPixel == 2)
|
|
srcColor = *((const uint16 *)src->getBasePtr(x + xSrc, y + ySrc));
|
|
else
|
|
srcColor = *((const uint32 *)src->getBasePtr(x + xSrc, y + ySrc));
|
|
|
|
if (srcColor == transColor)
|
|
continue;
|
|
|
|
int srcCycles, dstCycles;
|
|
switch (opacityValue) {
|
|
case 50:
|
|
srcCycles = 1;
|
|
dstCycles = 3;
|
|
break;
|
|
case 85:
|
|
srcCycles = 17;
|
|
dstCycles = 3;
|
|
break;
|
|
default:
|
|
srcCycles = 1;
|
|
dstCycles = 0;
|
|
break;
|
|
}
|
|
|
|
byte rSrc, gSrc, bSrc;
|
|
g_system->getScreenFormat().colorToRGB(srcColor, rSrc, gSrc, bSrc);
|
|
|
|
uint32 dstColor;
|
|
if (dst->format.bytesPerPixel == 2)
|
|
dstColor = *((uint16 *)dst->getBasePtr(x + xDst, y + yDst));
|
|
else
|
|
dstColor = *((uint32 *)dst->getBasePtr(x + xDst, y + yDst));
|
|
|
|
byte rDst, gDst, bDst;
|
|
g_system->getScreenFormat().colorToRGB(dstColor, rDst, gDst, bDst);
|
|
|
|
byte r = (((int)rSrc * srcCycles) + ((int)rDst * dstCycles)) / (srcCycles + dstCycles);
|
|
byte g = (((int)gSrc * srcCycles) + ((int)gDst * dstCycles)) / (srcCycles + dstCycles);
|
|
byte b = (((int)bSrc * srcCycles) + ((int)bDst * dstCycles)) / (srcCycles + dstCycles);
|
|
uint32 color = g_system->getScreenFormat().RGBToColor(r, g, b);
|
|
|
|
if (dst->format.bytesPerPixel == 2)
|
|
*((uint16 *)dst->getBasePtr(x + xDst, y + yDst)) = color;
|
|
else
|
|
*((uint32 *)dst->getBasePtr(x + xDst, y + yDst)) = color;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
keyBlit(dst, xDst, yDst, w, h, src, xSrc, ySrc, rTrans, gTrans, bTrans);
|
|
}
|
|
}
|
|
|
|
bool GraphicsManager::checkPointAgainstMaskedBitmap(const Graphics::Surface *bitmap, int x, int y, const Common::Point &point, byte rTrans, byte gTrans, byte bTrans) {
|
|
if (_vm->isTrueColor()) {
|
|
uint32 transColor = getColor(rTrans, gTrans, bTrans);
|
|
uint32 color;
|
|
|
|
if (bitmap->format.bytesPerPixel == 2)
|
|
color = *((const uint16 *)bitmap->getBasePtr(point.x - x, point.y - y));
|
|
else
|
|
color = *((const uint32 *)bitmap->getBasePtr(point.x - x, point.y - y));
|
|
|
|
return transColor != color;
|
|
} else {
|
|
// Find the palette index of the color
|
|
int paletteIndex = -1;
|
|
for (int i = 0; i < 256; i++) {
|
|
if (_palette[i * 3] == rTrans && _palette[i * 3 + 1] == gTrans && _palette[i * 3 + 2] == bTrans) {
|
|
paletteIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
assert(paletteIndex >= 0);
|
|
|
|
return *((const byte *)bitmap->getBasePtr(point.x - x, point.y - y)) != paletteIndex;
|
|
}
|
|
}
|
|
|
|
byte *GraphicsManager::createDefaultPalette() const {
|
|
Common::SeekableReadStream *stream = _vm->getBitmapStream(700);
|
|
|
|
if (!stream)
|
|
error("Couldn't find bitmap 700");
|
|
|
|
stream->skip(14);
|
|
|
|
if (stream->readUint16LE() != 8)
|
|
error("Trying to load palette from non-8bpp image 700");
|
|
|
|
stream->skip(16);
|
|
|
|
uint32 colorsUsed = stream->readUint32LE();
|
|
|
|
if (colorsUsed != 0 && colorsUsed != 256)
|
|
error("Bitmap 700 is missing a full palette");
|
|
|
|
stream->skip(4);
|
|
byte *palette = new byte[256 * 3];
|
|
byte *ptr = palette;
|
|
|
|
for (uint32 i = 0; i < 256; i++) {
|
|
ptr[2] = stream->readByte();
|
|
ptr[1] = stream->readByte();
|
|
ptr[0] = stream->readByte();
|
|
stream->readByte();
|
|
ptr += 3;
|
|
}
|
|
|
|
delete stream;
|
|
|
|
// Make sure the first entry is black and the last is white
|
|
palette[0 * 3] = palette[0 * 3 + 1] = palette[0 * 3 + 2] = 0x00;
|
|
palette[255 * 3] = palette[255 * 3 + 1] = palette[255 * 3 + 2] = 0xFF;
|
|
|
|
return palette;
|
|
}
|
|
|
|
Graphics::Surface *GraphicsManager::remapPalettedFrame(const Graphics::Surface *frame, const byte *palette) {
|
|
// This is pretty much the same as the Cinepak one
|
|
// It seems to work for the one video I know that needs it (SWLOGO.BTV)
|
|
// TODO: Merge some of this with getColor()
|
|
|
|
byte palMap[256];
|
|
const byte *screenPal = getDefaultPalette();
|
|
|
|
for (int i = 0; i < 256; i++) {
|
|
int r = palette[i * 3];
|
|
int g = palette[i * 3 + 1];
|
|
int b = palette[i * 3 + 2];
|
|
|
|
int diff = 0x7FFFFFFF;
|
|
byte result = 0;
|
|
|
|
for (int j = 0; j < 256; j++) {
|
|
int bDiff = b - (int)screenPal[j * 3 + 2];
|
|
int curDiffB = diff - (bDiff * bDiff);
|
|
|
|
if (curDiffB > 0) {
|
|
int gDiff = g - (int)screenPal[j * 3 + 1];
|
|
int curDiffG = curDiffB - (gDiff * gDiff);
|
|
|
|
if (curDiffG > 0) {
|
|
int rDiff = r - (int)screenPal[j * 3];
|
|
int curDiffR = curDiffG - (rDiff * rDiff);
|
|
|
|
if (curDiffR > 0) {
|
|
diff -= curDiffR;
|
|
result = j;
|
|
|
|
if (diff == 0)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
palMap[i] = result;
|
|
}
|
|
|
|
Graphics::Surface *convertedSurface = new Graphics::Surface();
|
|
convertedSurface->create(frame->w, frame->h, frame->format);
|
|
|
|
for (int y = 0; y < frame->h; y++) {
|
|
for (int x = 0; x < frame->w; x++)
|
|
*((byte *)convertedSurface->getBasePtr(x, y)) = palMap[*((const byte *)frame->getBasePtr(x, y))];
|
|
}
|
|
|
|
return convertedSurface;
|
|
}
|
|
|
|
int GraphicsManager::computeHPushOffset(int speed) {
|
|
switch (speed) {
|
|
case 3:
|
|
return 432;
|
|
case 2:
|
|
return 72;
|
|
case 1:
|
|
return 36;
|
|
case 0:
|
|
return 12;
|
|
}
|
|
|
|
return 432;
|
|
}
|
|
|
|
int GraphicsManager::computeVPushOffset(int speed) {
|
|
switch (speed) {
|
|
case 3:
|
|
return 189;
|
|
case 2:
|
|
return 63;
|
|
case 1:
|
|
return 21;
|
|
case 0:
|
|
return 7;
|
|
}
|
|
|
|
return 189;
|
|
}
|
|
|
|
void GraphicsManager::crossBlit(Graphics::Surface *dst, int xDst, int yDst, uint w, uint h, const Graphics::Surface *src, uint xSrc, uint ySrc) {
|
|
assert(dst->format.bytesPerPixel == src->format.bytesPerPixel);
|
|
|
|
for (uint y = 0; y < h; y++)
|
|
memcpy(dst->getBasePtr(xDst, yDst + y), src->getBasePtr(xSrc, ySrc + y), w * src->format.bytesPerPixel);
|
|
}
|
|
|
|
Graphics::Font *GraphicsManager::createMSGothicFont(int size, bool bold) const {
|
|
switch (size) {
|
|
case 10:
|
|
case 11:
|
|
size = 8;
|
|
break;
|
|
case 12:
|
|
size = 9;
|
|
break;
|
|
case 20:
|
|
size = 16;
|
|
break;
|
|
default:
|
|
error("Unknown MS Gothic font size %d", size);
|
|
}
|
|
|
|
Graphics::Font *font;
|
|
|
|
// Try to see if the user supplied a font
|
|
Common::SeekableReadStream *stream = SearchMan.createReadStreamForMember("msgothic.ttc");
|
|
|
|
// TODO: Fake a bold version
|
|
if (stream) {
|
|
// Force monochrome, since the original uses the bitmap glyphs in the font
|
|
font = Graphics::loadTTFFont(*stream, size, Graphics::kTTFSizeModeCharacter, 96, 96, Graphics::kTTFRenderModeMonochrome);
|
|
} else {
|
|
font = Graphics::loadTTFFontFromArchive("VL-Gothic-Regular.ttf", size, Graphics::kTTFSizeModeCharacter, 96, 96, Graphics::kTTFRenderModeMonochrome);
|
|
}
|
|
|
|
if (!font)
|
|
error("Failed to load MS Gothic font");
|
|
|
|
delete stream;
|
|
return font;
|
|
}
|
|
|
|
void GraphicsManager::renderText(Graphics::Surface *dst, Graphics::Font *font, const Common::String &text, int x, int y, int w, int h, uint32 color, int lineHeight, TextAlign textAlign, bool centerVertically) {
|
|
if (text.empty())
|
|
return;
|
|
|
|
// Convert to UTF-32 for drawing. Choose the codepage based on the language.
|
|
Common::CodePage srcFormat = (_vm->getLanguage() == Common::JA_JPN) ? Common::kWindows932 : Common::kWindows1252;
|
|
Common::U32String convString = text.decode(srcFormat);
|
|
|
|
renderText(dst, font, convString, x, y, w, h, color, lineHeight, textAlign, centerVertically);
|
|
}
|
|
|
|
void GraphicsManager::renderText(Graphics::Surface *dst, Graphics::Font *font, const Common::U32String &text, int x, int y, int w, int h, uint32 color, int lineHeight, TextAlign textAlign, bool centerVertically) {
|
|
if (text.empty())
|
|
return;
|
|
|
|
Common::U32StringArray lines;
|
|
font->wordWrapText(text, w, lines);
|
|
|
|
Graphics::TextAlign align = Graphics::kTextAlignLeft;
|
|
switch (textAlign) {
|
|
case kTextAlignLeft:
|
|
align = Graphics::kTextAlignLeft;
|
|
break;
|
|
case kTextAlignCenter:
|
|
align = Graphics::kTextAlignCenter;
|
|
break;
|
|
case kTextAlignRight:
|
|
align = Graphics::kTextAlignRight;
|
|
break;
|
|
}
|
|
|
|
if (centerVertically)
|
|
y += (h - (lines.size() * lineHeight)) / 2;
|
|
|
|
// Why is this needed? Dunno, but I guess Windows adds one row of space on the top
|
|
y++;
|
|
|
|
for (uint32 i = 0; i < lines.size(); i++) {
|
|
font->drawString(dst, lines[i], x, y, w, color, align);
|
|
y += lineHeight;
|
|
}
|
|
}
|
|
|
|
void GraphicsManager::drawEllipse(const Common::Rect &rect, uint32 color) {
|
|
// HACK: This just hardcodes the sizes of the rows of the ellipses
|
|
// for the one thing in the game that needs it.
|
|
|
|
static const int rows7[7] = { 7, 13, 15, 15, 15, 13, 7 };
|
|
static const int rows10[10] = { 7, 11, 13, 15, 15, 15, 15, 13, 11, 7 };
|
|
static const int rows12[12] = { 7, 11, 13, 13, 15, 15, 15, 15, 13, 13, 11, 7 };
|
|
static const int rows15[15] = { 5, 9, 11, 13, 13, 15, 15, 15, 15, 15, 13, 13, 11, 9, 5 };
|
|
|
|
const int *table = nullptr;
|
|
switch (rect.height()) {
|
|
case 7:
|
|
table = rows7;
|
|
break;
|
|
case 10:
|
|
table = rows10;
|
|
break;
|
|
case 12:
|
|
table = rows12;
|
|
break;
|
|
case 15:
|
|
table = rows15;
|
|
break;
|
|
}
|
|
|
|
assert(table);
|
|
|
|
for (int y = 0; y < rect.height(); y++) {
|
|
int width = table[y];
|
|
int x = rect.left + (rect.width() - width) / 2;
|
|
_screen->hLine(x, y + rect.top, x + width, color);
|
|
}
|
|
}
|
|
|
|
TempCursorChange::TempCursorChange(Cursor cursor) {
|
|
_prevCursor = static_cast<BuriedEngine*>(g_engine)->_gfx->setCursor(cursor);
|
|
}
|
|
|
|
TempCursorChange::~TempCursorChange() {
|
|
static_cast<BuriedEngine*>(g_engine)->_gfx->setCursor(_prevCursor);
|
|
}
|
|
|
|
} // End of namespace Buried
|