2014-03-15 16:59:03 +00:00
// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
2016-05-18 00:42:11 +00:00
// Copyright (C) 2014-2016 by John "JTE" Muniz.
2018-11-25 12:35:38 +00:00
// Copyright (C) 2014-2018 by Sonic Team Junior.
2014-03-15 16:59:03 +00:00
//
// This program is free software distributed under the
// terms of the GNU General Public License, version 2.
// See the 'LICENSE' file for more details.
//-----------------------------------------------------------------------------
/// \file lua_hudlib.c
/// \brief custom HUD rendering library for Lua scripting
# include "doomdef.h"
# ifdef HAVE_BLUA
# include "r_defs.h"
2014-11-12 00:55:07 +00:00
# include "r_local.h"
2014-03-15 16:59:03 +00:00
# include "st_stuff.h" // hudinfo[]
# include "g_game.h"
2016-01-31 21:53:14 +00:00
# include "i_video.h" // rendermode
2014-08-04 03:49:33 +00:00
# include "p_local.h" // camera_t
2016-01-31 21:53:14 +00:00
# include "screen.h" // screen width/height
2014-03-15 16:59:03 +00:00
# include "v_video.h"
# include "w_wad.h"
# include "z_zone.h"
# include "lua_script.h"
# include "lua_libs.h"
# include "lua_hud.h"
2014-08-04 03:49:33 +00:00
# define HUDONLY if (!hud_running) return luaL_error(L, "HUD rendering code should not be called outside of rendering hooks!");
2014-03-15 16:59:03 +00:00
boolean hud_running = false ;
static UINT8 hud_enabled [ ( hud_MAX / 8 ) + 1 ] ;
2014-04-14 05:14:58 +00:00
static UINT8 hudAvailable ; // hud hooks field
2019-01-28 17:28:15 +00:00
static UINT8 camnum = 1 ;
2014-03-15 16:59:03 +00:00
// must match enum hud in lua_hud.h
static const char * const hud_disable_options [ ] = {
" stagetitle " ,
" textspectator " ,
" time " ,
2018-10-30 17:24:03 +00:00
" gametypeinfo " , // Bumpers / Karma / Laps depending on gametype
" minimap " ,
" item " ,
" position " ,
2018-10-31 14:18:05 +00:00
" minirankings " , // Gametype rankings to the left
2019-01-13 13:40:48 +00:00
" battlerankingsbumpers " , // bumper drawer for battle. Useful if you want to make a custom battle gamemode without bumpers being involved.
2019-02-17 22:39:00 +00:00
" battlefullscreen " , // battlefullscreen func (WAIT, ATTACK OR PROTECT ...)
" battlecomebacktimer " , // come back timer in battlefullscreen
2018-10-30 17:24:03 +00:00
" wanted " ,
" speedometer " ,
2019-01-13 13:40:48 +00:00
" freeplay " ,
2014-03-15 16:59:03 +00:00
" rankings " ,
NULL } ;
enum hudinfo {
hudinfo_x = 0 ,
hudinfo_y
} ;
static const char * const hudinfo_opt [ ] = {
" x " ,
" y " ,
NULL } ;
enum patch {
patch_valid = 0 ,
patch_width ,
patch_height ,
patch_leftoffset ,
patch_topoffset
} ;
static const char * const patch_opt [ ] = {
" valid " ,
" width " ,
" height " ,
" leftoffset " ,
" topoffset " ,
NULL } ;
enum hudhook {
hudhook_game = 0 ,
hudhook_scores
} ;
static const char * const hudhook_opt [ ] = {
" game " ,
" scores " ,
NULL } ;
2014-08-04 03:49:33 +00:00
// alignment types for v.drawString
2014-03-15 16:59:03 +00:00
enum align {
align_left = 0 ,
align_center ,
align_right ,
2014-08-04 03:49:33 +00:00
align_fixed ,
align_small ,
align_smallright ,
align_thin ,
align_thinright
2014-03-15 16:59:03 +00:00
} ;
static const char * const align_opt [ ] = {
" left " ,
" center " ,
" right " ,
" fixed " ,
2014-08-04 03:49:33 +00:00
" small " ,
" small-right " ,
" thin " ,
" thin-right " ,
NULL } ;
// width types for v.stringWidth
enum widtht {
widtht_normal = 0 ,
widtht_small ,
widtht_thin
} ;
static const char * const widtht_opt [ ] = {
" normal " ,
" small " ,
" thin " ,
NULL } ;
enum cameraf {
camera_chase = 0 ,
camera_aiming ,
camera_x ,
camera_y ,
camera_z ,
camera_angle ,
camera_subsector ,
camera_floorz ,
camera_ceilingz ,
camera_radius ,
camera_height ,
camera_momx ,
camera_momy ,
2019-01-28 17:28:15 +00:00
camera_momz ,
camera_pnum
2014-08-04 03:49:33 +00:00
} ;
static const char * const camera_opt [ ] = {
" chase " ,
" aiming " ,
" x " ,
" y " ,
" z " ,
" angle " ,
" subsector " ,
" floorz " ,
" ceilingz " ,
" radius " ,
" height " ,
" momx " ,
" momy " ,
" momz " ,
2019-01-28 17:28:15 +00:00
" pnum " ,
2014-03-15 16:59:03 +00:00
NULL } ;
static int lib_getHudInfo ( lua_State * L )
{
2014-08-27 03:56:30 +00:00
UINT32 i ;
lua_remove ( L , 1 ) ;
i = luaL_checkinteger ( L , 1 ) ;
2016-05-01 21:14:42 +00:00
if ( i > = NUMHUDITEMS )
return luaL_error ( L , " hudinfo[] index %d out of range (0 - %d) " , i, NUMHUDITEMS-1) ;
2014-08-27 03:56:30 +00:00
LUA_PushUserdata ( L , & hudinfo [ i ] , META_HUDINFO ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_hudinfolen ( lua_State * L )
{
lua_pushinteger ( L , NUMHUDITEMS ) ;
return 1 ;
}
static int hudinfo_get ( lua_State * L )
{
hudinfo_t * info = * ( ( hudinfo_t * * ) luaL_checkudata ( L , 1 , META_HUDINFO ) ) ;
enum hudinfo field = luaL_checkoption ( L , 2 , hudinfo_opt [ 0 ] , hudinfo_opt ) ;
I_Assert ( info ! = NULL ) ; // huditems are always valid
switch ( field )
{
case hudinfo_x :
lua_pushinteger ( L , info - > x ) ;
break ;
case hudinfo_y :
lua_pushinteger ( L , info - > y ) ;
break ;
}
return 1 ;
}
static int hudinfo_set ( lua_State * L )
{
hudinfo_t * info = * ( ( hudinfo_t * * ) luaL_checkudata ( L , 1 , META_HUDINFO ) ) ;
enum hudinfo field = luaL_checkoption ( L , 2 , hudinfo_opt [ 0 ] , hudinfo_opt ) ;
I_Assert ( info ! = NULL ) ;
switch ( field )
{
case hudinfo_x :
info - > x = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
break ;
case hudinfo_y :
info - > y = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
break ;
}
return 0 ;
}
static int hudinfo_num ( lua_State * L )
{
hudinfo_t * info = * ( ( hudinfo_t * * ) luaL_checkudata ( L , 1 , META_HUDINFO ) ) ;
lua_pushinteger ( L , info - hudinfo ) ;
return 1 ;
}
2014-11-12 00:55:07 +00:00
static int colormap_get ( lua_State * L )
{
return luaL_error ( L , " colormap is not a struct. " ) ;
}
2014-03-15 16:59:03 +00:00
static int patch_get ( lua_State * L )
{
patch_t * patch = * ( ( patch_t * * ) luaL_checkudata ( L , 1 , META_PATCH ) ) ;
enum patch field = luaL_checkoption ( L , 2 , NULL , patch_opt ) ;
// patches are CURRENTLY always valid, expected to be cached with PU_STATIC
// this may change in the future, so patch.valid still exists
I_Assert ( patch ! = NULL ) ;
switch ( field )
{
case patch_valid :
lua_pushboolean ( L , patch ! = NULL ) ;
break ;
case patch_width :
lua_pushinteger ( L , SHORT ( patch - > width ) ) ;
break ;
case patch_height :
lua_pushinteger ( L , SHORT ( patch - > height ) ) ;
break ;
case patch_leftoffset :
lua_pushinteger ( L , SHORT ( patch - > leftoffset ) ) ;
break ;
case patch_topoffset :
lua_pushinteger ( L , SHORT ( patch - > topoffset ) ) ;
break ;
}
return 1 ;
}
static int patch_set ( lua_State * L )
{
return luaL_error ( L , LUA_QL ( " patch_t " ) " struct cannot be edited by Lua. " ) ;
}
2014-08-04 03:49:33 +00:00
static int camera_get ( lua_State * L )
{
camera_t * cam = * ( ( camera_t * * ) luaL_checkudata ( L , 1 , META_CAMERA ) ) ;
enum cameraf field = luaL_checkoption ( L , 2 , NULL , camera_opt ) ;
// cameras should always be valid unless I'm a nutter
I_Assert ( cam ! = NULL ) ;
switch ( field )
{
case camera_chase :
lua_pushboolean ( L , cam - > chase ) ;
break ;
case camera_aiming :
lua_pushinteger ( L , cam - > aiming ) ;
break ;
case camera_x :
lua_pushinteger ( L , cam - > x ) ;
break ;
case camera_y :
lua_pushinteger ( L , cam - > y ) ;
break ;
case camera_z :
lua_pushinteger ( L , cam - > z ) ;
break ;
case camera_angle :
lua_pushinteger ( L , cam - > angle ) ;
break ;
case camera_subsector :
LUA_PushUserdata ( L , cam - > subsector , META_SUBSECTOR ) ;
break ;
case camera_floorz :
lua_pushinteger ( L , cam - > floorz ) ;
break ;
case camera_ceilingz :
lua_pushinteger ( L , cam - > ceilingz ) ;
break ;
case camera_radius :
lua_pushinteger ( L , cam - > radius ) ;
break ;
case camera_height :
lua_pushinteger ( L , cam - > height ) ;
break ;
case camera_momx :
lua_pushinteger ( L , cam - > momx ) ;
break ;
case camera_momy :
lua_pushinteger ( L , cam - > momy ) ;
break ;
case camera_momz :
lua_pushinteger ( L , cam - > momz ) ;
break ;
2019-01-28 17:28:15 +00:00
case camera_pnum :
lua_pushinteger ( L , camnum ) ;
break ;
2014-08-04 03:49:33 +00:00
}
return 1 ;
}
2014-03-15 16:59:03 +00:00
//
// lib_draw
//
2014-03-23 16:00:29 +00:00
static int libd_patchExists ( lua_State * L )
{
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-25 02:17:59 +00:00
lua_pushboolean ( L , W_LumpExists ( luaL_checkstring ( L , 1 ) ) ) ;
2014-03-23 16:00:29 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int libd_cachePatch ( lua_State * L )
{
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-15 16:59:03 +00:00
LUA_PushUserdata ( L , W_CachePatchName ( luaL_checkstring ( L , 1 ) , PU_STATIC ) , META_PATCH ) ;
return 1 ;
}
static int libd_draw ( lua_State * L )
{
INT32 x , y , flags ;
patch_t * patch ;
const UINT8 * colormap = NULL ;
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-15 16:59:03 +00:00
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
patch = * ( ( patch_t * * ) luaL_checkudata ( L , 3 , META_PATCH ) ) ;
flags = luaL_optinteger ( L , 4 , 0 ) ;
if ( ! lua_isnoneornil ( L , 5 ) )
2014-11-12 00:55:07 +00:00
colormap = * ( ( UINT8 * * ) luaL_checkudata ( L , 5 , META_COLORMAP ) ) ;
2014-03-15 16:59:03 +00:00
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
2014-03-21 18:42:55 +00:00
V_DrawFixedPatch ( x < < FRACBITS , y < < FRACBITS , FRACUNIT , flags , patch , colormap ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int libd_drawScaled ( lua_State * L )
{
fixed_t x , y , scale ;
INT32 flags ;
patch_t * patch ;
2014-03-21 18:42:55 +00:00
const UINT8 * colormap = NULL ;
2014-03-15 16:59:03 +00:00
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-15 16:59:03 +00:00
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
2014-03-21 18:42:55 +00:00
scale = luaL_checkinteger ( L , 3 ) ;
2016-10-18 21:07:20 +00:00
if ( scale < 0 )
return luaL_error ( L , " negative scale " ) ;
2014-03-21 18:42:55 +00:00
patch = * ( ( patch_t * * ) luaL_checkudata ( L , 4 , META_PATCH ) ) ;
flags = luaL_optinteger ( L , 5 , 0 ) ;
if ( ! lua_isnoneornil ( L , 6 ) )
2014-11-12 00:55:07 +00:00
colormap = * ( ( UINT8 * * ) luaL_checkudata ( L , 6 , META_COLORMAP ) ) ;
2014-03-21 18:42:55 +00:00
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
V_DrawFixedPatch ( x , y , scale , flags , patch , colormap ) ;
return 0 ;
}
2019-05-26 13:05:10 +00:00
// KART: draw patch on minimap from x, y coordinates on the map
static int libd_drawOnMinimap ( lua_State * L )
{
fixed_t x , y , scale ; // coordinates of the object
patch_t * patch ; // patch we want to draw
const UINT8 * colormap = NULL ; // do we want to colormap this patch?
boolean centered ; // the patch is centered and doesn't need readjusting on x/y coordinates.
// variables used to replicate k_kart's mmap drawer:
INT32 lumpnum ;
patch_t * AutomapPic ;
INT32 mx , my ;
INT32 splitflags , minimaptrans ;
// base position of the minimap which also takes splits into account:
INT32 MM_X , MM_Y ;
// variables used for actually drawing the icon:
fixed_t amnumxpos , amnumypos ;
INT32 amxpos , amypos ;
node_t * bsp = & nodes [ numnodes - 1 ] ;
fixed_t maxx , minx , maxy , miny ;
fixed_t mapwidth , mapheight ;
fixed_t xoffset , yoffset ;
fixed_t xscale , yscale , zoom ;
fixed_t patchw , patchh ;
HUDONLY // only run this function in hud hooks
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
scale = luaL_checkinteger ( L , 3 ) ;
patch = * ( ( patch_t * * ) luaL_checkudata ( L , 4 , META_PATCH ) ) ;
if ( ! lua_isnoneornil ( L , 5 ) )
colormap = * ( ( UINT8 * * ) luaL_checkudata ( L , 5 , META_COLORMAP ) ) ;
centered = lua_optboolean ( L , 6 ) ;
// replicate exactly what source does for its minimap drawer; AKA hardcoded garbo.
// first, check what position the mmap is supposed to be in (pasted from k_kart.c):
MM_X = BASEVIDWIDTH - 50 ; // 270
MM_Y = ( BASEVIDHEIGHT / 2 ) - 16 ; // 84
if ( splitscreen )
{
MM_Y = ( BASEVIDHEIGHT / 2 ) ;
if ( splitscreen > 1 ) // 3P : bottom right
{
MM_X = ( 3 * BASEVIDWIDTH / 4 ) ;
MM_Y = ( 3 * BASEVIDHEIGHT / 4 ) ;
if ( splitscreen > 2 ) // 4P: centered
{
MM_X = ( BASEVIDWIDTH / 2 ) ;
MM_Y = ( BASEVIDHEIGHT / 2 ) ;
}
}
}
// splitscreen flags
splitflags = ( splitscreen = = 3 ? 0 : V_SNAPTORIGHT ) ; // flags should only be 0 when it's centered (4p split)
// translucency:
if ( timeinmap > 105 )
{
minimaptrans = cv_kartminimap . value ;
if ( timeinmap < = 113 )
minimaptrans = ( ( ( ( INT32 ) timeinmap ) - 105 ) * minimaptrans ) / ( 113 - 105 ) ;
if ( ! minimaptrans )
return 0 ;
}
else
return 0 ;
minimaptrans = ( ( 10 - minimaptrans ) < < FF_TRANSSHIFT ) ;
splitflags | = minimaptrans ;
if ( ! ( splitscreen = = 2 ) )
{
splitflags & = ~ minimaptrans ;
splitflags | = V_HUDTRANSHALF ;
}
splitflags & = ~ V_HUDTRANSHALF ;
splitflags | = V_HUDTRANS ;
// Draw the HUD only when playing in a level.
// hu_stuff needs this, unlike st_stuff.
if ( gamestate ! = GS_LEVEL )
return 0 ;
if ( stplyr ! = & players [ displayplayers [ 0 ] ] )
return 0 ;
lumpnum = W_CheckNumForName ( va ( " %sR " , G_BuildMapName ( gamemap ) ) ) ;
if ( lumpnum ! = - 1 )
AutomapPic = W_CachePatchName ( va ( " %sR " , G_BuildMapName ( gamemap ) ) , PU_HUDGFX ) ;
else
return 0 ; // no pic, just get outta here
mx = MM_X - ( AutomapPic - > width / 2 ) ;
my = MM_Y - ( AutomapPic - > height / 2 ) ;
// let offsets transfer to the heads, too!
if ( encoremode )
mx + = SHORT ( AutomapPic - > leftoffset ) ;
else
mx - = SHORT ( AutomapPic - > leftoffset ) ;
my - = SHORT ( AutomapPic - > topoffset ) ;
// now that we have replicated this behavior, we can draw an icon from our supplied x, y coordinates by replicating k_kart.c's totally understandable uncommented code!!!
// get map boundaries using nodes
maxx = maxy = INT32_MAX ;
minx = miny = INT32_MIN ;
minx = bsp - > bbox [ 0 ] [ BOXLEFT ] ;
maxx = bsp - > bbox [ 0 ] [ BOXRIGHT ] ;
miny = bsp - > bbox [ 0 ] [ BOXBOTTOM ] ;
maxy = bsp - > bbox [ 0 ] [ BOXTOP ] ;
if ( bsp - > bbox [ 1 ] [ BOXLEFT ] < minx )
minx = bsp - > bbox [ 1 ] [ BOXLEFT ] ;
if ( bsp - > bbox [ 1 ] [ BOXRIGHT ] > maxx )
maxx = bsp - > bbox [ 1 ] [ BOXRIGHT ] ;
if ( bsp - > bbox [ 1 ] [ BOXBOTTOM ] < miny )
miny = bsp - > bbox [ 1 ] [ BOXBOTTOM ] ;
if ( bsp - > bbox [ 1 ] [ BOXTOP ] > maxy )
maxy = bsp - > bbox [ 1 ] [ BOXTOP ] ;
// You might be wondering why these are being bitshift here
// it's because mapwidth and height would otherwise overflow for maps larger than half the size possible...
// map boundaries and sizes will ALWAYS be whole numbers thankfully
// later calculations take into consideration that these are actually not in terms of FRACUNIT though
minx > > = FRACBITS ;
maxx > > = FRACBITS ;
miny > > = FRACBITS ;
maxy > > = FRACBITS ;
// these are our final map boundaries:
mapwidth = maxx - minx ;
mapheight = maxy - miny ;
// These should always be small enough to be bitshift back right now
xoffset = ( minx + mapwidth / 2 ) < < FRACBITS ;
yoffset = ( miny + mapheight / 2 ) < < FRACBITS ;
xscale = FixedDiv ( AutomapPic - > width , mapwidth ) ;
yscale = FixedDiv ( AutomapPic - > height , mapheight ) ;
zoom = FixedMul ( min ( xscale , yscale ) , FRACUNIT - FRACUNIT / 20 ) ;
amnumxpos = ( FixedMul ( x , zoom ) - FixedMul ( xoffset , zoom ) ) ;
amnumypos = - ( FixedMul ( y , zoom ) - FixedMul ( yoffset , zoom ) ) ;
if ( encoremode )
amnumxpos = - amnumxpos ;
// scale patch coords
patchw = patch - > width * scale / 2 ;
patchh = patch - > height * scale / 2 ;
if ( centered )
patchw = patchh = 0 ; // patch is supposedly already centered, don't butt in.
amxpos = amnumxpos + ( ( mx + AutomapPic - > width / 2 ) < < FRACBITS ) - patchw ;
amypos = amnumypos + ( ( my + AutomapPic - > height / 2 ) < < FRACBITS ) - patchh ;
// and NOW we can FINALLY DRAW OUR GOD DAMN PATCH :V
V_DrawFixedPatch ( amxpos , amypos , scale , splitflags , patch , colormap ) ;
return 0 ;
}
2014-03-21 18:42:55 +00:00
static int libd_drawNum ( lua_State * L )
{
INT32 x , y , flags , num ;
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-21 18:42:55 +00:00
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
num = luaL_checkinteger ( L , 3 ) ;
flags = luaL_optinteger ( L , 4 , 0 ) ;
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
V_DrawTallNum ( x , y , flags , num ) ;
return 0 ;
}
static int libd_drawPaddedNum ( lua_State * L )
{
INT32 x , y , flags , num , digits ;
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-21 18:42:55 +00:00
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
2016-03-30 04:22:12 +00:00
num = labs ( luaL_checkinteger ( L , 3 ) ) ;
2014-03-21 18:42:55 +00:00
digits = luaL_optinteger ( L , 4 , 2 ) ;
2014-03-15 16:59:03 +00:00
flags = luaL_optinteger ( L , 5 , 0 ) ;
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
2014-03-21 18:42:55 +00:00
V_DrawPaddedTallNum ( x , y , flags , num , digits ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2019-02-16 03:08:08 +00:00
static int libd_drawPingNum ( lua_State * L )
{
INT32 x , y , flags , num ;
const UINT8 * colormap = NULL ;
HUDONLY
x = luaL_checkinteger ( L , 1 ) ;
y = luaL_checkinteger ( L , 2 ) ;
num = luaL_checkinteger ( L , 3 ) ;
flags = luaL_optinteger ( L , 4 , 0 ) ;
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
if ( ! lua_isnoneornil ( L , 5 ) )
colormap = * ( ( UINT8 * * ) luaL_checkudata ( L , 5 , META_COLORMAP ) ) ;
V_DrawPingNum ( x , y , flags , num , colormap ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int libd_drawFill ( lua_State * L )
{
INT32 x = luaL_optinteger ( L , 1 , 0 ) ;
INT32 y = luaL_optinteger ( L , 2 , 0 ) ;
INT32 w = luaL_optinteger ( L , 3 , BASEVIDWIDTH ) ;
INT32 h = luaL_optinteger ( L , 4 , BASEVIDHEIGHT ) ;
INT32 c = luaL_optinteger ( L , 5 , 31 ) ;
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-15 16:59:03 +00:00
V_DrawFill ( x , y , w , h , c ) ;
return 0 ;
}
2018-06-26 14:14:59 +00:00
static int libd_fadeScreen ( lua_State * L )
{
UINT16 color = luaL_checkinteger ( L , 1 ) ;
UINT8 strength = luaL_checkinteger ( L , 2 ) ;
const UINT8 maxstrength = ( ( color & 0xFF00 ) ? 32 : 10 ) ;
HUDONLY
if ( ! strength )
return 0 ;
if ( strength > maxstrength )
return luaL_error ( L , " %s fade strength %d out of range (0 - %d) " , ((color & 0xFF00) ? " COLORMAP " : " TRANSMAP " ), strength, maxstrength) ;
if ( strength = = maxstrength ) // Allow as a shortcut for drawfill...
{
V_DrawFill ( 0 , 0 , BASEVIDWIDTH , BASEVIDHEIGHT , ( ( color & 0xFF00 ) ? 31 : color ) ) ;
return 0 ;
}
V_DrawFadeScreen ( color , strength ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int libd_drawString ( lua_State * L )
{
fixed_t x = luaL_checkinteger ( L , 1 ) ;
fixed_t y = luaL_checkinteger ( L , 2 ) ;
const char * str = luaL_checkstring ( L , 3 ) ;
INT32 flags = luaL_optinteger ( L , 4 , V_ALLOWLOWERCASE ) ;
enum align align = luaL_checkoption ( L , 5 , " left " , align_opt ) ;
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
2014-08-04 03:49:33 +00:00
HUDONLY
2014-03-15 16:59:03 +00:00
switch ( align )
{
2014-08-04 03:49:33 +00:00
// hu_font
2014-03-15 16:59:03 +00:00
case align_left :
V_DrawString ( x , y , flags , str ) ;
break ;
case align_center :
V_DrawCenteredString ( x , y , flags , str ) ;
break ;
case align_right :
V_DrawRightAlignedString ( x , y , flags , str ) ;
break ;
case align_fixed :
V_DrawStringAtFixed ( x , y , flags , str ) ;
break ;
2014-08-04 03:49:33 +00:00
// hu_font, 0.5x scale
case align_small :
V_DrawSmallString ( x , y , flags , str ) ;
break ;
case align_smallright :
V_DrawRightAlignedSmallString ( x , y , flags , str ) ;
break ;
// tny_font
case align_thin :
V_DrawThinString ( x , y , flags , str ) ;
break ;
case align_thinright :
V_DrawRightAlignedThinString ( x , y , flags , str ) ;
break ;
2014-03-15 16:59:03 +00:00
}
return 0 ;
}
2019-01-13 13:40:48 +00:00
static int libd_drawKartString ( lua_State * L )
{
fixed_t x = luaL_checkinteger ( L , 1 ) ;
fixed_t y = luaL_checkinteger ( L , 2 ) ;
const char * str = luaL_checkstring ( L , 3 ) ;
INT32 flags = luaL_optinteger ( L , 4 , V_ALLOWLOWERCASE ) ;
flags & = ~ V_PARAMMASK ; // Don't let crashes happen.
HUDONLY
V_DrawKartString ( x , y , flags , str ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int libd_stringWidth ( lua_State * L )
{
const char * str = luaL_checkstring ( L , 1 ) ;
INT32 flags = luaL_optinteger ( L , 2 , V_ALLOWLOWERCASE ) ;
2014-08-04 03:49:33 +00:00
enum widtht widtht = luaL_checkoption ( L , 3 , " normal " , widtht_opt ) ;
2014-03-15 16:59:03 +00:00
2014-08-04 03:49:33 +00:00
HUDONLY
switch ( widtht )
{
case widtht_normal : // hu_font
lua_pushinteger ( L , V_StringWidth ( str , flags ) ) ;
break ;
case widtht_small : // hu_font, 0.5x scale
lua_pushinteger ( L , V_SmallStringWidth ( str , flags ) ) ;
break ;
case widtht_thin : // tny_font
lua_pushinteger ( L , V_ThinStringWidth ( str , flags ) ) ;
break ;
}
2014-03-15 16:59:03 +00:00
return 1 ;
}
2014-11-12 00:55:07 +00:00
static int libd_getColormap ( lua_State * L )
{
INT32 skinnum = TC_DEFAULT ;
skincolors_t color = luaL_optinteger ( L , 2 , 0 ) ;
UINT8 * colormap = NULL ;
2016-01-31 22:52:02 +00:00
HUDONLY
2014-11-12 00:55:07 +00:00
if ( lua_isnoneornil ( L , 1 ) )
; // defaults to TC_DEFAULT
else if ( lua_type ( L , 1 ) = = LUA_TNUMBER ) // skin number
{
skinnum = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
2018-11-24 00:15:14 +00:00
if ( skinnum < TC_BLINK | | skinnum > = MAXSKINS )
return luaL_error ( L , " skin number %d is out of range (%d - %d) " , skinnum, TC_BLINK, MAXSKINS-1) ;
2014-11-12 00:55:07 +00:00
}
else // skin name
{
const char * skinname = luaL_checkstring ( L , 1 ) ;
INT32 i = R_SkinAvailable ( skinname ) ;
if ( i ! = - 1 ) // if -1, just default to TC_DEFAULT as above
skinnum = i ;
}
// all was successful above, now we generate the colormap at last!
colormap = R_GetTranslationColormap ( skinnum , color , GTC_CACHE ) ;
LUA_PushUserdata ( L , colormap , META_COLORMAP ) ; // push as META_COLORMAP userdata, specifically for patches to use!
return 1 ;
}
2016-01-31 21:53:14 +00:00
static int libd_width ( lua_State * L )
{
HUDONLY
lua_pushinteger ( L , vid . width ) ; // push screen width
return 1 ;
}
static int libd_height ( lua_State * L )
{
HUDONLY
lua_pushinteger ( L , vid . height ) ; // push screen height
return 1 ;
}
2016-05-03 05:25:00 +00:00
static int libd_dupx ( lua_State * L )
{
HUDONLY
lua_pushinteger ( L , vid . dupx ) ; // push integral scale (patch scale)
lua_pushfixed ( L , vid . fdupx ) ; // push fixed point scale (position scale)
return 2 ;
}
static int libd_dupy ( lua_State * L )
{
HUDONLY
lua_pushinteger ( L , vid . dupy ) ; // push integral scale (patch scale)
lua_pushfixed ( L , vid . fdupy ) ; // push fixed point scale (position scale)
return 2 ;
}
2016-01-31 21:53:14 +00:00
static int libd_renderer ( lua_State * L )
{
HUDONLY
2016-01-31 22:15:17 +00:00
switch ( rendermode ) {
case render_opengl : lua_pushliteral ( L , " opengl " ) ; break ; // OpenGL renderer
case render_soft : lua_pushliteral ( L , " software " ) ; break ; // Software renderer
default : lua_pushliteral ( L , " none " ) ; break ; // render_none (for dedicated), in case there's any reason this should be run
}
2016-01-31 21:53:14 +00:00
return 1 ;
}
2018-10-30 18:40:59 +00:00
// 30/10/18 Lat': Get cv_translucenthud's value for HUD rendering as a normal V_xxTRANS int
2018-10-30 17:24:03 +00:00
// Could as well be thrown in global vars for ease of access but I guess it makes sense for it to be a HUD fn
static int libd_getlocaltransflag ( lua_State * L )
{
HUDONLY
lua_pushinteger ( L , ( 10 - cv_translucenthud . value ) * V_10TRANS ) ; // A bit weird that it's called "translucenthud" yet 10 is fully opaque :V
return 1 ;
}
2014-03-15 16:59:03 +00:00
static luaL_Reg lib_draw [ ] = {
2014-03-23 16:00:29 +00:00
{ " patchExists " , libd_patchExists } ,
2014-03-15 16:59:03 +00:00
{ " cachePatch " , libd_cachePatch } ,
{ " draw " , libd_draw } ,
{ " drawScaled " , libd_drawScaled } ,
2014-03-21 18:42:55 +00:00
{ " drawNum " , libd_drawNum } ,
{ " drawPaddedNum " , libd_drawPaddedNum } ,
2019-02-16 03:08:08 +00:00
{ " drawPingNum " , libd_drawPingNum } ,
2014-03-15 16:59:03 +00:00
{ " drawFill " , libd_drawFill } ,
2018-06-26 14:14:59 +00:00
{ " fadeScreen " , libd_fadeScreen } ,
2014-03-15 16:59:03 +00:00
{ " drawString " , libd_drawString } ,
2019-01-13 13:40:48 +00:00
{ " drawKartString " , libd_drawKartString } ,
2014-03-15 16:59:03 +00:00
{ " stringWidth " , libd_stringWidth } ,
2014-11-12 00:55:07 +00:00
{ " getColormap " , libd_getColormap } ,
2016-01-31 21:53:14 +00:00
{ " width " , libd_width } ,
{ " height " , libd_height } ,
2016-05-03 05:25:00 +00:00
{ " dupx " , libd_dupx } ,
{ " dupy " , libd_dupy } ,
2016-01-31 21:53:14 +00:00
{ " renderer " , libd_renderer } ,
2018-10-30 17:24:03 +00:00
{ " localTransFlag " , libd_getlocaltransflag } ,
2019-05-26 13:05:10 +00:00
{ " drawOnMinimap " , libd_drawOnMinimap } ,
2014-03-15 16:59:03 +00:00
{ NULL , NULL }
} ;
//
// lib_hud
//
// enable vanilla HUD element
static int lib_hudenable ( lua_State * L )
{
enum hud option = luaL_checkoption ( L , 1 , NULL , hud_disable_options ) ;
hud_enabled [ option / 8 ] | = 1 < < ( option % 8 ) ;
return 0 ;
}
// disable vanilla HUD element
static int lib_huddisable ( lua_State * L )
{
enum hud option = luaL_checkoption ( L , 1 , NULL , hud_disable_options ) ;
hud_enabled [ option / 8 ] & = ~ ( 1 < < ( option % 8 ) ) ;
return 0 ;
}
2018-10-30 18:40:59 +00:00
// 30/10/18: Lat': How come this wasn't here before?
2018-10-30 22:08:58 +00:00
static int lib_hudenabled ( lua_State * L )
2018-10-30 17:24:03 +00:00
{
enum hud option = luaL_checkoption ( L , 1 , NULL , hud_disable_options ) ;
2018-10-30 22:08:58 +00:00
if ( hud_enabled [ option / 8 ] & ( 1 < < ( option % 8 ) ) )
2018-10-30 17:24:03 +00:00
lua_pushboolean ( L , true ) ;
else
lua_pushboolean ( L , false ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
// add a HUD element for rendering
static int lib_hudadd ( lua_State * L )
{
enum hudhook field ;
luaL_checktype ( L , 1 , LUA_TFUNCTION ) ;
field = luaL_checkoption ( L , 2 , " game " , hudhook_opt ) ;
lua_getfield ( L , LUA_REGISTRYINDEX , " HUD " ) ;
I_Assert ( lua_istable ( L , - 1 ) ) ;
lua_rawgeti ( L , - 1 , field + 2 ) ; // HUD[2+]
I_Assert ( lua_istable ( L , - 1 ) ) ;
lua_remove ( L , - 2 ) ;
lua_pushvalue ( L , 1 ) ;
lua_rawseti ( L , - 2 , ( int ) ( lua_objlen ( L , - 2 ) + 1 ) ) ;
2014-04-14 05:14:58 +00:00
hudAvailable | = 1 < < field ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static luaL_Reg lib_hud [ ] = {
{ " enable " , lib_hudenable } ,
{ " disable " , lib_huddisable } ,
2018-10-30 22:10:56 +00:00
{ " enabled " , lib_hudenabled } ,
2014-03-15 16:59:03 +00:00
{ " add " , lib_hudadd } ,
{ NULL , NULL }
} ;
//
//
//
int LUA_HudLib ( lua_State * L )
{
memset ( hud_enabled , 0xff , ( hud_MAX / 8 ) + 1 ) ;
lua_newtable ( L ) ; // HUD registry table
lua_newtable ( L ) ;
luaL_register ( L , NULL , lib_draw ) ;
lua_rawseti ( L , - 2 , 1 ) ; // HUD[1] = lib_draw
lua_newtable ( L ) ;
lua_rawseti ( L , - 2 , 2 ) ; // HUD[2] = game rendering functions array
lua_newtable ( L ) ;
lua_rawseti ( L , - 2 , 3 ) ; // HUD[2] = scores rendering functions array
lua_setfield ( L , LUA_REGISTRYINDEX , " HUD " ) ;
luaL_newmetatable ( L , META_HUDINFO ) ;
lua_pushcfunction ( L , hudinfo_get ) ;
lua_setfield ( L , - 2 , " __index " ) ;
lua_pushcfunction ( L , hudinfo_set ) ;
lua_setfield ( L , - 2 , " __newindex " ) ;
lua_pushcfunction ( L , hudinfo_num ) ;
lua_setfield ( L , - 2 , " __len " ) ;
lua_pop ( L , 1 ) ;
lua_newuserdata ( L , 0 ) ;
lua_createtable ( L , 0 , 2 ) ;
lua_pushcfunction ( L , lib_getHudInfo ) ;
lua_setfield ( L , - 2 , " __index " ) ;
lua_pushcfunction ( L , lib_hudinfolen ) ;
lua_setfield ( L , - 2 , " __len " ) ;
lua_setmetatable ( L , - 2 ) ;
lua_setglobal ( L , " hudinfo " ) ;
2014-11-12 00:55:07 +00:00
luaL_newmetatable ( L , META_COLORMAP ) ;
lua_pushcfunction ( L , colormap_get ) ;
lua_setfield ( L , - 2 , " __index " ) ;
lua_pop ( L , 1 ) ;
2014-03-15 16:59:03 +00:00
luaL_newmetatable ( L , META_PATCH ) ;
lua_pushcfunction ( L , patch_get ) ;
lua_setfield ( L , - 2 , " __index " ) ;
lua_pushcfunction ( L , patch_set ) ;
lua_setfield ( L , - 2 , " __newindex " ) ;
lua_pop ( L , 1 ) ;
2014-08-04 03:49:33 +00:00
luaL_newmetatable ( L , META_CAMERA ) ;
lua_pushcfunction ( L , camera_get ) ;
lua_setfield ( L , - 2 , " __index " ) ;
lua_pop ( L , 1 ) ;
2014-03-15 16:59:03 +00:00
luaL_register ( L , " hud " , lib_hud ) ;
return 0 ;
}
boolean LUA_HudEnabled ( enum hud option )
{
if ( ! gL | | hud_enabled [ option / 8 ] & ( 1 < < ( option % 8 ) ) )
return true ;
return false ;
}
// Hook for HUD rendering
2014-08-04 03:49:33 +00:00
void LUAh_GameHUD ( player_t * stplayr )
2014-03-15 16:59:03 +00:00
{
2014-04-14 05:14:58 +00:00
if ( ! gL | | ! ( hudAvailable & ( 1 < < hudhook_game ) ) )
2014-03-15 16:59:03 +00:00
return ;
hud_running = true ;
lua_pop ( gL , - 1 ) ;
lua_getfield ( gL , LUA_REGISTRYINDEX , " HUD " ) ;
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_rawgeti ( gL , - 1 , 2 ) ; // HUD[2] = rendering funcs
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_rawgeti ( gL , - 2 , 1 ) ; // HUD[1] = lib_draw
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_remove ( gL , - 3 ) ; // pop HUD
2014-08-04 03:49:33 +00:00
LUA_PushUserdata ( gL , stplayr , META_PLAYER ) ;
2019-04-18 07:08:11 +00:00
if ( splitscreen > 2 & & stplayr = = & players [ displayplayers [ 3 ] ] )
2019-01-28 17:28:15 +00:00
{
2019-04-18 07:08:11 +00:00
LUA_PushUserdata ( gL , & camera [ 3 ] , META_CAMERA ) ;
2019-01-28 17:28:15 +00:00
camnum = 4 ;
}
2019-04-18 07:08:11 +00:00
else if ( splitscreen > 1 & & stplayr = = & players [ displayplayers [ 2 ] ] )
2019-01-28 17:28:15 +00:00
{
2019-04-18 07:08:11 +00:00
LUA_PushUserdata ( gL , & camera [ 2 ] , META_CAMERA ) ;
2019-01-28 17:28:15 +00:00
camnum = 3 ;
}
2019-04-18 07:08:11 +00:00
else if ( splitscreen & & stplayr = = & players [ displayplayers [ 1 ] ] )
2019-01-28 17:28:15 +00:00
{
2019-04-18 07:08:11 +00:00
LUA_PushUserdata ( gL , & camera [ 1 ] , META_CAMERA ) ;
2019-01-28 17:28:15 +00:00
camnum = 2 ;
}
2014-08-04 03:49:33 +00:00
else
2019-01-28 17:28:15 +00:00
{
2019-04-18 07:08:11 +00:00
LUA_PushUserdata ( gL , & camera [ 0 ] , META_CAMERA ) ;
2019-01-28 17:28:15 +00:00
camnum = 1 ;
}
2014-03-15 16:59:03 +00:00
lua_pushnil ( gL ) ;
2014-08-04 03:49:33 +00:00
while ( lua_next ( gL , - 5 ) ! = 0 ) {
lua_pushvalue ( gL , - 5 ) ; // graphics library (HUD[1])
lua_pushvalue ( gL , - 5 ) ; // stplayr
lua_pushvalue ( gL , - 5 ) ; // camera
LUA_Call ( gL , 3 ) ;
2014-03-15 16:59:03 +00:00
}
lua_pop ( gL , - 1 ) ;
hud_running = false ;
}
void LUAh_ScoresHUD ( void )
{
2014-04-14 05:14:58 +00:00
if ( ! gL | | ! ( hudAvailable & ( 1 < < hudhook_scores ) ) )
2014-03-15 16:59:03 +00:00
return ;
hud_running = true ;
lua_pop ( gL , - 1 ) ;
lua_getfield ( gL , LUA_REGISTRYINDEX , " HUD " ) ;
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_rawgeti ( gL , - 1 , 3 ) ; // HUD[3] = rendering funcs
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_rawgeti ( gL , - 2 , 1 ) ; // HUD[1] = lib_draw
I_Assert ( lua_istable ( gL , - 1 ) ) ;
lua_remove ( gL , - 3 ) ; // pop HUD
lua_pushnil ( gL ) ;
while ( lua_next ( gL , - 3 ) ! = 0 ) {
lua_pushvalue ( gL , - 3 ) ; // graphics library (HUD[1])
LUA_Call ( gL , 1 ) ;
}
lua_pop ( gL , - 1 ) ;
hud_running = false ;
}
# endif