mirror of
https://github.com/mapbase-source/source-sdk-2013.git
synced 2025-01-15 00:07:58 +03:00
Made followup responses more reliable with generic NPCs and added "vscript_file" response type
This commit is contained in:
parent
d6b959899c
commit
d4a91fe027
@ -74,6 +74,10 @@ static void DispatchComeback( CAI_ExpresserWithFollowup *pExpress, CBaseEntity *
|
|||||||
// add in the FROM context so dispatchee knows was from me
|
// add in the FROM context so dispatchee knows was from me
|
||||||
const char * RESTRICT pszSpeakerName = GetResponseName( pSpeaker );
|
const char * RESTRICT pszSpeakerName = GetResponseName( pSpeaker );
|
||||||
criteria.AppendCriteria( "From", pszSpeakerName );
|
criteria.AppendCriteria( "From", pszSpeakerName );
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// See DispatchFollowupThroughQueue()
|
||||||
|
criteria.AppendCriteria( "From_idx", CNumStr( pSpeaker->entindex() ) );
|
||||||
|
#endif
|
||||||
// if a SUBJECT criteria is missing, put it back in.
|
// if a SUBJECT criteria is missing, put it back in.
|
||||||
if ( criteria.FindCriterionIndex( "Subject" ) == -1 )
|
if ( criteria.FindCriterionIndex( "Subject" ) == -1 )
|
||||||
{
|
{
|
||||||
@ -140,8 +144,17 @@ static CBaseEntity *AscertainSpeechSubjectFromContext( AI_Response *response, AI
|
|||||||
const char *subject = criteria.GetValue( criteria.FindCriterionIndex( pContextName ) );
|
const char *subject = criteria.GetValue( criteria.FindCriterionIndex( pContextName ) );
|
||||||
if (subject)
|
if (subject)
|
||||||
{
|
{
|
||||||
|
CBaseEntity *pEnt = gEntList.FindEntityByName( NULL, subject );
|
||||||
|
|
||||||
return gEntList.FindEntityByName( NULL, subject );
|
#ifdef MAPBASE
|
||||||
|
// Allow entity indices to be used (see DispatchFollowupThroughQueue() for one particular use case)
|
||||||
|
if (!pEnt && atoi(subject))
|
||||||
|
{
|
||||||
|
pEnt = CBaseEntity::Instance( atoi( subject ) );
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return pEnt;
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -166,7 +179,12 @@ static CResponseQueue::CFollowupTargetSpec_t ResolveFollowupTargetToEntity( AICo
|
|||||||
}
|
}
|
||||||
else if ( Q_stricmp(szTarget, "from") == 0 )
|
else if ( Q_stricmp(szTarget, "from") == 0 )
|
||||||
{
|
{
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// See DispatchFollowupThroughQueue()
|
||||||
|
return CResponseQueue::CFollowupTargetSpec_t( AscertainSpeechSubjectFromContext( response, criteria, "From_idx" ) );
|
||||||
|
#else
|
||||||
return CResponseQueue::CFollowupTargetSpec_t( AscertainSpeechSubjectFromContext( response, criteria, "From" ) );
|
return CResponseQueue::CFollowupTargetSpec_t( AscertainSpeechSubjectFromContext( response, criteria, "From" ) );
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
else if ( Q_stricmp(szTarget, "any") == 0 )
|
else if ( Q_stricmp(szTarget, "any") == 0 )
|
||||||
{
|
{
|
||||||
@ -400,6 +418,14 @@ void CAI_ExpresserWithFollowup::DispatchFollowupThroughQueue( const AIConcept_t
|
|||||||
// Don't add my own criteria! GatherCriteria( &criteria, followup.followup_concept, followup.followup_contexts );
|
// Don't add my own criteria! GatherCriteria( &criteria, followup.followup_concept, followup.followup_contexts );
|
||||||
|
|
||||||
criteria.AppendCriteria( "From", STRING( pOuter->GetEntityName() ) );
|
criteria.AppendCriteria( "From", STRING( pOuter->GetEntityName() ) );
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// The index of the "From" entity.
|
||||||
|
// In HL2 mods, many followup users would be generic NPCs (e.g. citizens) who might not have any particular significance.
|
||||||
|
// Those generic NPCs are quite likely to have no name or have a name in common with other entities. As a result, Mapbase
|
||||||
|
// changes internal operations of the "From" context to search for an entity index. This won't be 100% reliable if the source
|
||||||
|
// talker dies and another entity is created immediately afterwards, but it's a lot more reliable than a simple entity name search.
|
||||||
|
criteria.AppendCriteria( "From_idx", CNumStr( pOuter->entindex() ) );
|
||||||
|
#endif
|
||||||
|
|
||||||
criteria.Merge( criteriaStr );
|
criteria.Merge( criteriaStr );
|
||||||
g_ResponseQueueManager.GetQueue()->Add( concept, &criteria, gpGlobals->curtime + delay, target, pOuter );
|
g_ResponseQueueManager.GetQueue()->Add( concept, &criteria, gpGlobals->curtime + delay, target, pOuter );
|
||||||
@ -436,6 +462,12 @@ void CAI_ExpresserWithFollowup::OnSpeechFinished()
|
|||||||
{
|
{
|
||||||
if (m_pPostponedFollowup && m_pPostponedFollowup->IsValid())
|
if (m_pPostponedFollowup && m_pPostponedFollowup->IsValid())
|
||||||
{
|
{
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// HACKHACK: Non-scene speech (e.g. noscene speak/sentence) fire OnSpeechFinished() immediately,
|
||||||
|
// so add the actual speech time to the followup delay
|
||||||
|
if (GetTimeSpeechCompleteWithoutDelay() > gpGlobals->curtime)
|
||||||
|
m_pPostponedFollowup->followup_delay += GetTimeSpeechCompleteWithoutDelay() - gpGlobals->curtime;
|
||||||
|
#endif
|
||||||
return SpeakDispatchFollowup(*m_pPostponedFollowup);
|
return SpeakDispatchFollowup(*m_pPostponedFollowup);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -750,6 +750,10 @@ bool CAI_Expresser::SpeakDispatchResponse( AIConcept_t &concept, AI_Response *re
|
|||||||
DevMsg( 2, "SpeakDispatchResponse: Entity ( %i/%s ) playing sound '%s'\n", GetOuter()->entindex(), STRING( GetOuter()->GetEntityName() ), response );
|
DevMsg( 2, "SpeakDispatchResponse: Entity ( %i/%s ) playing sound '%s'\n", GetOuter()->entindex(), STRING( GetOuter()->GetEntityName() ), response );
|
||||||
NoteSpeaking( speakTime, delay );
|
NoteSpeaking( speakTime, delay );
|
||||||
spoke = true;
|
spoke = true;
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// Not really any other way of doing this
|
||||||
|
OnSpeechFinished();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -757,6 +761,10 @@ bool CAI_Expresser::SpeakDispatchResponse( AIConcept_t &concept, AI_Response *re
|
|||||||
case ResponseRules::RESPONSE_SENTENCE:
|
case ResponseRules::RESPONSE_SENTENCE:
|
||||||
{
|
{
|
||||||
spoke = ( -1 != SpeakRawSentence( response, delay, VOL_NORM, soundlevel ) ) ? true : false;
|
spoke = ( -1 != SpeakRawSentence( response, delay, VOL_NORM, soundlevel ) ) ? true : false;
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// Not really any other way of doing this
|
||||||
|
OnSpeechFinished();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -781,17 +789,30 @@ bool CAI_Expresser::SpeakDispatchResponse( AIConcept_t &concept, AI_Response *re
|
|||||||
NDebugOverlay::Text( vPrintPos, response, true, 1.5 );
|
NDebugOverlay::Text( vPrintPos, response, true, 1.5 );
|
||||||
}
|
}
|
||||||
spoke = true;
|
spoke = true;
|
||||||
|
#ifdef MAPBASE
|
||||||
|
OnSpeechFinished();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ResponseRules::RESPONSE_ENTITYIO:
|
case ResponseRules::RESPONSE_ENTITYIO:
|
||||||
{
|
{
|
||||||
return FireEntIOFromResponse( response, GetOuter() );
|
spoke = FireEntIOFromResponse( response, GetOuter() );
|
||||||
|
#ifdef MAPBASE
|
||||||
|
OnSpeechFinished();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
#ifdef MAPBASE
|
#ifdef MAPBASE_VSCRIPT
|
||||||
case ResponseRules::RESPONSE_VSCRIPT:
|
case ResponseRules::RESPONSE_VSCRIPT:
|
||||||
{
|
{
|
||||||
return GetOuter()->RunScript( response, "ResponseScript" );
|
spoke = RunScriptResponse( GetOuter(), response, criteria, false );
|
||||||
|
OnSpeechFinished();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ResponseRules::RESPONSE_VSCRIPT_FILE:
|
||||||
|
{
|
||||||
|
spoke = RunScriptResponse( GetOuter(), response, criteria, true );
|
||||||
|
OnSpeechFinished();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
@ -918,6 +939,47 @@ bool CAI_Expresser::FireEntIOFromResponse( char *response, CBaseEntity *pInitiat
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef MAPBASE_VSCRIPT
|
||||||
|
bool CAI_Expresser::RunScriptResponse( CBaseEntity *pTarget, const char *response, AI_CriteriaSet *criteria, bool file )
|
||||||
|
{
|
||||||
|
if (!pTarget->ValidateScriptScope())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
ScriptVariant_t varCriteriaTable;
|
||||||
|
g_pScriptVM->CreateTable( varCriteriaTable );
|
||||||
|
|
||||||
|
if (criteria)
|
||||||
|
{
|
||||||
|
// Sort all of the criteria into a table.
|
||||||
|
// Letting VScript have access to this is important because not all criteria is appended in ModifyOrAppendCriteria() and
|
||||||
|
// not all contexts are actually appended as contexts. This is specifically important for followup responses.
|
||||||
|
int count = criteria->GetCount();
|
||||||
|
for ( int i = 0 ; i < count ; ++i )
|
||||||
|
{
|
||||||
|
// TODO: Weight?
|
||||||
|
g_pScriptVM->SetValue( varCriteriaTable, criteria->GetName(i), criteria->GetValue(i) );
|
||||||
|
}
|
||||||
|
|
||||||
|
g_pScriptVM->SetValue( "criteria", varCriteriaTable );
|
||||||
|
}
|
||||||
|
|
||||||
|
bool bSuccess = false;
|
||||||
|
if (file)
|
||||||
|
{
|
||||||
|
bSuccess = pTarget->RunScriptFile( response );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
bSuccess = pTarget->RunScript( response, "ResponseScript" );
|
||||||
|
}
|
||||||
|
|
||||||
|
g_pScriptVM->ClearValue( "criteria" );
|
||||||
|
g_pScriptVM->ReleaseScript( varCriteriaTable );
|
||||||
|
|
||||||
|
return bSuccess;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// Purpose:
|
// Purpose:
|
||||||
// Input : *response -
|
// Input : *response -
|
||||||
@ -967,6 +1029,37 @@ float CAI_Expresser::GetResponseDuration( AI_Response *result )
|
|||||||
return 0.0f;
|
return 0.0f;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
//-----------------------------------------------------------------------------
|
||||||
|
// Purpose:
|
||||||
|
//-----------------------------------------------------------------------------
|
||||||
|
void CAI_Expresser::SetUsingProspectiveResponses( bool bToggle )
|
||||||
|
{
|
||||||
|
VPROF("CAI_Expresser::SetUsingProspectiveResponses");
|
||||||
|
IResponseSystem *rs = GetOuter()->GetResponseSystem();
|
||||||
|
if ( !rs )
|
||||||
|
{
|
||||||
|
Assert( !"No response system installed for CAI_Expresser::GetOuter()!!!" );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
rs->SetProspective( bToggle );
|
||||||
|
}
|
||||||
|
|
||||||
|
void CAI_Expresser::MarkResponseAsUsed( AI_Response *response )
|
||||||
|
{
|
||||||
|
VPROF("CAI_Expresser::MarkResponseAsUsed");
|
||||||
|
IResponseSystem *rs = GetOuter()->GetResponseSystem();
|
||||||
|
if ( !rs )
|
||||||
|
{
|
||||||
|
Assert( !"No response system installed for CAI_Expresser::GetOuter()!!!" );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
rs->MarkResponseAsUsed( response->GetInternalIndices()[0], response->GetInternalIndices()[1] );
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
// Purpose: Placeholder for rules based response system
|
// Purpose: Placeholder for rules based response system
|
||||||
// Input : concept -
|
// Input : concept -
|
||||||
|
@ -178,6 +178,11 @@ public:
|
|||||||
virtual bool SpeakDispatchResponse( AIConcept_t &concept, AI_Response *response, AI_CriteriaSet *criteria, IRecipientFilter *filter = NULL );
|
virtual bool SpeakDispatchResponse( AIConcept_t &concept, AI_Response *response, AI_CriteriaSet *criteria, IRecipientFilter *filter = NULL );
|
||||||
float GetResponseDuration( AI_Response *response );
|
float GetResponseDuration( AI_Response *response );
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
void SetUsingProspectiveResponses( bool bToggle );
|
||||||
|
void MarkResponseAsUsed( AI_Response *response );
|
||||||
|
#endif
|
||||||
|
|
||||||
virtual int SpeakRawSentence( const char *pszSentence, float delay, float volume = VOL_NORM, soundlevel_t soundlevel = SNDLVL_TALKING, CBaseEntity *pListener = NULL );
|
virtual int SpeakRawSentence( const char *pszSentence, float delay, float volume = VOL_NORM, soundlevel_t soundlevel = SNDLVL_TALKING, CBaseEntity *pListener = NULL );
|
||||||
|
|
||||||
bool SemaphoreIsAvailable( CBaseEntity *pTalker );
|
bool SemaphoreIsAvailable( CBaseEntity *pTalker );
|
||||||
@ -194,6 +199,9 @@ public:
|
|||||||
bool CanSpeak();
|
bool CanSpeak();
|
||||||
bool CanSpeakAfterMyself();
|
bool CanSpeakAfterMyself();
|
||||||
float GetTimeSpeechComplete() const { return m_flStopTalkTime; }
|
float GetTimeSpeechComplete() const { return m_flStopTalkTime; }
|
||||||
|
#ifdef MAPBASE
|
||||||
|
float GetTimeSpeechCompleteWithoutDelay() const { return m_flStopTalkTimeWithoutDelay; }
|
||||||
|
#endif
|
||||||
void BlockSpeechUntil( float time );
|
void BlockSpeechUntil( float time );
|
||||||
|
|
||||||
// --------------------------------
|
// --------------------------------
|
||||||
@ -227,6 +235,11 @@ public:
|
|||||||
// returns false on failure (eg, couldn't match parse contents)
|
// returns false on failure (eg, couldn't match parse contents)
|
||||||
static bool FireEntIOFromResponse( char *response, CBaseEntity *pInitiator );
|
static bool FireEntIOFromResponse( char *response, CBaseEntity *pInitiator );
|
||||||
|
|
||||||
|
#ifdef MAPBASE_VSCRIPT
|
||||||
|
// Used for RESPONSE_VSCRIPT(_FILE)
|
||||||
|
static bool RunScriptResponse( CBaseEntity *pTarget, const char *response, AI_CriteriaSet *criteria, bool file );
|
||||||
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
CAI_TimedSemaphore *GetMySpeechSemaphore( CBaseEntity *pNpc );
|
CAI_TimedSemaphore *GetMySpeechSemaphore( CBaseEntity *pNpc );
|
||||||
|
|
||||||
|
@ -374,6 +374,10 @@ bool CResponseQueue::DispatchOneResponse_ThenANY( CDeferredResponse &response, A
|
|||||||
}
|
}
|
||||||
AI_Response prospectiveResponse;
|
AI_Response prospectiveResponse;
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
pEx->SetUsingProspectiveResponses( true );
|
||||||
|
#endif
|
||||||
|
|
||||||
if ( pEx->FindResponse( prospectiveResponse, response.m_concept, &characterCriteria ) )
|
if ( pEx->FindResponse( prospectiveResponse, response.m_concept, &characterCriteria ) )
|
||||||
{
|
{
|
||||||
float score = prospectiveResponse.GetMatchScore();
|
float score = prospectiveResponse.GetMatchScore();
|
||||||
@ -391,7 +395,12 @@ bool CResponseQueue::DispatchOneResponse_ThenANY( CDeferredResponse &response, A
|
|||||||
else if ( score >= bestScore - slop ) // if this score is at least as good as the best we've seen, but not better than all
|
else if ( score >= bestScore - slop ) // if this score is at least as good as the best we've seen, but not better than all
|
||||||
{
|
{
|
||||||
if ( numExFound >= EXARRAYMAX )
|
if ( numExFound >= EXARRAYMAX )
|
||||||
|
{
|
||||||
|
#ifdef MAPBASE
|
||||||
|
pEx->SetUsingProspectiveResponses( false );
|
||||||
|
#endif
|
||||||
continue; // SAFETY: don't overflow the array
|
continue; // SAFETY: don't overflow the array
|
||||||
|
}
|
||||||
|
|
||||||
responseToSay[numExFound] = prospectiveResponse;
|
responseToSay[numExFound] = prospectiveResponse;
|
||||||
pBestEx[numExFound] = pEx;
|
pBestEx[numExFound] = pEx;
|
||||||
@ -400,6 +409,10 @@ bool CResponseQueue::DispatchOneResponse_ThenANY( CDeferredResponse &response, A
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
pEx->SetUsingProspectiveResponses( false );
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
// if I have a response, dispatch it.
|
// if I have a response, dispatch it.
|
||||||
@ -410,6 +423,9 @@ bool CResponseQueue::DispatchOneResponse_ThenANY( CDeferredResponse &response, A
|
|||||||
|
|
||||||
if ( pBestEx[iSelect] != NULL )
|
if ( pBestEx[iSelect] != NULL )
|
||||||
{
|
{
|
||||||
|
#ifdef MAPBASE
|
||||||
|
pBestEx[iSelect]->MarkResponseAsUsed( responseToSay + iSelect );
|
||||||
|
#endif
|
||||||
return pBestEx[iSelect]->SpeakDispatchResponse( response.m_concept, responseToSay + iSelect, pDeferredCriteria );
|
return pBestEx[iSelect]->SpeakDispatchResponse( response.m_concept, responseToSay + iSelect, pDeferredCriteria );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -98,6 +98,7 @@ namespace ResponseRules
|
|||||||
RESPONSE_ENTITYIO, // poke an input on an entity
|
RESPONSE_ENTITYIO, // poke an input on an entity
|
||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
RESPONSE_VSCRIPT, // Run VScript code
|
RESPONSE_VSCRIPT, // Run VScript code
|
||||||
|
RESPONSE_VSCRIPT_FILE, // Run a VScript file (bypasses ugliness and character limits when just using IncludeScript() with RESPONSE_VSCRIPT)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
NUM_RESPONSES,
|
NUM_RESPONSES,
|
||||||
@ -351,6 +352,9 @@ namespace ResponseRules
|
|||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
int GetContextFlags() { return m_iContextFlags; }
|
int GetContextFlags() { return m_iContextFlags; }
|
||||||
bool IsApplyContextToWorld( void ) { return (m_iContextFlags & APPLYCONTEXT_WORLD) != 0; }
|
bool IsApplyContextToWorld( void ) { return (m_iContextFlags & APPLYCONTEXT_WORLD) != 0; }
|
||||||
|
|
||||||
|
inline short *GetInternalIndices() { return m_InternalIndices; }
|
||||||
|
inline void SetInternalIndices( short iGroup, short iWithinGroup ) { m_InternalIndices[0] = iGroup; m_InternalIndices[1] = iWithinGroup; }
|
||||||
#else
|
#else
|
||||||
bool IsApplyContextToWorld( void ) { return m_bApplyContextToWorld; }
|
bool IsApplyContextToWorld( void ) { return m_bApplyContextToWorld; }
|
||||||
#endif
|
#endif
|
||||||
@ -393,6 +397,10 @@ namespace ResponseRules
|
|||||||
char * m_szContext; // context data we apply to character after running
|
char * m_szContext; // context data we apply to character after running
|
||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
int m_iContextFlags;
|
int m_iContextFlags;
|
||||||
|
|
||||||
|
// The response's original indices in the system. [0] is the group's index, [1] is the index within the group.
|
||||||
|
// For now, this is only set in prospecctive mode. It's used to call back to the ParserResponse and mark a prospectively chosen response as used.
|
||||||
|
short m_InternalIndices[2];
|
||||||
#else
|
#else
|
||||||
bool m_bApplyContextToWorld;
|
bool m_bApplyContextToWorld;
|
||||||
#endif
|
#endif
|
||||||
@ -418,6 +426,15 @@ namespace ResponseRules
|
|||||||
virtual bool FindBestResponse( const CriteriaSet& set, CRR_Response& response, IResponseFilter *pFilter = NULL ) = 0;
|
virtual bool FindBestResponse( const CriteriaSet& set, CRR_Response& response, IResponseFilter *pFilter = NULL ) = 0;
|
||||||
virtual void GetAllResponses( CUtlVector<CRR_Response> *pResponses ) = 0;
|
virtual void GetAllResponses( CUtlVector<CRR_Response> *pResponses ) = 0;
|
||||||
virtual void PrecacheResponses( bool bEnable ) = 0;
|
virtual void PrecacheResponses( bool bEnable ) = 0;
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// (Optional) Call this before and after using FindBestResponse() for a prospective lookup, e.g. a response that might not actually be used
|
||||||
|
// and should not trigger displayfirst, etc.
|
||||||
|
virtual void SetProspective( bool bToggle ) {};
|
||||||
|
|
||||||
|
// (Optional) Marks a prospective response as used
|
||||||
|
virtual void MarkResponseAsUsed( short iGroup, short iWithinGroup ) {};
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -942,6 +942,10 @@ int CResponseSystem::SelectWeightedResponseFromResponseGroup( ResponseGroup *g,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ( slot != -1 )
|
if ( slot != -1 )
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// Don't mark responses as used in prospective mode
|
||||||
|
if (m_bInProspective == false)
|
||||||
|
#endif
|
||||||
g->MarkResponseUsed( slot );
|
g->MarkResponseUsed( slot );
|
||||||
|
|
||||||
// Revert fake depletion of unavailable choices
|
// Revert fake depletion of unavailable choices
|
||||||
@ -1284,6 +1288,26 @@ bool CResponseSystem::FindBestResponse( const CriteriaSet& set, CRR_Response& re
|
|||||||
context = r->GetContext();
|
context = r->GetContext();
|
||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
contextflags = r->GetContextFlags();
|
contextflags = r->GetContextFlags();
|
||||||
|
|
||||||
|
// Sets the internal indices for the response to call back to later for prospective responses
|
||||||
|
// (NOTE: Performance not tested; Be wary of turning off the m_bInProspective check!)
|
||||||
|
if (m_bInProspective)
|
||||||
|
{
|
||||||
|
for ( int i = 0; i < (int)m_Responses.Count(); i++ )
|
||||||
|
{
|
||||||
|
if (&m_Responses[i] == result.group)
|
||||||
|
{
|
||||||
|
ResponseGroup &group = m_Responses[i];
|
||||||
|
for ( int j = 0; j < group.group.Count(); j++)
|
||||||
|
{
|
||||||
|
if (&group.group[j] == result.action)
|
||||||
|
{
|
||||||
|
response.SetInternalIndices( i, j );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
bcontexttoworld = r->IsApplyContextToWorld();
|
bcontexttoworld = r->IsApplyContextToWorld();
|
||||||
#endif
|
#endif
|
||||||
@ -1369,6 +1393,22 @@ void CResponseSystem::GetAllResponses( CUtlVector<CRR_Response> *pResponses )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
void CResponseSystem::MarkResponseAsUsed( short iGroup, short iWithinGroup )
|
||||||
|
{
|
||||||
|
if (m_Responses.Count() > (unsigned int)iGroup)
|
||||||
|
{
|
||||||
|
ResponseGroup &group = m_Responses[iGroup];
|
||||||
|
if (group.group.Count() > (int)iWithinGroup)
|
||||||
|
{
|
||||||
|
group.MarkResponseUsed( iWithinGroup );
|
||||||
|
|
||||||
|
Msg("Marked response %s (%i) used\n", group.group[iWithinGroup].value, iWithinGroup);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
void CResponseSystem::ParseInclude()
|
void CResponseSystem::ParseInclude()
|
||||||
{
|
{
|
||||||
char includefile[ 256 ];
|
char includefile[ 256 ];
|
||||||
@ -1521,6 +1561,9 @@ inline ResponseType_t ComputeResponseType( const char *s )
|
|||||||
return RESPONSE_ENTITYIO;
|
return RESPONSE_ENTITYIO;
|
||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
case 'v':
|
case 'v':
|
||||||
|
if (*(s + 7) == '_')
|
||||||
|
return RESPONSE_VSCRIPT_FILE;
|
||||||
|
else
|
||||||
return RESPONSE_VSCRIPT;
|
return RESPONSE_VSCRIPT;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -44,6 +44,12 @@ namespace ResponseRules
|
|||||||
// IResponseSystem
|
// IResponseSystem
|
||||||
virtual bool FindBestResponse( const CriteriaSet& set, CRR_Response& response, IResponseFilter *pFilter = NULL );
|
virtual bool FindBestResponse( const CriteriaSet& set, CRR_Response& response, IResponseFilter *pFilter = NULL );
|
||||||
virtual void GetAllResponses( CUtlVector<CRR_Response> *pResponses );
|
virtual void GetAllResponses( CUtlVector<CRR_Response> *pResponses );
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
virtual void SetProspective( bool bToggle ) { m_bInProspective = bToggle; }
|
||||||
|
|
||||||
|
virtual void MarkResponseAsUsed( short iGroup, short iWithinGroup );
|
||||||
|
#endif
|
||||||
#pragma endregion Implement interface from IResponseSystem
|
#pragma endregion Implement interface from IResponseSystem
|
||||||
|
|
||||||
virtual void Release() = 0;
|
virtual void Release() = 0;
|
||||||
@ -283,6 +289,13 @@ public:
|
|||||||
|
|
||||||
bool m_bCustomManagable;
|
bool m_bCustomManagable;
|
||||||
|
|
||||||
|
#ifdef MAPBASE
|
||||||
|
// This is a hack specifically designed to fix displayfirst, speakonce, etc. in "prospective" response searches,
|
||||||
|
// especially the prospective lookups in followup responses.
|
||||||
|
// It works by preventing responses from being marked as "used".
|
||||||
|
bool m_bInProspective;
|
||||||
|
#endif
|
||||||
|
|
||||||
struct ScriptEntry
|
struct ScriptEntry
|
||||||
{
|
{
|
||||||
unsigned char *buffer;
|
unsigned char *buffer;
|
||||||
|
@ -244,6 +244,8 @@ const char *CRR_Response::DescribeResponse( ResponseType_t type )
|
|||||||
#ifdef MAPBASE
|
#ifdef MAPBASE
|
||||||
case ResponseRules::RESPONSE_VSCRIPT:
|
case ResponseRules::RESPONSE_VSCRIPT:
|
||||||
return "RESPONSE_VSCRIPT";
|
return "RESPONSE_VSCRIPT";
|
||||||
|
case ResponseRules::RESPONSE_VSCRIPT_FILE:
|
||||||
|
return "RESPONSE_VSCRIPT_FILE";
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user