2009-04-15 14:03:31 +00:00
/*
* SDL_Extensions . h , part of VCMI engine
*
* Authors : listed in file AUTHORS in main folder
*
* License : GNU General Public License v2 .0 or later
* Full text of license available in license . txt file , in main folder
*
2009-04-16 11:14:13 +00:00
*/
2014-05-23 14:51:38 +04:00
# pragma once
# include <SDL_version.h>
# include <SDL_render.h>
# include <SDL_video.h>
2014-07-15 10:14:49 +03:00
# include <SDL_events.h>
2014-05-23 14:51:38 +04:00
# include "../../lib/int3.h"
# include "Geometries.h"
2014-07-15 10:14:49 +03:00
# include "../../lib/GameConstants.h"
2014-05-23 14:51:38 +04:00
2009-04-16 11:14:13 +00:00
2014-05-21 20:04:34 +04:00
extern SDL_Window * mainWindow ;
extern SDL_Renderer * mainRenderer ;
extern SDL_Texture * screenTexture ;
inline void SDL_SetColors ( SDL_Surface * surface , SDL_Color * colors , int firstcolor , int ncolors )
{
SDL_SetPaletteColors ( surface - > format - > palette , colors , firstcolor , ncolors ) ;
}
inline void SDL_WarpMouse ( int x , int y )
{
SDL_WarpMouseInWindow ( mainWindow , x , y ) ;
}
2014-06-01 19:31:37 +04:00
void SDL_UpdateRect ( SDL_Surface * surface , int x , int y , int w , int h ) ;
2014-05-21 20:04:34 +04:00
2014-05-21 23:14:05 +04:00
inline bool isCtrlKeyDown ( )
{
return SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_LCTRL ] | | SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_RCTRL ] ;
}
inline bool isAltKeyDown ( )
{
return SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_LALT ] | | SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_RALT ] ;
}
inline bool isShiftKeyDown ( )
{
return SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_LSHIFT ] | | SDL_GetKeyboardState ( nullptr ) [ SDL_SCANCODE_RSHIFT ] ;
2014-05-23 14:51:38 +04:00
}
namespace CSDL_Ext
{
2022-11-25 16:32:23 +02:00
template < typename Int >
Int lerp ( Int a , Int b , float f )
{
return a + std : : round ( ( b - a ) * f ) ;
}
2014-05-23 14:51:38 +04:00
//todo: should this better be assignment operator?
STRONG_INLINE void colorAssign ( SDL_Color & dest , const SDL_Color & source )
{
2015-02-18 15:31:55 +01:00
dest . r = source . r ;
2014-05-23 14:51:38 +04:00
dest . g = source . g ;
2015-02-18 15:31:55 +01:00
dest . b = source . b ;
2014-05-23 14:51:38 +04:00
dest . a = source . a ;
2015-02-18 15:31:55 +01:00
}
inline void setAlpha ( SDL_Surface * bg , int value )
{
SDL_SetSurfaceAlphaMod ( bg , value ) ;
2014-05-23 14:51:38 +04:00
}
2014-05-21 23:14:05 +04:00
}
2010-08-05 11:02:18 +00:00
struct Rect ;
2009-05-24 23:21:55 +00:00
extern SDL_Surface * screen , * screen2 , * screenBuf ;
2009-04-16 11:14:13 +00:00
void blitAt ( SDL_Surface * src , int x , int y , SDL_Surface * dst = screen ) ;
2009-06-17 09:38:03 +00:00
void blitAt ( SDL_Surface * src , const SDL_Rect & pos , SDL_Surface * dst = screen ) ;
2009-04-16 11:14:13 +00:00
bool isItIn ( const SDL_Rect * rect , int x , int y ) ;
2017-07-08 23:35:43 +02:00
bool isItInOrLowerBounds ( const SDL_Rect * rect , int x , int y ) ;
2009-04-17 14:01:22 +00:00
2012-11-11 12:23:31 +00:00
/**
* The colors class defines color constants of type SDL_Color .
*/
class Colors
2011-12-22 13:05:19 +00:00
{
2012-11-11 12:23:31 +00:00
public :
/** the h3 yellow color, typically used for headlines */
static const SDL_Color YELLOW ;
/** the standard h3 white color */
static const SDL_Color WHITE ;
2012-11-20 17:53:45 +00:00
/** the metallic gold color used mostly as a border around buttons */
static const SDL_Color METALLIC_GOLD ;
2012-12-19 17:24:53 +00:00
/** green color used for in-game console */
static const SDL_Color GREEN ;
2016-11-08 00:19:53 +03:00
2018-01-05 20:21:07 +03:00
/** the h3 orange color, used for blocked buttons */
static const SDL_Color ORANGE ;
/** the h3 bright yellow color, used for selection border */
static const SDL_Color BRIGHT_YELLOW ;
2014-07-02 22:20:54 +04:00
/** default key color for all 8 & 24 bit graphics */
static const SDL_Color DEFAULT_KEY_COLOR ;
2012-11-11 12:23:31 +00:00
} ;
2011-12-22 13:05:19 +00:00
2012-07-20 21:42:25 +00:00
//MSVC gives an error when calling abs with ui64 -> we add template that will match calls with unsigned arg and return it
template < typename T >
typename boost : : enable_if_c < boost : : is_unsigned < T > : : type , T > : : type abs ( T arg )
{
return arg ;
}
2009-06-26 15:41:19 +00:00
template < typename IntType >
2014-07-04 12:48:09 +03:00
std : : string makeNumberShort ( IntType number , IntType maxLength = 3 ) //the output is a string containing at most 5 characters [4 if positive] (eg. intead 10000 it gives 10k)
2009-06-26 15:41:19 +00:00
{
2014-07-04 12:48:09 +03:00
IntType max = pow ( 10 , maxLength ) ;
2015-10-12 17:08:18 +03:00
if ( std : : abs ( number ) < max )
2012-07-19 18:52:44 +00:00
return boost : : lexical_cast < std : : string > ( number ) ;
2009-06-26 15:41:19 +00:00
2014-07-04 12:48:09 +03:00
std : : string symbols = " kMGTPE " ;
2012-07-19 18:52:44 +00:00
auto iter = symbols . begin ( ) ;
2009-06-26 15:41:19 +00:00
2014-07-04 12:48:09 +03:00
while ( number > = max )
2009-06-26 15:41:19 +00:00
{
2012-07-19 18:52:44 +00:00
number / = 1000 ;
iter + + ;
2009-06-26 15:41:19 +00:00
2012-07-19 18:52:44 +00:00
assert ( iter ! = symbols . end ( ) ) ; //should be enough even for int64
}
return boost : : lexical_cast < std : : string > ( number ) + * iter ;
2009-06-26 15:41:19 +00:00
}
2010-08-05 11:02:18 +00:00
typedef void ( * TColorPutter ) ( Uint8 * & ptr , const Uint8 & R , const Uint8 & G , const Uint8 & B ) ;
2010-08-12 05:22:48 +00:00
typedef void ( * TColorPutterAlpha ) ( Uint8 * & ptr , const Uint8 & R , const Uint8 & G , const Uint8 & B , const Uint8 & A ) ;
2010-08-05 11:02:18 +00:00
2009-04-16 11:14:13 +00:00
inline SDL_Rect genRect ( const int & hh , const int & ww , const int & xx , const int & yy )
{
SDL_Rect ret ;
ret . h = hh ;
ret . w = ww ;
ret . x = xx ;
ret . y = yy ;
return ret ;
}
2010-08-05 11:02:18 +00:00
2010-08-12 05:22:48 +00:00
template < int bpp , int incrementPtr >
struct ColorPutter
{
static STRONG_INLINE void PutColor ( Uint8 * & ptr , const Uint8 & R , const Uint8 & G , const Uint8 & B ) ;
static STRONG_INLINE void PutColor ( Uint8 * & ptr , const Uint8 & R , const Uint8 & G , const Uint8 & B , const Uint8 & A ) ;
static STRONG_INLINE void PutColorAlphaSwitch ( Uint8 * & ptr , const Uint8 & R , const Uint8 & G , const Uint8 & B , const Uint8 & A ) ;
static STRONG_INLINE void PutColor ( Uint8 * & ptr , const SDL_Color & Color ) ;
static STRONG_INLINE void PutColorAlpha ( Uint8 * & ptr , const SDL_Color & Color ) ;
2011-02-20 09:24:53 +00:00
static STRONG_INLINE void PutColorRow ( Uint8 * & ptr , const SDL_Color & Color , size_t count ) ;
2010-08-12 05:22:48 +00:00
} ;
2010-08-05 11:02:18 +00:00
typedef void ( * BlitterWithRotationVal ) ( SDL_Surface * src , SDL_Rect srcRect , SDL_Surface * dst , SDL_Rect dstRect , ui8 rotation ) ;
2022-12-11 23:43:43 +02:00
/// Base class for applying palette transformation on images
2020-01-25 01:21:26 -08:00
class ColorShifter
{
public :
2022-11-25 00:26:14 +02:00
~ ColorShifter ( ) = default ;
virtual SDL_Color shiftColor ( SDL_Color input ) const = 0 ;
2020-01-25 01:21:26 -08:00
} ;
2022-12-11 23:43:43 +02:00
/// Generic class for palette transformation
/// formula:
/// result = input * factor + added
class ColorShifterMultiplyAndAdd : public ColorShifter
2020-01-25 01:21:26 -08:00
{
2022-12-11 23:43:43 +02:00
SDL_Color added ;
SDL_Color factor ;
2022-11-25 00:26:14 +02:00
2022-12-11 23:43:43 +02:00
public :
ColorShifterMultiplyAndAdd ( SDL_Color factor , SDL_Color added ) :
factor ( factor ) ,
added ( added )
2022-11-25 00:26:14 +02:00
{ }
SDL_Color shiftColor ( SDL_Color input ) const override
2020-01-25 01:21:26 -08:00
{
2022-11-25 00:26:14 +02:00
return {
2022-12-11 23:43:43 +02:00
uint8_t ( std : : min ( 255.f , std : : round ( input . r * float ( factor . r ) / 255.f + added . r ) ) ) ,
uint8_t ( std : : min ( 255.f , std : : round ( input . g * float ( factor . g ) / 255.f + added . g ) ) ) ,
uint8_t ( std : : min ( 255.f , std : : round ( input . b * float ( factor . b ) / 255.f + added . b ) ) ) ,
2022-12-12 00:04:46 +02:00
uint8_t ( std : : min ( 255.f , std : : round ( input . a * float ( factor . a ) / 255.f + added . a ) ) )
2022-11-25 00:26:14 +02:00
} ;
2020-01-25 01:21:26 -08:00
}
} ;
2022-12-11 23:43:43 +02:00
/// Color shifter that allows to specify color to be excempt from changes
class ColorShifterMultiplyAndAddExcept : public ColorShifterMultiplyAndAdd
2020-01-25 01:21:26 -08:00
{
2022-11-25 00:26:14 +02:00
SDL_Color ignored ;
2020-01-25 01:21:26 -08:00
public :
2022-12-11 23:43:43 +02:00
ColorShifterMultiplyAndAddExcept ( SDL_Color factor , SDL_Color added , SDL_Color ignored ) :
ColorShifterMultiplyAndAdd ( factor , added ) ,
2022-11-27 02:26:02 +02:00
ignored ( ignored )
2022-11-25 00:26:14 +02:00
{ }
SDL_Color shiftColor ( SDL_Color input ) const override
2020-01-25 01:21:26 -08:00
{
2022-11-25 00:26:14 +02:00
if ( input . r = = ignored . r & & input . g = = ignored . g & & input . b = = ignored . b & & input . a = = ignored . a )
return input ;
2022-12-11 23:43:43 +02:00
return ColorShifterMultiplyAndAdd : : shiftColor ( input ) ;
2020-01-25 01:21:26 -08:00
}
} ;
2009-04-16 11:14:13 +00:00
namespace CSDL_Ext
{
2012-07-21 20:16:54 +00:00
/// helper that will safely set and un-set ClipRect for SDL_Surface
class CClipRectGuard
{
SDL_Surface * surf ;
SDL_Rect oldRect ;
public :
CClipRectGuard ( SDL_Surface * surface , const SDL_Rect & rect ) :
2012-11-11 12:23:31 +00:00
surf ( surface )
2012-07-21 20:16:54 +00:00
{
SDL_GetClipRect ( surf , & oldRect ) ;
SDL_SetClipRect ( surf , & rect ) ;
}
~ CClipRectGuard ( )
{
SDL_SetClipRect ( surf , & oldRect ) ;
}
} ;
2016-11-08 00:19:53 +03:00
void blitSurface ( SDL_Surface * src , const SDL_Rect * srcRect , SDL_Surface * dst , SDL_Rect * dstRect ) ;
2010-08-23 16:13:30 +00:00
void fillRect ( SDL_Surface * dst , SDL_Rect * dstrect , Uint32 color ) ;
2014-07-05 01:46:55 +04:00
void fillRectBlack ( SDL_Surface * dst , SDL_Rect * dstrect ) ;
2012-07-21 20:16:54 +00:00
//fill dest image with source texture.
void fillTexture ( SDL_Surface * dst , SDL_Surface * sourceTexture ) ;
2009-04-16 11:14:13 +00:00
2010-08-05 11:02:18 +00:00
void SDL_PutPixelWithoutRefresh ( SDL_Surface * ekran , const int & x , const int & y , const Uint8 & R , const Uint8 & G , const Uint8 & B , Uint8 A = 255 ) ;
void SDL_PutPixelWithoutRefreshIfInSurf ( SDL_Surface * ekran , const int & x , const int & y , const Uint8 & R , const Uint8 & G , const Uint8 & B , Uint8 A = 255 ) ;
2009-04-16 11:14:13 +00:00
2013-11-06 13:42:58 +00:00
SDL_Surface * verticalFlip ( SDL_Surface * toRot ) ; //vertical flip
SDL_Surface * horizontalFlip ( SDL_Surface * toRot ) ; //horizontal flip
2009-04-16 11:14:13 +00:00
Uint32 SDL_GetPixel ( SDL_Surface * surface , const int & x , const int & y , bool colorByte = false ) ;
2010-02-18 12:44:59 +00:00
bool isTransparent ( SDL_Surface * srf , int x , int y ) ; //checks if surface is transparent at given position
2009-12-22 21:53:50 +00:00
2012-05-19 16:22:34 +00:00
Uint8 * getPxPtr ( const SDL_Surface * const & srf , const int x , const int y ) ;
2012-05-28 19:29:32 +00:00
TColorPutter getPutterFor ( SDL_Surface * const & dest , int incrementing ) ; //incrementing: -1, 0, 1
TColorPutterAlpha getPutterAlphaFor ( SDL_Surface * const & dest , int incrementing ) ; //incrementing: -1, 0, 1
2009-12-22 21:53:50 +00:00
2010-08-12 05:22:48 +00:00
template < int bpp >
int blit8bppAlphaTo24bppT ( const SDL_Surface * src , const SDL_Rect * srcRect , SDL_Surface * dst , SDL_Rect * dstRect ) ; //blits 8 bpp surface with alpha channel to 24 bpp surface
2009-05-28 02:58:29 +00:00
int blit8bppAlphaTo24bpp ( const SDL_Surface * src , const SDL_Rect * srcRect , SDL_Surface * dst , SDL_Rect * dstRect ) ; //blits 8 bpp surface with alpha channel to 24 bpp surface
2009-04-16 11:14:13 +00:00
Uint32 colorToUint32 ( const SDL_Color * color ) ; //little endian only
2013-07-07 08:27:27 +00:00
SDL_Color makeColor ( ui8 r , ui8 g , ui8 b , ui8 a ) ;
2009-08-17 08:50:31 +00:00
2009-04-16 11:14:13 +00:00
void update ( SDL_Surface * what = screen ) ; //updates whole surface (default - main screen)
2022-11-15 21:42:16 +02:00
void drawLine ( SDL_Surface * sur , int x1 , int y1 , int x2 , int y2 , const SDL_Color & color1 , const SDL_Color & color2 ) ;
2009-09-20 12:47:40 +00:00
void drawBorder ( SDL_Surface * sur , int x , int y , int w , int h , const int3 & color ) ;
void drawBorder ( SDL_Surface * sur , const SDL_Rect & r , const int3 & color ) ;
2011-12-22 13:05:19 +00:00
void drawDashedBorder ( SDL_Surface * sur , const Rect & r , const int3 & color ) ;
2013-03-03 17:06:03 +00:00
void setPlayerColor ( SDL_Surface * sur , PlayerColor player ) ; //sets correct color of flags; -1 for neutral
2009-04-16 11:14:13 +00:00
std : : string processStr ( std : : string str , std : : vector < std : : string > & tor ) ; //replaces %s in string
SDL_Surface * newSurface ( int w , int h , SDL_Surface * mod = screen ) ; //creates new surface, with flags/format same as in surface given
SDL_Surface * copySurface ( SDL_Surface * mod ) ; //returns copy of given surface
2012-05-18 17:35:46 +00:00
template < int bpp >
SDL_Surface * createSurfaceWithBpp ( int width , int height ) ; //create surface with give bits per pixels value
2010-07-15 17:13:17 +00:00
void VflipSurf ( SDL_Surface * surf ) ; //fluipis given surface by vertical axis
2012-05-17 10:44:48 +00:00
2012-06-15 17:08:19 +00:00
//scale surface to required size.
//nearest neighbour algorithm
SDL_Surface * scaleSurfaceFast ( SDL_Surface * surf , int width , int height ) ;
// bilinear filtering. Uses fallback to scaleSurfaceFast in case of indexed surfaces
2012-05-19 21:38:01 +00:00
SDL_Surface * scaleSurface ( SDL_Surface * surf , int width , int height ) ;
2012-05-17 10:44:48 +00:00
template < int bpp >
void applyEffectBpp ( SDL_Surface * surf , const SDL_Rect * rect , int mode ) ;
2010-08-17 12:48:34 +00:00
void applyEffect ( SDL_Surface * surf , const SDL_Rect * rect , int mode ) ; //mode: 0 - sepia, 1 - grayscale
2016-11-08 00:19:53 +03:00
2014-07-02 19:41:30 +04:00
void startTextInput ( SDL_Rect * where ) ;
void stopTextInput ( ) ;
2016-11-08 00:19:53 +03:00
2014-07-02 22:20:54 +04:00
void setColorKey ( SDL_Surface * surface , SDL_Color color ) ;
2017-09-08 14:25:12 +03:00
2014-07-02 22:20:54 +04:00
///set key-color to 0,255,255
void setDefaultColorKey ( SDL_Surface * surface ) ;
2017-09-08 14:25:12 +03:00
2014-07-02 22:20:54 +04:00
///set key-color to 0,255,255 only if it exactly mapped
void setDefaultColorKeyPresize ( SDL_Surface * surface ) ;
2012-09-15 19:16:16 +00:00
}