⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 enhancedgideon.cpp

📁 Blood 2全套源码
💻 CPP
📖 第 1 页 / 共 2 页
字号:
// ----------------------------------------------------------------------- //
//
// MODULE  : EnhancedGideon.cpp
//
// PURPOSE : EnhancedGideon AI - Implementation
//
// CREATED : 10/1/98 (version 3!)
//
// ----------------------------------------------------------------------- //

#include <stdio.h>
#include "EnhancedGideon.h"
#include "cpp_server_de.h"
#include "SFXMsgIds.h"

BEGIN_CLASS(EnhancedGideon)
    ADD_REALPROP(RandomHitPoints, 0.0f) \
	ADD_STRINGPROP(AIState, "IDLE")     \
	ADD_BOOLPROP(Enhanced,DFALSE)	\
END_CLASS_DEFAULT(EnhancedGideon, AI_Mgr, NULL, NULL)

//static data member initialization
DBOOL		EnhancedGideon::m_bLoadAnims = DTRUE;
CAnim_Sound	EnhancedGideon::m_Anim_Sound;

// ----------------------------------------------------------------------- //
//
//	ROUTINE:	
//
//	PURPOSE:	Constructor
//
// ----------------------------------------------------------------------- //

EnhancedGideon::EnhancedGideon() : AI_Mgr()
{
	m_fHearingDist	= 0.0f;
	m_fSensingDist	= 10000.0f;
	m_fSmellingDist	= 0.0f;
	m_fSeeingDist	= 10000.0f;

	m_fWalkSpeed	= 175.0f;
	m_fRunSpeed		= 380.0f;
	m_fRollSpeed	= 200.0f;

	m_fAIMass		= AI_DEFAULT_MASS;

	m_nAIStrength	= 10;

	strcpy(m_szAIWeapon[0], "GIDEON_SHIELD" );
	strcpy(m_szAIWeapon[1], "ENERGY_BLAST" );
	strcpy(m_szAIWeapon[2], "NAGA_BLAST");
	strcpy(m_szAIWeapon[3], "GIDEON_WIND");

	m_nState = STATE_Idle; 
	m_nLastState = STATE_Idle; 
	m_dwFlags = FLAG_ALWAYSRECOIL;

	m_bCabal = DFALSE; // Should this be true?
	m_bMoveToGround = DFALSE; // We can fly, don't move to floor on creation.

	sprintf(m_szFireNode,"rr_gun");

	// [blg]
	m_fAIHitPoints   = 9000;
	m_fAIRandomHP    = 0;
	m_fAIArmorPoints = 0;

	m_fFullHealth = 0.0f;

	m_bCreateHealth = DTRUE;

	m_fShieldDuration = 60.0f; // We can only shield ourselves once every 60 seonds
	m_fLastShield = 0.0f;
	m_fLastPanic = 0.0f;

	m_damage.SetApplyDamagePhysics(DFALSE);

	m_bNoFire = DFALSE;

	strcpy(m_szAIState, "IDLE");
}


// ----------------------------------------------------------------------- //
//
//	ROUTINE:	EngineMessageFn
//
//	PURPOSE:	Handle engine messages
//
// ----------------------------------------------------------------------- //

DDWORD EnhancedGideon::EngineMessageFn(DDWORD messageID, void *pData, DFLOAT fData)
{
	switch(messageID)
	{
		case MID_PRECREATE:
		{
			// Need to call base class to have the object name read in before
			// we call PostPropRead()

			DDWORD dwRet = AI_Mgr::EngineMessageFn(messageID, pData, fData);

			if(fData != PRECREATE_SAVEGAME)
				PostPropRead((ObjectCreateStruct*)pData);

			return dwRet;
		}
		break;
        
		case MID_INITIALUPDATE:
		{
			InitialUpdate((DVector *)pData);
			CacheFiles();
			break;
		}

		case MID_UPDATE:
		{
			if (m_bCreateHealth)
			{
				if (!m_fFullHealth)
				{
					HMESSAGEWRITE hWrite = m_pServerDE->StartMessage(NULL, SMSG_BOSSHEALTH);
					m_pServerDE->WriteToMessageFloat(hWrite,1.0f);
					m_pServerDE->EndMessage(hWrite);
					m_bCreateHealth = DFALSE;
					m_fFullHealth = m_damage.GetHitPoints();
				}
				else
				{
					DFLOAT fTemp = m_damage.GetHitPoints() / m_fFullHealth;	
					HMESSAGEWRITE hWrite = m_pServerDE->StartMessage(NULL, SMSG_BOSSHEALTH);
					m_pServerDE->WriteToMessageFloat(hWrite,fTemp);
					m_pServerDE->EndMessage(hWrite);
				}
			}
		}
		break;

		case MID_SAVEOBJECT:
			Save((HMESSAGEWRITE)pData, (DDWORD)fData);
			break;

		case MID_LOADOBJECT:
			Load((HMESSAGEREAD)pData, (DDWORD)fData);
			break;

		default : break;
	}

	// Store result of parent class function
	DDWORD dwResult = AI_Mgr::EngineMessageFn(messageID, pData, fData);

	// If this is initial update, we need to turn off gravity AFTER the
	// MID_INITIALUPDATE call in AI_Mgr...
	if (messageID == MID_INITIALUPDATE) 
	{
		DDWORD dwFlags = m_pServerDE->GetObjectFlags(m_hObject);
		dwFlags &= ~FLAG_GRAVITY;
		m_pServerDE->SetObjectFlags(m_hObject, dwFlags);	
	}

	// ...and return the stored result.
	return dwResult;
}

DDWORD EnhancedGideon::ObjectMessageFn(HOBJECT hSender, DDWORD messageID, HMESSAGEREAD hRead)
{
	switch (messageID)
	{
		case MID_DAMAGE:
		{
			if (!m_pServerDE) break;
			
			CBaseCharacter::ObjectMessageFn(hSender, messageID, hRead);
			
			DFLOAT fTemp = m_damage.GetHitPoints() / m_fFullHealth;	
			HMESSAGEWRITE hWrite = m_pServerDE->StartMessage(NULL, SMSG_BOSSHEALTH);
			m_pServerDE->WriteToMessageFloat(hWrite,fTemp);
			m_pServerDE->EndMessage(hWrite);
		}

		default: break;
	}

	return AI_Mgr::ObjectMessageFn(hSender, messageID, hRead);
}


// ----------------------------------------------------------------------- //
//
//	ROUTINE:	PostPropRead()
//
//	PURPOSE:	Update properties
//
// ----------------------------------------------------------------------- //

void EnhancedGideon::PostPropRead(ObjectCreateStruct *pStruct)
{
	if (!pStruct) return;

	char* pFilename = "Models\\Enemies\\Gideon.abc";
	char *pSkin = "Skins\\Enemies\\GideonE.dtx";

	strcpy(pStruct->m_Filename, pFilename);
	strcpy(pStruct->m_SkinName, pSkin);
}


// ----------------------------------------------------------------------- //
//
//	ROUTINE:	InitialUpdate()
//
//	PURPOSE:	Handle initial update
//
// ----------------------------------------------------------------------- //

DBOOL EnhancedGideon::InitialUpdate(DVector *pMovement)
{
	m_pServerDE = BaseClass::GetServerDE();
	if (!m_pServerDE) return DFALSE;

    if(m_bLoadAnims)
	{
	    m_Anim_Sound.SetAnimationIndexes(m_hObject);
		m_Anim_Sound.GenerateHitSpheres(m_hObject);
	    m_Anim_Sound.SetSoundRoot("sounds\\enemies\\EnhancedGideon");
		m_bLoadAnims = DFALSE;
	}

	AI_Mgr::InitStatics(&m_Anim_Sound);

	m_bCreateHealth = DTRUE;

	return DTRUE;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::MC_Dodge_Left
// DESCRIPTION	: run the dodge left animation
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::MC_Dodge_Left()
{
    if (m_bAnimating == DFALSE || m_nCurMetacmd != MC_DODGE_LEFT)
    {
		m_fTimeStart = m_pServerDE->GetTime();

        SetAnimation( m_pAnim_Sound->m_nAnim_DODGE_LEFT);
        m_pServerDE->SetModelLooping(m_hObject, DFALSE);

		DVector vLeft;
		VEC_MULSCALAR(vLeft, m_MoveObj.GetRightVector(), -1.0f);
		Move(vLeft, m_fRollSpeed);

		m_bAnimating = DTRUE;
		m_nCurMetacmd = MC_DODGE_LEFT;
	}
	else
	{
		DVector vLeft;
		VEC_MULSCALAR(vLeft, m_MoveObj.GetRightVector(), -1.0f);
		Move(vLeft, m_fRollSpeed);

		//Are we done?
		if(m_pServerDE->GetTime() - m_fTimeStart >= 1.0f)
        {
            m_bAnimating = DFALSE;
            Metacmd++;
        }
	}

	return;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::MC_Dodge_Right
// DESCRIPTION	: run the dodge right animation
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::MC_Dodge_Right()
{
    if (m_bAnimating == DFALSE || m_nCurMetacmd != MC_DODGE_RIGHT)
    {
		m_fTimeStart = m_pServerDE->GetTime();

        SetAnimation( m_pAnim_Sound->m_nAnim_DODGE_RIGHT);
        m_pServerDE->SetModelLooping(m_hObject, DFALSE);

		Move(m_MoveObj.GetRightVector(),m_fRollSpeed);

		m_bAnimating = DTRUE;
		m_nCurMetacmd = MC_DODGE_RIGHT;
	}
	else
	{
		Move(m_MoveObj.GetRightVector(),m_fRollSpeed);

		//Are we done?
		if(m_pServerDE->GetTime() - m_fTimeStart >= 1.0f)
        {
            m_bAnimating = DFALSE;
            Metacmd++;
        }
	}

	return;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::ComputeState
// DESCRIPTION	: Compute actual substate
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::ComputeState(int nStimType)
{
	int nStim = nStimType;

	if(nStimType == -1)
		nStim = ComputeStimuli();

	if(!nStim)
	{
		switch(m_nState)
		{
			case STATE_Idle:				SetNewState(STATE_Special1);			break;
			case STATE_Teleport:			SetNewState(STATE_SearchVisualTarget);	break;
			case STATE_SearchVisualTarget:	SetNewState(STATE_Idle);				break;
			default:						SetNewState(STATE_Teleport);			break;
		}
	}
	else
	{
		//if health is low or threat is high...
		if(m_fStimuli[HEALTH] < 0.25f || m_fStimuli[THREAT] > 0.75f)
		{
			// ...and we can shield...
			if (m_pServerDE->GetTime() - m_fLastShield > m_fShieldDuration)
			{
				// ...do so.
				SetNewState(STATE_AttackClose);
			} 
			else
			{
				// ...or run like hell.
				if (m_fStimuli[HEALTH] < 0.05f)
				{
					if ((m_fLastPanic + 15.0f) >= m_pServerDE->GetTime())
					{
						m_fLastPanic = m_pServerDE->GetTime();
						SetNewState(STATE_Teleport);
					}
					else
					{
						SetNewState(STATE_AttackClose);
					}
				}
				else
				{
					// ...or use the wind attack.
					SetNewState(STATE_AttackClose);
				}
			}
			return;
		}

		switch(m_nState)
		{
			case STATE_SearchVisualTarget:
			{
				SetNewState(STATE_Special1);
				break;
			}

			default:
			{
				if(m_fStimuli[SIGHT] > 0.94f)
				{
					SetNewState(STATE_AttackClose);
				}
				else
				{
					if (m_fStimuli[SIGHT] > 0.75f)
					{
						SetNewState(STATE_AttackFar);
					}
					else
					{
						SetNewState(STATE_Special1);
					}
				}
				
				break;
			}
		}
	}

	return;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::MC_Fire_Stand
// DESCRIPTION	: Run the fire_stand animation
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::MC_Fire_Stand()
{
	// Hack.
	if (m_bNoFire)
	{
		SetNewState(STATE_Passive);
		return;
	}

    if (m_bAnimating == DFALSE || m_nCurMetacmd != MC_FIRE_STAND)
    {
		DBOOL bRet = DFALSE;

		DFLOAT fArmor = m_damage.GetArmorPoints();

		if ((fArmor <= 25) && ((m_pServerDE->GetTime() - m_fLastShield) > m_fShieldDuration))
		{
			m_InventoryMgr.ChangeWeapon(WEAP_GIDEON_SHIELD);
			m_damage.SetArmorPoints(fArmor + 100);
			m_fLastShield = m_pServerDE->GetTime();
			bRet = SetAnimation(m_pAnim_Sound->m_nAnim_FIRE_STAND[TYPE_RIFLE]);
		} 
		else if(m_nState == STATE_AttackFar)
		{
			if (IsRandomChance(70))
			{
				m_InventoryMgr.ChangeWeapon(WEAP_ZEALOT_ENERGYBLAST);
				bRet = SetAnimation(m_pAnim_Sound->m_nAnim_FIRE_STAND[TYPE_MAGIC]);
			} else
			{
				m_InventoryMgr.ChangeWeapon(WEAP_NAGA_EYEBEAM);
				bRet = SetAnimation(m_pAnim_Sound->m_nAnim_FIRE_STAND[TYPE_RIFLE]);
			}
		}
		else
		{
			if (IsRandomChance(50))
			{
				m_InventoryMgr.ChangeWeapon(WEAP_GIDEON_WIND);
				bRet = SetAnimation(m_pAnim_Sound->m_nAnim_FIRE_STAND[TYPE_RIFLE]);
			}
			else
			{
				m_InventoryMgr.ChangeWeapon(WEAP_NAGA_EYEBEAM);
				bRet = SetAnimation(m_pAnim_Sound->m_nAnim_FIRE_STAND[TYPE_RIFLE]);
			}
		}

        m_pServerDE->SetModelLooping(m_hObject, DFALSE);
        
		m_bAnimating = DTRUE; 
		m_nCurMetacmd = MC_FIRE_STAND;
    }
    else
	{
		if(m_pServerDE->GetModelPlaybackState(m_hObject) & MS_PLAYDONE)		
		{
			m_InventoryMgr.ChangeWeapon(WEAP_SOUL_HOOK);
			m_bAnimating = DFALSE;
			Metacmd++;
			return;
		}
    }      
	
    return;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::AI_STATE_AttackClose
// DESCRIPTION	: 
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::AI_STATE_AttackClose()
{
	//SCHLEGZ 4/22/98 4:51:21 PM: sanity check
	if(m_hTarget == DNULL)
	{
		SetNewState(STATE_SearchVisualTarget);
		return;
	}

	if (m_bNoFire)
	{
		SetNewState(STATE_Passive);
		return;
	}

	MC_FaceTarget();
	Metacmd--;

	switch(Metacmd)
	{
		case 1:		MC_FaceTarget();	break;
		case 2:		m_fStimuli[SIGHT] = VEC_DIST(m_MoveObj.GetPos(),m_vTargetPos);

					if((m_fStimuli[SIGHT] <= (m_fSeeingDist * 0.90f)) || m_nCurMetacmd == MC_FIRE_STAND)
						MC_Fire_Stand();
					else
					{
						if(m_fStimuli[SIGHT] > (m_fSeeingDist * 0.40))
						{
							SetNewState(STATE_Special1);		
						}
						else
						{
							MC_Walk();
						}
					}
					
					break;
		case 3:		if(m_nCurMetacmd == MC_DODGE_RIGHT)
						MC_Dodge_Right();
					else if(m_nCurMetacmd == MC_DODGE_LEFT)
						MC_Dodge_Left();
					else if(IsRandomChance(50))
						MC_Dodge_Right();
					else
						MC_Dodge_Left();

					break;
		case 4:		ComputeState();									break;
	}

	return;
}

// ----------------------------------------------------------------------- //
// ROUTINE		: EnhancedGideon::AI_STATE_AttackFar
// DESCRIPTION	: 
// RETURN TYPE	: void 
// ----------------------------------------------------------------------- //

void EnhancedGideon::AI_STATE_AttackFar()
{
	//SCHLEGZ 4/22/98 4:51:21 PM: sanity check
	if(m_hTarget == DNULL)
	{
		SetNewState(STATE_SearchVisualTarget);
		return;
	}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -