2021-05-17 20:47:39 +02: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.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Based on the original sources
|
|
|
|
* Faery Tale II -- The Halls of the Dead
|
|
|
|
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SAGA2_GDRAW_H
|
|
|
|
#define SAGA2_GDRAW_H
|
|
|
|
|
|
|
|
#include "saga2/rect.h"
|
|
|
|
|
|
|
|
namespace Saga2 {
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
Drawing Pens
|
|
|
|
* ===================================================================== */
|
|
|
|
|
|
|
|
// A drawing pen. Note that this should later be a class?
|
|
|
|
|
|
|
|
typedef uint8 gPen; // a pen index number
|
|
|
|
|
|
|
|
/* ============================================================================ *
|
|
|
|
PixelMap: defines a chunky bitmap
|
|
|
|
* ============================================================================ */
|
|
|
|
|
|
|
|
// A general purpose image
|
|
|
|
|
|
|
|
class gPixelMap {
|
|
|
|
public:
|
|
|
|
Extent16 size; // image size
|
|
|
|
uint8 *data;
|
|
|
|
|
|
|
|
// Compute the number of bytes in the pixel map
|
|
|
|
int32 bytes(void) {
|
|
|
|
return size.x * size.y;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// The simplest kind of pixel map is a static image
|
|
|
|
|
|
|
|
class gStaticImage : public gPixelMap {
|
|
|
|
public:
|
|
|
|
// constructors:
|
|
|
|
|
|
|
|
gStaticImage(void) {
|
|
|
|
size.x = size.y = 0;
|
|
|
|
data = NULL;
|
|
|
|
}
|
|
|
|
gStaticImage(int w, int h, uint8 *buffer) {
|
|
|
|
size.x = w;
|
|
|
|
size.y = h;
|
|
|
|
data = buffer;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Automatically-remapped images: Eacn image adds itself to a
|
|
|
|
// chain at the time it is constructed, and will automatically
|
|
|
|
// remap to a new palette by calling the global remap() function.
|
|
|
|
|
|
|
|
class gMappedImage : public gPixelMap {
|
|
|
|
static gMappedImage *head; // first image in map chain
|
|
|
|
|
|
|
|
gMappedImage *next; // next image to remap
|
|
|
|
gPixelMap original;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Constructor and destructor
|
|
|
|
gMappedImage(int w, int h, uint8 *buffer);
|
|
|
|
virtual ~gMappedImage() {
|
|
|
|
if (data) free(data);
|
|
|
|
}
|
|
|
|
static void remap(gPen[]);
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ============================================================================ *
|
|
|
|
gFont: A bitmapped font structure
|
|
|
|
* ============================================================================ */
|
|
|
|
|
|
|
|
struct gFont {
|
|
|
|
uint16 height; // height of the font
|
|
|
|
uint16 baseLine; // baseline of the font
|
|
|
|
uint16 rowMod; // row modulus for char data
|
|
|
|
|
2021-06-07 18:19:10 +02:00
|
|
|
uint16 charXOffset[256]; // offset of each char in data
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-07 18:19:10 +02:00
|
|
|
int8 charWidth[256], // width of character in pixels
|
|
|
|
charKern[256], // character kern value
|
|
|
|
charSpace[256]; // character space value
|
2021-05-17 20:47:39 +02:00
|
|
|
|
|
|
|
// followed by (row_mod * height) bytes of character data
|
|
|
|
byte *fontdata;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Prototypes
|
|
|
|
|
|
|
|
gFont *LoadFont(char *fontname);
|
|
|
|
void DisposeFont(gFont *font);
|
2021-05-27 15:44:09 +02:00
|
|
|
int16 TextWidth(gFont *font, const char *s, int16 length, int16 styles);
|
2021-05-17 20:47:39 +02:00
|
|
|
int16 WhichChar(gFont *font, uint8 *s, int16 length, int16 maxLen);
|
|
|
|
int16 WhichIChar(gFont *font, uint8 *s, int16 length, int16 maxLen);
|
|
|
|
int32 GTextWrap(gFont *font, char *mark, uint16 &count, uint16 width, int16 styles);
|
|
|
|
|
|
|
|
/* ============================================================================ *
|
|
|
|
gPenState: Used by gPorts to save the current state
|
|
|
|
* ============================================================================ */
|
|
|
|
|
|
|
|
struct gPenState {
|
|
|
|
gPen fgPen, // current foregroung pen
|
|
|
|
bgPen, // current drawing mode
|
|
|
|
olPen, // text outline pen
|
|
|
|
shPen; // text shadow pen
|
|
|
|
uint8 drawMode; // current drawing mode
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ============================================================================ *
|
|
|
|
gPort: Facilitates redering operations on PixelMaps
|
|
|
|
* ============================================================================ */
|
|
|
|
|
|
|
|
enum draw_modes {
|
|
|
|
drawModeMatte = 0, // use transparency
|
|
|
|
drawModeColor, // solid color, use transparency
|
|
|
|
drawModeReplace, // don't use transparency
|
|
|
|
drawModeComplement, // blit in complement mode
|
|
|
|
|
|
|
|
numDrawModes
|
|
|
|
};
|
|
|
|
|
|
|
|
enum text_styles {
|
|
|
|
textStyleOutline = (1 << 0), // outline the characters
|
|
|
|
textStyleShadow = (1 << 1), // drop shadow the characters
|
|
|
|
textStyleUnderScore = (1 << 2), // underscore all chars
|
|
|
|
textStyleUnderBar = (1 << 3), // underscore char after a '_'
|
|
|
|
textStyleHiLiteBar = (1 << 4), // highlight char after a '_'
|
|
|
|
textStyleThickOutline = (1 << 5), // extra-thick outline
|
|
|
|
textStyleBold = (1 << 6), // bold
|
|
|
|
textStyleItalics = (1 << 7) // italic
|
|
|
|
};
|
|
|
|
|
|
|
|
enum text_positions {
|
|
|
|
textPosLeft = (1 << 0),
|
|
|
|
textPosRight = (1 << 1),
|
|
|
|
textPosHigh = (1 << 2),
|
|
|
|
textPosLow = (1 << 3)
|
|
|
|
};
|
|
|
|
|
|
|
|
class gPort {
|
|
|
|
public:
|
|
|
|
gPixelMap *map; // pointer to map
|
|
|
|
|
|
|
|
// Added by Talin to speed up rendering and allow inverted
|
|
|
|
// gPorts for WinG compatibility
|
|
|
|
|
|
|
|
uint8 *baseRow; // address of row 0
|
|
|
|
int16 rowMod; // modulus or row
|
|
|
|
|
|
|
|
Point16 origin; // origin drawing point
|
|
|
|
Rect16 clip; // clip region DrawPort
|
|
|
|
gPen fgPen, // current foregroung pen
|
|
|
|
bgPen, // current drawing mode
|
|
|
|
olPen, // text outline pen
|
|
|
|
shPen; // text shadow pen
|
|
|
|
gPen *penMap; // indirect pen map
|
|
|
|
enum draw_modes drawMode; // current drawing mode
|
|
|
|
Point16 penPos; // current pen position
|
|
|
|
gFont *font; // current font
|
|
|
|
int16 textSpacing; // extra space between characters
|
|
|
|
uint16 textStyles; // text style bits
|
|
|
|
|
|
|
|
// Constructor
|
|
|
|
gPort(void) {
|
2021-05-27 18:39:28 +02:00
|
|
|
map = nullptr;
|
|
|
|
baseRow = nullptr;
|
|
|
|
|
|
|
|
rowMod = 0;
|
|
|
|
penMap = nullptr;
|
|
|
|
drawMode = drawModeMatte;
|
|
|
|
font = nullptr;
|
|
|
|
textSpacing = 0;
|
|
|
|
textStyles = 0;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual ~gPort() {}
|
|
|
|
|
|
|
|
// Set attributes
|
|
|
|
|
2021-06-13 16:56:52 +02:00
|
|
|
virtual void setMap(gPixelMap *newmap, bool inverted = false);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
|
|
|
// Direct colors
|
|
|
|
|
|
|
|
void setColor(gPen color) {
|
|
|
|
fgPen = color;
|
|
|
|
}
|
|
|
|
void setBgColor(gPen color) {
|
|
|
|
bgPen = color;
|
|
|
|
}
|
|
|
|
void setShadowColor(gPen color) {
|
|
|
|
shPen = color;
|
|
|
|
}
|
|
|
|
void setOutlineColor(gPen color) {
|
|
|
|
olPen = color;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Indirect colors
|
|
|
|
|
2021-05-27 11:03:05 +02:00
|
|
|
void setPenMap(gPen *pmap) {
|
|
|
|
penMap = pmap;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
void setIndirectColor(uint8 color) {
|
2021-06-07 18:19:10 +02:00
|
|
|
fgPen = penMap[color];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
void setIndirectBgColor(uint8 color) {
|
2021-06-07 18:19:10 +02:00
|
|
|
bgPen = penMap[color];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
void setIndirectShColor(uint8 color) {
|
2021-06-07 18:19:10 +02:00
|
|
|
shPen = penMap[color];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
void setIndirectOLColor(uint8 color) {
|
2021-06-07 18:19:10 +02:00
|
|
|
olPen = penMap[color];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// modes & styles
|
|
|
|
|
|
|
|
void setMode(enum draw_modes mode) {
|
|
|
|
drawMode = mode;
|
|
|
|
}
|
|
|
|
void setStyle(int style) {
|
|
|
|
textStyles = style;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pen states
|
|
|
|
|
|
|
|
void setState(gPenState &);
|
|
|
|
void getState(gPenState &);
|
|
|
|
|
|
|
|
// REM: calc intersection of pixel map rect...
|
|
|
|
|
|
|
|
void setClip(const Rect16 &newclip) {
|
|
|
|
clip = newclip;
|
|
|
|
}
|
|
|
|
void getClip(Rect16 &r) {
|
|
|
|
r = clip;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setOrigin(Point16 pt) {
|
|
|
|
origin = pt;
|
|
|
|
}
|
|
|
|
Point16 getOrigin(void) {
|
|
|
|
return origin;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pen position movement
|
|
|
|
|
|
|
|
void move(int16 x, int16 y) {
|
|
|
|
penPos.x += x;
|
|
|
|
penPos.y += y;
|
|
|
|
}
|
|
|
|
void move(Vector16 v) {
|
|
|
|
penPos += v;
|
|
|
|
}
|
|
|
|
void moveTo(int16 x, int16 y) {
|
|
|
|
penPos.x = x;
|
|
|
|
penPos.y = y;
|
|
|
|
}
|
|
|
|
void moveTo(Point16 p) {
|
|
|
|
penPos = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simple drawing functions
|
|
|
|
// REM: This should clip!
|
|
|
|
|
|
|
|
virtual void clear(void) {
|
|
|
|
memset(map->data, (int)fgPen, (int)map->bytes());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Functions to set a single pixel
|
|
|
|
// NOTE: should this actually be a function pointer to deal
|
|
|
|
// with drawing mode?
|
|
|
|
|
|
|
|
virtual void setPixel(int16 x, int16 y, gPen color) {
|
|
|
|
if (x >= clip.x && x < clip.x + clip.width
|
|
|
|
&& y >= clip.y && y < clip.y + clip.height) {
|
2021-06-07 18:19:10 +02:00
|
|
|
baseRow[(y + origin.y) * rowMod + x + origin.x] = color;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
void setPixel(int16 x, int16 y) {
|
|
|
|
setPixel(x, y, fgPen);
|
|
|
|
}
|
|
|
|
void setPixel(Point16 p, gPen color) {
|
|
|
|
setPixel(p.x, p.y, color);
|
|
|
|
}
|
|
|
|
void setPixel(Point16 p) {
|
|
|
|
setPixel(p.x, p.y, fgPen);
|
|
|
|
}
|
|
|
|
|
|
|
|
// pixel query functions
|
|
|
|
|
|
|
|
virtual gPen getPixel(int16 x, int16 y) {
|
2021-06-07 18:19:10 +02:00
|
|
|
return baseRow[(y + origin.y) * rowMod + x + origin.x];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
virtual gPen getPixel(Point16 p) {
|
2021-06-07 18:19:10 +02:00
|
|
|
return baseRow[(p.y + origin.y) * rowMod + p.x + origin.x];
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rectangle fill functions
|
|
|
|
|
|
|
|
virtual void fillRect(const Rect16 r);
|
|
|
|
void fillRect(int16 x, int16 y, int16 w, int16 h) {
|
|
|
|
fillRect(Rect16(x, y, w, h));
|
|
|
|
}
|
|
|
|
// Rectangle frame functions //
|
|
|
|
|
|
|
|
virtual void frameRect(const Rect16 r, int16 thick);
|
|
|
|
void frameRect(int16 x, int16 y, int16 w, int16 h, int16 thick) {
|
|
|
|
frameRect(Rect16(x, y, w, h), thick);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Horiztonal and vertical lines
|
|
|
|
|
|
|
|
virtual void hLine(int16 x, int16 y, int16 width);
|
|
|
|
virtual void vLine(int16 x, int16 y, int16 height);
|
|
|
|
|
|
|
|
// Bresenham line-drawing functions
|
|
|
|
|
|
|
|
virtual void line(int16 x1, int16 y1, int16 x2, int16 y2);
|
|
|
|
void line(Point16 from, Point16 to) {
|
|
|
|
line(from.x, from.y, to.x, to.y);
|
|
|
|
}
|
|
|
|
void drawTo(int16 x, int16 y) {
|
|
|
|
line(penPos.x, penPos.y, x, y);
|
|
|
|
penPos.x = x;
|
|
|
|
penPos.y = y;
|
|
|
|
}
|
|
|
|
void drawTo(Point16 to) {
|
|
|
|
line(penPos, to);
|
|
|
|
penPos = to;
|
|
|
|
}
|
|
|
|
void draw(int16 x, int16 y) {
|
|
|
|
line(penPos.x, penPos.y, penPos.x + x, penPos.y + y);
|
|
|
|
penPos.x += x;
|
|
|
|
penPos.y += y;
|
|
|
|
}
|
|
|
|
void draw(Vector16 v) {
|
|
|
|
line(penPos, v);
|
|
|
|
penPos += v;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Blitting functions
|
|
|
|
|
2021-06-21 22:26:31 +02:00
|
|
|
virtual void bltPixels(const gPixelMap &src,
|
2021-05-17 20:47:39 +02:00
|
|
|
int src_x, int src_y,
|
|
|
|
int dst_x, int dst_y,
|
|
|
|
int width, int height);
|
|
|
|
|
|
|
|
virtual void bltPixelMask(gPixelMap &src,
|
|
|
|
gPixelMap &msk,
|
|
|
|
int src_x, int src_y,
|
|
|
|
int dst_x, int dst_y,
|
|
|
|
int width, int height);
|
|
|
|
|
|
|
|
virtual void scrollPixels(const Rect16 r, int dx, int dy);
|
|
|
|
|
|
|
|
// Text rendering functions
|
|
|
|
|
|
|
|
void setFont(gFont *newFont) {
|
|
|
|
font = newFont;
|
|
|
|
}
|
|
|
|
void setTextSpacing(int16 fs) {
|
|
|
|
textSpacing = fs;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Unclipped text rendering
|
2021-05-27 15:44:09 +02:00
|
|
|
void drawStringChars(const char *, int16, gPixelMap &, int, int);
|
2021-05-17 20:47:39 +02:00
|
|
|
// Clipped but still low-level
|
2021-05-27 15:44:09 +02:00
|
|
|
int16 drawClippedString(const char *str, int16 len, int xpos, int ypos);
|
2021-05-17 20:47:39 +02:00
|
|
|
public:
|
|
|
|
|
|
|
|
// Draw a text string with the current settings at the
|
|
|
|
// current pen position. If the length is < 0, then use the
|
|
|
|
// natural string length.
|
2021-05-27 15:44:09 +02:00
|
|
|
void drawText(const char *str, int16 length = -1);
|
|
|
|
void drawTextInBox(const char *str, int16 length,
|
2021-05-17 20:47:39 +02:00
|
|
|
const Rect16 &r, int16 pos,
|
|
|
|
Point16 borders);
|
|
|
|
};
|
|
|
|
|
|
|
|
void mapImage(gPort &from, gPort &to, gPen map[]);
|
|
|
|
void mapImage(gPixelMap &from, gPixelMap &to, gPen map[]);
|
|
|
|
|
2021-06-22 12:42:03 +02:00
|
|
|
bool NewTempPort(gPort &, int width, int height);
|
2021-05-17 20:47:39 +02:00
|
|
|
void DisposeTempPort(gPort &);
|
|
|
|
|
|
|
|
/* ============================================================================ *
|
|
|
|
gSavePort: a class which auto-saves and auto-restores the port state
|
|
|
|
* ============================================================================ */
|
|
|
|
|
|
|
|
struct gSavePort {
|
|
|
|
gPenState state; // saved state of port
|
|
|
|
public:
|
|
|
|
gPort &port; // port which was saved
|
|
|
|
|
|
|
|
gSavePort(gPort &p) : port(p) {
|
|
|
|
port.getState(state);
|
|
|
|
}
|
|
|
|
~gSavePort() {
|
|
|
|
port.setState(state);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#define SAVE_GPORT_STATE(p) gSavePort sp( p )
|
|
|
|
|
|
|
|
} // end of namespace Saga2
|
|
|
|
|
|
|
|
#endif
|