2015-05-19 21:54:34 +00:00
//-------------------------------------------------------------------------
/*
Copyright ( C ) 1997 , 2005 - 3 D Realms Entertainment
2019-12-19 00:20:43 +00:00
Copyright ( C ) 2019 Christoph Oelckers
2015-05-19 21:54:34 +00:00
This file is part of Shadow Warrior version 1.2
Shadow Warrior 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
Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
Original Source : 1997 - Frank Maddin and Jim Norwood
Prepared for public release : 03 / 28 / 2005 - Charlie Wiederhold , 3 D Realms
*/
//-------------------------------------------------------------------------
2019-10-09 16:09:05 +00:00
# include "ns.h"
2017-02-25 08:15:36 +00:00
# include "compat.h"
2015-05-19 21:54:34 +00:00
# include "build.h"
2019-12-17 22:25:07 +00:00
2015-05-19 21:54:34 +00:00
# include "keys.h"
# include "names2.h"
# include "mytypes.h"
# include "gamedefs.h"
# include "config.h"
# include "panel.h"
# include "game.h"
# include "sounds.h"
# include "ai.h"
2019-03-21 02:24:19 +00:00
# include "network.h"
2015-05-19 21:54:34 +00:00
# include "cache.h"
# include "text.h"
# include "rts.h"
# include "menus.h"
2019-04-08 06:26:44 +00:00
# include "config.h"
2019-12-02 20:05:19 +00:00
# include "menu/menu.h"
2019-12-07 18:57:19 +00:00
# include "z_music.h"
2019-12-17 22:25:07 +00:00
# include "sound/s_soundinternal.h"
# include "filesystem/filesystem.h"
2020-02-23 16:13:18 +00:00
# include "serializer.h"
2015-05-19 22:06:04 +00:00
2019-10-09 16:09:05 +00:00
BEGIN_SW_NS
2019-12-18 10:09:01 +00:00
enum EChanExFlags
{
2019-12-18 18:17:37 +00:00
CHANEXF_NODOPPLER = 0x20000000 ,
2019-12-19 00:20:43 +00:00
CHANEXF_DONTPAN = 0x40000000 ,
2019-12-18 10:09:01 +00:00
} ;
2015-05-19 21:54:34 +00:00
// Parentally locked sounds list
int PLocked_Sounds [ ] =
{
483 , 328 , 334 , 335 , 338 , 478 , 450 , 454 , 452 , 453 , 456 , 457 , 458 , 455 , 460 , 462 ,
461 , 464 , 466 , 465 , 467 , 463 , 342 , 371 , 254 , 347 , 350 , 432 , 488 , 489 , 490 , 76 , 339 ,
499 , 500 , 506 , 479 , 480 , 481 , 482 , 78 , 600 , 467 , 548 , 547 , 544 , 546 , 545 , 542 , 541 , 540 ,
539 , 536 , 529 , 525 , 522 , 521 , 515 , 516 , 612 , 611 , 589 , 625 , 570 , 569 , 567 , 565 ,
558 , 557
} ;
//
// Includes digi.h to build the table
//
# define DIGI_TABLE
VOC_INFO voc [ ] =
{
# include "digi.h"
} ;
# undef DIGI_TABLE
//
// Includes ambient.h to build the table of ambient sounds for game
//
# define AMBIENT_TABLE
AMB_INFO ambarray [ ] =
{
# include "ambient.h"
} ;
# undef AMBIENT_TABLE
# define MAX_AMBIENT_SOUNDS 82
2019-12-18 21:13:19 +00:00
//==========================================================================
//
//
2015-05-19 21:54:34 +00:00
//
2019-12-17 22:25:07 +00:00
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
float S_ConvertPitch ( int lpitch )
2015-05-19 21:54:34 +00:00
{
2019-12-19 00:20:43 +00:00
return pow ( 2 , lpitch / 1200. ) ;
2015-05-19 21:54:34 +00:00
}
2019-12-17 22:25:07 +00:00
//==========================================================================
//
2019-12-18 18:17:37 +00:00
// Sound Distance Calculation
2019-12-17 22:25:07 +00:00
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
enum
{
MAXLEVLDIST = 19000 , // The higher the number, the further away you can hear sound
DECAY_CONST = 4000
} ;
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
short SoundDist ( int x , int y , int z , int basedist )
{
double tx , ty , tz ;
double sqrdist , retval ;
extern short screenpeek ;
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
tx = fabs ( Player [ screenpeek ] . posx - x ) ;
ty = fabs ( Player [ screenpeek ] . posy - y ) ;
tz = fabs ( ( Player [ screenpeek ] . posz - z ) > > 4 ) ;
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
// Use the Pythagreon Theorem to compute the magnitude of a 3D vector
sqrdist = fabs ( tx * tx + ty * ty + tz * tz ) ;
retval = sqrt ( sqrdist ) ;
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
if ( basedist < 0 ) // if basedist is negative
{
double decayshift = 2 ;
int decay = labs ( basedist ) / DECAY_CONST ;
for ( int i = 0 ; i < decay ; i + + )
decayshift * = 2 ;
if ( fabs ( double ( basedist ) / decayshift ) > = retval )
retval = 0 ;
else
retval * = decay ;
}
else
{
if ( basedist > retval )
retval = 0 ;
else
retval - = basedist ;
}
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
retval = retval * 256 / MAXLEVLDIST ;
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
if ( retval < 0 ) retval = 0 ;
if ( retval > 255 ) retval = 255 ;
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
return retval ;
}
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
//==========================================================================
//
// Calculate rolloff info.
//
//==========================================================================
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
FRolloffInfo GetRolloff ( int basedist )
{
FRolloffInfo info ;
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
if ( basedist < 0 ) // if basedist is negative
{
double decayshift = 2 ;
int decay = labs ( basedist ) / DECAY_CONST ;
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
for ( int i = 0 ; i < decay ; i + + )
decayshift * = 2 ;
2019-12-17 22:25:07 +00:00
2019-12-18 18:17:37 +00:00
info . RolloffType = ROLLOFF_Doom ;
info . MinDistance = ( float ) ( - basedist / decayshift / 16. ) ;
info . MaxDistance = MAXLEVLDIST / 16.f / decay ;
}
else
{
info . RolloffType = ROLLOFF_Doom ;
info . MinDistance = basedist / 16.f ;
info . MaxDistance = info . MinDistance + MAXLEVLDIST / 16.f ;
}
return info ;
}
2019-12-17 22:25:07 +00:00
2019-12-19 00:20:43 +00:00
//==========================================================================
//
//
// Ambient sounds
//
//
//==========================================================================
struct AmbientSound
{
SPRITEp sp ;
int ambIndex ;
int vocIndex ;
int ChanFlags ;
int maxIndex ;
int curIndex ;
bool intermit ;
} ;
static TArray < AmbientSound * > ambients ;
//==========================================================================
//
//
//
//==========================================================================
void StopAmbientSound ( void )
{
2020-02-23 14:41:57 +00:00
for ( auto amb : ambients )
2019-12-19 00:20:43 +00:00
{
2020-02-23 14:41:57 +00:00
soundEngine - > StopSound ( SOURCE_Ambient , amb , - 1 ) ;
2020-02-23 16:13:18 +00:00
delete amb ;
2019-12-19 00:20:43 +00:00
}
ambients . Clear ( ) ;
}
//==========================================================================
//
// Play a sound
//
//==========================================================================
void InitAmbient ( int num , SPRITEp sp )
{
VOC_INFOp vp ;
int pitch = 0 ;
short angle , sound_dist ;
int tx , ty , tz ;
uint8_t priority ;
int maxtics = 0 ;
if ( ! snd_ambience ) return ;
// Ambient sounds need special treatment
if ( num < 0 | | num > MAX_AMBIENT_SOUNDS )
{
sprintf ( ds , " Invalid or out of range ambient sound number %d \n " , num ) ;
PutStringInfo ( Player + screenpeek , ds ) ;
return ;
}
auto vnum = ambarray [ num ] . diginame ;
if ( ! soundEngine - > isValidSoundId ( vnum ) )
{
return ; // linked sound does not exist.
}
auto amb = new AmbientSound ;
amb - > sp = sp ;
amb - > ambIndex = num ;
amb - > vocIndex = vnum ;
2020-02-23 16:13:18 +00:00
amb - > ChanFlags = CHANF_TRANSIENT ;
2019-12-19 00:20:43 +00:00
if ( ambarray [ num ] . ambient_flags & v3df_dontpan ) amb - > ChanFlags | = EChanFlags : : FromInt ( CHANEXF_DONTPAN ) ;
if ( voc [ vnum ] . voc_flags & vf_loop ) amb - > ChanFlags | = CHANF_LOOP ;
amb - > maxIndex = ambarray [ num ] . maxtics * 8 ;
amb - > curIndex = 0 ;
amb - > intermit = ! ! ( ambarray [ num ] . ambient_flags & v3df_intermit ) ;
ambients . Push ( amb ) ;
}
//==========================================================================
//
//
//
//==========================================================================
void StartAmbientSound ( void )
{
short i , nexti ;
extern SWBOOL InMenuLevel ;
if ( InMenuLevel | | ! SoundEnabled ( ) ) return ; // Don't restart ambience if no level is active! Will crash game.
TRAVERSE_SPRITE_STAT ( headspritestat [ STAT_AMBIENT ] , i , nexti )
{
SPRITEp sp = & sprite [ i ] ;
InitAmbient ( sp - > lotag , sp ) ;
}
}
//==========================================================================
//
//
//
//==========================================================================
static void RestartAmbient ( AmbientSound * amb )
{
auto & vp = voc [ amb - > vocIndex ] ;
auto rolloff = GetRolloff ( vp . voc_distance ) ;
int pitch = 0 ;
if ( vp . pitch_hi < = vp . pitch_lo ) pitch = vp . pitch_lo ;
else pitch = vp . pitch_lo + ( STD_RANDOM_RANGE ( vp . pitch_hi - vp . pitch_lo ) ) ;
2020-02-27 22:11:03 +00:00
if ( ! soundEngine - > IsSourcePlayingSomething ( SOURCE_Ambient , amb , CHAN_BODY , amb - > vocIndex ) )
soundEngine - > StartSound ( SOURCE_Ambient , amb , nullptr , CHAN_BODY , EChanFlags : : FromInt ( amb - > ChanFlags ) , amb - > vocIndex , 1.f , ATTN_NORM , & rolloff , S_ConvertPitch ( pitch ) ) ;
2019-12-19 00:20:43 +00:00
}
2019-12-18 21:13:19 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
static int RandomizeAmbientSpecials ( int handle )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
# define MAXRNDAMB 12
int ambrand [ ] =
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
56 , 57 , 58 , 59 , 60 , 61 , 62 , 63 , 64 , 65 , 66 , 67
} ;
2019-12-18 21:13:19 +00:00
int i ;
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
// If ambient sound is found in the array, randomly pick a new sound
for ( i = 0 ; i < MAXRNDAMB ; i + + )
2015-05-19 21:54:34 +00:00
{
2019-12-18 21:13:19 +00:00
if ( handle = = ambarray [ ambrand [ i ] ] . diginame )
2019-12-18 10:09:01 +00:00
return ambrand [ STD_RANDOM_RANGE ( MAXRNDAMB - 1 ) ] ;
2015-05-19 21:54:34 +00:00
}
2019-12-18 21:13:19 +00:00
return - 1 ;
2015-05-19 21:54:34 +00:00
}
2019-12-18 21:13:19 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
static void DoTimedSound ( AmbientSound * amb )
2015-05-19 21:54:34 +00:00
{
2019-12-19 00:20:43 +00:00
amb - > curIndex + = synctics ;
if ( amb - > curIndex > = amb - > maxIndex )
2019-12-18 10:09:01 +00:00
{
2020-02-23 14:41:57 +00:00
if ( soundEngine - > EnumerateChannels ( [ = ] ( FSoundChan * tchan )
{
return ( tchan - > Source = = amb & & ! ( tchan - > ChanFlags & CHANF_FORGETTABLE ) ) ;
} ) )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
// Check for special case ambient sounds. Since the sound is stopped and doesn't occupy a real channel at this time we can just swap out the sound ID before restarting it.
2019-12-19 00:20:43 +00:00
int ambid = RandomizeAmbientSpecials ( amb - > vocIndex ) ;
2019-12-18 21:13:19 +00:00
if ( ambid ! = - 1 )
2019-12-18 10:09:01 +00:00
{
2019-12-19 00:20:43 +00:00
amb - > vocIndex = ambid ;
amb - > maxIndex = STD_RANDOM_RANGE ( ambarray [ ambid ] . maxtics ) ;
}
RestartAmbient ( amb ) ;
}
}
}
//==========================================================================
//
//
//
//==========================================================================
static void UpdateAmbients ( )
{
for ( auto & amb : ambients )
{
auto sp = amb - > sp ;
auto sdist = SoundDist ( sp - > pos . x , sp - > pos . y , sp - > pos . z , voc [ amb - > vocIndex ] . voc_distance ) ;
if ( sdist < 255 & & amb - > vocIndex = = DIGI_WHIPME )
{
PLAYERp pp = Player + screenpeek ;
if ( ! FAFcansee ( sp - > pos . x , sp - > pos . y , sp - > pos . z , sp - > sectnum , pp - > posx , pp - > posy , pp - > posz , pp - > cursectnum ) )
{
sdist = 255 ;
2019-12-18 10:09:01 +00:00
}
2019-12-19 00:20:43 +00:00
}
if ( sdist < 255 )
{
if ( amb - > intermit ) DoTimedSound ( amb ) ;
else RestartAmbient ( amb ) ;
2019-12-18 21:13:19 +00:00
2019-12-19 00:20:43 +00:00
}
else
{
2020-02-23 14:41:57 +00:00
soundEngine - > StopSound ( SOURCE_Ambient , amb , - 1 ) ;
2019-12-18 10:09:01 +00:00
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
}
2015-05-19 21:54:34 +00:00
}
2019-12-19 00:20:43 +00:00
//==========================================================================
//
// end of ambient sounds
//
//==========================================================================
//==========================================================================
//
//
//
//==========================================================================
class SWSoundEngine : public SoundEngine
{
// client specific parts of the sound engine go in this class.
void CalcPosVel ( int type , const void * source , const float pt [ 3 ] , int channum , int chanflags , FSoundID chanSound , FVector3 * pos , FVector3 * vel , FSoundChan * chan ) override ;
TArray < uint8_t > ReadSound ( int lumpnum ) override ;
public :
SWSoundEngine ( )
{
S_Rolloff . RolloffType = ROLLOFF_Doom ;
S_Rolloff . MinDistance = 0 ; // These are the default values, SW uses a few different rolloff settings.
S_Rolloff . MaxDistance = 1187 ;
}
} ;
//==========================================================================
//
//
//
//==========================================================================
TArray < uint8_t > SWSoundEngine : : ReadSound ( int lumpnum )
{
auto wlump = fileSystem . OpenFileReader ( lumpnum ) ;
return wlump . Read ( ) ;
}
//==========================================================================
//
//
//
//==========================================================================
void InitFX ( void )
{
if ( soundEngine ) return ; // just in case.
soundEngine = new SWSoundEngine ;
auto & S_sfx = soundEngine - > GetSounds ( ) ;
S_sfx . Resize ( countof ( voc ) ) ;
for ( auto & sfx : S_sfx ) { sfx . Clear ( ) ; sfx . lumpnum = sfx_empty ; }
for ( size_t i = 1 ; i < countof ( voc ) ; i + + )
{
auto & entry = voc [ i ] ;
2020-01-27 21:29:45 +00:00
auto lump = S_LookupSound ( entry . name ) ;
2019-12-19 00:20:43 +00:00
if ( lump > 0 )
{
auto & newsfx = S_sfx [ i ] ;
newsfx . name = entry . name ;
newsfx . lumpnum = lump ;
newsfx . NearLimit = 6 ;
newsfx . bTentative = false ;
}
}
soundEngine - > HashSounds ( ) ;
for ( auto & sfx : S_sfx )
{
soundEngine - > CacheSound ( & sfx ) ;
}
}
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
void SWSoundEngine : : CalcPosVel ( int type , const void * source , const float pt [ 3 ] , int channum , int chanflags , FSoundID chanSound , FVector3 * pos , FVector3 * vel , FSoundChan * chan )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
if ( pos ! = nullptr )
2015-05-19 21:54:34 +00:00
{
2019-12-18 21:13:19 +00:00
PLAYERp pp = Player + screenpeek ;
FVector3 campos = GetSoundPos ( ( vec3_t * ) pp ) ;
2019-12-19 00:20:43 +00:00
vec3_t * vpos = nullptr ;
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
if ( vel ) vel - > Zero ( ) ;
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
if ( type = = SOURCE_Unattached )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
pos - > X = pt [ 0 ] ;
pos - > Y = pt [ 1 ] ;
pos - > Z = pt [ 2 ] ;
2019-12-18 10:09:01 +00:00
}
2019-12-18 21:13:19 +00:00
else if ( type = = SOURCE_Actor | | type = = SOURCE_Player )
2019-12-18 10:09:01 +00:00
{
2019-12-19 00:20:43 +00:00
vpos = type = = SOURCE_Actor ? & ( ( SPRITEp ) source ) - > pos : ( vec3_t * ) & ( ( PLAYERp ) source ) - > posx ;
FVector3 npos = GetSoundPos ( vpos ) ;
* pos = npos ;
if ( ! ( chanflags & CHANEXF_NODOPPLER ) & & vel )
{
// Hack alert. Velocity may only be set if a) the sound is already running and b) an actual sound channel is modified.
// It remains to be seen if this is actually workable. I have my doubts. The velocity should be taken from a stable source.
if ( chan & & ! ( chanflags & ( CHANF_JUSTSTARTED | CHANF_EVICTED ) ) )
{
* vel = ( npos - FVector3 ( pt [ 0 ] , pt [ 1 ] , pt [ 2 ] ) ) * 40 ; // SW ticks 40 times a second.
chan - > Point [ 0 ] = npos . X ;
chan - > Point [ 1 ] = npos . Y ;
chan - > Point [ 2 ] = npos . Z ;
}
}
}
else if ( type = = SOURCE_Ambient )
{
auto sp = ( ( AmbientSound * ) source ) - > sp ;
2020-02-27 22:11:03 +00:00
vec3_t * vpos = & sp - > pos ;
2019-12-18 21:13:19 +00:00
FVector3 npos = GetSoundPos ( vpos ) ;
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
// Can the ambient sound see the player? If not, tone it down some.
2019-12-19 00:20:43 +00:00
if ( ( chanflags & CHANF_LOOP ) )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
if ( ! FAFcansee ( vpos - > x , vpos - > y , vpos - > z , sp - > sectnum , pp - > posx , pp - > posy , pp - > posz , pp - > cursectnum ) )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
auto distvec = npos - campos ;
npos = campos + distvec * 1.75f ; // Play more quietly
2019-12-18 10:09:01 +00:00
}
}
2019-12-19 00:20:43 +00:00
* pos = npos ;
}
2015-05-19 21:54:34 +00:00
2019-12-19 08:29:23 +00:00
if ( vpos & & chanflags & CHANEXF_DONTPAN )
2019-12-19 00:20:43 +00:00
{
// For unpanned sounds the volume must be set directly and the position taken from the listener.
* pos = campos ;
auto sdist = SoundDist ( vpos - > x , vpos - > y , vpos - > z , voc [ chanSound ] . voc_distance ) ;
if ( chan ) SetVolume ( chan , ( 255 - sdist ) * ( 1 / 255.f ) ) ;
2015-05-19 21:54:34 +00:00
}
2019-12-19 00:20:43 +00:00
2019-12-18 21:13:19 +00:00
if ( ( chanflags & CHANF_LISTENERZ ) & & campos ! = nullptr & & type ! = SOURCE_None )
2019-12-18 10:09:01 +00:00
{
2019-12-18 21:13:19 +00:00
pos - > Y = campos . Y ;
2019-12-18 10:09:01 +00:00
}
}
2015-05-19 21:54:34 +00:00
}
2019-12-18 21:13:19 +00:00
//==========================================================================
//
// Main function to update 3D sound array
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 21:13:19 +00:00
void DoUpdateSounds ( void )
{
2019-12-18 22:18:23 +00:00
PLAYERp pp = Player + screenpeek ;
SoundListener listener ;
listener . angle = - ( float ) pp - > pang * pi : : pi ( ) / 1024 ; // Build uses a period of 2048.
listener . velocity . Zero ( ) ;
listener . position = GetSoundPos ( ( vec3_t * ) & pp - > posx ) ;
listener . underwater = false ;
// This should probably use a real environment instead of the pitch hacking in S_PlaySound3D.
// listenactor->waterlevel == 3;
//assert(primaryLevel->Zones.Size() > listenactor->Sector->ZoneNumber);
listener . Environment = 0 ; // primaryLevel->Zones[listenactor->Sector->ZoneNumber].Environment;
listener . valid = true ;
listener . ListenerObject = pp ;
soundEngine - > SetListener ( listener ) ;
2019-12-19 00:20:43 +00:00
UpdateAmbients ( ) ;
2019-12-18 21:13:19 +00:00
soundEngine - > UpdateSounds ( ( int ) totalclock ) ;
}
2015-05-19 21:54:34 +00:00
2019-12-18 18:17:37 +00:00
//==========================================================================
//
2015-05-19 21:54:34 +00:00
// Play a sound
2019-12-18 18:17:37 +00:00
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2020-02-16 19:08:04 +00:00
int _PlaySound ( int num , SPRITEp sp , PLAYERp pp , vec3_t * pos , Voc3D_Flags flags , int channel , EChanFlags cflags )
2015-05-19 21:54:34 +00:00
{
2019-12-19 00:20:43 +00:00
if ( Prediction | | ! SoundEnabled ( ) | | ! soundEngine - > isValidSoundId ( num ) )
return - 1 ;
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
// Weed out parental lock sounds if PLock is active
if ( adult_lockout | | Global_PLock )
2015-05-19 21:54:34 +00:00
{
2019-12-19 00:20:43 +00:00
for ( unsigned i = 0 ; i < sizeof ( PLocked_Sounds ) ; i + + )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
if ( num = = PLocked_Sounds [ i ] )
return - 1 ;
2019-11-26 08:25:08 +00:00
}
2015-05-19 21:54:34 +00:00
}
2019-12-19 00:20:43 +00:00
auto vp = & voc [ num ] ;
int sourcetype = SOURCE_None ;
2020-02-16 19:08:04 +00:00
cflags | = channel = = 8 ? CHANF_OVERLAP : CHANF_NONE ; // for the default channel we do not want to have sounds stopping each other.
2019-12-19 00:20:43 +00:00
void * source = nullptr ;
// If the sound is not supposd to be positioned, it may not be linked to the launching actor.
if ( ! ( flags & v3df_follow ) )
2015-05-19 21:54:34 +00:00
{
2019-12-19 00:20:43 +00:00
if ( sp & & ! pos )
2019-12-18 10:09:01 +00:00
{
2019-12-19 00:20:43 +00:00
pos = & sp - > pos ;
sp = nullptr ;
2019-12-18 10:09:01 +00:00
}
2019-12-19 00:20:43 +00:00
else if ( pp & & ! pos )
2019-12-18 18:17:37 +00:00
{
2019-12-19 00:20:43 +00:00
pos = ( vec3_t * ) & pp - > posx ;
pp = nullptr ;
2019-12-18 10:09:01 +00:00
}
2019-12-19 00:20:43 +00:00
}
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
if ( pos ! = nullptr )
{
sourcetype = SOURCE_Unattached ;
}
else if ( sp ! = nullptr )
{
source = sp ;
sourcetype = SOURCE_Actor ;
}
else if ( pp ! = nullptr )
{
source = pp ;
sourcetype = SOURCE_Player ;
}
// Otherwise it's an unpositioned sound.
2019-12-18 18:17:37 +00:00
2019-12-19 00:20:43 +00:00
if ( flags & v3df_doppler ) cflags | = EChanFlags : : FromInt ( CHANEXF_NODOPPLER ) ; // this must ensure that CalcPosVel always zeros the velocity.
if ( flags & v3df_dontpan ) cflags | = EChanFlags : : FromInt ( CHANEXF_DONTPAN ) ; // beware of hackery to emulate this.
if ( vp - > voc_flags & vf_loop ) cflags | = CHANF_LOOP ; // with the new sound engine these can just be started and don't have to be stopped ever.
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
int pitch = 0 ;
if ( vp - > pitch_hi < = vp - > pitch_lo ) pitch = vp - > pitch_lo ;
else if ( vp - > pitch_hi ! = vp - > pitch_lo ) pitch = vp - > pitch_lo + ( STD_RANDOM_RANGE ( vp - > pitch_hi - vp - > pitch_lo ) ) ;
2015-05-19 21:54:34 +00:00
2019-12-19 00:20:43 +00:00
auto rolloff = GetRolloff ( vp - > voc_distance ) ;
FVector3 spos = pos ? GetSoundPos ( pos ) : FVector3 ( 0 , 0 , 0 ) ;
2019-12-19 08:29:23 +00:00
soundEngine - > StartSound ( sourcetype , source , & spos , channel , cflags , num , 1.f , ATTN_NORM , & rolloff , S_ConvertPitch ( pitch ) ) ;
2019-12-18 18:17:37 +00:00
return 1 ;
2019-12-18 10:09:01 +00:00
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void PlaySoundRTS ( int rts_num )
{
if ( ! adult_lockout & & SoundEnabled ( ) & & RTS_IsInitialized ( ) & & snd_speech )
{
auto sid = RTS_GetSoundID ( rts_num - 1 ) ;
if ( sid ! = - 1 )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
soundEngine - > StartSound ( SOURCE_Unattached , nullptr , nullptr , CHAN_VOICE , 0 , sid , 0.8f , ATTN_NONE ) ;
}
}
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void COVER_SetReverb ( int amt )
{
FX_SetReverb ( amt ) ;
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
// Deletes vocs in the 3D sound queue with no owners
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void DeleteNoSoundOwner ( short spritenum )
{
2019-12-25 10:26:19 +00:00
if ( ! soundEngine ) return ;
2019-12-18 10:09:01 +00:00
SPRITEp sp = & sprite [ spritenum ] ;
soundEngine - > EnumerateChannels ( [ = ] ( FSoundChan * chan )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
if ( chan - > Source = = sp & & chan - > ChanFlags & CHANF_LOOP )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
soundEngine - > StopChannel ( chan ) ;
2015-05-19 21:54:34 +00:00
}
2019-12-18 10:09:01 +00:00
return false ;
} ) ;
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
// This is called from KillSprite to kill a follow sound with no valid sprite owner
2019-12-18 21:13:19 +00:00
// Stops any active sound with the follow bit set, even play once sounds.
2019-12-18 10:09:01 +00:00
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void DeleteNoFollowSoundOwner ( short spritenum )
{
SPRITEp sp = & sprite [ spritenum ] ;
2019-12-18 21:13:19 +00:00
soundEngine - > StopSound ( SOURCE_Actor , sp , - 1 ) ; // all non-follow sounds are SOURCE_Unattached
2019-12-18 10:09:01 +00:00
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
// Terminate the sounds list
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void Terminate3DSounds ( void )
2015-05-19 21:54:34 +00:00
{
2019-12-18 10:09:01 +00:00
soundEngine - > EnumerateChannels ( [ ] ( FSoundChan * chan )
{
if ( chan - > SourceType = = SOURCE_Actor | | chan - > SourceType = = SOURCE_Unattached | |
( chan - > SourceType = = SOURCE_Player & & chan - > EntChannel ! = CHAN_VOICE ) )
{
soundEngine - > StopChannel ( chan ) ;
}
return false ;
} ) ;
}
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
//==========================================================================
//
// no longer needed, only left to avoid changing the game code
//
//==========================================================================
2015-05-19 21:54:34 +00:00
2019-12-18 10:09:01 +00:00
void Set3DSoundOwner ( short spritenum )
{
2015-05-19 21:54:34 +00:00
}
2019-12-18 10:09:01 +00:00
//==========================================================================
//
2015-05-19 21:54:34 +00:00
// Play a sound using special sprite setup
2019-12-18 10:09:01 +00:00
//
//==========================================================================
void PlaySpriteSound ( short spritenum , int attrib_ndx , Voc3D_Flags flags )
2015-05-19 21:54:34 +00:00
{
SPRITEp sp = & sprite [ spritenum ] ;
USERp u = User [ spritenum ] ;
ASSERT ( u ) ;
2019-12-18 10:09:01 +00:00
PlaySound ( u - > Attrib - > Sounds [ attrib_ndx ] , sp , flags ) ;
}
//==========================================================================
//
//
//
//==========================================================================
int _PlayerSound ( int num , PLAYERp pp )
{
int handle ;
VOC_INFOp vp ;
if ( Prediction )
return 0 ;
if ( pp < Player | | pp > = Player + MAX_SW_PLAYERS )
{
return 0 ;
}
if ( num < 0 | | num > = DIGI_MAX | | ! soundEngine - > isValidSoundId ( num ) )
return 0 ;
if ( TEST ( pp - > Flags , PF_DEAD ) ) return 0 ; // You're dead, no talking!
// If this is a player voice and he's already yacking, forget it.
vp = & voc [ num ] ;
// Not a player voice, bail.
if ( vp - > priority ! = PRI_PLAYERVOICE & & vp - > priority ! = PRI_PLAYERDEATH )
return 0 ;
// He wasn't talking, but he will be now.
if ( ! soundEngine - > IsSourcePlayingSomething ( SOURCE_Player , pp , CHAN_VOICE ) )
{
soundEngine - > StartSound ( SOURCE_Player , pp , nullptr , CHAN_VOICE , 0 , num , 1.f , ATTN_NORM ) ;
}
return 0 ;
2015-05-19 21:54:34 +00:00
}
2019-12-18 10:09:01 +00:00
void StopPlayerSound ( PLAYERp pp )
{
soundEngine - > StopSound ( SOURCE_Player , pp , CHAN_VOICE ) ;
}
2019-12-17 20:33:53 +00:00
2019-12-18 18:17:37 +00:00
bool SoundValidAndActive ( SPRITEp spr , int channel )
{
return soundEngine - > IsSourcePlayingSomething ( SOURCE_Actor , spr , channel ) ;
}
2019-12-17 20:33:53 +00:00
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
=
= High level sound code ( not directly engine related )
=
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
int PlayerPainVocs [ ] =
{
DIGI_PLAYERPAIN1 ,
DIGI_PLAYERPAIN2 ,
DIGI_PLAYERPAIN3 ,
DIGI_PLAYERPAIN4 ,
DIGI_PLAYERPAIN5
} ;
// Don't have these sounds yet
int PlayerLowHealthPainVocs [ ] =
{
DIGI_HURTBAD1 ,
DIGI_HURTBAD2 ,
DIGI_HURTBAD3 ,
DIGI_HURTBAD4 ,
DIGI_HURTBAD5
} ;
int TauntAIVocs [ ] =
{
DIGI_TAUNTAI1 ,
DIGI_TAUNTAI2 ,
DIGI_TAUNTAI3 ,
DIGI_TAUNTAI4 ,
DIGI_TAUNTAI5 ,
DIGI_TAUNTAI6 ,
DIGI_TAUNTAI7 ,
DIGI_TAUNTAI8 ,
DIGI_TAUNTAI9 ,
DIGI_TAUNTAI10 ,
DIGI_COWABUNGA ,
DIGI_NOCHARADE ,
DIGI_TIMETODIE ,
DIGI_EATTHIS ,
DIGI_FIRECRACKERUPASS ,
DIGI_HOLYCOW ,
DIGI_HAHA2 ,
DIGI_HOLYPEICESOFCOW ,
DIGI_HOLYSHIT ,
DIGI_HOLYPEICESOFSHIT ,
DIGI_PAYINGATTENTION ,
DIGI_EVERYBODYDEAD ,
DIGI_KUNGFU ,
DIGI_HOWYOULIKEMOVE ,
DIGI_HAHA3 ,
DIGI_NOMESSWITHWANG ,
DIGI_RAWREVENGE ,
DIGI_YOULOOKSTUPID ,
DIGI_TINYDICK ,
DIGI_NOTOURNAMENT ,
DIGI_WHOWANTSWANG ,
DIGI_MOVELIKEYAK ,
DIGI_ALLINREFLEXES
} ;
int PlayerGetItemVocs [ ] =
{
DIGI_GOTITEM1 ,
DIGI_HAHA1 ,
DIGI_BANZAI ,
DIGI_COWABUNGA ,
DIGI_TIMETODIE
} ;
int PlayerYellVocs [ ] =
{
DIGI_PLAYERYELL1 ,
DIGI_PLAYERYELL2 ,
DIGI_PLAYERYELL3
} ;
2019-12-17 22:25:07 +00:00
//==========================================================================
//
// PLays music
//
//==========================================================================
2019-12-17 20:33:53 +00:00
extern short Level ;
SWBOOL PlaySong ( const char * mapname , const char * song_file_name , int cdaudio_track , bool isThemeTrack ) //(nullptr, nullptr, -1, false) starts the normal level music.
{
// Play CD audio if enabled.
2020-02-08 20:36:57 +00:00
if ( cdaudio_track > = 0 & & ( mus_redbook | | * song_file_name = = 0 ) )
2019-12-17 20:33:53 +00:00
{
FStringf trackname ( " track%02d.ogg " , cdaudio_track ) ;
2019-12-26 12:04:29 +00:00
if ( ! Mus_Play ( mapname , trackname , true ) )
2019-12-17 20:33:53 +00:00
{
buildprintf ( " Can't find CD track %i! \n " , cdaudio_track ) ;
}
2020-02-08 20:36:57 +00:00
else return true ;
2019-12-17 20:33:53 +00:00
}
2019-12-26 12:04:29 +00:00
if ( ! Mus_Play ( mapname , song_file_name , true ) )
{
// try the CD track anyway if no MIDI could be found (the original game doesn't have any MIDI, it was CD Audio only, this avoids no music playing id mus_redbook is off.)
FStringf trackname ( " track%02d.ogg " , cdaudio_track ) ;
if ( ! Mus_Play ( nullptr , trackname , true ) ) return false ;
}
return true ;
2019-12-17 20:33:53 +00:00
}
2019-12-18 18:17:37 +00:00
void StopSound ( void )
{
2019-12-25 10:26:19 +00:00
// This gets also called on shutdown.
2020-02-13 23:52:57 +00:00
StopAmbientSound ( ) ;
2019-12-25 10:26:19 +00:00
if ( soundEngine ) soundEngine - > StopAllChannels ( ) ;
2019-12-18 18:17:37 +00:00
Mus_Stop ( ) ;
}
2020-02-11 18:28:25 +00:00
void StopFX ( )
{
2020-02-13 23:52:57 +00:00
StopAmbientSound ( ) ;
2020-02-11 18:28:25 +00:00
if ( soundEngine ) soundEngine - > StopAllChannels ( ) ;
}
2019-12-17 20:33:53 +00:00
2019-10-09 16:09:05 +00:00
END_SW_NS