2016-03-01 15:47:10 +00:00
/*
* * v_draw . cpp
* * Draw patches and blocks to a canvas
* *
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* * 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 .
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* *
*/
// #define NO_SWRENDER // set this if you want to exclude the software renderer. Without software renderer the base implementations of DrawTextureV and FillSimplePoly need to be disabled because they depend on it.
# include <stdio.h>
# include <stdarg.h>
# include "doomtype.h"
# include "v_video.h"
# include "m_swap.h"
# include "r_defs.h"
# include "r_utility.h"
# ifndef NO_SWRENDER
2016-12-27 05:31:55 +00:00
# include "swrenderer/drawers/r_draw.h"
# include "swrenderer/drawers/r_draw_rgba.h"
2017-01-12 15:21:46 +00:00
# include "swrenderer/scene/r_light.h"
# include "swrenderer/scene/r_viewport.h"
2016-03-01 15:47:10 +00:00
# endif
# include "r_data/r_translate.h"
# include "doomstat.h"
# include "v_palette.h"
# include "gi.h"
# include "g_level.h"
# include "st_stuff.h"
# include "sbar.h"
# include "i_system.h"
# include "i_video.h"
# include "templates.h"
# include "d_net.h"
# include "colormatcher.h"
# include "r_data/colormaps.h"
2017-01-08 17:45:30 +00:00
# include "g_levellocals.h"
2016-03-01 15:47:10 +00:00
2016-09-06 17:48:14 +00:00
CUSTOM_CVAR ( Int , uiscale , 2 , CVAR_ARCHIVE | CVAR_NOINITCALL )
{
if ( StatusBar ! = NULL )
{
StatusBar - > ScreenSizeChanged ( ) ;
}
}
2016-12-25 00:22:31 +00:00
EXTERN_CVAR ( Bool , r_blendmethod )
2016-09-06 17:48:14 +00:00
2016-03-01 15:47:10 +00:00
// [RH] Stretch values to make a 320x200 image best fit the screen
// without using fractional steppings
int CleanXfac , CleanYfac ;
// [RH] Effective screen sizes that the above scale values give you
int CleanWidth , CleanHeight ;
// Above minus 1 (or 1, if they are already 1)
int CleanXfac_1 , CleanYfac_1 , CleanWidth_1 , CleanHeight_1 ;
2016-09-06 17:48:14 +00:00
CVAR ( Bool , hud_scale , true , CVAR_ARCHIVE ) ;
2016-03-01 15:47:10 +00:00
// For routines that take RGB colors, cache the previous lookup in case there
// are several repetitions with the same color.
static int LastPal = - 1 ;
static uint32 LastRGB ;
static int PalFromRGB ( uint32 rgb )
{
if ( LastPal > = 0 & & LastRGB = = rgb )
{
return LastPal ;
}
// Quick check for black and white.
if ( rgb = = MAKEARGB ( 255 , 0 , 0 , 0 ) )
{
LastPal = GPalette . BlackIndex ;
}
else if ( rgb = = MAKEARGB ( 255 , 255 , 255 , 255 ) )
{
LastPal = GPalette . WhiteIndex ;
}
else
{
LastPal = ColorMatcher . Pick ( RPART ( rgb ) , GPART ( rgb ) , BPART ( rgb ) ) ;
}
LastRGB = rgb ;
return LastPal ;
}
2016-04-11 08:46:30 +00:00
void DCanvas : : DrawTexture ( FTexture * img , double x , double y , int tags_first , . . . )
2016-03-01 15:47:10 +00:00
{
va_list tags ;
va_start ( tags , tags_first ) ;
2016-04-09 10:55:12 +00:00
DrawParms parms ;
2016-04-09 18:47:54 +00:00
bool res = ParseDrawTextureTags ( img , x , y , tags_first , tags , & parms , false ) ;
va_end ( tags ) ;
if ( ! res )
2016-04-09 10:55:12 +00:00
{
return ;
}
2016-04-09 10:59:50 +00:00
DrawTextureParms ( img , parms ) ;
2016-03-01 15:47:10 +00:00
}
2017-01-15 22:21:38 +00:00
DEFINE_ACTION_FUNCTION ( _Screen , DrawHUDTexture )
{
PARAM_PROLOGUE ;
PARAM_INT ( texid ) ;
PARAM_FLOAT ( x ) ;
PARAM_FLOAT ( y ) ;
screen - > DrawTexture ( TexMan ( FSetTextureID ( texid ) ) , x , y , DTA_HUDRules , HUD_Normal , TAG_END ) ;
return 0 ;
}
2016-04-09 10:59:50 +00:00
void DCanvas : : DrawTextureParms ( FTexture * img , DrawParms & parms )
2016-03-01 15:47:10 +00:00
{
# ifndef NO_SWRENDER
2016-12-01 01:38:32 +00:00
using namespace swrenderer ;
2016-12-05 07:22:04 +00:00
using namespace drawerargs ;
2016-03-01 15:47:10 +00:00
static short bottomclipper [ MAXWIDTH ] , topclipper [ MAXWIDTH ] ;
2016-04-09 20:56:12 +00:00
const BYTE * translation = NULL ;
2016-03-01 15:47:10 +00:00
2016-06-01 06:54:39 +00:00
if ( r_swtruecolor ! = IsBgra ( ) )
{
r_swtruecolor = IsBgra ( ) ;
R_InitColumnDrawers ( ) ;
}
2016-03-01 15:47:10 +00:00
if ( APART ( parms . colorOverlay ) ! = 0 )
{
// The software renderer cannot invert the source without inverting the overlay
// too. That means if the source is inverted, we need to do the reverse of what
// the invert overlay flag says to do.
INTBOOL invertoverlay = ( parms . style . Flags & STYLEF_InvertOverlay ) ;
if ( parms . style . Flags & STYLEF_InvertSource )
{
invertoverlay = ! invertoverlay ;
}
if ( invertoverlay )
{
parms . colorOverlay = PalEntry ( parms . colorOverlay ) . InverseColor ( ) ;
}
// Note that this overrides DTA_Translation in software, but not in hardware.
2016-10-16 14:14:54 +00:00
if ( ! r_swtruecolor )
{
FDynamicColormap * colormap = GetSpecialLights ( MAKERGB ( 255 , 255 , 255 ) ,
parms . colorOverlay & MAKEARGB ( 0 , 255 , 255 , 255 ) , 0 ) ;
translation = & colormap - > Maps [ ( APART ( parms . colorOverlay ) * NUMCOLORMAPS / 255 ) * 256 ] ;
}
2016-04-09 20:56:12 +00:00
}
else if ( parms . remap ! = NULL )
{
2016-10-16 14:14:54 +00:00
if ( r_swtruecolor )
translation = ( const BYTE * ) parms . remap - > Palette ;
else
translation = parms . remap - > Remap ;
2016-03-01 15:47:10 +00:00
}
2016-04-09 20:56:12 +00:00
if ( translation ! = NULL )
2016-03-01 15:47:10 +00:00
{
2016-06-05 12:08:03 +00:00
R_SetTranslationMap ( ( lighttable_t * ) translation ) ;
2016-03-01 15:47:10 +00:00
}
else
{
2016-10-16 14:14:54 +00:00
if ( r_swtruecolor )
R_SetTranslationMap ( nullptr ) ;
else
R_SetTranslationMap ( identitymap ) ;
2016-03-01 15:47:10 +00:00
}
2016-06-05 12:08:03 +00:00
fixedcolormap = dc_fcolormap ;
2016-12-25 04:15:23 +00:00
bool visible ;
2017-01-12 20:29:19 +00:00
FDynamicColormap * basecolormap = nullptr ;
2016-10-16 14:14:54 +00:00
if ( r_swtruecolor )
2017-01-12 20:29:19 +00:00
visible = R_SetPatchStyle ( parms . style , parms . Alpha , - 1 , parms . fillcolor , basecolormap ) ;
2016-10-16 14:14:54 +00:00
else
2017-01-12 20:29:19 +00:00
visible = R_SetPatchStyle ( parms . style , parms . Alpha , 0 , parms . fillcolor , basecolormap ) ;
2016-03-01 15:47:10 +00:00
BYTE * destorgsave = dc_destorg ;
2016-10-17 10:36:01 +00:00
int destheightsave = dc_destheight ;
2016-03-01 15:47:10 +00:00
dc_destorg = screen - > GetBuffer ( ) ;
2016-10-17 10:36:01 +00:00
dc_destheight = screen - > GetHeight ( ) ;
2016-04-09 10:55:12 +00:00
if ( dc_destorg = = NULL )
{
I_FatalError ( " Attempt to write to buffer of hardware canvas " ) ;
}
2016-03-01 15:47:10 +00:00
double x0 = parms . x - parms . left * parms . destwidth / parms . texwidth ;
double y0 = parms . y - parms . top * parms . destheight / parms . texheight ;
2016-12-25 04:15:23 +00:00
if ( visible )
2016-03-01 15:47:10 +00:00
{
2016-04-14 17:40:14 +00:00
double centeryback = CenterY ;
CenterY = 0 ;
2016-03-01 15:47:10 +00:00
// There is not enough precision in the drawing routines to keep the full
// precision for y0. :(
2017-01-09 12:53:02 +00:00
double sprtopscreen ;
2016-04-24 02:29:40 +00:00
modf ( y0 , & sprtopscreen ) ;
2016-03-01 15:47:10 +00:00
double yscale = parms . destheight / img - > GetHeight ( ) ;
double iyscale = 1 / yscale ;
2017-01-09 12:53:02 +00:00
double spryscale = yscale ;
2016-04-16 01:30:05 +00:00
assert ( spryscale > 0 ) ;
2016-03-01 15:47:10 +00:00
2017-01-09 12:53:02 +00:00
bool sprflipvert = false ;
fixed_t iscale = FLOAT2FIXED ( 1 / spryscale ) ;
//dc_texturemid = (CenterY - 1 - sprtopscreen) * iscale / 65536;
2016-03-01 15:47:10 +00:00
fixed_t frac = 0 ;
double xiscale = img - > GetWidth ( ) / parms . destwidth ;
double x2 = x0 + parms . destwidth ;
2017-01-09 12:53:02 +00:00
short * mfloorclip ;
short * mceilingclip ;
2016-03-01 15:47:10 +00:00
if ( bottomclipper [ 0 ] ! = parms . dclip )
{
2016-12-09 12:04:53 +00:00
fillshort ( bottomclipper , screen - > GetWidth ( ) , ( short ) parms . dclip ) ;
2016-03-01 15:47:10 +00:00
}
if ( parms . uclip ! = 0 )
{
if ( topclipper [ 0 ] ! = parms . uclip )
{
2016-12-09 12:04:53 +00:00
fillshort ( topclipper , screen - > GetWidth ( ) , ( short ) parms . uclip ) ;
2016-03-01 15:47:10 +00:00
}
mceilingclip = topclipper ;
}
else
{
mceilingclip = zeroarray ;
}
mfloorclip = bottomclipper ;
if ( parms . flipX )
{
frac = ( img - > GetWidth ( ) < < FRACBITS ) - 1 ;
xiscale = - xiscale ;
}
if ( parms . windowleft > 0 | | parms . windowright < parms . texwidth )
{
2016-04-09 18:47:54 +00:00
double wi = MIN ( parms . windowright , parms . texwidth ) ;
2016-03-01 15:47:10 +00:00
double xscale = parms . destwidth / parms . texwidth ;
x0 + = parms . windowleft * xscale ;
frac + = FLOAT2FIXED ( parms . windowleft ) ;
2016-04-09 18:47:54 +00:00
x2 - = ( parms . texwidth - wi ) * xscale ;
2016-03-01 15:47:10 +00:00
}
if ( x0 < parms . lclip )
{
frac + = FLOAT2FIXED ( ( parms . lclip - x0 ) * xiscale ) ;
x0 = parms . lclip ;
}
if ( x2 > parms . rclip )
{
x2 = parms . rclip ;
}
2017-01-09 12:53:02 +00:00
int x = int ( x0 ) ;
2016-03-01 15:47:10 +00:00
int x2_i = int ( x2 ) ;
fixed_t xiscale_i = FLOAT2FIXED ( xiscale ) ;
2017-01-09 12:53:02 +00:00
while ( x < x2_i )
2016-03-01 15:47:10 +00:00
{
2017-01-09 12:53:02 +00:00
R_DrawMaskedColumn ( x , iscale , img , frac , spryscale , sprtopscreen , sprflipvert , mfloorclip , mceilingclip , ! parms . masked ) ;
x + + ;
2016-12-25 04:05:53 +00:00
frac + = xiscale_i ;
2016-03-01 15:47:10 +00:00
}
2016-04-14 17:40:14 +00:00
CenterY = centeryback ;
2016-03-01 15:47:10 +00:00
}
dc_destorg = destorgsave ;
2016-10-17 10:36:01 +00:00
dc_destheight = destheightsave ;
2016-03-01 15:47:10 +00:00
if ( ticdup ! = 0 & & menuactive = = MENU_Off )
{
NetUpdate ( ) ;
}
# endif
}
2016-04-09 18:47:54 +00:00
bool DCanvas : : SetTextureParms ( DrawParms * parms , FTexture * img , double xx , double yy ) const
{
if ( img ! = NULL )
{
parms - > x = xx ;
parms - > y = yy ;
parms - > texwidth = img - > GetScaledWidthDouble ( ) ;
parms - > texheight = img - > GetScaledHeightDouble ( ) ;
if ( parms - > top = = INT_MAX | | parms - > fortext )
{
parms - > top = img - > GetScaledTopOffset ( ) ;
}
if ( parms - > left = = INT_MAX | | parms - > fortext )
{
parms - > left = img - > GetScaledLeftOffset ( ) ;
}
if ( parms - > destwidth = = INT_MAX | | parms - > fortext )
{
parms - > destwidth = img - > GetScaledWidthDouble ( ) ;
}
if ( parms - > destheight = = INT_MAX | | parms - > fortext )
{
parms - > destheight = img - > GetScaledHeightDouble ( ) ;
}
switch ( parms - > cleanmode )
{
default :
break ;
case DTA_Clean :
parms - > x = ( parms - > x - 160.0 ) * CleanXfac + ( Width * 0.5 ) ;
parms - > y = ( parms - > y - 100.0 ) * CleanYfac + ( Height * 0.5 ) ;
parms - > destwidth = parms - > texwidth * CleanXfac ;
parms - > destheight = parms - > texheight * CleanYfac ;
break ;
case DTA_CleanNoMove :
parms - > destwidth = parms - > texwidth * CleanXfac ;
parms - > destheight = parms - > texheight * CleanYfac ;
break ;
case DTA_CleanNoMove_1 :
parms - > destwidth = parms - > texwidth * CleanXfac_1 ;
parms - > destheight = parms - > texheight * CleanYfac_1 ;
break ;
case DTA_Fullscreen :
parms - > x = parms - > y = 0 ;
break ;
case DTA_HUDRules :
case DTA_HUDRulesC :
{
bool xright = parms - > x < 0 ;
bool ybot = parms - > y < 0 ;
if ( hud_scale )
{
parms - > x * = CleanXfac ;
if ( parms - > cleanmode = = DTA_HUDRulesC )
parms - > x + = Width * 0.5 ;
else if ( xright )
parms - > x = Width + parms - > x ;
parms - > y * = CleanYfac ;
if ( ybot )
parms - > y = Height + parms - > y ;
parms - > destwidth = parms - > texwidth * CleanXfac ;
parms - > destheight = parms - > texheight * CleanYfac ;
}
else
{
if ( parms - > cleanmode = = DTA_HUDRulesC )
parms - > x + = Width * 0.5 ;
else if ( xright )
parms - > x = Width + parms - > x ;
if ( ybot )
parms - > y = Height + parms - > y ;
}
break ;
}
}
if ( parms - > virtWidth ! = Width | | parms - > virtHeight ! = Height )
{
VirtualToRealCoords ( parms - > x , parms - > y , parms - > destwidth , parms - > destheight ,
parms - > virtWidth , parms - > virtHeight , parms - > virtBottom , ! parms - > keepratio ) ;
}
}
2016-04-09 20:56:12 +00:00
2016-04-09 18:47:54 +00:00
return false ;
}
2016-04-09 10:55:12 +00:00
bool DCanvas : : ParseDrawTextureTags ( FTexture * img , double x , double y , DWORD tag , va_list tags , DrawParms * parms , bool fortext ) const
2016-03-01 15:47:10 +00:00
{
INTBOOL boolval ;
int intval ;
bool translationset = false ;
bool fillcolorset = false ;
2016-04-09 18:47:54 +00:00
if ( ! fortext )
2016-03-01 15:47:10 +00:00
{
2016-04-09 18:47:54 +00:00
if ( img = = NULL | | img - > UseType = = FTexture : : TEX_Null )
{
va_end ( tags ) ;
return false ;
}
2016-03-01 15:47:10 +00:00
}
// Do some sanity checks on the coordinates.
if ( x < - 16383 | | x > 16383 | | y < - 16383 | | y > 16383 )
{
va_end ( tags ) ;
return false ;
}
2016-04-09 18:47:54 +00:00
parms - > fortext = fortext ;
2016-03-01 15:47:10 +00:00
parms - > windowleft = 0 ;
2016-04-09 18:47:54 +00:00
parms - > windowright = INT_MAX ;
2016-03-01 15:47:10 +00:00
parms - > dclip = this - > GetHeight ( ) ;
parms - > uclip = 0 ;
parms - > lclip = 0 ;
parms - > rclip = this - > GetWidth ( ) ;
2016-04-09 18:47:54 +00:00
parms - > left = INT_MAX ;
parms - > top = INT_MAX ;
parms - > destwidth = INT_MAX ;
parms - > destheight = INT_MAX ;
2016-03-22 12:35:16 +00:00
parms - > Alpha = 1.f ;
2016-03-01 15:47:10 +00:00
parms - > fillcolor = - 1 ;
parms - > remap = NULL ;
parms - > colorOverlay = 0 ;
parms - > alphaChannel = false ;
parms - > flipX = false ;
parms - > shadowAlpha = 0 ;
parms - > shadowColor = 0 ;
parms - > virtWidth = this - > GetWidth ( ) ;
parms - > virtHeight = this - > GetHeight ( ) ;
parms - > keepratio = false ;
parms - > style . BlendOp = 255 ; // Dummy "not set" value
parms - > masked = true ;
parms - > bilinear = false ;
parms - > specialcolormap = NULL ;
parms - > colormapstyle = NULL ;
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
parms - > scalex = parms - > scaley = 1 ;
parms - > cellx = parms - > celly = 0 ;
parms - > maxstrlen = INT_MAX ;
parms - > virtBottom = false ;
2016-03-01 15:47:10 +00:00
// Parse the tag list for attributes. (For floating point attributes,
// consider that the C ABI dictates that all floats be promoted to
// doubles when passed as function arguments.)
while ( tag ! = TAG_DONE )
{
DWORD data ;
switch ( tag )
{
default :
data = va_arg ( tags , DWORD ) ;
break ;
case DTA_DestWidth :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > destwidth = va_arg ( tags , int ) ;
break ;
case DTA_DestWidthF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > destwidth = va_arg ( tags , double ) ;
break ;
case DTA_DestHeight :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > destheight = va_arg ( tags , int ) ;
break ;
case DTA_DestHeightF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > destheight = va_arg ( tags , double ) ;
break ;
case DTA_Clean :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
parms - > scalex = 1 ;
parms - > scaley = 1 ;
parms - > cleanmode = tag ;
2016-03-01 15:47:10 +00:00
}
break ;
case DTA_CleanNoMove :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
parms - > scalex = CleanXfac ;
parms - > scaley = CleanYfac ;
parms - > cleanmode = tag ;
2016-03-01 15:47:10 +00:00
}
break ;
case DTA_CleanNoMove_1 :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
parms - > scalex = CleanXfac_1 ;
parms - > scaley = CleanYfac_1 ;
parms - > cleanmode = tag ;
2016-03-01 15:47:10 +00:00
}
break ;
case DTA_320x200 :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
parms - > scalex = 1 ;
parms - > scaley = 1 ;
2016-03-01 15:47:10 +00:00
parms - > virtWidth = 320 ;
parms - > virtHeight = 200 ;
}
break ;
case DTA_Bottom320x200 :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
parms - > scalex = 1 ;
parms - > scaley = 1 ;
2016-03-01 15:47:10 +00:00
parms - > virtWidth = 320 ;
parms - > virtHeight = 200 ;
}
2016-04-09 18:47:54 +00:00
parms - > virtBottom = true ;
2016-03-01 15:47:10 +00:00
break ;
case DTA_HUDRules :
2016-04-09 18:47:54 +00:00
intval = va_arg ( tags , int ) ;
parms - > cleanmode = intval = = HUD_HorizCenter ? DTA_HUDRulesC : DTA_HUDRules ;
2016-03-01 15:47:10 +00:00
break ;
case DTA_VirtualWidth :
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > virtWidth = va_arg ( tags , int ) ;
break ;
case DTA_VirtualWidthF :
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > virtWidth = va_arg ( tags , double ) ;
break ;
case DTA_VirtualHeight :
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > virtHeight = va_arg ( tags , int ) ;
break ;
case DTA_VirtualHeightF :
2016-04-09 18:47:54 +00:00
parms - > cleanmode = DTA_Base ;
2016-03-01 15:47:10 +00:00
parms - > virtHeight = va_arg ( tags , double ) ;
break ;
case DTA_Fullscreen :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( img = = NULL ) return false ;
parms - > cleanmode = DTA_Fullscreen ;
2016-03-01 15:47:10 +00:00
parms - > virtWidth = img - > GetScaledWidthDouble ( ) ;
parms - > virtHeight = img - > GetScaledHeightDouble ( ) ;
}
break ;
case DTA_Alpha :
2016-03-22 12:35:16 +00:00
parms - > Alpha = FIXED2FLOAT ( MIN < fixed_t > ( OPAQUE , va_arg ( tags , fixed_t ) ) ) ;
2016-03-21 11:18:46 +00:00
break ;
case DTA_AlphaF :
2016-03-22 12:35:16 +00:00
parms - > Alpha = ( float ) ( MIN < double > ( 1. , va_arg ( tags , double ) ) ) ;
2016-03-01 15:47:10 +00:00
break ;
case DTA_AlphaChannel :
parms - > alphaChannel = va_arg ( tags , INTBOOL ) ;
break ;
case DTA_FillColor :
parms - > fillcolor = va_arg ( tags , uint32 ) ;
fillcolorset = true ;
break ;
case DTA_Translation :
parms - > remap = va_arg ( tags , FRemapTable * ) ;
if ( parms - > remap ! = NULL & & parms - > remap - > Inactive )
{ // If it's inactive, pretend we were passed NULL instead.
parms - > remap = NULL ;
}
break ;
case DTA_ColorOverlay :
parms - > colorOverlay = va_arg ( tags , DWORD ) ;
break ;
case DTA_FlipX :
parms - > flipX = va_arg ( tags , INTBOOL ) ;
break ;
case DTA_TopOffset :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > top = va_arg ( tags , int ) ;
break ;
case DTA_TopOffsetF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > top = va_arg ( tags , double ) ;
break ;
case DTA_LeftOffset :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > left = va_arg ( tags , int ) ;
break ;
case DTA_LeftOffsetF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > left = va_arg ( tags , double ) ;
break ;
case DTA_CenterOffset :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
if ( va_arg ( tags , int ) )
{
2016-04-09 18:47:54 +00:00
parms - > left = img - > GetScaledWidthDouble ( ) * 0.5 ;
parms - > top = img - > GetScaledHeightDouble ( ) * 0.5 ;
2016-03-01 15:47:10 +00:00
}
break ;
case DTA_CenterBottomOffset :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
if ( va_arg ( tags , int ) )
{
2016-04-09 18:47:54 +00:00
parms - > left = img - > GetScaledWidthDouble ( ) * 0.5 ;
parms - > top = img - > GetScaledHeightDouble ( ) ;
2016-03-01 15:47:10 +00:00
}
break ;
case DTA_WindowLeft :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > windowleft = va_arg ( tags , int ) ;
break ;
case DTA_WindowLeftF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > windowleft = va_arg ( tags , double ) ;
break ;
case DTA_WindowRight :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > windowright = va_arg ( tags , int ) ;
break ;
case DTA_WindowRightF :
2016-04-09 18:47:54 +00:00
assert ( fortext = = false ) ;
if ( fortext ) return false ;
2016-03-01 15:47:10 +00:00
parms - > windowright = va_arg ( tags , double ) ;
break ;
case DTA_ClipTop :
parms - > uclip = va_arg ( tags , int ) ;
if ( parms - > uclip < 0 )
{
parms - > uclip = 0 ;
}
break ;
case DTA_ClipBottom :
parms - > dclip = va_arg ( tags , int ) ;
if ( parms - > dclip > this - > GetHeight ( ) )
{
parms - > dclip = this - > GetHeight ( ) ;
}
break ;
case DTA_ClipLeft :
parms - > lclip = va_arg ( tags , int ) ;
if ( parms - > lclip < 0 )
{
parms - > lclip = 0 ;
}
break ;
case DTA_ClipRight :
parms - > rclip = va_arg ( tags , int ) ;
if ( parms - > rclip > this - > GetWidth ( ) )
{
parms - > rclip = this - > GetWidth ( ) ;
}
break ;
case DTA_ShadowAlpha :
2016-03-21 11:18:46 +00:00
parms - > shadowAlpha = MIN < fixed_t > ( OPAQUE , va_arg ( tags , fixed_t ) ) ;
2016-03-01 15:47:10 +00:00
break ;
case DTA_ShadowColor :
parms - > shadowColor = va_arg ( tags , int ) ;
break ;
case DTA_Shadow :
boolval = va_arg ( tags , INTBOOL ) ;
if ( boolval )
{
parms - > shadowAlpha = FRACUNIT / 2 ;
parms - > shadowColor = 0 ;
}
else
{
parms - > shadowAlpha = 0 ;
}
break ;
case DTA_Masked :
parms - > masked = va_arg ( tags , INTBOOL ) ;
break ;
case DTA_BilinearFilter :
parms - > bilinear = va_arg ( tags , INTBOOL ) ;
break ;
case DTA_KeepRatio :
// I think this is a terribly misleading name, since it actually turns
// *off* aspect ratio correction.
parms - > keepratio = va_arg ( tags , INTBOOL ) ;
break ;
case DTA_RenderStyle :
parms - > style . AsDWORD = va_arg ( tags , DWORD ) ;
break ;
case DTA_SpecialColormap :
parms - > specialcolormap = va_arg ( tags , FSpecialColormap * ) ;
break ;
case DTA_ColormapStyle :
parms - > colormapstyle = va_arg ( tags , FColormapStyle * ) ;
break ;
2016-04-09 18:47:54 +00:00
case DTA_TextLen :
parms - > maxstrlen = va_arg ( tags , int ) ;
break ;
case DTA_CellX :
parms - > cellx = va_arg ( tags , int ) ;
break ;
case DTA_CellY :
parms - > celly = va_arg ( tags , int ) ;
break ;
2016-03-01 15:47:10 +00:00
}
tag = va_arg ( tags , DWORD ) ;
}
va_end ( tags ) ;
if ( parms - > uclip > = parms - > dclip | | parms - > lclip > = parms - > rclip )
{
return false ;
}
2016-04-09 18:47:54 +00:00
if ( img ! = NULL )
2016-03-01 15:47:10 +00:00
{
2016-04-09 18:47:54 +00:00
SetTextureParms ( parms , img , x , y ) ;
2016-03-01 15:47:10 +00:00
2016-04-09 18:47:54 +00:00
if ( parms - > destwidth < = 0 | | parms - > destheight < = 0 )
{
return false ;
}
2016-03-01 15:47:10 +00:00
}
if ( parms - > style . BlendOp = = 255 )
{
if ( fillcolorset )
{
if ( parms - > alphaChannel )
{
parms - > style = STYLE_Shaded ;
}
2016-03-22 12:35:16 +00:00
else if ( parms - > Alpha < 1.f )
2016-03-01 15:47:10 +00:00
{
parms - > style = STYLE_TranslucentStencil ;
}
else
{
parms - > style = STYLE_Stencil ;
}
}
2016-03-22 12:35:16 +00:00
else if ( parms - > Alpha < 1.f )
2016-03-01 15:47:10 +00:00
{
parms - > style = STYLE_Translucent ;
}
else
{
parms - > style = STYLE_Normal ;
}
}
return true ;
}
void DCanvas : : VirtualToRealCoords ( double & x , double & y , double & w , double & h ,
double vwidth , double vheight , bool vbottom , bool handleaspect ) const
{
2016-09-12 13:51:50 +00:00
float myratio = handleaspect ? ActiveRatio ( Width , Height ) : ( 4.0f / 3.0f ) ;
2016-03-02 20:07:11 +00:00
// if 21:9 AR, map to 16:9 for all callers.
// this allows for black bars and stops the stretching of fullscreen images
2016-09-12 13:51:50 +00:00
if ( myratio > 1.7f ) {
myratio = 16.0f / 9.0f ;
2016-03-02 20:07:11 +00:00
}
2016-03-01 15:47:10 +00:00
double right = x + w ;
double bottom = y + h ;
2016-09-13 21:26:30 +00:00
if ( myratio > 1.334f )
2016-03-01 15:47:10 +00:00
{ // The target surface is either 16:9 or 16:10, so expand the
// specified virtual size to avoid undesired stretching of the
// image. Does not handle non-4:3 virtual sizes. I'll worry about
// those if somebody expresses a desire to use them.
2016-09-12 13:51:50 +00:00
x = ( x - vwidth * 0.5 ) * Width * 960 / ( vwidth * AspectBaseWidth ( myratio ) ) + Width * 0.5 ;
w = ( right - vwidth * 0.5 ) * Width * 960 / ( vwidth * AspectBaseWidth ( myratio ) ) + Width * 0.5 - x ;
2016-03-01 15:47:10 +00:00
}
else
{
x = x * Width / vwidth ;
w = right * Width / vwidth - x ;
}
2016-09-13 21:26:30 +00:00
if ( AspectTallerThanWide ( myratio ) )
2016-03-01 15:47:10 +00:00
{ // The target surface is 5:4
2016-09-12 13:51:50 +00:00
y = ( y - vheight * 0.5 ) * Height * 600 / ( vheight * AspectBaseHeight ( myratio ) ) + Height * 0.5 ;
h = ( bottom - vheight * 0.5 ) * Height * 600 / ( vheight * AspectBaseHeight ( myratio ) ) + Height * 0.5 - y ;
2016-03-01 15:47:10 +00:00
if ( vbottom )
{
2016-09-12 13:51:50 +00:00
y + = ( Height - Height * AspectMultiplier ( myratio ) / 48.0 ) * 0.5 ;
2016-03-01 15:47:10 +00:00
}
}
else
{
y = y * Height / vheight ;
h = bottom * Height / vheight - y ;
}
}
void DCanvas : : VirtualToRealCoordsFixed ( fixed_t & x , fixed_t & y , fixed_t & w , fixed_t & h ,
int vwidth , int vheight , bool vbottom , bool handleaspect ) const
{
double dx , dy , dw , dh ;
dx = FIXED2DBL ( x ) ;
dy = FIXED2DBL ( y ) ;
dw = FIXED2DBL ( w ) ;
dh = FIXED2DBL ( h ) ;
VirtualToRealCoords ( dx , dy , dw , dh , vwidth , vheight , vbottom , handleaspect ) ;
x = FLOAT2FIXED ( dx ) ;
y = FLOAT2FIXED ( dy ) ;
w = FLOAT2FIXED ( dw ) ;
h = FLOAT2FIXED ( dh ) ;
}
void DCanvas : : VirtualToRealCoordsInt ( int & x , int & y , int & w , int & h ,
int vwidth , int vheight , bool vbottom , bool handleaspect ) const
{
double dx , dy , dw , dh ;
dx = x ;
dy = y ;
dw = w ;
dh = h ;
VirtualToRealCoords ( dx , dy , dw , dh , vwidth , vheight , vbottom , handleaspect ) ;
x = int ( dx + 0.5 ) ;
y = int ( dy + 0.5 ) ;
w = int ( dx + dw + 0.5 ) - x ;
h = int ( dy + dh + 0.5 ) - y ;
}
void DCanvas : : FillBorder ( FTexture * img )
{
2016-09-12 13:51:50 +00:00
float myratio = ActiveRatio ( Width , Height ) ;
2016-03-02 20:33:06 +00:00
// if 21:9 AR, fill borders akin to 16:9, since all fullscreen
// images are being drawn to that scale.
2016-09-12 13:51:50 +00:00
if ( myratio > 1.7f ) {
myratio = 16 / 9.0f ;
2016-03-02 20:33:06 +00:00
}
2016-09-12 13:51:50 +00:00
if ( myratio > = 1.3f & & myratio < = 1.4f )
2016-03-01 15:47:10 +00:00
{ // This is a 4:3 display, so no border to show
return ;
}
int bordtop , bordbottom , bordleft , bordright , bord ;
2016-09-13 21:26:30 +00:00
if ( AspectTallerThanWide ( myratio ) )
2016-03-01 15:47:10 +00:00
{ // Screen is taller than it is wide
bordleft = bordright = 0 ;
2016-09-12 13:51:50 +00:00
bord = Height - Height * AspectMultiplier ( myratio ) / 48 ;
2016-03-01 15:47:10 +00:00
bordtop = bord / 2 ;
bordbottom = bord - bordtop ;
}
else
{ // Screen is wider than it is tall
bordtop = bordbottom = 0 ;
2016-09-12 13:51:50 +00:00
bord = Width - Width * AspectMultiplier ( myratio ) / 48 ;
2016-03-01 15:47:10 +00:00
bordleft = bord / 2 ;
bordright = bord - bordleft ;
}
if ( img ! = NULL )
{
FlatFill ( 0 , 0 , Width , bordtop , img ) ; // Top
FlatFill ( 0 , bordtop , bordleft , Height - bordbottom , img ) ; // Left
FlatFill ( Width - bordright , bordtop , Width , Height - bordbottom , img ) ; // Right
FlatFill ( 0 , Height - bordbottom , Width , Height , img ) ; // Bottom
}
else
{
Clear ( 0 , 0 , Width , bordtop , GPalette . BlackIndex , 0 ) ; // Top
Clear ( 0 , bordtop , bordleft , Height - bordbottom , GPalette . BlackIndex , 0 ) ; // Left
Clear ( Width - bordright , bordtop , Width , Height - bordbottom , GPalette . BlackIndex , 0 ) ; // Right
Clear ( 0 , Height - bordbottom , Width , Height , GPalette . BlackIndex , 0 ) ; // Bottom
}
}
void DCanvas : : PUTTRANSDOT ( int xx , int yy , int basecolor , int level )
{
static int oldyy ;
static int oldyyshifted ;
if ( yy = = oldyy + 1 )
{
oldyy + + ;
oldyyshifted + = GetPitch ( ) ;
}
else if ( yy = = oldyy - 1 )
{
oldyy - - ;
oldyyshifted - = GetPitch ( ) ;
}
else if ( yy ! = oldyy )
{
oldyy = yy ;
oldyyshifted = yy * GetPitch ( ) ;
}
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-05-30 23:49:39 +00:00
{
2016-05-31 03:31:32 +00:00
uint32_t * spot = ( uint32_t * ) GetBuffer ( ) + oldyyshifted + xx ;
2016-05-30 23:49:39 +00:00
2016-12-01 01:38:32 +00:00
uint32_t fg = swrenderer : : LightBgra : : shade_pal_index_simple ( basecolor , swrenderer : : LightBgra : : calc_light_multiplier ( 0 ) ) ;
2016-12-25 00:22:31 +00:00
uint32_t fg_red = ( ( ( fg > > 16 ) & 0xff ) * ( 63 - level ) ) > > 6 ;
uint32_t fg_green = ( ( ( fg > > 8 ) & 0xff ) * ( 63 - level ) ) > > 6 ;
uint32_t fg_blue = ( ( fg & 0xff ) * ( 63 - level ) ) > > 6 ;
2016-05-30 23:49:39 +00:00
2016-12-25 00:22:31 +00:00
uint32_t bg_red = ( ( ( * spot > > 16 ) & 0xff ) * level ) > > 6 ;
uint32_t bg_green = ( ( ( * spot > > 8 ) & 0xff ) * level ) > > 6 ;
uint32_t bg_blue = ( ( ( * spot ) & 0xff ) * level ) > > 6 ;
2016-05-30 23:49:39 +00:00
2016-12-25 00:22:31 +00:00
uint32_t red = fg_red + bg_red ;
uint32_t green = fg_green + bg_green ;
uint32_t blue = fg_blue + bg_blue ;
2016-05-30 23:49:39 +00:00
* spot = 0xff000000 | ( red < < 16 ) | ( green < < 8 ) | blue ;
}
2016-12-25 00:00:28 +00:00
else if ( ! r_blendmethod )
2016-05-30 23:49:39 +00:00
{
2016-05-31 03:31:32 +00:00
BYTE * spot = GetBuffer ( ) + oldyyshifted + xx ;
2016-05-30 23:49:39 +00:00
DWORD * bg2rgb = Col2RGB8 [ 1 + level ] ;
DWORD * fg2rgb = Col2RGB8 [ 63 - level ] ;
DWORD fg = fg2rgb [ basecolor ] ;
DWORD bg = bg2rgb [ * spot ] ;
bg = ( fg + bg ) | 0x1f07c1f ;
* spot = RGB32k . All [ bg & ( bg > > 15 ) ] ;
}
2016-12-25 00:00:28 +00:00
else
{
BYTE * spot = GetBuffer ( ) + oldyyshifted + xx ;
uint32_t r = ( GPalette . BaseColors [ * spot ] . r * ( 64 - level ) + GPalette . BaseColors [ basecolor ] . r * level ) / 64 ;
uint32_t g = ( GPalette . BaseColors [ * spot ] . g * ( 64 - level ) + GPalette . BaseColors [ basecolor ] . g * level ) / 64 ;
uint32_t b = ( GPalette . BaseColors [ * spot ] . b * ( 64 - level ) + GPalette . BaseColors [ basecolor ] . b * level ) / 64 ;
* spot = ( BYTE ) RGB256k . RGB [ r ] [ g ] [ b ] ;
}
2016-03-01 15:47:10 +00:00
}
void DCanvas : : DrawLine ( int x0 , int y0 , int x1 , int y1 , int palColor , uint32 realcolor )
//void DrawTransWuLine (int x0, int y0, int x1, int y1, BYTE palColor)
{
const int WeightingScale = 0 ;
const int WEIGHTBITS = 6 ;
const int WEIGHTSHIFT = 16 - WEIGHTBITS ;
const int NUMWEIGHTS = ( 1 < < WEIGHTBITS ) ;
const int WEIGHTMASK = ( NUMWEIGHTS - 1 ) ;
if ( palColor < 0 )
{
palColor = PalFromRGB ( realcolor ) ;
}
Lock ( ) ;
int deltaX , deltaY , xDir ;
if ( y0 > y1 )
{
int temp = y0 ; y0 = y1 ; y1 = temp ;
temp = x0 ; x0 = x1 ; x1 = temp ;
}
PUTTRANSDOT ( x0 , y0 , palColor , 0 ) ;
if ( ( deltaX = x1 - x0 ) > = 0 )
{
xDir = 1 ;
}
else
{
xDir = - 1 ;
deltaX = - deltaX ;
}
if ( ( deltaY = y1 - y0 ) = = 0 )
{ // horizontal line
if ( x0 > x1 )
{
swapvalues ( x0 , x1 ) ;
}
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-05-31 03:31:32 +00:00
{
2016-06-13 18:01:31 +00:00
uint32_t fillColor = GPalette . BaseColors [ palColor ] . d ;
2016-05-31 03:31:32 +00:00
uint32_t * spot = ( uint32_t * ) GetBuffer ( ) + y0 * GetPitch ( ) + x0 ;
for ( int i = 0 ; i < = deltaX ; i + + )
2016-06-13 18:01:31 +00:00
spot [ i ] = fillColor ;
2016-05-31 03:31:32 +00:00
}
else
{
memset ( GetBuffer ( ) + y0 * GetPitch ( ) + x0 , palColor , deltaX + 1 ) ;
}
2016-03-01 15:47:10 +00:00
}
else if ( deltaX = = 0 )
{ // vertical line
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-03-01 15:47:10 +00:00
{
2016-06-13 18:01:31 +00:00
uint32_t fillColor = GPalette . BaseColors [ palColor ] . d ;
2016-05-31 03:31:32 +00:00
uint32_t * spot = ( uint32_t * ) GetBuffer ( ) + y0 * GetPitch ( ) + x0 ;
int pitch = GetPitch ( ) ;
do
{
2016-06-13 18:01:31 +00:00
* spot = fillColor ;
2016-05-31 03:31:32 +00:00
spot + = pitch ;
} while ( - - deltaY ! = 0 ) ;
}
else
2016-03-01 15:47:10 +00:00
{
2016-05-31 03:31:32 +00:00
BYTE * spot = GetBuffer ( ) + y0 * GetPitch ( ) + x0 ;
int pitch = GetPitch ( ) ;
do
{
* spot = palColor ;
spot + = pitch ;
} while ( - - deltaY ! = 0 ) ;
}
2016-03-01 15:47:10 +00:00
}
else if ( deltaX = = deltaY )
{ // diagonal line.
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-05-31 03:31:32 +00:00
{
2016-06-13 18:01:31 +00:00
uint32_t fillColor = GPalette . BaseColors [ palColor ] . d ;
2016-05-31 03:31:32 +00:00
uint32_t * spot = ( uint32_t * ) GetBuffer ( ) + y0 * GetPitch ( ) + x0 ;
int advance = GetPitch ( ) + xDir ;
do
{
2016-06-13 18:01:31 +00:00
* spot = fillColor ;
2016-05-31 03:31:32 +00:00
spot + = advance ;
} while ( - - deltaY ! = 0 ) ;
}
else
2016-03-01 15:47:10 +00:00
{
2016-05-31 03:31:32 +00:00
BYTE * spot = GetBuffer ( ) + y0 * GetPitch ( ) + x0 ;
int advance = GetPitch ( ) + xDir ;
do
{
* spot = palColor ;
spot + = advance ;
} while ( - - deltaY ! = 0 ) ;
}
2016-03-01 15:47:10 +00:00
}
else
{
// line is not horizontal, diagonal, or vertical
fixed_t errorAcc = 0 ;
if ( deltaY > deltaX )
{ // y-major line
fixed_t errorAdj = ( ( ( unsigned ) deltaX < < 16 ) / ( unsigned ) deltaY ) & 0xffff ;
if ( xDir < 0 )
{
if ( WeightingScale = = 0 )
{
while ( - - deltaY )
{
errorAcc + = errorAdj ;
y0 + + ;
int weighting = ( errorAcc > > WEIGHTSHIFT ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 - ( errorAcc > > 16 ) , y0 , palColor , weighting ) ;
PUTTRANSDOT ( x0 - ( errorAcc > > 16 ) - 1 , y0 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
else
{
while ( - - deltaY )
{
errorAcc + = errorAdj ;
y0 + + ;
int weighting = ( ( errorAcc * WeightingScale ) > > ( WEIGHTSHIFT + 8 ) ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 - ( errorAcc > > 16 ) , y0 , palColor , weighting ) ;
PUTTRANSDOT ( x0 - ( errorAcc > > 16 ) - 1 , y0 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
}
else
{
if ( WeightingScale = = 0 )
{
while ( - - deltaY )
{
errorAcc + = errorAdj ;
y0 + + ;
int weighting = ( errorAcc > > WEIGHTSHIFT ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 + ( errorAcc > > 16 ) , y0 , palColor , weighting ) ;
PUTTRANSDOT ( x0 + ( errorAcc > > 16 ) + xDir , y0 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
else
{
while ( - - deltaY )
{
errorAcc + = errorAdj ;
y0 + + ;
int weighting = ( ( errorAcc * WeightingScale ) > > ( WEIGHTSHIFT + 8 ) ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 + ( errorAcc > > 16 ) , y0 , palColor , weighting ) ;
PUTTRANSDOT ( x0 + ( errorAcc > > 16 ) + xDir , y0 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
}
}
else
{ // x-major line
fixed_t errorAdj = ( ( ( DWORD ) deltaY < < 16 ) / ( DWORD ) deltaX ) & 0xffff ;
if ( WeightingScale = = 0 )
{
while ( - - deltaX )
{
errorAcc + = errorAdj ;
x0 + = xDir ;
int weighting = ( errorAcc > > WEIGHTSHIFT ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 , y0 + ( errorAcc > > 16 ) , palColor , weighting ) ;
PUTTRANSDOT ( x0 , y0 + ( errorAcc > > 16 ) + 1 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
else
{
while ( - - deltaX )
{
errorAcc + = errorAdj ;
x0 + = xDir ;
int weighting = ( ( errorAcc * WeightingScale ) > > ( WEIGHTSHIFT + 8 ) ) & WEIGHTMASK ;
PUTTRANSDOT ( x0 , y0 + ( errorAcc > > 16 ) , palColor , weighting ) ;
PUTTRANSDOT ( x0 , y0 + ( errorAcc > > 16 ) + 1 ,
palColor , WEIGHTMASK - weighting ) ;
}
}
}
PUTTRANSDOT ( x1 , y1 , palColor , 0 ) ;
}
Unlock ( ) ;
}
void DCanvas : : DrawPixel ( int x , int y , int palColor , uint32 realcolor )
{
if ( palColor < 0 )
{
palColor = PalFromRGB ( realcolor ) ;
}
Buffer [ Pitch * y + x ] = ( BYTE ) palColor ;
}
//==========================================================================
//
// DCanvas :: Clear
//
// Set an area to a specified color.
//
//==========================================================================
void DCanvas : : Clear ( int left , int top , int right , int bottom , int palcolor , uint32 color )
{
int x , y ;
if ( left = = right | | top = = bottom )
{
return ;
}
assert ( left < right ) ;
assert ( top < bottom ) ;
if ( left > = Width | | right < = 0 | | top > = Height | | bottom < = 0 )
{
return ;
}
left = MAX ( 0 , left ) ;
right = MIN ( Width , right ) ;
top = MAX ( 0 , top ) ;
bottom = MIN ( Height , bottom ) ;
if ( palcolor < 0 )
{
if ( APART ( color ) ! = 255 )
{
Dim ( color , APART ( color ) / 255.f , left , top , right - left , bottom - top ) ;
return ;
}
palcolor = PalFromRGB ( color ) ;
}
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-03-01 15:47:10 +00:00
{
2016-06-13 18:01:31 +00:00
uint32_t fill_color = GPalette . BaseColors [ palcolor ] ;
2016-05-31 03:31:32 +00:00
uint32_t * dest = ( uint32_t * ) Buffer + top * Pitch + left ;
x = right - left ;
for ( y = top ; y < bottom ; y + + )
{
for ( int i = 0 ; i < x ; i + + )
2016-06-13 18:01:31 +00:00
dest [ i ] = fill_color ;
2016-05-31 03:31:32 +00:00
dest + = Pitch ;
}
}
else
{
BYTE * dest = Buffer + top * Pitch + left ;
x = right - left ;
for ( y = top ; y < bottom ; y + + )
{
memset ( dest , palcolor , x ) ;
dest + = Pitch ;
}
2016-03-01 15:47:10 +00:00
}
}
2016-08-06 17:20:41 +00:00
//==========================================================================
//
// no-ops. This is so that renderer backends can better manage the
// processing of the subsector drawing in the automap
//
//==========================================================================
void DCanvas : : StartSimplePolys ( )
{ }
void DCanvas : : FinishSimplePolys ( )
{ }
2016-03-01 15:47:10 +00:00
//==========================================================================
//
// DCanvas :: FillSimplePoly
//
// Fills a simple polygon with a texture. Here, "simple" means that a
// horizontal scanline at any vertical position within the polygon will
// not cross it more than twice.
//
// The originx, originy, scale, and rotation parameters specify
// transformation of the filling texture, not of the points.
//
// The points must be specified in clockwise order.
//
//==========================================================================
void DCanvas : : 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 ,
2016-11-06 03:49:33 +00:00
FDynamicColormap * colormap , int lightlevel , int bottomclip )
2016-03-01 15:47:10 +00:00
{
# ifndef NO_SWRENDER
2016-12-01 01:38:32 +00:00
using namespace swrenderer ;
2016-12-05 07:22:04 +00:00
using namespace drawerargs ;
2016-12-01 01:38:32 +00:00
2016-03-01 15:47:10 +00:00
// Use an equation similar to player sprites to determine shade
fixed_t shade = LIGHT2SHADE ( lightlevel ) - 12 * FRACUNIT ;
float topy , boty , leftx , rightx ;
int toppt , botpt , pt1 , pt2 ;
int i ;
int y1 , y2 , y ;
fixed_t x ;
2016-03-24 15:36:43 +00:00
bool dorotate = rotation ! = 0. ;
2016-03-01 15:47:10 +00:00
double cosrot , sinrot ;
if ( - - npoints < 2 | | Buffer = = NULL )
{ // not a polygon or we're not locked
return ;
}
2016-11-06 03:49:33 +00:00
if ( bottomclip < = 0 )
{
bottomclip = Height ;
}
2016-03-01 15:47:10 +00:00
// Find the extents of the polygon, in particular the highest and lowest points.
for ( botpt = toppt = 0 , boty = topy = points [ 0 ] . Y , leftx = rightx = points [ 0 ] . X , i = 1 ; i < = npoints ; + + i )
{
if ( points [ i ] . Y < topy )
{
topy = points [ i ] . Y ;
toppt = i ;
}
if ( points [ i ] . Y > boty )
{
boty = points [ i ] . Y ;
botpt = i ;
}
if ( points [ i ] . X < leftx )
{
leftx = points [ i ] . X ;
}
if ( points [ i ] . X > rightx )
{
rightx = points [ i ] . X ;
}
}
2016-11-06 03:49:33 +00:00
if ( topy > = bottomclip | | // off the bottom of the screen
2016-03-01 15:47:10 +00:00
boty < = 0 | | // off the top of the screen
leftx > = Width | | // off the right of the screen
rightx < = 0 ) // off the left of the screen
{
return ;
}
2016-11-01 03:34:46 +00:00
BYTE * destorgsave = dc_destorg ;
dc_destorg = screen - > GetBuffer ( ) ;
if ( dc_destorg = = NULL )
{
I_FatalError ( " Attempt to write to buffer of hardware canvas " ) ;
}
2016-03-26 12:37:44 +00:00
scalex / = tex - > Scale . X ;
scaley / = tex - > Scale . Y ;
2016-03-01 15:47:10 +00:00
2016-03-24 15:36:43 +00:00
// Use the CRT's functions here.
2016-03-30 14:30:22 +00:00
cosrot = cos ( rotation . Radians ( ) ) ;
sinrot = sin ( rotation . Radians ( ) ) ;
2016-03-01 15:47:10 +00:00
// Setup constant texture mapping parameters.
2016-12-27 06:30:29 +00:00
R_SetSpanTexture ( tex ) ;
2016-06-05 12:08:03 +00:00
if ( colormap )
R_SetSpanColormap ( colormap , clamp ( shade > > FRACBITS , 0 , NUMCOLORMAPS - 1 ) ) ;
else
R_SetSpanColormap ( & identitycolormap , 0 ) ;
2016-11-06 03:29:02 +00:00
if ( ds_xbits ! = 0 )
{
scalex = double ( 1u < < ( 32 - ds_xbits ) ) / scalex ;
ds_xstep = xs_RoundToInt ( cosrot * scalex ) ;
}
else
{ // Texture is one pixel wide.
scalex = 0 ;
ds_xstep = 0 ;
}
if ( ds_ybits ! = 0 )
{
scaley = double ( 1u < < ( 32 - ds_ybits ) ) / scaley ;
ds_ystep = xs_RoundToInt ( sinrot * scaley ) ;
}
else
{ // Texture is one pixel tall.
scaley = 0 ;
ds_ystep = 0 ;
}
2016-03-01 15:47:10 +00:00
// Travel down the right edge and create an outline of that edge.
2016-12-31 10:42:49 +00:00
static short spanend [ MAXHEIGHT ] ;
2016-03-01 15:47:10 +00:00
pt1 = toppt ;
pt2 = toppt + 1 ; if ( pt2 > npoints ) pt2 = 0 ;
y1 = xs_RoundToInt ( points [ pt1 ] . Y + 0.5f ) ;
do
{
x = FLOAT2FIXED ( points [ pt1 ] . X + 0.5f ) ;
y2 = xs_RoundToInt ( points [ pt2 ] . Y + 0.5f ) ;
2016-11-06 03:49:33 +00:00
if ( y1 > = y2 | | ( y1 < 0 & & y2 < 0 ) | | ( y1 > = bottomclip & & y2 > = bottomclip ) )
2016-03-01 15:47:10 +00:00
{
}
else
{
fixed_t xinc = FLOAT2FIXED ( ( points [ pt2 ] . X - points [ pt1 ] . X ) / ( points [ pt2 ] . Y - points [ pt1 ] . Y ) ) ;
2016-11-06 03:49:33 +00:00
int y3 = MIN ( y2 , bottomclip ) ;
2016-03-01 15:47:10 +00:00
if ( y1 < 0 )
{
x + = xinc * - y1 ;
y1 = 0 ;
}
for ( y = y1 ; y < y3 ; + + y )
{
spanend [ y ] = clamp < short > ( x > > FRACBITS , - 1 , Width ) ;
x + = xinc ;
}
}
y1 = y2 ;
pt1 = pt2 ;
pt2 + + ; if ( pt2 > npoints ) pt2 = 0 ;
} while ( pt1 ! = botpt ) ;
// Travel down the left edge and fill it in.
pt1 = toppt ;
pt2 = toppt - 1 ; if ( pt2 < 0 ) pt2 = npoints ;
y1 = xs_RoundToInt ( points [ pt1 ] . Y + 0.5f ) ;
do
{
x = FLOAT2FIXED ( points [ pt1 ] . X + 0.5f ) ;
y2 = xs_RoundToInt ( points [ pt2 ] . Y + 0.5f ) ;
2016-11-06 03:49:33 +00:00
if ( y1 > = y2 | | ( y1 < 0 & & y2 < 0 ) | | ( y1 > = bottomclip & & y2 > = bottomclip ) )
2016-03-01 15:47:10 +00:00
{
}
else
{
fixed_t xinc = FLOAT2FIXED ( ( points [ pt2 ] . X - points [ pt1 ] . X ) / ( points [ pt2 ] . Y - points [ pt1 ] . Y ) ) ;
2016-11-06 03:49:33 +00:00
int y3 = MIN ( y2 , bottomclip ) ;
2016-03-01 15:47:10 +00:00
if ( y1 < 0 )
{
x + = xinc * - y1 ;
y1 = 0 ;
}
for ( y = y1 ; y < y3 ; + + y )
{
int x1 = x > > FRACBITS ;
int x2 = spanend [ y ] ;
if ( x2 > x1 & & x2 > 0 & & x1 < Width )
{
x1 = MAX ( x1 , 0 ) ;
x2 = MIN ( x2 , Width ) ;
#if 0
memset ( this - > Buffer + y * this - > Pitch + x1 , ( int ) tex , x2 - x1 ) ;
# else
ds_y = y ;
ds_x1 = x1 ;
ds_x2 = x2 - 1 ;
2016-03-10 19:44:53 +00:00
DVector2 tex ( x1 - originx , y - originy ) ;
2016-03-01 15:47:10 +00:00
if ( dorotate )
{
double t = tex . X ;
tex . X = t * cosrot - tex . Y * sinrot ;
tex . Y = tex . Y * cosrot + t * sinrot ;
}
ds_xfrac = xs_RoundToInt ( tex . X * scalex ) ;
ds_yfrac = xs_RoundToInt ( tex . Y * scaley ) ;
2016-12-27 22:53:47 +00:00
R_Drawers ( ) - > DrawSpan ( ) ;
2016-03-01 15:47:10 +00:00
# endif
}
x + = xinc ;
}
}
y1 = y2 ;
pt1 = pt2 ;
pt2 - - ; if ( pt2 < 0 ) pt2 = npoints ;
} while ( pt1 ! = botpt ) ;
2016-11-01 03:34:46 +00:00
dc_destorg = destorgsave ;
2016-03-01 15:47:10 +00:00
# endif
}
/********************************/
/* */
/* Other miscellaneous routines */
/* */
/********************************/
//
// V_DrawBlock
// Draw a linear block of pixels into the view buffer.
//
void DCanvas : : DrawBlock ( int x , int y , int _width , int _height , const BYTE * src ) const
{
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-05-31 03:31:32 +00:00
return ;
2016-03-01 15:47:10 +00:00
int srcpitch = _width ;
int destpitch ;
BYTE * dest ;
if ( ClipBox ( x , y , _width , _height , src , srcpitch ) )
{
return ; // Nothing to draw
}
destpitch = Pitch ;
dest = Buffer + y * Pitch + x ;
do
{
memcpy ( dest , src , _width ) ;
src + = srcpitch ;
dest + = destpitch ;
} while ( - - _height ) ;
}
//
// V_GetBlock
// Gets a linear block of pixels from the view buffer.
//
void DCanvas : : GetBlock ( int x , int y , int _width , int _height , BYTE * dest ) const
{
2016-05-31 07:36:18 +00:00
if ( IsBgra ( ) )
2016-05-31 03:31:32 +00:00
return ;
2016-03-01 15:47:10 +00:00
const BYTE * src ;
# ifdef RANGECHECK
if ( x < 0
| | x + _width > Width
| | y < 0
| | y + _height > Height )
{
I_Error ( " Bad V_GetBlock " ) ;
}
# endif
src = Buffer + y * Pitch + x ;
while ( _height - - )
{
memcpy ( dest , src , _width ) ;
src + = Pitch ;
dest + = _width ;
}
}
// Returns true if the box was completely clipped. False otherwise.
bool DCanvas : : ClipBox ( int & x , int & y , int & w , int & h , const BYTE * & src , const int srcpitch ) const
{
if ( x > = Width | | y > = Height | | x + w < = 0 | | y + h < = 0 )
{ // Completely clipped off screen
return true ;
}
if ( x < 0 ) // clip left edge
{
src - = x ;
w + = x ;
x = 0 ;
}
if ( x + w > Width ) // clip right edge
{
w = Width - x ;
}
if ( y < 0 ) // clip top edge
{
src - = y * srcpitch ;
h + = y ;
y = 0 ;
}
if ( y + h > Height ) // clip bottom edge
{
h = Height - y ;
}
return false ;
}
//==========================================================================
//
// V_SetBorderNeedRefresh
//
// Flag the border as in need of updating. (Probably because something that
// was on top of it has changed.
//
//==========================================================================
void V_SetBorderNeedRefresh ( )
{
if ( screen ! = NULL )
{
BorderNeedRefresh = screen - > GetPageCount ( ) ;
}
}
//==========================================================================
//
// V_DrawFrame
//
// Draw a frame around the specified area using the view border
// frame graphics. The border is drawn outside the area, not in it.
//
//==========================================================================
void V_DrawFrame ( int left , int top , int width , int height )
{
FTexture * p ;
const gameborder_t * border = & gameinfo . Border ;
// Sanity check for incomplete gameinfo
if ( border = = NULL )
return ;
int offset = border - > offset ;
int right = left + width ;
int bottom = top + height ;
// Draw top and bottom sides.
p = TexMan [ border - > t ] ;
screen - > FlatFill ( left , top - p - > GetHeight ( ) , right , top , p , true ) ;
p = TexMan [ border - > b ] ;
screen - > FlatFill ( left , bottom , right , bottom + p - > GetHeight ( ) , p , true ) ;
// Draw left and right sides.
p = TexMan [ border - > l ] ;
screen - > FlatFill ( left - p - > GetWidth ( ) , top , left , bottom , p , true ) ;
p = TexMan [ border - > r ] ;
screen - > FlatFill ( right , top , right + p - > GetWidth ( ) , bottom , p , true ) ;
// Draw beveled corners.
screen - > DrawTexture ( TexMan [ border - > tl ] , left - offset , top - offset , TAG_DONE ) ;
screen - > DrawTexture ( TexMan [ border - > tr ] , left + width , top - offset , TAG_DONE ) ;
screen - > DrawTexture ( TexMan [ border - > bl ] , left - offset , top + height , TAG_DONE ) ;
screen - > DrawTexture ( TexMan [ border - > br ] , left + width , top + height , TAG_DONE ) ;
}
//==========================================================================
//
// V_DrawBorder
//
//==========================================================================
void V_DrawBorder ( int x1 , int y1 , int x2 , int y2 )
{
FTextureID picnum ;
if ( level . info ! = NULL & & level . info - > BorderTexture . Len ( ) ! = 0 )
{
picnum = TexMan . CheckForTexture ( level . info - > BorderTexture , FTexture : : TEX_Flat ) ;
}
else
{
picnum = TexMan . CheckForTexture ( gameinfo . BorderFlat , FTexture : : TEX_Flat ) ;
}
if ( picnum . isValid ( ) )
{
screen - > FlatFill ( x1 , y1 , x2 , y2 , TexMan ( picnum ) ) ;
}
else
{
screen - > Clear ( x1 , y1 , x2 , y2 , 0 , 0 ) ;
}
}
//==========================================================================
//
// R_DrawViewBorder
//
// Draws the border around the view for different size windows
//
//==========================================================================
int BorderNeedRefresh ;
static void V_DrawViewBorder ( void )
{
// [RH] Redraw the status bar if SCREENWIDTH > status bar width.
// Will draw borders around itself, too.
if ( SCREENWIDTH > 320 )
{
ST_SetNeedRefresh ( ) ;
}
if ( viewwidth = = SCREENWIDTH )
{
return ;
}
V_DrawBorder ( 0 , 0 , SCREENWIDTH , viewwindowy ) ;
V_DrawBorder ( 0 , viewwindowy , viewwindowx , viewheight + viewwindowy ) ;
V_DrawBorder ( viewwindowx + viewwidth , viewwindowy , SCREENWIDTH , viewheight + viewwindowy ) ;
2017-01-20 09:51:21 +00:00
V_DrawBorder ( 0 , viewwindowy + viewheight , SCREENWIDTH , gST_Y ) ;
2016-03-01 15:47:10 +00:00
V_DrawFrame ( viewwindowx , viewwindowy , viewwidth , viewheight ) ;
2017-01-20 09:51:21 +00:00
V_MarkRect ( 0 , 0 , SCREENWIDTH , gST_Y ) ;
2016-03-01 15:47:10 +00:00
}
//==========================================================================
//
// R_DrawTopBorder
//
// Draws the top border around the view for different size windows
//
//==========================================================================
static void V_DrawTopBorder ( )
{
FTexture * p ;
int offset ;
if ( viewwidth = = SCREENWIDTH )
return ;
offset = gameinfo . Border . offset ;
if ( viewwindowy < 34 )
{
V_DrawBorder ( 0 , 0 , viewwindowx , 34 ) ;
V_DrawBorder ( viewwindowx , 0 , viewwindowx + viewwidth , viewwindowy ) ;
V_DrawBorder ( viewwindowx + viewwidth , 0 , SCREENWIDTH , 34 ) ;
p = TexMan ( gameinfo . Border . t ) ;
screen - > FlatFill ( viewwindowx , viewwindowy - p - > GetHeight ( ) ,
viewwindowx + viewwidth , viewwindowy , p , true ) ;
p = TexMan ( gameinfo . Border . l ) ;
screen - > FlatFill ( viewwindowx - p - > GetWidth ( ) , viewwindowy ,
viewwindowx , 35 , p , true ) ;
p = TexMan ( gameinfo . Border . r ) ;
screen - > FlatFill ( viewwindowx + viewwidth , viewwindowy ,
viewwindowx + viewwidth + p - > GetWidth ( ) , 35 , p , true ) ;
p = TexMan ( gameinfo . Border . tl ) ;
screen - > DrawTexture ( p , viewwindowx - offset , viewwindowy - offset , TAG_DONE ) ;
p = TexMan ( gameinfo . Border . tr ) ;
screen - > DrawTexture ( p , viewwindowx + viewwidth , viewwindowy - offset , TAG_DONE ) ;
}
else
{
V_DrawBorder ( 0 , 0 , SCREENWIDTH , 34 ) ;
}
}
//==========================================================================
//
// R_RefreshViewBorder
//
// Draws the border around the player view, if needed.
//
//==========================================================================
void V_RefreshViewBorder ( )
{
if ( setblocks < 10 )
{
if ( BorderNeedRefresh )
{
BorderNeedRefresh - - ;
if ( BorderTopRefresh )
{
BorderTopRefresh - - ;
}
V_DrawViewBorder ( ) ;
}
else if ( BorderTopRefresh )
{
BorderTopRefresh - - ;
V_DrawTopBorder ( ) ;
}
}
}