2004-04-12 21:40:49 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2005-01-01 16:20:17 +00:00
|
|
|
* Copyright (C) 2004-2005 The ScummVM project
|
2004-04-12 21:40:49 +00:00
|
|
|
*
|
|
|
|
* The ReInherit Engine is (C)2000-2003 by Daniel Balsom.
|
|
|
|
*
|
|
|
|
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2004-05-01 07:32:48 +00:00
|
|
|
// Misc. graphics routines
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-06-25 21:47:37 +00:00
|
|
|
#include "saga/saga.h"
|
|
|
|
#include "saga/gfx.h"
|
2005-01-20 13:59:12 +00:00
|
|
|
#include "saga/interface.h"
|
2004-06-25 21:47:37 +00:00
|
|
|
|
|
|
|
#include "common/system.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
namespace Saga {
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
Gfx::Gfx(SagaEngine *vm, OSystem *system, int width, int height, GameDetector &detector) : _vm(vm), _system(system) {
|
2004-11-24 00:14:21 +00:00
|
|
|
_system->beginGFXTransaction();
|
|
|
|
_vm->initCommonGFX(detector);
|
|
|
|
_system->initSize(width, height);
|
|
|
|
_system->endGFXTransaction();
|
2004-05-02 00:00:39 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(5, "Init screen %dx%d", width, height);
|
2004-07-31 12:39:26 +00:00
|
|
|
// Convert surface data to R surface data
|
2005-07-09 16:23:45 +00:00
|
|
|
_backBuffer.create(width, height, 1);
|
2004-05-02 00:00:39 +00:00
|
|
|
|
|
|
|
// Set module data
|
2004-08-02 11:27:50 +00:00
|
|
|
_init = 1;
|
2004-05-02 15:44:19 +00:00
|
|
|
|
|
|
|
// For now, always show the mouse cursor.
|
2005-01-15 13:41:57 +00:00
|
|
|
setCursor();
|
2004-11-24 00:14:21 +00:00
|
|
|
_system->showMouse(true);
|
2004-05-02 00:00:39 +00:00
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
|
|
|
|
Gfx::~Gfx() {
|
|
|
|
_backBuffer.free();
|
2004-05-02 00:00:39 +00:00
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
|
|
|
|
void Surface::drawPalette() {
|
2004-04-12 21:40:49 +00:00
|
|
|
int x;
|
|
|
|
int y;
|
|
|
|
int color = 0;
|
2005-07-09 16:23:45 +00:00
|
|
|
Rect palRect;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
for (y = 0; y < 16; y++) {
|
2005-07-09 16:23:45 +00:00
|
|
|
palRect.top = (y * 8) + 4;
|
|
|
|
palRect.bottom = palRect.top + 8;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
for (x = 0; x < 16; x++) {
|
2005-07-09 16:23:45 +00:00
|
|
|
palRect.left = (x * 8) + 4;
|
|
|
|
palRect.right = palRect.left + 8;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
drawRect(palRect, color);
|
2004-04-12 21:40:49 +00:00
|
|
|
color++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-01 07:32:48 +00:00
|
|
|
// * Copies a rectangle from a raw 8 bit pixel buffer to the specified surface.
|
|
|
|
// - The surface must match the logical dimensions of the buffer exactly.
|
2005-07-09 16:23:45 +00:00
|
|
|
void Surface::blit(const Common::Rect &destRect, const byte *sourceBuffer) {
|
|
|
|
const byte *readPointer;
|
|
|
|
byte *writePointer;
|
|
|
|
int row;
|
|
|
|
ClipData clipData;
|
|
|
|
|
|
|
|
clipData.sourceRect.left = 0;
|
|
|
|
clipData.sourceRect.top = 0;
|
|
|
|
clipData.sourceRect.right = destRect.width();
|
|
|
|
clipData.sourceRect.bottom = destRect.height();
|
|
|
|
|
|
|
|
clipData.destPoint.x = destRect.left;
|
|
|
|
clipData.destPoint.y = destRect.top;
|
|
|
|
clipData.destRect.left = 0;
|
|
|
|
clipData.destRect.right = w;
|
|
|
|
clipData.destRect.top = 0;
|
|
|
|
clipData.destRect.bottom = h;
|
|
|
|
|
|
|
|
if (!clipData.calcClip()) {
|
|
|
|
return;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2005-07-09 16:23:45 +00:00
|
|
|
|
2004-05-01 07:32:48 +00:00
|
|
|
// Transfer buffer data to surface
|
2005-07-09 16:23:45 +00:00
|
|
|
readPointer = (sourceBuffer + clipData.drawSource.x) +
|
|
|
|
(clipData.sourceRect.right * clipData.drawSource.y);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
writePointer = ((byte *)pixels + clipData.drawDest.x) + (pitch * clipData.drawDest.y);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
for (row = 0; row < clipData.drawHeight; row++) {
|
|
|
|
memcpy(writePointer, readPointer, clipData.drawWidth);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
writePointer += pitch;
|
|
|
|
readPointer += clipData.sourceRect.right;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
void Surface::drawPolyLine(const Point *points, int count, int color) {
|
|
|
|
int i;
|
|
|
|
if (count >= 3) {
|
|
|
|
for (i = 1; i < count; i++) {
|
|
|
|
drawLine(points[i].x, points[i].y, points[i - 1].x, points[i - 1].y, color);
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
drawLine(points[count - 1].x, points[count - 1].y, points->x, points->y, color);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-09 17:11:41 +00:00
|
|
|
/**
|
|
|
|
* Dissolve one image with another.
|
|
|
|
* If flags if set to 1, do zero masking.
|
|
|
|
*/
|
|
|
|
void Surface::transitionDissolve(const byte *sourceBuffer, const Common::Rect &sourceRect, int flags, double percent) {
|
|
|
|
#define XOR_MASK 0xB400;
|
|
|
|
int pixelcount = w * h;
|
|
|
|
int seqlimit = (int)(65535 * percent);
|
|
|
|
int seq = 1;
|
|
|
|
int i, x1, y1;
|
|
|
|
byte color;
|
|
|
|
|
|
|
|
for (i = 0; i < seqlimit; i++) {
|
|
|
|
if (seq & 1) {
|
|
|
|
seq = (seq >> 1) ^ XOR_MASK;
|
|
|
|
} else {
|
|
|
|
seq = seq >> 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (seq == 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (seq >= pixelcount) {
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
x1 = seq % w;
|
|
|
|
y1 = seq / w;
|
|
|
|
|
|
|
|
if (sourceRect.contains(x1, y1)) {
|
|
|
|
color = sourceBuffer[(x1-sourceRect.left) + sourceRect.width()*(y1-sourceRect.top)];
|
|
|
|
if (flags == 0 || color)
|
|
|
|
((byte*)pixels)[seq] = color;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
void Gfx::setPalette(PalEntry *pal) {
|
2004-05-02 00:00:39 +00:00
|
|
|
int i;
|
|
|
|
byte *ppal;
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
for (i = 0, ppal = _currentPal; i < PAL_ENTRIES; i++, ppal += 4) {
|
2005-01-27 20:07:04 +00:00
|
|
|
ppal[0] = pal[i].red;
|
|
|
|
ppal[1] = pal[i].green;
|
|
|
|
ppal[2] = pal[i].blue;
|
2004-05-02 00:00:39 +00:00
|
|
|
ppal[3] = 0;
|
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
_system->setPalette(_currentPal, 0, PAL_ENTRIES);
|
2004-05-02 00:00:39 +00:00
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
void Gfx::getCurrentPal(PalEntry *src_pal) {
|
2004-05-02 00:00:39 +00:00
|
|
|
int i;
|
|
|
|
byte *ppal;
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
for (i = 0, ppal = _currentPal; i < PAL_ENTRIES; i++, ppal += 4) {
|
2004-05-02 00:00:39 +00:00
|
|
|
src_pal[i].red = ppal[0];
|
|
|
|
src_pal[i].green = ppal[1];
|
|
|
|
src_pal[i].blue = ppal[2];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
void Gfx::palToBlack(PalEntry *src_pal, double percent) {
|
2004-05-02 00:00:39 +00:00
|
|
|
int i;
|
|
|
|
//int fade_max = 255;
|
|
|
|
int new_entry;
|
|
|
|
byte *ppal;
|
|
|
|
|
|
|
|
double fpercent;
|
|
|
|
|
|
|
|
if (percent > 1.0) {
|
|
|
|
percent = 1.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exponential fade
|
|
|
|
fpercent = percent * percent;
|
|
|
|
|
|
|
|
fpercent = 1.0 - fpercent;
|
|
|
|
|
|
|
|
// Use the correct percentage change per frame for each palette entry
|
2005-07-09 16:23:45 +00:00
|
|
|
for (i = 0, ppal = _currentPal; i < PAL_ENTRIES; i++, ppal += 4) {
|
2004-05-02 00:00:39 +00:00
|
|
|
new_entry = (int)(src_pal[i].red * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[0] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[0] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_entry = (int)(src_pal[i].green * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[1] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[1] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_entry = (int)(src_pal[i].blue * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[2] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[2] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
ppal[3] = 0;
|
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
_system->setPalette(_currentPal, 0, PAL_ENTRIES);
|
2004-05-02 00:00:39 +00:00
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
void Gfx::blackToPal(PalEntry *src_pal, double percent) {
|
2004-05-02 00:00:39 +00:00
|
|
|
int new_entry;
|
|
|
|
double fpercent;
|
|
|
|
int color_delta;
|
|
|
|
int best_wdelta = 0;
|
|
|
|
int best_windex = 0;
|
|
|
|
int best_bindex = 0;
|
|
|
|
int best_bdelta = 1000;
|
|
|
|
byte *ppal;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (percent > 1.0) {
|
|
|
|
percent = 1.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exponential fade
|
|
|
|
fpercent = percent * percent;
|
|
|
|
|
|
|
|
fpercent = 1.0 - fpercent;
|
|
|
|
|
|
|
|
// Use the correct percentage change per frame for each palette entry
|
2005-07-09 16:23:45 +00:00
|
|
|
for (i = 0, ppal = _currentPal; i < PAL_ENTRIES; i++, ppal += 4) {
|
2004-05-02 00:00:39 +00:00
|
|
|
new_entry = (int)(src_pal[i].red - src_pal[i].red * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[0] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[0] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_entry = (int)(src_pal[i].green - src_pal[i].green * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[1] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[1] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_entry = (int)(src_pal[i].blue - src_pal[i].blue * fpercent);
|
|
|
|
|
|
|
|
if (new_entry < 0) {
|
|
|
|
ppal[2] = 0;
|
|
|
|
} else {
|
|
|
|
ppal[2] = (byte) new_entry;
|
|
|
|
}
|
|
|
|
ppal[3] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the best white and black color indices again
|
|
|
|
if (percent >= 1.0) {
|
2005-07-09 16:23:45 +00:00
|
|
|
for (i = 0, ppal = _currentPal; i < PAL_ENTRIES; i++, ppal += 4) {
|
2004-05-02 00:00:39 +00:00
|
|
|
color_delta = ppal[0];
|
|
|
|
color_delta += ppal[1];
|
|
|
|
color_delta += ppal[2];
|
|
|
|
|
|
|
|
if (color_delta < best_bdelta) {
|
|
|
|
best_bindex = i;
|
|
|
|
best_bdelta = color_delta;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (color_delta > best_wdelta) {
|
|
|
|
best_windex = i;
|
|
|
|
best_wdelta = color_delta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
_system->setPalette(_currentPal, 0, PAL_ENTRIES);
|
2004-05-02 00:00:39 +00:00
|
|
|
}
|
|
|
|
|
2004-11-20 00:05:50 +00:00
|
|
|
void Gfx::showCursor(bool state) {
|
|
|
|
updateCursor();
|
|
|
|
g_system->showMouse(state);
|
|
|
|
}
|
|
|
|
|
2005-01-15 13:41:57 +00:00
|
|
|
void Gfx::setCursor() {
|
2004-05-02 16:32:28 +00:00
|
|
|
// Set up the mouse cursor
|
2005-01-20 13:59:12 +00:00
|
|
|
const byte A = kITEColorLightGrey;
|
|
|
|
const byte B = kITEColorWhite;
|
|
|
|
|
|
|
|
const byte cursor_img[CURSOR_W * CURSOR_H] = {
|
|
|
|
0, 0, 0, A, 0, 0, 0,
|
|
|
|
0, 0, 0, A, 0, 0, 0,
|
|
|
|
0, 0, 0, A, 0, 0, 0,
|
2005-01-21 23:18:32 +00:00
|
|
|
A, A, A, B, A, A, A,
|
2005-01-20 13:59:12 +00:00
|
|
|
0, 0, 0, A, 0, 0, 0,
|
|
|
|
0, 0, 0, A, 0, 0, 0,
|
|
|
|
0, 0, 0, A, 0, 0, 0,
|
2004-05-02 16:32:28 +00:00
|
|
|
};
|
|
|
|
|
2005-01-15 13:41:57 +00:00
|
|
|
_system->setMouseCursor(cursor_img, CURSOR_W, CURSOR_H, 3, 3, 0);
|
2004-05-02 16:32:28 +00:00
|
|
|
}
|
|
|
|
|
2004-10-30 22:34:08 +00:00
|
|
|
bool hitTestPoly(const Point *points, unsigned int npoints, const Point& test_point) {
|
2004-10-08 01:22:39 +00:00
|
|
|
int yflag0;
|
|
|
|
int yflag1;
|
|
|
|
bool inside_flag = false;
|
|
|
|
unsigned int pt;
|
|
|
|
|
2004-10-08 19:58:49 +00:00
|
|
|
const Point *vtx0 = &points[npoints - 1];
|
|
|
|
const Point *vtx1 = &points[0];
|
2004-10-08 01:22:39 +00:00
|
|
|
|
|
|
|
yflag0 = (vtx0->y >= test_point.y);
|
|
|
|
for (pt = 0; pt < npoints; pt++, vtx1++) {
|
|
|
|
yflag1 = (vtx1->y >= test_point.y);
|
|
|
|
if (yflag0 != yflag1) {
|
|
|
|
if (((vtx1->y - test_point.y) * (vtx0->x - vtx1->x) >=
|
|
|
|
(vtx1->x - test_point.x) * (vtx0->y - vtx1->y)) == yflag1) {
|
|
|
|
inside_flag = !inside_flag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
yflag0 = yflag1;
|
|
|
|
vtx0 = vtx1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return inside_flag;
|
|
|
|
}
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
} // End of namespace Saga
|