2016-03-01 15:47:10 +00:00
|
|
|
/*
|
|
|
|
** v_video.h
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2008 Randy Heit
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __V_VIDEO_H__
|
|
|
|
#define __V_VIDEO_H__
|
|
|
|
|
|
|
|
#include "doomtype.h"
|
2016-03-10 21:36:28 +00:00
|
|
|
#include "vectors.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
#include "doomdef.h"
|
|
|
|
#include "dobject.h"
|
|
|
|
#include "r_data/renderstyle.h"
|
|
|
|
#include "c_cvars.h"
|
|
|
|
|
|
|
|
extern int CleanWidth, CleanHeight, CleanXfac, CleanYfac;
|
|
|
|
extern int CleanWidth_1, CleanHeight_1, CleanXfac_1, CleanYfac_1;
|
|
|
|
extern int DisplayWidth, DisplayHeight, DisplayBits;
|
|
|
|
|
|
|
|
bool V_DoModeSetup (int width, int height, int bits);
|
2016-09-12 11:04:36 +00:00
|
|
|
void V_UpdateModeSize (int width, int height);
|
|
|
|
void V_OutputResized (int width, int height);
|
2016-03-01 15:47:10 +00:00
|
|
|
void V_CalcCleanFacs (int designwidth, int designheight, int realwidth, int realheight, int *cleanx, int *cleany, int *cx1=NULL, int *cx2=NULL);
|
|
|
|
|
|
|
|
class FTexture;
|
2017-03-15 15:47:42 +00:00
|
|
|
struct FColormap;
|
2018-03-24 07:40:24 +00:00
|
|
|
enum FTextureFormat : uint32_t;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// TagItem definitions for DrawTexture. As far as I know, tag lists
|
|
|
|
// originated on the Amiga.
|
|
|
|
//
|
|
|
|
// Think of TagItems as an array of the following structure:
|
|
|
|
//
|
|
|
|
// struct TagItem {
|
2017-03-09 19:19:55 +00:00
|
|
|
// uint32_t ti_Tag;
|
|
|
|
// uint32_t ti_Data;
|
2016-03-01 15:47:10 +00:00
|
|
|
// };
|
|
|
|
|
|
|
|
#define TAG_DONE (0) /* Used to indicate the end of the Tag list */
|
|
|
|
#define TAG_END (0) /* Ditto */
|
|
|
|
/* list pointed to in ti_Data */
|
|
|
|
|
2017-03-09 19:19:55 +00:00
|
|
|
#define TAG_USER ((uint32_t)(1u<<30))
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
DTA_Base = TAG_USER + 5000,
|
|
|
|
DTA_DestWidth, // width of area to draw to
|
|
|
|
DTA_DestHeight, // height of area to draw to
|
|
|
|
DTA_Alpha, // alpha value for translucency
|
|
|
|
DTA_FillColor, // color to stencil onto the destination (RGB is the color for truecolor drawers, A is the palette index for paletted drawers)
|
2017-02-04 23:17:29 +00:00
|
|
|
DTA_TranslationIndex, // translation table to recolor the source
|
2016-03-01 15:47:10 +00:00
|
|
|
DTA_AlphaChannel, // bool: the source is an alpha channel; used with DTA_FillColor
|
|
|
|
DTA_Clean, // bool: scale texture size and position by CleanXfac and CleanYfac
|
|
|
|
DTA_320x200, // bool: scale texture size and position to fit on a virtual 320x200 screen
|
|
|
|
DTA_Bottom320x200, // bool: same as DTA_320x200 but centers virtual screen on bottom for 1280x1024 targets
|
|
|
|
DTA_CleanNoMove, // bool: like DTA_Clean but does not reposition output position
|
|
|
|
DTA_CleanNoMove_1, // bool: like DTA_CleanNoMove, but uses Clean[XY]fac_1 instead
|
|
|
|
DTA_FlipX, // bool: flip image horizontally //FIXME: Does not work with DTA_Window(Left|Right)
|
|
|
|
DTA_ShadowColor, // color of shadow
|
|
|
|
DTA_ShadowAlpha, // alpha of shadow
|
|
|
|
DTA_Shadow, // set shadow color and alphas to defaults
|
|
|
|
DTA_VirtualWidth, // pretend the canvas is this wide
|
|
|
|
DTA_VirtualHeight, // pretend the canvas is this tall
|
|
|
|
DTA_TopOffset, // override texture's top offset
|
|
|
|
DTA_LeftOffset, // override texture's left offset
|
|
|
|
DTA_CenterOffset, // bool: override texture's left and top offsets and set them for the texture's middle
|
|
|
|
DTA_CenterBottomOffset,// bool: override texture's left and top offsets and set them for the texture's bottom middle
|
|
|
|
DTA_WindowLeft, // don't draw anything left of this column (on source, not dest)
|
|
|
|
DTA_WindowRight, // don't draw anything at or to the right of this column (on source, not dest)
|
|
|
|
DTA_ClipTop, // don't draw anything above this row (on dest, not source)
|
|
|
|
DTA_ClipBottom, // don't draw anything at or below this row (on dest, not source)
|
|
|
|
DTA_ClipLeft, // don't draw anything to the left of this column (on dest, not source)
|
|
|
|
DTA_ClipRight, // don't draw anything at or to the right of this column (on dest, not source)
|
|
|
|
DTA_Masked, // true(default)=use masks from texture, false=ignore masks
|
|
|
|
DTA_HUDRules, // use fullscreen HUD rules to position and size textures
|
2016-04-09 18:47:54 +00:00
|
|
|
DTA_HUDRulesC, // only used internally for marking HUD_HorizCenter
|
2016-03-01 15:47:10 +00:00
|
|
|
DTA_KeepRatio, // doesn't adjust screen size for DTA_Virtual* if the aspect ratio is not 4:3
|
|
|
|
DTA_RenderStyle, // same as render style for actors
|
2017-03-09 19:19:55 +00:00
|
|
|
DTA_ColorOverlay, // uint32_t: ARGB to overlay on top of image; limited to black for software
|
2016-03-01 15:47:10 +00:00
|
|
|
DTA_BilinearFilter, // bool: apply bilinear filtering to the image
|
|
|
|
DTA_SpecialColormap,// pointer to FSpecialColormapParameters (likely to be forever hardware-only)
|
|
|
|
DTA_ColormapStyle, // pointer to FColormapStyle (hardware-only)
|
|
|
|
DTA_Fullscreen, // Draw image fullscreen (same as DTA_VirtualWidth/Height with graphics size.)
|
|
|
|
|
|
|
|
// floating point duplicates of some of the above:
|
|
|
|
DTA_DestWidthF,
|
|
|
|
DTA_DestHeightF,
|
|
|
|
DTA_TopOffsetF,
|
|
|
|
DTA_LeftOffsetF,
|
|
|
|
DTA_VirtualWidthF,
|
|
|
|
DTA_VirtualHeightF,
|
|
|
|
DTA_WindowLeftF,
|
|
|
|
DTA_WindowRightF,
|
|
|
|
|
|
|
|
// For DrawText calls:
|
|
|
|
DTA_TextLen, // stop after this many characters, even if \0 not hit
|
|
|
|
DTA_CellX, // horizontal size of character cell
|
|
|
|
DTA_CellY, // vertical size of character cell
|
2017-03-29 19:22:05 +00:00
|
|
|
|
|
|
|
// New additions.
|
|
|
|
DTA_Color,
|
2016-03-01 15:47:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
HUD_Normal,
|
|
|
|
HUD_HorizCenter
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class FFont;
|
|
|
|
struct FRemapTable;
|
|
|
|
class player_t;
|
2017-03-09 18:31:45 +00:00
|
|
|
typedef uint32_t angle_t;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-04-09 10:55:12 +00:00
|
|
|
struct DrawParms
|
|
|
|
{
|
|
|
|
double x, y;
|
|
|
|
double texwidth;
|
|
|
|
double texheight;
|
|
|
|
double destwidth;
|
|
|
|
double destheight;
|
|
|
|
double virtWidth;
|
|
|
|
double virtHeight;
|
|
|
|
double windowleft;
|
|
|
|
double windowright;
|
2016-04-09 18:47:54 +00:00
|
|
|
int cleanmode;
|
2016-04-09 10:55:12 +00:00
|
|
|
int dclip;
|
|
|
|
int uclip;
|
|
|
|
int lclip;
|
|
|
|
int rclip;
|
|
|
|
double top;
|
|
|
|
double left;
|
|
|
|
float Alpha;
|
2017-03-09 18:31:45 +00:00
|
|
|
uint32_t fillcolor;
|
2016-04-09 10:55:12 +00:00
|
|
|
FRemapTable *remap;
|
2017-03-09 18:31:45 +00:00
|
|
|
uint32_t colorOverlay;
|
2017-03-29 19:22:05 +00:00
|
|
|
PalEntry color;
|
2016-04-09 10:55:12 +00:00
|
|
|
INTBOOL alphaChannel;
|
|
|
|
INTBOOL flipX;
|
2017-02-25 18:10:49 +00:00
|
|
|
//float shadowAlpha;
|
2016-04-09 10:55:12 +00:00
|
|
|
int shadowColor;
|
|
|
|
INTBOOL keepratio;
|
|
|
|
INTBOOL masked;
|
|
|
|
INTBOOL bilinear;
|
|
|
|
FRenderStyle style;
|
|
|
|
struct FSpecialColormap *specialcolormap;
|
|
|
|
struct FColormapStyle *colormapstyle;
|
2016-04-09 18:47:54 +00:00
|
|
|
int scalex, scaley;
|
|
|
|
int cellx, celly;
|
|
|
|
int maxstrlen;
|
|
|
|
bool fortext;
|
|
|
|
bool virtBottom;
|
2016-04-09 10:55:12 +00:00
|
|
|
};
|
|
|
|
|
2017-02-05 20:54:09 +00:00
|
|
|
struct Va_List
|
|
|
|
{
|
|
|
|
va_list list;
|
|
|
|
};
|
|
|
|
|
2017-02-04 21:09:49 +00:00
|
|
|
struct VMVa_List
|
|
|
|
{
|
|
|
|
VMValue *args;
|
|
|
|
int curindex;
|
|
|
|
int numargs;
|
|
|
|
};
|
2016-03-01 15:47:10 +00:00
|
|
|
//
|
|
|
|
// VIDEO
|
|
|
|
//
|
|
|
|
// [RH] Made screens more implementation-independant:
|
|
|
|
//
|
2017-04-14 08:48:18 +00:00
|
|
|
class DCanvas
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
public:
|
2016-05-31 07:36:18 +00:00
|
|
|
DCanvas (int width, int height, bool bgra);
|
2016-03-01 15:47:10 +00:00
|
|
|
virtual ~DCanvas ();
|
|
|
|
|
|
|
|
// Member variable access
|
2017-03-09 18:54:41 +00:00
|
|
|
inline uint8_t *GetBuffer () const { return Buffer; }
|
2016-03-01 15:47:10 +00:00
|
|
|
inline int GetWidth () const { return Width; }
|
|
|
|
inline int GetHeight () const { return Height; }
|
|
|
|
inline int GetPitch () const { return Pitch; }
|
2016-05-31 07:36:18 +00:00
|
|
|
inline bool IsBgra() const { return Bgra; }
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
virtual bool IsValid ();
|
|
|
|
|
|
|
|
// Access control
|
|
|
|
virtual bool Lock (bool buffered=true) = 0; // Returns true if the surface was lost since last time
|
|
|
|
virtual void Unlock () = 0;
|
|
|
|
virtual bool IsLocked () { return Buffer != NULL; } // Returns true if the surface is locked
|
|
|
|
|
|
|
|
// Draw a linear block of pixels into the canvas
|
2017-03-09 18:54:41 +00:00
|
|
|
virtual void DrawBlock (int x, int y, int width, int height, const uint8_t *src) const;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Reads a linear block of pixels into the view buffer.
|
2017-03-09 18:54:41 +00:00
|
|
|
virtual void GetBlock (int x, int y, int width, int height, uint8_t *dest) const;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Dim the entire canvas for the menus
|
|
|
|
virtual void Dim (PalEntry color = 0);
|
|
|
|
|
|
|
|
// Dim part of the canvas
|
2017-03-28 11:06:24 +00:00
|
|
|
virtual void Dim (PalEntry color, float amount, int x1, int y1, int w, int h) final;
|
|
|
|
virtual void DoDim(PalEntry color, float amount, int x1, int y1, int w, int h);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Fill an area with a texture
|
|
|
|
virtual void FlatFill (int left, int top, int right, int bottom, FTexture *src, bool local_origin=false);
|
|
|
|
|
|
|
|
// Fill a simple polygon with a texture
|
|
|
|
virtual void FillSimplePoly(FTexture *tex, FVector2 *points, int npoints,
|
2016-03-24 15:36:43 +00:00
|
|
|
double originx, double originy, double scalex, double scaley, DAngle rotation,
|
2017-03-15 15:47:42 +00:00
|
|
|
const FColormap &colormap, PalEntry flatcolor, int lightlevel, int bottomclip);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Set an area to a specified color
|
2017-03-28 11:06:24 +00:00
|
|
|
virtual void Clear (int left, int top, int right, int bottom, int palcolor, uint32_t color) final;
|
|
|
|
virtual void DoClear(int left, int top, int right, int bottom, int palcolor, uint32_t color);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Draws a line
|
2017-03-09 18:31:45 +00:00
|
|
|
virtual void DrawLine(int x0, int y0, int x1, int y1, int palColor, uint32_t realcolor);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Draws a single pixel
|
2017-03-09 18:31:45 +00:00
|
|
|
virtual void DrawPixel(int x, int y, int palcolor, uint32_t rgbcolor);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Calculate gamma table
|
2017-03-09 18:54:41 +00:00
|
|
|
void CalcGamma (float gamma, uint8_t gammalookup[256]);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Retrieves a buffer containing image data for a screenshot.
|
|
|
|
// Hint: Pitch can be negative for upside-down images, in which case buffer
|
|
|
|
// points to the last row in the buffer, which will be the first row output.
|
2017-12-06 08:59:20 +00:00
|
|
|
virtual void GetScreenshotBuffer(const uint8_t *&buffer, int &pitch, ESSType &color_type, float &gamma);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Releases the screenshot buffer.
|
|
|
|
virtual void ReleaseScreenshotBuffer();
|
|
|
|
|
|
|
|
// Text drawing functions -----------------------------------------------
|
|
|
|
|
|
|
|
// 2D Texture drawing
|
2017-03-28 11:06:24 +00:00
|
|
|
void ClearClipRect() { clipleft = cliptop = 0; clipwidth = clipheight = -1; }
|
|
|
|
void SetClipRect(int x, int y, int w, int h);
|
|
|
|
void GetClipRect(int *x, int *y, int *w, int *h);
|
|
|
|
|
2016-04-09 18:47:54 +00:00
|
|
|
bool SetTextureParms(DrawParms *parms, FTexture *img, double x, double y) const;
|
2016-04-11 08:46:30 +00:00
|
|
|
void DrawTexture (FTexture *img, double x, double y, int tags, ...);
|
2017-02-04 21:09:49 +00:00
|
|
|
void DrawTexture(FTexture *img, double x, double y, VMVa_List &);
|
2016-03-01 15:47:10 +00:00
|
|
|
void FillBorder (FTexture *img); // Fills the border around a 4:3 part of the screen on non-4:3 displays
|
|
|
|
void VirtualToRealCoords(double &x, double &y, double &w, double &h, double vwidth, double vheight, bool vbottom=false, bool handleaspect=true) const;
|
|
|
|
|
|
|
|
// Code that uses these (i.e. SBARINFO) should probably be evaluated for using doubles all around instead.
|
|
|
|
void VirtualToRealCoordsInt(int &x, int &y, int &w, int &h, int vwidth, int vheight, bool vbottom=false, bool handleaspect=true) const;
|
|
|
|
|
2016-04-09 18:47:54 +00:00
|
|
|
#ifdef DrawText
|
|
|
|
#undef DrawText // See WinUser.h for the definition of DrawText as a macro
|
2016-03-01 15:47:10 +00:00
|
|
|
#endif
|
2016-04-09 18:47:54 +00:00
|
|
|
// 2D Text drawing
|
2017-02-05 15:47:33 +00:00
|
|
|
void DrawText(FFont *font, int normalcolor, double x, double y, const char *string, int tag_first, ...);
|
|
|
|
void DrawText(FFont *font, int normalcolor, double x, double y, const char *string, VMVa_List &args);
|
|
|
|
void DrawChar(FFont *font, int normalcolor, double x, double y, int character, int tag_first, ...);
|
|
|
|
void DrawChar(FFont *font, int normalcolor, double x, double y, int character, VMVa_List &args);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
protected:
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t *Buffer;
|
2016-03-01 15:47:10 +00:00
|
|
|
int Width;
|
|
|
|
int Height;
|
|
|
|
int Pitch;
|
|
|
|
int LockCount;
|
2016-05-31 07:36:18 +00:00
|
|
|
bool Bgra;
|
2017-03-28 11:06:24 +00:00
|
|
|
int clipleft = 0, cliptop = 0, clipwidth = -1, clipheight = -1;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-02-05 15:47:33 +00:00
|
|
|
void DrawTextCommon(FFont *font, int normalcolor, double x, double y, const char *string, DrawParms &parms);
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
bool ClipBox (int &left, int &top, int &width, int &height, const uint8_t *&src, const int srcpitch) const;
|
2017-03-09 18:31:45 +00:00
|
|
|
void DrawTextureV(FTexture *img, double x, double y, uint32_t tag, va_list tags) = delete;
|
2016-04-09 10:59:50 +00:00
|
|
|
virtual void DrawTextureParms(FTexture *img, DrawParms &parms);
|
2017-02-04 21:09:49 +00:00
|
|
|
|
|
|
|
template<class T>
|
2017-03-09 19:19:55 +00:00
|
|
|
bool ParseDrawTextureTags(FTexture *img, double x, double y, uint32_t tag, T& tags, DrawParms *parms, bool fortext) const;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
DCanvas() {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Keep track of canvases, for automatic destruction at exit
|
|
|
|
DCanvas *Next;
|
|
|
|
static DCanvas *CanvasChain;
|
|
|
|
};
|
|
|
|
|
|
|
|
// A canvas in system memory.
|
|
|
|
|
|
|
|
class DSimpleCanvas : public DCanvas
|
|
|
|
{
|
2017-04-14 08:48:18 +00:00
|
|
|
typedef DCanvas Super;
|
2016-03-01 15:47:10 +00:00
|
|
|
public:
|
2016-05-31 07:36:18 +00:00
|
|
|
DSimpleCanvas (int width, int height, bool bgra);
|
2016-03-01 15:47:10 +00:00
|
|
|
~DSimpleCanvas ();
|
|
|
|
|
|
|
|
bool IsValid ();
|
|
|
|
bool Lock (bool buffered=true);
|
|
|
|
void Unlock ();
|
|
|
|
|
|
|
|
protected:
|
2016-09-12 11:04:36 +00:00
|
|
|
void Resize(int width, int height);
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t *MemBuffer;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
DSimpleCanvas() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
// This class represents a native texture, as opposed to an FTexture.
|
|
|
|
class FNativeTexture
|
|
|
|
{
|
2018-03-23 22:04:30 +00:00
|
|
|
protected:
|
|
|
|
FTexture * mGameTex;
|
|
|
|
FTextureFormat mFormat;
|
2016-03-01 15:47:10 +00:00
|
|
|
public:
|
2018-03-23 22:04:30 +00:00
|
|
|
FNativeTexture(FTexture *tex, FTextureFormat fmt) : mGameTex(tex), mFormat(fmt) {}
|
2016-03-01 15:47:10 +00:00
|
|
|
virtual ~FNativeTexture();
|
|
|
|
virtual bool Update() = 0;
|
|
|
|
virtual bool CheckWrapping(bool wrapping);
|
|
|
|
};
|
|
|
|
|
|
|
|
// This class represents a texture lookup palette.
|
|
|
|
class FNativePalette
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~FNativePalette();
|
|
|
|
virtual bool Update() = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
// A canvas that represents the actual display. The video code is responsible
|
|
|
|
// for actually implementing this. Built on top of SimpleCanvas, because it
|
|
|
|
// needs a system memory buffer when buffered output is enabled.
|
|
|
|
|
|
|
|
class DFrameBuffer : public DSimpleCanvas
|
|
|
|
{
|
2017-04-14 08:48:18 +00:00
|
|
|
typedef DSimpleCanvas Super;
|
2016-03-01 15:47:10 +00:00
|
|
|
public:
|
2016-05-31 07:36:18 +00:00
|
|
|
DFrameBuffer (int width, int height, bool bgra);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Force the surface to use buffered output if true is passed.
|
|
|
|
virtual bool Lock (bool buffered) = 0;
|
|
|
|
|
|
|
|
// Make the surface visible. Also implies Unlock().
|
|
|
|
virtual void Update () = 0;
|
|
|
|
|
|
|
|
// Return a pointer to 256 palette entries that can be written to.
|
|
|
|
virtual PalEntry *GetPalette () = 0;
|
|
|
|
|
|
|
|
// Stores the palette with flash blended in into 256 dwords
|
|
|
|
virtual void GetFlashedPalette (PalEntry palette[256]) = 0;
|
|
|
|
|
|
|
|
// Mark the palette as changed. It will be updated on the next Update().
|
|
|
|
virtual void UpdatePalette () = 0;
|
|
|
|
|
|
|
|
// Sets the gamma level. Returns false if the hardware does not support
|
|
|
|
// gamma changing. (Always true for now, since palettes can always be
|
|
|
|
// gamma adjusted.)
|
|
|
|
virtual bool SetGamma (float gamma) = 0;
|
|
|
|
|
|
|
|
// Sets a color flash. RGB is the color, and amount is 0-256, with 256
|
|
|
|
// being all flash and 0 being no flash. Returns false if the hardware
|
|
|
|
// does not support this. (Always true for now, since palettes can always
|
|
|
|
// be flashed.)
|
|
|
|
virtual bool SetFlash (PalEntry rgb, int amount) = 0;
|
|
|
|
|
|
|
|
// Converse of SetFlash
|
|
|
|
virtual void GetFlash (PalEntry &rgb, int &amount) = 0;
|
|
|
|
|
|
|
|
// Returns the number of video pages the frame buffer is using.
|
|
|
|
virtual int GetPageCount () = 0;
|
|
|
|
|
|
|
|
// Returns true if running fullscreen.
|
|
|
|
virtual bool IsFullscreen () = 0;
|
|
|
|
|
|
|
|
// Changes the vsync setting, if supported by the device.
|
|
|
|
virtual void SetVSync (bool vsync);
|
|
|
|
|
|
|
|
// Tells the device to recreate itself with the new setting from vid_refreshrate.
|
|
|
|
virtual void NewRefreshRate ();
|
|
|
|
|
|
|
|
// Set the rect defining the area affected by blending.
|
|
|
|
virtual void SetBlendingRect (int x1, int y1, int x2, int y2);
|
|
|
|
|
|
|
|
bool Accel2D; // If true, 2D drawing can be accelerated.
|
2018-02-13 22:06:59 +00:00
|
|
|
virtual bool LegacyHardware() const { return false; } // only for reporting SM1.4 support to the stat collector
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Begin 2D drawing operations. This is like Update, but it doesn't end
|
|
|
|
// the scene, and it doesn't present the image yet. If you are going to
|
|
|
|
// be covering the entire screen with 2D elements, then pass false to
|
|
|
|
// avoid copying the software buffer to the screen.
|
|
|
|
// Returns true if hardware-accelerated 2D has been entered, false if not.
|
|
|
|
virtual bool Begin2D(bool copy3d);
|
2017-04-01 10:59:58 +00:00
|
|
|
void End2D() { isIn2D = false; }
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-03-28 22:45:53 +00:00
|
|
|
// Returns true if Begin2D has been called and 2D drawing is now active
|
2017-04-01 10:59:58 +00:00
|
|
|
bool HasBegun2D() { return isIn2D; }
|
2017-03-28 22:45:53 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
// DrawTexture calls after Begin2D use native textures.
|
|
|
|
|
|
|
|
// Draws the blending rectangle over the viewwindow if in hardware-
|
|
|
|
// accelerated 2D mode.
|
|
|
|
virtual void DrawBlendingRect();
|
|
|
|
|
|
|
|
// Create a native texture from a game texture.
|
2018-03-23 22:04:30 +00:00
|
|
|
virtual FNativeTexture *CreateTexture(FTexture *gametex, FTextureFormat fmt, bool wrapping);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Create a palette texture from a remap/palette table.
|
|
|
|
virtual FNativePalette *CreatePalette(FRemapTable *remap);
|
|
|
|
|
|
|
|
// Precaches or unloads a texture
|
2016-05-01 20:47:36 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
// Report a game restart
|
|
|
|
virtual void GameRestart();
|
|
|
|
|
|
|
|
// Screen wiping
|
|
|
|
virtual bool WipeStartScreen(int type);
|
|
|
|
virtual void WipeEndScreen();
|
|
|
|
virtual bool WipeDo(int ticks);
|
|
|
|
virtual void WipeCleanup();
|
|
|
|
|
2017-03-09 18:31:45 +00:00
|
|
|
virtual void ScaleCoordsFromWindow(int16_t &x, int16_t &y) {}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-11-16 01:33:08 +00:00
|
|
|
uint64_t GetLastFPS() const { return LastCount; }
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
virtual void PaletteChanged () = 0;
|
|
|
|
virtual int QueryNewPalette () = 0;
|
|
|
|
virtual bool Is8BitMode() = 0;
|
|
|
|
#endif
|
|
|
|
|
2016-09-14 21:38:11 +00:00
|
|
|
// The original size of the framebuffer as selected in the video menu.
|
|
|
|
int VideoWidth = 0;
|
|
|
|
int VideoHeight = 0;
|
2017-11-14 20:52:54 +00:00
|
|
|
uint64_t FrameTime = 0;
|
2016-09-14 21:38:11 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
protected:
|
|
|
|
void DrawRateStuff ();
|
2017-03-09 18:54:41 +00:00
|
|
|
void CopyFromBuff (uint8_t *src, int srcPitch, int width, int height, uint8_t *dest);
|
|
|
|
void CopyWithGammaBgra(void *output, int pitch, const uint8_t *gammared, const uint8_t *gammagreen, const uint8_t *gammablue, PalEntry flash, int flash_amount);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
DFrameBuffer () {}
|
|
|
|
|
|
|
|
private:
|
2017-11-16 01:33:08 +00:00
|
|
|
uint64_t LastMS, LastSec, FrameCount, LastCount, LastTic;
|
2017-04-01 10:59:58 +00:00
|
|
|
bool isIn2D = false;
|
2016-03-01 15:47:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// This is the screen updated by I_FinishUpdate.
|
|
|
|
extern DFrameBuffer *screen;
|
|
|
|
|
|
|
|
#define SCREENWIDTH (screen->GetWidth ())
|
|
|
|
#define SCREENHEIGHT (screen->GetHeight ())
|
|
|
|
#define SCREENPITCH (screen->GetPitch ())
|
|
|
|
|
|
|
|
EXTERN_CVAR (Float, Gamma)
|
|
|
|
|
|
|
|
// Translucency tables
|
|
|
|
|
|
|
|
// RGB32k is a normal R5G5B5 -> palette lookup table.
|
|
|
|
|
|
|
|
// Use a union so we can "overflow" without warnings.
|
|
|
|
// Otherwise, we get stuff like this from Clang (when compiled
|
|
|
|
// with -fsanitize=bounds) while running:
|
2017-03-09 18:54:41 +00:00
|
|
|
// src/v_video.cpp:390:12: runtime error: index 1068 out of bounds for type 'uint8_t [32]'
|
|
|
|
// src/r_draw.cpp:273:11: runtime error: index 1057 out of bounds for type 'uint8_t [32]'
|
2016-03-01 15:47:10 +00:00
|
|
|
union ColorTable32k
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t RGB[32][32][32];
|
|
|
|
uint8_t All[32 *32 *32];
|
2016-03-01 15:47:10 +00:00
|
|
|
};
|
2017-03-11 10:31:09 +00:00
|
|
|
extern ColorTable32k RGB32k;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-19 12:18:20 +00:00
|
|
|
// [SP] RGB666 support
|
|
|
|
union ColorTable256k
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t RGB[64][64][64];
|
|
|
|
uint8_t All[64 *64 *64];
|
2016-12-19 12:18:20 +00:00
|
|
|
};
|
2017-03-11 10:31:09 +00:00
|
|
|
extern ColorTable256k RGB256k;
|
2016-12-19 12:18:20 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
// Col2RGB8 is a pre-multiplied palette for color lookup. It is stored in a
|
|
|
|
// special R10B10G10 format for efficient blending computation.
|
|
|
|
// --RRRRRrrr--BBBBBbbb--GGGGGggg-- at level 64
|
|
|
|
// --------rrrr------bbbb------gggg at level 1
|
2017-03-11 10:31:09 +00:00
|
|
|
extern uint32_t Col2RGB8[65][256];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Col2RGB8_LessPrecision is the same as Col2RGB8, but the LSB for red
|
|
|
|
// and blue are forced to zero, so if the blend overflows, it won't spill
|
|
|
|
// over into the next component's value.
|
|
|
|
// --RRRRRrrr-#BBBBBbbb-#GGGGGggg-- at level 64
|
|
|
|
// --------rrr#------bbb#------gggg at level 1
|
2017-03-11 10:31:09 +00:00
|
|
|
extern uint32_t *Col2RGB8_LessPrecision[65];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Col2RGB8_Inverse is the same as Col2RGB8_LessPrecision, except the source
|
|
|
|
// palette has been inverted.
|
2017-03-11 10:31:09 +00:00
|
|
|
extern uint32_t Col2RGB8_Inverse[65][256];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// "Magic" numbers used during the blending:
|
|
|
|
// --000001111100000111110000011111 = 0x01f07c1f
|
|
|
|
// -0111111111011111111101111111111 = 0x3FEFFBFF
|
|
|
|
// -1000000000100000000010000000000 = 0x40100400
|
|
|
|
// ------10000000001000000000100000 = 0x40100400 >> 5
|
|
|
|
// --11111-----11111-----11111----- = 0x40100400 - (0x40100400 >> 5) aka "white"
|
|
|
|
// --111111111111111111111111111111 = 0x3FFFFFFF
|
|
|
|
|
|
|
|
// Allocates buffer screens, call before R_Init.
|
|
|
|
void V_Init (bool restart);
|
|
|
|
|
|
|
|
// Initializes graphics mode for the first time.
|
|
|
|
void V_Init2 ();
|
|
|
|
|
|
|
|
void V_Shutdown ();
|
|
|
|
|
2016-12-03 15:27:53 +00:00
|
|
|
class FScanner;
|
2017-04-12 23:12:04 +00:00
|
|
|
struct FScriptPosition;
|
2016-03-01 15:47:10 +00:00
|
|
|
// Returns the closest color to the one desired. String
|
|
|
|
// should be of the form "rr gg bb".
|
2017-03-09 19:19:55 +00:00
|
|
|
int V_GetColorFromString (const uint32_t *palette, const char *colorstring, FScriptPosition *sc = nullptr);
|
2016-03-01 15:47:10 +00:00
|
|
|
// Scans through the X11R6RGB lump for a matching color
|
|
|
|
// and returns a color string suitable for V_GetColorFromString.
|
2016-12-03 15:27:53 +00:00
|
|
|
FString V_GetColorStringByName (const char *name, FScriptPosition *sc = nullptr);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
// Tries to get color by name, then by string
|
2017-03-09 19:19:55 +00:00
|
|
|
int V_GetColor (const uint32_t *palette, const char *str, FScriptPosition *sc = nullptr);
|
|
|
|
int V_GetColor(const uint32_t *palette, FScanner &sc);
|
2016-03-01 15:47:10 +00:00
|
|
|
void V_DrawFrame (int left, int top, int width, int height);
|
|
|
|
|
|
|
|
// If the view size is not full screen, draws a border around it.
|
|
|
|
void V_DrawBorder (int x1, int y1, int x2, int y2);
|
|
|
|
void V_RefreshViewBorder ();
|
|
|
|
|
|
|
|
void V_SetBorderNeedRefresh();
|
|
|
|
|
|
|
|
int CheckRatio (int width, int height, int *trueratio=NULL);
|
|
|
|
static inline int CheckRatio (double width, double height) { return CheckRatio(int(width), int(height)); }
|
2016-09-12 13:51:50 +00:00
|
|
|
inline bool IsRatioWidescreen(int ratio) { return (ratio & 3) != 0; }
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-09-12 11:59:01 +00:00
|
|
|
float ActiveRatio (int width, int height, float *trueratio = NULL);
|
|
|
|
static inline double ActiveRatio (double width, double height) { return ActiveRatio(int(width), int(height)); }
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-09-12 12:37:10 +00:00
|
|
|
int AspectBaseWidth(float aspect);
|
|
|
|
int AspectBaseHeight(float aspect);
|
2016-09-13 09:46:05 +00:00
|
|
|
double AspectPspriteOffset(float aspect);
|
2016-09-12 12:37:10 +00:00
|
|
|
int AspectMultiplier(float aspect);
|
2016-09-13 21:26:30 +00:00
|
|
|
bool AspectTallerThanWide(float aspect);
|
2017-05-13 21:39:54 +00:00
|
|
|
void ScaleWithAspect(int &w, int &h, int Width, int Height);
|
|
|
|
|
2017-03-30 00:16:23 +00:00
|
|
|
int GetUIScale(int altval);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-09-06 17:48:14 +00:00
|
|
|
EXTERN_CVAR(Int, uiscale);
|
2017-03-30 00:16:23 +00:00
|
|
|
EXTERN_CVAR(Int, con_scaletext);
|
|
|
|
EXTERN_CVAR(Int, con_scale);
|
|
|
|
|
|
|
|
inline int active_con_scaletext()
|
|
|
|
{
|
|
|
|
return GetUIScale(con_scaletext);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int active_con_scale()
|
|
|
|
{
|
|
|
|
return GetUIScale(con_scale);
|
|
|
|
}
|
|
|
|
|
2016-09-06 17:48:14 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
#endif // __V_VIDEO_H__
|