2014-04-29 07:50:40 +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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This code is based on original Sfinx source code
|
|
|
|
* Copyright (c) 1994-1997 Janus B. Wisniewski and L.K. Avalon
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CGE2_VGA13H_H
|
|
|
|
#define CGE2_VGA13H_H
|
|
|
|
|
|
|
|
#include "common/serializer.h"
|
|
|
|
#include "common/events.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
#include "cge2/general.h"
|
|
|
|
#include "cge2/bitmap.h"
|
2014-05-09 11:11:50 +00:00
|
|
|
#include "cge2/snail.h"
|
2014-05-17 21:20:21 +00:00
|
|
|
#include "cge2/spare.h"
|
2014-05-26 07:57:41 +00:00
|
|
|
#include "cge2/cge2.h"
|
2014-04-29 07:50:40 +00:00
|
|
|
|
|
|
|
namespace CGE2 {
|
|
|
|
|
|
|
|
#define kFadeStep 2
|
|
|
|
#define kVgaColDark 207
|
|
|
|
#define kVgaColDarkGray 225 /*219*/
|
|
|
|
#define kVgaColGray 231
|
|
|
|
#define kVgaColLightGray 237
|
|
|
|
#define kPixelTransp 0xFE
|
|
|
|
#define kNoSeq (-1)
|
|
|
|
#define kNoPtr ((uint8)-1)
|
|
|
|
#define kSprExt ".SPR"
|
|
|
|
#define kPalCount 256
|
|
|
|
#define kPalSize (kPalCount * 3)
|
|
|
|
|
2014-06-12 05:20:55 +00:00
|
|
|
// From FXP.H
|
|
|
|
class FXP // fixed point
|
|
|
|
{
|
|
|
|
uint16 f;
|
|
|
|
int16 i;
|
|
|
|
long& Joined (void) const { return *(long *)&f; }
|
|
|
|
public:
|
2014-06-13 03:49:06 +00:00
|
|
|
FXP (void): f(0), i(0) { }
|
2014-06-14 23:34:30 +00:00
|
|
|
FXP (int i0, int f0 = 0) : i(i0), f(0) { }
|
2014-06-12 05:20:55 +00:00
|
|
|
FXP& operator = (const int& x) { i = x; f = 0; return *this; }
|
|
|
|
FXP operator + (const FXP& x) const { FXP y; y.Joined() = Joined()+x.Joined(); return y; }
|
|
|
|
FXP operator - (const FXP& x) const { FXP y; y.Joined() = Joined()-x.Joined(); return y; }
|
|
|
|
FXP operator * (const FXP& x) const {
|
|
|
|
FXP y; long t;
|
|
|
|
y.i = i * x.i;
|
|
|
|
t = ((long) f * x.f) >> 16;
|
|
|
|
t += ((long) i * x.f) + ((long) f * x.i);
|
|
|
|
y.f = t & 0xFFFF;
|
|
|
|
y.i += t >> 16;
|
|
|
|
return y;
|
|
|
|
}
|
|
|
|
FXP operator / (const FXP& x) const {
|
|
|
|
FXP y; bool sign = false;
|
2014-06-13 03:49:06 +00:00
|
|
|
if (!x.empty()) {
|
|
|
|
long j = Joined(), jx = x.Joined();
|
|
|
|
if (j < 0) {
|
|
|
|
j = -j;
|
|
|
|
sign ^= 1;
|
|
|
|
}
|
|
|
|
if (jx < 0) {
|
|
|
|
jx = -jx;
|
|
|
|
sign ^= 1;
|
|
|
|
}
|
|
|
|
y.i = signed(j / jx);
|
|
|
|
long r = j - jx * y.i;
|
|
|
|
//-- binary division
|
|
|
|
y.f = unsigned((r << 4) / (jx >> 12));
|
|
|
|
//------------------
|
|
|
|
if (sign)
|
|
|
|
y.Joined() = -y.Joined();
|
2014-06-12 05:20:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return y;
|
|
|
|
}
|
|
|
|
//int& operator = (int& a, const FXP& b) { return a = b.i; }
|
|
|
|
friend int& operator += (int& a, const FXP& b) { return a += b.i; }
|
|
|
|
friend int& operator -= (int& a, const FXP& b) { return a -= b.i; }
|
|
|
|
friend FXP& operator += (FXP& a, const int& b) { a.i += b; return a; }
|
|
|
|
friend FXP& operator -= (FXP& a, const int& b) { a.i -= b; return a; }
|
|
|
|
friend bool operator == (const FXP &a, const FXP &b) { return (a.i == b.i) && (a.f == b.f); }
|
|
|
|
friend bool operator != (const FXP &a, const FXP &b) { return (a.i != b.i) || (a.f != b.f); }
|
|
|
|
friend bool operator < (const FXP &a, const FXP &b) { return (a.i < b.i) || ((a.i == b.i) && (a.f < b.f)); }
|
|
|
|
friend bool operator > (const FXP &a, const FXP &b) { return (a.i > b.i) || ((a.i == b.i) && (a.f > b.f)); }
|
|
|
|
int trunc(void) const { return i; }
|
|
|
|
int round(void) const { return i + (f > 0x7FFF); }
|
2014-06-13 03:49:06 +00:00
|
|
|
bool empty() const { return i == 0 && f == 0; }
|
2014-06-12 05:20:55 +00:00
|
|
|
};
|
|
|
|
|
2014-05-26 07:57:41 +00:00
|
|
|
// From CGETYPE.H:
|
|
|
|
class V3D {
|
|
|
|
public:
|
2014-06-12 05:20:55 +00:00
|
|
|
FXP _x, _y, _z;
|
2014-06-04 05:10:27 +00:00
|
|
|
V3D() { }
|
2014-06-12 05:20:55 +00:00
|
|
|
V3D(FXP x, FXP y, FXP z = 0) : _x(x), _y(y), _z(z) { }
|
2014-05-26 07:57:41 +00:00
|
|
|
V3D(const V3D &p) : _x(p._x), _y(p._y), _z(p._z) { }
|
|
|
|
V3D operator+(const V3D &p) const { return V3D(_x + p._x, _y + p._y, _z + p._z); }
|
|
|
|
V3D operator-(const V3D &p) const { return V3D(_x - p._x, _y - p._y, _z - p._z); }
|
|
|
|
V3D operator*(long n) const { return V3D(_x * n, _y * n, _z * n); }
|
|
|
|
V3D operator/ (long n) const { return V3D(_x / n, _y / n, _z / n); }
|
2014-06-12 05:20:55 +00:00
|
|
|
bool operator==(const V3D &p) const { return _x == p._x && _y == p._y && _z == p._z; }
|
|
|
|
bool operator!=(const V3D &p) const { return _x != p._x || _y != p._y || _z != p._z; }
|
2014-05-26 07:57:41 +00:00
|
|
|
V3D& operator+=(const V3D &x) { return *this = *this + x; }
|
|
|
|
V3D& operator-=(const V3D &x) { return *this = *this - x; }
|
|
|
|
};
|
|
|
|
|
|
|
|
class V2D : public Common::Point {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
public:
|
|
|
|
V2D& operator=(const V3D &p3) {
|
2014-06-12 05:20:55 +00:00
|
|
|
FXP m = _vm->_eye->_z / (p3._z - _vm->_eye->_z);
|
|
|
|
FXP posx = _vm->_eye->_x + (_vm->_eye->_x - p3._x) * m;
|
|
|
|
x = posx.round();
|
|
|
|
FXP posy = _vm->_eye->_y + (_vm->_eye->_y - p3._y) * m;
|
|
|
|
y = posy.round();
|
2014-05-26 07:57:41 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
V2D(CGE2Engine *vm) : _vm(vm) { }
|
|
|
|
V2D(CGE2Engine *vm, const V3D &p3) : _vm(vm) { *this = p3; }
|
2014-06-03 20:34:39 +00:00
|
|
|
V2D(CGE2Engine *vm, int posx, int posy) : _vm(vm), Common::Point(posx, posy) { }
|
2014-05-26 07:57:41 +00:00
|
|
|
bool operator<(const V2D &p) const { return (x < p.x) && (y < p.y); }
|
|
|
|
bool operator<=(const V2D &p) const { return (x <= p.x) && (y <= p.y); }
|
|
|
|
bool operator>(const V2D &p) const { return (x > p.x) && (y > p.y); }
|
|
|
|
bool operator>=(const V2D &p) const { return (x >= p.x) && (y >= p.y); }
|
|
|
|
V2D operator+(const V2D &p) const { return V2D(_vm, x + p.x, y + p.y); }
|
|
|
|
V2D operator-(const V2D &p) const { return V2D(_vm, x - p.x, y - p.y); }
|
2014-06-12 05:20:55 +00:00
|
|
|
bool operator==(const V3D &p) const { V3D tmp(x, y); return tmp._x == p._x && tmp._y == p._y && tmp._z == p._z; }
|
|
|
|
bool operator!=(const V3D &p) const { V3D tmp(x, y); return tmp._x != p._x || tmp._y != p._y || tmp._z == p._z; }
|
2014-06-22 11:26:55 +00:00
|
|
|
bool operator==(const V2D &p) const { return x == p.x && y == p.y; }
|
2014-06-04 05:10:27 +00:00
|
|
|
uint16 area() { return x * y; }
|
2014-05-26 07:57:41 +00:00
|
|
|
bool limited(const V2D &p) {
|
2014-06-12 05:20:55 +00:00
|
|
|
return ((x < p.x) && (y < p.y));
|
2014-05-26 07:57:41 +00:00
|
|
|
}
|
2014-06-12 05:20:55 +00:00
|
|
|
V2D scale (int z) {
|
|
|
|
FXP m = _vm->_eye->_z / (_vm->_eye->_z - z);
|
|
|
|
FXP posx = m * x;
|
|
|
|
FXP posy = m * y;
|
|
|
|
return V2D(_vm, posx.trunc(), posy.trunc());
|
2014-05-26 07:57:41 +00:00
|
|
|
}
|
|
|
|
static double trunc(double d) { return (d > 0) ? floor(d) : ceil(d); }
|
2014-06-04 05:10:27 +00:00
|
|
|
static double round(double number) { return number < 0.0 ? ceil(number) : floor(number); }
|
2014-05-26 07:57:41 +00:00
|
|
|
};
|
|
|
|
|
2014-04-29 07:50:40 +00:00
|
|
|
struct Seq {
|
|
|
|
uint8 _now;
|
|
|
|
uint8 _next;
|
|
|
|
int8 _dx;
|
|
|
|
int8 _dy;
|
2014-05-13 15:27:12 +00:00
|
|
|
int8 _dz;
|
2014-04-29 07:50:40 +00:00
|
|
|
int _dly;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SprExt {
|
|
|
|
public:
|
2014-05-13 09:58:42 +00:00
|
|
|
V2D _p0;
|
|
|
|
V2D _p1;
|
2014-04-29 07:50:40 +00:00
|
|
|
BitmapPtr _b0;
|
|
|
|
BitmapPtr _b1;
|
2014-06-06 12:57:34 +00:00
|
|
|
BitmapPtr _shpList;
|
2014-05-13 09:58:42 +00:00
|
|
|
int _location;
|
2014-04-29 07:50:40 +00:00
|
|
|
Seq *_seq;
|
|
|
|
char *_name;
|
2014-05-13 09:58:42 +00:00
|
|
|
CommandHandler::Command *_actions[kActions];
|
2014-05-18 12:15:13 +00:00
|
|
|
SprExt(CGE2Engine *vm);
|
2014-04-29 07:50:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class Sprite {
|
|
|
|
protected:
|
|
|
|
SprExt *_ext;
|
2014-05-08 14:51:03 +00:00
|
|
|
CGE2Engine *_vm;
|
2014-04-29 07:50:40 +00:00
|
|
|
public:
|
|
|
|
int _ref;
|
|
|
|
signed char _scene;
|
|
|
|
struct Flags {
|
2014-06-15 20:48:23 +00:00
|
|
|
bool _hide; // general visibility switch
|
|
|
|
bool _drag; // sprite is moveable
|
|
|
|
bool _hold; // sprite is held with mouse
|
|
|
|
bool _trim; // Trim flag
|
|
|
|
bool _slav; // slave object
|
|
|
|
bool _kill; // dispose memory after remove
|
|
|
|
bool _xlat; // 2nd way display: xlat table
|
|
|
|
bool _port; // portable
|
|
|
|
bool _kept; // kept in pocket
|
|
|
|
bool _frnt; // stay in front of sprite
|
|
|
|
bool _east; // talk to east (in opposite to west)
|
|
|
|
bool _near; // Near action lock
|
|
|
|
bool _shad; // shadow
|
|
|
|
bool _back; // 'send to background' request
|
|
|
|
bool _zmov; // sprite needs Z-update in queue
|
|
|
|
bool _tran; // transparent (untouchable)
|
2014-04-29 07:50:40 +00:00
|
|
|
} _flags;
|
2014-05-13 14:47:55 +00:00
|
|
|
V2D _pos2D;
|
|
|
|
V3D _pos3D;
|
|
|
|
V2D _siz;
|
2014-04-29 07:50:40 +00:00
|
|
|
uint16 _time;
|
2014-05-13 14:47:55 +00:00
|
|
|
struct { byte _ptr, _cnt; } _actionCtrl[kActions];
|
2014-04-29 07:50:40 +00:00
|
|
|
int _seqPtr;
|
2014-05-13 14:47:55 +00:00
|
|
|
int _seqCnt;
|
2014-04-29 07:50:40 +00:00
|
|
|
int _shpCnt;
|
|
|
|
char _file[kMaxFile];
|
2014-05-13 14:47:55 +00:00
|
|
|
// Following trailer is not saved with the game:
|
2014-04-29 07:50:40 +00:00
|
|
|
Sprite *_prev;
|
|
|
|
Sprite *_next;
|
2014-05-10 09:12:30 +00:00
|
|
|
static byte _constY;
|
|
|
|
static byte _follow;
|
2014-05-13 15:27:12 +00:00
|
|
|
static Seq _stdSeq8[];
|
2014-04-29 07:50:40 +00:00
|
|
|
|
|
|
|
bool works(Sprite *spr);
|
|
|
|
bool seqTest(int n);
|
|
|
|
inline bool active() {
|
|
|
|
return _ext != NULL;
|
|
|
|
}
|
2014-05-09 11:11:50 +00:00
|
|
|
Sprite(CGE2Engine *vm);
|
2014-06-06 12:57:34 +00:00
|
|
|
Sprite(CGE2Engine *vm, BitmapPtr shp, int cnt);
|
2014-04-29 07:50:40 +00:00
|
|
|
virtual ~Sprite();
|
2014-06-03 15:39:10 +00:00
|
|
|
BitmapPtr getShp();
|
2014-06-06 12:57:34 +00:00
|
|
|
void setShapeList(BitmapPtr shp, int cnt);
|
2014-06-04 05:10:27 +00:00
|
|
|
void moveShapesHi();
|
|
|
|
void moveShapesLo();
|
2014-05-19 21:24:23 +00:00
|
|
|
int labVal(Action snq, int lab);
|
2014-04-29 07:50:40 +00:00
|
|
|
Sprite *expand();
|
|
|
|
Sprite *contract();
|
2014-06-04 05:10:27 +00:00
|
|
|
void backShow();
|
2014-04-29 07:50:40 +00:00
|
|
|
void setName(char *newName);
|
|
|
|
inline char *name() {
|
|
|
|
return (_ext) ? _ext->_name : NULL;
|
|
|
|
}
|
2014-05-09 11:11:50 +00:00
|
|
|
void gotoxyz(int x, int y, int z = 0);
|
2014-06-04 05:10:27 +00:00
|
|
|
void gotoxyz();
|
2014-05-09 11:11:50 +00:00
|
|
|
void gotoxyz(V2D pos);
|
|
|
|
void gotoxyz_(V2D pos);
|
|
|
|
void gotoxyz(V3D pos);
|
2014-04-29 07:50:40 +00:00
|
|
|
void center();
|
2014-05-13 14:47:55 +00:00
|
|
|
void show(uint16 pg);
|
|
|
|
void hide(uint16 pg);
|
2014-04-29 07:50:40 +00:00
|
|
|
void show();
|
|
|
|
void hide();
|
|
|
|
BitmapPtr ghost();
|
|
|
|
void step(int nr = -1);
|
|
|
|
Seq *setSeq(Seq *seq);
|
2014-05-13 14:47:55 +00:00
|
|
|
CommandHandler::Command *snList(Action type);
|
2014-06-15 19:29:32 +00:00
|
|
|
virtual void touch(uint16 mask, V2D pos, Common::KeyCode keyCode);
|
2014-05-23 08:55:58 +00:00
|
|
|
virtual void tick();
|
2014-06-13 20:36:32 +00:00
|
|
|
virtual void setScene(int c);
|
2014-06-04 05:10:27 +00:00
|
|
|
void clrHide() { if (_ext) _ext->_b0 = NULL; }
|
2014-04-29 07:50:40 +00:00
|
|
|
void sync(Common::Serializer &s);
|
2014-05-13 14:47:55 +00:00
|
|
|
|
2014-06-04 05:10:27 +00:00
|
|
|
static void (*notify) ();
|
2014-04-29 07:50:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class Queue {
|
|
|
|
Sprite *_head;
|
|
|
|
Sprite *_tail;
|
|
|
|
public:
|
|
|
|
Queue(bool show);
|
|
|
|
~Queue();
|
|
|
|
|
|
|
|
void append(Sprite *spr);
|
|
|
|
void insert(Sprite *spr, Sprite *nxt);
|
|
|
|
void insert(Sprite *spr);
|
|
|
|
Sprite *remove(Sprite *spr);
|
|
|
|
Sprite *first() {
|
|
|
|
return _head;
|
|
|
|
}
|
|
|
|
Sprite *last() {
|
|
|
|
return _tail;
|
|
|
|
}
|
|
|
|
Sprite *locate(int ref);
|
2014-05-17 21:14:19 +00:00
|
|
|
bool locate(Sprite *spr);
|
2014-04-29 07:50:40 +00:00
|
|
|
void clear();
|
|
|
|
};
|
|
|
|
|
|
|
|
class Vga {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
bool _setPal;
|
|
|
|
Dac *_oldColors;
|
|
|
|
Dac *_newColors;
|
|
|
|
const char *_msg;
|
|
|
|
const char *_name;
|
|
|
|
|
|
|
|
void updateColors();
|
|
|
|
void setColors();
|
|
|
|
void waitVR();
|
|
|
|
uint8 closest(Dac *pal, const uint8 colR, const uint8 colG, const uint8 colB);
|
|
|
|
|
|
|
|
public:
|
|
|
|
uint32 _frmCnt;
|
|
|
|
Queue *_showQ;
|
|
|
|
int _mono;
|
|
|
|
Graphics::Surface *_page[4];
|
|
|
|
Dac *_sysPal;
|
|
|
|
|
|
|
|
Vga(CGE2Engine *vm);
|
|
|
|
~Vga();
|
|
|
|
|
|
|
|
uint8 *glass(Dac *pal, const uint8 colR, const uint8 colG, const uint8 colB);
|
|
|
|
void getColors(Dac *tab);
|
|
|
|
void setColors(Dac *tab, int lum);
|
|
|
|
void clear(uint8 color);
|
|
|
|
void copyPage(uint16 d, uint16 s);
|
|
|
|
void sunrise(Dac *tab);
|
|
|
|
void sunset();
|
|
|
|
void show();
|
|
|
|
void update();
|
|
|
|
|
|
|
|
void palToDac(const byte *palData, Dac *tab);
|
|
|
|
void dacToPal(const Dac *tab, byte *palData);
|
|
|
|
};
|
|
|
|
|
|
|
|
class HorizLine: public Sprite {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
public:
|
|
|
|
HorizLine(CGE2Engine *vm);
|
|
|
|
};
|
|
|
|
|
|
|
|
class SceneLight: public Sprite {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
public:
|
|
|
|
SceneLight(CGE2Engine *vm);
|
|
|
|
};
|
|
|
|
|
|
|
|
class Speaker: public Sprite {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
public:
|
|
|
|
Speaker(CGE2Engine *vm);
|
|
|
|
};
|
|
|
|
|
|
|
|
class PocLight: public Sprite {
|
|
|
|
CGE2Engine *_vm;
|
|
|
|
public:
|
|
|
|
PocLight(CGE2Engine *vm);
|
|
|
|
};
|
|
|
|
|
|
|
|
} // End of namespace CGE2
|
|
|
|
|
|
|
|
#endif // CGE2_VGA13H_H
|