2008-04-20 14:47:37 +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.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef M4_GRAPHICS_H
|
|
|
|
#define M4_GRAPHICS_H
|
|
|
|
|
|
|
|
#include "common/rect.h"
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/stream.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
|
|
|
|
#include "m4/globals.h"
|
|
|
|
|
|
|
|
namespace M4 {
|
|
|
|
|
2010-06-03 09:34:32 +00:00
|
|
|
#define MADS_SURFACE_WIDTH 320
|
|
|
|
#define MADS_SURFACE_HEIGHT 156
|
|
|
|
#define MADS_SCREEN_HEIGHT 200
|
|
|
|
#define MADS_Y_OFFSET ((MADS_SCREEN_HEIGHT - MADS_SURFACE_HEIGHT) / 2)
|
|
|
|
|
2010-06-20 07:04:58 +00:00
|
|
|
#define TRANSPARENT_COLOUR_INDEX 0xFF
|
2010-06-03 09:34:32 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
struct BGR8 {
|
|
|
|
uint8 b, g, r;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RGB8 {
|
|
|
|
uint8 r, g, b, u;
|
|
|
|
};
|
|
|
|
|
|
|
|
//later use ScummVM's Rect?
|
|
|
|
struct M4Rect {
|
|
|
|
int32 x1, y1, x2, y2;
|
|
|
|
};
|
|
|
|
|
|
|
|
class M4Surface;
|
|
|
|
|
|
|
|
// RGBList
|
2009-01-01 15:06:43 +00:00
|
|
|
// Implements a list of RGB entries
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
class RGBList {
|
|
|
|
private:
|
|
|
|
int _size;
|
|
|
|
RGB8 *_data;
|
|
|
|
byte *_palIndexes;
|
|
|
|
bool _freeData;
|
|
|
|
public:
|
|
|
|
RGBList(int numEntries = 256, RGB8 *srcData = NULL, bool freeData = true);
|
|
|
|
~RGBList();
|
|
|
|
|
|
|
|
RGB8 *data() { return _data; }
|
|
|
|
byte *palIndexes() { return _palIndexes; }
|
|
|
|
int size() { return _size; }
|
2009-12-12 11:59:38 +00:00
|
|
|
RGB8 &operator[](int idx) { return _data[idx]; }
|
|
|
|
void setRange(int start, int count, const RGB8 *src);
|
2010-07-04 00:38:16 +00:00
|
|
|
RGBList *clone() const;
|
2008-04-20 14:47:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// M4Surface
|
|
|
|
// Class representing either a buffered surface or the physical screen.
|
|
|
|
|
|
|
|
class M4Sprite;
|
|
|
|
|
|
|
|
struct SpriteInfo {
|
|
|
|
M4Sprite *sprite;
|
|
|
|
int hotX, hotY;
|
|
|
|
int width, height;
|
|
|
|
int scaleX, scaleY;
|
|
|
|
uint8 encoding;
|
2010-03-14 05:39:47 +00:00
|
|
|
byte *inverseColourTable;
|
2008-04-20 14:47:37 +00:00
|
|
|
RGB8 *palette;
|
|
|
|
};
|
|
|
|
|
2009-12-07 18:22:18 +00:00
|
|
|
class M4Surface : protected Graphics::Surface {
|
2008-04-20 14:47:37 +00:00
|
|
|
private:
|
|
|
|
byte _color;
|
|
|
|
bool _isScreen;
|
2010-06-03 09:34:32 +00:00
|
|
|
RGBList *_rgbList;
|
2010-06-30 10:13:20 +00:00
|
|
|
bool _ownsData;
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
void rexLoadBackground(Common::SeekableReadStream *source, RGBList **palData = NULL);
|
|
|
|
void madsLoadBackground(int roomNumber, RGBList **palData = NULL);
|
|
|
|
void m4LoadBackground(Common::SeekableReadStream *source);
|
|
|
|
public:
|
2009-01-01 15:06:43 +00:00
|
|
|
M4Surface(bool isScreen = false) {
|
2010-06-03 09:34:32 +00:00
|
|
|
create(g_system->getWidth(), isScreen ? g_system->getHeight() : MADS_SURFACE_HEIGHT, 1);
|
2008-04-20 14:47:37 +00:00
|
|
|
_isScreen = isScreen;
|
2010-06-03 09:34:32 +00:00
|
|
|
_rgbList = NULL;
|
2010-06-30 10:13:20 +00:00
|
|
|
_ownsData = true;
|
2008-04-20 14:47:37 +00:00
|
|
|
}
|
2009-12-07 18:22:18 +00:00
|
|
|
M4Surface(int width_, int height_) {
|
|
|
|
create(width_, height_, 1);
|
|
|
|
_isScreen = false;
|
2010-06-03 09:34:32 +00:00
|
|
|
_rgbList = NULL;
|
2010-06-30 10:13:20 +00:00
|
|
|
_ownsData = true;
|
2009-12-07 18:22:18 +00:00
|
|
|
}
|
2010-06-30 10:13:20 +00:00
|
|
|
M4Surface(int width_, int height_, byte *srcPixels, int pitch_) {
|
|
|
|
bytesPerPixel = 1;
|
|
|
|
w = width_;
|
|
|
|
h = height_;
|
|
|
|
pitch = pitch_;
|
|
|
|
pixels = srcPixels;
|
|
|
|
_rgbList = NULL;
|
|
|
|
_ownsData = false;
|
|
|
|
}
|
|
|
|
|
2010-06-03 09:34:32 +00:00
|
|
|
virtual ~M4Surface();
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
// loads a .COD file into the M4Surface
|
|
|
|
// TODO: maybe move this to the rail system? check where it makes sense
|
|
|
|
// The sprite drawing needs this, too, so should be more global.
|
|
|
|
void loadCodesM4(Common::SeekableReadStream *source);
|
|
|
|
void loadCodesMads(Common::SeekableReadStream *source);
|
|
|
|
|
|
|
|
// loads the specified background
|
|
|
|
void loadBackground(int sceneNumber, RGBList **palData = NULL);
|
|
|
|
void loadBackgroundRiddle(const char *sceneName);
|
2010-06-03 10:46:55 +00:00
|
|
|
void madsLoadInterface(int index, RGBList **palData = NULL);
|
|
|
|
void madsLoadInterface(const Common::String &filename);
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
void setColor(byte value) { _color = value; }
|
2009-12-12 11:59:38 +00:00
|
|
|
void setColour(byte value) { _color = value; }
|
2009-12-07 18:22:18 +00:00
|
|
|
inline byte getColor() const { return _color; }
|
2008-04-20 14:47:37 +00:00
|
|
|
void vLine(int x, int y1, int y2);
|
|
|
|
void hLine(int x1, int x2, int y);
|
|
|
|
void vLineXor(int x, int y1, int y2);
|
|
|
|
void hLineXor(int x1, int x2, int y);
|
2009-12-11 17:14:55 +00:00
|
|
|
void drawLine(int x1, int y1, int x2, int y2, byte color);
|
2008-04-20 14:47:37 +00:00
|
|
|
void frameRect(int x1, int y1, int x2, int y2);
|
|
|
|
void fillRect(int x1, int y1, int x2, int y2);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-04-20 14:47:37 +00:00
|
|
|
void drawSprite(int x, int y, SpriteInfo &info, const Common::Rect &clipRect);
|
|
|
|
|
|
|
|
// Surface methods
|
2010-03-21 10:51:11 +00:00
|
|
|
inline Common::Rect bounds() const { return Common::Rect(0, 0, width(), height()); }
|
2009-12-07 18:22:18 +00:00
|
|
|
inline int width() const { return w; }
|
|
|
|
inline int height() const { return h; }
|
2010-06-30 10:13:20 +00:00
|
|
|
inline int getPitch() const { return pitch; }
|
2008-04-20 14:47:37 +00:00
|
|
|
void setSize(int sizeX, int sizeY) { create(sizeX, sizeY, 1); }
|
2009-12-07 18:20:20 +00:00
|
|
|
inline byte *getBasePtr() {
|
|
|
|
return (byte *)pixels;
|
|
|
|
}
|
|
|
|
inline byte *getBasePtr(int x, int y) {
|
|
|
|
return (byte *)Graphics::Surface::getBasePtr(x, y);
|
|
|
|
}
|
|
|
|
inline const byte *getBasePtr(int x, int y) const {
|
|
|
|
return (const byte *)Graphics::Surface::getBasePtr(x, y);
|
|
|
|
}
|
2008-04-20 14:47:37 +00:00
|
|
|
void freeData();
|
2008-06-18 21:02:52 +00:00
|
|
|
void clear();
|
2010-06-10 10:20:25 +00:00
|
|
|
void reset();
|
2008-04-20 14:47:37 +00:00
|
|
|
void frameRect(const Common::Rect &r, uint8 color);
|
|
|
|
void fillRect(const Common::Rect &r, uint8 color);
|
2010-06-15 11:19:32 +00:00
|
|
|
void copyFrom(M4Surface *src, const Common::Rect &srcBounds, int destX, int destY, int transparentColour = -1);
|
2010-06-30 10:17:54 +00:00
|
|
|
void copyFrom(M4Surface *src, int destX, int destY, int depth, M4Surface *depthSurface,
|
2010-06-15 11:19:32 +00:00
|
|
|
int scale, int transparentColour = -1);
|
2008-04-20 14:47:37 +00:00
|
|
|
|
2009-01-01 15:06:43 +00:00
|
|
|
void update() {
|
2008-04-20 14:47:37 +00:00
|
|
|
if (_isScreen) {
|
|
|
|
g_system->copyRectToScreen((const byte *)pixels, pitch, 0, 0, w, h);
|
2009-01-01 15:06:43 +00:00
|
|
|
g_system->updateScreen();
|
2008-04-20 14:47:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// copyTo methods
|
2010-05-23 07:20:40 +00:00
|
|
|
inline void copyTo(M4Surface *dest, int transparentColour = -1) {
|
|
|
|
dest->copyFrom(this, Common::Rect(width(), height()), 0, 0, transparentColour);
|
2008-04-20 14:47:37 +00:00
|
|
|
}
|
2010-05-23 07:20:40 +00:00
|
|
|
inline void copyTo(M4Surface *dest, int x, int y, int transparentColour = -1) {
|
|
|
|
dest->copyFrom(this, Common::Rect(width(), height()), x, y, transparentColour);
|
2008-04-20 14:47:37 +00:00
|
|
|
}
|
2009-12-07 18:22:18 +00:00
|
|
|
inline void copyTo(M4Surface *dest, const Common::Rect &srcBounds, int destX, int destY,
|
2010-05-23 07:20:40 +00:00
|
|
|
int transparentColour = -1) {
|
|
|
|
dest->copyFrom(this, srcBounds, destX, destY, transparentColour);
|
2008-04-20 14:47:37 +00:00
|
|
|
}
|
2010-05-23 07:20:40 +00:00
|
|
|
inline void copyTo(M4Surface *dest, int destX, int destY, int depth, M4Surface *depthsSurface, int scale,
|
|
|
|
int transparentColour = -1) {
|
2010-06-30 10:17:54 +00:00
|
|
|
dest->copyFrom(this, destX, destY, depth, depthsSurface, scale, transparentColour);
|
2010-05-23 07:20:40 +00:00
|
|
|
}
|
|
|
|
|
2010-05-28 10:17:16 +00:00
|
|
|
void scrollX(int xAmount);
|
|
|
|
void scrollY(int yAmount);
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
void translate(RGBList *list, bool isTransparent = false);
|
2010-07-04 00:38:16 +00:00
|
|
|
M4Surface *flipHorizontal() const;
|
2008-04-20 14:47:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum FadeType {FT_TO_GREY, FT_TO_COLOR, FT_TO_BLOCK};
|
|
|
|
|
|
|
|
class Palette {
|
|
|
|
private:
|
2010-01-30 09:21:07 +00:00
|
|
|
MadsM4Engine *_vm;
|
2008-04-20 14:47:37 +00:00
|
|
|
bool _colorsChanged;
|
|
|
|
bool _fading_in_progress;
|
|
|
|
byte _originalPalette[256 * 4];
|
|
|
|
byte _fadedPalette[256 * 4];
|
|
|
|
int _usageCount[256];
|
|
|
|
|
|
|
|
void reset();
|
|
|
|
public:
|
2010-01-30 09:21:07 +00:00
|
|
|
Palette(MadsM4Engine *vm);
|
2008-04-20 14:47:37 +00:00
|
|
|
|
|
|
|
void setPalette(const byte *colors, uint start, uint num);
|
|
|
|
void setPalette(const RGB8 *colors, uint start, uint num);
|
|
|
|
void grabPalette(byte *colors, uint start, uint num);
|
|
|
|
void grabPalette(RGB8 *colors, uint start, uint num) {
|
|
|
|
grabPalette((byte *)colors, start, num);
|
|
|
|
}
|
2010-02-13 11:49:18 +00:00
|
|
|
void setEntry(uint index, uint8 r, uint8 g, uint8 b);
|
2008-04-20 14:47:37 +00:00
|
|
|
uint8 palIndexFromRgb(byte r, byte g, byte b, RGB8 *paletteData = NULL);
|
|
|
|
|
|
|
|
void fadeToGreen(int numSteps, uint delayAmount);
|
|
|
|
void fadeFromGreen(int numSteps, uint delayAmount, bool fadeToBlack);
|
|
|
|
void fadeIn(int numSteps, uint delayAmount, RGB8 *destPalette, int numColors);
|
|
|
|
void fadeIn(int numSteps, uint delayAmount, RGBList *destPalette);
|
|
|
|
static RGB8 *decodeMadsPalette(Common::SeekableReadStream *palStream, int *numColors);
|
|
|
|
int setMadsPalette(Common::SeekableReadStream *palStream, int indexStart = 0);
|
|
|
|
void setMadsSystemPalette();
|
|
|
|
|
|
|
|
// Methods used for reference counting color usage
|
|
|
|
void resetColorCounts();
|
|
|
|
void blockRange(int startIndex, int size);
|
|
|
|
void addRange(RGBList *list);
|
|
|
|
void deleteRange(RGBList *list);
|
|
|
|
void deleteAllRanges();
|
|
|
|
|
|
|
|
// Color indexes
|
|
|
|
uint8 BLACK;
|
|
|
|
uint8 BLUE;
|
|
|
|
uint8 GREEN;
|
|
|
|
uint8 CYAN;
|
|
|
|
uint8 RED;
|
|
|
|
uint8 VIOLET;
|
|
|
|
uint8 BROWN;
|
|
|
|
uint8 LIGHT_GRAY;
|
|
|
|
uint8 DARK_GRAY;
|
|
|
|
uint8 LIGHT_BLUE;
|
|
|
|
uint8 LIGHT_GREEN;
|
|
|
|
uint8 LIGHT_CYAN;
|
|
|
|
uint8 LIGHT_RED;
|
|
|
|
uint8 PINK;
|
|
|
|
uint8 YELLOW;
|
|
|
|
uint8 WHITE;
|
|
|
|
};
|
|
|
|
|
|
|
|
void decompressRle(byte *rleData, int rleSize, byte *celData, int w, int h);
|
|
|
|
void decompressRle(Common::SeekableReadStream &rleData, byte *celData, int w, int h);
|
|
|
|
int scaleValue(int value, int scale, int err);
|
|
|
|
|
|
|
|
} // End of namespace M4
|
|
|
|
|
|
|
|
#endif
|