172 lines
5.3 KiB
C++
172 lines
5.3 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#ifndef AI_SENTENCE_H
|
|
#define AI_SENTENCE_H
|
|
|
|
#include "ai_component.h"
|
|
#include "ai_basenpc.h"
|
|
#include "SoundEmitterSystem/isoundemittersystembase.h"
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Sentence helper class used by humanoids sometimes. To use:
|
|
// 1) Include it into the leaf class
|
|
// 2) Use DEFINE_EMBEDDED to save/load its state
|
|
// 3) Call Init in the CreateComponents method
|
|
// 4) Use Speak() when it's time to speak
|
|
// 5) Add m_Sentences.UpdateSentenceQueue(); to the PrescheduleThink method of an NPC
|
|
// to get queued sentence support working
|
|
//-----------------------------------------------------------------------------
|
|
enum SentenceCriteria_t
|
|
{
|
|
SENTENCE_CRITERIA_ALWAYS = 0,
|
|
SENTENCE_CRITERIA_NORMAL, // Obeys gag rules
|
|
SENTENCE_CRITERIA_IN_SQUAD,
|
|
SENTENCE_CRITERIA_SQUAD_LEADER,
|
|
};
|
|
|
|
enum SentencePriority_t
|
|
{
|
|
SENTENCE_PRIORITY_INVALID = -1,
|
|
SENTENCE_PRIORITY_NORMAL = 0,
|
|
SENTENCE_PRIORITY_MEDIUM = 1,
|
|
SENTENCE_PRIORITY_HIGH = 2,
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// This is the met of the class
|
|
//-----------------------------------------------------------------------------
|
|
abstract_class CAI_SentenceBase : public CAI_Component
|
|
{
|
|
DECLARE_CLASS_NOBASE( CAI_SentenceBase );
|
|
DECLARE_SIMPLE_DATADESC();
|
|
|
|
public:
|
|
CAI_SentenceBase();
|
|
|
|
void SetVoicePitch( int voicePitch );
|
|
int GetVoicePitch() const;
|
|
|
|
// Check for queued-up-sentences + speak them
|
|
void UpdateSentenceQueue();
|
|
|
|
// Returns the sentence index played, which can be used to determine
|
|
// the sentence length of time using engine->SentenceLength
|
|
int Speak( const char *pSentence, SentencePriority_t nSoundPriority = SENTENCE_PRIORITY_NORMAL, SentenceCriteria_t nCriteria = SENTENCE_CRITERIA_IN_SQUAD );
|
|
|
|
// Returns the sentence index played, which can be used to determine
|
|
// the sentence length of time using engine->SentenceLength. If the sentence
|
|
// was queued, then -1 is returned, which is the same result as if the sound wasn't played
|
|
int SpeakQueued( const char *pSentence, SentencePriority_t nSoundPriority = SENTENCE_PRIORITY_NORMAL, SentenceCriteria_t nCriteria = SENTENCE_CRITERIA_IN_SQUAD );
|
|
|
|
// Clears the sentence queue
|
|
void ClearQueue();
|
|
|
|
protected:
|
|
virtual float GetVolume() = 0;
|
|
virtual soundlevel_t GetSoundLevel() = 0;
|
|
|
|
private:
|
|
// Speech criteria
|
|
bool MatchesCriteria( SentenceCriteria_t nCriteria );
|
|
|
|
// Play the actual sentence
|
|
int PlaySentence( const char *pSentence );
|
|
|
|
// Debug output
|
|
void SentenceMsg( const char *pStatus, const char *pSentence );
|
|
|
|
int m_voicePitch;
|
|
int m_nQueuedSentenceIndex;
|
|
float m_flQueueTimeout;
|
|
int m_nQueueSoundPriority;
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// NOTE: This is a template class so each user has a different set of globals
|
|
//-----------------------------------------------------------------------------
|
|
template< class NPC_CLASS >
|
|
class CAI_Sentence : public CAI_SentenceBase
|
|
{
|
|
DECLARE_CLASS_NOFRIEND( CAI_Sentence, CAI_SentenceBase );
|
|
|
|
public:
|
|
void Init( NPC_CLASS *pOuter, const char *pGameSound );
|
|
|
|
protected:
|
|
virtual float GetVolume() { return m_sentenceVolume; }
|
|
virtual soundlevel_t GetSoundLevel() { return m_sentenceSoundlevel; }
|
|
|
|
private:
|
|
static float m_sentenceVolume;
|
|
static soundlevel_t m_sentenceSoundlevel;
|
|
static int m_voicePitchMin;
|
|
static int m_voicePitchMax;
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Voice pitch
|
|
//-----------------------------------------------------------------------------
|
|
inline void CAI_SentenceBase::SetVoicePitch( int voicePitch )
|
|
{
|
|
m_voicePitch = voicePitch;
|
|
}
|
|
|
|
inline int CAI_SentenceBase::GetVoicePitch() const
|
|
{
|
|
return m_voicePitch;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Set up the class's sentence information
|
|
//-----------------------------------------------------------------------------
|
|
template< class NPC_CLASS >
|
|
void CAI_Sentence< NPC_CLASS >::Init( NPC_CLASS *pOuter, const char *pGameSound )
|
|
{
|
|
SetOuter( pOuter );
|
|
|
|
if ( m_voicePitchMin <= 0 || m_voicePitchMax <= 0 )
|
|
{
|
|
// init the sentence parameters using a dummy gamesounds entry
|
|
CSoundParameters params;
|
|
if ( GetOuter()->GetParametersForSound( pGameSound, params, NULL ) )
|
|
{
|
|
m_sentenceVolume = params.volume;
|
|
m_sentenceSoundlevel = params.soundlevel;
|
|
m_voicePitchMin = params.pitchlow;
|
|
m_voicePitchMax = params.pitchhigh;
|
|
}
|
|
}
|
|
|
|
// Select a voice pitch
|
|
if ( random->RandomInt(0,1) )
|
|
{
|
|
SetVoicePitch( random->RandomInt( m_voicePitchMin, m_voicePitchMax ) );
|
|
}
|
|
else
|
|
{
|
|
SetVoicePitch( 100 );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Global instantiation
|
|
//-----------------------------------------------------------------------------
|
|
template< class NPC_CLASS > float CAI_Sentence< NPC_CLASS >::m_sentenceVolume = 1.0f;
|
|
template< class NPC_CLASS > soundlevel_t CAI_Sentence< NPC_CLASS >::m_sentenceSoundlevel = SNDLVL_NORM;
|
|
template< class NPC_CLASS > int CAI_Sentence< NPC_CLASS >::m_voicePitchMin = 0;
|
|
template< class NPC_CLASS > int CAI_Sentence< NPC_CLASS >::m_voicePitchMax = 0;
|
|
|
|
|
|
|
|
#endif // AI_SENTENCE_H
|