scummvm/engines/cge/vga13h.cpp

1019 lines
22 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.
*
*/
/*
* This code is based on original Soltys source code
* Copyright (c) 1994-1995 Janus B. Wisniewski and L.K. Avalon
*/
#include "common/array.h"
#include "common/rect.h"
#include "graphics/palette.h"
2011-06-13 11:57:24 +02:00
#include "cge/general.h"
#include "cge/vga13h.h"
#include "cge/bitmap.h"
#include "cge/text.h"
#include "cge/cge_main.h"
#include "cge/cge.h"
2011-06-10 22:57:09 +02:00
namespace CGE {
Seq *getConstantSeq(bool seqFlag) {
const Seq seq1[] = { { 0, 0, 0, 0, 0 } };
const Seq seq2[] = { { 0, 1, 0, 0, 0 }, { 1, 0, 0, 0, 0 } };
Seq *seq;
if (seqFlag) {
seq = (Seq *)malloc(1 * sizeof(Seq));
*seq = seq1[0];
} else {
seq = (Seq *)malloc(2 * sizeof(Seq));
seq[0] = seq2[0];
seq[1] = seq2[1];
}
return seq;
}
Sprite::Sprite(CGEEngine *vm, BitmapPtr *shpP)
2011-07-02 01:02:14 +02:00
: _x(0), _y(0), _z(0), _nearPtr(0), _takePtr(0),
2011-08-21 11:15:28 +02:00
_next(NULL), _prev(NULL), _seqPtr(kNoSeq), _time(0),
2011-09-14 08:15:30 +02:00
_ext(NULL), _ref(-1), _scene(0), _vm(vm) {
2011-07-02 01:02:14 +02:00
memset(_file, 0, sizeof(_file));
memset(&_flags, 0, sizeof(_flags));
2011-07-05 22:06:26 +10:00
_ref = 0;
_x = _y = 0;
_w = _h = 0;
_time = 0;
_seqPtr = 0;
_shpCnt = 0;
_prev = _next = NULL;
2011-07-03 14:55:49 +10:00
setShapeList(shpP);
}
Sprite::~Sprite() {
if (_sprite == this)
_sprite = NULL;
2011-07-02 01:02:14 +02:00
contract();
}
BitmapPtr Sprite::shp() {
2011-08-19 23:54:24 +02:00
SprExt *e = _ext;
if (!e || !e->_seq)
return NULL;
int i = e->_seq[_seqPtr]._now;
if (i >= _shpCnt)
2011-08-19 23:54:24 +02:00
error("Invalid PHASE in SPRITE::Shp() %s", _file);
return e->_shpList[i];
}
BitmapPtr *Sprite::setShapeList(BitmapPtr *shpP) {
BitmapPtr *r = (_ext) ? _ext->_shpList : NULL;
2011-06-29 16:13:17 +02:00
_shpCnt = 0;
_w = 0;
_h = 0;
2011-07-03 14:55:49 +10:00
if (shpP) {
BitmapPtr *p;
2011-07-03 14:55:49 +10:00
for (p = shpP; *p; p++) {
BitmapPtr b = (*p); // ->Code();
2011-06-29 16:13:17 +02:00
if (b->_w > _w)
_w = b->_w;
if (b->_h > _h)
_h = b->_h;
_shpCnt++;
2011-06-13 11:57:24 +02:00
}
2011-07-02 01:02:14 +02:00
expand();
2011-07-03 14:55:49 +10:00
_ext->_shpList = shpP;
_flags._bDel = true;
if (!_ext->_seq)
setSeq(getConstantSeq(_shpCnt < 2));
}
2011-06-13 11:57:24 +02:00
return r;
}
2011-07-02 01:02:14 +02:00
void Sprite::moveShapes(uint8 *buf) {
BitmapPtr *p;
for (p = _ext->_shpList; *p; p++) {
2011-06-30 08:30:23 +02:00
buf += (*p)->moveVmap(buf);
}
}
2011-07-02 01:02:14 +02:00
bool Sprite::works(Sprite *spr) {
2011-08-19 23:54:24 +02:00
if (!spr || !spr->_ext)
return false;
Snail::Com *c = spr->_ext->_take;
if (c != NULL) {
c += spr->_takePtr;
if (c->_ref == _ref)
if (c->_com != kSnLabel || (c->_val == 0 || c->_val == _vm->_now))
return true;
}
2011-06-13 11:57:24 +02:00
return false;
}
2011-07-02 01:02:14 +02:00
Seq *Sprite::setSeq(Seq *seq) {
if (_ext) {
free(_ext->_seq);
_ext->_seq = NULL;
}
2011-07-02 01:02:14 +02:00
expand();
2011-08-19 23:54:24 +02:00
Seq *s = _ext->_seq;
_ext->_seq = seq;
2011-08-21 11:15:28 +02:00
if (_seqPtr == kNoSeq)
2011-07-02 01:02:14 +02:00
step(0);
2011-06-29 16:13:17 +02:00
else if (_time == 0)
2011-07-02 01:02:14 +02:00
step(_seqPtr);
2011-06-13 11:57:24 +02:00
return s;
}
2011-07-02 01:02:14 +02:00
bool Sprite::seqTest(int n) {
2011-06-13 13:07:45 +02:00
if (n >= 0)
2011-06-29 16:13:17 +02:00
return (_seqPtr == n);
if (_ext)
2011-07-03 11:28:22 +02:00
return (_ext->_seq[_seqPtr]._next == _seqPtr);
2011-06-13 11:57:24 +02:00
return true;
}
2011-07-13 08:44:58 +02:00
Snail::Com *Sprite::snList(SnList type) {
2011-08-19 23:54:24 +02:00
SprExt *e = _ext;
2011-06-13 13:07:45 +02:00
if (e)
2011-07-13 08:44:58 +02:00
return (type == kNear) ? e->_near : e->_take;
2011-06-13 11:57:24 +02:00
return NULL;
}
void Sprite::setName(char *newName) {
2011-08-19 23:54:24 +02:00
if (!_ext)
return;
if (_ext->_name) {
delete[] _ext->_name;
_ext->_name = NULL;
}
if (newName) {
_ext->_name = new char[strlen(newName) + 1];
assert(_ext->_name != NULL);
strcpy(_ext->_name, newName);
}
}
2011-07-02 01:02:14 +02:00
Sprite *Sprite::expand() {
2011-08-19 23:54:24 +02:00
if (_ext)
return this;
_ext = new SprExt;
assert(_ext != NULL);
if (!*_file)
return this;
static const char *Comd[] = { "Name", "Phase", "Seq", "Near", "Take", NULL };
2011-09-11 22:13:00 +02:00
char fname[kPathMax];
2011-08-19 23:54:24 +02:00
Common::Array<BitmapPtr> shplist;
2011-08-21 11:15:28 +02:00
for (int i = 0; i < _shpCnt + 1; ++i)
shplist.push_back(NULL);
2011-08-19 23:54:24 +02:00
Seq *seq = NULL;
2011-09-11 22:13:00 +02:00
int shapeCount = 0,
seqCount = 0,
nearCount = 0,
takeCount = 0,
2011-08-19 23:54:24 +02:00
maxnow = 0,
maxnxt = 0;
Snail::Com *nearList = NULL;
Snail::Com *takeList = NULL;
_vm->mergeExt(fname, _file, kSprExt);
2011-09-14 00:10:59 +02:00
if (_resman->exist(fname)) { // sprite description file exist
2011-09-11 22:13:00 +02:00
EncryptedStream sprf(fname);
if (sprf.err())
2011-08-19 23:54:24 +02:00
error("Bad SPR [%s]", fname);
2011-09-11 22:13:00 +02:00
Common::String line;
char tmpStr[kLineMax + 1];
2011-08-19 23:54:24 +02:00
int len = 0, lcnt = 0;
2011-09-11 22:13:00 +02:00
for (line = sprf.readLine(); !sprf.eos(); line = sprf.readLine()) {
len = line.size();
strcpy(tmpStr, line.c_str());
2011-08-19 23:54:24 +02:00
lcnt++;
2011-09-11 22:13:00 +02:00
if (len == 0 || *tmpStr == '.')
2011-08-19 23:54:24 +02:00
continue;
Snail::Com *c;
switch (_vm->takeEnum(Comd, strtok(tmpStr, " =\t"))) {
2011-08-19 23:54:24 +02:00
case 0:
// Name
setName(strtok(NULL, ""));
break;
case 1:
// Phase
// In case the shape index gets too high, increase the array size
2011-09-11 22:13:00 +02:00
while ((shapeCount + 1) >= (int)shplist.size()) {
2011-08-19 23:54:24 +02:00
shplist.push_back(NULL);
++_shpCnt;
}
2011-09-16 23:54:08 +02:00
shplist[shapeCount++] = new Bitmap(_vm, strtok(NULL, " \t,;/"));
2011-08-19 23:54:24 +02:00
break;
case 2:
// Seq
2011-09-11 22:13:00 +02:00
seq = (Seq *)realloc(seq, (seqCount + 1) * sizeof(*seq));
2011-08-19 23:54:24 +02:00
assert(seq != NULL);
Seq *s;
2011-09-11 22:13:00 +02:00
s = &seq[seqCount++];
2011-08-19 23:54:24 +02:00
s->_now = atoi(strtok(NULL, " \t,;/"));
if (s->_now > maxnow)
maxnow = s->_now;
s->_next = atoi(strtok(NULL, " \t,;/"));
switch (s->_next) {
case 0xFF:
2011-09-11 22:13:00 +02:00
s->_next = seqCount;
2011-08-19 23:54:24 +02:00
break;
case 0xFE:
2011-09-11 22:13:00 +02:00
s->_next = seqCount - 1;
2011-06-13 11:57:24 +02:00
break;
}
2011-08-19 23:54:24 +02:00
if (s->_next > maxnxt)
maxnxt = s->_next;
s->_dx = atoi(strtok(NULL, " \t,;/"));
s->_dy = atoi(strtok(NULL, " \t,;/"));
s->_dly = atoi(strtok(NULL, " \t,;/"));
break;
case 3:
// Near
2011-08-21 11:15:28 +02:00
if (_nearPtr == kNoPtr)
2011-08-19 23:54:24 +02:00
break;
nearList = (Snail::Com *)realloc(nearList, (nearCount + 1) * sizeof(*nearList));
assert(nearList != NULL);
c = &nearList[nearCount++];
if ((c->_com = (SnCom)_vm->takeEnum(Snail::_comText, strtok(NULL, " \t,;/"))) < 0)
2011-08-19 23:54:24 +02:00
error("Bad NEAR in %d [%s]", lcnt, fname);
c->_ref = atoi(strtok(NULL, " \t,;/"));
c->_val = atoi(strtok(NULL, " \t,;/"));
c->_ptr = NULL;
break;
case 4:
// Take
2011-08-21 11:15:28 +02:00
if (_takePtr == kNoPtr)
2011-08-19 23:54:24 +02:00
break;
takeList = (Snail::Com *)realloc(takeList, (takeCount + 1) * sizeof(*takeList));
assert(takeList != NULL);
c = &takeList[takeCount++];
if ((c->_com = (SnCom)_vm->takeEnum(Snail::_comText, strtok(NULL, " \t,;/"))) < 0)
2011-08-19 23:54:24 +02:00
error("Bad NEAR in %d [%s]", lcnt, fname);
c->_ref = atoi(strtok(NULL, " \t,;/"));
c->_val = atoi(strtok(NULL, " \t,;/"));
c->_ptr = NULL;
break;
2011-06-13 11:57:24 +02:00
}
}
2011-08-19 23:54:24 +02:00
} else {
// no sprite description: try to read immediately from .BMP
2011-09-16 23:54:08 +02:00
shplist[shapeCount++] = new Bitmap(_vm, _file);
}
2011-08-19 23:54:24 +02:00
2011-09-11 22:13:00 +02:00
shplist[shapeCount] = NULL;
2011-08-19 23:54:24 +02:00
if (seq) {
2011-09-11 22:13:00 +02:00
if (maxnow >= shapeCount)
2011-08-19 23:54:24 +02:00
error("Bad PHASE in SEQ [%s]", fname);
2011-09-11 22:13:00 +02:00
if (maxnxt >= seqCount)
2011-08-19 23:54:24 +02:00
error("Bad JUMP in SEQ [%s]", fname);
setSeq(seq);
} else
setSeq(getConstantSeq(_shpCnt == 1));
// Set the shape list
BitmapPtr *shapeList = new BitmapPtr[shplist.size()];
for (uint i = 0; i < shplist.size(); ++i)
shapeList[i] = shplist[i];
setShapeList(shapeList);
if (nearList)
nearList[nearCount - 1]._ptr = _ext->_near = nearList;
2011-08-19 23:54:24 +02:00
else
2011-08-21 11:15:28 +02:00
_nearPtr = kNoPtr;
if (takeList)
takeList[takeCount - 1]._ptr = _ext->_take = takeList;
2011-08-19 23:54:24 +02:00
else
2011-08-21 11:15:28 +02:00
_takePtr = kNoPtr;
2011-08-19 23:54:24 +02:00
2011-06-13 11:57:24 +02:00
return this;
}
2011-07-02 01:02:14 +02:00
Sprite *Sprite::contract() {
2011-08-19 23:54:24 +02:00
SprExt *e = _ext;
if (!e)
return this;
if (e->_name)
delete[] e->_name;
if (_flags._bDel && e->_shpList) {
for (int i = 0; e->_shpList[i]; i++)
delete e->_shpList[i];
delete[] e->_shpList;
}
2011-08-19 23:54:24 +02:00
free(e->_seq);
free(e->_near);
free(e->_take);
delete e;
_ext = NULL;
2011-06-13 11:57:24 +02:00
return this;
}
2011-07-02 01:02:14 +02:00
Sprite *Sprite::backShow(bool fast) {
expand();
show(2);
show(1);
2011-06-13 13:07:45 +02:00
if (fast)
2011-07-02 01:02:14 +02:00
show(0);
contract();
2011-06-13 11:57:24 +02:00
return this;
}
2011-07-02 01:02:14 +02:00
void Sprite::step(int nr) {
2011-06-13 13:07:45 +02:00
if (nr >= 0)
2011-06-29 16:13:17 +02:00
_seqPtr = nr;
if (_ext) {
Seq *seq;
2011-06-13 13:07:45 +02:00
if (nr < 0)
2011-07-03 11:28:22 +02:00
_seqPtr = _ext->_seq[_seqPtr]._next;
2011-06-29 16:13:17 +02:00
seq = _ext->_seq + _seqPtr;
2011-07-03 11:28:22 +02:00
if (seq->_dly >= 0) {
gotoxy(_x + (seq->_dx), _y + (seq->_dy));
_time = seq->_dly;
2011-06-13 11:57:24 +02:00
}
}
}
2011-07-02 01:02:14 +02:00
void Sprite::tick() {
step();
}
2011-07-02 01:02:14 +02:00
void Sprite::makeXlat(uint8 *x) {
2011-08-19 23:54:24 +02:00
if (!_ext)
return;
2011-08-19 23:54:24 +02:00
if (_flags._xlat)
killXlat();
for (BitmapPtr *b = _ext->_shpList; *b; b++)
(*b)->_m = x;
_flags._xlat = true;
}
2011-07-02 01:02:14 +02:00
void Sprite::killXlat() {
2011-08-19 23:54:24 +02:00
if (!_flags._xlat || !_ext)
return;
2011-08-19 23:54:24 +02:00
uint8 *m = (*_ext->_shpList)->_m;
free(m);
for (BitmapPtr *b = _ext->_shpList; *b; b++)
(*b)->_m = NULL;
_flags._xlat = false;
}
2011-07-02 01:02:14 +02:00
void Sprite::gotoxy(int x, int y) {
2011-06-29 16:13:17 +02:00
int xo = _x, yo = _y;
if (_x < kScrWidth) {
2011-06-13 13:07:45 +02:00
if (x < 0)
2011-06-13 11:57:24 +02:00
x = 0;
if (x + _w > kScrWidth)
x = (kScrWidth - _w);
2011-06-29 16:13:17 +02:00
_x = x;
2011-06-13 11:57:24 +02:00
}
if (_h < kScrHeight) {
2011-06-13 13:07:45 +02:00
if (y < 0)
2011-06-13 11:57:24 +02:00
y = 0;
if (y + _h > kScrHeight)
y = (kScrHeight - _h);
2011-06-29 16:13:17 +02:00
_y = y;
2011-06-13 11:57:24 +02:00
}
2011-06-29 16:13:17 +02:00
if (_next)
if (_next->_flags._slav)
2011-07-02 01:02:14 +02:00
_next->gotoxy(_next->_x - xo + _x, _next->_y - yo + _y);
if (_flags._shad)
2011-07-02 01:02:14 +02:00
_prev->gotoxy(_prev->_x - xo + _x, _prev->_y - yo + _y);
}
2011-07-02 01:02:14 +02:00
void Sprite::center() {
gotoxy((kScrWidth - _w) / 2, (kScrHeight - _h) / 2);
}
2011-07-02 01:02:14 +02:00
void Sprite::show() {
2011-08-19 23:54:24 +02:00
SprExt *e;
e = _ext;
e->_x0 = e->_x1;
e->_y0 = e->_y1;
e->_b0 = e->_b1;
2011-06-29 16:13:17 +02:00
e->_x1 = _x;
e->_y1 = _y;
2011-07-02 01:02:14 +02:00
e->_b1 = shp();
if (!_flags._hide) {
if (_flags._xlat)
2011-06-30 08:30:23 +02:00
e->_b1->xShow(e->_x1, e->_y1);
else
2011-06-30 08:30:23 +02:00
e->_b1->show(e->_x1, e->_y1);
2011-06-13 11:57:24 +02:00
}
}
2011-07-02 01:02:14 +02:00
void Sprite::show(uint16 pg) {
2011-09-16 23:54:08 +02:00
Graphics::Surface *a = _vm->_vga->_page[1];
_vm->_vga->_page[1] = _vm->_vga->_page[pg & 3];
2011-07-02 01:02:14 +02:00
shp()->show(_x, _y);
2011-09-16 23:54:08 +02:00
_vm->_vga->_page[1] = a;
}
2011-07-02 01:02:14 +02:00
void Sprite::hide() {
2011-08-19 23:54:24 +02:00
SprExt *e = _ext;
if (e->_b0)
2011-06-30 08:30:23 +02:00
e->_b0->hide(e->_x0, e->_y0);
}
BitmapPtr Sprite::ghost() {
2011-08-19 23:54:24 +02:00
SprExt *e = _ext;
if (!e->_b1)
return NULL;
2011-09-16 23:54:08 +02:00
BitmapPtr bmp = new Bitmap(_vm, 0, 0, (uint8 *)NULL);
2011-08-19 23:54:24 +02:00
assert(bmp != NULL);
bmp->_w = e->_b1->_w;
bmp->_h = e->_b1->_h;
bmp->_b = new HideDesc[bmp->_h];
assert(bmp->_b != NULL);
bmp->_v = (uint8 *) memcpy(bmp->_b, e->_b1->_b, sizeof(HideDesc) * bmp->_h);
bmp->_map = (e->_y1 << 16) + e->_x1;
return bmp;
}
void Sprite::sync(Common::Serializer &s) {
uint16 unused = 0;
s.syncAsUint16LE(unused);
s.syncAsUint16LE(unused); // _ext
s.syncAsUint16LE(_ref);
2011-09-14 08:15:30 +02:00
s.syncAsByte(_scene);
// bitfield in-memory storage is unpredictable, so to avoid
// any issues, pack/unpack everything manually
uint16 flags = 0;
if (s.isLoading()) {
s.syncAsUint16LE(flags);
_flags._hide = flags & 0x0001 ? true : false;
_flags._near = flags & 0x0002 ? true : false;
_flags._drag = flags & 0x0004 ? true : false;
_flags._hold = flags & 0x0008 ? true : false;
_flags._____ = flags & 0x0010 ? true : false;
_flags._slav = flags & 0x0020 ? true : false;
_flags._syst = flags & 0x0040 ? true : false;
_flags._kill = flags & 0x0080 ? true : false;
_flags._xlat = flags & 0x0100 ? true : false;
_flags._port = flags & 0x0200 ? true : false;
_flags._kept = flags & 0x0400 ? true : false;
_flags._east = flags & 0x0800 ? true : false;
_flags._shad = flags & 0x1000 ? true : false;
_flags._back = flags & 0x2000 ? true : false;
_flags._bDel = flags & 0x4000 ? true : false;
_flags._tran = flags & 0x8000 ? true : false;
} else {
flags = (flags << 1) | _flags._tran;
flags = (flags << 1) | _flags._bDel;
flags = (flags << 1) | _flags._back;
flags = (flags << 1) | _flags._shad;
flags = (flags << 1) | _flags._east;
flags = (flags << 1) | _flags._kept;
flags = (flags << 1) | _flags._port;
flags = (flags << 1) | _flags._xlat;
flags = (flags << 1) | _flags._kill;
flags = (flags << 1) | _flags._syst;
flags = (flags << 1) | _flags._slav;
flags = (flags << 1) | _flags._____;
flags = (flags << 1) | _flags._hold;
flags = (flags << 1) | _flags._drag;
flags = (flags << 1) | _flags._near;
flags = (flags << 1) | _flags._hide;
s.syncAsUint16LE(flags);
}
s.syncAsUint16LE(_x);
s.syncAsUint16LE(_y);
s.syncAsByte(_z);
s.syncAsUint16LE(_w);
s.syncAsUint16LE(_h);
s.syncAsUint16LE(_time);
s.syncAsByte(_nearPtr);
s.syncAsByte(_takePtr);
s.syncAsSint16LE(_seqPtr);
s.syncAsUint16LE(_shpCnt);
s.syncBytes((byte *)&_file[0], 9);
_file[8] = '\0';
s.syncAsUint16LE(unused); // _prev
s.syncAsUint16LE(unused); // _next
}
2011-07-03 11:28:22 +02:00
Queue::Queue(bool show) : _head(NULL), _tail(NULL), _show(show) {
}
2011-07-03 11:28:22 +02:00
Queue::~Queue() {
clear();
}
2011-07-03 11:28:22 +02:00
void Queue::clear() {
while (_head) {
Sprite *s = remove(_head);
if (s->_flags._kill)
2011-06-13 11:57:24 +02:00
delete s;
}
}
2011-07-03 11:28:22 +02:00
void Queue::forAll(void (*fun)(Sprite *)) {
Sprite *s = _head;
2011-06-13 11:57:24 +02:00
while (s) {
2011-06-29 16:13:17 +02:00
Sprite *n = s->_next;
2011-06-13 11:57:24 +02:00
fun(s);
s = n;
}
}
2011-07-03 11:28:22 +02:00
void Queue::append(Sprite *spr) {
if (_tail) {
spr->_prev = _tail;
_tail->_next = spr;
2011-06-13 13:07:45 +02:00
} else
2011-07-03 11:28:22 +02:00
_head = spr;
_tail = spr;
if (_show)
2011-07-02 01:02:14 +02:00
spr->expand();
2011-06-13 13:07:45 +02:00
else
2011-07-02 01:02:14 +02:00
spr->contract();
2011-06-13 11:57:24 +02:00
}
2011-07-03 11:28:22 +02:00
void Queue::insert(Sprite *spr, Sprite *nxt) {
if (nxt == _head) {
spr->_next = _head;
_head = spr;
if (!_tail)
_tail = spr;
2011-06-13 11:57:24 +02:00
} else {
assert(nxt);
2011-06-29 16:13:17 +02:00
spr->_next = nxt;
spr->_prev = nxt->_prev;
if (spr->_prev)
spr->_prev->_next = spr;
2011-06-13 11:57:24 +02:00
}
2011-06-29 16:13:17 +02:00
if (spr->_next)
spr->_next->_prev = spr;
2011-07-03 11:28:22 +02:00
if (_show)
2011-07-02 01:02:14 +02:00
spr->expand();
2011-06-13 13:07:45 +02:00
else
2011-07-02 01:02:14 +02:00
spr->contract();
2011-06-13 11:57:24 +02:00
}
2011-07-03 11:28:22 +02:00
void Queue::insert(Sprite *spr) {
Sprite *s;
2011-07-03 11:28:22 +02:00
for (s = _head; s; s = s->_next)
2011-06-29 16:13:17 +02:00
if (s->_z > spr->_z)
2011-06-13 11:57:24 +02:00
break;
2011-06-13 13:07:45 +02:00
if (s)
2011-07-03 11:28:22 +02:00
insert(spr, s);
2011-06-13 13:07:45 +02:00
else
2011-07-03 11:28:22 +02:00
append(spr);
if (_show)
2011-07-02 01:02:14 +02:00
spr->expand();
2011-06-13 13:07:45 +02:00
else
2011-07-02 01:02:14 +02:00
spr->contract();
2011-06-13 11:57:24 +02:00
}
template<typename T>
inline bool contains(const Common::List<T> &l, const T &v) {
return (Common::find(l.begin(), l.end(), v) != l.end());
}
2011-06-13 11:57:24 +02:00
2011-07-03 11:28:22 +02:00
Sprite *Queue::remove(Sprite *spr) {
if (spr == _head)
_head = spr->_next;
if (spr == _tail)
_tail = spr->_prev;
2011-06-29 16:13:17 +02:00
if (spr->_next)
spr->_next->_prev = spr->_prev;
if (spr->_prev)
spr->_prev->_next = spr->_next;
spr->_prev = NULL;
spr->_next = NULL;
2011-06-13 11:57:24 +02:00
return spr;
}
2011-07-03 11:28:22 +02:00
Sprite *Queue::locate(int ref) {
2011-08-19 23:54:24 +02:00
for (Sprite *spr = _head; spr; spr = spr->_next) {
if (spr->_ref == ref)
2011-06-13 11:57:24 +02:00
return spr;
}
2011-06-13 11:57:24 +02:00
return NULL;
}
Vga::Vga() : _frmCnt(0), _msg(NULL), _name(NULL), _setPal(false), _mono(0) {
_oldColors = NULL;
_newColors = NULL;
_showQ = new Queue(true);
_spareQ = new Queue(false);
_sysPal = new Dac[kPalCount];
for (int idx = 0; idx < 4; idx++) {
2011-07-03 16:22:26 +02:00
_page[idx] = new Graphics::Surface();
_page[idx]->create(320, 200, Graphics::PixelFormat::createFormatCLUT8());
}
#if 0
// This part was used to display credits at the beginning of the game
for (int i = 10; i < 20; i++) {
char *text = _text->getText(i);
if (text) {
debugN(1, "%s\n", text);
2011-06-13 11:57:24 +02:00
}
}
#endif
_oldColors = (Dac *)malloc(sizeof(Dac) * kPalCount);
_newColors = (Dac *)malloc(sizeof(Dac) * kPalCount);
2011-07-12 08:02:18 +02:00
getColors(_oldColors);
sunset();
setColors();
clear(0);
}
2011-07-03 16:22:26 +02:00
Vga::~Vga() {
_mono = 0;
2011-07-12 08:02:18 +02:00
Common::String buffer = "";
/*
2011-07-12 08:02:18 +02:00
clear(0);
setMode(_oldMode);
setColors();
restoreScreen(_oldScreen);
sunrise(_oldColors);
*/
2011-07-12 08:02:18 +02:00
free(_oldColors);
free(_newColors);
if (_msg)
buffer = Common::String(_msg);
if (_name)
buffer = buffer + " [" + _name + "]";
2011-07-12 08:02:18 +02:00
debugN("%s", buffer.c_str());
2011-07-03 17:14:47 +10:00
2011-07-03 11:28:22 +02:00
delete _showQ;
delete _spareQ;
delete[] _sysPal;
for (int idx = 0; idx < 4; idx++) {
_page[idx]->free();
delete _page[idx];
}
}
2011-08-21 11:15:28 +02:00
void Vga::waitVR() {
// Since some of the game parts rely on using vertical sync as a delay mechanism,
// we're introducing a short delay to simulate it
g_system->delayMillis(5);
}
2011-07-03 16:22:26 +02:00
void Vga::getColors(Dac *tab) {
byte palData[kPalSize];
g_system->getPaletteManager()->grabPalette(palData, 0, kPalCount);
2011-07-03 16:22:26 +02:00
palToDac(palData, tab);
}
2011-09-16 07:55:50 +02:00
uint8 Vga::closest(Dac *pal, const uint8 colR, const uint8 colG, const uint8 colB) {
#define f(col, lum) ((((uint16)(col)) << 8) / lum)
uint16 i, dif = 0xFFFF, found = 0;
uint16 L = colR + colG + colB;
if (!L)
L++;
uint16 R = f(colR, L), G = f(colG, L), B = f(colB, L);
for (i = 0; i < 256; i++) {
uint16 l = pal[i]._r + pal[i]._g + pal[i]._b;
if (!l)
l++;
int r = f(pal[i]._r, l), g = f(pal[i]._g, l), b = f(pal[i]._b, l);
uint16 D = ((r > R) ? (r - R) : (R - r)) +
((g > G) ? (g - G) : (G - g)) +
((b > B) ? (b - B) : (B - b)) +
((l > L) ? (l - L) : (L - l)) * 10 ;
if (D < dif) {
found = i;
dif = D;
if (D == 0)
break; // exact!
}
}
return found;
#undef f
}
uint8 *Vga::glass(Dac *pal, const uint8 colR, const uint8 colG, const uint8 colB) {
uint8 *x = (uint8 *)malloc(256);
if (x) {
uint16 i;
for (i = 0; i < 256; i++) {
x[i] = closest(pal, ((uint16)(pal[i]._r) * colR) / 255,
((uint16)(pal[i]._g) * colG) / 255,
((uint16)(pal[i]._b) * colB) / 255);
}
}
return x;
}
2011-07-03 16:22:26 +02:00
void Vga::palToDac(const byte *palData, Dac *tab) {
const byte *colP = palData;
for (int idx = 0; idx < kPalCount; idx++, colP += 3) {
tab[idx]._r = *colP >> 2;
tab[idx]._g = *(colP + 1) >> 2;
tab[idx]._b = *(colP + 2) >> 2;
}
}
2011-07-03 16:22:26 +02:00
void Vga::dacToPal(const Dac *tab, byte *palData) {
for (int idx = 0; idx < kPalCount; idx++, palData += 3) {
2011-07-01 08:37:40 +02:00
*palData = tab[idx]._r << 2;
*(palData + 1) = tab[idx]._g << 2;
*(palData + 2) = tab[idx]._b << 2;
}
}
2011-07-03 16:22:26 +02:00
void Vga::setColors(Dac *tab, int lum) {
Dac *palP = tab, *destP = _newColors;
for (int idx = 0; idx < kPalCount; idx++, palP++, destP++) {
destP->_r = (palP->_r * lum) >> 6;
destP->_g = (palP->_g * lum) >> 6;
destP->_b = (palP->_b * lum) >> 6;
}
2011-07-03 16:22:26 +02:00
if (_mono) {
destP = _newColors;
for (int idx = 0; idx < kPalCount; idx++, destP++) {
// Form a greyscalce colour from 30% R, 59% G, 11% B
uint8 intensity = (((int)destP->_r * 77) + ((int)destP->_g * 151) + ((int)destP->_b * 28)) >> 8;
destP->_r = intensity;
destP->_g = intensity;
destP->_b = intensity;
}
}
2011-07-03 16:22:26 +02:00
_setPal = true;
}
void Vga::setColors() {
memset(_newColors, 0, kPalSize);
2011-07-03 16:22:26 +02:00
updateColors();
}
2011-07-03 16:22:26 +02:00
void Vga::sunrise(Dac *tab) {
2011-08-21 11:15:28 +02:00
for (int i = 0; i <= 64; i += kFadeStep) {
2011-07-03 16:22:26 +02:00
setColors(tab, i);
2011-08-21 11:15:28 +02:00
waitVR();
2011-07-03 16:22:26 +02:00
updateColors();
2011-06-13 11:57:24 +02:00
}
}
2011-07-03 16:22:26 +02:00
void Vga::sunset() {
2011-07-01 08:37:40 +02:00
Dac tab[256];
2011-07-03 16:22:26 +02:00
getColors(tab);
2011-08-21 11:15:28 +02:00
for (int i = 64; i >= 0; i -= kFadeStep) {
2011-07-03 16:22:26 +02:00
setColors(tab, i);
2011-08-21 11:15:28 +02:00
waitVR();
2011-07-03 16:22:26 +02:00
updateColors();
2011-06-13 11:57:24 +02:00
}
}
2011-07-03 16:22:26 +02:00
void Vga::show() {
2011-08-19 23:54:24 +02:00
for (Sprite *spr = _showQ->first(); spr; spr = spr->_next)
2011-07-02 01:02:14 +02:00
spr->show();
2011-07-03 16:22:26 +02:00
update();
2011-08-19 23:54:24 +02:00
for (Sprite *spr = _showQ->first(); spr; spr = spr->_next)
2011-07-02 01:02:14 +02:00
spr->hide();
2011-07-03 16:22:26 +02:00
_frmCnt++;
}
2011-07-03 16:22:26 +02:00
void Vga::updateColors() {
byte palData[kPalSize];
2011-07-03 16:22:26 +02:00
dacToPal(_newColors, palData);
g_system->getPaletteManager()->setPalette(palData, 0, 256);
}
2011-07-03 16:22:26 +02:00
void Vga::update() {
SWAP(Vga::_page[0], Vga::_page[1]);
2011-07-03 16:22:26 +02:00
if (_setPal) {
updateColors();
_setPal = false;
2011-06-13 11:57:24 +02:00
}
g_system->copyRectToScreen((const byte *)Vga::_page[0]->getBasePtr(0, 0), kScrWidth, 0, 0, kScrWidth, kScrHeight);
g_system->updateScreen();
}
2011-07-03 16:22:26 +02:00
void Vga::clear(uint8 color) {
for (int paneNum = 0; paneNum < 4; paneNum++)
_page[paneNum]->fillRect(Common::Rect(0, 0, kScrWidth, kScrHeight), color);
}
2011-07-03 16:22:26 +02:00
void Vga::copyPage(uint16 d, uint16 s) {
_page[d]->copyFrom(*_page[s]);
}
//--------------------------------------------------------------------------
void Bitmap::xShow(int16 x, int16 y) {
2011-07-20 14:22:56 +02:00
debugC(4, kCGEDebugBitmap, "Bitmap::xShow(%d, %d)", x, y);
2011-07-18 18:05:57 +02:00
2011-07-13 20:42:30 +10:00
const byte *srcP = (const byte *)_v;
2011-09-16 23:54:08 +02:00
byte *destEndP = (byte *)_vm->_vga->_page[1]->pixels + (kScrWidth * kScrHeight);
2011-07-13 20:42:30 +10:00
byte *lookupTable = _m;
// Loop through processing data for each plane. The game originally ran in plane mapped mode, where a
// given plane holds each fourth pixel sequentially. So to handle an entire picture, each plane's data
// must be decompressed and inserted into the surface
for (int planeCtr = 0; planeCtr < 4; planeCtr++) {
2011-09-16 23:54:08 +02:00
byte *destP = (byte *)_vm->_vga->_page[1]->getBasePtr(x + planeCtr, y);
2011-07-13 20:42:30 +10:00
for (;;) {
uint16 v = READ_LE_UINT16(srcP);
srcP += 2;
int cmd = v >> 14;
int count = v & 0x3FFF;
if (cmd == 0) {
// End of image
break;
}
assert(destP < destEndP);
if (cmd == 2)
srcP++;
2011-07-13 20:42:30 +10:00
else if (cmd == 3)
srcP += count;
// Handle a set of pixels
while (count-- > 0) {
// Transfer operation
switch (cmd) {
case 1:
// SKIP
break;
case 2:
case 3:
// TINT
*destP = lookupTable[*destP];
break;
}
// Move to next dest position
destP += 4;
2011-07-13 20:42:30 +10:00
}
}
}
}
void Bitmap::show(int16 x, int16 y) {
2011-07-20 14:22:56 +02:00
debugC(5, kCGEDebugBitmap, "Bitmap::show(%d, %d)", x, y);
2011-07-18 18:05:57 +02:00
2011-06-29 16:13:17 +02:00
const byte *srcP = (const byte *)_v;
2011-09-16 23:54:08 +02:00
byte *destEndP = (byte *)_vm->_vga->_page[1]->pixels + (kScrWidth * kScrHeight);
// Loop through processing data for each plane. The game originally ran in plane mapped mode, where a
// given plane holds each fourth pixel sequentially. So to handle an entire picture, each plane's data
// must be decompressed and inserted into the surface
for (int planeCtr = 0; planeCtr < 4; planeCtr++) {
2011-09-16 23:54:08 +02:00
byte *destP = (byte *)_vm->_vga->_page[1]->getBasePtr(x + planeCtr, y);
2011-06-28 21:58:03 +10:00
for (;;) {
uint16 v = READ_LE_UINT16(srcP);
srcP += 2;
int cmd = v >> 14;
int count = v & 0x3FFF;
if (cmd == 0) {
// End of image
break;
}
2011-06-28 21:58:03 +10:00
assert(destP < destEndP);
// Handle a set of pixels
while (count-- > 0) {
// Transfer operation
switch (cmd) {
case 1:
// SKIP
break;
case 2:
// REPEAT
*destP = *srcP;
break;
case 3:
// COPY
*destP = *srcP++;
break;
}
// Move to next dest position
destP += 4;
}
if (cmd == 2)
srcP++;
}
}
}
void Bitmap::hide(int16 x, int16 y) {
2011-07-20 14:22:56 +02:00
debugC(5, kCGEDebugBitmap, "Bitmap::hide(%d, %d)", x, y);
2011-07-18 18:05:57 +02:00
for (int yp = y; yp < y + _h; yp++) {
2011-09-16 23:54:08 +02:00
const byte *srcP = (const byte *)_vm->_vga->_page[2]->getBasePtr(x, yp);
byte *destP = (byte *)_vm->_vga->_page[1]->getBasePtr(x, yp);
2011-06-13 11:57:24 +02:00
2011-07-03 11:43:59 +10:00
Common::copy(srcP, srcP + _w, destP);
}
}
/*--------------------------------------------------------------------------*/
2011-09-16 23:54:08 +02:00
HorizLine::HorizLine(CGEEngine *vm) : Sprite(vm, NULL), _vm(vm) {
// Set the sprite list
BitmapPtr *HL = new BitmapPtr[2];
2011-09-16 23:54:08 +02:00
HL[0] = new Bitmap(_vm, "HLINE");
HL[1] = NULL;
setShapeList(HL);
}
2011-09-16 23:54:08 +02:00
SceneLight::SceneLight(CGEEngine *vm) : Sprite(vm, NULL), _vm(vm) {
// Set the sprite list
BitmapPtr *PR = new BitmapPtr[2];
2011-09-16 23:54:08 +02:00
PR[0] = new Bitmap(_vm, "PRESS");
PR[1] = NULL;
setShapeList(PR);
}
2011-09-16 23:54:08 +02:00
Spike::Spike(CGEEngine *vm): Sprite(vm, NULL), _vm(vm) {
// Set the sprite list
BitmapPtr *SP = new BitmapPtr[3];
2011-09-16 23:54:08 +02:00
SP[0] = new Bitmap(_vm, "SPK_L");
SP[1] = new Bitmap(_vm, "SPK_R");
SP[2] = NULL;
setShapeList(SP);
}
2011-09-16 23:54:08 +02:00
PocLight::PocLight(CGEEngine *vm): Sprite(vm, NULL), _vm(vm) {
// Set the sprite list
BitmapPtr *LI = new BitmapPtr[5];
2011-09-16 23:54:08 +02:00
LI[0] = new Bitmap(_vm, "LITE0");
LI[1] = new Bitmap(_vm, "LITE1");
LI[2] = new Bitmap(_vm, "LITE2");
LI[3] = new Bitmap(_vm, "LITE3");
LI[4] = NULL;
setShapeList(LI);
2011-07-10 20:07:45 +10:00
_flags._kill = false;
}
2011-07-18 18:05:57 +02:00
} // End of namespace CGE