2020-05-17 06:51:49 +00:00
//-------------------------------------------------------------------------
/*
Copyright ( C ) 1996 , 2003 - 3 D Realms Entertainment
Copyright ( C ) 2000 , 2003 - Matt Saettler ( EDuke Enhancements )
2020-06-28 07:03:31 +00:00
Copyright ( C ) 2020 - Christoph Oelckers
2020-05-17 06:51:49 +00:00
This file is part of Enhanced Duke Nukem 3 D version 1.5 - Atomic Edition
Duke Nukem 3 D is free software ; you can redistribute it and / or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation ; either version 2
of the License , or ( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
See the GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
2020-05-17 11:51:18 +00:00
Foundation , Inc . , 59 Temple Place - Suite 330 , Boston , MA 02111 - 1307 , USA .
2020-05-17 06:51:49 +00:00
Original Source : 1996 - Todd Replogle
Prepared for public release : 03 / 21 / 2003 - Charlie Wiederhold , 3 D Realms
EDuke enhancements integrated : 04 / 13 / 2003 - Matt Saettler
Note : EDuke source was in transition . Changes are in - progress in the
source as it is released .
*/
//-------------------------------------------------------------------------
# include "ns.h"
# include "global.h"
2020-08-24 18:34:18 +00:00
# include "gamecontrol.h"
2020-08-31 17:18:53 +00:00
# include "v_video.h"
2020-05-17 06:51:49 +00:00
2020-07-17 18:56:10 +00:00
BEGIN_DUKE_NS
2020-05-17 06:51:49 +00:00
2020-07-18 11:27:24 +00:00
// State timer counters.
static int nonsharedtimer ;
static int turnheldtime ;
static int lastcontroltime ;
static double lastCheck ;
2020-08-29 19:20:10 +00:00
static InputPacket loc ; // input accumulation buffer.
2020-07-18 11:27:24 +00:00
2020-08-24 18:20:15 +00:00
void GameInterface : : ResetFollowPos ( bool message )
{
if ( automapFollow )
{
ud . folx = ps [ screenpeek ] . oposx ;
ud . foly = ps [ screenpeek ] . oposy ;
ud . fola = ps [ screenpeek ] . getoang ( ) ;
}
if ( message ) FTA ( automapFollow ? QUOTE_MAP_FOLLOW_ON : QUOTE_MAP_FOLLOW_OFF , & ps [ myconnectindex ] ) ;
}
2020-07-15 17:48:04 +00:00
//---------------------------------------------------------------------------
//
2020-08-31 17:18:53 +00:00
//
2020-07-15 17:48:04 +00:00
//
//---------------------------------------------------------------------------
void nonsharedkeys ( void )
{
2020-08-31 17:18:53 +00:00
int ms = screen - > FrameTime ;
int interval ;
if ( nonsharedtimer > 0 | | ms < nonsharedtimer )
{
interval = ms - nonsharedtimer ;
}
else
{
interval = 0 ;
}
nonsharedtimer = screen - > FrameTime ;
2020-07-15 17:48:04 +00:00
if ( System_WantGuiCapture ( ) )
return ;
2020-08-24 18:34:18 +00:00
if ( automapMode ! = am_off )
2020-07-15 17:48:04 +00:00
{
2020-08-31 17:18:53 +00:00
double j = interval * ( 120. / 1000 ) ;
2020-08-24 18:34:18 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Enlarge_Screen ) )
2020-08-31 17:18:53 +00:00
ps [ myconnectindex ] . zoom + = ( int ) fmulscale6 ( j , max ( ps [ myconnectindex ] . zoom , 256 ) ) ;
2020-08-24 18:34:18 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Shrink_Screen ) )
2020-08-31 17:18:53 +00:00
ps [ myconnectindex ] . zoom - = ( int ) fmulscale6 ( j , max ( ps [ myconnectindex ] . zoom , 256 ) ) ;
2020-08-24 18:34:18 +00:00
ps [ myconnectindex ] . zoom = clamp ( ps [ myconnectindex ] . zoom , 48 , 2048 ) ;
2020-07-15 17:48:04 +00:00
}
}
2020-05-17 06:51:49 +00:00
//---------------------------------------------------------------------------
//
2020-05-17 11:25:39 +00:00
// handles all HUD related input, i.e. inventory item selection and activation plus weapon selection.
//
// Note: This doesn't restrict the events to WW2GI - since the other games do
// not define any by default there is no harm done keeping the code clean.
2020-05-17 06:51:49 +00:00
//
//---------------------------------------------------------------------------
2020-05-17 11:25:39 +00:00
void hud_input ( int snum )
{
int i , k ;
uint8_t dainv ;
struct player_struct * p ;
short unk ;
unk = 0 ;
p = & ps [ snum ] ;
i = p - > aim_mode ;
2020-08-31 18:51:22 +00:00
p - > aim_mode = ! PlayerInput ( snum , SB_AIMMODE ) ;
2020-05-17 11:25:39 +00:00
if ( p - > aim_mode < i )
p - > return_to_center = 9 ;
2020-08-03 03:28:16 +00:00
// Backup weapon here as hud_input() is the first function where any one of the weapon variables can change.
backupweapon ( p ) ;
2020-08-02 13:00:53 +00:00
2020-05-17 11:25:39 +00:00
if ( isRR ( ) )
{
2020-08-29 11:32:14 +00:00
if ( PlayerInput ( snum , SB_QUICK_KICK ) & & p - > last_pissed_time = = 0 )
2020-05-17 11:25:39 +00:00
{
if ( ! isRRRA ( ) | | sprite [ p - > i ] . extra > 0 )
{
p - > last_pissed_time = 4000 ;
2020-07-29 20:43:06 +00:00
S_PlayActorSound ( 437 , p - > i ) ;
2020-05-17 11:25:39 +00:00
if ( sprite [ p - > i ] . extra < = max_player_health - max_player_health / 10 )
{
sprite [ p - > i ] . extra + = 2 ;
p - > last_extra = sprite [ p - > i ] . extra ;
}
else if ( sprite [ p - > i ] . extra < max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
}
}
}
else
{
2020-08-29 11:32:14 +00:00
if ( PlayerInput ( snum , SB_QUICK_KICK ) & & p - > quick_kick = = 0 & & ( p - > curr_weapon ! = KNEE_WEAPON | | p - > kickback_pic = = 0 ) )
2020-05-17 11:25:39 +00:00
{
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_QUICKKICK , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
{
p - > quick_kick = 14 ;
2020-07-28 18:43:33 +00:00
if ( ! p - > quick_kick_msg & & snum = = screenpeek ) FTA ( QUOTE_MIGHTY_FOOT , p ) ;
p - > quick_kick_msg = true ;
2020-05-17 11:25:39 +00:00
}
}
}
2020-08-29 11:32:14 +00:00
if ( ! PlayerInput ( snum , SB_QUICK_KICK ) ) p - > quick_kick_msg = false ;
2020-05-17 11:25:39 +00:00
2020-08-29 11:32:14 +00:00
if ( ! PlayerInputBits ( snum , SB_INTERFACE_BITS ) )
2020-05-17 11:25:39 +00:00
p - > interface_toggle_flag = 0 ;
else if ( p - > interface_toggle_flag = = 0 )
{
p - > interface_toggle_flag = 1 ;
2020-05-17 11:51:18 +00:00
// Don't go on if paused or dead.
2020-07-15 16:10:31 +00:00
if ( paused ) return ;
2020-05-17 11:51:18 +00:00
if ( sprite [ p - > i ] . extra < = 0 ) return ;
// Activate an inventory item. This just forwards to the other inventory bits. If the inventory selector was taken out of the playsim this could be removed.
2020-08-27 20:19:24 +00:00
if ( PlayerInput ( snum , SB_INVUSE ) & & p - > newowner = = - 1 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_INVENTORY , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-08-27 19:25:09 +00:00
if ( p - > inven_icon > ICON_NONE & & p - > inven_icon < = ICON_HEATS ) PlayerSetItemUsed ( snum , p - > inven_icon ) ;
2020-05-17 11:25:39 +00:00
}
}
2020-08-27 19:25:09 +00:00
if ( ! isRR ( ) & & PlayerUseItem ( snum , ICON_HEATS ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_USENIGHTVISION , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 & & p - > heat_amount > 0 )
{
p - > heat_on = ! p - > heat_on ;
setpal ( p ) ;
p - > inven_icon = 5 ;
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( NITEVISION_ONOFF , p - > i ) ;
2020-05-17 11:51:18 +00:00
FTA ( 106 + ( ! p - > heat_on ) , p ) ;
}
2020-05-17 11:25:39 +00:00
}
2020-08-27 19:25:09 +00:00
if ( PlayerUseItem ( snum , ICON_STEROIDS ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_USESTEROIDS , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > steroids_amount = = 400 )
{
p - > steroids_amount - - ;
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( DUKE_TAKEPILLS , p - > i ) ;
2020-05-17 11:51:18 +00:00
p - > inven_icon = ICON_STEROIDS ;
FTA ( 12 , p ) ;
}
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
return ;
2020-05-17 11:25:39 +00:00
}
2020-08-27 20:19:24 +00:00
if ( PlayerInput ( snum , SB_INVPREV ) | | PlayerInput ( snum , SB_INVNEXT ) )
2020-05-17 11:51:18 +00:00
{
p - > invdisptime = 26 * 2 ;
2020-05-17 11:25:39 +00:00
2020-08-27 20:19:24 +00:00
if ( PlayerInput ( snum , SB_INVNEXT ) ) k = 1 ;
2020-05-17 11:51:18 +00:00
else k = 0 ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
dainv = p - > inven_icon ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
i = 0 ;
CHECKINV1 :
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
if ( i < 9 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
i + + ;
switch ( dainv )
{
case 4 :
if ( p - > jetpack_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 5 ;
else dainv = 3 ;
goto CHECKINV1 ;
case 6 :
if ( p - > scuba_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 7 ;
else dainv = 5 ;
goto CHECKINV1 ;
case 2 :
if ( p - > steroids_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 3 ;
else dainv = 1 ;
goto CHECKINV1 ;
case 3 :
if ( p - > holoduke_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 4 ;
else dainv = 2 ;
goto CHECKINV1 ;
case 0 :
case 1 :
if ( p - > firstaid_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 2 ;
else dainv = 7 ;
goto CHECKINV1 ;
case 5 :
if ( p - > heat_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 6 ;
else dainv = 4 ;
goto CHECKINV1 ;
case 7 :
if ( p - > boot_amount > 0 & & i > 1 )
break ;
if ( k ) dainv = 1 ;
else dainv = 6 ;
goto CHECKINV1 ;
}
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else dainv = 0 ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
// These events force us to keep the inventory selector in the playsim as opposed to the UI where it really belongs.
2020-08-27 20:19:24 +00:00
if ( PlayerInput ( snum , SB_INVPREV ) )
2020-05-17 11:51:18 +00:00
{
SetGameVarID ( g_iReturnVarID , dainv , - 1 , snum ) ;
OnEvent ( EVENT_INVENTORYLEFT , - 1 , snum , - 1 ) ;
dainv = GetGameVarID ( g_iReturnVarID , - 1 , snum ) ;
}
2020-08-27 20:19:24 +00:00
if ( PlayerInput ( snum , SB_INVNEXT ) )
2020-05-17 11:51:18 +00:00
{
SetGameVarID ( g_iReturnVarID , dainv , - 1 , snum ) ;
OnEvent ( EVENT_INVENTORYRIGHT , - 1 , snum , - 1 ) ;
dainv = GetGameVarID ( g_iReturnVarID , - 1 , snum ) ;
}
p - > inven_icon = dainv ;
// Someone must have really hated constant data, doing this with a switch/case (and of course also with literal numbers...)
static const uint8_t invquotes [ ] = { QUOTE_MEDKIT , QUOTE_STEROIDS , QUOTE_HOLODUKE , QUOTE_JETPACK , QUOTE_NVG , QUOTE_SCUBA , QUOTE_BOOTS } ;
if ( dainv > = 1 & & dainv < 8 ) FTA ( invquotes [ dainv - 1 ] , p ) ;
2020-05-17 11:25:39 +00:00
}
2020-08-26 22:53:35 +00:00
int weap = PlayerNewWeapon ( snum ) ;
if ( weap > 1 & & p - > kickback_pic > 0 )
p - > wantweaponfire = weap - 1 ;
2020-05-17 11:25:39 +00:00
// Here we have to be extra careful that the weapons do not get mixed up, so let's keep the code for Duke and RR completely separate.
2020-08-26 22:53:35 +00:00
fi . selectweapon ( snum , weap ) ;
2020-05-17 11:25:39 +00:00
2020-08-27 22:03:35 +00:00
if ( PlayerInput ( snum , SB_HOLSTER ) )
2020-05-17 11:25:39 +00:00
{
if ( p - > curr_weapon > KNEE_WEAPON )
{
if ( p - > holster_weapon = = 0 & & p - > weapon_pos = = 0 )
{
p - > holster_weapon = 1 ;
p - > weapon_pos = - 1 ;
FTA ( QUOTE_WEAPON_LOWERED , p ) ;
}
else if ( p - > holster_weapon = = 1 & & p - > weapon_pos = = - 9 )
{
p - > holster_weapon = 0 ;
p - > weapon_pos = 10 ;
FTA ( QUOTE_WEAPON_RAISED , p ) ;
}
}
}
2020-08-27 19:25:09 +00:00
if ( PlayerUseItem ( snum , ICON_HOLODUKE ) & & ( isRR ( ) | | p - > newowner = = - 1 ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_HOLODUKEON , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( ! isRR ( ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > holoduke_on = = - 1 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > holoduke_amount > 0 )
{
p - > inven_icon = 3 ;
p - > holoduke_on = i =
EGS ( p - > cursectnum ,
p - > posx ,
p - > posy ,
p - > posz + ( 30 < < 8 ) , TILE_APLAYER , - 64 , 0 , 0 , p - > getang ( ) , 0 , 0 , - 1 , 10 ) ;
hittype [ i ] . temp_data [ 3 ] = hittype [ i ] . temp_data [ 4 ] = 0 ;
sprite [ i ] . yvel = snum ;
sprite [ i ] . extra = 0 ;
2020-08-31 18:01:55 +00:00
FTA ( QUOTE_HOLODUKE_ON , p ) ;
2020-08-31 18:04:20 +00:00
S_PlayActorSound ( TELEPORTER , p - > holoduke_on ) ;
2020-05-17 11:51:18 +00:00
}
2020-08-31 18:01:55 +00:00
else FTA ( QUOTE_HOLODUKE_NOT_FOUND , p ) ;
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else
{
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( TELEPORTER , p - > holoduke_on ) ;
2020-05-17 11:51:18 +00:00
p - > holoduke_on = - 1 ;
2020-08-31 18:01:55 +00:00
FTA ( QUOTE_HOLODUKE_OFF , p ) ;
2020-05-17 11:51:18 +00:00
}
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else // In RR this means drinking whiskey.
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > holoduke_amount > 0 & & sprite [ p - > i ] . extra < max_player_health )
{
p - > holoduke_amount - = 400 ;
sprite [ p - > i ] . extra + = 5 ;
if ( sprite [ p - > i ] . extra > max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
p - > drink_amt + = 5 ;
p - > inven_icon = 3 ;
if ( p - > holoduke_amount = = 0 )
checkavailinven ( p ) ;
2020-07-25 07:32:54 +00:00
if ( p - > drink_amt < 99 & & ! S_CheckActorSoundPlaying ( p - > i , 425 ) )
S_PlayActorSound ( 425 , p - > i ) ;
2020-05-17 11:51:18 +00:00
}
2020-05-17 11:25:39 +00:00
}
}
}
2020-08-27 19:25:09 +00:00
if ( isRR ( ) & & PlayerUseItem ( snum , ICON_HEATS ) & & p - > newowner = = - 1 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_USENIGHTVISION , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > yehaa_timer = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
p - > yehaa_timer = 126 ;
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( 390 , p - > i ) ;
2020-05-17 11:51:18 +00:00
p - > noise_radius = 16384 ;
madenoise ( snum ) ;
if ( sector [ p - > cursectnum ] . lotag = = 857 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( sprite [ p - > i ] . extra < = max_player_health )
{
sprite [ p - > i ] . extra + = 10 ;
if ( sprite [ p - > i ] . extra > = max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
}
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( sprite [ p - > i ] . extra + 1 < = max_player_health )
{
sprite [ p - > i ] . extra + + ;
}
2020-05-17 11:25:39 +00:00
}
}
}
}
2020-08-27 19:25:09 +00:00
if ( PlayerUseItem ( snum , ICON_FIRSTAID ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_USEMEDKIT , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > firstaid_amount > 0 & & sprite [ p - > i ] . extra < max_player_health )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( ! isRR ( ) )
2020-05-17 11:25:39 +00:00
{
2020-08-26 22:53:35 +00:00
int j = max_player_health - sprite [ p - > i ] . extra ;
2020-05-17 11:51:18 +00:00
if ( ( unsigned int ) p - > firstaid_amount > j )
{
p - > firstaid_amount - = j ;
sprite [ p - > i ] . extra = max_player_health ;
p - > inven_icon = 1 ;
}
else
{
sprite [ p - > i ] . extra + = p - > firstaid_amount ;
p - > firstaid_amount = 0 ;
checkavailinven ( p ) ;
}
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( DUKE_USEMEDKIT , p - > i ) ;
2020-05-17 11:25:39 +00:00
}
else
{
2020-08-26 22:53:35 +00:00
int j = 10 ;
2020-05-17 11:51:18 +00:00
if ( p - > firstaid_amount > j )
{
p - > firstaid_amount - = j ;
sprite [ p - > i ] . extra + = j ;
if ( sprite [ p - > i ] . extra > max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
p - > inven_icon = 1 ;
}
else
{
sprite [ p - > i ] . extra + = p - > firstaid_amount ;
p - > firstaid_amount = 0 ;
checkavailinven ( p ) ;
}
2020-05-17 11:25:39 +00:00
if ( sprite [ p - > i ] . extra > max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
2020-05-17 11:51:18 +00:00
p - > drink_amt + = 10 ;
2020-07-25 07:32:54 +00:00
if ( p - > drink_amt < = 100 & & ! S_CheckActorSoundPlaying ( p - > i , DUKE_USEMEDKIT ) )
S_PlayActorSound ( DUKE_USEMEDKIT , p - > i ) ;
2020-05-17 11:25:39 +00:00
}
}
}
}
2020-08-27 19:25:09 +00:00
if ( PlayerUseItem ( snum , ICON_JETPACK ) & & ( isRR ( ) | | p - > newowner = = - 1 ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_USEJETPACK , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( ! isRR ( ) )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
if ( p - > jetpack_amount > 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
p - > jetpack_on = ! p - > jetpack_on ;
if ( p - > jetpack_on )
{
p - > inven_icon = 4 ;
2020-07-25 07:32:54 +00:00
S_StopSound ( - 1 , p - > i , CHAN_VOICE ) ; // this will stop the falling scream
S_PlayActorSound ( DUKE_JETPACK_ON , p - > i ) ;
2020-05-17 11:51:18 +00:00
FTA ( QUOTE_JETPACK_ON , p ) ;
}
else
{
p - > hard_landing = 0 ;
p - > poszv = 0 ;
2020-07-25 07:32:54 +00:00
S_PlayActorSound ( DUKE_JETPACK_OFF , p - > i ) ;
S_StopSound ( DUKE_JETPACK_IDLE , p - > i ) ;
S_StopSound ( DUKE_JETPACK_ON , p - > i ) ;
2020-05-17 11:51:18 +00:00
FTA ( QUOTE_JETPACK_OFF , p ) ;
}
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else FTA ( QUOTE_JETPACK_NOT_FOUND , p ) ;
2020-05-17 11:25:39 +00:00
}
2020-05-17 11:51:18 +00:00
else
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
// eat cow pie
if ( p - > jetpack_amount > 0 & & sprite [ p - > i ] . extra < max_player_health )
2020-05-17 11:25:39 +00:00
{
2020-07-25 07:32:54 +00:00
if ( ! S_CheckActorSoundPlaying ( p - > i , 429 ) )
S_PlayActorSound ( 429 , p - > i ) ;
2020-05-17 11:51:18 +00:00
p - > jetpack_amount - = 100 ;
if ( p - > drink_amt > 0 )
{
p - > drink_amt - = 5 ;
if ( p - > drink_amt < 0 )
p - > drink_amt = 0 ;
}
if ( p - > eat < 100 )
{
p - > eat + = 5 ;
if ( p - > eat > 100 )
p - > eat = 100 ;
}
sprite [ p - > i ] . extra + = 5 ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
p - > inven_icon = 4 ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
if ( sprite [ p - > i ] . extra > max_player_health )
sprite [ p - > i ] . extra = max_player_health ;
2020-05-17 11:25:39 +00:00
2020-05-17 11:51:18 +00:00
if ( p - > jetpack_amount < = 0 )
checkavailinven ( p ) ;
}
2020-05-17 11:25:39 +00:00
}
}
}
2020-08-27 22:03:35 +00:00
if ( PlayerInput ( snum , SB_TURNAROUND ) & & p - > one_eighty_count = = 0 )
2020-05-17 11:25:39 +00:00
{
2020-05-17 11:51:18 +00:00
SetGameVarID ( g_iReturnVarID , 0 , - 1 , snum ) ;
OnEvent ( EVENT_TURNAROUND , - 1 , snum , - 1 ) ;
if ( GetGameVarID ( g_iReturnVarID , - 1 , snum ) = = 0 )
{
2020-07-17 01:00:43 +00:00
p - > one_eighty_count = - F16 ( 1024 ) ;
2020-05-17 11:51:18 +00:00
}
2020-05-17 11:25:39 +00:00
}
}
}
2020-07-16 16:58:31 +00:00
2020-07-17 22:34:20 +00:00
//---------------------------------------------------------------------------
//
2020-07-18 07:59:16 +00:00
// Main input routine.
// This includes several input improvements from EDuke32, but this code
// has been mostly rewritten completely to make it clearer and reduce redundancy.
2020-07-17 22:34:20 +00:00
//
//---------------------------------------------------------------------------
2020-07-16 16:58:31 +00:00
enum
{
2020-07-17 22:34:20 +00:00
TURBOTURNTIME = ( TICRATE / 8 ) , // 7
NORMALTURN = 15 ,
PREAMBLETURN = 5 ,
NORMALKEYMOVE = 40 ,
MAXVEL = ( ( NORMALKEYMOVE * 2 ) + 10 ) ,
MAXSVEL = ( ( NORMALKEYMOVE * 2 ) + 10 ) ,
MAXANGVEL = 1024 ,
MAXHORIZVEL = 256 ,
ONEEIGHTYSCALE = 4 ,
MOTOTURN = 20 ,
MAXVELMOTO = 120 ,
2020-07-16 16:58:31 +00:00
} ;
2020-07-16 20:49:26 +00:00
//---------------------------------------------------------------------------
//
2020-07-17 22:34:20 +00:00
// handles the input bits
2020-07-16 20:49:26 +00:00
//
//---------------------------------------------------------------------------
2020-08-29 19:09:04 +00:00
static void processInputBits ( player_struct * p , ControlInfo & info )
2020-07-16 20:49:26 +00:00
{
2020-08-27 22:03:35 +00:00
ApplyGlobalInput ( loc , & info ) ;
2020-08-29 19:09:04 +00:00
if ( isRR ( ) & & ( loc . actions & SB_CROUCH ) ) loc . actions & = ~ SB_JUMP ;
2020-08-28 20:51:05 +00:00
2020-08-29 11:32:14 +00:00
if ( p - > OnMotorcycle | | p - > OnBoat )
2020-08-27 22:03:35 +00:00
{
// mask out all actions not compatible with vehicles.
2020-08-29 19:09:04 +00:00
loc . actions & = ~ ( SB_WEAPONMASK_BITS | SB_TURNAROUND | SB_CENTERVIEW | SB_HOLSTER | SB_JUMP | SB_CROUCH | SB_RUN |
2020-08-29 11:32:14 +00:00
SB_AIM_UP | SB_AIM_DOWN | SB_AIMMODE | SB_LOOK_UP | SB_LOOK_DOWN | SB_LOOK_LEFT | SB_LOOK_RIGHT ) ;
2020-07-16 20:49:26 +00:00
}
2020-08-29 11:32:14 +00:00
else
{
if ( buttonMap . ButtonDown ( gamefunc_Quick_Kick ) ) // this shares a bit with another function so cannot be in the common code.
2020-08-29 19:09:04 +00:00
loc . actions | = SB_QUICK_KICK ;
2020-07-17 18:56:10 +00:00
2020-08-29 11:32:14 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Toggle_Crouch ) | | p - > crouch_toggle )
{
2020-08-29 19:09:04 +00:00
loc . actions | = SB_CROUCH ;
2020-08-29 11:32:14 +00:00
}
2020-08-29 19:09:04 +00:00
if ( ( isRR ( ) & & p - > drink_amt > 88 ) ) loc . actions | = SB_LOOK_LEFT ;
if ( ( isRR ( ) & & p - > drink_amt > 99 ) ) loc . actions | = SB_LOOK_DOWN ;
2020-08-29 11:32:14 +00:00
}
2020-07-16 21:32:00 +00:00
}
2020-07-18 07:59:16 +00:00
//---------------------------------------------------------------------------
//
// split off so that it can later be integrated into the other games more easily.
//
//---------------------------------------------------------------------------
static void checkCrouchToggle ( player_struct * p )
{
int const sectorLotag = p - > cursectnum ! = - 1 ? sector [ p - > cursectnum ] . lotag : 0 ;
int const crouchable = sectorLotag ! = ST_2_UNDERWATER & & ( sectorLotag ! = ST_1_ABOVE_WATER | | p - > spritebridge ) ;
if ( buttonMap . ButtonDown ( gamefunc_Toggle_Crouch ) )
{
p - > crouch_toggle = ! p - > crouch_toggle & & crouchable ;
if ( crouchable )
buttonMap . ClearButton ( gamefunc_Toggle_Crouch ) ;
}
if ( buttonMap . ButtonDown ( gamefunc_Crouch ) | | buttonMap . ButtonDown ( gamefunc_Jump ) | | p - > jetpack_on | | ( ! crouchable & & p - > on_ground ) )
p - > crouch_toggle = 0 ;
}
2020-07-18 11:27:24 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int getticssincelastupdate ( )
{
2020-08-31 17:23:05 +00:00
int tics = lastcontroltime = = 0 | | ud . levelclock < lastcontroltime ? 0 : ud . levelclock - lastcontroltime ;
lastcontroltime = ud . levelclock ;
2020-07-18 11:27:24 +00:00
return tics ;
}
2020-07-16 21:32:00 +00:00
//---------------------------------------------------------------------------
//
2020-07-17 22:34:20 +00:00
// handles movement
2020-07-16 21:32:00 +00:00
//
//---------------------------------------------------------------------------
2020-08-29 19:09:04 +00:00
static void processMovement ( player_struct * p , InputPacket & input , ControlInfo & info , double scaleFactor )
2020-07-16 21:32:00 +00:00
{
2020-08-31 18:51:22 +00:00
bool mouseaim = ! ( loc . actions & SB_AIMMODE ) ;
2020-07-17 22:34:20 +00:00
// JBF: Run key behaviour is selectable
2020-08-29 19:09:04 +00:00
int running = ! ! ( loc . actions & SB_RUN ) ;
2020-07-17 22:34:20 +00:00
int turnamount = NORMALTURN < < running ;
int keymove = NORMALKEYMOVE < < running ;
if ( buttonMap . ButtonDown ( gamefunc_Strafe ) )
input . svel - = info . mousex * 4.f + scaleFactor * info . dyaw * keymove ;
else
input . q16avel + = fix16_from_float ( info . mousex + scaleFactor * info . dyaw ) ;
if ( mouseaim )
input . q16horz + = fix16_from_float ( info . mousey ) ;
else
input . fvel - = info . mousey * 8.f ;
if ( ! in_mouseflip ) input . q16horz = - input . q16horz ;
input . q16horz - = fix16_from_dbl ( scaleFactor * ( info . dpitch ) ) ;
input . svel - = scaleFactor * ( info . dx * keymove ) ;
input . fvel - = scaleFactor * ( info . dz * keymove ) ;
if ( buttonMap . ButtonDown ( gamefunc_Strafe ) )
{
2020-08-29 19:09:04 +00:00
if ( ! loc . svel )
2020-07-17 22:34:20 +00:00
{
if ( buttonMap . ButtonDown ( gamefunc_Turn_Left ) )
input . svel = keymove ;
2020-07-17 18:56:10 +00:00
2020-07-17 22:34:20 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Turn_Right ) )
input . svel = - keymove ;
}
}
else
{
2020-07-18 11:27:24 +00:00
int tics = getticssincelastupdate ( ) ;
2020-07-17 22:34:20 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Turn_Left ) )
{
turnheldtime + = tics ;
2020-07-18 19:28:57 +00:00
input . q16avel - = fix16_from_dbl ( 2 * scaleFactor * ( turnheldtime > = TURBOTURNTIME ? turnamount : PREAMBLETURN ) ) ;
2020-07-17 22:34:20 +00:00
}
else if ( buttonMap . ButtonDown ( gamefunc_Turn_Right ) )
{
turnheldtime + = tics ;
2020-07-18 19:28:57 +00:00
input . q16avel + = fix16_from_dbl ( 2 * scaleFactor * ( turnheldtime > = TURBOTURNTIME ? turnamount : PREAMBLETURN ) ) ;
2020-07-17 22:34:20 +00:00
}
else
2020-07-18 11:27:24 +00:00
{
2020-07-17 22:34:20 +00:00
turnheldtime = 0 ;
2020-07-18 11:27:24 +00:00
lastcontroltime = 0 ;
}
2020-07-17 22:34:20 +00:00
}
2020-08-29 19:09:04 +00:00
if ( abs ( loc . svel ) < keymove )
2020-07-17 22:34:20 +00:00
{
if ( buttonMap . ButtonDown ( gamefunc_Strafe_Left ) )
input . svel + = keymove ;
2020-07-16 21:32:00 +00:00
2020-07-17 22:34:20 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Strafe_Right ) )
input . svel + = - keymove ;
}
2020-08-29 19:09:04 +00:00
if ( abs ( loc . fvel ) < keymove )
2020-07-17 22:34:20 +00:00
{
if ( isRR ( ) & & p - > drink_amt > = 66 & & p - > drink_amt < = 87 )
{
if ( buttonMap . ButtonDown ( gamefunc_Move_Forward ) )
{
input . fvel + = keymove ;
if ( p - > drink_amt & 1 )
input . svel + = keymove ;
else
input . svel - = keymove ;
}
if ( buttonMap . ButtonDown ( gamefunc_Move_Backward ) )
{
input . fvel + = - keymove ;
if ( p - > drink_amt & 1 )
input . svel - = keymove ;
else
input . svel + = keymove ;
}
}
else
{
if ( buttonMap . ButtonDown ( gamefunc_Move_Forward ) )
input . fvel + = keymove ;
2020-07-16 20:49:26 +00:00
2020-07-17 22:34:20 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Move_Backward ) )
input . fvel + = - keymove ;
}
}
2020-07-16 20:49:26 +00:00
}
2020-07-16 17:16:56 +00:00
//---------------------------------------------------------------------------
//
2020-07-17 22:34:20 +00:00
// split out for readability
2020-07-16 17:16:56 +00:00
//
//---------------------------------------------------------------------------
2020-07-24 10:46:20 +00:00
static double motoApplyTurn ( player_struct * p , int turnl , int turnr , int bike_turn , bool goback , double factor )
2020-07-16 17:16:56 +00:00
{
2020-07-24 10:46:20 +00:00
int turnvel = 0 ;
2020-08-04 07:10:44 +00:00
p - > oTiltStatus = p - > TiltStatus ;
2020-07-24 10:46:20 +00:00
2020-07-16 17:16:56 +00:00
if ( p - > MotoSpeed = = 0 | | ! p - > on_ground )
{
2020-07-18 11:27:24 +00:00
turnheldtime = 0 ;
lastcontroltime = 0 ;
2020-07-16 17:16:56 +00:00
if ( turnl )
{
p - > TiltStatus - = ( float ) factor ;
if ( p - > TiltStatus < - 10 )
p - > TiltStatus = - 10 ;
}
else if ( turnr )
{
p - > TiltStatus + = ( float ) factor ;
if ( p - > TiltStatus > 10 )
p - > TiltStatus = 10 ;
}
}
else
{
2020-07-18 11:27:24 +00:00
int tics = getticssincelastupdate ( ) ;
2020-07-24 10:46:20 +00:00
if ( turnl | | turnr | | p - > moto_drink ! = 0 )
2020-07-16 17:16:56 +00:00
{
2020-07-24 10:46:20 +00:00
if ( turnl | | p - > moto_drink < 0 )
2020-07-16 17:16:56 +00:00
{
2020-07-24 10:46:20 +00:00
turnheldtime + = tics ;
p - > TiltStatus - = ( float ) factor ;
if ( p - > TiltStatus < - 10 )
p - > TiltStatus = - 10 ;
if ( turnheldtime > = TURBOTURNTIME & & p - > MotoSpeed > 0 )
{
if ( goback ) turnvel + = bike_turn ? 40 : 20 ;
else turnvel + = bike_turn ? - 40 : - 20 ;
}
else
{
if ( goback ) turnvel + = bike_turn ? 20 : 6 ;
else turnvel + = bike_turn ? - 20 : - 6 ;
}
2020-07-16 17:16:56 +00:00
}
2020-07-24 10:46:20 +00:00
if ( turnr | | p - > moto_drink > 0 )
2020-07-16 17:16:56 +00:00
{
2020-07-24 10:46:20 +00:00
turnheldtime + = tics ;
p - > TiltStatus + = ( float ) factor ;
if ( p - > TiltStatus > 10 )
p - > TiltStatus = 10 ;
if ( turnheldtime > = TURBOTURNTIME & & p - > MotoSpeed > 0 )
{
if ( goback ) turnvel + = bike_turn ? - 40 : - 20 ;
else turnvel + = bike_turn ? 40 : 20 ;
}
else
{
if ( goback ) turnvel + = bike_turn ? - 20 : - 6 ;
else turnvel + = bike_turn ? 20 : 6 ;
}
2020-07-16 17:16:56 +00:00
}
}
else
{
turnheldtime = 0 ;
2020-07-18 11:27:24 +00:00
lastcontroltime = 0 ;
2020-07-16 17:16:56 +00:00
if ( p - > TiltStatus > 0 )
p - > TiltStatus - = ( float ) factor ;
else if ( p - > TiltStatus < 0 )
p - > TiltStatus + = ( float ) factor ;
}
}
2020-07-24 23:51:52 +00:00
if ( fabs ( p - > TiltStatus ) < factor )
p - > TiltStatus = 0 ;
2020-07-24 10:46:20 +00:00
return turnvel * factor ;
2020-07-16 17:16:56 +00:00
}
2020-07-16 16:58:31 +00:00
//---------------------------------------------------------------------------
//
2020-07-16 17:16:56 +00:00
// same for the boat
2020-07-16 16:58:31 +00:00
//
//---------------------------------------------------------------------------
2020-07-24 11:27:47 +00:00
static double boatApplyTurn ( player_struct * p , int turnl , int turnr , int boat_turn , double factor )
2020-07-16 16:58:31 +00:00
{
2020-07-24 11:27:47 +00:00
int turnvel = 0 ;
2020-07-18 11:27:24 +00:00
int tics = getticssincelastupdate ( ) ;
2020-07-16 16:58:31 +00:00
if ( p - > MotoSpeed )
{
2020-07-24 11:27:47 +00:00
if ( turnl | | turnr | | p - > moto_drink ! = 0 )
2020-07-16 16:58:31 +00:00
{
2020-07-24 11:27:47 +00:00
if ( turnl | | p - > moto_drink < 0 )
2020-07-16 16:58:31 +00:00
{
2020-07-24 11:27:47 +00:00
turnheldtime + = tics ;
if ( ! p - > NotOnWater )
{
p - > TiltStatus - = ( float ) factor ;
if ( p - > TiltStatus < - 10 )
p - > TiltStatus = - 10 ;
}
if ( turnheldtime > = TURBOTURNTIME )
{
if ( p - > NotOnWater ) turnvel + = boat_turn ? - 12 : - 6 ;
else turnvel + = boat_turn ? - 40 : - 20 ;
}
else
{
if ( p - > NotOnWater ) turnvel + = boat_turn ? - 4 : - 2 ;
else turnvel + = boat_turn ? - 12 : - 6 ;
}
2020-07-16 16:58:31 +00:00
}
2020-07-24 11:27:47 +00:00
if ( turnr | | p - > moto_drink > 0 )
2020-07-16 16:58:31 +00:00
{
2020-07-24 11:27:47 +00:00
turnheldtime + = tics ;
if ( ! p - > NotOnWater )
{
p - > TiltStatus + = ( float ) factor ;
if ( p - > TiltStatus > 10 )
p - > TiltStatus = 10 ;
}
if ( turnheldtime > = TURBOTURNTIME )
{
if ( p - > NotOnWater ) turnvel + = boat_turn ? 12 : 6 ;
else turnvel + = boat_turn ? 40 : 20 ;
}
else
{
if ( p - > NotOnWater ) turnvel + = boat_turn ? 4 : 2 ;
else turnvel + = boat_turn ? 12 : 6 ;
}
2020-07-16 16:58:31 +00:00
}
}
else if ( ! p - > NotOnWater )
{
turnheldtime = 0 ;
2020-07-18 11:27:24 +00:00
lastcontroltime = 0 ;
2020-07-16 16:58:31 +00:00
if ( p - > TiltStatus > 0 )
p - > TiltStatus - = ( float ) factor ;
else if ( p - > TiltStatus < 0 )
p - > TiltStatus + = ( float ) factor ;
}
}
2020-07-24 11:27:47 +00:00
else if ( ! p - > NotOnWater )
2020-07-18 11:27:24 +00:00
{
turnheldtime = 0 ;
lastcontroltime = 0 ;
2020-07-24 11:27:47 +00:00
if ( p - > TiltStatus > 0 )
p - > TiltStatus - = ( float ) factor ;
else if ( p - > TiltStatus < 0 )
p - > TiltStatus + = ( float ) factor ;
2020-07-18 11:27:24 +00:00
}
2020-07-24 11:27:47 +00:00
2020-07-24 11:38:11 +00:00
if ( fabs ( p - > TiltStatus ) < factor )
2020-07-24 11:27:47 +00:00
p - > TiltStatus = 0 ;
return turnvel * factor ;
2020-07-16 16:58:31 +00:00
}
2020-07-17 18:56:10 +00:00
//---------------------------------------------------------------------------
//
// much of this was rewritten from scratch to make the logic easier to follow.
//
//---------------------------------------------------------------------------
2020-08-29 19:09:04 +00:00
static void processVehicleInput ( player_struct * p , ControlInfo & info , InputPacket & input , double scaleAdjust )
2020-07-17 18:56:10 +00:00
{
2020-07-17 20:09:01 +00:00
auto turnspeed = info . mousex + scaleAdjust * info . dyaw * ( 1. / 32 ) ; // originally this was 64, not 32. Why the change?
2020-07-17 18:56:10 +00:00
int turnl = buttonMap . ButtonDown ( gamefunc_Turn_Left ) | | buttonMap . ButtonDown ( gamefunc_Strafe_Left ) ;
int turnr = buttonMap . ButtonDown ( gamefunc_Turn_Right ) | | buttonMap . ButtonDown ( gamefunc_Strafe_Right ) ;
// Cancel out micro-movement
const double turn_threshold = 1 / 65536. ;
2020-07-17 20:09:01 +00:00
if ( turnspeed < - turn_threshold )
2020-07-17 18:56:10 +00:00
turnl = 1 ;
2020-07-17 20:09:01 +00:00
else if ( turnspeed > turn_threshold )
2020-07-17 18:56:10 +00:00
turnr = 1 ;
else
2020-07-17 20:09:01 +00:00
turnspeed = 0 ;
2020-07-17 18:56:10 +00:00
2020-07-17 20:09:01 +00:00
if ( p - > OnBoat | | ! p - > moto_underwater )
{
if ( buttonMap . ButtonDown ( gamefunc_Move_Forward ) | | buttonMap . ButtonDown ( gamefunc_Strafe ) )
2020-08-29 19:09:04 +00:00
loc . actions | = SB_JUMP ;
2020-07-17 20:09:01 +00:00
if ( buttonMap . ButtonDown ( gamefunc_Move_Backward ) )
2020-08-29 19:09:04 +00:00
loc . actions | = SB_AIM_UP ;
if ( loc . actions & SB_RUN )
loc . actions | = SB_CROUCH ;
2020-07-17 20:09:01 +00:00
}
2020-07-17 18:56:10 +00:00
if ( turnl )
2020-08-29 19:09:04 +00:00
loc . actions | = SB_AIM_DOWN ;
2020-07-17 18:56:10 +00:00
if ( turnr )
2020-08-29 19:09:04 +00:00
loc . actions | = SB_LOOK_LEFT ;
2020-07-17 18:56:10 +00:00
2020-07-17 20:09:01 +00:00
double turnvel ;
if ( p - > OnMotorcycle )
{
2020-07-17 22:34:20 +00:00
bool backward = buttonMap . ButtonDown ( gamefunc_Move_Backward ) & & p - > MotoSpeed < = 0 ;
2020-07-17 20:09:01 +00:00
2020-07-24 10:46:20 +00:00
turnvel = motoApplyTurn ( p , turnl , turnr , turnspeed , backward , scaleAdjust ) ;
2020-07-17 20:09:01 +00:00
if ( p - > moto_underwater ) p - > MotoSpeed = 0 ;
}
else
{
2020-07-24 11:27:47 +00:00
turnvel = boatApplyTurn ( p , turnl , turnr , turnspeed ! = 0 , scaleAdjust ) ;
2020-07-17 20:09:01 +00:00
}
2020-07-17 18:56:10 +00:00
// What is this? Optimization for playing with a mouse which the original did not have?
2020-07-17 20:09:01 +00:00
if ( turnspeed )
turnvel * = clamp ( turnspeed * turnspeed , 0. , 1. ) ;
2020-07-17 18:56:10 +00:00
input . fvel = p - > MotoSpeed ;
input . q16avel = fix16_from_dbl ( turnvel ) ;
}
2020-07-18 07:59:16 +00:00
//---------------------------------------------------------------------------
//
// finalizes the input and passes it to the global input buffer
//
//---------------------------------------------------------------------------
2020-08-29 19:09:04 +00:00
static void FinalizeInput ( int playerNum , InputPacket & input , bool vehicle )
2020-07-18 07:59:16 +00:00
{
auto p = & ps [ playerNum ] ;
bool blocked = movementBlocked ( playerNum ) | | sprite [ p - > i ] . extra < = 0 | | ( p - > dead_flag & & ! ud . god ) ;
2020-08-24 17:31:43 +00:00
if ( ( automapFollow & & automapMode ! = am_off ) | | blocked )
2020-07-18 07:59:16 +00:00
{
2020-08-24 17:31:43 +00:00
if ( automapFollow & & automapMode ! = am_off )
2020-07-18 07:59:16 +00:00
{
ud . folfvel = input . fvel ;
ud . folavel = fix16_to_int ( input . q16avel ) ;
}
2020-08-29 19:09:04 +00:00
loc . fvel = loc . svel = 0 ;
loc . q16avel = loc . q16horz = 0 ;
2020-08-04 03:02:19 +00:00
input . q16avel = input . q16horz = 0 ;
2020-07-18 07:59:16 +00:00
}
else
{
if ( p - > on_crane < 0 )
{
if ( ! vehicle )
{
2020-08-29 19:09:04 +00:00
loc . fvel = clamp ( loc . fvel + input . fvel , - MAXVEL , MAXVEL ) ;
loc . svel = clamp ( loc . svel + input . svel , - MAXSVEL , MAXSVEL ) ;
2020-07-18 07:59:16 +00:00
}
else
2020-08-29 19:09:04 +00:00
loc . fvel = clamp ( input . fvel , - ( MAXVELMOTO / 8 ) , MAXVELMOTO ) ;
2020-07-18 07:59:16 +00:00
}
2020-08-05 21:25:04 +00:00
else
{
2020-08-29 19:09:04 +00:00
loc . fvel = input . fvel = 0 ;
loc . svel = input . svel = 0 ;
2020-08-05 21:25:04 +00:00
}
2020-07-18 07:59:16 +00:00
if ( p - > on_crane < 0 & & p - > newowner = = - 1 )
{
2020-08-29 19:09:04 +00:00
loc . q16avel = fix16_clamp ( loc . q16avel + input . q16avel , F16 ( - MAXANGVEL ) , F16 ( MAXANGVEL ) ) ;
2020-08-03 11:19:45 +00:00
if ( ! cl_syncinput & & input . q16avel )
2020-07-18 07:59:16 +00:00
{
2020-08-03 11:19:45 +00:00
p - > one_eighty_count = 0 ;
2020-07-18 07:59:16 +00:00
}
}
2020-08-05 21:25:04 +00:00
else
{
2020-08-29 19:09:04 +00:00
loc . q16avel = input . q16avel = 0 ;
2020-08-05 21:25:04 +00:00
}
2020-07-18 07:59:16 +00:00
if ( p - > newowner = = - 1 & & p - > return_to_center < = 0 )
{
2020-08-29 19:09:04 +00:00
loc . q16horz = fix16_clamp ( loc . q16horz + input . q16horz , F16 ( - MAXHORIZVEL ) , F16 ( MAXHORIZVEL ) ) ;
2020-07-18 07:59:16 +00:00
}
2020-08-05 21:25:04 +00:00
else
{
2020-08-29 19:09:04 +00:00
loc . q16horz = input . q16horz = 0 ;
2020-08-05 21:25:04 +00:00
}
2020-07-18 07:59:16 +00:00
}
}
2020-07-17 22:59:10 +00:00
//---------------------------------------------------------------------------
//
// main input handler routine
//
//---------------------------------------------------------------------------
2020-08-29 19:20:10 +00:00
static void GetInputInternal ( InputPacket & locInput )
2020-07-17 22:59:10 +00:00
{
double elapsedInputTicks ;
auto const p = & ps [ myconnectindex ] ;
auto now = I_msTimeF ( ) ;
// do not let this become too large - it would create overflows resulting in undefined behavior. The very first tic must not use the timer difference at all because the timer has not been set yet.
// This really needs to have the timer fixed to be robust, doing it ad-hoc here is not really safe.
2020-07-24 09:38:09 +00:00
if ( lastCheck > 0 ) elapsedInputTicks = min ( now - lastCheck , 1000.0 / REALGAMETICSPERSEC ) ;
2020-07-17 22:59:10 +00:00
else elapsedInputTicks = 1 ;
lastCheck = now ;
if ( paused )
{
loc = { } ;
return ;
}
if ( numplayers = = 1 )
{
setlocalplayerinput ( p ) ;
}
2020-08-04 07:11:32 +00:00
double scaleAdjust = ! cl_syncinput ? elapsedInputTicks * REALGAMETICSPERSEC / 1000.0 : 1 ;
2020-07-17 22:59:10 +00:00
ControlInfo info ;
CONTROL_GetInput ( & info ) ;
2020-08-26 15:12:48 +00:00
InputPacket input { } ;
2020-07-17 22:59:10 +00:00
if ( isRRRA ( ) & & ( p - > OnMotorcycle | | p - > OnBoat ) )
{
p - > crouch_toggle = 0 ;
2020-08-29 19:09:04 +00:00
processInputBits ( p , info ) ;
processVehicleInput ( p , info , input , scaleAdjust ) ;
FinalizeInput ( myconnectindex , input , true ) ;
2020-07-17 22:59:10 +00:00
2020-08-05 07:53:22 +00:00
if ( ! cl_syncinput & & sprite [ p - > i ] . extra > 0 )
2020-07-17 22:59:10 +00:00
{
2020-08-05 07:53:22 +00:00
apply_seasick ( p , scaleAdjust ) ;
2020-07-17 22:59:10 +00:00
}
}
else
{
2020-08-29 19:09:04 +00:00
processInputBits ( p , info ) ;
processMovement ( p , input , info , scaleAdjust ) ;
2020-07-17 22:59:10 +00:00
checkCrouchToggle ( p ) ;
2020-08-29 19:09:04 +00:00
FinalizeInput ( myconnectindex , input , false ) ;
2020-08-05 07:53:22 +00:00
}
2020-07-17 22:59:10 +00:00
2020-08-05 07:53:22 +00:00
if ( ! cl_syncinput )
{
// Do these in the same order as the old code.
calcviewpitch ( p , scaleAdjust ) ;
applylook ( myconnectindex , scaleAdjust , input . q16avel ) ;
2020-08-29 19:09:04 +00:00
sethorizon ( myconnectindex , loc . actions , scaleAdjust , input . q16horz ) ;
2020-07-17 22:59:10 +00:00
}
}
2020-07-17 20:09:01 +00:00
2020-08-29 19:20:10 +00:00
//---------------------------------------------------------------------------
//
// External entry point
//
//---------------------------------------------------------------------------
void GameInterface : : GetInput ( InputPacket * packet )
{
GetInputInternal ( loc ) ;
if ( packet )
{
auto const pPlayer = & ps [ myconnectindex ] ;
auto const q16ang = fix16_to_int ( pPlayer - > q16ang ) ;
* packet = loc ;
auto fvel = loc . fvel ;
auto svel = loc . svel ;
packet - > fvel = mulscale9 ( fvel , sintable [ ( q16ang + 2560 ) & 2047 ] ) +
mulscale9 ( svel , sintable [ ( q16ang + 2048 ) & 2047 ] ) +
pPlayer - > fric . x ;
packet - > svel = mulscale9 ( fvel , sintable [ ( q16ang + 2048 ) & 2047 ] ) +
mulscale9 ( svel , sintable [ ( q16ang + 1536 ) & 2047 ] ) +
pPlayer - > fric . y ;
loc = { } ;
}
}
//---------------------------------------------------------------------------
//
2020-07-18 11:27:24 +00:00
// This is called from ImputState::ClearAllInput and resets all static state being used here.
2020-08-29 19:20:10 +00:00
//
//---------------------------------------------------------------------------
2020-07-17 18:56:10 +00:00
void GameInterface : : clearlocalinputstate ( )
{
2020-08-29 19:20:10 +00:00
loc = { } ;
2020-07-18 11:27:24 +00:00
turnheldtime = 0 ;
lastcontroltime = 0 ;
lastCheck = 0 ;
2020-07-17 18:56:10 +00:00
}
2020-08-29 19:20:10 +00:00
2020-05-17 11:25:39 +00:00
END_DUKE_NS