scummvm/engines/tony/gfxcore.h
2021-12-26 18:48:43 +01:00

503 lines
13 KiB
C++

/* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
/*
* This code is based on original Tony Tough source code
*
* Copyright (c) 1997-2003 Nayma Software
*/
#ifndef TONY_GFXCORE_H
#define TONY_GFXCORE_H
#include "common/system.h"
#include "common/coroutines.h"
#include "tony/utils.h"
namespace Tony {
/****************************************************************************\
* Class prototype
\****************************************************************************/
// Class Name Family Treee Abstract?
class RMGfxTask; // Yes
class RMGfxTaskSetPrior; // Task Yes
class RMGfxBuffer; //
class RMGfxSourceBuffer; // TaskP+[Buffer] Yes
class RMGfxTargetBuffer; // [Buffer]
class RMGfxSourceBufferPal; // Source Yes
class RMGfxSourceBuffer4; // SourcePal
class RMGfxSourceBuffer8; // SourcePal
class RMGfxSourceBuffer16; // Source
class RMGfxWoodyBuffer; // Source16+Target
class RMGfxClearTask; // Task
/**
* Graphics buffer
*/
class RMGfxBuffer {
protected:
int _dimx, _dimy;
byte *_buf;
byte *_origBuf;
public:
RMGfxBuffer();
RMGfxBuffer(int dimx, int dimy, int nBpp);
virtual ~RMGfxBuffer();
// Attributes
int getDimx();
int getDimy();
// Creation
void create(int dimx, int dimy, int nBpp);
virtual void destroy();
// These are valid only if the buffer is locked
operator byte *();
operator void *();
// Getting the offset for a given Y position
void offsetY(int nLines, int nBpp);
};
/**
* Graphics primitive
*/
class RMGfxPrimitive {
public:
RMGfxTask *_task;
protected:
RMRect _src;
RMRect _dst;
bool _bStretch;
byte _bFlag;
public:
RMGfxPrimitive();
RMGfxPrimitive(RMGfxTask *task);
RMGfxPrimitive(RMGfxTask *task, const RMRect &src, RMRect &dst);
RMGfxPrimitive(RMGfxTask *task, const RMPoint &src, RMRect &dst);
RMGfxPrimitive(RMGfxTask *task, const RMPoint &src, RMPoint &dst);
RMGfxPrimitive(RMGfxTask *task, const RMRect &src, RMPoint &dst);
RMGfxPrimitive(RMGfxTask *task, const RMRect &dst);
RMGfxPrimitive(RMGfxTask *task, const RMPoint &dst);
virtual ~RMGfxPrimitive();
void setFlag(byte bFlag);
void setTask(RMGfxTask *task);
void setSrc(const RMRect &src);
void setSrc(const RMPoint &src);
void setDst(const RMRect &dst);
void setDst(const RMPoint &dst);
void setStretch(bool bStretch);
bool haveDst();
RMRect &getDst();
bool haveSrc();
RMRect &getSrc();
// Flags
bool isFlipped();
// Duplicate
virtual RMGfxPrimitive *duplicate();
};
/**
* Graphic drawing task
*/
class RMGfxTask {
protected:
int _nPrior;
int _nInList;
public:
// Standard constructor
RMGfxTask();
virtual ~RMGfxTask() { }
virtual int priority();
virtual void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) = 0;
virtual void removeThis(CORO_PARAM, bool &result);
// Registration
virtual void Register();
virtual void unregister();
};
/**
* Graphic drawing with priority
*/
class RMGfxTaskSetPrior : public RMGfxTask {
public:
~RMGfxTaskSetPrior() override { }
void setPriority(int nPrior);
};
/**
* Task that cleans the destination buffer
*/
class RMGfxClearTask : public RMGfxTask {
public:
~RMGfxClearTask() override { }
int priority() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
void removeThis(CORO_PARAM, bool &result) override;
};
/**
* Task that draws a colored box
*/
class RMGfxBox : public RMGfxTaskSetPrior {
protected:
uint16 _wFillColor;
public:
~RMGfxBox() override { }
void setColor(byte r, byte g, byte b);
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
void removeThis(CORO_PARAM, bool &result) override;
};
/**
* Buffer source for the design, which is a task. This is an abstract base.
*/
class RMGfxSourceBuffer : public virtual RMGfxBuffer, public RMGfxTaskSetPrior {
public:
// Load the data for the surface
virtual int init(uint32 resID, int dimx, int dimy, bool bLoadPalette = false);
virtual int init(const byte *buf, int dimx, int dimy, bool bLoadPalette = false);
virtual void init(Common::ReadStream &ds, int dimx, int dimy, bool bLoadPalette = false);
~RMGfxSourceBuffer() override;
protected:
virtual void prepareImage();
bool clip2D(int &x1, int &y1, int &u, int &v, int &width, int &height, bool bUseSrc, RMGfxTargetBuffer *buf);
void offsetY(int nLines);
public:
virtual int getBpp() = 0;
};
/**
* 16-bit color source
*/
class RMGfxSourceBuffer16 : public RMGfxSourceBuffer {
public:
void prepareImage() override;
protected:
bool _bTrasp0;
public:
RMGfxSourceBuffer16(bool bUseTrasp = false);
RMGfxSourceBuffer16(int dimx, int dimy);
~RMGfxSourceBuffer16() override;
// Initialization
void create(int dimx, int dimy);
int getBpp() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
/**
* Buffer source with palette
*/
class RMGfxSourceBufferPal : public RMGfxSourceBuffer {
protected:
// The size of the palette is (1 << Bpp()) * 4
byte _pal[256 * 3];
uint16 _palFinal[256];
// Post process to prepare the palette for drawing
virtual void preparePalette();
public:
~RMGfxSourceBufferPal() override;
int init(const byte *buf, int dimx, int dimy, bool bLoadPalette = false) override;
void init(Common::ReadStream &ds, int dimx, int dimy, bool bLoadPalette = false) override;
int loadPaletteWA(uint32 resID, bool bSwapped = false);
int loadPaletteWA(const byte *buf, bool bSwapped = false);
int loadPalette(uint32 resID);
int loadPalette(const byte *buf);
};
/**
* Buffer source with a 256 color palette
*/
class RMGfxSourceBuffer8 : public RMGfxSourceBufferPal {
protected:
bool _bTrasp0;
public:
RMGfxSourceBuffer8(bool bTrasp0 = true);
RMGfxSourceBuffer8(int dimx, int dimy);
~RMGfxSourceBuffer8() override;
// Initialization
void create(int dimx, int dimy);
int getBpp() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
/**
* Buffer source with a 256 color palette, and alpha blending
*/
class RMGfxSourceBuffer8AB : public RMGfxSourceBuffer8 {
protected:
int calcTrasp(int f, int b);
public:
~RMGfxSourceBuffer8AB() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
/**
* Buffer source with a 256 color palette, RLE compressed
*/
class RMGfxSourceBuffer8RLE : public virtual RMGfxSourceBuffer8 {
protected:
int _alphaBlendColor;
int _alphaR, _alphaB, _alphaG;
bool _bNeedRLECompress;
protected:
static byte _megaRLEBuf[];
virtual void rleWriteTrasp(byte *&cur, int rep) = 0;
virtual void rleWriteData(byte *&cur, int rep, byte *src) = 0;
virtual void rleWriteEOL(byte *&cur) = 0;
virtual void rleWriteAlphaBlend(byte *&cur, int rep) = 0;
virtual void rleDecompressLine(uint16 *dst, byte *src, int nStartSkip, int nLength) = 0;
virtual void rleDecompressLineFlipped(uint16 *dst, byte *src, int nStartSkip, int nLength) = 0;
// Perform image compression in RLE
void compressRLE();
protected:
// Overriding initialization methods
void prepareImage() override;
void preparePalette() override;
public:
RMGfxSourceBuffer8RLE();
~RMGfxSourceBuffer8RLE() override;
// Overload of the initialization method
void init(Common::ReadStream &ds, int dimx, int dimy, bool bLoadPalette = false) override;
int init(const byte *buf, int dimx, int dimy, bool bLoadPalette = false) override;
// Draw image with RLE decompression
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// Sets the color that will be alpha blended
void setAlphaBlendColor(int color);
// Warn if the data is already compressed
void setAlreadyCompressed();
};
class RMGfxSourceBuffer8RLEByte : public RMGfxSourceBuffer8RLE {
protected:
void rleWriteTrasp(byte * &cur, int rep) override;
void rleWriteAlphaBlend(byte * &cur, int rep) override;
void rleWriteData(byte * &cur, int rep, byte *src) override;
void rleWriteEOL(byte * &cur) override;
void rleDecompressLine(uint16 *dst, byte *src, int nStartSkip, int nLength) override;
void rleDecompressLineFlipped(uint16 *dst, byte *src, int nStartSkip, int nLength) override;
public:
~RMGfxSourceBuffer8RLEByte() override;
};
class RMGfxSourceBuffer8RLEWord : public RMGfxSourceBuffer8RLE {
protected:
void rleWriteTrasp(byte * &cur, int rep) override;
void rleWriteAlphaBlend(byte * &cur, int rep) override;
void rleWriteData(byte * &cur, int rep, byte *src) override;
void rleWriteEOL(byte * &cur) override;
void rleDecompressLine(uint16 *dst, byte *src, int nStartSkip, int nLength) override;
void rleDecompressLineFlipped(uint16 *dst, byte *src, int nStartSkip, int nLength) override;
public:
~RMGfxSourceBuffer8RLEWord() override;
};
class RMGfxSourceBuffer8RLEWordAB : public RMGfxSourceBuffer8RLEWord {
protected:
void rleDecompressLine(uint16 *dst, byte *src, int nStartSkip, int nLength) override;
public:
~RMGfxSourceBuffer8RLEWordAB() override;
};
/**
* Buffer source with a 256 color palette, with anti-aliasing
*/
class RMGfxSourceBuffer8AA : public virtual RMGfxSourceBuffer8 {
protected:
static byte _megaAABuf[];
static byte _megaAABuf2[];
byte *_aabuf;
// Calculate the buffer for the anti-aliasing
void calculateAA();
// Draw the AA
void drawAA(RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim);
protected:
void prepareImage() override;
public:
RMGfxSourceBuffer8AA();
~RMGfxSourceBuffer8AA() override;
// Draw with anti-aliasing
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
class RMGfxSourceBuffer8RLEByteAA : public RMGfxSourceBuffer8RLEByte, public RMGfxSourceBuffer8AA {
protected:
void prepareImage() override;
public:
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// Overloaded initialization methods
void init(Common::ReadStream &ds, int dimx, int dimy, bool bLoadPalette = false) override;
int init(const byte *buf, int dimx, int dimy, bool bLoadPalette = false) override;
~RMGfxSourceBuffer8RLEByteAA() override;
};
class RMGfxSourceBuffer8RLEWordAA : public RMGfxSourceBuffer8RLEWord, public RMGfxSourceBuffer8AA {
protected:
void prepareImage() override;
public:
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
// Overloaded initialization methods
void init(Common::ReadStream &ds, int dimx, int dimy, bool bLoadPalette = false) override;
int init(const byte *buf, int dimx, int dimy, bool bLoadPalette = false) override;
~RMGfxSourceBuffer8RLEWordAA() override;
};
/**
* Source buffer with 16 colors
*/
class RMGfxSourceBuffer4 : public RMGfxSourceBufferPal {
public:
RMGfxSourceBuffer4();
RMGfxSourceBuffer4(int dimx, int dimy);
// Initialization
void create(int dimx, int dimy);
int getBpp() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
/**
* Destination buffer which manages its own internal list of tasks
*/
class RMGfxTargetBuffer : public virtual RMGfxBuffer {
private:
struct OTList {
RMGfxPrimitive *_prim;
OTList *_next;
OTList();
OTList(RMGfxPrimitive *pr) {
_prim = pr;
_next = NULL;
}
};
bool _trackDirtyRects;
Common::List<Common::Rect> _currentDirtyRects, _previousDirtyRects, _dirtyRects;
void mergeDirtyRects();
private:
//Common::Mutex csModifyingOT;
protected:
OTList *_otlist;
int _otSize;
public:
RMGfxTargetBuffer();
~RMGfxTargetBuffer() override;
static uint16 *_precalcTable;
static void createBWPrecalcTable();
static void freeBWPrecalcTable();
// management of the OT list
void clearOT();
void drawOT(CORO_PARAM);
void addPrim(RMGfxPrimitive *prim); // The pointer must be delted
operator byte *();
operator void *();
operator uint16 *();
// Offseting buffer
void offsetY(int nLines);
// Dirty rect methods
void addDirtyRect(const Common::Rect &r);
Common::List<Common::Rect> &getDirtyRects();
void clearDirtyRects();
void setTrackDirtyRects(bool v);
bool getTrackDirtyRects() const;
};
/**
* Ring buffer, which is both source and by destination
*/
class RMGfxWoodyBuffer: public RMGfxSourceBuffer16, public RMGfxTargetBuffer {
public:
RMGfxWoodyBuffer();
RMGfxWoodyBuffer(int dimx, int dimy);
~RMGfxWoodyBuffer() override;
void draw(CORO_PARAM, RMGfxTargetBuffer &bigBuf, RMGfxPrimitive *prim) override;
};
} // End of namespace Tony
#endif