2008-09-02 20:15:42 +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.
|
|
|
|
*
|
|
|
|
* 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/endian.h"
|
2008-12-17 16:06:16 +00:00
|
|
|
#include "common/savefile.h"
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2010-09-30 13:03:22 +00:00
|
|
|
#include "graphics/conversion.h"
|
|
|
|
|
2008-09-02 20:15:42 +00:00
|
|
|
#include "gob/gob.h"
|
|
|
|
#include "gob/video.h"
|
2008-12-14 04:33:28 +00:00
|
|
|
#include "gob/util.h"
|
2009-04-24 21:30:40 +00:00
|
|
|
#include "gob/draw.h"
|
|
|
|
#include "gob/global.h"
|
2008-09-02 20:15:42 +00:00
|
|
|
|
|
|
|
namespace Gob {
|
|
|
|
|
|
|
|
Video_v6::Video_v6(GobEngine *vm) : Video_v2(vm) {
|
|
|
|
}
|
|
|
|
|
|
|
|
char Video_v6::spriteUncompressor(byte *sprBuf, int16 srcWidth, int16 srcHeight,
|
2010-09-30 13:02:16 +00:00
|
|
|
int16 x, int16 y, int16 transp, Surface &destDesc) {
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2008-12-11 21:57:47 +00:00
|
|
|
if ((sprBuf[0] == 1) && (sprBuf[1] == 3)) {
|
|
|
|
drawPacked(sprBuf, x, y, destDesc);
|
|
|
|
return 1;
|
|
|
|
}
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2009-08-22 13:11:30 +00:00
|
|
|
if (srcWidth & 0xC000) {
|
|
|
|
warning("Playtoons Stub: srcWidth & 0xC000 == %04X", srcWidth & 0xC000);
|
|
|
|
srcWidth &= 0x3FFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sprBuf[0] == 1) && (sprBuf[1] == 2)) {
|
|
|
|
if (Video_v2::spriteUncompressor(sprBuf, srcWidth, srcHeight, x, y, transp, destDesc))
|
|
|
|
return 1;
|
|
|
|
|
2010-09-30 13:02:16 +00:00
|
|
|
Video::drawPacked(sprBuf, srcWidth, srcHeight, x, y, transp, destDesc);
|
2009-08-22 13:11:30 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
warning("Urban Stub: spriteUncompressor(), sprBuf[0,1,2] = %d,%d,%d",
|
|
|
|
sprBuf[0], sprBuf[1], sprBuf[2]);
|
2008-12-11 21:57:47 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2010-09-30 13:02:16 +00:00
|
|
|
void Video_v6::drawPacked(const byte *sprBuf, int16 x, int16 y, Surface &surfDesc) {
|
2008-12-11 21:57:47 +00:00
|
|
|
const byte *data = sprBuf + 2;
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2008-12-11 21:57:47 +00:00
|
|
|
int16 width = READ_LE_UINT16(data);
|
|
|
|
int16 height = READ_LE_UINT16(data + 2);
|
|
|
|
data += 4;
|
2008-09-02 20:15:42 +00:00
|
|
|
|
2008-12-11 21:57:47 +00:00
|
|
|
const byte *srcData = data;
|
|
|
|
byte *uncBuf = 0;
|
|
|
|
|
2008-12-12 00:10:39 +00:00
|
|
|
if (*srcData++ != 0) {
|
2008-12-11 21:57:47 +00:00
|
|
|
uint32 size = READ_LE_UINT32(data);
|
|
|
|
|
|
|
|
uncBuf = new byte[size];
|
|
|
|
|
2009-04-27 18:56:28 +00:00
|
|
|
//spriteUncompressor(data, buf);
|
|
|
|
warning("Urban Stub: drawPacked: spriteUncompressor(data, uncBuf)");
|
2008-12-11 21:57:47 +00:00
|
|
|
|
|
|
|
srcData = uncBuf;
|
2008-09-02 20:15:42 +00:00
|
|
|
}
|
|
|
|
|
2008-12-11 21:57:47 +00:00
|
|
|
drawYUVData(srcData, surfDesc, width, height, x, y);
|
|
|
|
|
|
|
|
delete[] uncBuf;
|
2008-09-02 20:15:42 +00:00
|
|
|
}
|
|
|
|
|
2010-09-30 13:02:16 +00:00
|
|
|
void Video_v6::drawYUVData(const byte *srcData, Surface &destDesc,
|
2008-12-11 21:57:47 +00:00
|
|
|
int16 width, int16 height, int16 x, int16 y) {
|
|
|
|
|
|
|
|
int16 dataWidth = width;
|
|
|
|
int16 dataHeight = height;
|
|
|
|
|
|
|
|
if (dataWidth & 0xF)
|
|
|
|
dataWidth = (dataWidth & 0xFFF0) + 16;
|
|
|
|
if (dataHeight & 0xF)
|
|
|
|
dataHeight = (dataHeight & 0xFFF0) + 16;
|
|
|
|
|
|
|
|
const byte *dataY = srcData;
|
|
|
|
const byte *dataU = dataY + (dataWidth * dataHeight);
|
|
|
|
const byte *dataV = dataU + ((dataWidth * dataHeight) >> 4);
|
|
|
|
|
|
|
|
drawYUV(destDesc, x, y, dataWidth, dataHeight, width, height, dataY, dataU, dataV);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-09-30 13:02:16 +00:00
|
|
|
void Video_v6::drawYUV(Surface &destDesc, int16 x, int16 y,
|
2008-12-11 21:57:47 +00:00
|
|
|
int16 dataWidth, int16 dataHeight, int16 width, int16 height,
|
|
|
|
const byte *dataY, const byte *dataU, const byte *dataV) {
|
|
|
|
|
2010-09-30 13:03:22 +00:00
|
|
|
const Graphics::PixelFormat &pixelFormat = _vm->getPixelFormat();
|
2008-12-11 21:57:47 +00:00
|
|
|
|
2009-06-06 20:03:13 +00:00
|
|
|
if ((x + width - 1) >= destDesc.getWidth())
|
|
|
|
width = destDesc.getWidth() - x;
|
|
|
|
if ((y + height - 1) >= destDesc.getHeight())
|
|
|
|
height = destDesc.getHeight() - y;
|
2008-12-11 21:57:47 +00:00
|
|
|
|
2010-09-30 13:03:22 +00:00
|
|
|
Pixel dst = destDesc.get(x, y);
|
2008-12-11 21:57:47 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < height; i++) {
|
2010-09-30 13:03:22 +00:00
|
|
|
Pixel dstRow = dst;
|
|
|
|
|
2011-01-18 00:56:48 +00:00
|
|
|
int nextChromaLine = (i < ((height - 1) & ~3) ) ? dataWidth : 0;
|
2011-06-19 22:59:48 +00:00
|
|
|
|
2010-12-02 21:40:17 +00:00
|
|
|
for (int j = 0; j < width; j++, dstRow++) {
|
2011-01-18 00:56:48 +00:00
|
|
|
|
|
|
|
int nextChromaColumn = (j < ((width - 1) & ~3)) ? 1 : 0;
|
|
|
|
|
|
|
|
// Get (7bit) Y data. It is at full res, does not need to be interpolated
|
|
|
|
byte dY = dataY[j] << 1;
|
|
|
|
|
|
|
|
// do linear interpolation on chroma values (7bits)
|
|
|
|
// to avoid blockiness
|
|
|
|
byte dU0 = dataU[j >> 2];
|
|
|
|
byte dV0 = dataV[j >> 2];
|
|
|
|
|
|
|
|
byte dU1 = dataU[(j >> 2) + nextChromaColumn];
|
|
|
|
byte dV1 = dataV[(j >> 2) + nextChromaColumn];
|
|
|
|
|
|
|
|
byte dU2 = dataU[(j + nextChromaLine) >> 2];
|
|
|
|
byte dV2 = dataV[(j + nextChromaLine) >> 2];
|
|
|
|
|
|
|
|
byte dU3 = dataU[((j + nextChromaLine) >> 2) + nextChromaColumn];
|
|
|
|
byte dV3 = dataV[((j + nextChromaLine) >> 2) + nextChromaColumn];
|
|
|
|
|
|
|
|
byte tX = j & 3;
|
|
|
|
byte tY = i & 3;
|
|
|
|
byte invtX = 4 - tX;
|
|
|
|
byte invtY = 4 - tY;
|
|
|
|
|
|
|
|
int16 dUX1 = dU0 * invtX + dU1 * tX;
|
|
|
|
int16 dUX2 = dU2 * invtX + dU3 * tX;
|
|
|
|
byte dU = (dUX1 * invtY + dUX2 * tY) >> 3;
|
|
|
|
|
|
|
|
int16 dVY1 = dV0 * invtX + dV1 * tX;
|
|
|
|
int16 dVY2 = dV2 * invtX + dV3 * tX;
|
|
|
|
byte dV = (dVY1 * invtY + dVY2 * tY) >> 3;
|
2008-12-11 21:57:47 +00:00
|
|
|
|
2010-12-02 21:40:17 +00:00
|
|
|
byte r, g, b;
|
|
|
|
Graphics::YUV2RGB(dY, dU, dV, r, g, b);
|
2008-12-11 21:57:47 +00:00
|
|
|
|
2010-12-02 21:40:17 +00:00
|
|
|
if (dY != 0) {
|
|
|
|
// Solid pixel
|
|
|
|
uint32 c = pixelFormat.RGBToColor(r, g, b);
|
2010-09-30 13:03:22 +00:00
|
|
|
|
2010-12-02 21:40:17 +00:00
|
|
|
// If the solid pixel's value is 0, we'll fudge it to 1
|
|
|
|
dstRow.set((c == 0) ? 1 : c);
|
|
|
|
} else
|
|
|
|
// Transparent pixel, we'll use pixel value 0
|
|
|
|
dstRow.set(0);
|
2010-09-30 13:03:22 +00:00
|
|
|
|
2008-12-11 21:57:47 +00:00
|
|
|
}
|
|
|
|
|
2010-12-02 21:40:17 +00:00
|
|
|
dst += destDesc.getWidth();
|
|
|
|
dataY += dataWidth;
|
|
|
|
|
|
|
|
if ((i & 3) == 3) {
|
|
|
|
// Next line of chroma data
|
|
|
|
dataU += dataWidth >> 2;
|
|
|
|
dataV += dataWidth >> 2;
|
|
|
|
}
|
2008-12-11 21:57:47 +00:00
|
|
|
}
|
2008-12-14 04:33:28 +00:00
|
|
|
|
2009-01-01 15:06:43 +00:00
|
|
|
}
|
2008-12-11 21:57:47 +00:00
|
|
|
|
2008-09-02 20:15:42 +00:00
|
|
|
} // End of namespace Gob
|