/*==LICENSE==*
CyanWorlds . com Engine - MMOG client , server and tools
Copyright ( C ) 2011 Cyan Worlds , Inc .
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/>.
Additional permissions under GNU GPL version 3 section 7
If you modify this Program , or any covered work , by linking or
combining it with any of RAD Game Tools Bink SDK , Autodesk 3 ds Max SDK ,
NVIDIA PhysX SDK , Microsoft DirectX SDK , OpenSSL library , Independent
JPEG Group JPEG library , Microsoft Windows Media SDK , or Apple QuickTime SDK
( or a modified version of those libraries ) ,
containing parts covered by the terms of the Bink SDK EULA , 3 ds Max EULA ,
PhysX SDK EULA , DirectX SDK EULA , OpenSSL and SSLeay licenses , IJG
JPEG Library README , Windows Media SDK EULA , or QuickTime SDK EULA , the
licensors of this Program grant you additional
permission to convey the resulting work . Corresponding Source for a
non - source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work .
You can contact Cyan Worlds , Inc . by email legal @ cyan . com
or by snail mail at :
Cyan Worlds , Inc .
14617 N Newport Hwy
Mead , WA 99021
* = = LICENSE = = */
///////////////////////////////////////////////////////////////////////////////
// //
// plDynSurfaceWriter Class Header //
// Abstract class wrapping around Windows GDI functionality for writing to //
// a generic RGBA surface. Allows us to create one writer per DTMap or a //
// single shared writer to conserve OS resources on 98/ME. //
// //
// Cyan, Inc. //
// //
//// Version History //////////////////////////////////////////////////////////
// //
// 10.28.2002 mcn - Created. //
// //
///////////////////////////////////////////////////////////////////////////////
# include "HeadSpin.h"
# include "plDynSurfaceWriter.h"
# include "plDynamicTextMap.h"
# include "hsExceptions.h"
# include "hsMatrix44.h"
# include "plMessage/plDynamicTextMsg.h"
# include "pnKeyedObject/plKey.h"
# include "plProfile.h"
# include "plStatusLog/plStatusLog.h"
# include "plWinFontCache.h"
//// plWinSurface Helper Functions ////////////////////////////////////////////
# if HS_BUILD_FOR_WIN32
static uint32_t sNumDCsAllocated ;
static uint32_t sNumBitmapsAllocated ;
plDynSurfaceWriter : : plWinSurface : : plWinSurface ( )
{
fDC = nil ;
fBitmap = nil ;
fFont = nil ;
fBits = nil ;
fTextColor = RGB ( 255 , 255 , 255 ) ;
fWidth = fHeight = 0 ;
fSaveNum = 0 ;
fFontFace = nil ;
fFontSize = 0 ;
fFontFlags = 0 ;
fFontAntiAliasRGB = false ;
fFontBlockedRGB = false ;
}
plDynSurfaceWriter : : plWinSurface : : ~ plWinSurface ( )
{
Release ( ) ;
}
void plDynSurfaceWriter : : plWinSurface : : Allocate ( uint16_t w , uint16_t h )
{
int i ;
BITMAPINFO * bmi ;
Release ( ) ;
fWidth = w ;
fHeight = h ;
/// Initialize a bitmap info struct to describe our surface
if ( IBitsPerPixel ( ) = = 8 )
bmi = ( BITMAPINFO * ) ( TRACKED_NEW uint8_t [ sizeof ( BITMAPINFOHEADER ) + sizeof ( RGBQUAD ) * 256 ] ) ;
else
bmi = TRACKED_NEW BITMAPINFO ;
memset ( & bmi - > bmiHeader , 0 , sizeof ( BITMAPINFOHEADER ) ) ;
bmi - > bmiHeader . biSize = sizeof ( BITMAPINFOHEADER ) ;
bmi - > bmiHeader . biWidth = ( int ) fWidth ;
bmi - > bmiHeader . biHeight = - ( int ) fHeight ;
bmi - > bmiHeader . biPlanes = 1 ;
bmi - > bmiHeader . biCompression = BI_RGB ;
bmi - > bmiHeader . biBitCount = IBitsPerPixel ( ) ;
if ( IBitsPerPixel ( ) = = 8 )
{
// Set up map for grayscale bitmap
for ( i = 0 ; i < 256 ; i + + )
{
bmi - > bmiColors [ i ] . rgbRed = i ;
bmi - > bmiColors [ i ] . rgbGreen = i ;
bmi - > bmiColors [ i ] . rgbBlue = i ;
bmi - > bmiColors [ i ] . rgbReserved = i ;
}
}
/// Create a screen-compatible DC
fDC = CreateCompatibleDC ( nil ) ;
if ( fDC = = nil )
{
char msg [ 256 ] ;
FormatMessage ( FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS , nil , GetLastError ( ) , MAKELANGID ( LANG_NEUTRAL , SUBLANG_DEFAULT ) , msg , sizeof ( msg ) , nil ) ;
char * ret = strrchr ( msg , ' \n ' ) ;
if ( ret ! = nil )
* ret = 0 ;
plStatusLog : : AddLineS ( " pipeline.log " , 0xffff0000 , " Unable to allocate DC for dynamic text map (%s, %d DCs allocated already) " , msg , sNumDCsAllocated ) ;
if ( IBitsPerPixel ( ) = = 8 )
delete [ ] bmi ;
else
delete bmi ;
return ;
}
sNumDCsAllocated + + ;
/// Create a bitmap using the DC and the bitmapInfo struct we filled out
fBitmap = CreateDIBSection ( fDC , bmi , DIB_RGB_COLORS , ( void * * ) & fBits , nil , 0 ) ;
if ( fBitmap = = nil )
{
char msg [ 256 ] ;
FormatMessage ( FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS , nil , GetLastError ( ) , MAKELANGID ( LANG_NEUTRAL , SUBLANG_DEFAULT ) , msg , sizeof ( msg ) , nil ) ;
char * ret = strrchr ( msg , ' \n ' ) ;
if ( ret ! = nil )
* ret = 0 ;
plStatusLog : : AddLineS ( " pipeline.log " , 0xffff0000 , " Unable to allocate RGB DIB section for dynamic text map (%s, %d bitmaps allocated already) " , msg , sNumBitmapsAllocated ) ;
if ( IBitsPerPixel ( ) = = 8 )
delete [ ] bmi ;
else
delete bmi ;
return ;
}
sNumBitmapsAllocated + + ;
/// Set up some basic props
SetMapMode ( fDC , MM_TEXT ) ;
SetBkMode ( fDC , TRANSPARENT ) ;
SetTextAlign ( fDC , TA_TOP | TA_LEFT ) ;
fSaveNum = SaveDC ( fDC ) ;
SelectObject ( fDC , fBitmap ) ;
if ( IBitsPerPixel ( ) = = 8 )
delete [ ] bmi ;
else
delete bmi ;
}
void plDynSurfaceWriter : : plWinSurface : : Release ( void )
{
if ( fBitmap ! = nil )
sNumBitmapsAllocated - - ;
if ( fDC ! = nil )
sNumDCsAllocated - - ;
if ( fSaveNum ! = 0 )
RestoreDC ( fDC , fSaveNum ) ;
fSaveNum = 0 ;
DeleteObject ( fBitmap ) ;
DeleteDC ( fDC ) ;
fDC = nil ;
fBitmap = nil ;
fFont = nil ;
fBits = nil ;
fWidth = fHeight = 0 ;
delete [ ] fFontFace ;
fFontFace = nil ;
fFontSize = 0 ;
fFontFlags = 0 ;
fFontAntiAliasRGB = false ;
fFontBlockedRGB = false ;
}
hsBool plDynSurfaceWriter : : plWinSurface : : WillFit ( uint16_t w , uint16_t h )
{
if ( fWidth > = w & & fHeight > = h )
return true ;
return false ;
}
static int SafeStrCmp ( const char * str1 , const char * str2 )
{
if ( str1 = = nil & & str2 = = nil )
return - 1 ;
if ( str1 ! = nil & & str2 ! = nil )
return strcmp ( str1 , str2 ) ;
return - 1 ;
}
hsBool plDynSurfaceWriter : : plWinSurface : : FontMatches ( const char * face , uint16_t size , uint8_t flags , hsBool aaRGB )
{
if ( SafeStrCmp ( face , fFontFace ) = = 0 & & fFontSize = = size & &
fFontFlags = = flags & & fFontAntiAliasRGB = = aaRGB )
return true ;
return false ;
}
void plDynSurfaceWriter : : plWinSurface : : SetFont ( const char * face , uint16_t size , uint8_t flags , hsBool aaRGB )
{
delete [ ] fFontFace ;
fFontFace = ( face ! = nil ) ? hsStrcpy ( face ) : nil ;
fFontSize = size ;
fFontFlags = flags ;
fFontAntiAliasRGB = aaRGB ;
bool hadAFont = false ;
if ( fFont ! = nil )
{
hadAFont = true ;
plWinFontCache : : GetInstance ( ) . FreeFont ( fFont ) ;
fFont = nil ;
}
if ( face = = nil )
return ;
bool bold = ( fFontFlags & plDynSurfaceWriter : : kFontBold ) ? true : false ;
bool italic = ( fFontFlags & plDynSurfaceWriter : : kFontItalic ) ? true : false ;
int nHeight = - MulDiv ( size , GetDeviceCaps ( fDC , LOGPIXELSY ) , 72 ) ;
fFont = plWinFontCache : : GetInstance ( ) . GetMeAFont ( face , nHeight , bold ? FW_BOLD : FW_NORMAL , italic ,
fFontAntiAliasRGB ? ANTIALIASED_QUALITY : DEFAULT_QUALITY ) ;
if ( fFont = = nil & & fFontAntiAliasRGB )
{
static bool warnedCantAntiAlias = false ;
// Creation of font failed; could be that we can't do anti-aliasing? Try not doing it...
if ( ! warnedCantAntiAlias )
{
plStatusLog : : AddLineS ( " pipeline.log " , " WARNING: Cannot allocate anti-aliased font. Falling back to non-anti-aliased. This will be the only warning " ) ;
warnedCantAntiAlias = true ;
}
fFont = plWinFontCache : : GetInstance ( ) . GetMeAFont ( face , nHeight , bold ? FW_BOLD : FW_NORMAL , italic ,
fFontAntiAliasRGB ? ANTIALIASED_QUALITY : DEFAULT_QUALITY ) ;
}
if ( fFont = = nil )
{
hsAssert ( false , " Cannot create Windows font for plDynSurfaceWriter " ) ;
plStatusLog : : AddLineS ( " pipeline.log " , " ERROR: Cannot allocate font for RGB surface! (face: %s, size: %d %s %s) " , face , nHeight , bold ? " bold " : " " , italic ? " italic " : " " ) ;
delete [ ] fFontFace ;
fFontFace = nil ;
fFontSize = 0 ;
return ;
}
if ( SelectObject ( fDC , fFont ) = = nil & & hadAFont )
{
char msg [ 256 ] ;
FormatMessage ( FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS , nil , GetLastError ( ) , MAKELANGID ( LANG_NEUTRAL , SUBLANG_DEFAULT ) , msg , sizeof ( msg ) , nil ) ;
char * ret = strrchr ( msg , ' \n ' ) ;
if ( ret ! = nil )
* ret = 0 ;
plStatusLog : : AddLineS ( " pipeline.log " , 0xffff0000 , " SelectObject() FAILED (%s) " , msg ) ;
}
}
# endif // BUILD_FOR_WIN32
//// StupidStatic /////////////////////////////////////////////////////////////
hsBool plDynSurfaceWriter : : fForceSharedSurfaces = false ;
hsBool plDynSurfaceWriter : : fOSDetected = false ;
hsBool plDynSurfaceWriter : : fOSCanShareSurfaces = false ;
hsBool plDynSurfaceWriter : : CanHandleLotsOfThem ( void )
{
if ( fOSDetected )
return fOSCanShareSurfaces ;
fOSDetected = true ;
# if HS_BUILD_FOR_WIN32
OSVERSIONINFO versionInfo ;
memset ( & versionInfo , 0 , sizeof ( versionInfo ) ) ;
versionInfo . dwOSVersionInfoSize = sizeof ( versionInfo ) ;
if ( GetVersionEx ( & versionInfo ) )
{
plStatusLog : : AddLineS ( " pipeline.log " , " OS version detection results: " ) ;
plStatusLog : : AddLineS ( " pipeline.log " , " Version: %d.%d " , versionInfo . dwMajorVersion , versionInfo . dwMinorVersion ) ;
plStatusLog : : AddLineS ( " pipeline.log " , " Build #: %d " , versionInfo . dwBuildNumber ) ;
plStatusLog : : AddLineS ( " pipeline.log " , " Platform ID: %d " , versionInfo . dwPlatformId ) ;
if ( versionInfo . dwPlatformId = = VER_PLATFORM_WIN32_NT )
{
if ( fForceSharedSurfaces )
{
plStatusLog : : AddLineS ( " pipeline.log " , " Detected NT-based platform, but sharing surfaces due to override " ) ;
fOSCanShareSurfaces = false ;
}
else
{
plStatusLog : : AddLineS ( " pipeline.log " , " Detected NT-based platform, allowing separate surfaces " ) ;
fOSCanShareSurfaces = true ;
}
}
else
{
plStatusLog : : AddLineS ( " pipeline.log " , " Detected non-NT-based platform: sharing surfaces " ) ;
fOSCanShareSurfaces = false ;
}
}
else
{
plStatusLog : : AddLineS ( " pipeline.log " , " OS version detection failed " ) ;
fOSCanShareSurfaces = false ;
}
# endif
return fOSCanShareSurfaces ;
}
//// Constructor & Destructor /////////////////////////////////////////////////
plDynSurfaceWriter : : plDynSurfaceWriter ( )
{
IInit ( ) ;
}
plDynSurfaceWriter : : ~ plDynSurfaceWriter ( )
{
Reset ( ) ;
}
plDynSurfaceWriter : : plDynSurfaceWriter ( plDynamicTextMap * target , uint32_t flags )
{
IInit ( ) ;
fFlags = flags ;
SwitchTarget ( target ) ;
}
void plDynSurfaceWriter : : IInit ( void )
{
fCurrTarget = 0 ;
fJustify = kLeftJustify ;
fFlags = 0 ;
fFlushed = true ;
fFontFace = nil ;
fFontSize = 0 ;
fFontBlockedRGB = false ;
}
//// Reset ////////////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : Reset ( void )
{
# if HS_BUILD_FOR_WIN32
fRGBSurface . Release ( ) ;
fAlphaSurface . Release ( ) ;
# endif
fCurrTarget = nil ;
fFlushed = true ;
delete [ ] fFontFace ;
fFontFace = nil ;
fFontSize = 0 ;
}
///////////////////////////////////////////////////////////////////////////////
//// Target Switching /////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//// FlushToTarget ////////////////////////////////////////////////////////////
// Flushes all ops to the target.
void plDynSurfaceWriter : : FlushToTarget ( void )
{
int x , y ;
if ( fCurrTarget ! = nil & & ! fFlushed )
{
# if HS_BUILD_FOR_WIN32
// Flush the GDI so we can grab the bits
GdiFlush ( ) ;
uint32_t * destBits = ( uint32_t * ) fCurrTarget - > GetImage ( ) ;
// Are we merging in the alpha bits?
if ( fFlags & kSupportAlpha )
{
// Yup, munge 'em
uint32_t * srcRGBBits = fRGBSurface . GetBits ( ) ;
uint8_t * srcAlphaBits = fAlphaSurface . GetBits ( ) ;
uint32_t destWidth = fCurrTarget - > GetWidth ( ) ;
for ( y = 0 ; y < fCurrTarget - > GetHeight ( ) ; y + + )
{
for ( x = 0 ; x < destWidth ; x + + )
destBits [ x ] = ( srcRGBBits [ x ] & 0x00ffffff ) | ( ( uint32_t ) srcAlphaBits [ x ] < < 24 ) ;
destBits + = destWidth ;
srcRGBBits + = fRGBSurface . fWidth ;
srcAlphaBits + = fAlphaSurface . fWidth ;
}
}
else
{
// Nope, just a 24-bit copy and set alphas to ff
uint32_t * srcBits = fRGBSurface . GetBits ( ) ;
uint32_t destWidth = fCurrTarget - > GetWidth ( ) ;
for ( y = 0 ; y < fCurrTarget - > GetHeight ( ) ; y + + )
{
memcpy ( destBits , srcBits , destWidth * sizeof ( uint32_t ) ) ;
// Fill in 0xff
for ( x = 0 ; x < destWidth ; x + + )
destBits [ x ] | = 0xff000000 ;
destBits + = destWidth ;
srcBits + = fRGBSurface . fWidth ;
}
}
# endif
}
fFlushed = true ;
}
//// SwitchTarget /////////////////////////////////////////////////////////////
// Switches targets. Will flush to old target before switching. Also, if
// kDiscard isn't specified, will copy contents of new target to working
// surface.
void plDynSurfaceWriter : : SwitchTarget ( plDynamicTextMap * target )
{
if ( target = = fCurrTarget )
return ;
if ( ! fFlushed )
FlushToTarget ( ) ;
fCurrTarget = target ;
fFlushed = true ; // Will force a copy next IEnsureSurfaceUpdated()
// Make sure our surfaces fit
bool hadToAllocate = false ;
# if HS_BUILD_FOR_WIN32
if ( target ! = nil )
{
if ( ! fRGBSurface . WillFit ( ( uint16_t ) ( target - > GetWidth ( ) ) , ( uint16_t ) ( target - > GetHeight ( ) ) ) )
{
fRGBSurface . Allocate ( ( uint16_t ) ( target - > GetWidth ( ) ) , ( uint16_t ) ( target - > GetHeight ( ) ) ) ;
hadToAllocate = true ;
}
if ( fFlags & kSupportAlpha )
{
if ( ! fAlphaSurface . WillFit ( ( uint16_t ) ( target - > GetWidth ( ) ) , ( uint16_t ) ( target - > GetHeight ( ) ) ) )
{
fAlphaSurface . Allocate ( ( uint16_t ) ( target - > GetWidth ( ) ) , ( uint16_t ) ( target - > GetHeight ( ) ) ) ;
hadToAllocate = true ;
}
}
}
else
{
fRGBSurface . Release ( ) ;
fAlphaSurface . Release ( ) ;
hadToAllocate = true ;
}
# endif
if ( hadToAllocate )
{
delete [ ] fFontFace ;
fFontFace = nil ;
fFontSize = 0 ;
fFontFlags = 0 ;
}
}
//// IEnsureSurfaceUpdated ////////////////////////////////////////////////////
// Makes sure our surfaces are ready to write to.
void plDynSurfaceWriter : : IEnsureSurfaceUpdated ( void )
{
uint32_t x , y ;
// If we're flushed, then we haven't drawn since the last flush,
// which means we want to copy from our target before we start drawing.
// If we've already drawn, then we won't be flushed and we don't want
// to be copying over what we've already drawn
if ( fCurrTarget ! = nil & & fFlushed )
{
uint32_t * srcBits = ( uint32_t * ) fCurrTarget - > GetImage ( ) ;
# if HS_BUILD_FOR_WIN32
// Are we merging in the alpha bits?
if ( fFlags & kSupportAlpha )
{
// Yup, de-munge 'em
uint32_t * destRGBBits = fRGBSurface . GetBits ( ) ;
uint8_t * destAlphaBits = fAlphaSurface . GetBits ( ) ;
uint32_t srcWidth = fCurrTarget - > GetWidth ( ) ;
for ( y = 0 ; y < fCurrTarget - > GetHeight ( ) ; y + + )
{
for ( x = 0 ; x < srcWidth ; x + + )
{
destRGBBits [ x ] = srcBits [ x ] ; // Windows GDI probably doesn't care about the alpha bits here. Hopefully...
destAlphaBits [ x ] = ( uint8_t ) ( srcBits [ x ] > > 24 ) ;
}
srcBits + = srcWidth ;
destRGBBits + = fRGBSurface . fWidth ;
destAlphaBits + = fAlphaSurface . fWidth ;
}
}
else
{
// Nope, just do a straight memcopy
uint32_t * destBits = fRGBSurface . GetBits ( ) ;
uint32_t srcWidth = fCurrTarget - > GetWidth ( ) ;
for ( y = 0 ; y < fCurrTarget - > GetHeight ( ) ; y + + )
{
memcpy ( destBits , srcBits , srcWidth * sizeof ( uint32_t ) ) ;
srcBits + = srcWidth ;
destBits + = fRGBSurface . fWidth ;
}
}
# endif
// ALSO, we need to re-update our settings, since different targets
// can have different fonts or justifications
ISetFont ( fCurrTarget - > GetFontFace ( ) , fCurrTarget - > GetFontSize ( ) , 0 /*fCurrTarget->GetWriterFontFlags()*/ , fCurrTarget - > GetFontAARGB ( ) ) ;
SetJustify ( ( Justify ) fCurrTarget - > GetFontJustify ( ) ) ;
hsColorRGBA col = fCurrTarget - > GetFontColor ( ) ;
ISetTextColor ( col , fCurrTarget - > GetFontBlockRGB ( ) ) ;
fFlushed = false ;
}
}
hsBool plDynSurfaceWriter : : IsValid ( void ) const
{
if ( fCurrTarget = = nil )
return false ;
# if HS_BUILD_FOR_WIN32
if ( fRGBSurface . fDC = = nil | | fRGBSurface . fBitmap = = nil )
return false ;
if ( ( fFlags & kSupportAlpha ) & & ( fAlphaSurface . fDC = = nil | | fAlphaSurface . fBitmap = = nil ) )
return false ;
# endif
return true ;
}
///////////////////////////////////////////////////////////////////////////////
//// Rendering Functions //////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//// SetBitsFromBuffer ////////////////////////////////////////////////////////
/*
void plDynSurfaceWriter : : SetBitsFromBuffer ( uint32_t * clearBuffer , uint16_t width , uint16_t height )
{
int x , y ;
uint32_t * data = ( uint32_t * ) fImage , * srcData = clearBuffer ;
uint8_t * destAlpha = nil ;
if ( ! IsValid ( ) )
return ;
# if HS_BUILD_FOR_WIN32
GdiFlush ( ) ;
# endif
// Clear *all* to zero
memset ( data , 0 , fWidth * fHeight * sizeof ( uint32_t ) ) ;
if ( fHasAlpha )
{
# if HS_BUILD_FOR_WIN32
memset ( fWinAlphaBits , 0 , fWidth * fHeight ) ;
destAlpha = fWinAlphaBits ;
# endif
}
// Buffer is of size fVisWidth x fVisHeight, so we need a bit of work to do this right
for ( y = 0 ; y < fVisHeight ; y + + )
{
for ( x = 0 ; x < fVisWidth ; x + + )
{
data [ x ] = srcData [ x ] ;
}
if ( destAlpha ! = nil )
{
for ( x = 0 ; x < fVisWidth ; x + + )
destAlpha [ x ] = srcData [ x ] > > 24 ;
destAlpha + = fWidth ;
}
data + = fWidth ;
srcData + = fVisWidth ;
}
}
*/
//// ClearToColor /////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : ClearToColor ( hsColorRGBA & color )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
uint32_t i , hexColor = color . ToARGB32 ( ) ;
// Flush the GDI first, so it doesn't decide to overwrite us later
GdiFlush ( ) ;
uint32_t * rgbBits = fRGBSurface . GetBits ( ) ;
for ( i = 0 ; i < fRGBSurface . fWidth * fRGBSurface . fHeight ; i + + )
rgbBits [ i ] = hexColor ;
if ( fFlags & kSupportAlpha )
{
uint8_t * alphaBits = fAlphaSurface . GetBits ( ) , alpha = ( uint8_t ) ( hexColor > > 24 ) ;
for ( i = 0 ; i < fAlphaSurface . fWidth * fAlphaSurface . fHeight ; i + + )
alphaBits [ i ] = alpha ;
}
# endif
}
//// SetFont //////////////////////////////////////////////////////////////////
// OS-specific. Load the given font for drawing the text with.
void plDynSurfaceWriter : : SetFont ( const char * face , uint16_t size , uint8_t fontFlags , hsBool antiAliasRGB )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
ISetFont ( face , size , fontFlags , antiAliasRGB ) ;
}
//// ISetFont /////////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : ISetFont ( const char * face , uint16_t size , uint8_t fontFlags , hsBool antiAliasRGB )
{
fFlags = ( fFlags & ~ kFontShadowed ) | ( fontFlags & kFontShadowed ) ;
# if HS_BUILD_FOR_WIN32
if ( ! fRGBSurface . FontMatches ( face , size , fontFlags , antiAliasRGB ) )
fRGBSurface . SetFont ( face , size , fontFlags , antiAliasRGB ) ;
if ( fFlags & kSupportAlpha )
{
if ( ! fAlphaSurface . FontMatches ( face , size , fontFlags , ! antiAliasRGB ) )
fAlphaSurface . SetFont ( face , size , fontFlags , ! antiAliasRGB ) ;
}
# endif
}
//// SetTextColor /////////////////////////////////////////////////////////////
// blockRGB basically forces the RGB channel to write in blocks instead of
// actual characters. This isn't useful unless you're relying on the alpha
// channel to do the text (opaque text and transparent background), in which
// case you want plenty of block color in your RGB channel because it'll get
// alpha-ed out by the alpha channel.
void plDynSurfaceWriter : : SetTextColor ( hsColorRGBA & color , hsBool blockRGB )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
ISetTextColor ( color , blockRGB ) ;
}
//// IRefreshTextColor ////////////////////////////////////////////////////////
void plDynSurfaceWriter : : ISetTextColor ( hsColorRGBA & color , hsBool blockRGB )
{
# if HS_BUILD_FOR_WIN32
int r = ( int ) ( color . r * 255.f ) ;
int g = ( int ) ( color . g * 255.f ) ;
int b = ( int ) ( color . b * 255.f ) ;
fRGBSurface . fTextColor = RGB ( r , g , b ) ;
if ( fFlags & kSupportAlpha )
{
int a = ( int ) ( color . a * 255.f ) ;
fAlphaSurface . fTextColor = RGB ( a , a , a ) ;
}
fFontBlockedRGB = blockRGB ;
if ( fFontBlockedRGB & & ! ( fFlags & kFontShadowed ) )
{
: : SetBkColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
: : SetBkMode ( fRGBSurface . fDC , OPAQUE ) ;
}
else
: : SetBkMode ( fRGBSurface . fDC , TRANSPARENT ) ;
: : SetTextColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
if ( fFlags & kSupportAlpha )
: : SetTextColor ( fAlphaSurface . fDC , fAlphaSurface . fTextColor ) ;
# endif
}
//// SetJustify ///////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : SetJustify ( Justify j )
{
fJustify = j ;
}
//// IRefreshOSJustify ////////////////////////////////////////////////////////
void plDynSurfaceWriter : : IRefreshOSJustify ( void )
{
if ( ! IsValid ( ) )
return ;
# if HS_BUILD_FOR_WIN32
uint32_t justMode ;
switch ( fJustify )
{
case kLeftJustify : justMode = TA_LEFT ; break ;
case kCenter : justMode = TA_CENTER ; break ;
case kRightJustify : justMode = TA_RIGHT ; break ;
}
: : SetTextAlign ( fRGBSurface . fDC , justMode ) ;
if ( fFlags & kSupportAlpha )
: : SetTextAlign ( fAlphaSurface . fDC , justMode ) ;
# endif
}
//// DrawString ///////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawString ( uint16_t x , uint16_t y , const char * text )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
IRefreshOSJustify ( ) ;
# if HS_BUILD_FOR_WIN32
if ( fFlags & kFontShadowed )
{
: : SetTextColor ( fRGBSurface . fDC , RGB ( 0 , 0 , 0 ) ) ;
: : TextOut ( fRGBSurface . fDC , x + 1 , y + 1 , text , strlen ( text ) ) ;
: : SetTextColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
: : TextOut ( fRGBSurface . fDC , x , y , text , strlen ( text ) ) ;
if ( fFlags & kSupportAlpha )
{
: : TextOut ( fAlphaSurface . fDC , x + 1 , y + 1 , text , strlen ( text ) ) ;
: : TextOut ( fAlphaSurface . fDC , x , y , text , strlen ( text ) ) ;
}
}
else
{
: : TextOut ( fRGBSurface . fDC , x , y , text , strlen ( text ) ) ;
if ( fFlags & kSupportAlpha )
: : TextOut ( fAlphaSurface . fDC , x , y , text , strlen ( text ) ) ;
}
# endif
}
//// DrawClippedString ////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawClippedString ( int16_t x , int16_t y , const char * text , uint16_t width , uint16_t height )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
IRefreshOSJustify ( ) ;
# if HS_BUILD_FOR_WIN32
RECT r ;
: : SetRect ( & r , x , y , x + width , y + height ) ;
if ( fJustify = = kRightJustify )
x + = width - 1 ;
else if ( fJustify = = kCenter )
x + = width > > 1 ;
if ( fFlags & kFontShadowed )
{
: : SetTextColor ( fRGBSurface . fDC , RGB ( 0 , 0 , 0 ) ) ;
: : OffsetRect ( & r , 1 , 1 ) ;
: : ExtTextOut ( fRGBSurface . fDC , x + 1 , y + 1 , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
if ( fFlags & kSupportAlpha )
: : ExtTextOut ( fAlphaSurface . fDC , x + 1 , y + 1 , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
: : OffsetRect ( & r , - 1 , - 1 ) ;
: : SetTextColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
}
: : ExtTextOut ( fRGBSurface . fDC , x , y , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
if ( fFlags & kSupportAlpha )
: : ExtTextOut ( fAlphaSurface . fDC , x , y , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
# endif
}
//// DrawClippedString ////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawClippedString ( int16_t x , int16_t y , const char * text , uint16_t clipX , uint16_t clipY , uint16_t width , uint16_t height )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
IRefreshOSJustify ( ) ;
# if HS_BUILD_FOR_WIN32
RECT r ;
: : SetRect ( & r , clipX , clipY , clipX + width , clipY + height ) ;
if ( fFlags & kFontShadowed )
{
: : SetTextColor ( fRGBSurface . fDC , RGB ( 0 , 0 , 0 ) ) ;
: : OffsetRect ( & r , 1 , 1 ) ;
: : ExtTextOut ( fRGBSurface . fDC , x + 1 , y + 1 , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
if ( fFlags & kSupportAlpha )
: : ExtTextOut ( fAlphaSurface . fDC , x + 1 , y + 1 , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
: : OffsetRect ( & r , - 1 , - 1 ) ;
: : SetTextColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
}
: : ExtTextOut ( fRGBSurface . fDC , x , y , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
if ( fFlags & kSupportAlpha )
: : ExtTextOut ( fAlphaSurface . fDC , x , y , ETO_CLIPPED , & r , text , strlen ( text ) , nil ) ;
# endif
}
//// DrawWrappedString ////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawWrappedString ( uint16_t x , uint16_t y , const char * text , uint16_t width , uint16_t height )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
RECT r ;
: : SetRect ( & r , x , y , x + width , y + height ) ;
UINT format = DT_TOP | DT_NOPREFIX | DT_WORDBREAK ;
switch ( fJustify )
{
case kLeftJustify : format | = DT_LEFT ; break ;
case kCenter : format | = DT_CENTER ; break ;
case kRightJustify : format | = DT_RIGHT ; break ;
}
if ( fFlags & kFontShadowed )
{
: : SetTextColor ( fRGBSurface . fDC , RGB ( 0 , 0 , 0 ) ) ;
: : OffsetRect ( & r , 1 , 1 ) ;
: : DrawText ( fRGBSurface . fDC , text , strlen ( text ) , & r , format ) ;
if ( fFlags & kSupportAlpha )
: : DrawText ( fAlphaSurface . fDC , text , strlen ( text ) , & r , format ) ;
: : OffsetRect ( & r , - 1 , - 1 ) ;
: : SetTextColor ( fRGBSurface . fDC , fRGBSurface . fTextColor ) ;
}
: : DrawText ( fRGBSurface . fDC , text , strlen ( text ) , & r , format ) ;
if ( fFlags & kSupportAlpha )
: : DrawText ( fAlphaSurface . fDC , text , strlen ( text ) , & r , format ) ;
# endif
}
//// CalcStringWidth //////////////////////////////////////////////////////////
uint16_t plDynSurfaceWriter : : CalcStringWidth ( const char * text , uint16_t * height )
{
if ( ! IsValid ( ) )
return 0 ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
SIZE size ;
: : GetTextExtentPoint32 ( fRGBSurface . fDC , text , strlen ( text ) , & size ) ;
if ( height ! = nil )
* height = ( uint16_t ) size . cy ;
return ( uint16_t ) size . cx ;
# endif
}
//// CalcWrappedStringSize ////////////////////////////////////////////////////
void plDynSurfaceWriter : : CalcWrappedStringSize ( const char * text , uint16_t * width , uint16_t * height )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
RECT r ;
: : SetRect ( & r , 0 , 0 , * width , 0 ) ;
UINT format = DT_TOP | DT_NOPREFIX | DT_WORDBREAK | DT_CALCRECT ;
switch ( fJustify )
{
case kLeftJustify : format | = DT_LEFT ; break ;
case kCenter : format | = DT_CENTER ; break ;
case kRightJustify : format | = DT_RIGHT ; break ;
}
: : DrawText ( fRGBSurface . fDC , text , strlen ( text ) , & r , format ) ;
* width = ( uint16_t ) ( r . right ) ;
if ( height ! = nil )
* height = ( uint16_t ) r . bottom ;
# endif
}
//// FillRect /////////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : FillRect ( uint16_t x , uint16_t y , uint16_t width , uint16_t height , hsColorRGBA & color )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
RECT rc ;
: : SetRect ( & rc , x , y , x + width , y + height ) ;
int r = ( int ) ( color . r * 255.f ) ;
int g = ( int ) ( color . g * 255.f ) ;
int b = ( int ) ( color . b * 255.f ) ;
int a = ( int ) ( color . a * 255.f ) ;
HBRUSH brush = : : CreateSolidBrush ( RGB ( r , g , b ) ) ;
: : FillRect ( fRGBSurface . fDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
if ( fFlags & kSupportAlpha )
{
brush = : : CreateSolidBrush ( RGB ( a , a , a ) ) ;
: : FillRect ( fAlphaSurface . fDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
}
# endif
}
//// FrameRect ////////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : FrameRect ( uint16_t x , uint16_t y , uint16_t width , uint16_t height , hsColorRGBA & color )
{
if ( ! IsValid ( ) )
return ;
IEnsureSurfaceUpdated ( ) ;
# if HS_BUILD_FOR_WIN32
RECT rc ;
: : SetRect ( & rc , x , y , x + width , y + height ) ;
int r = ( int ) ( color . r * 255.f ) ;
int g = ( int ) ( color . g * 255.f ) ;
int b = ( int ) ( color . b * 255.f ) ;
int a = ( int ) ( color . a * 255.f ) ;
HBRUSH brush = : : CreateSolidBrush ( RGB ( r , g , b ) ) ;
: : FrameRect ( fRGBSurface . fDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
if ( fFlags & kSupportAlpha )
{
brush = : : CreateSolidBrush ( RGB ( a , a , a ) ) ;
: : FrameRect ( fAlphaSurface . fDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
}
# endif
}
/*
//// DrawImage ////////////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawImage ( uint16_t x , uint16_t y , plMipmap * image , hsBool respectAlpha )
{
if ( ! IsValid ( ) )
return ;
# if HS_BUILD_FOR_WIN32
// Flush the GDI first, to make sure it's done
GdiFlush ( ) ;
# endif
plMipmap : : CompositeOptions opts ( respectAlpha ? 0 : plMipmap : : kForceOpaque ) ;
if ( ! fHasAlpha )
opts . fFlags = plMipmap : : kCopySrcAlpha ; // Don't care, this is fastest
Composite ( image , x , y , & opts ) ;
/// HACK for now, since the alpha in the mipmap gets copied straight into the
/// 32-bit color buffer, but our separate hacked alpha buffer hasn't been updated
if ( fHasAlpha & & ! respectAlpha )
{
HBRUSH brush = : : CreateSolidBrush ( RGB ( 255 , 255 , 255 ) ) ;
RECT rc ;
: : SetRect ( & rc , x , y , x + image - > GetWidth ( ) , y + image - > GetHeight ( ) ) ;
: : FillRect ( fWinAlphaDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
}
}
//// DrawClippedImage /////////////////////////////////////////////////////////
void plDynSurfaceWriter : : DrawClippedImage ( uint16_t x , uint16_t y , plMipmap * image ,
uint16_t srcClipX , uint16_t srcClipY ,
uint16_t srcClipWidth , uint16_t srcClipHeight ,
hsBool respectAlpha )
{
if ( ! IsValid ( ) )
return ;
# if HS_BUILD_FOR_WIN32
// Flush the GDI first, to make sure it's done
GdiFlush ( ) ;
# endif
plMipmap : : CompositeOptions opts ( respectAlpha ? 0 : plMipmap : : kForceOpaque ) ;
if ( ! fHasAlpha )
opts . fFlags = plMipmap : : kCopySrcAlpha ; // Don't care, this is fastest
opts . fSrcClipX = srcClipX ;
opts . fSrcClipY = srcClipY ;
opts . fSrcClipWidth = srcClipWidth ;
opts . fSrcClipHeight = srcClipHeight ;
Composite ( image , x , y , & opts ) ;
/// HACK for now, since the alpha in the mipmap gets copied straight into the
/// 32-bit color buffer, but our separate hacked alpha buffer hasn't been updated
if ( fHasAlpha & & ! respectAlpha )
{
HBRUSH brush = : : CreateSolidBrush ( RGB ( 255 , 255 , 255 ) ) ;
RECT rc ;
: : SetRect ( & rc , x , y , x + ( srcClipWidth > 0 ? srcClipWidth : image - > GetWidth ( ) ) ,
y + ( srcClipHeight > 0 ? srcClipHeight : image - > GetHeight ( ) ) ) ;
: : FillRect ( fWinAlphaDC , & rc , brush ) ;
: : DeleteObject ( brush ) ;
}
}
*/