raze/source/games/exhumed/src/sound.cpp

810 lines
22 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 2010-2019 EDuke32 developers and contributors
Copyright (C) 2019 Nuke.YKT, sirlemonhead
This file is part of PCExhumed.
PCExhumed is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License version 2
as published by the Free Software Foundation.
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.
*/
//-------------------------------------------------------------------------
#include "ns.h"
2019-09-18 16:11:06 +00:00
#include "build.h"
#include "engine.h"
#include "exhumed.h"
#include "sound.h"
#include "aistuff.h"
2019-09-18 16:11:06 +00:00
#include "player.h"
#include "sequence.h"
#include "raze_sound.h"
#include "mapinfo.h"
BEGIN_PS_NS
2019-09-18 16:11:06 +00:00
const char *SoundFiles[kMaxSoundFiles] =
{
"spl_big",
"spl_smal",
"bubble_l",
"grn_drop",
"p_click",
"grn_roll",
"cosprite",
"m_chant0",
"anu_icu",
"item_reg",
"item_spe", // 10
"item_key",
"torch_on", // 12
"jon_bnst",
"jon_gasp",
"jon_land",
"jon_gags",
"jon_fall",
"jon_drwn",
"jon_air1",
"jon_glp1", // 20
"jon_bbwl",
"jon_pois",
"amb_ston",
"cat_icu",
"bubble_h",
"set_land",
"jon_hlnd",
"jon_laf2",
"spi_jump",
"jon_scub", // 30
"item_use",
"tr_arrow",
"swi_foot",
"swi_ston",
"swi_wtr1",
"tr_fire",
"m_skull5",
"spi_atak",
"anu_hit",
"fishdies", // 40
"scrp_icu",
"jon_wade",
"amb_watr",
"tele_1",
"wasp_stg",
"res",
"drum4",
"rex_icu",
"m_hits_u",
"q_tail", // 50
"vatr_mov",
"jon_hit3",
"jon_t_2", // 53
"jon_t_1",
"jon_t_5",
"jon_t_6",
"jon_t_8",
"jon_t_4",
"rasprit1",
"jon_fdie", // 60
"wijaf1",
"ship_1",
"saw_on",
"ra_on",
"amb_ston", // 65
"vatr_stp", // 66
"mana1",
"mana2",
"ammo",
"pot_pc1", // 70?
"pot_pc2",
"weapon",
"alarm",
"tick1",
"scrp_zap", // 75
"jon_t_3",
"jon_laf1",
"blasted",
"jon_air2" // 79
};
int nStopSound;
int nStoneSound;
int nSwitchSound;
int nElevSound;
int nCreepyTimer;
bool looped[kMaxSounds];
int16_t StaticSound[kMaxSounds];
int fakesources[] = { 0, 1, 2, 3 };
2020-02-24 23:40:51 +00:00
int swirlysources[4]= { 0, 1, 2, 3 };
FVector3 amb, creepy;
2019-09-18 16:11:06 +00:00
int nLocalChan = 0;
//==========================================================================
//
//
//
//==========================================================================
2019-09-18 16:11:06 +00:00
class EXSoundEngine : public RazeSoundEngine
{
// 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;
public:
EXSoundEngine()
{
S_Rolloff.RolloffType = ROLLOFF_Doom;
S_Rolloff.MinDistance = 0;
S_Rolloff.MaxDistance = 1536;
}
void StopChannel(FSoundChan* chan) override
{
if (chan && chan->SysChannel != NULL && !(chan->ChanFlags & CHANF_EVICTED) && chan->SourceType == SOURCE_Actor)
{
chan->Source = NULL;
chan->SourceType = SOURCE_Unattached;
}
SoundEngine::StopChannel(chan);
}
};
//==========================================================================
//
//
//
//==========================================================================
2019-09-18 16:11:06 +00:00
int LoadSound(const char* name)
{
char nname[9]{};
for (int i = 0; i < 8 && name[i]; i++) nname[i] = name[i];
2022-11-24 16:46:39 +00:00
FSoundID sndid = soundEngine->FindSoundNoHash(nname);
if (sndid.isvalid()) return sndid.index() - 1;
FStringf filename("%s.voc", nname);
auto lump = S_LookupSound(filename);
if (lump > 0)
{
2023-08-19 17:41:40 +00:00
auto check = fileSystem.ReadFile(lump);
bool loops = false;
2023-08-19 17:41:40 +00:00
if (check.GetSize() > 26 && check.GetString()[26] == 6 && !memcmp("Creative Voice File", check.GetString(), 19))
{
// This game uses the actual loop point information in the sound data as its only means to check if a sound is looped.
loops = true;
}
2022-11-24 16:46:39 +00:00
FSoundID retval = soundEngine->AddSoundLump(nname, lump, 0, -1, 6);
soundEngine->CacheSound(retval);
2022-11-24 16:46:39 +00:00
looped[retval.index() - 1] = loops;
return retval.index() - 1;
}
else if (!isShareware()) // demo tries to load sound files it doesn't have
{
Printf("Unable to open sound '%s'!\n", filename.GetChars());
}
return -1;
}
//==========================================================================
//
//
//
//==========================================================================
void GameInterface::StartSoundEngine()
2019-09-18 16:11:06 +00:00
{
soundEngine = new EXSoundEngine;
}
2019-09-18 16:11:06 +00:00
void InitFX(void)
{
auto& S_sfx = soundEngine->GetSounds();
S_sfx.Resize(1);
for (size_t i = 0; i < kMaxSoundFiles; i++)
2019-09-18 16:11:06 +00:00
{
StaticSound[i] = LoadSound(SoundFiles[i]);
2019-09-18 16:11:06 +00:00
}
soundEngine->HashSounds();
nCreepyTimer = kCreepyCount;
}
2019-09-18 16:11:06 +00:00
//==========================================================================
//
//
//
//==========================================================================
void GetSpriteSoundPitch(int* pVolume, int* pPitch)
{
auto pSoundSect = getPlayer(nLocalPlayer)->pPlayerViewSect;
2021-11-22 23:20:15 +00:00
int nLocalSectFlags = pSoundSect->Flag;
if (nLocalSectFlags & kSectUnderwater)
2019-09-18 16:11:06 +00:00
{
if (*pVolume == 255)
{
*pVolume >>= 1;
*pPitch -= 1200;
}
}
else
{
if (*pVolume < 255)
{
*pVolume = 255;
*pPitch += 1200;
}
2019-09-18 16:11:06 +00:00
}
}
//==========================================================================
//
//
//
//==========================================================================
2019-09-18 16:11:06 +00:00
void BendAmbientSound(void)
{
soundEngine->EnumerateChannels([](FSoundChan* chan)
{
if (chan->SourceType == SOURCE_Ambient && chan->Source == &amb)
{
soundEngine->SetPitch(chan, (nDronePitch + 11800) / 11025.f);
}
return 1;
});
}
//==========================================================================
//
//
//
//==========================================================================
void PlayLocalSound(int nSound, int nRate, bool unattached, EChanFlags cflags)
{
if (!SoundEnabled()) return;
2022-11-24 16:46:39 +00:00
auto soundid = FSoundID::fromInt(nSound + 1);
if (!soundEngine->isValidSoundId(soundid))
2019-09-18 16:11:06 +00:00
{
Printf("PlayLocalSound: Invalid sound nSound == %i, nRate == %i\n", nSound, nRate);
return;
2019-09-18 16:11:06 +00:00
}
if (looped[nSound]) cflags |= CHANF_LOOP;
FSoundChan* chan;
2019-12-27 22:09:05 +00:00
if (!unattached)
{
2022-11-24 16:46:39 +00:00
if (!(cflags & CHANF_UI) && soundEngine->IsSourcePlayingSomething(SOURCE_None, nullptr, CHAN_BODY, soundid)) return;
soundEngine->StopSound(SOURCE_None, nullptr, CHAN_BODY);
2022-11-24 16:46:39 +00:00
chan = soundEngine->StartSound(SOURCE_None, nullptr, nullptr, CHAN_BODY, cflags, soundid, 1.f, ATTN_NONE, nullptr);
}
else
{
2022-11-24 16:46:39 +00:00
chan = soundEngine->StartSound(SOURCE_None, nullptr, nullptr, CHAN_VOICE, CHANF_OVERLAP|cflags, soundid, 1.f, ATTN_NONE, nullptr);
2019-12-27 22:09:05 +00:00
}
2019-12-27 22:09:05 +00:00
if (nRate && chan)
{
float ratefac = (11025 + nRate) / 11025.f;
2019-12-27 22:09:05 +00:00
soundEngine->SetPitch(chan, ratefac);
}
2019-09-18 16:11:06 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
int LocalSoundPlaying(void)
2019-09-18 16:11:06 +00:00
{
return soundEngine->EnumerateChannels([](FSoundChan* chan)
{
return chan->SourceType == SOURCE_None;
});
}
//==========================================================================
//
//
//
//==========================================================================
void StopLocalSound(void)
{
soundEngine->StopSound(SOURCE_None, nullptr, -1);
}
//==========================================================================
//
//
//
//==========================================================================
int nNextFreq;
int nSwirlyFrames;
2019-09-18 16:11:06 +00:00
void StartSwirly(int nActiveSound)
{
if (!SoundEnabled()) return;
auto &swirly = swirlysources[nActiveSound];
2019-09-18 16:11:06 +00:00
int nPitch = nNextFreq - RandomSize(9);
nNextFreq = 25000 - RandomSize(10) * 6;
2019-09-18 16:11:06 +00:00
if (nNextFreq > 32000)
nNextFreq = 32000;
int nVolume = nSwirlyFrames + 1;
2019-09-18 16:11:06 +00:00
if (nVolume >= 220)
nVolume = 220;
soundEngine->StopSound(SOURCE_Swirly, &swirly, -1);
2022-11-24 16:46:39 +00:00
soundEngine->StartSound(SOURCE_Swirly, &swirly, nullptr, CHAN_BODY, CHANF_TRANSIENT, FSoundID::fromInt(StaticSound[kSoundMana1]+1), nVolume / 255.f, ATTN_NONE, nullptr, nPitch / 11025.f);
2019-09-18 16:11:06 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
2019-09-18 16:11:06 +00:00
void StartSwirlies()
{
StopAllSounds();
2019-09-18 16:11:06 +00:00
nNextFreq = 19000;
nSwirlyFrames = 0;
2022-12-13 11:13:08 +00:00
for (int i = 0; i < 4; i++)
2019-09-18 16:11:06 +00:00
StartSwirly(i);
}
//==========================================================================
//
//
//
//==========================================================================
2019-09-18 16:11:06 +00:00
void UpdateSwirlies()
{
2019-09-18 16:11:06 +00:00
nSwirlyFrames++;
2022-12-13 11:13:08 +00:00
for (int i = 0; i < 4; i++)
2019-09-18 16:11:06 +00:00
{
if (!soundEngine->IsSourcePlayingSomething(SOURCE_Swirly, &swirlysources[i], -1))
2019-09-18 16:11:06 +00:00
StartSwirly(i);
}
}
//==========================================================================
//
//
//
//==========================================================================
void SoundBigEntrance(void)
{
if (!SoundEnabled()) return;
StopAllSounds();
for (int i = 0; i < 4; i++)
{
int nPitch = 11025 + (i * 512 - 1200);
//pASound->snd_pitch = nPitch;
soundEngine->StopSound(SOURCE_EXBoss, &fakesources[i], -1);
2022-11-24 16:46:39 +00:00
soundEngine->StartSound(SOURCE_EXBoss, &fakesources[i], nullptr, CHAN_BODY, CHANF_TRANSIENT, FSoundID::fromInt(StaticSound[kSoundTorchOn]+1), 200 / 255.f, ATTN_NONE, nullptr, nPitch / 11025.f);
}
}
//==========================================================================
//
//
//
//==========================================================================
void EXSoundEngine::CalcPosVel(int type, const void* source, const float pt[3], int channum, int chanflags, FSoundID chanSound, FVector3* pos, FVector3* vel, FSoundChan* chan)
{
if (pos != nullptr)
{
2022-08-23 21:36:23 +00:00
DVector3 campos;
if (nSnakeCam > -1)
{
Snake* pSnake = &SnakeList[nSnakeCam];
2022-08-23 21:36:23 +00:00
campos = pSnake->pSprites[0]->spr.pos;
}
else if (const auto pActor = getPlayer(nLocalPlayer)->GetActor())
{
2023-03-18 01:49:44 +00:00
campos = pActor->spr.pos;
}
auto fcampos = GetSoundPos(campos);
if (vel) vel->Zero();
if (type == SOURCE_Ambient)
{
*pos = *(FVector3*)source;
}
else if (type == SOURCE_Unattached)
{
pos->X = pt[0];
pos->Y = pt[1];
pos->Z = pt[2];
}
// Do some angular magic. The original was just 2D panning which in a 3D sound field is not sufficient.
else if (type == SOURCE_Swirly)
{
int which = *(int*)source;
double phase = PlayClock << (4 + which);
pos->X = fcampos.X + float(256 * BobVal(phase + 512));
pos->Z = fcampos.Z + float(256 * BobVal(phase));
}
else if (type == SOURCE_EXBoss)
{
int which = *(int*)source;
*pos = fcampos;
2022-12-13 11:13:08 +00:00
// Should be positioned in 90° intervals.
switch (which)
{
default:
case 0: pos->X -= 256; break;
2020-02-24 23:40:51 +00:00
case 1: pos->Z -= 256; break;
case 2: pos->X += 256; break;
2020-02-24 23:40:51 +00:00
case 3: pos->Z += 256; break;
}
}
else if (type == SOURCE_Actor)
{
auto actor = (DExhumedActor*)source;
assert(actor != nullptr);
if (actor != nullptr)
{
2022-08-23 21:36:23 +00:00
*pos = GetSoundPos(actor->spr.pos);
}
}
if ((chanflags & CHANF_LISTENERZ) && type != SOURCE_None)
{
pos->Y = fcampos.Y;
}
2019-09-18 16:11:06 +00:00
}
}
//==========================================================================
//
//
//
//==========================================================================
void GameInterface::UpdateSounds()
{
if (nFreeze)
2019-09-18 16:11:06 +00:00
return;
const auto pActor = getPlayer(nLocalPlayer)->GetActor();
2023-03-18 01:44:37 +00:00
2022-08-23 21:36:23 +00:00
DVector3 pos;
DAngle ang;
2019-09-18 16:11:06 +00:00
if (nSnakeCam > -1)
{
Snake* pSnake = &SnakeList[nSnakeCam];
2022-08-23 21:36:23 +00:00
pos = pSnake->pSprites[0]->spr.pos;
ang = pSnake->pSprites[0]->spr.Angles.Yaw;
2019-09-18 16:11:06 +00:00
}
2023-03-18 01:44:37 +00:00
else if (pActor)
2019-09-18 16:11:06 +00:00
{
2023-03-18 01:49:44 +00:00
pos = pActor->spr.pos;
2023-03-18 01:44:37 +00:00
ang = pActor->spr.Angles.Yaw;
2019-09-18 16:11:06 +00:00
}
else pos.Zero();
SoundListener listener;
listener.angle = float(-ang.Radians()); // Build uses a period of 2048.
listener.velocity.Zero();
listener.position = GetSoundPos(pos);
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;
soundEngine->SetListener(listener);
soundEngine->EnumerateChannels([](FSoundChan* chan)
2019-09-18 16:11:06 +00:00
{
if (!(chan->ChanFlags & (CHANF_UI|CHANF_FORGETTABLE)))
{
int nVolume = int(chan->Volume * 255);
2020-02-24 17:34:15 +00:00
int nPitch = int(chan->Pitch * (11025.f / 128.f)) - 11025;
int oVolume = nVolume;
GetSpriteSoundPitch(&nVolume, &nPitch);
if (oVolume != nVolume)
{
soundEngine->SetPitch(chan, (11025 + nPitch) / 11025.f);
soundEngine->SetVolume(chan, nVolume / 255.f);
}
}
return 0;
});
}
//==========================================================================
//
//
//
//==========================================================================
void PlayFX2(int nSound, DExhumedActor* pActor, int sectf, EChanFlags chanflags, int sprflags, const DVector3* soundpos)
2019-09-18 16:11:06 +00:00
{
if (!SoundEnabled()) return;
2022-11-25 10:26:24 +00:00
auto soundid = FSoundID::fromInt((nSound & 0x1ff) + 1);
2022-11-24 16:46:39 +00:00
if (!soundEngine->isValidSoundId(soundid))
{
2021-10-22 16:31:28 +00:00
Printf("PlayFX2: Invalid sound nSound == %i\n", nSound);
return;
}
bool fullvol = false, hiprio = false;
2021-10-22 16:31:28 +00:00
if (pActor)
2019-09-18 16:11:06 +00:00
{
fullvol = (sprflags & 0x2000) != 0;
hiprio = (sprflags & 0x4000) != 0;
soundpos = &pActor->spr.pos;
2019-09-18 16:11:06 +00:00
}
if (!soundpos) return;
2019-09-18 16:11:06 +00:00
int nVolume = 255;
bool allowMultiple = (nSound & 0x2000) != 0;
bool forcePlay = (nSound & 0x1000) != 0;
bool midprio = (nSound & 0x4000) != 0;
int prio = 0;
if (forcePlay || midprio) prio = 1000;
2021-10-22 16:31:28 +00:00
else if (pActor != nullptr && hiprio) prio = 2000;
int v10 = (nSound&0xe00)>>9;
2019-09-18 16:11:06 +00:00
nSound &= 0x1ff;
int nPitch = 0;
if (v10) nPitch = -(totalmoves&((1<<v10)-1))*16;
2019-09-18 16:11:06 +00:00
GetSpriteSoundPitch(&nVolume, &nPitch);
2019-09-18 16:11:06 +00:00
FVector3 vv = GetSoundPos(*soundpos);
// Check if this sound is allowed to play or if it must stop some other sound.
if (!forcePlay)
{
bool res = soundEngine->EnumerateChannels([=](FSoundChan* chan)
{
2021-10-22 16:31:28 +00:00
if (chan->SourceType == SOURCE_Actor && pActor != nullptr)
{
if (prio >= chan->UserData)
{
2022-11-24 16:46:39 +00:00
if (chan->SoundID == soundid)
{
if (!allowMultiple && pActor == chan->Source)
return 1;
}
else if (pActor == chan->Source)
{
soundEngine->StopChannel(chan);
return -1;
}
}
}
2021-10-22 16:31:28 +00:00
else if (chan->SourceType == SOURCE_Unattached && pActor != nullptr)
{
2022-11-24 16:46:39 +00:00
if (chan->SoundID == soundid)
{
if (vv.X == chan->Point[0] && vv.Y == chan->Point[1] && vv.Z == chan->Point[2])
return 1;
}
}
return 0;
});
if (res) return;
}
FSoundChan* chan = nullptr;
2021-10-22 16:31:28 +00:00
if (pActor != nullptr)
2019-09-18 16:11:06 +00:00
{
2022-11-24 16:46:39 +00:00
chan = soundEngine->StartSound(SOURCE_Actor, pActor, nullptr, CHAN_BODY, chanflags| CHANF_OVERLAP, soundid, nVolume / 255.f,fullvol? 0.5f : ATTN_NORM, nullptr, (11025 + nPitch) / 11025.f);
2019-09-18 16:11:06 +00:00
}
else
{
2022-11-24 16:46:39 +00:00
chan = soundEngine->StartSound(SOURCE_Unattached, nullptr, &vv, CHAN_BODY, chanflags | CHANF_OVERLAP, soundid, nVolume / 255.f, ATTN_NORM, nullptr, (11025 + nPitch) / 11025.f);
}
if (chan)
{
if (sectf) chan->UserData = 2000;
else chan->UserData = prio;
2019-09-18 16:11:06 +00:00
}
// Nuke: added nSprite >= 0 check
if (pActor != getPlayer(nLocalPlayer)->GetActor() && pActor != nullptr && (pActor->spr.cstat & CSTAT_SPRITE_BLOCK_ALL))
nCreepyTimer = kCreepyCount;
}
//==========================================================================
//
//
//
//==========================================================================
void PlayFXAtXYZ(int ax, const DVector3& pos, EChanFlags chanflags, int sectf)
{
PlayFX2(ax, nullptr, sectf, chanflags, 0, &pos);
2019-09-18 16:11:06 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
2021-11-22 23:35:29 +00:00
void CheckAmbience(sectortype* sect)
{
if (!SoundEnabled()) return;
if (sect->Sound != -1)
{
2021-11-22 19:18:14 +00:00
auto pSector2 = sect->pSoundSect;
2022-11-15 14:44:33 +00:00
walltype* pWall = pSector2->walls.Data();
if (!soundEngine->IsSourcePlayingSomething(SOURCE_Ambient, &amb, 0))
{
2022-08-20 18:25:38 +00:00
DVector3 v = { pWall->pos.X, pWall->pos.Y, pSector2->floorz };
amb = GetSoundPos(v);
2022-11-24 16:46:39 +00:00
soundEngine->StartSound(SOURCE_Ambient, &amb, nullptr, CHAN_BODY, CHANF_TRANSIENT, FSoundID::fromInt(sect->Sound + 1), 1.f, ATTN_NORM);
return;
}
soundEngine->EnumerateChannels([=](FSoundChan* chan)
{
if (chan->SourceType == SOURCE_Ambient)
{
2021-11-22 19:18:14 +00:00
if (sect == pSector2)
{
amb = GetSoundPos(getPlayer(0)->GetActor()->spr.pos);
}
else
{
2022-08-20 18:25:38 +00:00
DVector3 v = { pWall->pos.X, pWall->pos.Y, pSector2->floorz };
amb = GetSoundPos(v);
}
return 1;
}
return 0;
});
}
else
{
soundEngine->StopSound(SOURCE_Ambient, &amb, -1);
}
}
//==========================================================================
//
//
//
//==========================================================================
void UpdateCreepySounds()
{
if ((currentLevel->gameflags & LEVEL_EX_COUNTDOWN) || nFreeze || !SoundEnabled())
return;
nCreepyTimer--;
if (nCreepyTimer <= 0)
{
if (Level.kills.got < Level.kills.max && !(getPlayer(nLocalPlayer)->pPlayerViewSect->Flag & 0x2000))
{
const auto creepySeq = getSequence("creepy");
const auto seqFrameSound = creepySeq->frames[totalmoves % creepySeq->frames.Size()].sound;
if (seqFrameSound >= 0 && (seqFrameSound & 0x1ff) < kMaxSounds)
{
2022-09-09 17:08:28 +00:00
DVector2 adder;
adder.X = ((totalmoves + 32) & 31) / 16.;
if (totalmoves & 1)
2022-09-09 17:08:28 +00:00
adder.X = -adder.X;
adder.Y = ((totalmoves + 32) & 63) / 16.;
if (totalmoves & 2)
2022-09-09 17:08:28 +00:00
adder.Y = -adder.Y;
auto sp = getPlayer(nLocalPlayer)->GetActor()->spr.pos + adder;
creepy = GetSoundPos(sp);
auto soundid = FSoundID::fromInt((seqFrameSound & 0x1ff) + 1);
2022-11-24 16:46:39 +00:00
if (!soundEngine->isValidSoundId(soundid))
{
return;
}
int nVolume = 255;
int v10 = (seqFrameSound & 0xe00) >> 9;
int nPitch = 0;
if (v10) nPitch = -(totalmoves & ((1 << v10) - 1)) * 16;
GetSpriteSoundPitch(&nVolume, &nPitch);
soundEngine->StopSound(SOURCE_Ambient, &creepy, CHAN_BODY);
2022-11-24 16:46:39 +00:00
soundEngine->StartSound(SOURCE_Ambient, &creepy, nullptr, CHAN_BODY, CHANF_TRANSIENT, soundid, nVolume / 255.f, ATTN_NONE, nullptr, (11025 + nPitch) / 11025.f);
}
}
nCreepyTimer = kCreepyCount;
}
}
//==========================================================================
//
//
//
//==========================================================================
2021-10-21 22:08:40 +00:00
void StopActorSound(DExhumedActor *pActor)
2019-09-18 16:11:06 +00:00
{
2021-10-21 22:08:40 +00:00
if (pActor)
soundEngine->StopSound(SOURCE_Actor, pActor, -1);
2019-09-18 16:11:06 +00:00
}
void StopAllSounds(void)
{
soundEngine->StopAllChannels();
2019-09-18 16:11:06 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
void PlayTitleSound(void)
{
PlayLocalSound(StaticSound[kSoundItemSpecial], 0, false, CHANF_UI);
}
void PlayGameOverSound(void)
{
PlayLocalSound(StaticSound[kSoundJonLaugh2], 0, false, CHANF_UI);
}
DEFINE_ACTION_FUNCTION(_Exhumed, PlayLocalSound)
{
PARAM_PROLOGUE;
PARAM_INT(snd);
PARAM_INT(pitch);
PARAM_BOOL(unatt);
PARAM_INT(flags);
PlayLocalSound(StaticSound[snd], pitch, unatt, EChanFlags::FromInt(flags));
return 0;
}
DEFINE_ACTION_FUNCTION_NATIVE(_Exhumed, StopLocalSound, StopLocalSound)
{
StopLocalSound();
return 0;
}
DEFINE_ACTION_FUNCTION(_Exhumed, LocalSoundPlaying)
{
2022-11-24 16:46:39 +00:00
ACTION_RETURN_BOOL(soundEngine->IsSourcePlayingSomething(SOURCE_None, nullptr, CHAN_AUTO));
}
DEFINE_ACTION_FUNCTION(_Exhumed, PlayCDTrack)
{
PARAM_PROLOGUE;
PARAM_INT(track);
PARAM_BOOL(loop);
playCDtrack(track, loop);
return 0;
}
END_PS_NS