2014-03-15 16:59:03 +00:00
// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
2016-05-18 00:42:11 +00:00
// Copyright (C) 2012-2016 by John "JTE" Muniz.
2023-03-31 12:53:31 +00:00
// Copyright (C) 2012-2023 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_baselib.c
/// \brief basic functions for Lua scripting
# include "doomdef.h"
2019-12-19 05:13:09 +00:00
# include "fastcmp.h"
2014-03-15 16:59:03 +00:00
# include "p_local.h"
# include "p_setup.h" // So we can have P_SetupLevelSky
2020-05-18 13:23:56 +00:00
# include "p_slopes.h" // P_GetSlopeZAt
2014-03-15 16:59:03 +00:00
# include "z_zone.h"
# include "r_main.h"
2020-02-15 22:35:00 +00:00
# include "r_draw.h"
2020-03-09 13:54:56 +00:00
# include "r_things.h" // R_Frame2Char etc
2014-03-15 16:59:03 +00:00
# include "m_random.h"
# include "s_sound.h"
# include "g_game.h"
2019-12-19 05:20:49 +00:00
# include "m_menu.h"
2019-12-19 05:13:09 +00:00
# include "y_inter.h"
2018-07-31 09:10:02 +00:00
# include "hu_stuff.h" // HU_AddChatText
2018-11-07 23:07:34 +00:00
# include "console.h"
2022-12-31 13:10:19 +00:00
# include "netcode/d_netcmd.h" // IsPlayerAdmin
2020-02-15 08:18:41 +00:00
# include "m_menu.h" // Player Setup menu color stuff
2021-05-27 22:35:15 +00:00
# include "m_misc.h" // M_MapNumber
2021-01-23 15:58:34 +00:00
# include "b_bot.h" // B_UpdateBotleader
2022-12-31 13:10:19 +00:00
# include "netcode/d_clisrv.h" // CL_RemovePlayer
2022-05-01 05:32:46 +00:00
# include "i_system.h" // I_GetPreciseTime, I_GetPrecisePrecision
2014-03-15 16:59:03 +00:00
# include "lua_script.h"
# include "lua_libs.h"
# include "lua_hud.h" // hud_running errors
2020-07-29 15:26:43 +00:00
# include "taglist.h" // P_FindSpecialLineFromTag
2020-07-17 05:08:38 +00:00
# include "lua_hook.h" // hook_cmd_running errors
2014-03-15 16:59:03 +00:00
2017-01-18 22:02:28 +00:00
# define NOHUD if (hud_running)\
2020-07-17 05:08:38 +00:00
return luaL_error ( L , " HUD rendering code should not call this function! " ) ; \
else if ( hook_cmd_running ) \
return luaL_error ( L , " CMD building code should not call this function! " ) ;
2014-03-15 16:59:03 +00:00
2024-02-11 12:30:15 +00:00
# define NOSPAWNNULL if (type >= NUMMOBJTYPES)\
return luaL_error ( L , " mobj type %d out of range (0 - %d) " , type, NUMMOBJTYPES-1) ; \
else if ( type = = MT_NULL ) \
{ \
if ( ! nospawnnull_seen ) { \
nospawnnull_seen = true ; \
CONS_Alert ( CONS_WARNING , " Spawning an \" MT_NULL \" mobj is deprecated and will be removed. \n Use \" MT_RAY \" instead. \n " ) ; \
} \
type = MT_RAY ; \
}
static boolean nospawnnull_seen = false ; // TODO: 2.3: Delete
// TODO: 2.3: Use the below NOSPAWNNULL define instead. P_SpawnMobj used to say "if MT_NULL, use MT_RAY instead", so the above define maintains Lua script compatibility until v2.3
/*#define NOSPAWNNULL if (type <= MT_NULL || type >= NUMMOBJTYPES)\
return luaL_error ( L , " mobj type %d out of range (1 - %d) " , type, NUMMOBJTYPES-1) ; */
2014-03-15 16:59:03 +00:00
boolean luaL_checkboolean ( lua_State * L , int narg ) {
luaL_checktype ( L , narg , LUA_TBOOLEAN ) ;
return lua_toboolean ( L , narg ) ;
}
// String concatination
static int lib_concat ( lua_State * L )
{
int n = lua_gettop ( L ) ; /* number of arguments */
int i ;
char * r = NULL ;
size_t rl = 0 , sl ;
lua_getglobal ( L , " tostring " ) ;
for ( i = 1 ; i < = n ; i + + ) {
const char * s ;
lua_pushvalue ( L , - 1 ) ; /* function to be called */
lua_pushvalue ( L , i ) ; /* value to print */
lua_call ( L , 1 , 1 ) ;
s = lua_tolstring ( L , - 1 , & sl ) ; /* get result */
if ( s = = NULL )
return luaL_error ( L , LUA_QL ( " tostring " ) " must return a string to "
LUA_QL ( " __add " ) ) ;
r = Z_Realloc ( r , rl + sl , PU_STATIC , NULL ) ;
M_Memcpy ( r + rl , s , sl ) ;
rl + = sl ;
lua_pop ( L , 1 ) ; /* pop result */
}
lua_pushlstring ( L , r , rl ) ;
Z_Free ( r ) ;
return 1 ;
}
// Wrapper for CONS_Printf
// Copied from base Lua code
static int lib_print ( lua_State * L )
{
int n = lua_gettop ( L ) ; /* number of arguments */
int i ;
//HUDSAFE
lua_getglobal ( L , " tostring " ) ;
for ( i = 1 ; i < = n ; i + + ) {
const char * s ;
lua_pushvalue ( L , - 1 ) ; /* function to be called */
lua_pushvalue ( L , i ) ; /* value to print */
lua_call ( L , 1 , 1 ) ;
s = lua_tostring ( L , - 1 ) ; /* get result */
if ( s = = NULL )
return luaL_error ( L , LUA_QL ( " tostring " ) " must return a string to "
LUA_QL ( " print " ) ) ;
if ( i > 1 ) CONS_Printf ( " \n " ) ;
CONS_Printf ( " %s " , s ) ;
lua_pop ( L , 1 ) ; /* pop result */
}
CONS_Printf ( " \n " ) ;
return 0 ;
}
2018-07-31 09:10:02 +00:00
// Print stuff in the chat, or in the console if we can't.
static int lib_chatprint ( lua_State * L )
{
const char * str = luaL_checkstring ( L , 1 ) ; // retrieve string
2018-12-22 11:34:17 +00:00
boolean sound = lua_optboolean ( L , 2 ) ; // retrieve sound boolean
2018-12-18 00:02:22 +00:00
int len = strlen ( str ) ;
2018-07-31 09:10:02 +00:00
if ( str = = NULL ) // error if we don't have a string!
return luaL_error ( L , LUA_QL ( " tostring " ) " must return a string to " LUA_QL ( " chatprint " ) ) ;
2018-12-18 00:02:22 +00:00
2018-07-31 09:10:02 +00:00
if ( len > 255 ) // string is too long!!!
return luaL_error ( L , " String exceeds the 255 characters limit of the chat buffer. " ) ;
2018-12-17 19:43:59 +00:00
HU_AddChatText ( str , sound ) ;
2018-07-31 09:10:02 +00:00
return 0 ;
}
// Same as above, but do it for only one player.
static int lib_chatprintf ( lua_State * L )
{
int n = lua_gettop ( L ) ; /* number of arguments */
2018-12-18 00:02:22 +00:00
const char * str = luaL_checkstring ( L , 2 ) ; // retrieve string
2018-12-22 11:34:17 +00:00
boolean sound = lua_optboolean ( L , 3 ) ; // sound?
2018-12-18 00:02:22 +00:00
int len = strlen ( str ) ;
2018-07-31 09:10:02 +00:00
player_t * plr ;
2018-12-18 00:02:22 +00:00
2018-07-31 09:10:02 +00:00
if ( n < 2 )
return luaL_error ( L , " chatprintf requires at least two arguments: player and text. " ) ;
2018-12-17 19:43:59 +00:00
2018-07-31 09:10:02 +00:00
plr = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ; // retrieve player
if ( ! plr )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( plr ! = & players [ consoleplayer ] )
return 0 ;
2018-12-17 19:43:59 +00:00
2018-07-31 09:10:02 +00:00
if ( str = = NULL ) // error if we don't have a string!
return luaL_error ( L , LUA_QL ( " tostring " ) " must return a string to " LUA_QL ( " chatprintf " ) ) ;
2018-12-18 00:02:22 +00:00
2018-07-31 09:10:02 +00:00
if ( len > 255 ) // string is too long!!!
return luaL_error ( L , " String exceeds the 255 characters limit of the chat buffer. " ) ;
2018-12-17 19:43:59 +00:00
HU_AddChatText ( str , sound ) ;
2018-07-31 09:10:02 +00:00
return 0 ;
}
2017-01-19 18:30:55 +00:00
static const struct {
const char * meta ;
2017-01-19 21:35:53 +00:00
const char * utype ;
2017-01-19 18:30:55 +00:00
} meta2utype [ ] = {
2017-01-19 21:35:53 +00:00
{ META_STATE , " state_t " } ,
{ META_MOBJINFO , " mobjinfo_t " } ,
{ META_SFXINFO , " sfxinfo_t " } ,
2020-02-23 17:17:52 +00:00
{ META_SKINCOLOR , " skincolor_t " } ,
2020-02-15 08:18:41 +00:00
{ META_COLORRAMP , " skincolor_t.ramp " } ,
2019-11-07 06:11:16 +00:00
{ META_SPRITEINFO , " spriteinfo_t " } ,
2019-11-08 02:42:14 +00:00
{ META_PIVOTLIST , " spriteframepivot_t[] " } ,
{ META_FRAMEPIVOT , " spriteframepivot_t " } ,
2017-01-19 21:35:53 +00:00
2020-12-05 10:02:06 +00:00
{ META_TAGLIST , " taglist " } ,
2017-01-19 21:35:53 +00:00
{ META_MOBJ , " mobj_t " } ,
{ META_MAPTHING , " mapthing_t " } ,
{ META_PLAYER , " player_t " } ,
{ META_TICCMD , " ticcmd_t " } ,
{ META_SKIN , " skin_t " } ,
{ META_POWERS , " player_t.powers " } ,
{ META_SOUNDSID , " skin_t.soundsid " } ,
2024-06-03 00:40:14 +00:00
{ META_SKINSPRITES , " skin_t.skinsprites " } ,
{ META_SKINSPRITESLIST , " skin_t.skinsprites[] " } ,
{ META_SKINSPRITESCOMPAT , " skin_t.sprites " } , // TODO: 2.3: Delete
2017-01-19 21:35:53 +00:00
{ META_VERTEX , " vertex_t " } ,
{ META_LINE , " line_t " } ,
{ META_SIDE , " side_t " } ,
{ META_SUBSECTOR , " subsector_t " } ,
{ META_SECTOR , " sector_t " } ,
{ META_FFLOOR , " ffloor_t " } ,
2017-01-19 18:30:55 +00:00
# ifdef HAVE_LUA_SEGS
2017-01-19 21:35:53 +00:00
{ META_SEG , " seg_t " } ,
{ META_NODE , " node_t " } ,
2017-01-19 18:30:55 +00:00
# endif
2020-09-09 20:24:07 +00:00
{ META_SLOPE , " slope_t " } ,
{ META_VECTOR2 , " vector2_t " } ,
{ META_VECTOR3 , " vector3_t " } ,
2017-01-19 21:35:53 +00:00
{ META_MAPHEADER , " mapheader_t " } ,
2017-01-19 18:30:55 +00:00
2020-09-09 20:24:07 +00:00
{ META_POLYOBJ , " polyobj_t " } ,
2021-10-22 23:31:37 +00:00
{ META_POLYOBJVERTICES , " polyobj_t.vertices " } ,
{ META_POLYOBJLINES , " polyobj_t.lines " } ,
2020-09-09 20:24:07 +00:00
2017-01-19 21:35:53 +00:00
{ META_CVAR , " consvar_t " } ,
2017-01-19 18:30:55 +00:00
2017-01-19 21:35:53 +00:00
{ META_SECTORLINES , " sector_t.lines " } ,
2020-12-05 10:02:06 +00:00
# ifdef MUTABLE_TAGS
2020-12-04 21:47:22 +00:00
{ META_SECTORTAGLIST , " sector_t.taglist " } ,
2020-12-05 10:02:06 +00:00
# endif
2017-01-19 21:35:53 +00:00
{ META_SIDENUM , " line_t.sidenum " } ,
2020-01-02 11:23:14 +00:00
{ META_LINEARGS , " line_t.args " } ,
2020-01-08 07:42:35 +00:00
{ META_LINESTRINGARGS , " line_t.stringargs " } ,
2020-04-11 10:54:34 +00:00
{ META_THINGARGS , " mapthing.args " } ,
{ META_THINGSTRINGARGS , " mapthing.stringargs " } ,
2017-01-19 18:30:55 +00:00
# ifdef HAVE_LUA_SEGS
2017-01-19 21:35:53 +00:00
{ META_NODEBBOX , " node_t.bbox " } ,
{ META_NODECHILDREN , " node_t.children " } ,
2017-01-19 18:30:55 +00:00
# endif
2017-01-19 21:35:53 +00:00
{ META_BBOX , " bbox " } ,
2017-01-19 18:30:55 +00:00
2017-01-19 21:35:53 +00:00
{ META_HUDINFO , " hudinfo_t " } ,
{ META_PATCH , " patch_t " } ,
{ META_COLORMAP , " colormap " } ,
2023-08-04 03:31:51 +00:00
{ META_EXTRACOLORMAP , " extracolormap_t " } ,
2017-01-19 21:35:53 +00:00
{ META_CAMERA , " camera_t " } ,
2017-05-17 19:36:40 +00:00
{ META_ACTION , " action " } ,
2019-08-24 17:25:27 +00:00
{ META_LUABANKS , " luabanks[] " } ,
2021-03-26 03:21:53 +00:00
2021-10-22 23:31:37 +00:00
{ META_KEYEVENT , " keyevent_t " } ,
2021-03-24 08:54:11 +00:00
{ META_MOUSE , " mouse_t " } ,
2017-01-19 21:35:53 +00:00
{ NULL , NULL }
2017-01-19 18:30:55 +00:00
} ;
2017-01-19 21:35:53 +00:00
// goes through the above list and returns the utype string for the userdata type
2017-09-28 15:10:24 +00:00
// returns "unknown" instead if we couldn't find the right userdata type
2017-01-19 21:35:53 +00:00
static const char * GetUserdataUType ( lua_State * L )
2017-01-19 18:30:55 +00:00
{
UINT8 i ;
lua_getmetatable ( L , - 1 ) ;
for ( i = 0 ; meta2utype [ i ] . meta ; i + + )
{
luaL_getmetatable ( L , meta2utype [ i ] . meta ) ;
if ( lua_rawequal ( L , - 1 , - 2 ) )
{
lua_pop ( L , 2 ) ;
return meta2utype [ i ] . utype ;
}
lua_pop ( L , 1 ) ;
}
lua_pop ( L , 1 ) ;
2017-01-19 21:35:53 +00:00
return " unknown " ;
2017-01-19 18:30:55 +00:00
}
2017-01-19 21:35:53 +00:00
// Return a string representing the type of userdata the given var is
// e.g. players[0] -> "player_t"
// or players[0].powers -> "player_t.powers"
2017-01-19 18:30:55 +00:00
static int lib_userdataType ( lua_State * L )
{
lua_settop ( L , 1 ) ; // pop everything except arg 1 (in case somebody decided to add more)
2021-05-03 04:32:07 +00:00
luaL_checktype ( L , 1 , LUA_TUSERDATA ) ;
lua_pushstring ( L , GetUserdataUType ( L ) ) ;
return 1 ;
2017-01-19 18:30:55 +00:00
}
2020-10-03 14:31:04 +00:00
// Takes a metatable as first and only argument
// Only callable during script loading
2020-10-03 16:40:37 +00:00
static int lib_registerMetatable ( lua_State * L )
2020-10-03 14:31:04 +00:00
{
2020-11-08 16:20:25 +00:00
static UINT16 nextid = 1 ;
2020-10-03 14:31:04 +00:00
if ( ! lua_lumploading )
return luaL_error ( L , " This function cannot be called from within a hook or coroutine! " ) ;
luaL_checktype ( L , 1 , LUA_TTABLE ) ;
2020-11-08 16:20:25 +00:00
if ( nextid = = 0 )
2020-11-08 23:16:40 +00:00
return luaL_error ( L , " Too many metatables registered?! Please consider rewriting your script once you are sober again. \n " ) ;
2020-11-08 16:20:25 +00:00
2020-10-03 14:31:04 +00:00
lua_getfield ( L , LUA_REGISTRYINDEX , LREG_METATABLES ) ; // 2
// registry.metatables[metatable] = nextid
lua_pushvalue ( L , 1 ) ; // 3
lua_pushnumber ( L , nextid ) ; // 4
lua_settable ( L , 2 ) ;
// registry.metatables[nextid] = metatable
lua_pushnumber ( L , nextid ) ; // 3
lua_pushvalue ( L , 1 ) ; // 4
lua_settable ( L , 2 ) ;
lua_pop ( L , 1 ) ;
nextid + + ;
return 0 ;
}
2020-10-03 16:40:37 +00:00
// Takes a string as only argument and returns the metatable
// associated to the userdata type this string refers to
// Returns nil if the string does not refer to a valid userdata type
static int lib_userdataMetatable ( lua_State * L )
{
2020-10-30 14:00:13 +00:00
UINT32 i ;
2020-10-03 16:40:37 +00:00
const char * udname = luaL_checkstring ( L , 1 ) ;
2020-10-30 14:00:13 +00:00
// Find internal metatable name
for ( i = 0 ; meta2utype [ i ] . meta ; i + + )
if ( ! strcmp ( udname , meta2utype [ i ] . utype ) )
{
luaL_getmetatable ( L , meta2utype [ i ] . meta ) ;
return 1 ;
}
lua_pushnil ( L ) ;
2020-10-03 16:40:37 +00:00
return 1 ;
}
2018-11-07 23:07:34 +00:00
static int lib_isPlayerAdmin ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//HUDSAFE
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , IsPlayerAdmin ( player - players ) ) ;
return 1 ;
}
2019-08-24 17:25:27 +00:00
static int lib_reserveLuabanks ( lua_State * L )
{
static boolean reserved = false ;
if ( ! lua_lumploading )
return luaL_error ( L , " luabanks[] cannot be reserved from within a hook or coroutine! " ) ;
if ( reserved )
return luaL_error ( L , " luabanks[] has already been reserved! Only one savedata-enabled mod at a time may use this feature. " ) ;
reserved = true ;
LUA_PushUserdata ( L , & luabanks , META_LUABANKS ) ;
return 1 ;
}
2024-05-01 15:26:02 +00:00
static int lib_tofixed ( lua_State * L )
{
const char * arg = luaL_checkstring ( L , 1 ) ;
char * end ;
float f = strtof ( arg , & end ) ;
if ( * end ! = ' \0 ' )
lua_pushnil ( L ) ;
else
lua_pushnumber ( L , FLOAT_TO_FIXED ( f ) ) ;
return 1 ;
}
2020-02-15 08:18:41 +00:00
// M_MENU
//////////////
static int lib_pMoveColorBefore ( lua_State * L )
{
UINT16 color = ( UINT16 ) luaL_checkinteger ( L , 1 ) ;
UINT16 targ = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
M_MoveColorBefore ( color , targ ) ;
return 0 ;
}
static int lib_pMoveColorAfter ( lua_State * L )
{
UINT16 color = ( UINT16 ) luaL_checkinteger ( L , 1 ) ;
UINT16 targ = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
M_MoveColorAfter ( color , targ ) ;
return 0 ;
}
static int lib_pGetColorBefore ( lua_State * L )
{
UINT16 color = ( UINT16 ) luaL_checkinteger ( L , 1 ) ;
lua_pushinteger ( L , M_GetColorBefore ( color ) ) ;
return 1 ;
}
static int lib_pGetColorAfter ( lua_State * L )
{
UINT16 color = ( UINT16 ) luaL_checkinteger ( L , 1 ) ;
lua_pushinteger ( L , M_GetColorAfter ( color ) ) ;
return 1 ;
}
2021-05-27 22:35:15 +00:00
// M_MISC
//////////////
static int lib_mMapNumber ( lua_State * L )
{
const char * arg = luaL_checkstring ( L , 1 ) ;
size_t len = strlen ( arg ) ;
if ( len = = 2 | | len = = 5 ) {
char first = arg [ len - 2 ] ;
char second = arg [ len - 1 ] ;
lua_pushinteger ( L , M_MapNumber ( first , second ) ) ;
} else {
lua_pushinteger ( L , 0 ) ;
}
return 1 ;
}
2014-03-15 16:59:03 +00:00
// M_RANDOM
//////////////
2016-03-27 14:33:15 +00:00
static int lib_pRandomFixed ( lua_State * L )
2014-03-15 16:59:03 +00:00
{
NOHUD
2016-03-27 14:33:15 +00:00
lua_pushfixed ( L , P_RandomFixed ( ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
2016-03-27 14:33:15 +00:00
static int lib_pRandomByte ( lua_State * L )
2014-03-15 16:59:03 +00:00
{
NOHUD
2016-03-27 14:33:15 +00:00
lua_pushinteger ( L , P_RandomByte ( ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pRandomKey ( lua_State * L )
{
INT32 a = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
NOHUD
2016-05-11 21:33:50 +00:00
if ( a > 65536 )
LUA_UsageWarning ( L , " P_RandomKey: range > 65536 is undefined behavior " ) ;
2014-03-15 16:59:03 +00:00
lua_pushinteger ( L , P_RandomKey ( a ) ) ;
return 1 ;
}
static int lib_pRandomRange ( lua_State * L )
{
INT32 a = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
INT32 b = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
if ( b < a ) {
INT32 c = a ;
a = b ;
b = c ;
}
2016-05-11 21:33:50 +00:00
if ( ( b - a + 1 ) > 65536 )
LUA_UsageWarning ( L , " P_RandomRange: range > 65536 is undefined behavior " ) ;
2014-03-15 16:59:03 +00:00
lua_pushinteger ( L , P_RandomRange ( a , b ) ) ;
return 1 ;
}
2018-03-20 15:00:27 +00:00
// Macros.
2016-03-27 14:33:15 +00:00
static int lib_pSignedRandom ( lua_State * L )
{
NOHUD
lua_pushinteger ( L , P_SignedRandom ( ) ) ;
return 1 ;
}
static int lib_pRandomChance ( lua_State * L )
{
fixed_t p = luaL_checkfixed ( L , 1 ) ;
NOHUD
lua_pushboolean ( L , P_RandomChance ( p ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
// P_MAPUTIL
///////////////
static int lib_pAproxDistance ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t dx = luaL_checkfixed ( L , 1 ) ;
fixed_t dy = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2021-06-03 04:36:29 +00:00
lua_pushfixed ( L , P_AproxDistance ( dx , dy ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pClosestPointOnLine ( lua_State * L )
{
2016-03-02 20:31:04 +00:00
int n = lua_gettop ( L ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
vertex_t result ;
//HUDSAFE
2016-03-02 20:31:04 +00:00
if ( lua_isuserdata ( L , 3 ) ) // use a real linedef to get our points
{
line_t * line = * ( ( line_t * * ) luaL_checkudata ( L , 3 , META_LINE ) ) ;
if ( ! line )
return LUA_ErrInvalid ( L , " line_t " ) ;
P_ClosestPointOnLine ( x , y , line , & result ) ;
}
else // use custom coordinates of our own!
{
vertex_t v1 , v2 ; // fake vertexes
line_t junk ; // fake linedef
if ( n < 6 )
return luaL_error ( L , " arguments 3 to 6 not all given (expected 4 fixed-point integers) " ) ;
v1 . x = luaL_checkfixed ( L , 3 ) ;
v1 . y = luaL_checkfixed ( L , 4 ) ;
v2 . x = luaL_checkfixed ( L , 5 ) ;
v2 . y = luaL_checkfixed ( L , 6 ) ;
junk . v1 = & v1 ;
junk . v2 = & v2 ;
junk . dx = v2 . x - v1 . x ;
junk . dy = v2 . y - v1 . y ;
P_ClosestPointOnLine ( x , y , & junk , & result ) ;
}
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , result . x ) ;
lua_pushfixed ( L , result . y ) ;
2014-03-15 16:59:03 +00:00
return 2 ;
}
2016-05-25 15:34:09 +00:00
static int lib_pPointOnLineSide ( lua_State * L )
{
2016-11-24 22:01:51 +00:00
int n = lua_gettop ( L ) ;
2016-05-25 15:34:09 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
//HUDSAFE
2016-11-24 22:01:51 +00:00
if ( lua_isuserdata ( L , 3 ) ) // use a real linedef to get our points
{
line_t * line = * ( ( line_t * * ) luaL_checkudata ( L , 3 , META_LINE ) ) ;
if ( ! line )
return LUA_ErrInvalid ( L , " line_t " ) ;
lua_pushinteger ( L , P_PointOnLineSide ( x , y , line ) ) ;
}
else // use custom coordinates of our own!
{
vertex_t v1 , v2 ; // fake vertexes
line_t junk ; // fake linedef
if ( n < 6 )
return luaL_error ( L , " arguments 3 to 6 not all given (expected 4 fixed-point integers) " ) ;
v1 . x = luaL_checkfixed ( L , 3 ) ;
v1 . y = luaL_checkfixed ( L , 4 ) ;
v2 . x = luaL_checkfixed ( L , 5 ) ;
v2 . y = luaL_checkfixed ( L , 6 ) ;
junk . v1 = & v1 ;
junk . v2 = & v2 ;
junk . dx = v2 . x - v1 . x ;
junk . dy = v2 . y - v1 . y ;
lua_pushinteger ( L , P_PointOnLineSide ( x , y , & junk ) ) ;
}
2016-05-25 15:34:09 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
// P_ENEMY
/////////////
static int lib_pCheckMeleeRange ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_CheckMeleeRange ( actor ) ) ;
return 1 ;
}
static int lib_pJetbCheckMeleeRange ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_JetbCheckMeleeRange ( actor ) ) ;
return 1 ;
}
static int lib_pFaceStabCheckMeleeRange ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_FaceStabCheckMeleeRange ( actor ) ) ;
return 1 ;
}
static int lib_pSkimCheckMeleeRange ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_SkimCheckMeleeRange ( actor ) ) ;
return 1 ;
}
static int lib_pCheckMissileRange ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_CheckMissileRange ( actor ) ) ;
return 1 ;
}
static int lib_pNewChaseDir ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_NewChaseDir ( actor ) ;
return 0 ;
}
static int lib_pLookForPlayers ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2014-03-18 17:56:54 +00:00
fixed_t dist = ( fixed_t ) luaL_optinteger ( L , 2 , 0 ) ;
2014-03-15 16:59:03 +00:00
boolean allaround = lua_optboolean ( L , 3 ) ;
boolean tracer = lua_optboolean ( L , 4 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_LookForPlayers ( actor , allaround , tracer , dist ) ) ;
return 1 ;
}
// P_MOBJ
////////////
static int lib_pSpawnMobj ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
fixed_t z = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
mobjtype_t type = luaL_checkinteger ( L , 4 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2024-01-23 21:43:25 +00:00
LUA_PushUserdata ( L , P_SpawnMobj ( x , y , z , type , NULL ) , META_MOBJ ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
2017-07-08 10:41:20 +00:00
static int lib_pSpawnMobjFromMobj ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
mobjtype_t type = luaL_checkinteger ( L , 5 ) ;
NOHUD
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2017-07-08 10:41:20 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnMobjFromMobj ( actor , x , y , z , type ) , META_MOBJ ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pRemoveMobj ( lua_State * L )
{
mobj_t * th = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! th )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( th - > player )
return luaL_error ( L , " Attempt to remove player mobj with P_RemoveMobj. " ) ;
P_RemoveMobj ( th ) ;
return 0 ;
}
2016-11-11 23:23:41 +00:00
static int lib_pIsValidSprite2 ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2023-11-13 00:46:49 +00:00
UINT16 spr2 = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
2016-11-11 23:23:41 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2016-11-11 23:23:41 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2023-11-13 00:46:49 +00:00
lua_pushboolean ( L , mobj - > skin & & P_IsValidSprite2 ( mobj - > skin , spr2 ) ) ;
2016-11-11 23:23:41 +00:00
return 1 ;
}
2017-03-23 19:11:22 +00:00
// P_SpawnLockOn doesn't exist either, but we want to expose making a local mobj without encouraging hacks.
static int lib_pSpawnLockOn ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * lockon = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
statenum_t state = luaL_checkinteger ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! lockon )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2017-08-25 17:00:20 +00:00
if ( state > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , state, NUMSTATES-1) ;
2023-06-29 22:09:20 +00:00
if ( P_IsLocalPlayer ( player ) ) // Only display it on your own view. Don't display it for spectators
2017-03-23 19:11:22 +00:00
{
mobj_t * visual = P_SpawnMobj ( lockon - > x , lockon - > y , lockon - > z , MT_LOCKON ) ; // positioning, flip handled in P_SceneryThinker
2019-07-12 23:42:03 +00:00
P_SetTarget ( & visual - > target , lockon ) ;
2019-06-19 11:09:02 +00:00
visual - > flags2 | = MF2_DONTDRAW ;
2023-06-29 22:09:20 +00:00
visual - > drawonlyforplayer = player ; // Hide it from the other player in splitscreen, and yourself when spectating
2017-03-23 19:11:22 +00:00
P_SetMobjStateNF ( visual , state ) ;
}
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSpawnMissile ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * dest = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 3 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source | | ! dest )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnMissile ( source , dest , type ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pSpawnXYZMissile ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * dest = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 3 ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 4 ) ;
fixed_t y = luaL_checkfixed ( L , 5 ) ;
fixed_t z = luaL_checkfixed ( L , 6 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source | | ! dest )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnXYZMissile ( source , dest , type , x , y , z ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pSpawnPointMissile ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t xa = luaL_checkfixed ( L , 2 ) ;
fixed_t ya = luaL_checkfixed ( L , 3 ) ;
fixed_t za = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
mobjtype_t type = luaL_checkinteger ( L , 5 ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 6 ) ;
fixed_t y = luaL_checkfixed ( L , 7 ) ;
fixed_t z = luaL_checkfixed ( L , 8 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnPointMissile ( source , xa , ya , za , type , x , y , z ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pSpawnAlteredDirectionMissile ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 2 ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 3 ) ;
fixed_t y = luaL_checkfixed ( L , 4 ) ;
fixed_t z = luaL_checkfixed ( L , 5 ) ;
2014-03-15 16:59:03 +00:00
INT32 shiftingAngle = ( INT32 ) luaL_checkinteger ( L , 5 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnAlteredDirectionMissile ( source , type , x , y , z , shiftingAngle ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pColorTeamMissile ( lua_State * L )
{
mobj_t * missile = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
player_t * source = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! missile )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( ! source )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_ColorTeamMissile ( missile , source ) ;
return 0 ;
}
static int lib_pSPMAngle ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 2 ) ;
2015-05-21 03:54:04 +00:00
angle_t angle = luaL_checkangle ( L , 3 ) ;
2014-03-18 17:56:54 +00:00
UINT8 allowaim = ( UINT8 ) luaL_optinteger ( L , 4 , 0 ) ;
UINT32 flags2 = ( UINT32 ) luaL_optinteger ( L , 5 , 0 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SPMAngle ( source , type , angle , allowaim , flags2 ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pSpawnPlayerMissile ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 2 ) ;
2014-03-18 17:56:54 +00:00
UINT32 flags2 = ( UINT32 ) luaL_optinteger ( L , 3 , 0 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnPlayerMissile ( source , type , flags2 ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pMobjFlip ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushinteger ( L , P_MobjFlip ( mobj ) ) ;
return 1 ;
}
2016-06-01 12:19:44 +00:00
static int lib_pGetMobjGravity ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2016-06-01 12:19:44 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2016-06-01 13:49:14 +00:00
lua_pushfixed ( L , P_GetMobjGravity ( mobj ) ) ;
2016-06-01 12:19:44 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pWeaponOrPanel ( lua_State * L )
{
mobjtype_t type = luaL_checkinteger ( L , 1 ) ;
//HUDSAFE
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , P_WeaponOrPanel ( type ) ) ;
return 1 ;
}
static int lib_pFlashPal ( lua_State * L )
{
player_t * pl = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
UINT16 type = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
UINT16 duration = ( UINT16 ) luaL_checkinteger ( L , 3 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! pl )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_FlashPal ( pl , type , duration ) ;
return 0 ;
}
static int lib_pGetClosestAxis ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_GetClosestAxis ( source ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pSpawnParaloop ( lua_State * L )
{
2023-05-07 15:33:12 +00:00
mobj_t * ptmthing = tmthing ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
fixed_t z = luaL_checkfixed ( L , 3 ) ;
fixed_t radius = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
INT32 number = ( INT32 ) luaL_checkinteger ( L , 5 ) ;
mobjtype_t type = luaL_checkinteger ( L , 6 ) ;
2015-05-21 03:54:04 +00:00
angle_t rotangle = luaL_checkangle ( L , 7 ) ;
2014-03-15 16:59:03 +00:00
statenum_t nstate = luaL_optinteger ( L , 8 , S_NULL ) ;
boolean spawncenter = lua_optboolean ( L , 9 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2016-05-01 21:14:42 +00:00
if ( nstate > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , nstate, NUMSTATES-1) ;
2014-03-15 16:59:03 +00:00
P_SpawnParaloop ( x , y , z , radius , number , type , nstate , rotangle , spawncenter ) ;
2023-05-07 15:33:12 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pBossTargetPlayer ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
boolean closest = lua_optboolean ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_BossTargetPlayer ( actor , closest ) ) ;
return 1 ;
}
static int lib_pSupermanLook4Players ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_SupermanLook4Players ( actor ) ) ;
return 1 ;
}
static int lib_pSetScale ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t newscale = luaL_checkfixed ( L , 2 ) ;
2024-01-02 16:51:43 +00:00
boolean instant = lua_optboolean ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( newscale < FRACUNIT / 100 )
newscale = FRACUNIT / 100 ;
2024-01-02 16:51:43 +00:00
P_SetScale ( mobj , newscale , instant ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2014-08-04 03:49:33 +00:00
static int lib_pInsideANonSolidFFloor ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
ffloor_t * rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 2 , META_FFLOOR ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( ! rover )
return LUA_ErrInvalid ( L , " ffloor_t " ) ;
lua_pushboolean ( L , P_InsideANonSolidFFloor ( mobj , rover ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pCheckDeathPitCollide ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_CheckDeathPitCollide ( mo ) ) ;
return 1 ;
}
2014-08-04 03:49:33 +00:00
static int lib_pCheckSolidLava ( lua_State * L )
{
ffloor_t * rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 2 , META_FFLOOR ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! rover )
return LUA_ErrInvalid ( L , " ffloor_t " ) ;
2019-10-01 19:58:18 +00:00
lua_pushboolean ( L , P_CheckSolidLava ( rover ) ) ;
2014-08-04 03:49:33 +00:00
return 1 ;
}
static int lib_pCanRunOnWater ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
ffloor_t * rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 2 , META_FFLOOR ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! rover )
return LUA_ErrInvalid ( L , " ffloor_t " ) ;
lua_pushboolean ( L , P_CanRunOnWater ( player , rover ) ) ;
return 1 ;
}
2018-04-01 19:54:19 +00:00
static int lib_pMaceRotate ( lua_State * L )
{
mobj_t * center = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
INT32 baserot = luaL_checkinteger ( L , 2 ) ;
INT32 baseprevrot = luaL_checkinteger ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! center )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_MaceRotate ( center , baserot , baseprevrot ) ;
return 0 ;
}
2020-10-14 16:07:02 +00:00
static int lib_pCreateFloorSpriteSlope ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , ( pslope_t * ) P_CreateFloorSpriteSlope ( mobj ) , META_SLOPE ) ;
return 1 ;
}
2020-11-05 03:42:14 +00:00
static int lib_pRemoveFloorSpriteSlope ( lua_State * L )
2020-10-14 16:07:02 +00:00
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2020-11-05 03:42:14 +00:00
P_RemoveFloorSpriteSlope ( mobj ) ;
2020-10-14 16:07:02 +00:00
return 1 ;
}
2020-07-12 05:36:22 +00:00
static int lib_pRailThinker ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2020-10-03 18:24:47 +00:00
mobj_t * ptmthing = tmthing ;
2020-07-12 05:36:22 +00:00
NOHUD
INLEVEL
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_RailThinker ( mobj ) ) ;
2020-10-03 18:24:47 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-07-12 05:36:22 +00:00
return 1 ;
}
2023-06-01 04:03:22 +00:00
static int lib_pCheckSkyHit ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
line_t * line = * ( ( line_t * * ) luaL_checkudata ( L , 2 , META_LINE ) ) ;
//HUDSAFE
INLEVEL
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( ! line )
return LUA_ErrInvalid ( L , " line_t " ) ;
lua_pushboolean ( L , P_CheckSkyHit ( mobj , line ) ) ;
return 1 ;
}
2020-04-19 23:41:03 +00:00
static int lib_pXYMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2020-10-03 18:24:47 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-19 23:41:03 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_XYMovement ( actor ) ;
2020-10-03 18:24:47 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-19 23:41:03 +00:00
return 0 ;
}
2020-04-20 01:01:29 +00:00
static int lib_pRingXYMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2020-10-03 18:24:47 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 01:01:29 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_RingXYMovement ( actor ) ;
2020-10-03 18:24:47 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 01:01:29 +00:00
return 0 ;
}
2020-04-20 01:22:41 +00:00
static int lib_pSceneryXYMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2020-10-03 18:24:47 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 01:22:41 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_SceneryXYMovement ( actor ) ;
2020-10-03 18:24:47 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 01:22:41 +00:00
return 0 ;
}
2020-04-20 01:41:30 +00:00
static int lib_pZMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2021-05-28 17:47:09 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 01:41:30 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_ZMovement ( actor ) ) ;
2023-07-16 12:19:19 +00:00
if ( ! P_MobjWasRemoved ( actor ) )
P_CheckPosition ( actor , actor - > x , actor - > y ) ;
2021-05-28 17:47:09 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 01:41:30 +00:00
return 1 ;
}
2020-04-20 03:55:01 +00:00
static int lib_pRingZMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2021-05-28 17:47:09 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 03:55:01 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_RingZMovement ( actor ) ;
2020-10-03 18:24:47 +00:00
P_CheckPosition ( actor , actor - > x , actor - > y ) ;
2021-05-28 17:47:09 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 03:55:01 +00:00
return 0 ;
}
2020-04-20 01:41:30 +00:00
2020-04-20 04:08:33 +00:00
static int lib_pSceneryZMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2021-05-28 17:47:09 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 04:08:33 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_SceneryZMovement ( actor ) ) ;
2023-07-16 12:19:19 +00:00
if ( ! P_MobjWasRemoved ( actor ) )
P_CheckPosition ( actor , actor - > x , actor - > y ) ;
2021-05-28 17:47:09 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 04:08:33 +00:00
return 1 ;
}
2020-04-20 04:15:46 +00:00
static int lib_pPlayerZMovement ( lua_State * L )
{
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2021-05-28 17:47:09 +00:00
mobj_t * ptmthing = tmthing ;
2020-04-20 04:15:46 +00:00
NOHUD
INLEVEL
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_PlayerZMovement ( actor ) ;
2020-10-03 18:24:47 +00:00
P_CheckPosition ( actor , actor - > x , actor - > y ) ;
2021-05-28 17:47:09 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-04-20 04:15:46 +00:00
return 0 ;
}
2014-03-15 16:59:03 +00:00
// P_USER
////////////
static int lib_pGetPlayerHeight ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , P_GetPlayerHeight ( player ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pGetPlayerSpinHeight ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , P_GetPlayerSpinHeight ( player ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pGetPlayerControlDirection ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushinteger ( L , P_GetPlayerControlDirection ( player ) ) ;
return 1 ;
}
static int lib_pAddPlayerScore ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
UINT32 amount = ( UINT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_AddPlayerScore ( player , amount ) ;
return 0 ;
}
2015-05-27 06:08:18 +00:00
static int lib_pStealPlayerScore ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
UINT32 amount = ( UINT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2015-05-27 06:08:18 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_StealPlayerScore ( player , amount ) ;
return 0 ;
}
2017-04-03 17:28:02 +00:00
static int lib_pGetJumpFlags ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushinteger ( L , P_GetJumpFlags ( player ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pPlayerInPain ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_PlayerInPain ( player ) ) ;
return 1 ;
}
static int lib_pDoPlayerPain ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * source = NULL , * inflictor = NULL ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
2014-03-15 16:59:03 +00:00
source = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
2014-03-15 16:59:03 +00:00
inflictor = * ( ( mobj_t * * ) luaL_checkudata ( L , 3 , META_MOBJ ) ) ;
P_DoPlayerPain ( player , source , inflictor ) ;
return 0 ;
}
static int lib_pResetPlayer ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_ResetPlayer ( player ) ;
return 0 ;
}
2019-06-19 11:09:02 +00:00
static int lib_pPlayerCanDamage ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
"PlayerCanDamage" hook!
* Takes function(player, mo) input.
* Return TRUE for stating that yes, the player is in a state that can cause contact damage, do with that what you will.
* Return FALSE for stating that no, the player is weak and vulnerable and cannot cause contact damage, do with that what you will.
* Return NIL for allowing the function to continue regular operation.
Fills a different ideological niche than ShouldDamage - that's for determining whether damage dished between two objects should happen, this is for determining which way around damage should be dished when considering a player-object interaction.
Or, in other words, think of it as "ShouldDamage is whether damage that has been requested should be granted, for object-object interaction, while PlayerCanDamage is for whether global player properties should cause damage to enemies and monitors in the first place, like spinning, hammering or stomping."
2019-06-19 11:55:05 +00:00
NOHUD // was hud safe but then i added a lua hook
2019-06-19 11:09:02 +00:00
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_PlayerCanDamage ( player , thing ) ) ;
return 1 ;
}
2020-07-20 14:24:16 +00:00
static int lib_pPlayerFullbright ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_PlayerFullbright ( player ) ) ;
return 1 ;
}
2019-06-19 11:09:02 +00:00
2014-08-04 03:49:33 +00:00
static int lib_pIsObjectInGoop ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_IsObjectInGoop ( mo ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pIsObjectOnGround ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_IsObjectOnGround ( mo ) ) ;
return 1 ;
}
2014-08-04 03:49:33 +00:00
static int lib_pInSpaceSector ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_InSpaceSector ( mo ) ) ;
return 1 ;
}
static int lib_pInQuicksand ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_InQuicksand ( mo ) ) ;
return 1 ;
}
2022-09-09 13:35:05 +00:00
static int lib_pInJumpFlipSector ( lua_State * L )
2014-08-04 03:49:33 +00:00
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2022-09-09 13:35:05 +00:00
lua_pushboolean ( L , P_InJumpFlipSector ( mo ) ) ;
2014-08-04 03:49:33 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSetObjectMomZ ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t value = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
boolean relative = lua_optboolean ( L , 3 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_SetObjectMomZ ( mo , value , relative ) ;
return 0 ;
}
2023-06-01 03:45:00 +00:00
static int lib_pIsLocalPlayer ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
//NOHUD
//INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_IsLocalPlayer ( player ) ) ;
return 1 ;
}
2020-02-23 23:19:18 +00:00
static int lib_pPlayJingle ( lua_State * L )
{
player_t * player = NULL ;
2020-03-19 03:40:11 +00:00
jingletype_t jingletype = luaL_checkinteger ( L , 2 ) ;
2020-02-23 23:19:18 +00:00
//NOHUD
//INLEVEL
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( jingletype > = NUMJINGLES )
return luaL_error ( L , " jingletype %d out of range (0 - %d) " , jingletype, NUMJINGLES-1) ;
P_PlayJingle ( player , jingletype ) ;
return 0 ;
}
static int lib_pPlayJingleMusic ( lua_State * L )
{
player_t * player = NULL ;
const char * musnamearg = luaL_checkstring ( L , 2 ) ;
char musname [ 7 ] , * p = musname ;
UINT16 musflags = luaL_optinteger ( L , 3 , 0 ) ;
boolean looping = lua_opttrueboolean ( L , 4 ) ;
2020-03-19 03:40:11 +00:00
jingletype_t jingletype = luaL_optinteger ( L , 5 , JT_OTHER ) ;
2020-02-23 23:19:18 +00:00
//NOHUD
//INLEVEL
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( jingletype > = NUMJINGLES )
return luaL_error ( L , " jingletype %d out of range (0 - %d) " , jingletype, NUMJINGLES-1) ;
musname [ 6 ] = ' \0 ' ;
strncpy ( musname , musnamearg , 6 ) ;
while ( * p ) {
* p = tolower ( * p ) ;
+ + p ;
}
P_PlayJingleMusic ( player , musname , musflags , looping , jingletype ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pRestoreMusic ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2020-02-23 23:19:18 +00:00
//NOHUD
//INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2018-09-18 19:10:42 +00:00
if ( P_IsLocalPlayer ( player ) )
2018-09-18 14:22:17 +00:00
P_RestoreMusic ( player ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2014-08-04 03:49:33 +00:00
static int lib_pSpawnShieldOrb ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_SpawnShieldOrb ( player ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSpawnGhostMobj ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_SpawnGhostMobj ( mobj ) , META_MOBJ ) ;
return 1 ;
}
static int lib_pGivePlayerRings ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 num_rings = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_GivePlayerRings ( player , num_rings ) ;
return 0 ;
}
2023-06-01 03:39:08 +00:00
static int lib_pGivePlayerSpheres ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 num_spheres = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_GivePlayerSpheres ( player , num_spheres ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pGivePlayerLives ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 numlives = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_GivePlayerLives ( player , numlives ) ;
return 0 ;
}
2017-05-29 21:18:02 +00:00
static int lib_pGiveCoopLives ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 numlives = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
boolean sound = ( boolean ) lua_opttrueboolean ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_GiveCoopLives ( player , numlives , sound ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pResetScore ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_ResetScore ( player ) ;
return 0 ;
}
2014-08-04 03:49:33 +00:00
static int lib_pDoJumpShield ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoJumpShield ( player ) ;
return 0 ;
}
2016-10-24 12:33:10 +00:00
static int lib_pDoBubbleBounce ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2016-10-24 12:33:10 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoBubbleBounce ( player ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pBlackOw ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_BlackOw ( player ) ;
return 0 ;
}
2016-07-08 21:56:17 +00:00
static int lib_pElementalFire ( lua_State * L )
2014-03-15 16:59:03 +00:00
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2016-07-08 21:52:49 +00:00
boolean cropcircle = lua_optboolean ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2016-07-08 21:56:17 +00:00
P_ElementalFire ( player , cropcircle ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2020-07-03 17:54:00 +00:00
static int lib_pSpawnSkidDust ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
fixed_t radius = luaL_checkfixed ( L , 2 ) ;
boolean sound = lua_optboolean ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_SpawnSkidDust ( player , radius , sound ) ;
return 0 ;
}
2020-05-23 14:50:35 +00:00
static int lib_pMovePlayer ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2021-05-28 17:47:09 +00:00
mobj_t * ptmthing = tmthing ;
2020-05-23 14:50:35 +00:00
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_MovePlayer ( player ) ;
2021-05-28 17:47:09 +00:00
P_SetTarget ( & tmthing , ptmthing ) ;
2020-05-23 14:50:35 +00:00
return 0 ;
}
2019-11-15 14:35:28 +00:00
static int lib_pDoPlayerFinish ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoPlayerFinish ( player ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pDoPlayerExit ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2023-09-20 02:48:01 +00:00
boolean finishedflag = lua_opttrueboolean ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2023-09-20 02:48:01 +00:00
P_DoPlayerExit ( player , finishedflag ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pInstaThrust ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
angle_t angle = luaL_checkangle ( L , 2 ) ;
fixed_t move = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_InstaThrust ( mo , angle , move ) ;
return 0 ;
}
2021-05-01 18:05:07 +00:00
static int lib_pInstaThrustEvenIn2D ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
angle_t angle = luaL_checkangle ( L , 2 ) ;
fixed_t move = luaL_checkfixed ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_InstaThrustEvenIn2D ( mo , angle , move ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pReturnThrustX ( lua_State * L )
{
angle_t angle ;
fixed_t move ;
if ( lua_isnil ( L , 1 ) | | lua_isuserdata ( L , 1 ) )
lua_remove ( L , 1 ) ; // ignore mobj as arg1
2015-05-21 03:54:04 +00:00
angle = luaL_checkangle ( L , 1 ) ;
move = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , P_ReturnThrustX ( NULL , angle , move ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pReturnThrustY ( lua_State * L )
{
angle_t angle ;
fixed_t move ;
if ( lua_isnil ( L , 1 ) | | lua_isuserdata ( L , 1 ) )
lua_remove ( L , 1 ) ; // ignore mobj as arg1
2015-05-21 03:54:04 +00:00
angle = luaL_checkangle ( L , 1 ) ;
move = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , P_ReturnThrustY ( NULL , angle , move ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pLookForEnemies ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2015-05-27 09:24:31 +00:00
boolean nonenemies = lua_opttrueboolean ( L , 2 ) ;
2017-03-21 23:24:57 +00:00
boolean bullet = lua_optboolean ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2017-03-22 18:51:30 +00:00
LUA_PushUserdata ( L , P_LookForEnemies ( player , nonenemies , bullet ) , META_MOBJ ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pNukeEnemies ( lua_State * L )
{
mobj_t * inflictor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t radius = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! inflictor | | ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_NukeEnemies ( inflictor , source , radius ) ;
return 0 ;
}
2020-07-03 17:54:00 +00:00
static int lib_pEarthquake ( lua_State * L )
{
mobj_t * inflictor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
fixed_t radius = luaL_checkfixed ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! inflictor | | ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_Earthquake ( inflictor , source , radius ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pHomingAttack ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * enemy = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! source | | ! enemy )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2019-07-02 23:58:02 +00:00
lua_pushboolean ( L , P_HomingAttack ( source , enemy ) ) ;
return 1 ;
2014-03-15 16:59:03 +00:00
}
2023-06-03 04:04:18 +00:00
static int lib_pResetCamera ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
camera_t * cam = * ( ( camera_t * * ) luaL_checkudata ( L , 2 , META_CAMERA ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! cam )
return LUA_ErrInvalid ( L , " camera_t " ) ;
P_ResetCamera ( player , cam ) ;
return 0 ;
}
2014-04-14 05:14:58 +00:00
static int lib_pSuperReady ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2023-11-26 20:38:49 +00:00
boolean transform = ( boolean ) lua_opttrueboolean ( L , 2 ) ;
2014-04-14 05:14:58 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-04-14 05:14:58 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2023-11-26 20:38:49 +00:00
lua_pushboolean ( L , P_SuperReady ( player , transform ) ) ;
2014-04-14 05:14:58 +00:00
return 1 ;
}
static int lib_pDoJump ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
boolean soundandstate = ( boolean ) lua_opttrueboolean ( L , 2 ) ;
2023-09-19 16:54:51 +00:00
boolean allowflip = ( boolean ) lua_opttrueboolean ( L , 3 ) ;
2014-04-14 05:14:58 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-04-14 05:14:58 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2023-09-19 16:54:51 +00:00
P_DoJump ( player , soundandstate , allowflip ) ;
2014-04-14 05:14:58 +00:00
return 0 ;
}
2023-06-01 03:49:30 +00:00
static int lib_pDoSpinDashDust ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoSpinDashDust ( player ) ;
return 0 ;
}
2014-05-23 20:46:49 +00:00
static int lib_pSpawnThokMobj ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-05-23 20:46:49 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_SpawnThokMobj ( player ) ;
return 0 ;
}
2014-05-23 22:08:35 +00:00
static int lib_pSpawnSpinMobj ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobjtype_t type = luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
2014-05-23 22:08:35 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_SpawnSpinMobj ( player , type ) ;
return 0 ;
}
2014-05-26 02:41:05 +00:00
static int lib_pTelekinesis ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t thrust = luaL_checkfixed ( L , 2 ) ;
fixed_t range = luaL_checkfixed ( L , 3 ) ;
2014-05-26 02:41:05 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-05-26 02:41:05 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_Telekinesis ( player , thrust , range ) ;
return 0 ;
}
2017-08-10 12:57:09 +00:00
static int lib_pSwitchShield ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
UINT16 shield = luaL_checkinteger ( L , 2 ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_SwitchShield ( player , shield ) ;
return 0 ;
}
2023-06-01 03:28:53 +00:00
static int lib_pDoTailsOverlay ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * tails = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! tails )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_DoTailsOverlay ( player , tails ) ;
return 0 ;
}
static int lib_pDoMetalJetFume ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * fume = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! fume )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_DoMetalJetFume ( player , fume ) ;
return 0 ;
}
static int lib_pDoFollowMobj ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
mobj_t * followmobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( ! followmobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_DoFollowMobj ( player , followmobj ) ;
return 0 ;
}
2021-03-23 03:49:22 +00:00
static int lib_pPlayerCanEnterSpinGaps ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_PlayerCanEnterSpinGaps ( player ) ) ;
return 1 ;
}
static int lib_pPlayerShouldUseSpinHeight ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_PlayerShouldUseSpinHeight ( player ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
// P_MAP
///////////
static int lib_pCheckPosition ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_CheckPosition ( thing , x , y ) ) ;
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
static int lib_pTryMove ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
boolean allowdropoff = lua_optboolean ( L , 4 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_TryMove ( thing , x , y , allowdropoff ) ) ;
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
static int lib_pMove ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t speed = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! actor )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_Move ( actor , speed ) ) ;
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
2023-10-28 13:19:35 +00:00
// TODO: 2.3: Delete
2014-03-15 16:59:03 +00:00
static int lib_pTeleportMove ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2022-04-27 22:08:40 +00:00
LUA_Deprecated ( L , " P_TeleportMove " , " P_SetOrigin \" or \" P_MoveOrigin " ) ;
lua_pushboolean ( L , P_MoveOrigin ( thing , x , y , z ) ) ;
2021-11-29 13:20:27 +00:00
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
static int lib_pSetOrigin ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
NOHUD
INLEVEL
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_SetOrigin ( thing , x , y , z ) ) ;
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
static int lib_pMoveOrigin ( lua_State * L )
{
mobj_t * ptmthing = tmthing ;
mobj_t * thing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
NOHUD
INLEVEL
if ( ! thing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_MoveOrigin ( thing , x , y , z ) ) ;
2014-03-15 16:59:03 +00:00
LUA_PushUserdata ( L , tmthing , META_MOBJ ) ;
P_SetTarget ( & tmthing , ptmthing ) ;
return 2 ;
}
2023-09-17 23:45:57 +00:00
static int lib_pLineIsBlocking ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
line_t * line = * ( ( line_t * * ) luaL_checkudata ( L , 2 , META_LINE ) ) ;
NOHUD
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
if ( ! line )
return LUA_ErrInvalid ( L , " line_t " ) ;
// P_LineOpening in P_LineIsBlocking sets these variables.
// We want to keep their old values after so that whatever
// map collision code uses them doesn't get messed up.
fixed_t oldopentop = opentop ;
fixed_t oldopenbottom = openbottom ;
fixed_t oldopenrange = openrange ;
fixed_t oldlowfloor = lowfloor ;
fixed_t oldhighceiling = highceiling ;
pslope_t * oldopentopslope = opentopslope ;
pslope_t * oldopenbottomslope = openbottomslope ;
ffloor_t * oldopenfloorrover = openfloorrover ;
ffloor_t * oldopenceilingrover = openceilingrover ;
lua_pushboolean ( L , P_LineIsBlocking ( mo , line ) ) ;
opentop = oldopentop ;
openbottom = oldopenbottom ;
openrange = oldopenrange ;
lowfloor = oldlowfloor ;
highceiling = oldhighceiling ;
opentopslope = oldopentopslope ;
openbottomslope = oldopenbottomslope ;
openfloorrover = oldopenfloorrover ;
openceilingrover = oldopenceilingrover ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSlideMove ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_SlideMove ( mo ) ;
return 0 ;
}
static int lib_pBounceMove ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_BounceMove ( mo ) ;
return 0 ;
}
static int lib_pCheckSight ( lua_State * L )
{
mobj_t * t1 = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * t2 = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
//HUDSAFE?
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! t1 | | ! t2 )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
lua_pushboolean ( L , P_CheckSight ( t1 , t2 ) ) ;
return 1 ;
}
static int lib_pCheckHoopPosition ( lua_State * L )
{
mobj_t * hoopthing = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
fixed_t radius = luaL_checkfixed ( L , 5 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! hoopthing )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_CheckHoopPosition ( hoopthing , x , y , z , radius ) ;
return 0 ;
}
static int lib_pRadiusAttack ( lua_State * L )
{
mobj_t * spot = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
fixed_t damagedist = luaL_checkfixed ( L , 3 ) ;
2018-05-08 22:26:36 +00:00
UINT8 damagetype = luaL_optinteger ( L , 4 , 0 ) ;
2020-07-04 10:27:06 +00:00
boolean sightcheck = lua_opttrueboolean ( L , 5 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! spot | | ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2020-07-04 10:27:06 +00:00
P_RadiusAttack ( spot , source , damagedist , damagetype , sightcheck ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pFloorzAtPos ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
fixed_t z = luaL_checkfixed ( L , 3 ) ;
fixed_t height = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , P_FloorzAtPos ( x , y , z , height ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
2014-05-23 22:55:07 +00:00
2020-09-30 06:58:41 +00:00
static int lib_pCeilingzAtPos ( lua_State * L )
{
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
fixed_t z = luaL_checkfixed ( L , 3 ) ;
fixed_t height = luaL_checkfixed ( L , 4 ) ;
//HUDSAFE
INLEVEL
lua_pushfixed ( L , P_CeilingzAtPos ( x , y , z , height ) ) ;
return 1 ;
}
2023-11-30 22:50:22 +00:00
static int lib_pGetSectorLightLevelAt ( lua_State * L )
{
boolean has_sector = false ;
sector_t * sector = NULL ;
if ( ! lua_isnoneornil ( L , 1 ) )
{
has_sector = true ;
sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
}
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
INLEVEL
if ( has_sector & & ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
if ( sector )
lua_pushinteger ( L , P_GetLightLevelFromSectorAt ( sector , x , y , z ) ) ;
else
lua_pushinteger ( L , P_GetSectorLightLevelAt ( x , y , z ) ) ;
return 1 ;
}
2023-08-04 03:31:51 +00:00
static int lib_pGetSectorColormapAt ( lua_State * L )
{
2023-11-22 00:15:09 +00:00
boolean has_sector = false ;
2023-08-04 03:31:51 +00:00
sector_t * sector = NULL ;
2023-11-22 00:15:09 +00:00
if ( ! lua_isnoneornil ( L , 1 ) )
{
has_sector = true ;
2023-08-04 03:31:51 +00:00
sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
2023-11-22 00:15:09 +00:00
}
2023-08-04 03:31:51 +00:00
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
INLEVEL
2023-11-22 00:15:09 +00:00
if ( has_sector & & ! sector )
2023-08-04 03:31:51 +00:00
return LUA_ErrInvalid ( L , " sector_t " ) ;
extracolormap_t * exc ;
if ( sector )
exc = P_GetColormapFromSectorAt ( sector , x , y , z ) ;
else
exc = P_GetSectorColormapAt ( x , y , z ) ;
LUA_PushUserdata ( L , exc , META_EXTRACOLORMAP ) ;
return 1 ;
}
2014-05-23 22:55:07 +00:00
static int lib_pDoSpring ( lua_State * L )
{
mobj_t * spring = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * object = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-05-23 22:55:07 +00:00
if ( ! spring | | ! object )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2015-07-28 18:28:51 +00:00
lua_pushboolean ( L , P_DoSpring ( spring , object ) ) ;
return 1 ;
2014-05-23 22:55:07 +00:00
}
2021-07-11 06:23:50 +00:00
static int lib_pTryCameraMove ( lua_State * L )
{
camera_t * cam = * ( ( camera_t * * ) luaL_checkudata ( L , 1 , META_CAMERA ) ) ;
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
if ( ! cam )
return LUA_ErrInvalid ( L , " camera_t " ) ;
lua_pushboolean ( L , P_TryCameraMove ( x , y , cam ) ) ;
return 1 ;
}
static int lib_pTeleportCameraMove ( lua_State * L )
{
camera_t * cam = * ( ( camera_t * * ) luaL_checkudata ( L , 1 , META_CAMERA ) ) ;
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
fixed_t z = luaL_checkfixed ( L , 4 ) ;
if ( ! cam )
return LUA_ErrInvalid ( L , " camera_t " ) ;
cam - > x = x ;
cam - > y = y ;
cam - > z = z ;
P_CheckCameraPosition ( x , y , cam ) ;
cam - > subsector = R_PointInSubsector ( x , y ) ;
cam - > floorz = tmfloorz ;
cam - > ceilingz = tmceilingz ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
// P_INTER
////////////
static int lib_pRemoveShield ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_RemoveShield ( player ) ;
return 0 ;
}
static int lib_pDamageMobj ( lua_State * L )
{
mobj_t * target = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) , * inflictor = NULL , * source = NULL ;
INT32 damage ;
2015-02-13 16:15:58 +00:00
UINT8 damagetype ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! target )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
2014-03-15 16:59:03 +00:00
inflictor = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
2014-03-15 16:59:03 +00:00
source = * ( ( mobj_t * * ) luaL_checkudata ( L , 3 , META_MOBJ ) ) ;
damage = ( INT32 ) luaL_optinteger ( L , 4 , 1 ) ;
2015-02-13 16:15:58 +00:00
damagetype = ( UINT8 ) luaL_optinteger ( L , 5 , 0 ) ;
lua_pushboolean ( L , P_DamageMobj ( target , inflictor , source , damage , damagetype ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_pKillMobj ( lua_State * L )
{
mobj_t * target = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) , * inflictor = NULL , * source = NULL ;
2015-02-13 16:15:58 +00:00
UINT8 damagetype ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! target )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
2014-03-15 16:59:03 +00:00
inflictor = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
2014-03-15 16:59:03 +00:00
source = * ( ( mobj_t * * ) luaL_checkudata ( L , 3 , META_MOBJ ) ) ;
2015-02-13 16:15:58 +00:00
damagetype = ( UINT8 ) luaL_optinteger ( L , 4 , 0 ) ;
P_KillMobj ( target , inflictor , source , damagetype ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pPlayerRingBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 num_rings = ( INT32 ) luaL_optinteger ( L , 2 , - 1 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
if ( num_rings = = - 1 )
2015-08-15 20:07:16 +00:00
num_rings = player - > rings ;
2014-03-15 16:59:03 +00:00
P_PlayerRingBurst ( player , num_rings ) ;
return 0 ;
}
static int lib_pPlayerWeaponPanelBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_PlayerWeaponPanelBurst ( player ) ;
return 0 ;
}
static int lib_pPlayerWeaponAmmoBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_PlayerWeaponAmmoBurst ( player ) ;
return 0 ;
}
2015-05-27 04:45:34 +00:00
static int lib_pPlayerWeaponPanelOrAmmoBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2015-05-27 04:45:34 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_PlayerWeaponPanelOrAmmoBurst ( player ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pPlayerEmeraldBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
boolean toss = lua_optboolean ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_PlayerEmeraldBurst ( player , toss ) ;
return 0 ;
}
static int lib_pPlayerFlagBurst ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
boolean toss = lua_optboolean ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_PlayerFlagBurst ( player , toss ) ;
return 0 ;
}
static int lib_pPlayRinglossSound ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
P_PlayRinglossSound ( source ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pPlayDeathSound ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
P_PlayDeathSound ( source ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pPlayVictorySound ( lua_State * L )
{
mobj_t * source = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! source )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
P_PlayVictorySound ( source ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pPlayLivesJingle ( lua_State * L )
{
2020-02-23 21:40:59 +00:00
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
//INLEVEL
2020-02-23 21:40:59 +00:00
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
2014-03-15 16:59:03 +00:00
P_PlayLivesJingle ( player ) ;
return 0 ;
}
static int lib_pCanPickupItem ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
boolean weapon = lua_optboolean ( L , 2 ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
lua_pushboolean ( L , P_CanPickupItem ( player , weapon ) ) ;
return 1 ;
}
static int lib_pDoNightsScore ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoNightsScore ( player ) ;
return 0 ;
}
2015-05-27 06:08:18 +00:00
static int lib_pDoMatchSuper ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2015-05-27 06:08:18 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoMatchSuper ( player ) ;
return 0 ;
}
2023-06-01 03:21:31 +00:00
static int lib_pTouchSpecialThing ( lua_State * L )
{
mobj_t * special = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
mobj_t * toucher = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
boolean heightcheck = lua_optboolean ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! special | | ! toucher )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2023-07-02 03:24:11 +00:00
if ( ! toucher - > player )
return luaL_error ( L , " P_TouchSpecialThing requires a valid toucher.player. " ) ;
2023-06-01 03:21:31 +00:00
P_TouchSpecialThing ( special , toucher , heightcheck ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
// P_SPEC
////////////
static int lib_pThrust ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
2015-05-21 03:54:04 +00:00
angle_t angle = luaL_checkangle ( L , 2 ) ;
fixed_t move = luaL_checkfixed ( L , 3 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_Thrust ( mo , angle , move ) ;
return 0 ;
}
2021-05-01 18:05:07 +00:00
static int lib_pThrustEvenIn2D ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
angle_t angle = luaL_checkangle ( L , 2 ) ;
fixed_t move = luaL_checkfixed ( L , 3 ) ;
NOHUD
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_ThrustEvenIn2D ( mo , angle , move ) ;
return 0 ;
}
static int lib_pVectorInstaThrust ( lua_State * L )
{
fixed_t xa = luaL_checkfixed ( L , 1 ) ;
fixed_t xb = luaL_checkfixed ( L , 2 ) ;
fixed_t xc = luaL_checkfixed ( L , 3 ) ;
fixed_t ya = luaL_checkfixed ( L , 4 ) ;
fixed_t yb = luaL_checkfixed ( L , 5 ) ;
fixed_t yc = luaL_checkfixed ( L , 6 ) ;
fixed_t za = luaL_checkfixed ( L , 7 ) ;
fixed_t zb = luaL_checkfixed ( L , 8 ) ;
fixed_t zc = luaL_checkfixed ( L , 9 ) ;
fixed_t momentum = luaL_checkfixed ( L , 10 ) ;
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 11 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_VectorInstaThrust ( xa , xb , xc , ya , yb , yc , za , zb , zc , momentum , mo ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSetMobjStateNF ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
statenum_t state = luaL_checkinteger ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
2016-05-01 21:14:42 +00:00
if ( state > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , state, NUMSTATES-1) ;
2014-03-15 16:59:03 +00:00
if ( mobj - > player & & state = = S_NULL )
return luaL_error ( L , " Attempt to remove player mobj with S_NULL. " ) ;
lua_pushboolean ( L , P_SetMobjStateNF ( mobj , state ) ) ;
return 1 ;
}
static int lib_pDoSuperTransformation ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
boolean giverings = lua_optboolean ( L , 2 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoSuperTransformation ( player , giverings ) ;
return 0 ;
}
2024-02-24 14:41:36 +00:00
static int lib_pDoSuperDetransformation ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
NOHUD
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
P_DoSuperDetransformation ( player ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pExplodeMissile ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_ExplodeMissile ( mo ) ;
return 0 ;
}
2021-12-08 20:07:26 +00:00
static int lib_pMobjTouchingSectorSpecial ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
INT32 section = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
INT32 number = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
//HUDSAFE
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_MobjTouchingSectorSpecial ( mo , section , number ) , META_SECTOR ) ;
return 1 ;
}
2023-10-28 13:19:35 +00:00
// TODO: 2.3: Delete
2023-01-01 10:31:44 +00:00
static int lib_pThingOnSpecial3DFloor ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_Deprecated ( L , " P_ThingOnSpecial3DFloor " , " P_MobjTouchingSectorSpecial \" or \" P_MobjTouchingSectorSpecialFlag " ) ;
LUA_PushUserdata ( L , P_ThingOnSpecial3DFloor ( mo ) , META_SECTOR ) ;
return 1 ;
}
2021-12-30 23:03:24 +00:00
static int lib_pMobjTouchingSectorSpecialFlag ( lua_State * L )
{
mobj_t * mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
sectorspecialflags_t flag = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
//HUDSAFE
INLEVEL
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
LUA_PushUserdata ( L , P_MobjTouchingSectorSpecialFlag ( mo , flag ) , META_SECTOR ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pPlayerTouchingSectorSpecial ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 section = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
INT32 number = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
LUA_PushUserdata ( L , P_PlayerTouchingSectorSpecial ( player , section , number ) , META_SECTOR ) ;
return 1 ;
}
2021-12-30 23:03:24 +00:00
static int lib_pPlayerTouchingSectorSpecialFlag ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
sectorspecialflags_t flag = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
//HUDSAFE
INLEVEL
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
LUA_PushUserdata ( L , P_PlayerTouchingSectorSpecialFlag ( player , flag ) , META_SECTOR ) ;
return 1 ;
}
2017-10-30 20:09:41 +00:00
static int lib_pFindLowestFloorSurrounding ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
lua_pushfixed ( L , P_FindLowestFloorSurrounding ( sector ) ) ;
return 1 ;
}
static int lib_pFindHighestFloorSurrounding ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
lua_pushfixed ( L , P_FindHighestFloorSurrounding ( sector ) ) ;
return 1 ;
}
static int lib_pFindNextHighestFloor ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
fixed_t currentheight ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
// defaults to floorheight of sector arg
currentheight = ( fixed_t ) luaL_optinteger ( L , 2 , sector - > floorheight ) ;
lua_pushfixed ( L , P_FindNextHighestFloor ( sector , currentheight ) ) ;
return 1 ;
}
static int lib_pFindNextLowestFloor ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
fixed_t currentheight ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
// defaults to floorheight of sector arg
currentheight = ( fixed_t ) luaL_optinteger ( L , 2 , sector - > floorheight ) ;
lua_pushfixed ( L , P_FindNextLowestFloor ( sector , currentheight ) ) ;
return 1 ;
}
static int lib_pFindLowestCeilingSurrounding ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
lua_pushfixed ( L , P_FindLowestCeilingSurrounding ( sector ) ) ;
return 1 ;
}
static int lib_pFindHighestCeilingSurrounding ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
//HUDSAFE
INLEVEL
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
lua_pushfixed ( L , P_FindHighestCeilingSurrounding ( sector ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pFindSpecialLineFromTag ( lua_State * L )
{
INT16 special = ( INT16 ) luaL_checkinteger ( L , 1 ) ;
INT16 line = ( INT16 ) luaL_checkinteger ( L , 2 ) ;
INT32 start = ( INT32 ) luaL_optinteger ( L , 3 , - 1 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushinteger ( L , P_FindSpecialLineFromTag ( special , line , start ) ) ;
return 1 ;
}
static int lib_pSwitchWeather ( lua_State * L )
{
INT32 weathernum = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
2014-08-04 03:49:33 +00:00
player_t * user = NULL ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) ) // if a player, setup weather for only the player, otherwise setup weather for all players
user = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! user ) // global
globalweather = weathernum ;
if ( ! user | | P_IsLocalPlayer ( user ) )
P_SwitchWeather ( weathernum ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_pLinedefExecute ( lua_State * L )
{
INT32 tag = ( INT16 ) luaL_checkinteger ( L , 1 ) ;
mobj_t * actor = NULL ;
sector_t * caller = NULL ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
actor = * ( ( mobj_t * * ) luaL_checkudata ( L , 2 , META_MOBJ ) ) ;
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
caller = * ( ( sector_t * * ) luaL_checkudata ( L , 3 , META_SECTOR ) ) ;
P_LinedefExecute ( tag , actor , caller ) ;
return 0 ;
}
static int lib_pSpawnLightningFlash ( lua_State * L )
{
sector_t * sector = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! sector )
return LUA_ErrInvalid ( L , " sector_t " ) ;
P_SpawnLightningFlash ( sector ) ;
return 0 ;
}
static int lib_pFadeLight ( lua_State * L )
{
INT16 tag = ( INT16 ) luaL_checkinteger ( L , 1 ) ;
INT32 destvalue = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
INT32 speed = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
2018-09-09 02:10:51 +00:00
boolean ticbased = lua_optboolean ( L , 4 ) ;
2018-09-18 10:27:30 +00:00
boolean force = lua_optboolean ( L , 5 ) ;
2021-09-19 21:12:30 +00:00
boolean relative = lua_optboolean ( L , 6 ) ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2021-09-19 21:12:30 +00:00
P_FadeLight ( tag , destvalue , speed , ticbased , force , relative ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2014-05-26 02:41:05 +00:00
static int lib_pIsFlagAtBase ( lua_State * L )
{
2024-02-11 12:30:15 +00:00
mobjtype_t type = luaL_checkinteger ( L , 1 ) ;
2023-08-01 11:45:58 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2024-02-11 12:30:15 +00:00
NOSPAWNNULL
lua_pushboolean ( L , P_IsFlagAtBase ( type ) ) ;
2014-05-26 02:41:05 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_pSetupLevelSky ( lua_State * L )
{
INT32 skynum = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
2014-08-04 03:49:33 +00:00
player_t * user = NULL ;
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) ) // if a player, setup sky for only the player, otherwise setup sky for all players
user = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! user ) // global
P_SetupLevelSky ( skynum , true ) ;
else if ( P_IsLocalPlayer ( user ) )
P_SetupLevelSky ( skynum , false ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
// Shhh, P_SetSkyboxMobj doesn't actually exist yet.
static int lib_pSetSkyboxMobj ( lua_State * L )
{
int n = lua_gettop ( L ) ;
mobj_t * mo = NULL ;
player_t * user = NULL ;
int w = 0 ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( ! lua_isnil ( L , 1 ) ) // nil leaves mo as NULL to remove the skybox rendering.
{
mo = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ; // otherwise it is a skybox mobj.
if ( ! mo )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
}
if ( n = = 1 )
;
else if ( lua_isuserdata ( L , 2 ) )
user = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
else if ( lua_isnil ( L , 2 ) )
w = 0 ;
else if ( lua_isboolean ( L , 2 ) )
{
if ( lua_toboolean ( L , 2 ) )
w = 1 ;
else
w = 0 ;
}
else
w = luaL_optinteger ( L , 2 , 0 ) ;
if ( n > 2 & & lua_isuserdata ( L , 3 ) )
{
user = * ( ( player_t * * ) luaL_checkudata ( L , 3 , META_PLAYER ) ) ;
if ( ! user )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( w > 1 | | w < 0 )
return luaL_error ( L , " skybox mobj index %d is out of range for P_SetSkyboxMobj argument #2 (expected 0 or 1) " , w) ;
if ( ! user | | P_IsLocalPlayer ( user ) )
skyboxmo [ w ] = mo ;
return 0 ;
}
// Shhh, neither does P_StartQuake.
static int lib_pStartQuake ( lua_State * L )
{
fixed_t q_intensity = luaL_checkinteger ( L , 1 ) ;
UINT16 q_time = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
static mappoint_t q_epicenter = { 0 , 0 , 0 } ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
// While technically we don't support epicenter and radius,
// we get their values anyway if they exist.
// This way when support is added we won't have to change anything.
if ( ! lua_isnoneornil ( L , 3 ) )
{
luaL_checktype ( L , 3 , LUA_TTABLE ) ;
lua_getfield ( L , 3 , " x " ) ;
if ( lua_isnil ( L , - 1 ) )
{
lua_pop ( L , 1 ) ;
lua_rawgeti ( L , 3 , 1 ) ;
}
if ( ! lua_isnil ( L , - 1 ) )
q_epicenter . x = luaL_checkinteger ( L , - 1 ) ;
else
q_epicenter . x = 0 ;
lua_pop ( L , 1 ) ;
lua_getfield ( L , 3 , " y " ) ;
if ( lua_isnil ( L , - 1 ) )
{
lua_pop ( L , 1 ) ;
lua_rawgeti ( L , 3 , 2 ) ;
}
if ( ! lua_isnil ( L , - 1 ) )
q_epicenter . y = luaL_checkinteger ( L , - 1 ) ;
else
q_epicenter . y = 0 ;
lua_pop ( L , 1 ) ;
lua_getfield ( L , 3 , " z " ) ;
if ( lua_isnil ( L , - 1 ) )
{
lua_pop ( L , 1 ) ;
lua_rawgeti ( L , 3 , 3 ) ;
}
if ( ! lua_isnil ( L , - 1 ) )
q_epicenter . z = luaL_checkinteger ( L , - 1 ) ;
else
q_epicenter . z = 0 ;
lua_pop ( L , 1 ) ;
quake . epicenter = & q_epicenter ;
}
else
quake . epicenter = NULL ;
quake . radius = luaL_optinteger ( L , 4 , 512 * FRACUNIT ) ;
// These things are actually used in 2.1.
quake . intensity = q_intensity ;
quake . time = q_time ;
return 0 ;
}
2014-08-04 03:49:33 +00:00
static int lib_evCrumbleChain ( lua_State * L )
{
2019-07-28 21:45:20 +00:00
sector_t * sec = NULL ;
ffloor_t * rover = NULL ;
2014-08-04 03:49:33 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2019-07-28 21:53:27 +00:00
if ( ! lua_isnone ( L , 2 ) )
2019-07-28 21:45:20 +00:00
{
2019-07-28 21:53:27 +00:00
if ( ! lua_isnil ( L , 1 ) )
{
sec = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
if ( ! sec )
return LUA_ErrInvalid ( L , " sector_t " ) ;
}
2019-07-28 21:45:20 +00:00
rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 2 , META_FFLOOR ) ) ;
}
else
rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 1 , META_FFLOOR ) ) ;
2014-08-04 03:49:33 +00:00
if ( ! rover )
return LUA_ErrInvalid ( L , " ffloor_t " ) ;
EV_CrumbleChain ( sec , rover ) ;
return 0 ;
}
2016-07-01 19:27:09 +00:00
static int lib_evStartCrumble ( lua_State * L )
{
sector_t * sec = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
ffloor_t * rover = * ( ( ffloor_t * * ) luaL_checkudata ( L , 2 , META_FFLOOR ) ) ;
boolean floating = lua_optboolean ( L , 3 ) ;
player_t * player = NULL ;
fixed_t origalpha ;
boolean crumblereturn = lua_optboolean ( L , 6 ) ;
NOHUD
if ( ! sec )
return LUA_ErrInvalid ( L , " sector_t " ) ;
if ( ! rover )
return LUA_ErrInvalid ( L , " ffloor_t " ) ;
if ( ! lua_isnone ( L , 4 ) & & lua_isuserdata ( L , 4 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 4 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! lua_isnone ( L , 5 ) )
origalpha = luaL_checkfixed ( L , 5 ) ;
else
origalpha = rover - > alpha ;
lua_pushboolean ( L , EV_StartCrumble ( sec , rover , floating , player , origalpha , crumblereturn ) ! = 0 ) ;
return 0 ;
}
2018-10-20 20:08:59 +00:00
// P_SLOPES
////////////
static int lib_pGetZAt ( lua_State * L )
{
fixed_t x = luaL_checkfixed ( L , 2 ) ;
fixed_t y = luaL_checkfixed ( L , 3 ) ;
//HUDSAFE
2020-05-18 14:16:45 +00:00
if ( lua_isnil ( L , 1 ) )
2020-05-18 13:23:56 +00:00
{
fixed_t z = luaL_checkfixed ( L , 4 ) ;
2020-05-18 14:16:45 +00:00
lua_pushfixed ( L , P_GetZAt ( NULL , x , y , z ) ) ;
}
else
{
pslope_t * slope = * ( ( pslope_t * * ) luaL_checkudata ( L , 1 , META_SLOPE ) ) ;
lua_pushfixed ( L , P_GetSlopeZAt ( slope , x , y ) ) ;
2020-05-18 13:23:56 +00:00
}
2018-10-20 20:08:59 +00:00
return 1 ;
}
2021-02-11 12:06:40 +00:00
static int lib_pButteredSlope ( lua_State * L )
{
mobj_t * mobj = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
NOHUD
INLEVEL
if ( ! mobj )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
P_ButteredSlope ( mobj ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
// R_DEFS
////////////
static int lib_rPointToAngle ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushangle ( L , R_PointToAngle ( x , y ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_rPointToAngle2 ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t px2 = luaL_checkfixed ( L , 1 ) ;
fixed_t py2 = luaL_checkfixed ( L , 2 ) ;
fixed_t px1 = luaL_checkfixed ( L , 3 ) ;
fixed_t py1 = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushangle ( L , R_PointToAngle2 ( px2 , py2 , px1 , py1 ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_rPointToDist ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , R_PointToDist ( x , y ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_rPointToDist2 ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t px2 = luaL_checkfixed ( L , 1 ) ;
fixed_t py2 = luaL_checkfixed ( L , 2 ) ;
fixed_t px1 = luaL_checkfixed ( L , 3 ) ;
fixed_t py1 = luaL_checkfixed ( L , 4 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2015-05-21 03:54:04 +00:00
lua_pushfixed ( L , R_PointToDist2 ( px2 , py2 , px1 , py1 ) ) ;
2014-03-15 16:59:03 +00:00
return 1 ;
}
static int lib_rPointInSubsector ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
LUA_PushUserdata ( L , R_PointInSubsector ( x , y ) , META_SUBSECTOR ) ;
return 1 ;
}
2020-01-22 15:58:57 +00:00
static int lib_rPointInSubsectorOrNil ( lua_State * L )
2020-01-17 03:49:43 +00:00
{
fixed_t x = luaL_checkfixed ( L , 1 ) ;
fixed_t y = luaL_checkfixed ( L , 2 ) ;
2020-01-22 15:58:57 +00:00
subsector_t * sub = R_PointInSubsectorOrNull ( x , y ) ;
2020-01-17 03:49:43 +00:00
//HUDSAFE
INLEVEL
if ( sub )
LUA_PushUserdata ( L , sub , META_SUBSECTOR ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
2014-04-14 05:14:58 +00:00
// R_THINGS
////////////
static int lib_rChar2Frame ( lua_State * L )
{
const char * p = luaL_checkstring ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , R_Char2Frame ( * p ) ) ; // first character only
return 1 ;
}
static int lib_rFrame2Char ( lua_State * L )
{
UINT8 ch = ( UINT8 ) luaL_checkinteger ( L , 1 ) ;
char c [ 2 ] = " " ;
//HUDSAFE
c [ 0 ] = R_Frame2Char ( ch ) ;
2024-04-03 18:39:53 +00:00
if ( c [ 0 ] = = ' \xFF ' )
return luaL_error ( L , " frame %u cannot be represented by a character " , ch ) ;
2014-04-14 05:14:58 +00:00
c [ 1 ] = 0 ;
lua_pushstring ( L , c ) ;
lua_pushinteger ( L , c [ 0 ] ) ;
return 2 ;
}
2023-11-13 15:13:44 +00:00
// R_SKINS
////////////
2014-08-04 03:49:33 +00:00
// R_SetPlayerSkin technically doesn't exist either, although it's basically just SetPlayerSkin and SetPlayerSkinByNum handled in one place for convenience
static int lib_rSetPlayerSkin ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
2019-11-10 16:10:34 +00:00
INT32 i = - 1 , j = - 1 ;
2014-08-04 03:49:33 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-08-04 03:49:33 +00:00
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
2019-11-10 16:10:34 +00:00
j = ( player - players ) ;
2014-08-04 03:49:33 +00:00
if ( lua_isnoneornil ( L , 2 ) )
return luaL_error ( L , " argument #2 not given (expected number or string) " ) ;
else if ( lua_type ( L , 2 ) = = LUA_TNUMBER ) // skin number
{
2019-11-10 16:10:34 +00:00
i = luaL_checkinteger ( L , 2 ) ;
if ( i < 0 | | i > = numskins )
return luaL_error ( L , " skin %d (argument #2) out of range ( 0 - % d ) " , i, numskins-1) ;
2014-08-04 03:49:33 +00:00
}
else // skin name
{
const char * skinname = luaL_checkstring ( L , 2 ) ;
2019-11-10 16:10:34 +00:00
i = R_SkinAvailable ( skinname ) ;
if ( i = = - 1 )
return luaL_error ( L , " skin %s (argument 2) is not loaded " , skinname) ;
2014-08-04 03:49:33 +00:00
}
2019-11-10 16:10:34 +00:00
if ( ! R_SkinUsable ( j , i ) )
return luaL_error ( L , " skin %d (argument 2) not usable - check with R_SkinUsable ( player_t , skin ) first . " , i) ;
SetPlayerSkinByNum ( j , i ) ;
2014-08-04 03:49:33 +00:00
return 0 ;
}
2019-11-10 16:10:34 +00:00
static int lib_rSkinUsable ( lua_State * L )
{
player_t * player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
INT32 i = - 1 , j = - 1 ;
if ( player )
j = ( player - players ) ;
else if ( netgame | | multiplayer )
return luaL_error ( L , " player_t (argument #1) must be provided in multiplayer games " ) ;
if ( lua_isnoneornil ( L , 2 ) )
return luaL_error ( L , " argument #2 not given (expected number or string) " ) ;
else if ( lua_type ( L , 2 ) = = LUA_TNUMBER ) // skin number
{
i = luaL_checkinteger ( L , 2 ) ;
if ( i < 0 | | i > = numskins )
return luaL_error ( L , " skin %d (argument #2) out of range ( 0 - % d ) " , i, numskins-1) ;
}
else // skin name
{
const char * skinname = luaL_checkstring ( L , 2 ) ;
i = R_SkinAvailable ( skinname ) ;
if ( i = = - 1 )
return luaL_error ( L , " skin %s (argument 2) is not loaded " , skinname) ;
}
lua_pushboolean ( L , R_SkinUsable ( j , i ) ) ;
return 1 ;
}
2023-11-13 15:13:44 +00:00
static int lib_pGetStateSprite2 ( lua_State * L )
{
int statenum = luaL_checkinteger ( L , 1 ) ;
if ( statenum < 0 | | statenum > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , statenum, NUMSTATES-1) ;
lua_pushinteger ( L , P_GetStateSprite2 ( & states [ statenum ] ) ) ;
return 1 ;
}
static int lib_pGetSprite2StateFrame ( lua_State * L )
{
int statenum = luaL_checkinteger ( L , 1 ) ;
if ( statenum < 0 | | statenum > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , statenum, NUMSTATES-1) ;
lua_pushinteger ( L , P_GetSprite2StateFrame ( & states [ statenum ] ) ) ;
return 1 ;
}
static int lib_pIsStateSprite2Super ( lua_State * L )
{
int statenum = luaL_checkinteger ( L , 1 ) ;
if ( statenum < 0 | | statenum > = NUMSTATES )
return luaL_error ( L , " state %d out of range (0 - %d) " , statenum, NUMSTATES-1) ;
lua_pushboolean ( L , P_IsStateSprite2Super ( & states [ statenum ] ) ) ;
return 1 ;
}
2023-11-13 17:08:08 +00:00
// Not a real function. Who cares? I know I don't.
2023-11-13 15:19:03 +00:00
static int lib_pGetSuperSprite2 ( lua_State * L )
{
int animID = luaL_checkinteger ( L , 1 ) & SPR2F_MASK ;
if ( animID < 0 | | animID > = NUMPLAYERSPRITES )
return luaL_error ( L , " sprite2 %d out of range (0 - %d) " , animID, NUMPLAYERSPRITES-1) ;
lua_pushinteger ( L , animID | SPR2F_SUPER ) ;
return 1 ;
}
2018-12-17 02:36:54 +00:00
// R_DATA
////////////
static int lib_rCheckTextureNumForName ( lua_State * L )
{
const char * name = luaL_checkstring ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , R_CheckTextureNumForName ( name ) ) ;
return 1 ;
}
static int lib_rTextureNumForName ( lua_State * L )
{
const char * name = luaL_checkstring ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , R_TextureNumForName ( name ) ) ;
return 1 ;
}
2022-11-20 20:25:53 +00:00
static int lib_rCheckTextureNameForNum ( lua_State * L )
{
2022-11-20 21:07:14 +00:00
INT32 num = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
2022-11-20 20:25:53 +00:00
//HUDSAFE
lua_pushstring ( L , R_CheckTextureNameForNum ( num ) ) ;
return 1 ;
}
static int lib_rTextureNameForNum ( lua_State * L )
{
2022-11-20 21:07:14 +00:00
INT32 num = ( INT32 ) luaL_checkinteger ( L , 1 ) ;
2022-11-20 20:25:53 +00:00
//HUDSAFE
lua_pushstring ( L , R_TextureNameForNum ( num ) ) ;
return 1 ;
}
2020-02-15 22:35:00 +00:00
// R_DRAW
2014-03-15 16:59:03 +00:00
////////////
2020-02-15 22:35:00 +00:00
static int lib_rGetColorByName ( lua_State * L )
{
const char * colorname = luaL_checkstring ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , R_GetColorByName ( colorname ) ) ;
return 1 ;
}
2020-07-12 11:39:52 +00:00
static int lib_rGetSuperColorByName ( lua_State * L )
{
const char * colorname = luaL_checkstring ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , R_GetSuperColorByName ( colorname ) ) ;
return 1 ;
}
2020-02-15 22:35:00 +00:00
// Lua exclusive function, returns the name of a color from the SKINCOLOR_ constant.
// SKINCOLOR_GREEN > "Green" for example
static int lib_rGetNameByColor ( lua_State * L )
{
2020-05-24 00:29:07 +00:00
UINT16 colornum = ( UINT16 ) luaL_checkinteger ( L , 1 ) ;
if ( ! colornum | | colornum > = numskincolors )
return luaL_error ( L , " skincolor %d out of range (1 - %d) . " , colornum, numskincolors-1) ;
lua_pushstring ( L , skincolors [ colornum ] . name ) ;
2020-02-15 22:35:00 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
2020-02-15 22:55:25 +00:00
// S_SOUND
////////////
2020-09-14 20:33:15 +00:00
static int GetValidSoundOrigin ( lua_State * L , void * * origin )
{
const char * type ;
lua_settop ( L , 1 ) ;
type = GetUserdataUType ( L ) ;
if ( fasticmp ( type , " mobj_t " ) )
{
* origin = * ( ( mobj_t * * ) luaL_checkudata ( L , 1 , META_MOBJ ) ) ;
if ( ! ( * origin ) )
return LUA_ErrInvalid ( L , " mobj_t " ) ;
return 1 ;
}
else if ( fasticmp ( type , " sector_t " ) )
{
* origin = * ( ( sector_t * * ) luaL_checkudata ( L , 1 , META_SECTOR ) ) ;
if ( ! ( * origin ) )
return LUA_ErrInvalid ( L , " sector_t " ) ;
* origin = & ( ( sector_t * ) ( * origin ) ) - > soundorg ;
return 1 ;
}
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
}
2014-03-15 16:59:03 +00:00
static int lib_sStartSound ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2014-03-15 16:59:03 +00:00
sfxenum_t sound_id = luaL_checkinteger ( L , 2 ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2020-09-13 15:33:18 +00:00
2016-05-01 21:14:42 +00:00
if ( sound_id > = NUMSFX )
return luaL_error ( L , " sfx %d out of range (0 - %d) " , sound_id, NUMSFX-1) ;
2020-09-13 15:33:18 +00:00
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 3 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
2020-09-13 15:33:18 +00:00
if ( ! lua_isnil ( L , 1 ) )
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
return 0 ;
2014-08-04 03:49:33 +00:00
if ( ! player | | P_IsLocalPlayer ( player ) )
2018-07-31 09:10:02 +00:00
{
2020-07-17 05:08:38 +00:00
if ( hud_running | | hook_cmd_running )
origin = NULL ; // HUD rendering and CMD building startsound shouldn't have an origin, just remove it instead of having a retarded error.
2018-12-17 19:43:59 +00:00
2014-08-04 03:49:33 +00:00
S_StartSound ( origin , sound_id ) ;
2018-12-17 19:43:59 +00:00
}
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_sStartSoundAtVolume ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2014-03-15 16:59:03 +00:00
sfxenum_t sound_id = luaL_checkinteger ( L , 2 ) ;
INT32 volume = ( INT32 ) luaL_checkinteger ( L , 3 ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2016-05-01 21:14:42 +00:00
if ( sound_id > = NUMSFX )
return luaL_error ( L , " sfx %d out of range (0 - %d) " , sound_id, NUMSFX-1) ;
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 4 ) & & lua_isuserdata ( L , 4 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 4 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
2020-09-13 15:33:18 +00:00
if ( ! lua_isnil ( L , 1 ) )
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
2020-09-13 15:33:18 +00:00
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
2020-09-14 20:33:15 +00:00
2014-08-04 03:49:33 +00:00
if ( ! player | | P_IsLocalPlayer ( player ) )
2020-09-14 20:33:15 +00:00
S_StartSoundAtVolume ( origin , sound_id , volume ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_sStopSound ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
2020-09-13 15:33:18 +00:00
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
2014-03-15 16:59:03 +00:00
S_StopSound ( origin ) ;
return 0 ;
}
2020-05-06 13:22:04 +00:00
static int lib_sStopSoundByID ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2020-05-16 15:14:47 +00:00
sfxenum_t sound_id = luaL_checkinteger ( L , 2 ) ;
2020-05-06 13:22:04 +00:00
//NOHUD
2020-09-13 15:33:18 +00:00
2020-05-06 13:22:04 +00:00
if ( sound_id > = NUMSFX )
return luaL_error ( L , " sfx %d out of range (0 - %d) " , sound_id, NUMSFX-1) ;
2020-09-13 15:33:18 +00:00
if ( ! lua_isnil ( L , 1 ) )
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
2020-09-13 15:33:18 +00:00
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
2020-05-06 13:22:04 +00:00
S_StopSoundByID ( origin , sound_id ) ;
return 0 ;
}
2014-03-15 16:59:03 +00:00
static int lib_sChangeMusic ( lua_State * L )
{
2020-12-12 04:11:15 +00:00
UINT32 position , prefadems , fadeinms ;
2016-03-03 10:54:07 +00:00
2016-01-08 03:48:20 +00:00
const char * music_name = luaL_checkstring ( L , 1 ) ;
2014-03-15 16:59:03 +00:00
boolean looping = ( boolean ) lua_opttrueboolean ( L , 2 ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2016-01-08 03:48:20 +00:00
UINT16 music_flags = 0 ;
2016-03-03 10:54:07 +00:00
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 3 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
2016-03-03 10:54:07 +00:00
2016-01-08 03:48:20 +00:00
music_flags = ( UINT16 ) luaL_optinteger ( L , 4 , 0 ) ;
2018-09-18 14:22:17 +00:00
position = ( UINT32 ) luaL_optinteger ( L , 5 , 0 ) ;
prefadems = ( UINT32 ) luaL_optinteger ( L , 6 , 0 ) ;
fadeinms = ( UINT32 ) luaL_optinteger ( L , 7 , 0 ) ;
2014-08-04 03:49:33 +00:00
if ( ! player | | P_IsLocalPlayer ( player ) )
2018-09-18 14:22:17 +00:00
S_ChangeMusicEx ( music_name , music_flags , looping , position , prefadems , fadeinms ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
static int lib_sSpeedMusic ( lua_State * L )
{
2015-05-21 03:54:04 +00:00
fixed_t fixedspeed = luaL_checkfixed ( L , 1 ) ;
2014-03-15 16:59:03 +00:00
float speed = FIXED_TO_FLOAT ( fixedspeed ) ;
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
2018-09-18 14:32:38 +00:00
S_SpeedMusic ( speed ) ;
return 0 ;
2014-03-15 16:59:03 +00:00
}
static int lib_sStopMusic ( lua_State * L )
{
2014-08-04 03:49:33 +00:00
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2014-08-04 03:49:33 +00:00
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
S_StopMusic ( ) ;
2014-03-15 16:59:03 +00:00
return 0 ;
}
2019-01-02 09:58:03 +00:00
static int lib_sSetInternalMusicVolume ( lua_State * L )
{
UINT32 volume = ( UINT32 ) luaL_checkinteger ( L , 1 ) ;
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2019-01-02 09:58:03 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
{
S_SetInternalMusicVolume ( volume ) ;
lua_pushboolean ( L , true ) ;
}
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sStopFadingMusic ( lua_State * L )
{
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2019-01-02 09:58:03 +00:00
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
{
S_StopFadingMusic ( ) ;
lua_pushboolean ( L , true ) ;
}
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sFadeMusic ( lua_State * L )
{
UINT32 target_volume = ( UINT32 ) luaL_checkinteger ( L , 1 ) ;
UINT32 ms ;
INT32 source_volume ;
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2019-01-02 09:58:03 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 3 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
ms = ( UINT32 ) luaL_checkinteger ( L , 2 ) ;
source_volume = - 1 ;
}
else if ( ! lua_isnone ( L , 4 ) & & lua_isuserdata ( L , 4 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 4 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
source_volume = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
ms = ( UINT32 ) luaL_checkinteger ( L , 3 ) ;
}
else if ( luaL_optinteger ( L , 3 , INT32_MAX ) = = INT32_MAX )
{
ms = ( UINT32 ) luaL_checkinteger ( L , 2 ) ;
source_volume = - 1 ;
}
else
{
source_volume = ( INT32 ) luaL_checkinteger ( L , 2 ) ;
ms = ( UINT32 ) luaL_checkinteger ( L , 3 ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushboolean ( L , S_FadeMusicFromVolume ( target_volume , source_volume , ms ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sFadeOutStopMusic ( lua_State * L )
{
UINT32 ms = ( UINT32 ) luaL_checkinteger ( L , 1 ) ;
player_t * player = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2019-01-02 09:58:03 +00:00
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
{
lua_pushboolean ( L , S_FadeOutStopMusic ( ms ) ) ;
}
else
lua_pushnil ( L ) ;
return 1 ;
}
2020-02-23 23:58:32 +00:00
static int lib_sGetMusicLength ( lua_State * L )
{
lua_pushinteger ( L , S_GetMusicLength ( ) ) ;
return 1 ;
}
static int lib_sGetMusicPosition ( lua_State * L )
{
lua_pushinteger ( L , S_GetMusicPosition ( ) ) ;
return 1 ;
}
static int lib_sSetMusicPosition ( lua_State * L )
{
UINT32 pos = ( UINT32 ) luaL_checkinteger ( L , 1 ) ;
lua_pushboolean ( L , S_SetMusicPosition ( pos ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_sOriginPlaying ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2020-02-23 18:45:43 +00:00
//NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
2020-09-13 15:33:18 +00:00
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , S_OriginPlaying ( origin ) ) ;
return 1 ;
}
static int lib_sIdPlaying ( lua_State * L )
{
sfxenum_t id = luaL_checkinteger ( L , 1 ) ;
2020-02-23 18:45:43 +00:00
//NOHUD
2016-05-01 21:14:42 +00:00
if ( id > = NUMSFX )
return luaL_error ( L , " sfx %d out of range (0 - %d) " , id, NUMSFX-1) ;
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , S_IdPlaying ( id ) ) ;
return 1 ;
}
static int lib_sSoundPlaying ( lua_State * L )
{
2020-09-13 15:33:18 +00:00
void * origin = NULL ;
2014-03-15 16:59:03 +00:00
sfxenum_t id = luaL_checkinteger ( L , 2 ) ;
2020-02-23 18:45:43 +00:00
//NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2016-05-01 21:14:42 +00:00
if ( id > = NUMSFX )
return luaL_error ( L , " sfx %d out of range (0 - %d) " , id, NUMSFX-1) ;
2020-09-14 20:33:15 +00:00
if ( ! GetValidSoundOrigin ( L , & origin ) )
return LUA_ErrInvalid ( L , " mobj_t/sector_t " ) ;
2020-09-13 15:33:18 +00:00
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , S_SoundPlaying ( origin , id ) ) ;
return 1 ;
}
2017-04-29 15:27:52 +00:00
// This doesn't really exist, but we're providing it as a handy netgame-safe wrapper for stuff that should be locally handled.
static int lib_sStartMusicCaption ( lua_State * L )
{
player_t * player = NULL ;
const char * caption = luaL_checkstring ( L , 1 ) ;
UINT16 lifespan = ( UINT16 ) luaL_checkinteger ( L , 2 ) ;
//HUDSAFE
2020-02-23 18:45:43 +00:00
//INLEVEL
2017-04-29 15:27:52 +00:00
if ( ! lua_isnone ( L , 3 ) & & lua_isuserdata ( L , 3 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 3 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( lifespan & & ( ! player | | P_IsLocalPlayer ( player ) ) )
{
strlcpy ( S_sfx [ sfx_None ] . caption , caption , sizeof ( S_sfx [ sfx_None ] . caption ) ) ;
S_StartCaption ( sfx_None , - 1 , lifespan ) ;
}
return 0 ;
}
2020-11-01 01:15:41 +00:00
static int lib_sMusicType ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushinteger ( L , S_MusicType ( ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sMusicPlaying ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushboolean ( L , S_MusicPlaying ( ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sMusicPaused ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushboolean ( L , S_MusicPaused ( ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sMusicName ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushstring ( L , S_MusicName ( ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sMusicExists ( lua_State * L )
{
boolean checkMIDI = lua_opttrueboolean ( L , 2 ) ;
boolean checkDigi = lua_opttrueboolean ( L , 3 ) ;
const char * music_name = luaL_checkstring ( L , 1 ) ;
NOHUD
lua_pushboolean ( L , S_MusicExists ( music_name , checkMIDI , checkDigi ) ) ;
return 1 ;
}
static int lib_sSetMusicLoopPoint ( lua_State * L )
{
UINT32 looppoint = ( UINT32 ) luaL_checkinteger ( L , 1 ) ;
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 2 ) & & lua_isuserdata ( L , 2 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 2 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushboolean ( L , S_SetMusicLoopPoint ( looppoint ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sGetMusicLoopPoint ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
lua_pushinteger ( L , ( int ) S_GetMusicLoopPoint ( ) ) ;
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sPauseMusic ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
{
S_PauseAudio ( ) ;
lua_pushboolean ( L , true ) ;
}
else
lua_pushnil ( L ) ;
return 1 ;
}
static int lib_sResumeMusic ( lua_State * L )
{
player_t * player = NULL ;
NOHUD
if ( ! lua_isnone ( L , 1 ) & & lua_isuserdata ( L , 1 ) )
{
player = * ( ( player_t * * ) luaL_checkudata ( L , 1 , META_PLAYER ) ) ;
if ( ! player )
return LUA_ErrInvalid ( L , " player_t " ) ;
}
if ( ! player | | P_IsLocalPlayer ( player ) )
{
S_ResumeAudio ( ) ;
lua_pushboolean ( L , true ) ;
}
else
lua_pushnil ( L ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
// G_GAME
////////////
2019-12-19 05:13:09 +00:00
// Copypasted from lib_cvRegisterVar :]
static int lib_gAddGametype ( lua_State * L )
{
const char * k ;
lua_Integer i ;
const char * gtname = NULL ;
2019-12-23 21:54:25 +00:00
const char * gtconst = NULL ;
2019-12-19 05:13:09 +00:00
const char * gtdescription = NULL ;
INT16 newgtidx = 0 ;
UINT32 newgtrules = 0 ;
UINT32 newgttol = 0 ;
INT32 newgtpointlimit = 0 ;
INT32 newgttimelimit = 0 ;
UINT8 newgtleftcolor = 0 ;
UINT8 newgtrightcolor = 0 ;
INT16 newgtrankingstype = - 1 ;
int newgtinttype = 0 ;
luaL_checktype ( L , 1 , LUA_TTABLE ) ;
lua_settop ( L , 1 ) ; // Clear out all other possible arguments, leaving only the first one.
if ( ! lua_lumploading )
return luaL_error ( L , " This function cannot be called from within a hook or coroutine! " ) ;
2019-12-23 22:00:46 +00:00
// Ran out of gametype slots
if ( gametypecount = = NUMGAMETYPEFREESLOTS )
return luaL_error ( L , " Ran out of free gametype slots! " ) ;
2019-12-19 05:13:09 +00:00
# define FIELDERROR(f, e) luaL_error(L, "bad value for " LUA_QL(f) " in table passed to " LUA_QL("G_AddGametype") " (%s)", e);
# define TYPEERROR(f, t) FIELDERROR(f, va("%s expected, got %s", lua_typename(L, t), luaL_typename(L, -1)))
lua_pushnil ( L ) ;
while ( lua_next ( L , 1 ) ) {
// stack: gametype table, key/index, value
// 1 2 3
i = 0 ;
k = NULL ;
if ( lua_isnumber ( L , 2 ) )
i = lua_tointeger ( L , 2 ) ;
else if ( lua_isstring ( L , 2 ) )
k = lua_tostring ( L , 2 ) ;
// Sorry, no gametype rules as key names.
if ( i = = 1 | | ( k & & fasticmp ( k , " name " ) ) ) {
if ( ! lua_isstring ( L , 3 ) )
TYPEERROR ( " name " , LUA_TSTRING )
gtname = Z_StrDup ( lua_tostring ( L , 3 ) ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 2 | | ( k & & fasticmp ( k , " identifier " ) ) ) {
if ( ! lua_isstring ( L , 3 ) )
TYPEERROR ( " identifier " , LUA_TSTRING )
gtconst = Z_StrDup ( lua_tostring ( L , 3 ) ) ;
} else if ( i = = 3 | | ( k & & fasticmp ( k , " rules " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " rules " , LUA_TNUMBER )
newgtrules = ( UINT32 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 4 | | ( k & & fasticmp ( k , " typeoflevel " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " typeoflevel " , LUA_TNUMBER )
newgttol = ( UINT32 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 5 | | ( k & & fasticmp ( k , " rankingtype " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " rankingtype " , LUA_TNUMBER )
newgtrankingstype = ( INT16 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 6 | | ( k & & fasticmp ( k , " intermissiontype " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " intermissiontype " , LUA_TNUMBER )
newgtinttype = ( int ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 7 | | ( k & & fasticmp ( k , " defaultpointlimit " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " defaultpointlimit " , LUA_TNUMBER )
newgtpointlimit = ( INT32 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 8 | | ( k & & fasticmp ( k , " defaulttimelimit " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " defaulttimelimit " , LUA_TNUMBER )
newgttimelimit = ( INT32 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 9 | | ( k & & fasticmp ( k , " description " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isstring ( L , 3 ) )
TYPEERROR ( " description " , LUA_TSTRING )
gtdescription = Z_StrDup ( lua_tostring ( L , 3 ) ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 10 | | ( k & & fasticmp ( k , " headerleftcolor " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " headerleftcolor " , LUA_TNUMBER )
newgtleftcolor = ( UINT8 ) lua_tointeger ( L , 3 ) ;
2019-12-23 21:54:25 +00:00
} else if ( i = = 11 | | ( k & & fasticmp ( k , " headerrightcolor " ) ) ) {
2019-12-19 05:13:09 +00:00
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " headerrightcolor " , LUA_TNUMBER )
newgtrightcolor = ( UINT8 ) lua_tointeger ( L , 3 ) ;
// Key name specified
} else if ( ( ! i ) & & ( k & & fasticmp ( k , " headercolor " ) ) ) {
if ( ! lua_isnumber ( L , 3 ) )
TYPEERROR ( " headercolor " , LUA_TNUMBER )
newgtleftcolor = newgtrightcolor = ( UINT8 ) lua_tointeger ( L , 3 ) ;
}
lua_pop ( L , 1 ) ;
}
# undef FIELDERROR
# undef TYPEERROR
// pop gametype table
lua_pop ( L , 1 ) ;
// Set defaults
if ( gtname = = NULL )
gtname = Z_StrDup ( " Unnamed gametype " ) ;
if ( gtdescription = = NULL )
gtdescription = Z_StrDup ( " ??? " ) ;
// Add the new gametype
newgtidx = G_AddGametype ( newgtrules ) ;
G_AddGametypeTOL ( newgtidx , newgttol ) ;
2019-12-19 05:20:49 +00:00
G_SetGametypeDescription ( newgtidx , NULL , newgtleftcolor , newgtrightcolor ) ;
strncpy ( gametypedesc [ newgtidx ] . notes , gtdescription , 441 ) ;
2019-12-19 05:13:09 +00:00
// Not covered by G_AddGametype alone.
if ( newgtrankingstype = = - 1 )
newgtrankingstype = newgtidx ;
gametyperankings [ newgtidx ] = newgtrankingstype ;
intermissiontypes [ newgtidx ] = newgtinttype ;
pointlimits [ newgtidx ] = newgtpointlimit ;
timelimits [ newgtidx ] = newgttimelimit ;
// Write the new gametype name.
Gametype_Names [ newgtidx ] = gtname ;
// Write the constant name.
2019-12-23 21:54:25 +00:00
if ( gtconst = = NULL )
gtconst = gtname ;
G_AddGametypeConstant ( newgtidx , gtconst ) ;
2019-12-19 05:13:09 +00:00
// Update gametype_cons_t accordingly.
G_UpdateGametypeSelections ( ) ;
// done
CONS_Printf ( " Added gametype %s \n " , Gametype_Names [ newgtidx ] ) ;
return 0 ;
}
2021-01-15 22:45:41 +00:00
// Bot adding function!
// Partly lifted from Got_AddPlayer
static int lib_gAddPlayer ( lua_State * L )
{
2023-10-16 00:35:04 +00:00
INT16 i , newplayernum ;
2021-01-15 22:45:41 +00:00
player_t * newplayer ;
2021-08-31 06:18:25 +00:00
SINT8 skinnum = 0 , bot ;
2021-01-15 22:45:41 +00:00
2024-06-09 21:33:24 +00:00
for ( i = 1 ; i < MAXPLAYERS ; i + + )
2021-01-15 22:45:41 +00:00
{
if ( ! playeringame [ i ] )
break ;
}
2023-10-16 00:35:04 +00:00
2021-01-15 22:45:41 +00:00
if ( i > = MAXPLAYERS )
{
lua_pushnil ( L ) ;
return 1 ;
}
2022-02-08 13:58:09 +00:00
2021-01-15 22:45:41 +00:00
newplayernum = i ;
2021-01-23 15:13:16 +00:00
CL_ClearPlayer ( newplayernum ) ;
2021-01-15 22:45:41 +00:00
playeringame [ newplayernum ] = true ;
G_AddPlayer ( newplayernum ) ;
newplayer = & players [ newplayernum ] ;
newplayer - > jointime = 0 ;
newplayer - > quittime = 0 ;
2023-10-14 19:50:59 +00:00
newplayer - > lastinputtime = 0 ;
2021-01-15 22:45:41 +00:00
2021-01-23 15:13:16 +00:00
// Read the skin argument (defaults to Sonic)
2021-01-15 22:45:41 +00:00
if ( ! lua_isnoneornil ( L , 1 ) )
{
skinnum = R_SkinAvailable ( luaL_checkstring ( L , 1 ) ) ;
skinnum = skinnum < 0 ? 0 : skinnum ;
}
2021-01-23 15:13:16 +00:00
// Read the color (defaults to skin prefcolor)
2021-01-15 22:45:41 +00:00
if ( ! lua_isnoneornil ( L , 2 ) )
newplayer - > skincolor = R_GetColorByName ( luaL_checkstring ( L , 2 ) ) ;
else
2023-10-28 22:09:42 +00:00
newplayer - > skincolor = skins [ skinnum ] - > prefcolor ;
2023-03-15 20:50:13 +00:00
// Set the bot default name as the skin
2023-10-28 22:09:42 +00:00
strcpy ( player_names [ newplayernum ] , skins [ skinnum ] - > realname ) ;
2021-01-15 22:45:41 +00:00
2021-01-23 15:13:16 +00:00
// Read the bot name, if given
2021-01-15 22:45:41 +00:00
if ( ! lua_isnoneornil ( L , 3 ) )
2022-02-08 13:58:09 +00:00
strlcpy ( player_names [ newplayernum ] , luaL_checkstring ( L , 3 ) , sizeof ( * player_names ) ) ;
2021-01-15 22:45:41 +00:00
bot = luaL_optinteger ( L , 4 , 3 ) ;
2021-01-24 01:01:48 +00:00
newplayer - > bot = ( bot > = BOT_NONE & & bot < = BOT_MPAI ) ? bot : BOT_MPAI ;
2022-02-08 13:58:09 +00:00
2021-01-23 15:13:16 +00:00
// If our bot is a 2P type, we'll need to set its leader so it can spawn
if ( newplayer - > bot = = BOT_2PAI | | newplayer - > bot = = BOT_2PHUMAN )
B_UpdateBotleader ( newplayer ) ;
2022-02-08 13:58:09 +00:00
2021-01-23 15:13:16 +00:00
// Set the skin (can't do this until AFTER bot type is set!)
2021-01-15 22:45:41 +00:00
SetPlayerSkinByNum ( newplayernum , skinnum ) ;
if ( netgame )
{
char joinmsg [ 256 ] ;
2023-03-15 20:50:13 +00:00
// Truncate bot name
2023-05-07 15:42:39 +00:00
player_names [ newplayernum ] [ sizeof ( * player_names ) - 8 ] = ' \0 ' ; // The length of colored [BOT] + 1
2023-03-15 20:50:13 +00:00
2021-01-15 22:45:41 +00:00
strcpy ( joinmsg , M_GetText ( " \x82 *Bot %s has joined the game (player %d) " ) ) ;
strcpy ( joinmsg , va ( joinmsg , player_names [ newplayernum ] , newplayernum ) ) ;
HU_AddChatText ( joinmsg , false ) ;
2023-03-15 20:50:13 +00:00
// Append blue [BOT] tag at the end
2023-05-07 15:42:39 +00:00
strlcat ( player_names [ newplayernum ] , " \x84 [BOT] \x80 " , sizeof ( * player_names ) ) ;
2021-01-15 22:45:41 +00:00
}
2022-02-08 13:58:09 +00:00
2021-01-15 22:45:41 +00:00
LUA_PushUserdata ( L , newplayer , META_PLAYER ) ;
2021-01-16 03:50:15 +00:00
return 1 ;
2021-01-15 22:45:41 +00:00
}
2021-01-24 01:01:48 +00:00
// Bot removing function
static int lib_gRemovePlayer ( lua_State * L )
{
UINT8 pnum = - 1 ;
if ( ! lua_isnoneornil ( L , 1 ) )
pnum = luaL_checkinteger ( L , 1 ) ;
2021-07-15 22:19:47 +00:00
else // No argument
return luaL_error ( L , " argument #1 not given (expected number) " ) ;
if ( pnum > = MAXPLAYERS ) // Out of range
return luaL_error ( L , " playernum %d out of range (0 - %d) " , pnum, MAXPLAYERS-1) ;
if ( playeringame [ pnum ] ) // Found player
2021-01-24 01:01:48 +00:00
{
2021-07-15 22:19:47 +00:00
if ( players [ pnum ] . bot = = BOT_NONE ) // Can't remove clients.
return luaL_error ( L , " G_RemovePlayer can only be used on players with a bot value other than BOT_NONE. " ) ;
else
2021-01-24 01:01:48 +00:00
{
2021-07-15 22:19:47 +00:00
players [ pnum ] . removing = true ;
2021-01-24 01:01:48 +00:00
lua_pushboolean ( L , true ) ;
return 1 ;
}
}
2021-07-15 22:19:47 +00:00
// Fell through. Invalid player
return LUA_ErrInvalid ( L , " player_t " ) ;
2021-01-24 01:01:48 +00:00
}
2022-02-28 17:43:00 +00:00
static int lib_gSetUsedCheats ( lua_State * L )
{
// Let large-scale level packs using Lua be able to add cheat commands.
boolean silent = lua_optboolean ( L , 1 ) ;
//NOHUD
//INLEVEL
G_SetUsedCheats ( silent ) ;
return 0 ;
}
2020-01-16 04:04:50 +00:00
static int Lcheckmapnumber ( lua_State * L , int idx , const char * fun )
2020-01-16 03:13:41 +00:00
{
if ( ISINLEVEL )
return luaL_optinteger ( L , idx , gamemap ) ;
else
{
if ( lua_isnoneornil ( L , idx ) )
{
return luaL_error ( L ,
2020-01-16 04:04:50 +00:00
" %s can only be used without a parameter while in a level. " ,
fun
2020-01-16 03:13:41 +00:00
) ;
}
else
return luaL_checkinteger ( L , idx ) ;
}
}
2014-03-15 16:59:03 +00:00
static int lib_gBuildMapName ( lua_State * L )
{
2020-01-16 04:04:50 +00:00
INT32 map = Lcheckmapnumber ( L , 1 , " G_BuildMapName " ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
lua_pushstring ( L , G_BuildMapName ( map ) ) ;
return 1 ;
}
2020-01-16 04:04:50 +00:00
static int lib_gBuildMapTitle ( lua_State * L )
{
2020-01-18 23:56:03 +00:00
INT32 map = Lcheckmapnumber ( L , 1 , " G_BuildMapTitle " ) ;
2020-01-16 04:04:50 +00:00
char * name ;
if ( map < 1 | | map > NUMMAPS )
{
return luaL_error ( L ,
" map number %d out of range (1 - %d) " ,
map ,
NUMMAPS
) ;
}
name = G_BuildMapTitle ( map ) ;
lua_pushstring ( L , name ) ;
Z_Free ( name ) ;
return 1 ;
}
2020-01-16 02:49:22 +00:00
static void
Lpushdim ( lua_State * L , int c , struct searchdim * v )
{
int i ;
lua_createtable ( L , c , 0 ) ; /* I guess narr is numeric indices??? */
for ( i = 0 ; i < c ; + + i )
{
lua_createtable ( L , 0 , 2 ) ; /* and hashed indices (field)... */
lua_pushnumber ( L , v [ i ] . pos ) ;
lua_setfield ( L , - 2 , " pos " ) ;
lua_pushnumber ( L , v [ i ] . siz ) ;
lua_setfield ( L , - 2 , " siz " ) ;
lua_rawseti ( L , - 2 , 1 + i ) ;
}
}
/*
I decided to make this return a table because userdata
is scary and tables let the user set their own fields .
*/
2020-01-16 03:01:28 +00:00
/*
Returns :
[ 1 ] = > map number
[ 2 ] = > map title
[ 3 ] = > search frequency table
The frequency table is unsorted . It has the following format :
{
[ ' mapnum ' ] ,
[ ' matchd ' ] = > matches in map title string
[ ' keywhd ' ] = > matches in map keywords
The above two tables have the following format :
{
[ ' pos ' ] = > offset from start of string
[ ' siz ' ] = > length of match
} . . .
[ ' total ' ] = > the total matches
} . . .
*/
2020-01-16 02:49:22 +00:00
static int lib_gFindMap ( lua_State * L )
{
const char * query = luaL_checkstring ( L , 1 ) ;
INT32 map ;
char * realname ;
INT32 frc ;
mapsearchfreq_t * frv ;
INT32 i ;
map = G_FindMap ( query , & realname , & frv , & frc ) ;
lua_settop ( L , 0 ) ;
lua_pushnumber ( L , map ) ;
lua_pushstring ( L , realname ) ;
lua_createtable ( L , frc , 0 ) ;
for ( i = 0 ; i < frc ; + + i )
{
lua_createtable ( L , 0 , 4 ) ;
lua_pushnumber ( L , frv [ i ] . mapnum ) ;
lua_setfield ( L , - 2 , " mapnum " ) ;
Lpushdim ( L , frv [ i ] . matchc , frv [ i ] . matchd ) ;
lua_setfield ( L , - 2 , " matchd " ) ;
Lpushdim ( L , frv [ i ] . keywhc , frv [ i ] . keywhd ) ;
lua_setfield ( L , - 2 , " keywhd " ) ;
lua_pushnumber ( L , frv [ i ] . total ) ;
lua_setfield ( L , - 2 , " total " ) ;
lua_rawseti ( L , - 2 , 1 + i ) ;
}
G_FreeMapSearch ( frv , frc ) ;
Z_Free ( realname ) ;
return 3 ;
}
2020-01-16 04:32:40 +00:00
/*
Returns :
[ 1 ] = > map number
[ 2 ] = > map title
*/
static int lib_gFindMapByNameOrCode ( lua_State * L )
{
const char * query = luaL_checkstring ( L , 1 ) ;
INT32 map ;
char * realname ;
map = G_FindMapByNameOrCode ( query , & realname ) ;
lua_pushnumber ( L , map ) ;
if ( map )
{
lua_pushstring ( L , realname ) ;
Z_Free ( realname ) ;
return 2 ;
}
else
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_gDoReborn ( lua_State * L )
{
INT32 playernum = luaL_checkinteger ( L , 1 ) ;
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
if ( playernum > = MAXPLAYERS )
2016-05-01 21:14:42 +00:00
return luaL_error ( L , " playernum %d out of range (0 - %d) " , playernum, MAXPLAYERS-1) ;
2014-03-15 16:59:03 +00:00
G_DoReborn ( playernum ) ;
return 0 ;
}
2018-12-18 19:48:04 +00:00
// Another Lua function that doesn't actually exist!
2023-08-06 01:15:38 +00:00
// Sets nextmapoverride, skipstats and nextgametype without instantly ending the level, for instances where other sources should be exiting the level, like normal signposts.
2018-12-18 19:48:04 +00:00
static int lib_gSetCustomExitVars ( lua_State * L )
2014-03-15 16:59:03 +00:00
{
2014-11-12 00:55:07 +00:00
int n = lua_gettop ( L ) ; // Num arguments
2014-03-15 16:59:03 +00:00
NOHUD
2017-01-18 22:02:28 +00:00
INLEVEL
2014-11-12 00:55:07 +00:00
// LUA EXTENSION: Custom exit like support
// Supported:
2023-08-06 01:15:38 +00:00
// G_SetCustomExitVars(); [reset to defaults]
// G_SetCustomExitVars(int) [nextmap override only]
// G_SetCustomExitVars(nil, int) [skipstats only]
// G_SetCustomExitVars(int, int) [both of the above]
// G_SetCustomExitVars(int, int, int) [nextmapoverride, skipstats and nextgametype]
2019-12-13 21:36:33 +00:00
nextmapoverride = 0 ;
skipstats = 0 ;
2023-08-06 01:15:38 +00:00
nextgametype = - 1 ;
2019-12-13 21:36:33 +00:00
2014-11-12 00:55:07 +00:00
if ( n > = 1 )
{
2019-12-13 21:36:33 +00:00
nextmapoverride = ( INT16 ) luaL_optinteger ( L , 1 , 0 ) ;
skipstats = ( INT16 ) luaL_optinteger ( L , 2 , 0 ) ;
2023-08-07 17:30:39 +00:00
nextgametype = ( INT16 ) luaL_optinteger ( L , 3 , - 1 ) ;
2018-12-18 19:48:04 +00:00
}
2014-11-12 00:55:07 +00:00
2018-12-18 19:48:04 +00:00
return 0 ;
}
2019-11-15 14:35:28 +00:00
static int lib_gEnoughPlayersFinished ( lua_State * L )
{
INLEVEL
lua_pushboolean ( L , G_EnoughPlayersFinished ( ) ) ;
return 1 ;
}
2018-12-18 20:03:54 +00:00
static int lib_gExitLevel ( lua_State * L )
{
int n = lua_gettop ( L ) ; // Num arguments
NOHUD
// Moved this bit to G_SetCustomExitVars
if ( n > = 1 ) // Don't run the reset to defaults option
lib_gSetCustomExitVars ( L ) ;
2014-03-15 16:59:03 +00:00
G_ExitLevel ( ) ;
return 0 ;
}
static int lib_gIsSpecialStage ( lua_State * L )
{
2014-04-19 17:41:29 +00:00
INT32 mapnum = luaL_optinteger ( L , 1 , gamemap ) ;
2014-03-15 16:59:03 +00:00
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_IsSpecialStage ( mapnum ) ) ;
return 1 ;
}
static int lib_gGametypeUsesLives ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_GametypeUsesLives ( ) ) ;
return 1 ;
}
2019-12-27 20:08:20 +00:00
static int lib_gGametypeUsesCoopLives ( lua_State * L )
{
//HUDSAFE
INLEVEL
lua_pushboolean ( L , G_GametypeUsesCoopLives ( ) ) ;
return 1 ;
}
2019-12-27 20:14:56 +00:00
static int lib_gGametypeUsesCoopStarposts ( lua_State * L )
{
//HUDSAFE
INLEVEL
lua_pushboolean ( L , G_GametypeUsesCoopStarposts ( ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_gGametypeHasTeams ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_GametypeHasTeams ( ) ) ;
return 1 ;
}
static int lib_gGametypeHasSpectators ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_GametypeHasSpectators ( ) ) ;
return 1 ;
}
static int lib_gRingSlingerGametype ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_RingSlingerGametype ( ) ) ;
return 1 ;
}
static int lib_gPlatformGametype ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_PlatformGametype ( ) ) ;
return 1 ;
}
2020-02-29 03:57:22 +00:00
static int lib_gCoopGametype ( lua_State * L )
{
//HUDSAFE
INLEVEL
lua_pushboolean ( L , G_CoopGametype ( ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_gTagGametype ( lua_State * L )
{
//HUDSAFE
2017-01-18 22:02:28 +00:00
INLEVEL
2014-03-15 16:59:03 +00:00
lua_pushboolean ( L , G_TagGametype ( ) ) ;
return 1 ;
}
2019-12-27 20:08:20 +00:00
static int lib_gCompetitionGametype ( lua_State * L )
{
//HUDSAFE
INLEVEL
lua_pushboolean ( L , G_CompetitionGametype ( ) ) ;
return 1 ;
}
2014-03-15 16:59:03 +00:00
static int lib_gTicsToHours ( lua_State * L )
{
tic_t rtic = luaL_checkinteger ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , G_TicsToHours ( rtic ) ) ;
return 1 ;
}
static int lib_gTicsToMinutes ( lua_State * L )
{
tic_t rtic = luaL_checkinteger ( L , 1 ) ;
boolean rfull = lua_optboolean ( L , 2 ) ;
//HUDSAFE
lua_pushinteger ( L , G_TicsToMinutes ( rtic , rfull ) ) ;
return 1 ;
}
static int lib_gTicsToSeconds ( lua_State * L )
{
tic_t rtic = luaL_checkinteger ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , G_TicsToSeconds ( rtic ) ) ;
return 1 ;
}
static int lib_gTicsToCentiseconds ( lua_State * L )
{
tic_t rtic = luaL_checkinteger ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , G_TicsToCentiseconds ( rtic ) ) ;
return 1 ;
}
static int lib_gTicsToMilliseconds ( lua_State * L )
{
tic_t rtic = luaL_checkinteger ( L , 1 ) ;
//HUDSAFE
lua_pushinteger ( L , G_TicsToMilliseconds ( rtic ) ) ;
return 1 ;
}
2021-11-08 18:28:35 +00:00
static int lib_getTimeMicros ( lua_State * L )
2021-11-07 23:17:30 +00:00
{
2022-05-01 05:32:46 +00:00
lua_pushinteger ( L , I_GetPreciseTime ( ) / ( I_GetPrecisePrecision ( ) / 1000000 ) ) ;
2021-11-07 23:17:30 +00:00
return 1 ;
}
2014-03-15 16:59:03 +00:00
static luaL_Reg lib [ ] = {
{ " print " , lib_print } ,
2018-07-31 09:10:02 +00:00
{ " chatprint " , lib_chatprint } ,
{ " chatprintf " , lib_chatprintf } ,
2017-01-19 18:30:55 +00:00
{ " userdataType " , lib_userdataType } ,
2020-10-03 16:40:37 +00:00
{ " registerMetatable " , lib_registerMetatable } ,
{ " userdataMetatable " , lib_userdataMetatable } ,
2018-11-07 23:07:34 +00:00
{ " IsPlayerAdmin " , lib_isPlayerAdmin } ,
2019-08-24 17:25:27 +00:00
{ " reserveLuabanks " , lib_reserveLuabanks } ,
2024-05-01 15:26:02 +00:00
{ " tofixed " , lib_tofixed } ,
2014-03-15 16:59:03 +00:00
2020-02-15 08:18:41 +00:00
// m_menu
{ " M_MoveColorAfter " , lib_pMoveColorAfter } ,
{ " M_MoveColorBefore " , lib_pMoveColorBefore } ,
{ " M_GetColorAfter " , lib_pGetColorAfter } ,
{ " M_GetColorBefore " , lib_pGetColorBefore } ,
2021-05-27 22:35:15 +00:00
// m_misc
{ " M_MapNumber " , lib_mMapNumber } ,
2014-03-15 16:59:03 +00:00
// m_random
2016-03-27 14:33:15 +00:00
{ " P_RandomFixed " , lib_pRandomFixed } ,
{ " P_RandomByte " , lib_pRandomByte } ,
2014-03-15 16:59:03 +00:00
{ " P_RandomKey " , lib_pRandomKey } ,
{ " P_RandomRange " , lib_pRandomRange } ,
2016-03-27 14:33:15 +00:00
{ " P_SignedRandom " , lib_pSignedRandom } , // MACRO
{ " P_RandomChance " , lib_pRandomChance } , // MACRO
2014-03-15 16:59:03 +00:00
// p_maputil
{ " P_AproxDistance " , lib_pAproxDistance } ,
{ " P_ClosestPointOnLine " , lib_pClosestPointOnLine } ,
2016-05-25 15:34:09 +00:00
{ " P_PointOnLineSide " , lib_pPointOnLineSide } ,
2014-03-15 16:59:03 +00:00
// p_enemy
{ " P_CheckMeleeRange " , lib_pCheckMeleeRange } ,
{ " P_JetbCheckMeleeRange " , lib_pJetbCheckMeleeRange } ,
{ " P_FaceStabCheckMeleeRange " , lib_pFaceStabCheckMeleeRange } ,
{ " P_SkimCheckMeleeRange " , lib_pSkimCheckMeleeRange } ,
{ " P_CheckMissileRange " , lib_pCheckMissileRange } ,
{ " P_NewChaseDir " , lib_pNewChaseDir } ,
{ " P_LookForPlayers " , lib_pLookForPlayers } ,
// p_mobj
// don't add P_SetMobjState or P_SetPlayerMobjState, use "mobj.state = S_NEWSTATE" instead.
{ " P_SpawnMobj " , lib_pSpawnMobj } ,
2017-07-08 10:41:20 +00:00
{ " P_SpawnMobjFromMobj " , lib_pSpawnMobjFromMobj } ,
2014-03-15 16:59:03 +00:00
{ " P_RemoveMobj " , lib_pRemoveMobj } ,
2016-11-11 23:23:41 +00:00
{ " P_IsValidSprite2 " , lib_pIsValidSprite2 } ,
2017-03-23 19:11:22 +00:00
{ " P_SpawnLockOn " , lib_pSpawnLockOn } ,
2014-03-15 16:59:03 +00:00
{ " P_SpawnMissile " , lib_pSpawnMissile } ,
{ " P_SpawnXYZMissile " , lib_pSpawnXYZMissile } ,
{ " P_SpawnPointMissile " , lib_pSpawnPointMissile } ,
{ " P_SpawnAlteredDirectionMissile " , lib_pSpawnAlteredDirectionMissile } ,
{ " P_ColorTeamMissile " , lib_pColorTeamMissile } ,
{ " P_SPMAngle " , lib_pSPMAngle } ,
{ " P_SpawnPlayerMissile " , lib_pSpawnPlayerMissile } ,
{ " P_MobjFlip " , lib_pMobjFlip } ,
2016-06-01 12:19:44 +00:00
{ " P_GetMobjGravity " , lib_pGetMobjGravity } ,
2014-03-15 16:59:03 +00:00
{ " P_WeaponOrPanel " , lib_pWeaponOrPanel } ,
{ " P_FlashPal " , lib_pFlashPal } ,
{ " P_GetClosestAxis " , lib_pGetClosestAxis } ,
{ " P_SpawnParaloop " , lib_pSpawnParaloop } ,
{ " P_BossTargetPlayer " , lib_pBossTargetPlayer } ,
{ " P_SupermanLook4Players " , lib_pSupermanLook4Players } ,
{ " P_SetScale " , lib_pSetScale } ,
2014-08-04 03:49:33 +00:00
{ " P_InsideANonSolidFFloor " , lib_pInsideANonSolidFFloor } ,
2014-03-15 16:59:03 +00:00
{ " P_CheckDeathPitCollide " , lib_pCheckDeathPitCollide } ,
2014-08-04 03:49:33 +00:00
{ " P_CheckSolidLava " , lib_pCheckSolidLava } ,
{ " P_CanRunOnWater " , lib_pCanRunOnWater } ,
2018-04-01 19:54:19 +00:00
{ " P_MaceRotate " , lib_pMaceRotate } ,
2020-10-14 16:07:02 +00:00
{ " P_CreateFloorSpriteSlope " , lib_pCreateFloorSpriteSlope } ,
2020-11-05 03:42:14 +00:00
{ " P_RemoveFloorSpriteSlope " , lib_pRemoveFloorSpriteSlope } ,
2020-07-12 05:36:22 +00:00
{ " P_RailThinker " , lib_pRailThinker } ,
2023-06-01 04:03:22 +00:00
{ " P_CheckSkyHit " , lib_pCheckSkyHit } ,
2020-04-19 23:41:03 +00:00
{ " P_XYMovement " , lib_pXYMovement } ,
2020-04-20 01:01:29 +00:00
{ " P_RingXYMovement " , lib_pRingXYMovement } ,
2020-04-20 01:22:41 +00:00
{ " P_SceneryXYMovement " , lib_pSceneryXYMovement } ,
2020-04-20 01:41:30 +00:00
{ " P_ZMovement " , lib_pZMovement } ,
2020-04-20 03:55:01 +00:00
{ " P_RingZMovement " , lib_pRingZMovement } ,
2020-04-20 04:08:33 +00:00
{ " P_SceneryZMovement " , lib_pSceneryZMovement } ,
2020-04-20 04:15:46 +00:00
{ " P_PlayerZMovement " , lib_pPlayerZMovement } ,
2014-03-15 16:59:03 +00:00
// p_user
{ " P_GetPlayerHeight " , lib_pGetPlayerHeight } ,
{ " P_GetPlayerSpinHeight " , lib_pGetPlayerSpinHeight } ,
{ " P_GetPlayerControlDirection " , lib_pGetPlayerControlDirection } ,
{ " P_AddPlayerScore " , lib_pAddPlayerScore } ,
2015-05-27 06:08:18 +00:00
{ " P_StealPlayerScore " , lib_pStealPlayerScore } ,
2017-04-03 17:28:02 +00:00
{ " P_GetJumpFlags " , lib_pGetJumpFlags } ,
2014-03-15 16:59:03 +00:00
{ " P_PlayerInPain " , lib_pPlayerInPain } ,
{ " P_DoPlayerPain " , lib_pDoPlayerPain } ,
{ " P_ResetPlayer " , lib_pResetPlayer } ,
2019-06-19 11:09:02 +00:00
{ " P_PlayerCanDamage " , lib_pPlayerCanDamage } ,
2020-07-20 14:24:16 +00:00
{ " P_PlayerFullbright " , lib_pPlayerFullbright } ,
2014-08-04 03:49:33 +00:00
{ " P_IsObjectInGoop " , lib_pIsObjectInGoop } ,
2014-03-15 16:59:03 +00:00
{ " P_IsObjectOnGround " , lib_pIsObjectOnGround } ,
2014-08-04 03:49:33 +00:00
{ " P_InSpaceSector " , lib_pInSpaceSector } ,
{ " P_InQuicksand " , lib_pInQuicksand } ,
2022-09-09 13:35:05 +00:00
{ " P_InJumpFlipSector " , lib_pInJumpFlipSector } ,
2014-03-15 16:59:03 +00:00
{ " P_SetObjectMomZ " , lib_pSetObjectMomZ } ,
2023-06-01 03:45:00 +00:00
{ " P_IsLocalPlayer " , lib_pIsLocalPlayer } ,
2020-02-23 23:19:18 +00:00
{ " P_PlayJingle " , lib_pPlayJingle } ,
{ " P_PlayJingleMusic " , lib_pPlayJingleMusic } ,
2014-03-15 16:59:03 +00:00
{ " P_RestoreMusic " , lib_pRestoreMusic } ,
2014-08-04 03:49:33 +00:00
{ " P_SpawnShieldOrb " , lib_pSpawnShieldOrb } ,
2014-03-15 16:59:03 +00:00
{ " P_SpawnGhostMobj " , lib_pSpawnGhostMobj } ,
{ " P_GivePlayerRings " , lib_pGivePlayerRings } ,
2023-06-01 03:39:08 +00:00
{ " P_GivePlayerSpheres " , lib_pGivePlayerSpheres } ,
2014-03-15 16:59:03 +00:00
{ " P_GivePlayerLives " , lib_pGivePlayerLives } ,
2017-05-29 21:18:02 +00:00
{ " P_GiveCoopLives " , lib_pGiveCoopLives } ,
2014-03-15 16:59:03 +00:00
{ " P_ResetScore " , lib_pResetScore } ,
2014-08-04 03:49:33 +00:00
{ " P_DoJumpShield " , lib_pDoJumpShield } ,
2016-10-24 12:33:10 +00:00
{ " P_DoBubbleBounce " , lib_pDoBubbleBounce } ,
2014-03-15 16:59:03 +00:00
{ " P_BlackOw " , lib_pBlackOw } ,
2016-07-08 21:56:17 +00:00
{ " P_ElementalFire " , lib_pElementalFire } ,
2020-07-03 17:54:00 +00:00
{ " P_SpawnSkidDust " , lib_pSpawnSkidDust } ,
2020-05-23 14:50:35 +00:00
{ " P_MovePlayer " , lib_pMovePlayer } ,
2019-11-15 14:35:28 +00:00
{ " P_DoPlayerFinish " , lib_pDoPlayerFinish } ,
2014-03-15 16:59:03 +00:00
{ " P_DoPlayerExit " , lib_pDoPlayerExit } ,
{ " P_InstaThrust " , lib_pInstaThrust } ,
2021-05-01 18:05:07 +00:00
{ " P_InstaThrustEvenIn2D " , lib_pInstaThrustEvenIn2D } ,
2014-03-15 16:59:03 +00:00
{ " P_ReturnThrustX " , lib_pReturnThrustX } ,
{ " P_ReturnThrustY " , lib_pReturnThrustY } ,
{ " P_LookForEnemies " , lib_pLookForEnemies } ,
{ " P_NukeEnemies " , lib_pNukeEnemies } ,
2020-07-03 17:54:00 +00:00
{ " P_Earthquake " , lib_pEarthquake } ,
2014-03-15 16:59:03 +00:00
{ " P_HomingAttack " , lib_pHomingAttack } ,
2023-06-03 04:04:18 +00:00
{ " P_ResetCamera " , lib_pResetCamera } ,
2014-04-14 05:14:58 +00:00
{ " P_SuperReady " , lib_pSuperReady } ,
{ " P_DoJump " , lib_pDoJump } ,
2023-06-01 03:49:30 +00:00
{ " P_DoSpinDashDust " , lib_pDoSpinDashDust } ,
2014-05-23 20:46:49 +00:00
{ " P_SpawnThokMobj " , lib_pSpawnThokMobj } ,
2014-05-23 22:08:35 +00:00
{ " P_SpawnSpinMobj " , lib_pSpawnSpinMobj } ,
2014-05-26 02:41:05 +00:00
{ " P_Telekinesis " , lib_pTelekinesis } ,
2017-08-10 12:57:09 +00:00
{ " P_SwitchShield " , lib_pSwitchShield } ,
2023-06-01 03:28:53 +00:00
{ " P_DoTailsOverlay " , lib_pDoTailsOverlay } ,
{ " P_DoMetalJetFume " , lib_pDoMetalJetFume } ,
{ " P_DoFollowMobj " , lib_pDoFollowMobj } ,
2021-03-23 03:49:22 +00:00
{ " P_PlayerCanEnterSpinGaps " , lib_pPlayerCanEnterSpinGaps } ,
{ " P_PlayerShouldUseSpinHeight " , lib_pPlayerShouldUseSpinHeight } ,
2014-03-15 16:59:03 +00:00
// p_map
{ " P_CheckPosition " , lib_pCheckPosition } ,
{ " P_TryMove " , lib_pTryMove } ,
{ " P_Move " , lib_pMove } ,
{ " P_TeleportMove " , lib_pTeleportMove } ,
2021-11-29 13:20:27 +00:00
{ " P_SetOrigin " , lib_pSetOrigin } ,
{ " P_MoveOrigin " , lib_pMoveOrigin } ,
2023-09-17 23:45:57 +00:00
{ " P_LineIsBlocking " , lib_pLineIsBlocking } ,
2014-03-15 16:59:03 +00:00
{ " P_SlideMove " , lib_pSlideMove } ,
{ " P_BounceMove " , lib_pBounceMove } ,
{ " P_CheckSight " , lib_pCheckSight } ,
{ " P_CheckHoopPosition " , lib_pCheckHoopPosition } ,
{ " P_RadiusAttack " , lib_pRadiusAttack } ,
{ " P_FloorzAtPos " , lib_pFloorzAtPos } ,
2020-09-30 07:10:28 +00:00
{ " P_CeilingzAtPos " , lib_pCeilingzAtPos } ,
2023-11-30 22:50:22 +00:00
{ " P_GetSectorLightLevelAt " , lib_pGetSectorLightLevelAt } ,
2023-08-04 03:31:51 +00:00
{ " P_GetSectorColormapAt " , lib_pGetSectorColormapAt } ,
2014-05-23 22:55:07 +00:00
{ " P_DoSpring " , lib_pDoSpring } ,
2023-06-01 03:21:31 +00:00
{ " P_TouchSpecialThing " , lib_pTouchSpecialThing } ,
2021-07-11 06:23:50 +00:00
{ " P_TryCameraMove " , lib_pTryCameraMove } ,
{ " P_TeleportCameraMove " , lib_pTeleportCameraMove } ,
2014-03-15 16:59:03 +00:00
// p_inter
{ " P_RemoveShield " , lib_pRemoveShield } ,
{ " P_DamageMobj " , lib_pDamageMobj } ,
{ " P_KillMobj " , lib_pKillMobj } ,
{ " P_PlayerRingBurst " , lib_pPlayerRingBurst } ,
{ " P_PlayerWeaponPanelBurst " , lib_pPlayerWeaponPanelBurst } ,
{ " P_PlayerWeaponAmmoBurst " , lib_pPlayerWeaponAmmoBurst } ,
2015-05-27 04:45:34 +00:00
{ " P_PlayerWeaponPanelOrAmmoBurst " , lib_pPlayerWeaponPanelOrAmmoBurst } ,
2014-03-15 16:59:03 +00:00
{ " P_PlayerEmeraldBurst " , lib_pPlayerEmeraldBurst } ,
{ " P_PlayerFlagBurst " , lib_pPlayerFlagBurst } ,
{ " P_PlayRinglossSound " , lib_pPlayRinglossSound } ,
{ " P_PlayDeathSound " , lib_pPlayDeathSound } ,
{ " P_PlayVictorySound " , lib_pPlayVictorySound } ,
{ " P_PlayLivesJingle " , lib_pPlayLivesJingle } ,
{ " P_CanPickupItem " , lib_pCanPickupItem } ,
{ " P_DoNightsScore " , lib_pDoNightsScore } ,
2015-05-27 06:08:18 +00:00
{ " P_DoMatchSuper " , lib_pDoMatchSuper } ,
2014-03-15 16:59:03 +00:00
// p_spec
{ " P_Thrust " , lib_pThrust } ,
2021-05-01 18:05:07 +00:00
{ " P_ThrustEvenIn2D " , lib_pThrustEvenIn2D } ,
{ " P_VectorInstaThrust " , lib_pVectorInstaThrust } ,
2014-03-15 16:59:03 +00:00
{ " P_SetMobjStateNF " , lib_pSetMobjStateNF } ,
{ " P_DoSuperTransformation " , lib_pDoSuperTransformation } ,
2024-02-24 14:41:36 +00:00
{ " P_DoSuperDetransformation " , lib_pDoSuperDetransformation } ,
2014-03-15 16:59:03 +00:00
{ " P_ExplodeMissile " , lib_pExplodeMissile } ,
2021-12-08 20:07:26 +00:00
{ " P_MobjTouchingSectorSpecial " , lib_pMobjTouchingSectorSpecial } ,
2023-01-01 10:31:44 +00:00
{ " P_ThingOnSpecial3DFloor " , lib_pThingOnSpecial3DFloor } ,
2021-12-30 23:03:24 +00:00
{ " P_MobjTouchingSectorSpecialFlag " , lib_pMobjTouchingSectorSpecialFlag } ,
2014-03-15 16:59:03 +00:00
{ " P_PlayerTouchingSectorSpecial " , lib_pPlayerTouchingSectorSpecial } ,
2021-12-30 23:03:24 +00:00
{ " P_PlayerTouchingSectorSpecialFlag " , lib_pPlayerTouchingSectorSpecialFlag } ,
2017-10-30 20:09:41 +00:00
{ " P_FindLowestFloorSurrounding " , lib_pFindLowestFloorSurrounding } ,
{ " P_FindHighestFloorSurrounding " , lib_pFindHighestFloorSurrounding } ,
{ " P_FindNextHighestFloor " , lib_pFindNextHighestFloor } ,
{ " P_FindNextLowestFloor " , lib_pFindNextLowestFloor } ,
{ " P_FindLowestCeilingSurrounding " , lib_pFindLowestCeilingSurrounding } ,
{ " P_FindHighestCeilingSurrounding " , lib_pFindHighestCeilingSurrounding } ,
2014-03-15 16:59:03 +00:00
{ " P_FindSpecialLineFromTag " , lib_pFindSpecialLineFromTag } ,
{ " P_SwitchWeather " , lib_pSwitchWeather } ,
{ " P_LinedefExecute " , lib_pLinedefExecute } ,
{ " P_SpawnLightningFlash " , lib_pSpawnLightningFlash } ,
{ " P_FadeLight " , lib_pFadeLight } ,
2014-05-26 02:41:05 +00:00
{ " P_IsFlagAtBase " , lib_pIsFlagAtBase } ,
2014-03-15 16:59:03 +00:00
{ " P_SetupLevelSky " , lib_pSetupLevelSky } ,
{ " P_SetSkyboxMobj " , lib_pSetSkyboxMobj } ,
{ " P_StartQuake " , lib_pStartQuake } ,
2014-08-04 03:49:33 +00:00
{ " EV_CrumbleChain " , lib_evCrumbleChain } ,
2016-07-01 19:27:09 +00:00
{ " EV_StartCrumble " , lib_evStartCrumble } ,
2014-03-15 16:59:03 +00:00
2018-10-20 20:08:59 +00:00
// p_slopes
{ " P_GetZAt " , lib_pGetZAt } ,
2021-02-11 12:06:40 +00:00
{ " P_ButteredSlope " , lib_pButteredSlope } ,
2018-10-20 20:08:59 +00:00
2014-03-15 16:59:03 +00:00
// r_defs
{ " R_PointToAngle " , lib_rPointToAngle } ,
{ " R_PointToAngle2 " , lib_rPointToAngle2 } ,
{ " R_PointToDist " , lib_rPointToDist } ,
{ " R_PointToDist2 " , lib_rPointToDist2 } ,
{ " R_PointInSubsector " , lib_rPointInSubsector } ,
2020-01-22 15:58:57 +00:00
{ " R_PointInSubsectorOrNil " , lib_rPointInSubsectorOrNil } ,
2014-03-15 16:59:03 +00:00
2014-04-14 05:14:58 +00:00
// r_things (sprite)
{ " R_Char2Frame " , lib_rChar2Frame } ,
{ " R_Frame2Char " , lib_rFrame2Char } ,
2014-08-04 03:49:33 +00:00
{ " R_SetPlayerSkin " , lib_rSetPlayerSkin } ,
2023-11-13 15:13:44 +00:00
// r_skins
2019-11-10 16:10:34 +00:00
{ " R_SkinUsable " , lib_rSkinUsable } ,
2023-11-13 15:13:44 +00:00
{ " P_GetStateSprite2 " , lib_pGetStateSprite2 } ,
{ " P_GetSprite2StateFrame " , lib_pGetSprite2StateFrame } ,
{ " P_IsStateSprite2Super " , lib_pIsStateSprite2Super } ,
2023-11-13 15:19:03 +00:00
{ " P_GetSuperSprite2 " , lib_pGetSuperSprite2 } ,
2014-04-14 05:14:58 +00:00
2018-12-16 20:58:47 +00:00
// r_data
2018-12-16 23:19:53 +00:00
{ " R_CheckTextureNumForName " , lib_rCheckTextureNumForName } ,
{ " R_TextureNumForName " , lib_rTextureNumForName } ,
2022-11-20 20:25:53 +00:00
{ " R_CheckTextureNameForNum " , lib_rCheckTextureNameForNum } ,
{ " R_TextureNameForNum " , lib_rTextureNameForNum } ,
2018-12-16 20:58:47 +00:00
2020-02-15 22:35:00 +00:00
// r_draw
{ " R_GetColorByName " , lib_rGetColorByName } ,
2020-07-12 11:39:52 +00:00
{ " R_GetSuperColorByName " , lib_rGetSuperColorByName } ,
2020-02-15 22:35:00 +00:00
{ " R_GetNameByColor " , lib_rGetNameByColor } ,
2014-03-15 16:59:03 +00:00
// s_sound
{ " S_StartSound " , lib_sStartSound } ,
{ " S_StartSoundAtVolume " , lib_sStartSoundAtVolume } ,
{ " S_StopSound " , lib_sStopSound } ,
2020-05-06 13:22:04 +00:00
{ " S_StopSoundByID " , lib_sStopSoundByID } ,
2014-03-15 16:59:03 +00:00
{ " S_ChangeMusic " , lib_sChangeMusic } ,
{ " S_SpeedMusic " , lib_sSpeedMusic } ,
{ " S_StopMusic " , lib_sStopMusic } ,
2019-01-02 09:58:03 +00:00
{ " S_SetInternalMusicVolume " , lib_sSetInternalMusicVolume } ,
{ " S_StopFadingMusic " , lib_sStopFadingMusic } ,
{ " S_FadeMusic " , lib_sFadeMusic } ,
{ " S_FadeOutStopMusic " , lib_sFadeOutStopMusic } ,
2020-02-23 23:58:32 +00:00
{ " S_GetMusicLength " , lib_sGetMusicLength } ,
{ " S_GetMusicPosition " , lib_sGetMusicPosition } ,
{ " S_SetMusicPosition " , lib_sSetMusicPosition } ,
2014-03-15 16:59:03 +00:00
{ " S_OriginPlaying " , lib_sOriginPlaying } ,
{ " S_IdPlaying " , lib_sIdPlaying } ,
{ " S_SoundPlaying " , lib_sSoundPlaying } ,
2017-04-29 15:27:52 +00:00
{ " S_StartMusicCaption " , lib_sStartMusicCaption } ,
2020-11-01 01:15:41 +00:00
{ " S_MusicType " , lib_sMusicType } ,
{ " S_MusicPlaying " , lib_sMusicPlaying } ,
{ " S_MusicPaused " , lib_sMusicPaused } ,
{ " S_MusicName " , lib_sMusicName } ,
{ " S_MusicExists " , lib_sMusicExists } ,
{ " S_SetMusicLoopPoint " , lib_sSetMusicLoopPoint } ,
{ " S_GetMusicLoopPoint " , lib_sGetMusicLoopPoint } ,
{ " S_PauseMusic " , lib_sPauseMusic } ,
{ " S_ResumeMusic " , lib_sResumeMusic } ,
2014-03-15 16:59:03 +00:00
// g_game
2019-12-19 05:13:09 +00:00
{ " G_AddGametype " , lib_gAddGametype } ,
2021-01-15 22:45:41 +00:00
{ " G_AddPlayer " , lib_gAddPlayer } ,
2021-01-24 01:01:48 +00:00
{ " G_RemovePlayer " , lib_gRemovePlayer } ,
2022-02-28 17:43:00 +00:00
{ " G_SetUsedCheats " , lib_gSetUsedCheats } ,
2014-03-15 16:59:03 +00:00
{ " G_BuildMapName " , lib_gBuildMapName } ,
2020-01-16 04:04:50 +00:00
{ " G_BuildMapTitle " , lib_gBuildMapTitle } ,
2020-01-16 02:49:22 +00:00
{ " G_FindMap " , lib_gFindMap } ,
2020-01-16 04:32:40 +00:00
{ " G_FindMapByNameOrCode " , lib_gFindMapByNameOrCode } ,
2014-03-15 16:59:03 +00:00
{ " G_DoReborn " , lib_gDoReborn } ,
2018-12-18 19:48:04 +00:00
{ " G_SetCustomExitVars " , lib_gSetCustomExitVars } ,
2019-11-15 14:35:28 +00:00
{ " G_EnoughPlayersFinished " , lib_gEnoughPlayersFinished } ,
2014-03-15 16:59:03 +00:00
{ " G_ExitLevel " , lib_gExitLevel } ,
{ " G_IsSpecialStage " , lib_gIsSpecialStage } ,
{ " G_GametypeUsesLives " , lib_gGametypeUsesLives } ,
2019-12-27 20:08:20 +00:00
{ " G_GametypeUsesCoopLives " , lib_gGametypeUsesCoopLives } ,
2019-12-27 20:14:56 +00:00
{ " G_GametypeUsesCoopStarposts " , lib_gGametypeUsesCoopStarposts } ,
2014-03-15 16:59:03 +00:00
{ " G_GametypeHasTeams " , lib_gGametypeHasTeams } ,
{ " G_GametypeHasSpectators " , lib_gGametypeHasSpectators } ,
{ " G_RingSlingerGametype " , lib_gRingSlingerGametype } ,
{ " G_PlatformGametype " , lib_gPlatformGametype } ,
2020-02-29 03:57:22 +00:00
{ " G_CoopGametype " , lib_gCoopGametype } ,
2014-03-15 16:59:03 +00:00
{ " G_TagGametype " , lib_gTagGametype } ,
2019-12-27 20:08:20 +00:00
{ " G_CompetitionGametype " , lib_gCompetitionGametype } ,
2014-03-15 16:59:03 +00:00
{ " G_TicsToHours " , lib_gTicsToHours } ,
{ " G_TicsToMinutes " , lib_gTicsToMinutes } ,
{ " G_TicsToSeconds " , lib_gTicsToSeconds } ,
{ " G_TicsToCentiseconds " , lib_gTicsToCentiseconds } ,
{ " G_TicsToMilliseconds " , lib_gTicsToMilliseconds } ,
2021-11-08 18:28:35 +00:00
{ " getTimeMicros " , lib_getTimeMicros } ,
2021-11-07 23:17:30 +00:00
2014-03-15 16:59:03 +00:00
{ NULL , NULL }
} ;
int LUA_BaseLib ( lua_State * L )
{
// Set metatable for string
lua_pushliteral ( L , " " ) ; // dummy string
lua_getmetatable ( L , - 1 ) ; // get string metatable
2023-10-28 09:06:30 +00:00
LUA_SetCFunctionField ( L , " __add " , lib_concat ) ;
2014-03-15 16:59:03 +00:00
lua_pop ( L , 2 ) ; // pop metatable and dummy string
lua_newtable ( L ) ;
lua_setfield ( L , LUA_REGISTRYINDEX , LREG_EXTVARS ) ;
// Set global functions
lua_pushvalue ( L , LUA_GLOBALSINDEX ) ;
luaL_register ( L , NULL , lib ) ;
return 0 ;
}