scummvm/engines/director/images.cpp

300 lines
7.8 KiB
C++
Raw Normal View History

/* 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.
*
*/
#include "common/substream.h"
#include "image/codecs/bmp_raw.h"
#include "director/director.h"
2016-08-21 11:04:18 +02:00
#include "director/images.h"
namespace Director {
DIBDecoder::DIBDecoder() {
_surface = 0;
_palette = 0;
_paletteColorCount = 0;
_codec = 0;
}
DIBDecoder::~DIBDecoder() {
destroy();
}
void DIBDecoder::destroy() {
_surface = 0; // It is deleted by BitmapRawDecoder
delete[] _palette;
_palette = 0;
_paletteColorCount = 0;
delete _codec;
_codec = 0;
}
void DIBDecoder::loadPalette(Common::SeekableReadStream &stream) {
2016-06-16 15:31:46 +02:00
uint16 steps = stream.size() / 6;
2016-05-18 04:07:14 +03:00
uint16 index = (steps * 3) - 1;
2016-06-10 13:35:46 +03:00
_paletteColorCount = steps;
2016-08-17 11:39:42 +02:00
_palette = new byte[index + 1];
2016-06-16 15:31:46 +02:00
2016-05-17 01:34:55 +03:00
for (uint8 i = 0; i < steps; i++) {
2016-05-20 23:33:55 +03:00
_palette[index - 2] = stream.readByte();
2016-05-17 01:34:55 +03:00
stream.readByte();
2016-05-20 23:33:55 +03:00
_palette[index - 1] = stream.readByte();
2016-05-17 01:34:55 +03:00
stream.readByte();
2016-05-20 23:33:55 +03:00
_palette[index] = stream.readByte();
2016-05-17 01:34:55 +03:00
stream.readByte();
2016-05-20 23:33:55 +03:00
index -= 3;
}
}
bool DIBDecoder::loadStream(Common::SeekableReadStream &stream) {
uint32 headerSize = stream.readUint32LE();
if (headerSize != 40)
return false;
uint32 width = stream.readUint32LE();
uint32 height = stream.readUint32LE();
stream.readUint16LE(); // planes
uint16 bitsPerPixel = stream.readUint16LE();
2016-05-19 00:04:22 +02:00
uint32 compression = stream.readUint32BE();
2016-08-23 08:14:39 +02:00
/* uint32 imageSize = */ stream.readUint32LE();
/* uint32 pixelsPerMeterX = */ stream.readUint32LE();
/* uint32 pixelsPerMeterY = */ stream.readUint32LE();
_paletteColorCount = stream.readUint32LE();
/* uint32 colorsImportant = */ stream.readUint32LE();
_paletteColorCount = (_paletteColorCount == 0) ? 255: _paletteColorCount;
2016-05-19 00:04:22 +02:00
Common::SeekableSubReadStream subStream(&stream, 40, stream.size());
2016-05-18 04:07:14 +03:00
_codec = Image::createBitmapCodec(compression, 0, width, height, bitsPerPixel);
2016-06-16 15:31:46 +02:00
if (!_codec)
return false;
2016-06-16 15:31:46 +02:00
2016-05-18 04:07:14 +03:00
_surface = _codec->decodeFrame(subStream);
2016-06-16 15:31:46 +02:00
return true;
}
/****************************
2019-12-24 18:22:43 +01:00
* BITD
****************************/
BITDDecoder::BITDDecoder(int w, int h, uint16 bitsPerPixel, uint16 pitch, const byte *palette) {
_surface = new Graphics::Surface();
if (pitch < w) {
warning("BITDDecoder: pitch is too small: %d < %d", pitch, w);
pitch = w;
}
Graphics::PixelFormat pf = Graphics::PixelFormat::createFormatCLUT8();
switch (bitsPerPixel) {
case 2:
break;
case 4:
break;
case 8:
break;
case 16:
break;
case 32:
//pf = Graphics::PixelFormat::PixelFormat(bitsPerPixel / 8, 8, 8, 8, 8, 24, 16, 8, 0);
break;
default:
break;
}
2017-01-13 08:32:31 +01:00
// HACK: Create a padded surface by adjusting w after create()
_surface->create(pitch, h, pf);
_surface->w = w;
_palette = palette;
// TODO: Bring this in from the main surface?
_paletteColorCount = 255;
_bitsPerPixel = bitsPerPixel;
}
2019-12-24 18:22:43 +01:00
BITDDecoder::~BITDDecoder() {
destroy();
}
2019-12-24 18:22:43 +01:00
void BITDDecoder::destroy() {
_surface->free();
2020-01-06 13:24:02 +01:00
delete _surface;
_surface = 0;
_paletteColorCount = 0;
}
2019-12-24 18:22:43 +01:00
void BITDDecoder::loadPalette(Common::SeekableReadStream &stream) {
// no op
}
void BITDDecoder::convertPixelIntoSurface(void* surfacePointer, uint fromBpp, uint toBpp, int red, int green, int blue) {
// Initial implementation of 32-bit images to palettised sprites.
switch (fromBpp) {
case 4:
switch (toBpp) {
case 1:
if (red == 255 && blue == 255 && green == 255) {
*((byte*)surfacePointer) = 255;
} else if (red == 0 && blue == 0 && green == 0) {
*((byte*)surfacePointer) = 0;
} else {
2020-04-11 13:00:41 +02:00
for (byte p = 0; p < _paletteColorCount; p++) {
if (_palette[p * 3 + 0] == red &&
_palette[p * 3 + 1] == green &&
2020-04-11 13:00:41 +02:00
_palette[p * 3 + 2] == blue) {
*((byte*)surfacePointer) = p;
}
}
}
break;
default:
warning("BITDDecoder::convertPixelIntoSurface(): conversion from %d to %d not implemented",
fromBpp, toBpp);
}
break;
default:
warning("BITDDecoder::convertPixelIntoSurface(): could not convert from %d to %d",
fromBpp, toBpp);
break;
}
}
2019-12-24 18:22:43 +01:00
bool BITDDecoder::loadStream(Common::SeekableReadStream &stream) {
int x = 0, y = 0;
// If the stream has exactly the required number of bits for this image,
// we assume it is uncompressed.
if (stream.size() * 8 == _surface->pitch * _surface->h) {
2017-02-14 18:57:53 +01:00
debugC(6, kDebugImages, "Skipping compression");
for (y = 0; y < _surface->h; y++) {
for (x = 0; x < _surface->pitch; ) {
byte color = stream.readByte();
for (int c = 0; c < 8; c++)
*((byte *)_surface->getBasePtr(x++, y)) = (color & (1 << (7 - c))) ? 0 : 0xff;
}
}
return true;
}
Common::Array<int> pixels;
while (!stream.eos()) {
// TODO: D3 32-bit bitmap casts seem to just be ARGB pixels in a row and not RLE.
// Determine how to distinguish these different types. Maybe stage version.
if (_bitsPerPixel == 32) {
int data = stream.readByte();
pixels.push_back(data);
} else {
int data = stream.readByte();
int len = data + 1;
if ((data & 0x80) != 0) {
len = ((data ^ 0xFF) & 0xff) + 2;
data = stream.readByte();
for (int p = 0; p < len; p++) {
pixels.push_back(data);
}
} else {
for (int p = 0; p < len; p++) {
data = stream.readByte();
pixels.push_back(data);
}
}
if (_bitsPerPixel == 32 && pixels.size() % (_surface->w * 3) == 0)
stream.readUint16BE();
}
}
2020-01-29 23:40:22 +02:00
if (pixels.size() < (uint32)_surface->w * _surface->h) {
int tail = (_surface->w * _surface->h * _bitsPerPixel / 8) - pixels.size();
warning("BITDDecoder::loadStream(): premature end of stream (%d of %d pixels)",
pixels.size(), pixels.size() + tail);
for (int i = 0; i < tail; i++)
pixels.push_back(0);
}
int offset = 0;
2017-02-11 20:18:17 +01:00
if (_surface->w < (pixels.size() / _surface->h))
offset = (pixels.size() / _surface->h) - _surface->w;
if (pixels.size() > 0) {
for (y = 0; y < _surface->h; y++) {
for (x = 0; x < _surface->w;) {
switch (_bitsPerPixel) {
case 1:
2019-12-31 16:37:51 +01:00
for (int c = 0; c < 8 && x < _surface->w; c++, x++) {
*((byte *)_surface->getBasePtr(x, y)) = (pixels[(((y * _surface->pitch) + x) / 8)] & (1 << (7 - c))) ? 0 : 0xff;
}
break;
case 8:
2017-02-11 20:18:17 +01:00
// this calculation is wrong.. need a demo with colours.
*((byte *)_surface->getBasePtr(x, y)) = g_director->transformColor(pixels[(y * _surface->w) + x + (y * offset)]);
2017-01-13 08:32:31 +01:00
x++;
break;
case 16:
2017-01-13 08:32:31 +01:00
*((uint16*)_surface->getBasePtr(x, y)) = _surface->format.RGBToColor(
(pixels[((y * _surface->w) * 2) + x] & 0x7c) << 1,
(pixels[((y * _surface->w) * 2) + x] & 0x03) << 6 |
(pixels[((y * _surface->w) * 2) + (_surface->w) + x] & 0xe0) >> 2,
(pixels[((y * _surface->w) * 2) + (_surface->w) + x] & 0x1f) << 3);
2017-01-13 08:32:31 +01:00
x++;
break;
case 32:
convertPixelIntoSurface(_surface->getBasePtr(x, y),
(_bitsPerPixel / 8),
_surface->format.bytesPerPixel,
pixels[(((y * (_surface->w * 4))) + ((x * 4) + 1))],
pixels[(((y * (_surface->w * 4))) + ((x * 4) + 2))],
pixels[(((y * (_surface->w * 4))) + ((x * 4) + 3))]);
2017-01-13 08:32:31 +01:00
x++;
break;
default:
x++;
break;
}
}
}
}
return true;
}
} // End of namespace Director