summaryrefslogblamecommitdiffstats
path: root/src/objects/Object.cpp
blob: 42021054f9cb5580d49eea94eb5dcc452f35197e (plain) (tree)
1
2
3
4
5
6
7
8
                   
 

                   
                  

                   
                        




                       
                      

                       

                      
 
                              

                                        
 



                                                                                                    
 




                                     

                                                             
                                         
                             

                                                            

                              
                          
                         
                          
                                      
                            

                              


                                   
                                                            

                                 


                                 

 








                                          
                                     
 
                                                      









                                                    
                               

 



                                                                                         
                                   

                                                                        
                                                                  


                                 


                             





                                                                                
                                   





                                                                    

                                                                                                                      
                                                                





                                                           

                                                          
                                                                                      



                                                                            










































                                                                                                                                                                                            

 
    
                                 
 




                                        

 


                     
                         

                       
                                                                                             
                                                                                                       
                                                             


                                                           



































































































                                                                                                                                   


                          


                            
                              

                                                                        

                             

                                 

                             


                                                         






                                   
                    


                                        

 

                                   
 



                                                         
     
                                             
                                   
                                                                                   





                                                     
      
                                                                                                                    



                                                               
                                                        
                                                      


































                                                                                                                                                     
                         





                                                                                                                     

                                                       


                                                               
                                                  
                                                       

                                                               
                                      
                         


































                                                                                                                                                                 
                         

























                                                                                                                                                                    
                         
































                                                                                                                                                           
                         


























                                                                                                                                                           
                         

























































































































                                                                                                                                                                       
                 

         






                                                    
 


                   
                                    
                                                           

                                      
                          
                          
                                      








                                   
                          
                         







                                                                                                                                  
 
                                       
                                      
 
                                                                                       

                                                        

                                 

 











                                     

                                           




                                    


                                  







                                                                            

 

                               
 







                                                                         

 

                                                                 
 
                                                          
                                                           
                                                          

                                                                                                                    



                                                
 














                                                                         
#include "common.h"

#include "main.h"
#include "Lights.h"
#include "Pools.h"
#include "Radar.h"
#include "Object.h"
#include "DummyObject.h"
#include "Particle.h"
#include "General.h"
#include "ObjectData.h"
#include "World.h"
#include "Floater.h"
#include "soundlist.h"
#include "WaterLevel.h"
#include "Timecycle.h"
#include "Stats.h"
#include "SpecialFX.h"

int16 CObject::nNoTempObjects;
//int16 CObject::nBodyCastHealth = 1000;
float CObject::fDistToNearestTree;

void* CObject::operator new(size_t sz) { return CPools::GetObjectPool()->New(); }
void* CObject::operator new(size_t sz, int handle) { return CPools::GetObjectPool()->New(handle); };
void CObject::operator delete(void* p, size_t sz) { CPools::GetObjectPool()->Delete((CObject*)p); }
void CObject::operator delete(void* p, int handle) { CPools::GetObjectPool()->Delete((CObject*)p); }

CObject::CObject(void)
{
	m_type = ENTITY_TYPE_OBJECT;
	m_fUprootLimit = 0.0f;
	m_nCollisionDamageEffect = 0;
	m_nSpecialCollisionResponseCases = COLLRESPONSE_NONE;
	m_bCameraToAvoidThisObject = false;
	ObjectCreatedBy = UNKNOWN_OBJECT;
	m_nEndOfLifeTime = 0;
	//	m_nRefModelIndex = -1;	// duplicate
	//	bUseVehicleColours = false;	// duplicate
	m_colour2 = 0;
	m_colour1 = m_colour2;
	m_nBonusValue = 0;
	m_nCostValue = 0;
	bIsPickup = false;
	bPickupObjWithMessage = false;
	bOutOfStock = false;
	bGlassCracked = false;
	bGlassBroken = false;
	bHasBeenDamaged = false;
	m_nRefModelIndex = -1;
	bUseVehicleColours = false;
	//	bIsStreetLight = false;		// duplicate
	m_pCurSurface = nil;
	m_pCollidingEntity = nil;
	m_nBeachballBounces = 0;
	bIsStreetLight = false;
	m_area = AREA_EVERYWHERE;
}

CObject::CObject(int32 mi, bool createRW)
{
	if (createRW)
		SetModelIndex(mi);
	else
		SetModelIndexNoCreate(mi);
	Init();
}

CObject::CObject(CDummyObject* dummy)
{
	SetModelIndexNoCreate(dummy->GetModelIndex());

	if (dummy->m_rwObject)
		AttachToRwObject(dummy->m_rwObject);
	else
		GetMatrix() = dummy->GetMatrix();

	m_objectMatrix = dummy->GetMatrix();
	dummy->DetachFromRwObject();
	Init();
	m_level = dummy->m_level;
	m_area = dummy->m_area;
}

CObject::~CObject(void)
{
	CRadar::ClearBlipForEntity(BLIP_OBJECT, CPools::GetObjectPool()->GetIndex(this));

	if (m_nRefModelIndex != -1)
		CModelInfo::GetModelInfo(m_nRefModelIndex)->RemoveRef();

	if (ObjectCreatedBy == TEMP_OBJECT && nNoTempObjects != 0)
		nNoTempObjects--;
}

void
CObject::ProcessControl(void)
{
	CVector point, impulse;
	if (m_nCollisionDamageEffect)
		ObjectDamage(m_fDamageImpulse);
	CPhysical::ProcessControl();
	if (mod_Buoyancy.ProcessBuoyancy(this, m_fBuoyancy, &point, &impulse)) {
		bIsInWater = true;
		SetIsStatic(false);
		ApplyMoveForce(impulse);
		ApplyTurnForce(impulse, point);
		float fTimeStep = Pow(0.97f, CTimer::GetTimeStep());
		m_vecMoveSpeed *= fTimeStep;
		m_vecTurnSpeed *= fTimeStep;
	}
	auto mi = GetModelIndex();
	if ((mi == MI_EXPLODINGBARREL || mi == MI_PETROLPUMP || mi == MI_PETROLPUMP2) && bHasBeenDamaged && bIsVisible
		&& (CGeneral::GetRandomNumber() & 0x1F) == 10) {
		bExplosionProof = true;
		bIsVisible = false;
		bUsesCollision = false;
		bAffectedByGravity = false;
		m_vecMoveSpeed = CVector(0.0f, 0.0f, 0.0f);
	}
	if (mi == MI_RCBOMB) {
		float fTurnForce = -(m_fTurnMass / 20.0f);
		CPhysical::ApplyTurnForce(m_vecMoveSpeed * fTurnForce, -GetForward());
		float fScalar = 1.0f - m_vecMoveSpeed.MagnitudeSqr() / 5.0f;
		float fScalarTimed = Pow(fScalar, CTimer::GetTimeStep());
		m_vecMoveSpeed *= fScalarTimed;
	}
	if (mi == MI_BEACHBALL) {
		constexpr uint8 BEACHBALL_MAX_SCORE = 250;
		constexpr float BEACHBALL_DEACCELERATION = 2.5f;
		float fMoveSpeedMag = m_vecMoveSpeed.Magnitude2D();
		float fTimeScale = powf(0.95, CTimer::GetTimeStep());
		m_vecMoveSpeed.x *= fTimeScale;
		m_vecMoveSpeed.y *= fTimeScale;
		m_vecMoveSpeed.z += fMoveSpeedMag - m_vecMoveSpeed.Magnitude2D();
		if (!FindPlayerVehicle()) {
			CVector distance = FindPlayerCoors() - GetPosition();
			float distanceMagnitude = distance.Magnitude2D();
			if (distance.z > 0.0 && distance.z < 1.5f && distanceMagnitude < 1.0) {
				CVector playerSpeed = FindPlayerSpeed();
				if (fMoveSpeedMag < 0.05 && playerSpeed.Magnitude() > 0.1) {
					playerSpeed.z = 0.0f;
					playerSpeed.Normalise();
					playerSpeed.z = 0.3;
					m_vecMoveSpeed = CVector(playerSpeed.x / BEACHBALL_DEACCELERATION, playerSpeed.y / BEACHBALL_DEACCELERATION, 1.0f / BEACHBALL_DEACCELERATION * 0.3);
					PlayOneShotScriptObject(SCRIPT_SOUND_HIT_BALL, GetPosition());
					if (m_nBeachballBounces > 0) {
						m_nBeachballBounces++;
						sprintf(gString, "%d", m_nBeachballBounces);
						CMoneyMessages::RegisterOne(GetPosition(), gString, 255, 50, 0, 0.6f, 0.5f);
						CStats::RegisterHighestScore(3, m_nBeachballBounces);
					}
				}
			}
			if (distance.z > -1.05 && distance.z < -0.6 && distanceMagnitude < 0.9 && m_vecMoveSpeed.z < 0.0f) {
				m_vecMoveSpeed.x += CGeneral::GetRandomNumberInRangeInc(-3, 4) / 100.0;
				m_vecMoveSpeed.y += CGeneral::GetRandomNumberInRangeInc(-3, 4) / 100.0;
				m_vecMoveSpeed.z = Max(m_vecMoveSpeed.z + 0.3f, 0.2f);
				PlayOneShotScriptObject(SCRIPT_SOUND_HIT_BALL, GetPosition());
				m_vecTurnSpeed.x += CGeneral::GetRandomNumberInRangeInc(-7, 8) / 10.0f;
				m_vecTurnSpeed.y += CGeneral::GetRandomNumberInRangeInc(-7, 8) / 10.0f;
				if (++m_nBeachballBounces >= BEACHBALL_MAX_SCORE) {
					m_nBeachballBounces = BEACHBALL_MAX_SCORE;
				}
				sprintf(gString, "%d", m_nBeachballBounces);
				CMoneyMessages::RegisterOne(GetPosition(), gString, 255, 50, 0, 0.6f, 0.5f);
				CStats::RegisterHighestScore(3, m_nBeachballBounces);
			}
		}
	}
}

void
CObject::Teleport(CVector vecPos)
{
	CWorld::Remove(this);
	m_matrix.GetPosition() = vecPos;
	m_matrix.UpdateRW();
	UpdateRwFrame();
	CWorld::Add(this);
}

void
CObject::Render(void)
{
	if (bDoNotRender)
		return;

	if (m_nRefModelIndex != -1 && ObjectCreatedBy == TEMP_OBJECT && bUseVehicleColours) {
		CVehicleModelInfo* mi = (CVehicleModelInfo*)CModelInfo::GetModelInfo(m_nRefModelIndex);
		assert(mi->GetModelType() == MITYPE_VEHICLE);
		mi->SetVehicleColour(m_colour1, m_colour2);
	}

	float red = (0.8f * CTimeCycle::GetDirectionalRed() + CTimeCycle::GetAmbientRed_Obj()) * 165.75f;
	float green = (0.8f * CTimeCycle::GetDirectionalGreen() + CTimeCycle::GetAmbientGreen_Obj()) * 165.75f;
	float blue = (0.8f * CTimeCycle::GetDirectionalBlue() + CTimeCycle::GetAmbientBlue_Obj()) * 165.75f;

	red = clamp(red, 0.0f, 255.0f);
	green = clamp(green, 0.0f, 255.0f);
	blue = clamp(blue, 0.0f, 255.0f);

	int alpha = CGeneral::GetRandomNumberInRange(196, 225);

	RwRGBA color = { (uint8)red, (uint8)green, (uint8)blue, (uint8)alpha };

	if (this->GetModelIndex() == MI_YT_MAIN_BODY) {
		float moveSpeedMagnitude = this->GetMoveSpeed().Magnitude();
		if (moveSpeedMagnitude > 0.0f) {
			float scaleMax = GetColModel()->boundingBox.max.y * 0.85f;

			CVector dir = this->GetMoveSpeed() + 0.3f * this->GetRight() - 0.5f * this->GetForward();
			dir.z += 0.05f * moveSpeedMagnitude;

			CVector pos = scaleMax * this->GetForward() + 2.25f * this->GetRight() + this->GetPosition();

			float fWaterLevel;
			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.75f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 1.2f * moveSpeedMagnitude, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);

			float scaleMin = GetColModel()->boundingBox.min.y;

			dir = this->GetMoveSpeed() - 0.5f * this->GetForward();
			dir.z += 0.05f * moveSpeedMagnitude;

			pos = scaleMin * this->GetForward() + 4.5f * this->GetRight() + this->GetPosition();

			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.55f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 0.9f, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);

			pos = scaleMin * 1.1f * this->GetForward() + 2.25f * this->GetRight() + this->GetPosition();

			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.55f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 0.9f, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);

			pos = scaleMin * 1.1f * this->GetForward() - 0.05f * this->GetRight() + this->GetPosition();

			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.55f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 0.9f, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);
		}
	}

	if (this->GetModelIndex() == MI_YT_MAIN_BODY2) {
		float moveSpeedMagnitude = this->GetMoveSpeed().Magnitude();
		if (moveSpeedMagnitude > 0.0f) {
			float scaleMax = GetColModel()->boundingBox.max.y * 0.85f;

			CVector dir = this->GetMoveSpeed() - 0.3f * this->GetRight() - 0.5f * this->GetForward();
			dir.z += 0.05f * moveSpeedMagnitude;

			CVector pos = scaleMax * this->GetForward() - 2.25f * this->GetRight() + this->GetPosition();

			float fWaterLevel;
			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.75f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 1.2f * moveSpeedMagnitude, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);

			float scaleMin = GetColModel()->boundingBox.min.y;

			dir = this->GetMoveSpeed() - 0.5f * this->GetForward();
			dir.z += 0.05f * moveSpeedMagnitude;

			pos = scaleMin * this->GetForward() - 4.5f * this->GetRight() + this->GetPosition();

			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.55f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 0.9f, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);

			pos = scaleMin * 1.1f * this->GetForward() - 2.25f * this->GetRight() + this->GetPosition();

			CWaterLevel::GetWaterLevel(pos.x, pos.y, pos.z, &fWaterLevel, true);
			pos.z = fWaterLevel + 0.55f;

			CParticle::AddParticle(PARTICLE_BOAT_SPLASH, pos, dir, nil, 0.9f, color,
				CGeneral::GetRandomNumberInRange(0.0f, 0.4f), CGeneral::GetRandomNumberInRange(0.0f, 45.0f), 0, 0);
		}
	}

	CEntity::Render();
}

bool
CObject::SetupLighting(void)
{
	if (bRenderScorched) {
		WorldReplaceNormalLightsWithScorched(Scene.world, 0.1f);
		return true;
	}
	else if (bIsPickup) {
		SetFullAmbient();
		return true;
	}
	else if (bIsWeapon) {
		ActivateDirectional();
		SetAmbientColoursForPedsCarsAndObjects();
		return true;
	}
	return false;
}

void
CObject::RemoveLighting(bool reset)
{
	if (reset) {
		SetAmbientColours();
		DeActivateDirectional();
	}
}

void
CObject::ObjectDamage(float amount)
{
	if (!m_nCollisionDamageEffect || !bUsesCollision)
		return;
	static int8 nFrameGen = 0;
	bool bBodyCastDamageEffect = false;
#if 0
	if (GetModelIndex() == MI_BODYCAST) {
		if (amount > 50.0f)
			nBodyCastHealth = (int16)(nBodyCastHealth - 0.5f * amount);
		if (nBodyCastHealth < 0)
			nBodyCastHealth = 0;
		if (nBodyCastHealth < 200)
			bBodyCastDamageEffect = true;
		amount = 0.0f;
	}
#endif
	if ((amount * m_fCollisionDamageMultiplier > 150.0f || bBodyCastDamageEffect) && m_nCollisionDamageEffect) {
		const CVector& vecPos = m_matrix.GetPosition();
		const float fDirectionZ = 0.0002f * amount;
		switch (m_nCollisionDamageEffect)
		{
			case DAMAGE_EFFECT_CHANGE_MODEL:
				bRenderDamaged = true;
				return;
			case DAMAGE_EFFECT_SPLIT_MODEL:
				return;
			case DAMAGE_EFFECT_SMASH_AND_DAMAGE_TRAFFICLIGHTS: {
				static RwRGBA debrisColor = { 0xc8,0xc8,0xc8,0xff };
				if (bRenderDamaged) {
					break;
				}
				bRenderDamaged = true;
				CBaseModelInfo* modelInfo = CModelInfo::GetModelInfo(GetModelIndex());
				CVector min = modelInfo->GetColModel()->boundingBox.min * 0.85f;
				CVector max = modelInfo->GetColModel()->boundingBox.max * 0.85f;
				min.z = max.z;
				min = GetMatrix() * min;
				max = GetMatrix() * max;
				CVector temp = (max - min) * 0.02;

				for (int32 i = 0; i < 50; i++) {
					float fDirX = CGeneral::GetRandomNumberInRange(-0.35f, 0.35f);
					float fDirY = CGeneral::GetRandomNumberInRange(-0.35f, 0.35f);
					float fDirZ = CGeneral::GetRandomNumberInRange(0.10f, 0.25f);
					++nFrameGen;
					int32 currentFrame = nFrameGen & 3;
					CVector pos = min + temp * (float)i;
					CVector dir = CVector(fDirX, fDirY, fDirZ);
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					float fColorFactor = CGeneral::GetRandomNumberInRange(0.6f, 1.2f);
					RwRGBA color = debrisColor;
					color.red *= fColorFactor;
					color.green *= fColorFactor;
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-0.40f, 0.40f);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, pos, dir, nil, fSize, color, nRotationSpeed, 0, currentFrame, 0);
				}
				PlayOneShotScriptObject(SCRIPT_SOUND_METAL_COLLISION, min);
				break;
			}
			case DAMAGE_EFFECT_SMASH_COMPLETELY:
			case DAMAGE_EFFECT_CHANGE_THEN_SMASH: {
				if (m_nCollisionDamageEffect == DAMAGE_EFFECT_CHANGE_THEN_SMASH && !bRenderDamaged) {
					bRenderDamaged = true;
					return;
				}
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				break;
			}
			case DAMAGE_EFFECT_SMASH_CARDBOARD_COMPLETELY:
			case DAMAGE_EFFECT_SMASH_YELLOW_TARGET_COMPLETELY: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				const RwRGBA color = { 96, 48, 0, 255 };
				for (int32 i = 0; i < 25; i++) {
					CVector vecDir(CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(0.10f, 0.25f) + fDirectionZ);
					++nFrameGen;
					int32 currentFrame = nFrameGen & 3;
					float fRandom = CGeneral::GetRandomNumberInRange(0.01f, 1.0f);
					RwRGBA randomColor = color;
					if (m_nCollisionDamageEffect == DAMAGE_EFFECT_SMASH_CARDBOARD_COMPLETELY) {
						randomColor.red *= fRandom;
						randomColor.green *= fRandom;
						randomColor.blue *= fRandom;
					}
					else {
						randomColor.red = 0xff;
						randomColor.blue = 0xfc;
					}
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, randomColor, nRotationSpeed, 0, currentFrame, 0);
				}
				PlayOneShotScriptObject(SCRIPT_SOUND_BOX_DESTROYED_2, vecPos);
				break;
			}
			case DAMAGE_EFFECT_SMASH_WOODENBOX_COMPLETELY: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				static const RwRGBA color = { 128, 128, 128, 255 };
				CVector position = GetPosition();
				for (int32 i = 0; i < 45; i++) {
					CVector vecDir(CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(0.10f, 0.25f) + fDirectionZ);
					++nFrameGen;
					int32 currentFrame = nFrameGen & 3;
					float fRandom = CGeneral::GetRandomNumberInRange(0.5f, 1.0f);
					RwRGBA randomColor = { uint8(color.red * fRandom), uint8(color.green * fRandom), uint8(color.blue * fRandom), color.alpha };
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, randomColor, nRotationSpeed, 0, currentFrame, 0);
				}
				PlayOneShotScriptObject(SCRIPT_SOUND_BOX_DESTROYED_1, vecPos);
				break;
			}
			case DAMAGE_EFFECT_SMASH_TRAFFICCONE_COMPLETELY:
			case DAMAGE_EFFECT_BURST_BEACHBALL: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				const RwRGBA color1 = { 200, 0, 0, 255 };
				const RwRGBA color2 = { 200, 200, 200, 255 };
				for (int32 i = 0; i < 10; i++) {
					CVector vecDir(CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(0.10f, 0.25f) + fDirectionZ);
					++nFrameGen;
					int32 currentFrame = nFrameGen & 3;
					RwRGBA color = color2;
					if (nFrameGen & 1)
						color = color1;
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, color, nRotationSpeed, 0, currentFrame, 0);
				}
				if (m_nCollisionDamageEffect == DAMAGE_EFFECT_BURST_BEACHBALL) {
					PlayOneShotScriptObject(SCRIPT_SOUND_HIT_BALL, vecPos);
				}
				else {
					PlayOneShotScriptObject(SCRIPT_SOUND_TIRE_COLLISION, vecPos);
				}
				break;
			}
			case DAMAGE_EFFECT_SMASH_BARPOST_COMPLETELY: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				const RwRGBA color1 = { 200, 0, 0, 255 };
				const RwRGBA color2 = { 200, 200, 200, 255 };
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				for (int32 i = 0; i < 32; i++) {
					CVector vecDir(CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(0.10f, 0.25f) + fDirectionZ);
					++nFrameGen;
					int32 currentFrame = nFrameGen & 3;
					const RwRGBA& color = nFrameGen & 1 ? color1 : color2;
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, color, nRotationSpeed, 0, currentFrame, 0);
				}
				PlayOneShotScriptObject(SCRIPT_SOUND_METAL_COLLISION, vecPos);
				break;
			}
			case DAMAGE_EFFECT_SMASH_NEWSTANDNEW1:
			case DAMAGE_EFFECT_SMASH_NEWSTANDNEW2:
			case DAMAGE_EFFECT_SMASH_NEWSTANDNEW3:
			case DAMAGE_EFFECT_SMASH_NEWSTANDNEW4:
			case DAMAGE_EFFECT_SMASH_NEWSTANDNEW5: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				CRGBA possibleColor1;
				CRGBA possibleColor2;
				switch (m_nCollisionDamageEffect) {
					case DAMAGE_EFFECT_SMASH_NEWSTANDNEW1:
						possibleColor1 = CRGBA(0xC0, 0x3E, 0xC, 0xFF);
						possibleColor2 = possibleColor1;
						break;
					case DAMAGE_EFFECT_SMASH_NEWSTANDNEW2:
						possibleColor1 = CRGBA(0xA3, 0x36, 0x21, 0xFF);
						possibleColor2 = possibleColor1;
						break;
					case DAMAGE_EFFECT_SMASH_NEWSTANDNEW3:
						possibleColor1 = CRGBA(0x12, 0x31, 0x24, 0xFF);
						possibleColor2 = possibleColor1;
						break;
					case DAMAGE_EFFECT_SMASH_NEWSTANDNEW4:
						possibleColor1 = CRGBA(0xC0, 0xC8, 0xBE, 0xFF);
						possibleColor2 = CRGBA(0x10, 0x57, 0x85, 0xFF);
						break;
					case DAMAGE_EFFECT_SMASH_NEWSTANDNEW5:
						possibleColor1 = CRGBA(0xD0, 0x94, 0x1B, 0xFF);
						possibleColor2 = possibleColor1;
						break;
				}
				for (int32 i = 0; i < 16; i++) {
					CVector vecDir(
						CGeneral::GetRandomNumberInRange(-0.35f, 0.7f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.7f),
						CGeneral::GetRandomNumberInRange(0.1f, 0.15f) + fDirectionZ
					);
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					nFrameGen++;
					int32 nCurFrame = nFrameGen & 0x3;
					CRGBA& selectedColor = nFrameGen & 0x1 ? possibleColor1 : possibleColor2;
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, selectedColor, nRotationSpeed, 0, nCurFrame, 0);
					if ((i % 7) == 0)
					{
						static CRGBA secondParticleColors[4] = {
							CRGBA(0xA0, 0x60, 0x60, 0xFF),
							CRGBA(0x60, 0xA0, 0x60, 0xFF),
							CRGBA(0x60, 0x60, 0xA0, 0xFF),
							CRGBA(0xA0, 0xA0, 0xA0, 0xFF)
						};
						vecDir *= 0.5f;
						CRGBA& secondParticleColor = secondParticleColors[nFrameGen & 3];
						int32 nSecondRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
						CParticle::AddParticle(PARTICLE_DEBRIS, vecPos, vecDir, nil, 0.1f, secondParticleColor, nSecondRotationSpeed, 0, 1, 0);
					}
				}
				PlayOneShotScriptObject(SCRIPT_SOUND_METAL_COLLISION, vecPos);
				break;
			}
			case DAMAGE_EFFECT_SMASH_BLACKBAG:
			case DAMAGE_EFFECT_SMASH_BEACHLOUNGE_WOOD:
			case DAMAGE_EFFECT_SMASH_BEACHLOUNGE_TOWEL: {
				bIsVisible = false;
				bUsesCollision = false;
				if (!GetIsStatic()) {
					RemoveFromMovingList();
				}
				SetIsStatic(true);
				bExplosionProof = true;
				SetMoveSpeed(0.0f, 0.0f, 0.0f);
				SetTurnSpeed(0.0f, 0.0f, 0.0f);
				CRGBA possibleColor1;
				CRGBA possibleColor2;
				switch (m_nCollisionDamageEffect)
				{
					case DAMAGE_EFFECT_SMASH_BLACKBAG:
						possibleColor1 = CRGBA(0, 0, 0, 0xFF);
						possibleColor2 = possibleColor1;
						break;
					case DAMAGE_EFFECT_SMASH_BEACHLOUNGE_WOOD:
						possibleColor1 = CRGBA(0x8F, 0x8A, 0x8C, 0xFF);
						possibleColor2 = CRGBA(0x73, 0x75, 0x7B, 0xFF);
						break;
					case DAMAGE_EFFECT_SMASH_BEACHLOUNGE_TOWEL:
						possibleColor1 = CRGBA(0x52, 0x92, 0x4A, 0xFF);
						possibleColor2 = CRGBA(0xCE, 0xCF, 0xCE, 0xFF);
						break;
				}
				for (int32 i = 0; i < 16; i++) {
					CVector vecDir(
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
						CGeneral::GetRandomNumberInRange(0.10f, 0.25f) + fDirectionZ
					);
					nFrameGen++;
					int32 nCurFrame = nFrameGen & 3;
					CRGBA& selectedColor = nFrameGen & 1 ? possibleColor1 : possibleColor2;
					float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
					int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
					CParticle::AddParticle(PARTICLE_CAR_DEBRIS, vecPos, vecDir, nil, fSize, selectedColor, nRotationSpeed, 0, nCurFrame, 0);
				}
				if (m_nCollisionDamageEffect == DAMAGE_EFFECT_SMASH_BLACKBAG)
				{
					PlayOneShotScriptObject(SCRIPT_SOUND_BOX_DESTROYED_2, vecPos);
				}
				else if (m_nCollisionDamageEffect == DAMAGE_EFFECT_SMASH_BEACHLOUNGE_WOOD)
				{
					PlayOneShotScriptObject(SCRIPT_SOUND_METAL_COLLISION, vecPos);
				}
				break;
			}
			default:
				DEV("Unhandled collision damage effect id: %d\n", m_nCollisionDamageEffect);
				return;
		}
	}
}

void
CObject::RefModelInfo(int32 modelId)
{
	m_nRefModelIndex = modelId;
	CModelInfo::GetModelInfo(modelId)->AddRef();
}

void
CObject::Init(void)
{
	m_type = ENTITY_TYPE_OBJECT;
	CObjectData::SetObjectData(GetModelIndex(), *this);
	m_nEndOfLifeTime = 0;
	ObjectCreatedBy = GAME_OBJECT;
	SetIsStatic(true);
	bIsPickup = false;
	bPickupObjWithMessage = false;
	bOutOfStock = false;
	bGlassCracked = false;
	bGlassBroken = false;
	bHasBeenDamaged = false;
	bUseVehicleColours = false;
	m_nRefModelIndex = -1;
	m_colour1 = 0;
	m_colour2 = 0;
	m_nBonusValue = 0;
	bIsWeapon = false;
	m_nCostValue = 0;
	m_pCollidingEntity = nil;
	CColPoint point;
	CEntity* outEntity = nil;
	const CVector& vecPos = m_matrix.GetPosition();
	if (CWorld::ProcessVerticalLine(vecPos, vecPos.z - 10.0f, point, outEntity, true, false, false, false, false, false, nil))
		m_pCurSurface = outEntity;
	else
		m_pCurSurface = nil;

	if (GetModelIndex() == MI_BUOY)
		bTouchingWater = true;

	if (CModelInfo::GetModelInfo(GetModelIndex())->GetModelType() == MITYPE_WEAPON)
		bIsWeapon = true;
	bIsStreetLight = IsLightObject(GetModelIndex());

	m_area = AREA_EVERYWHERE;
}

bool
CObject::CanBeDeleted(void)
{
	switch (ObjectCreatedBy) {
		case GAME_OBJECT:
			return true;
		case MISSION_OBJECT:
			return false;
		case TEMP_OBJECT:
			return true;
		case CUTSCENE_OBJECT:
			return false;
		case CONTROLLED_SUB_OBJECT:
			return false;
		default:
			return true;
	}
}

void
CObject::DeleteAllMissionObjects()
{
	CObjectPool* objectPool = CPools::GetObjectPool();
	for (int32 i = 0; i < objectPool->GetSize(); i++) {
		CObject* pObject = objectPool->GetSlot(i);
		if (pObject && pObject->ObjectCreatedBy == MISSION_OBJECT) {
			CWorld::Remove(pObject);
			delete pObject;
		}
	}
}

void
CObject::DeleteAllTempObjects()
{
	CObjectPool* objectPool = CPools::GetObjectPool();
	for (int32 i = 0; i < objectPool->GetSize(); i++) {
		CObject* pObject = objectPool->GetSlot(i);
		if (pObject && pObject->ObjectCreatedBy == TEMP_OBJECT) {
			CWorld::Remove(pObject);
			delete pObject;
		}
	}
}

void
CObject::DeleteAllTempObjectsInArea(CVector point, float fRadius)
{
	CObjectPool* objectPool = CPools::GetObjectPool();
	for (int32 i = 0; i < objectPool->GetSize(); i++) {
		CObject* pObject = objectPool->GetSlot(i);
		CVector dist = point - pObject->GetPosition();
		if (pObject && pObject->ObjectCreatedBy == TEMP_OBJECT && dist.MagnitudeSqr() < fRadius * fRadius) {
			CWorld::Remove(pObject);
			delete pObject;
		}
	}
}

bool
IsObjectPointerValid(CObject* pObject)
{
	if (!pObject)
		return false;
	int index = CPools::GetObjectPool()->GetJustIndex(pObject);
#ifdef FIX_BUGS
	if (index < 0 || index >= CPools::GetObjectPool()->GetSize())
#else
	if (index < 0 || index > CPools::GetObjectPool()->GetSize())
#endif
		return false;
	return pObject->bIsBIGBuilding || pObject->m_entryInfoList.first;
}