hlsdk-portable/dlls/satchel.cpp

531 lines
13 KiB
C++
Raw Normal View History

2016-06-04 13:24:23 +00:00
/***
*
* Copyright (c) 1996-2002, Valve LLC. All rights reserved.
*
* This product contains software technology licensed from Id
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
* All Rights Reserved.
*
* Use, distribution, and modification of this source code and/or resulting
* object code is restricted to non-commercial enhancements to products from
* Valve LLC. All other use, distribution, or modification is prohibited
* without written permission from Valve LLC.
*
****/
2021-06-07 14:17:22 +00:00
#if !OEM_BUILD && !HLDEMO_BUILD
2016-06-04 13:24:23 +00:00
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "weapons.h"
#include "nodes.h"
#include "player.h"
#include "gamerules.h"
enum satchel_state
{
SATCHEL_IDLE = 0,
SATCHEL_READY,
SATCHEL_RELOAD
};
2016-07-31 13:48:50 +00:00
enum satchel_e
{
2016-06-04 13:24:23 +00:00
SATCHEL_IDLE1 = 0,
SATCHEL_FIDGET1,
SATCHEL_DRAW,
SATCHEL_DROP
};
2016-07-31 13:48:50 +00:00
enum satchel_radio_e
{
2016-06-04 13:24:23 +00:00
SATCHEL_RADIO_IDLE1 = 0,
SATCHEL_RADIO_FIDGET1,
SATCHEL_RADIO_DRAW,
SATCHEL_RADIO_FIRE,
SATCHEL_RADIO_HOLSTER
};
LINK_ENTITY_TO_CLASS( monster_satchel, CPipebombCharge )
2016-06-04 13:24:23 +00:00
//=========================================================
// Deactivate - do whatever it is we do to an orphaned
// satchel when we don't want it in the world anymore.
//=========================================================
void CPipebombCharge::Deactivate( void )
2016-06-04 13:24:23 +00:00
{
pev->solid = SOLID_NOT;
UTIL_Remove( this );
}
void CPipebombCharge::PipebombUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value )
{
pev->owner = m_hOwner->edict();
SetThink( &CGrenade::Detonate );
pev->nextthink = gpGlobals->time;
}
void CPipebombCharge::Spawn( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
Precache();
2016-06-04 13:24:23 +00:00
// motor
pev->movetype = MOVETYPE_BOUNCE;
pev->solid = SOLID_BBOX;
SET_MODEL( ENT( pev ), "models/w_pipebomb.mdl" );
2016-07-31 13:48:50 +00:00
//UTIL_SetSize( pev, Vector( -16, -16, -4 ), Vector( 16, 16, 32 ) ); // Old box -- size of headcrab monsters/players get blocked by this
UTIL_SetSize( pev, Vector( -4, -4, -4 ), Vector( 4, 4, 4 ) ); // Uses point-sized, and can be stepped over
2016-06-04 13:24:23 +00:00
UTIL_SetOrigin( pev, pev->origin );
SetTouch( &CPipebombCharge::PipebombSlide );
SetUse( &CPipebombCharge::PipebombUse );
SetThink( &CPipebombCharge::PipebombThink );
2019-10-13 11:49:25 +00:00
pev->nextthink = gpGlobals->time + 0.1f;
2016-06-04 13:24:23 +00:00
2019-10-13 11:49:25 +00:00
pev->gravity = 0.5f;
2019-11-17 11:41:46 +00:00
pev->friction = 0.5f;
2016-06-04 13:24:23 +00:00
pev->dmg = gSkillData.plrDmgSatchel;
2016-07-31 13:48:50 +00:00
// ResetSequenceInfo();
2016-06-04 13:24:23 +00:00
pev->sequence = 1;
m_flDropTime = gpGlobals->time;
2016-06-04 13:24:23 +00:00
}
void CPipebombCharge::PipebombSlide( CBaseEntity *pOther )
2016-06-04 13:24:23 +00:00
{
2017-06-29 13:56:03 +00:00
//entvars_t *pevOther = pOther->pev;
2016-06-04 13:24:23 +00:00
// don't hit the guy that launched this grenade
if( pOther->edict() == m_hOwner->edict() )
{
if( pev->velocity.Length2D() < 10.0f
&& m_flDropTime + 0.5f <= gpGlobals->time
&& pOther->GiveAmmo( SATCHEL_DEFAULT_GIVE, "Satchel Charge", SATCHEL_MAX_CARRY ) != -1 )
{
CBasePlayer *pPlayer = static_cast<CBasePlayer*>( pOther );
EMIT_SOUND( ENT( pev ), CHAN_ITEM, "items/9mmclip1.wav", 1, ATTN_NORM );
2016-06-04 13:24:23 +00:00
UTIL_Remove( this );
CPipebomb *pPipebomb = static_cast<CPipebomb*>( pPlayer->GiveNamedPlayerItem( "weapon_pipebomb" ) );
if( pPipebomb )
pPipebomb->PipebombReload();
}
return;
}
2016-07-31 13:48:50 +00:00
// pev->avelocity = Vector( 300, 300, 300 );
pev->gravity = 0.7f;// normal gravity now
2016-06-04 13:24:23 +00:00
// HACKHACK - On ground isn't always set, so look for ground underneath
TraceResult tr;
2016-07-31 13:48:50 +00:00
UTIL_TraceLine( pev->origin, pev->origin - Vector( 0, 0, 10 ), ignore_monsters, edict(), &tr );
2016-06-04 13:24:23 +00:00
2019-10-13 11:49:25 +00:00
if( tr.flFraction < 1.0f )
2016-06-04 13:24:23 +00:00
{
// add a bit of static friction
pev->velocity = pev->velocity * 0.7;
2016-06-04 13:24:23 +00:00
pev->avelocity = pev->avelocity * 0.9;
// play sliding sound, volume based on velocity
}
2019-10-13 11:49:25 +00:00
if( !( pev->flags & FL_ONGROUND ) && pev->velocity.Length2D() > 10.0f )
2016-06-04 13:24:23 +00:00
{
// Fix for a bug in engine: when object isn't moving, but its speed isn't 0 and on ground isn't set
if( pev->origin != m_lastBounceOrigin )
BounceSound();
2016-06-04 13:24:23 +00:00
}
m_lastBounceOrigin = pev->origin;
// There is no model animation so commented this out to prevent net traffic
// StudioFrameAdvance();
2016-06-04 13:24:23 +00:00
}
void CPipebombCharge::PipebombThink( void )
2016-06-04 13:24:23 +00:00
{
// There is no model animation so commented this out to prevent net traffic
// StudioFrameAdvance();
2019-10-13 11:49:25 +00:00
pev->nextthink = gpGlobals->time + 0.1f;
2016-06-04 13:24:23 +00:00
if( pev->owner && m_flDropTime + 0.5f <= gpGlobals->time )
pev->owner = 0;
2016-07-31 13:48:50 +00:00
if( !IsInWorld() )
2016-06-04 13:24:23 +00:00
{
UTIL_Remove( this );
return;
}
2016-07-31 13:48:50 +00:00
if( pev->waterlevel == 3 )
2016-06-04 13:24:23 +00:00
{
pev->movetype = MOVETYPE_FLY;
2019-10-13 11:49:25 +00:00
pev->velocity = pev->velocity * 0.8f;
pev->avelocity = pev->avelocity * 0.9f;
2016-06-04 13:24:23 +00:00
pev->velocity.z += 8;
}
2016-07-31 13:48:50 +00:00
else if( pev->waterlevel == 0 )
2016-06-04 13:24:23 +00:00
{
pev->movetype = MOVETYPE_BOUNCE;
}
else
{
2019-10-13 11:49:25 +00:00
pev->velocity.z -= 8.0f;
2016-06-04 13:24:23 +00:00
}
}
void CPipebombCharge::Precache( void )
2016-06-04 13:24:23 +00:00
{
PRECACHE_SOUND( "weapons/pb_bounce1.wav" );
PRECACHE_SOUND( "weapons/pb_bounce2.wav" );
PRECACHE_SOUND( "weapons/pb_bounce3.wav" );
m_iTrail = PRECACHE_MODEL( "sprites/white.spr" );
2016-06-04 13:24:23 +00:00
}
void CPipebombCharge::BounceSound( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
switch( RANDOM_LONG( 0, 2 ) )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
case 0:
EMIT_SOUND( ENT( pev ), CHAN_VOICE, "weapons/pb_bounce1.wav", 1, ATTN_NORM );
2016-07-31 13:48:50 +00:00
break;
case 1:
EMIT_SOUND( ENT( pev ), CHAN_VOICE, "weapons/pb_bounce2.wav", 1, ATTN_NORM );
2016-07-31 13:48:50 +00:00
break;
case 2:
EMIT_SOUND( ENT( pev ), CHAN_VOICE, "weapons/pb_bounce3.wav", 1, ATTN_NORM );
2016-07-31 13:48:50 +00:00
break;
2016-06-04 13:24:23 +00:00
}
}
LINK_ENTITY_TO_CLASS( weapon_pipebomb, CPipebomb )
2016-06-04 13:24:23 +00:00
//=========================================================
// CALLED THROUGH the newly-touched weapon's instance. The existing player weapon is pOriginal
//=========================================================
int CPipebomb::AddDuplicate( CBasePlayerItem *pOriginal )
2016-06-04 13:24:23 +00:00
{
CPipebomb *pPipebomb;
2016-06-04 13:24:23 +00:00
2021-06-07 00:05:58 +00:00
#if CLIENT_DLL
2016-07-31 13:48:50 +00:00
if( bIsMultiplayer() )
2016-06-04 13:24:23 +00:00
#else
2016-07-31 13:48:50 +00:00
if( g_pGameRules->IsMultiplayer() )
2016-06-04 13:24:23 +00:00
#endif
{
pPipebomb = (CPipebomb *)pOriginal;
2016-06-04 13:24:23 +00:00
if( pPipebomb->m_chargeReady != SATCHEL_IDLE )
2016-06-04 13:24:23 +00:00
{
// player has some satchels deployed. Refuse to add more.
return FALSE;
}
}
2016-07-31 13:48:50 +00:00
return CBasePlayerWeapon::AddDuplicate( pOriginal );
2016-06-04 13:24:23 +00:00
}
//=========================================================
//=========================================================
int CPipebomb::AddToPlayer( CBasePlayer *pPlayer )
2016-06-04 13:24:23 +00:00
{
int bResult = CBasePlayerItem::AddToPlayer( pPlayer );
2016-07-31 13:48:50 +00:00
pPlayer->pev->weapons |= ( 1 << m_iId );
m_chargeReady = SATCHEL_IDLE;// this satchel charge weapon now forgets that any satchels are deployed by it.
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( bResult )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
return AddWeapon();
2016-06-04 13:24:23 +00:00
}
return FALSE;
}
void CPipebomb::Spawn()
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
Precache();
2016-06-04 13:24:23 +00:00
m_iId = WEAPON_SATCHEL;
SET_MODEL( ENT( pev ), "models/w_pipebomb.mdl" );
2016-06-04 13:24:23 +00:00
m_iDefaultAmmo = SATCHEL_DEFAULT_GIVE;
2016-06-04 13:24:23 +00:00
FallInit();// get ready to fall down.
}
void CPipebomb::Precache( void )
2016-06-04 13:24:23 +00:00
{
PRECACHE_MODEL( "models/v_pipebomb.mdl" );
PRECACHE_MODEL( "models/v_pipebomb_watch.mdl" );
PRECACHE_MODEL( "models/w_pipebomb.mdl" );
PRECACHE_MODEL( "models/p_pipebomb.mdl" );
PRECACHE_MODEL( "models/p_pipebomb_watch.mdl" );
2016-06-04 13:24:23 +00:00
UTIL_PrecacheOther( "monster_satchel" );
}
int CPipebomb::GetItemInfo( ItemInfo *p )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
p->pszName = STRING( pev->classname );
2016-06-04 13:24:23 +00:00
p->pszAmmo1 = "Satchel Charge";
p->iMaxAmmo1 = SATCHEL_MAX_CARRY;
p->pszAmmo2 = NULL;
p->iMaxAmmo2 = -1;
p->iMaxClip = WEAPON_NOCLIP;
p->iSlot = 4;
p->iPosition = 0;
2016-06-04 13:24:23 +00:00
p->iFlags = ITEM_FLAG_SELECTONEMPTY | ITEM_FLAG_LIMITINWORLD | ITEM_FLAG_EXHAUSTIBLE;
p->iId = m_iId = WEAPON_SATCHEL;
p->iWeight = SATCHEL_WEIGHT;
return 1;
}
//=========================================================
//=========================================================
BOOL CPipebomb::IsUseable( void )
2016-06-04 13:24:23 +00:00
{
return CanDeploy();
2016-06-04 13:24:23 +00:00
}
BOOL CPipebomb::CanDeploy( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( m_pPlayer->m_rgAmmo[PrimaryAmmoIndex()] > 0 )
2016-06-04 13:24:23 +00:00
{
// player is carrying some satchels
return TRUE;
}
if( m_chargeReady )
2016-06-04 13:24:23 +00:00
{
// player isn't carrying any satchels, but has some out
return TRUE;
}
return FALSE;
}
BOOL CPipebomb::Deploy()
2016-06-04 13:24:23 +00:00
{
2019-10-13 11:49:25 +00:00
m_pPlayer->m_flNextAttack = UTIL_WeaponTimeBase() + 1.0f;
BOOL result;
2016-06-04 13:24:23 +00:00
2016-07-31 13:48:50 +00:00
if( m_chargeReady )
2022-08-24 14:17:39 +00:00
result = DefaultDeploy( "models/v_pipebomb_watch.mdl", "models/p_pipebomb_watch.mdl", SATCHEL_RADIO_DRAW, "hive" );
2016-06-04 13:24:23 +00:00
else
2022-08-24 14:17:39 +00:00
result = DefaultDeploy( "models/v_pipebomb.mdl", "models/p_pipebomb.mdl", SATCHEL_DRAW, "trip" );
2016-06-04 13:24:23 +00:00
#if WEAPONS_ANIMATION_TIMES_FIX
if ( result )
{
m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + 2.0f;
}
#endif
return result;
2016-06-04 13:24:23 +00:00
}
void CPipebomb::Holster( int skiplocal /* = 0 */ )
2016-06-04 13:24:23 +00:00
{
2019-10-13 11:49:25 +00:00
m_pPlayer->m_flNextAttack = UTIL_WeaponTimeBase() + 0.5f;
2016-07-31 13:48:50 +00:00
if( m_chargeReady )
2016-06-04 13:24:23 +00:00
{
SendWeaponAnim( SATCHEL_RADIO_HOLSTER );
}
else
{
SendWeaponAnim( SATCHEL_DROP );
}
2019-10-13 11:49:25 +00:00
EMIT_SOUND( ENT( m_pPlayer->pev ), CHAN_WEAPON, "common/null.wav", 1.0f, ATTN_NORM );
2016-06-04 13:24:23 +00:00
if( !m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] && m_chargeReady != SATCHEL_READY )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
m_pPlayer->pev->weapons &= ~( 1 << WEAPON_SATCHEL );
DestroyItem();
2016-06-04 13:24:23 +00:00
}
}
void CPipebomb::PrimaryAttack()
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
switch( m_chargeReady )
2016-06-04 13:24:23 +00:00
{
case SATCHEL_IDLE:
2016-06-04 13:24:23 +00:00
{
Throw();
2016-06-04 13:24:23 +00:00
}
break;
case SATCHEL_READY:
2016-06-04 13:24:23 +00:00
{
SendWeaponAnim( SATCHEL_RADIO_FIRE );
2016-06-04 13:24:23 +00:00
edict_t *pPlayer = m_pPlayer->edict();
2016-06-04 13:24:23 +00:00
CBaseEntity *pEnt = NULL;
2016-06-04 13:24:23 +00:00
while( ( pEnt = UTIL_FindEntityInSphere( pEnt, m_pPlayer->pev->origin, 4096 ) ) != NULL )
2016-06-04 13:24:23 +00:00
{
if( FClassnameIs( pEnt->pev, "monster_satchel" ) )
2016-06-04 13:24:23 +00:00
{
CPipebombCharge *pPipebomb = (CPipebombCharge *)pEnt;
if( pPipebomb->m_hOwner->edict() == pPlayer )
{
pPipebomb->Use( m_pPlayer, m_pPlayer, USE_ON, 0 );
}
2016-06-04 13:24:23 +00:00
}
}
m_chargeReady = SATCHEL_RELOAD;
2019-11-17 11:41:46 +00:00
m_flNextPrimaryAttack = UTIL_WeaponTimeBase() + 0.5f;
2019-10-13 11:49:25 +00:00
m_flNextSecondaryAttack = UTIL_WeaponTimeBase() + 0.5f;
m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + 0.5f;
break;
2016-06-04 13:24:23 +00:00
}
case SATCHEL_RELOAD:
2016-06-04 13:24:23 +00:00
// we're reloading, don't allow fire
break;
}
}
void CPipebomb::SecondaryAttack( void )
2016-06-04 13:24:23 +00:00
{
if( m_chargeReady != SATCHEL_RELOAD )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
Throw();
2016-06-04 13:24:23 +00:00
}
}
void CPipebomb::Throw( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] )
2016-06-04 13:24:23 +00:00
{
2021-06-07 00:05:58 +00:00
#if !CLIENT_DLL
2016-06-04 13:24:23 +00:00
Vector vecSrc = m_pPlayer->pev->origin;
Vector vecThrow = gpGlobals->v_forward * 274 + m_pPlayer->pev->velocity;
CPipebombCharge *pPipebomb = (CPipebombCharge *)Create( "monster_satchel", vecSrc, g_vecZero, m_pPlayer->edict() );
pPipebomb->pev->velocity = vecThrow;
pPipebomb->pev->avelocity.y = 400;
pPipebomb->m_hOwner = m_pPlayer;
pPipebomb->pev->owner = pPipebomb->m_hOwner->edict();
2016-06-04 13:24:23 +00:00
m_pPlayer->pev->viewmodel = MAKE_STRING( "models/v_pipebomb_watch.mdl" );
m_pPlayer->pev->weaponmodel = MAKE_STRING( "models/p_pipebomb_watch.mdl" );
2016-06-04 13:24:23 +00:00
#else
LoadVModel( "models/v_pipebomb_watch.mdl", m_pPlayer );
2016-06-04 13:24:23 +00:00
#endif
SendWeaponAnim( SATCHEL_RADIO_DRAW );
// player "shoot" animation
m_pPlayer->SetAnimation( PLAYER_ATTACK1 );
m_chargeReady = SATCHEL_READY;
2016-06-04 13:24:23 +00:00
m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType]--;
2019-11-17 11:41:46 +00:00
m_flNextPrimaryAttack = UTIL_WeaponTimeBase() + 1.0f;
2019-10-13 11:49:25 +00:00
m_flNextSecondaryAttack = UTIL_WeaponTimeBase() + 0.5f;
2016-06-04 13:24:23 +00:00
}
}
void CPipebomb::WeaponIdle( void )
2016-06-04 13:24:23 +00:00
{
2016-07-31 13:48:50 +00:00
if( m_flTimeWeaponIdle > UTIL_WeaponTimeBase() )
2016-06-04 13:24:23 +00:00
return;
switch( m_chargeReady )
{
case SATCHEL_IDLE:
2016-06-04 13:24:23 +00:00
SendWeaponAnim( SATCHEL_FIDGET1 );
// use tripmine animations
strcpy( m_pPlayer->m_szAnimExtention, "trip" );
break;
case SATCHEL_READY:
2016-06-04 13:24:23 +00:00
SendWeaponAnim( SATCHEL_RADIO_FIDGET1 );
// use hivehand animations
strcpy( m_pPlayer->m_szAnimExtention, "hive" );
break;
case SATCHEL_RELOAD:
2016-07-31 13:48:50 +00:00
if( !m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] )
2016-06-04 13:24:23 +00:00
{
m_chargeReady = 0;
RetireWeapon();
return;
}
2021-06-07 00:05:58 +00:00
#if !CLIENT_DLL
m_pPlayer->pev->viewmodel = MAKE_STRING( "models/v_pipebomb.mdl" );
m_pPlayer->pev->weaponmodel = MAKE_STRING( "models/p_pipebomb.mdl" );
2016-06-04 13:24:23 +00:00
#else
LoadVModel( "models/v_pipebomb.mdl", m_pPlayer );
2016-06-04 13:24:23 +00:00
#endif
SendWeaponAnim( SATCHEL_DRAW );
// use tripmine animations
strcpy( m_pPlayer->m_szAnimExtention, "trip" );
2019-11-17 11:41:46 +00:00
m_flNextPrimaryAttack = UTIL_WeaponTimeBase() + 0.5f;
2019-10-13 11:49:25 +00:00
m_flNextSecondaryAttack = UTIL_WeaponTimeBase() + 0.5f;
m_chargeReady = SATCHEL_IDLE;
2016-06-04 13:24:23 +00:00
break;
}
m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + UTIL_SharedRandomFloat( m_pPlayer->random_seed, 10, 15 );// how long till we do this again.
}
void CPipebomb::PipebombReload()
{
int i = 0;
edict_t *pPlayer = m_pPlayer->edict();
CBaseEntity *pEntity = NULL;
while( ( pEntity = UTIL_FindEntityInSphere( pEntity, m_pPlayer->pev->origin, 4096 ) ) != NULL )
{
if( FClassnameIs( pEntity->pev, "monster_satchel" ) )
{
CPipebombCharge *pPipebomb = (CPipebombCharge *)pEntity;
if( pPipebomb->m_hOwner->edict() == pPlayer )
{
++i;
}
}
}
if( i < 2 )
{
2020-04-08 21:07:42 +00:00
m_flNextPrimaryAttack = m_flNextSecondaryAttack = UTIL_WeaponTimeBase() + 0.5f;
m_flTimeWeaponIdle = UTIL_WeaponTimeBase();
m_chargeReady = SATCHEL_RELOAD;
}
}
2016-06-04 13:24:23 +00:00
//=========================================================
// DeactivatePipebombs - removes all pipebombs owned by
2016-06-04 13:24:23 +00:00
// the provided player. Should only be used upon death.
//
// Made this global on purpose.
//=========================================================
void DeactivatePipebombs( CBasePlayer *pOwner )
2016-06-04 13:24:23 +00:00
{
edict_t *pFind;
pFind = FIND_ENTITY_BY_CLASSNAME( NULL, "monster_satchel" );
2016-07-31 13:48:50 +00:00
while( !FNullEnt( pFind ) )
2016-06-04 13:24:23 +00:00
{
CBaseEntity *pEnt = CBaseEntity::Instance( pFind );
CPipebombCharge *pPipebomb = (CPipebombCharge *)pEnt;
2016-06-04 13:24:23 +00:00
if( pPipebomb )
2016-06-04 13:24:23 +00:00
{
if( pPipebomb->m_hOwner->edict() == pOwner->edict() )
2016-06-04 13:24:23 +00:00
{
pPipebomb->Deactivate();
2016-06-04 13:24:23 +00:00
}
}
pFind = FIND_ENTITY_BY_CLASSNAME( pFind, "monster_satchel" );
}
}
2016-04-17 20:30:05 +00:00
#endif