mirror of
https://github.com/GTAmodding/re3.git
synced 2025-01-18 19:16:36 +00:00
ccbbde549e
garages revision + fixes
836 lines
28 KiB
C++
836 lines
28 KiB
C++
#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"
|
|
|
|
#define BEACHBALL_MAX_SCORE 250
|
|
// the proportion of the ball speed compared to the player speed when it hits the player
|
|
#define BEACHBALL_SPEED_PROPORTION 0.4f
|
|
|
|
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;
|
|
}
|
|
int16 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) {
|
|
float fTimeStep = Pow(0.95f, CTimer::GetTimeStep());
|
|
float fPreviousVecSpeedMag = m_vecMoveSpeed.Magnitude2D();
|
|
m_vecMoveSpeed.x *= fTimeStep;
|
|
m_vecMoveSpeed.y *= fTimeStep;
|
|
m_vecMoveSpeed.z += fPreviousVecSpeedMag - m_vecMoveSpeed.Magnitude2D();
|
|
if (!FindPlayerVehicle()) {
|
|
CVector distance;
|
|
distance.x = FindPlayerCoors().x - GetPosition().x;
|
|
distance.y = FindPlayerCoors().y - GetPosition().y;
|
|
distance.z = FindPlayerCoors().z - GetPosition().z;
|
|
if (distance.z > 0.0 && distance.z < 1.5f && distance.Magnitude2D() < 1.0f) {
|
|
CVector playerSpeed = FindPlayerSpeed();
|
|
if (fPreviousVecSpeedMag < 0.05f && playerSpeed.Magnitude() > 0.1f) {
|
|
playerSpeed.z = 0.0f;
|
|
playerSpeed.Normalise();
|
|
playerSpeed.z = 0.3f;
|
|
m_vecMoveSpeed = CVector(
|
|
playerSpeed.x * BEACHBALL_SPEED_PROPORTION,
|
|
playerSpeed.y * BEACHBALL_SPEED_PROPORTION,
|
|
0.3f * BEACHBALL_SPEED_PROPORTION
|
|
);
|
|
PlayOneShotScriptObject(SCRIPT_SOUND_HIT_BALL, GetPosition());
|
|
m_vecTurnSpeed += CVector(
|
|
((CGeneral::GetRandomNumber() % 16) - 7) / 10.0f,
|
|
((CGeneral::GetRandomNumber() % 16) - 7) / 10.0f,
|
|
0.0f);
|
|
if (m_nBeachballBounces > 0) {
|
|
m_nBeachballBounces++;
|
|
}
|
|
if (m_nBeachballBounces > 0) {
|
|
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 && m_vecMoveSpeed.z < 0.0f && distance.Magnitude2D() < 0.9f) {
|
|
m_vecMoveSpeed.x += (CGeneral::GetRandomNumber() % 8 - 3) / 100.0f;
|
|
m_vecMoveSpeed.y += (CGeneral::GetRandomNumber() % 8 - 3) / 100.0f;
|
|
m_vecMoveSpeed.z = Max(m_vecMoveSpeed.z + 0.3f, 0.2f);
|
|
PlayOneShotScriptObject(SCRIPT_SOUND_HIT_BALL, GetPosition());
|
|
m_vecTurnSpeed.x += (CGeneral::GetRandomNumber() % 16 - 7) / 10.0f;
|
|
m_vecTurnSpeed.y += (CGeneral::GetRandomNumber() % 16 - 7) / 10.0f;
|
|
m_nBeachballBounces++;
|
|
m_nBeachballBounces = Min(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);
|
|
}
|
|
}
|
|
}
|
|
if (bIsBIGBuilding) {
|
|
bIsInSafePosition = true;
|
|
}
|
|
}
|
|
|
|
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;
|
|
CVector min = 0.85f * GetColModel()->boundingBox.min;
|
|
CVector max = 0.85f * GetColModel()->boundingBox.max;
|
|
min.z = max.z;
|
|
min = GetMatrix() * min;
|
|
max = GetMatrix() * max;
|
|
CVector temp = (max - min) * 0.02f;
|
|
for (int32 i = 0; i < 50; i++) {
|
|
CVector vecDir = CVector(
|
|
CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
|
|
CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
|
|
CGeneral::GetRandomNumberInRange(0.10f, 0.25f)
|
|
);
|
|
++nFrameGen;
|
|
int32 currentFrame = nFrameGen & 3;
|
|
CVector pos = min + temp * (float)i;
|
|
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, vecDir, nil, fSize, color, nRotationSpeed, 0, currentFrame, 0);
|
|
}
|
|
PlayOneShotScriptObject(SCRIPT_SOUND_METAL_COLLISION, min);
|
|
break;
|
|
}
|
|
case DAMAGE_EFFECT_CHANGE_THEN_SMASH: {
|
|
if (!bRenderDamaged) {
|
|
bRenderDamaged = true;
|
|
return;
|
|
}
|
|
// fall through
|
|
}
|
|
case DAMAGE_EFFECT_SMASH_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);
|
|
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 = CVector(
|
|
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 randomColor = color;
|
|
switch (m_nCollisionDamageEffect) {
|
|
case DAMAGE_EFFECT_SMASH_CARDBOARD_COMPLETELY: {
|
|
float fRandom = CGeneral::GetRandomNumberInRange(0.01f, 1.0f);
|
|
randomColor.red *= fRandom;
|
|
randomColor.green *= fRandom;
|
|
randomColor.blue *= fRandom;
|
|
break;
|
|
}
|
|
case DAMAGE_EFFECT_SMASH_YELLOW_TARGET_COMPLETELY: {
|
|
randomColor.red = 0xff;
|
|
randomColor.green = 0xfc;
|
|
break;
|
|
}
|
|
}
|
|
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 = CVector(
|
|
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 = CVector(
|
|
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 = CVector(
|
|
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.35f),
|
|
CGeneral::GetRandomNumberInRange(-0.35f, 0.35f),
|
|
CGeneral::GetRandomNumberInRange(0.10f, 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)) {
|
|
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_VEGPALM:
|
|
{
|
|
static RwRGBA primaryColor1 = { 0x39, 0x4D, 0x29, 0xff };
|
|
static RwRGBA primaryColor2 = { 0x94, 0x7D, 0x73, 0xff };
|
|
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);
|
|
float fRadius = GetColModel()->boundingSphere.radius;
|
|
for (int32 i = 0; i < 32; i++) {
|
|
CVector particleDir = CVector(
|
|
CGeneral::GetRandomNumberInRange(-0.25f, 0.25f),
|
|
CGeneral::GetRandomNumberInRange(-0.25f, 0.25f),
|
|
CGeneral::GetRandomNumberInRange(-0.05f, 0.05f) + fDirectionZ
|
|
);
|
|
CVector particlePos = vecPos;
|
|
particlePos.z += CGeneral::GetRandomNumberInRange(0.0f, 1.0f) * fRadius;
|
|
++nFrameGen;
|
|
int32 nRotationSpeed = CGeneral::GetRandomNumberInRange(-40, 40);
|
|
int32 nCurFrame = nFrameGen & 0x3;
|
|
float fSize = CGeneral::GetRandomNumberInRange(0.02f, 0.20f);
|
|
RwRGBA& particleColor = nFrameGen & 1 ? primaryColor1 : primaryColor2;
|
|
CParticle::AddParticle(PARTICLE_CAR_DEBRIS, particlePos, particleDir, nil, fSize, particleColor, nRotationSpeed, 0, nCurFrame, 0);
|
|
if ((i % 7) == 0) {
|
|
static RwRGBA secondaryColor = { 0x9A, 0x99, 0x99, 0x3E };
|
|
CParticle::AddParticle(PARTICLE_DEBRIS, particlePos, particleDir, nil, 0.3, secondaryColor, nRotationSpeed, 0, 0, 0);
|
|
}
|
|
}
|
|
PlayOneShotScriptObject(SCRIPT_SOUND_BOX_DESTROYED_2, 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 & 0x3;
|
|
CRGBA &selectedColor = nFrameGen & 0x1 ? 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);
|
|
if (pObject && pObject->ObjectCreatedBy == TEMP_OBJECT && (point - pObject->GetPosition()).MagnitudeSqr() < SQR(fRadius)) {
|
|
CWorld::Remove(pObject);
|
|
delete pObject;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
IsObjectPointerValid(CObject *pObject)
|
|
{
|
|
if (!pObject)
|
|
return false;
|
|
int index = CPools::GetObjectPool()->GetJustIndex_NoFreeAssert(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;
|
|
}
|