2019-06-30 19:06:55 +00:00
|
|
|
#include "common.h"
|
2020-04-17 13:31:11 +00:00
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
#include "Camera.h"
|
|
|
|
#include "Clock.h"
|
|
|
|
#include "Coronas.h"
|
2020-04-19 16:34:08 +00:00
|
|
|
#include "General.h"
|
|
|
|
#include "PathFind.h"
|
|
|
|
#include "PointLights.h"
|
|
|
|
#include "Shadows.h"
|
2020-04-10 16:36:39 +00:00
|
|
|
#include "SpecialFX.h"
|
2020-04-19 16:34:08 +00:00
|
|
|
#include "Timecycle.h"
|
|
|
|
#include "Timer.h"
|
2020-04-10 16:36:39 +00:00
|
|
|
#include "TrafficLights.h"
|
2020-04-19 16:34:08 +00:00
|
|
|
#include "Vehicle.h"
|
|
|
|
#include "Weather.h"
|
|
|
|
#include "World.h"
|
2020-04-10 16:36:39 +00:00
|
|
|
|
2020-08-30 20:47:12 +00:00
|
|
|
bool CTrafficLights::bGreenLightsCheat;
|
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
void
|
|
|
|
CTrafficLights::DisplayActualLight(CEntity *ent)
|
|
|
|
{
|
|
|
|
if(ent->GetUp().z < 0.96f || ent->bRenderDamaged)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int phase;
|
|
|
|
if(FindTrafficLightType(ent) == 1)
|
2020-10-11 09:56:33 +00:00
|
|
|
phase = LightForCars1_Visual();
|
2020-04-10 16:36:39 +00:00
|
|
|
else
|
2020-10-11 09:56:33 +00:00
|
|
|
phase = LightForCars2_Visual();
|
|
|
|
|
|
|
|
int i, m = ent->GetModelIndex();
|
|
|
|
if (MI_TRAFFICLIGHTS == m) {
|
|
|
|
CBaseModelInfo* mi = CModelInfo::GetModelInfo(ent->GetModelIndex());
|
|
|
|
float x = mi->Get2dEffect(0)->pos.x;
|
|
|
|
float yMin = mi->Get2dEffect(0)->pos.y;
|
|
|
|
float yMax = mi->Get2dEffect(0)->pos.y;
|
|
|
|
float zMin = mi->Get2dEffect(0)->pos.z;
|
|
|
|
float zMax = mi->Get2dEffect(0)->pos.z;
|
|
|
|
for (i = 1; i < 6; i++) {
|
|
|
|
assert(mi->Get2dEffect(i));
|
|
|
|
yMin = Min(yMin, mi->Get2dEffect(i)->pos.y);
|
|
|
|
yMax = Max(yMax, mi->Get2dEffect(i)->pos.y);
|
|
|
|
zMin = Min(zMin, mi->Get2dEffect(i)->pos.z);
|
|
|
|
zMax = Max(zMax, mi->Get2dEffect(i)->pos.z);
|
|
|
|
}
|
|
|
|
|
|
|
|
CVector pos1, pos2;
|
|
|
|
uint8 r, g;
|
|
|
|
int id;
|
|
|
|
switch (phase) {
|
|
|
|
case CAR_LIGHTS_GREEN:
|
|
|
|
r = 0;
|
|
|
|
g = 255;
|
|
|
|
pos1 = ent->GetMatrix() * CVector(x, yMax, zMin);
|
|
|
|
pos2 = ent->GetMatrix() * CVector(x, yMin, zMin);
|
|
|
|
id = 0;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_YELLOW:
|
|
|
|
r = 255;
|
|
|
|
g = 128;
|
|
|
|
pos1 = ent->GetMatrix() * CVector(x, yMax, (zMin + zMax) / 2.0f);
|
|
|
|
pos2 = ent->GetMatrix() * CVector(x, yMin, (zMin + zMax) / 2.0f);
|
|
|
|
id = 1;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_RED:
|
|
|
|
r = 255;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * CVector(x, yMax, zMax);
|
|
|
|
pos2 = ent->GetMatrix() * CVector(x, yMin, zMax);
|
|
|
|
id = 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
r = 0;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * CVector(x, yMax, (zMin + zMax) / 2.0f);
|
|
|
|
pos2 = ent->GetMatrix() * CVector(x, yMin, (zMin + zMax) / 2.0f);
|
|
|
|
id = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.5f)
|
|
|
|
CPointLights::AddLight(CPointLights::LIGHT_POINT,
|
|
|
|
pos1, CVector(0.0f, 0.0f, 0.0f), 8.0f,
|
|
|
|
r / 255.0f, g / 255.0f, 0 / 255.0f, CPointLights::FOG_NORMAL, true);
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.05f)
|
|
|
|
CShadows::StoreStaticShadow((uintptr)ent,
|
|
|
|
SHADOWTYPE_ADDITIVE, gpShadowExplosionTex, &pos1,
|
|
|
|
8.0f, 0.0f, 0.0f, -8.0f, 128,
|
|
|
|
r * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
g * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
0 * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
12.0f, 1.0f, 40.0f, false, 0.0f);
|
|
|
|
|
|
|
|
if (DotProduct(TheCamera.GetForward(), ent->GetForward()) < 0.0f)
|
|
|
|
CCoronas::RegisterCorona((uintptr)ent + id,
|
|
|
|
r * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
g * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
0 * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
255,
|
|
|
|
pos1, 1.75f * CTimeCycle::GetSpriteSize(), 50.0f,
|
|
|
|
CCoronas::TYPE_STAR, CCoronas::FLARE_NONE, CCoronas::REFLECTION_ON,
|
|
|
|
CCoronas::LOSCHECK_OFF, CCoronas::STREAK_OFF, 0.0f);
|
|
|
|
else
|
|
|
|
CCoronas::RegisterCorona((uintptr)ent + id + 3,
|
|
|
|
r * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
g * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
0 * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
255,
|
|
|
|
pos2, 1.75f * CTimeCycle::GetSpriteSize(), 50.0f,
|
|
|
|
CCoronas::TYPE_STAR, CCoronas::FLARE_NONE, CCoronas::REFLECTION_ON,
|
|
|
|
CCoronas::LOSCHECK_OFF, CCoronas::STREAK_OFF, 0.0f);
|
|
|
|
|
|
|
|
CBrightLights::RegisterOne(pos1, ent->GetUp(), ent->GetRight(), CVector(0.0f, 0.0f, 0.0f), id + BRIGHTLIGHT_TRAFFIC_GREEN);
|
|
|
|
CBrightLights::RegisterOne(pos2, ent->GetUp(), -ent->GetRight(), CVector(0.0f, 0.0f, 0.0f), id + BRIGHTLIGHT_TRAFFIC_GREEN);
|
2021-01-24 10:42:45 +00:00
|
|
|
|
|
|
|
// TODO(LCS): check coordinates
|
|
|
|
static const float top = -0.127f;
|
|
|
|
static const float bot = -0.539f;
|
|
|
|
static const float mid = bot + (top - bot) / 3.0f;
|
|
|
|
static const float left = 1.256f;
|
|
|
|
static const float right = 0.706f;
|
|
|
|
phase = CTrafficLights::LightForPeds();
|
|
|
|
if (phase == PED_LIGHTS_DONT_WALK) {
|
|
|
|
CVector p0(2.7f, right, top);
|
|
|
|
CVector p1(2.7f, left, top);
|
|
|
|
CVector p2(2.7f, right, mid);
|
|
|
|
CVector p3(2.7f, left, mid);
|
|
|
|
CShinyTexts::RegisterOne(ent->GetMatrix() * p0, ent->GetMatrix() * p1, ent->GetMatrix() * p2, ent->GetMatrix() * p3,
|
|
|
|
1.0f, 0.0f, 0.0f, 0.0f, 1.0f, 1.0f, 0.0f, 1.0f,
|
|
|
|
SHINYTEXT_WALK, 255, 0, 0, 60.0f);
|
|
|
|
}
|
|
|
|
else if (phase == PED_LIGHTS_WALK || CTimer::GetTimeInMilliseconds() & 0x100) {
|
|
|
|
CVector p0(2.7f, right, mid);
|
|
|
|
CVector p1(2.7f, left, mid);
|
|
|
|
CVector p2(2.7f, right, bot);
|
|
|
|
CVector p3(2.7f, left, bot);
|
|
|
|
CShinyTexts::RegisterOne(ent->GetMatrix() * p0, ent->GetMatrix() * p1, ent->GetMatrix() * p2, ent->GetMatrix() * p3,
|
|
|
|
1.0f, 0.5f, 0.0f, 0.5f, 1.0f, 1.0f, 0.0f, 1.0f,
|
|
|
|
SHINYTEXT_WALK, 255, 255, 255, 60.0f);
|
|
|
|
}
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
2020-10-11 09:56:33 +00:00
|
|
|
else if (MI_TRAFFICLIGHTS_VERTICAL == m) {
|
|
|
|
CBaseModelInfo* mi = CModelInfo::GetModelInfo(ent->GetModelIndex());
|
|
|
|
float x = mi->Get2dEffect(0)->pos.x;
|
|
|
|
float yMin = mi->Get2dEffect(0)->pos.y;
|
|
|
|
float yMax = mi->Get2dEffect(0)->pos.y;
|
|
|
|
float zMin = mi->Get2dEffect(0)->pos.z;
|
|
|
|
float zMax = mi->Get2dEffect(0)->pos.z;
|
|
|
|
for (i = 1; i < 6; i++) {
|
|
|
|
assert(mi->Get2dEffect(i));
|
|
|
|
yMin = Min(yMin, mi->Get2dEffect(i)->pos.y);
|
|
|
|
yMax = Max(yMax, mi->Get2dEffect(i)->pos.y);
|
|
|
|
zMin = Min(zMin, mi->Get2dEffect(i)->pos.z);
|
|
|
|
zMax = Max(zMax, mi->Get2dEffect(i)->pos.z);
|
|
|
|
}
|
|
|
|
|
|
|
|
CVector pos1;
|
|
|
|
uint8 r, g;
|
|
|
|
int id;
|
|
|
|
switch (phase) {
|
|
|
|
case CAR_LIGHTS_GREEN:
|
|
|
|
r = 0;
|
|
|
|
g = 255;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(2)->pos;
|
|
|
|
id = 0;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_YELLOW:
|
|
|
|
r = 255;
|
|
|
|
g = 128;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(1)->pos;
|
|
|
|
id = 1;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_RED:
|
|
|
|
r = 255;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(0)->pos;
|
|
|
|
id = 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
r = 0;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(1)->pos;
|
|
|
|
id = -1;
|
|
|
|
break;
|
|
|
|
}
|
2020-04-10 16:36:39 +00:00
|
|
|
|
2020-10-11 09:56:33 +00:00
|
|
|
CBrightLights::RegisterOne(pos1, ent->GetUp(), ent->GetRight(), CVector(0.0f, 0.0f, 0.0f), id + BRIGHTLIGHT_TRAFFIC_GREEN);
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.5f)
|
|
|
|
CPointLights::AddLight(CPointLights::LIGHT_POINT,
|
|
|
|
pos1, CVector(0.0f, 0.0f, 0.0f), 8.0f,
|
|
|
|
r / 255.0f, g / 255.0f, 0 / 255.0f, CPointLights::FOG_NORMAL, true);
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.05f)
|
|
|
|
CShadows::StoreStaticShadow((uintptr)ent,
|
|
|
|
SHADOWTYPE_ADDITIVE, gpShadowExplosionTex, &pos1,
|
|
|
|
8.0f, 0.0f, 0.0f, -8.0f, 128,
|
|
|
|
r * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
g * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
0 * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
12.0f, 1.0f, 40.0f, false, 0.0f);
|
|
|
|
|
|
|
|
if (DotProduct(TheCamera.GetForward(), ent->GetForward()) < 0.0f)
|
|
|
|
CCoronas::RegisterCorona((uintptr)ent + id,
|
|
|
|
r * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
g * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
0 * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
255,
|
|
|
|
pos1, 1.75f * CTimeCycle::GetSpriteSize(), 50.0f,
|
|
|
|
CCoronas::TYPE_STAR, CCoronas::FLARE_NONE, CCoronas::REFLECTION_ON,
|
|
|
|
CCoronas::LOSCHECK_OFF, CCoronas::STREAK_OFF, 0.0f);
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
2020-10-11 09:56:33 +00:00
|
|
|
else if (MI_TRAFFICLIGHTS_MIAMI == m || MI_TRAFFICLIGHTS_TWOVERTICAL == m) {
|
|
|
|
CBaseModelInfo* mi = CModelInfo::GetModelInfo(ent->GetModelIndex());
|
|
|
|
CVector pos1, pos2;
|
|
|
|
uint8 r, g;
|
|
|
|
int id;
|
|
|
|
if (MI_TRAFFICLIGHTS_MIAMI == m) {
|
|
|
|
switch (phase) {
|
|
|
|
case CAR_LIGHTS_GREEN:
|
|
|
|
r = 0;
|
|
|
|
g = 255;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(4)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(5)->pos;
|
|
|
|
id = 0;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_YELLOW:
|
|
|
|
r = 255;
|
|
|
|
g = 128;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(2)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(3)->pos;
|
|
|
|
id = 1;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_RED:
|
|
|
|
r = 255;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(0)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(1)->pos;
|
|
|
|
id = 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
r = 0;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(2)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(3)->pos;
|
|
|
|
id = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
switch (phase) {
|
|
|
|
case CAR_LIGHTS_GREEN:
|
|
|
|
r = 0;
|
|
|
|
g = 255;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(2)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(5)->pos;
|
|
|
|
id = 0;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_YELLOW:
|
|
|
|
r = 255;
|
|
|
|
g = 128;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(1)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(4)->pos;
|
|
|
|
id = 1;
|
|
|
|
break;
|
|
|
|
case CAR_LIGHTS_RED:
|
|
|
|
r = 255;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(0)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(3)->pos;
|
|
|
|
id = 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
r = 0;
|
|
|
|
g = 0;
|
|
|
|
pos1 = ent->GetMatrix() * mi->Get2dEffect(1)->pos;
|
|
|
|
pos2 = ent->GetMatrix() * mi->Get2dEffect(4)->pos;
|
|
|
|
id = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-04-10 16:36:39 +00:00
|
|
|
|
2020-10-11 09:56:33 +00:00
|
|
|
CVector pos = (pos1 + pos2) / 2;
|
|
|
|
if (id >= 0) {
|
|
|
|
CBrightLights::RegisterOne(pos1, ent->GetUp(), ent->GetRight(), CVector(0.0f, 0.0f, 0.0f), id + BRIGHTLIGHT_TRAFFIC_GREEN);
|
2020-10-11 15:14:35 +00:00
|
|
|
CBrightLights::RegisterOne(pos2, ent->GetUp(), ent->GetRight(), CVector(0.0f, 0.0f, 0.0f), id + BRIGHTLIGHT_TRAFFIC_GREEN);
|
2020-10-11 09:56:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.5f)
|
|
|
|
CPointLights::AddLight(CPointLights::LIGHT_POINT,
|
|
|
|
pos, CVector(0.0f, 0.0f, 0.0f), 8.0f,
|
|
|
|
r / 255.0f, g / 255.0f, 0 / 255.0f, CPointLights::FOG_NORMAL, true);
|
|
|
|
|
|
|
|
if (CWeather::TrafficLightBrightness > 0.05f)
|
|
|
|
CShadows::StoreStaticShadow((uintptr)ent,
|
|
|
|
SHADOWTYPE_ADDITIVE, gpShadowExplosionTex, &pos,
|
|
|
|
8.0f, 0.0f, 0.0f, -8.0f, 128,
|
|
|
|
r * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
g * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
0 * CTimeCycle::GetLightOnGroundBrightness() * CWeather::TrafficLightBrightness / 8.0f,
|
|
|
|
12.0f, 1.0f, 40.0f, false, 0.0f);
|
|
|
|
|
|
|
|
if (id >= 0) {
|
|
|
|
if (DotProduct(TheCamera.GetForward(), ent->GetForward()) < 0.0f)
|
|
|
|
CCoronas::RegisterCorona((uintptr)ent + id,
|
|
|
|
r * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
g * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
0 * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
255,
|
|
|
|
pos1, 1.75f * CTimeCycle::GetSpriteSize(), 50.0f,
|
|
|
|
CCoronas::TYPE_STAR, CCoronas::FLARE_NONE, CCoronas::REFLECTION_ON,
|
|
|
|
CCoronas::LOSCHECK_OFF, CCoronas::STREAK_OFF, 0.0f);
|
|
|
|
else
|
|
|
|
CCoronas::RegisterCorona((uintptr)ent + id,
|
|
|
|
r * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
g * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
0 * CTimeCycle::GetSpriteBrightness() * 0.7f,
|
|
|
|
255,
|
|
|
|
pos2, 1.75f * CTimeCycle::GetSpriteSize(), 50.0f,
|
|
|
|
CCoronas::TYPE_STAR, CCoronas::FLARE_NONE, CCoronas::REFLECTION_ON,
|
|
|
|
CCoronas::LOSCHECK_OFF, CCoronas::STREAK_OFF, 0.0f);
|
|
|
|
}
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
2020-10-11 09:56:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DoesLineSegmentIntersect(float l1x1, float l1y1, float l1x2, float l1y2, float l2x1, float l2y1, float l2x2, float l2y2)
|
|
|
|
{
|
|
|
|
return ((l2y2 - l1y1) * (l1x2 - l1x1) + (l1x1 - l2x2) * (l1y2 - l1y1)) *
|
|
|
|
((l2y1 - l1y1) * (l1x2 - l1x1) + (l1x1 - l2x1) * (l1y2 - l1y1)) <= 0.0f &&
|
|
|
|
((l1y2 - l2y1) * (l2x2 - l2x1) + (l2y2 - l2y1) * (l2x1 - l1x2)) *
|
|
|
|
((l1y1 - l2y1) * (l2x2 - l2x1) + (l2y2 - l2y1) * (l2x1 - l1x1)) <= 0.0f;
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CTrafficLights::ScanForLightsOnMap(void)
|
|
|
|
{
|
|
|
|
int x, y;
|
2020-05-13 14:33:50 +00:00
|
|
|
int i, j, k, l;
|
2020-04-10 16:36:39 +00:00
|
|
|
CPtrNode *node;
|
|
|
|
|
|
|
|
for(x = 0; x < NUMSECTORS_X; x++)
|
|
|
|
for(y = 0; y < NUMSECTORS_Y; y++){
|
|
|
|
CPtrList &list = CWorld::GetSector(x, y)->m_lists[ENTITYLIST_DUMMIES];
|
|
|
|
for(node = list.first; node; node = node->next){
|
|
|
|
CEntity *light = (CEntity*)node->item;
|
2020-05-20 17:10:05 +00:00
|
|
|
if (!IsTrafficLight(light->GetModelIndex()))
|
2020-04-10 16:36:39 +00:00
|
|
|
continue;
|
|
|
|
|
2020-10-11 09:56:33 +00:00
|
|
|
CVector pos1 = light->GetMatrix() * CVector(17.0f, 0.0f, 0.0f);
|
|
|
|
CVector pos2 = light->GetMatrix() * CVector(-15.0f, 0.0f, 0.0f);
|
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
// Check cars
|
2020-10-11 09:56:33 +00:00
|
|
|
for(i = 0; i < ThePaths.m_numCarPathNodes; i++){
|
|
|
|
if ((ThePaths.m_pathNodes[i].GetPosition() - pos1).MagnitudeSqr() >= SQR(100.0f))
|
|
|
|
continue;
|
|
|
|
for (j = 0; j < ThePaths.m_pathNodes[i].numLinks; j++){
|
|
|
|
int con = ThePaths.ConnectedNode(ThePaths.m_pathNodes[i].firstLink + j);
|
|
|
|
if (i < con) {
|
|
|
|
CVector i_pos = ThePaths.m_pathNodes[i].GetPosition();
|
|
|
|
CVector con_pos = ThePaths.m_pathNodes[con].GetPosition();
|
|
|
|
if (Abs(pos1.z - (i_pos.z + con_pos.z) / 2) < 10.0f &&
|
|
|
|
DoesLineSegmentIntersect(pos1.x, pos1.y, pos2.x, pos2.y, i_pos.x, i_pos.y, con_pos.x, con_pos.y)) {
|
|
|
|
//debug("Setting up light: nodes %f %f %f - %f %f %f, light %f %f %f - %f %f %f\n", i_pos.x, i_pos.y, i_pos.z, con_pos.x, con_pos.y, con_pos.z, pos1.x, pos1.y, pos1.z, pos2.x, pos2.y, pos2.z);
|
|
|
|
int link = ThePaths.m_carPathConnections[ThePaths.m_pathNodes[i].firstLink + j];
|
|
|
|
ThePaths.m_carPathLinks[link].trafficLightType = FindTrafficLightType(light);
|
|
|
|
if (ThePaths.m_pathNodes[i].numLinks > ThePaths.m_pathNodes[con].numLinks)
|
|
|
|
con = i;
|
|
|
|
if (ThePaths.m_carPathLinks[link].pathNodeIndex != con)
|
|
|
|
ThePaths.m_carPathLinks[link].trafficLightDirection = true;
|
|
|
|
}
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check peds
|
|
|
|
for(i = ThePaths.m_numCarPathNodes; i < ThePaths.m_numPathNodes; i++){
|
|
|
|
float dist1, dist2;
|
2020-05-03 13:57:57 +00:00
|
|
|
dist1 = Abs(ThePaths.m_pathNodes[i].GetX() - light->GetPosition().x) +
|
|
|
|
Abs(ThePaths.m_pathNodes[i].GetY() - light->GetPosition().y);
|
2020-04-10 16:36:39 +00:00
|
|
|
if(dist1 < 50.0f){
|
|
|
|
for(l = 0; l < ThePaths.m_pathNodes[i].numLinks; l++){
|
|
|
|
j = ThePaths.m_pathNodes[i].firstLink + l;
|
2020-05-03 13:57:57 +00:00
|
|
|
if(ThePaths.ConnectionCrossesRoad(j)){
|
2020-05-13 14:33:50 +00:00
|
|
|
k = ThePaths.ConnectedNode(j);
|
|
|
|
dist2 = Abs(ThePaths.m_pathNodes[k].GetX() - light->GetPosition().x) +
|
|
|
|
Abs(ThePaths.m_pathNodes[k].GetY() - light->GetPosition().y);
|
2020-04-10 16:36:39 +00:00
|
|
|
if(dist1 < 15.0f || dist2 < 15.0f)
|
2020-05-03 13:57:57 +00:00
|
|
|
ThePaths.ConnectionSetTrafficLight(j);
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
CTrafficLights::ShouldCarStopForLight(CVehicle *vehicle, bool alwaysStop)
|
|
|
|
{
|
|
|
|
int node, type;
|
2020-10-11 09:56:33 +00:00
|
|
|
bool direction;
|
2019-06-30 19:06:55 +00:00
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
node = vehicle->AutoPilot.m_nNextPathNodeInfo;
|
|
|
|
type = ThePaths.m_carPathLinks[node].trafficLightType;
|
2020-10-11 09:56:33 +00:00
|
|
|
direction = ThePaths.m_carPathLinks[node].trafficLightDirection;
|
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
if(type){
|
2020-10-11 09:56:33 +00:00
|
|
|
if((direction || ThePaths.m_carPathLinks[node].pathNodeIndex == vehicle->AutoPilot.m_nNextRouteNode) &&
|
|
|
|
(!direction || ThePaths.m_carPathLinks[node].pathNodeIndex != vehicle->AutoPilot.m_nNextRouteNode))
|
2020-04-10 16:36:39 +00:00
|
|
|
if(alwaysStop ||
|
2020-10-11 09:56:33 +00:00
|
|
|
type == 1 && LightForCars1() != CAR_LIGHTS_GREEN ||
|
|
|
|
type == 2 && LightForCars2() != CAR_LIGHTS_GREEN){
|
2020-05-03 13:57:57 +00:00
|
|
|
float dist = DotProduct2D(CVector2D(vehicle->GetPosition()) - ThePaths.m_carPathLinks[node].GetPosition(),
|
|
|
|
ThePaths.m_carPathLinks[node].GetDirection());
|
2020-04-10 16:36:39 +00:00
|
|
|
if(vehicle->AutoPilot.m_nNextDirection == -1){
|
|
|
|
if(dist > 0.0f && dist < 8.0f)
|
|
|
|
return true;
|
|
|
|
}else{
|
|
|
|
if(dist < 0.0f && dist > -8.0f)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
node = vehicle->AutoPilot.m_nCurrentPathNodeInfo;
|
|
|
|
type = ThePaths.m_carPathLinks[node].trafficLightType;
|
2020-10-11 09:56:33 +00:00
|
|
|
direction = ThePaths.m_carPathLinks[node].trafficLightDirection;
|
2020-04-10 16:36:39 +00:00
|
|
|
if(type){
|
2020-10-11 09:56:33 +00:00
|
|
|
if((direction || ThePaths.m_carPathLinks[node].pathNodeIndex == vehicle->AutoPilot.m_nCurrentRouteNode) &&
|
|
|
|
(!direction || ThePaths.m_carPathLinks[node].pathNodeIndex != vehicle->AutoPilot.m_nCurrentRouteNode))
|
2020-04-10 16:36:39 +00:00
|
|
|
if(alwaysStop ||
|
2020-10-11 09:56:33 +00:00
|
|
|
type == 1 && LightForCars1() != CAR_LIGHTS_GREEN ||
|
|
|
|
type == 2 && LightForCars2() != CAR_LIGHTS_GREEN){
|
2020-05-03 13:57:57 +00:00
|
|
|
float dist = DotProduct2D(CVector2D(vehicle->GetPosition()) - ThePaths.m_carPathLinks[node].GetPosition(),
|
|
|
|
ThePaths.m_carPathLinks[node].GetDirection());
|
2020-04-10 16:36:39 +00:00
|
|
|
if(vehicle->AutoPilot.m_nCurrentDirection == -1){
|
|
|
|
if(dist > 0.0f && dist < 8.0f)
|
|
|
|
return true;
|
|
|
|
}else{
|
|
|
|
if(dist < 0.0f && dist > -8.0f)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-30 13:45:45 +00:00
|
|
|
if(vehicle->GetStatus() == STATUS_PHYSICS){
|
2020-04-10 16:36:39 +00:00
|
|
|
node = vehicle->AutoPilot.m_nPreviousPathNodeInfo;
|
|
|
|
type = ThePaths.m_carPathLinks[node].trafficLightType;
|
2020-10-11 09:56:33 +00:00
|
|
|
direction = ThePaths.m_carPathLinks[node].trafficLightDirection;
|
2020-04-10 16:36:39 +00:00
|
|
|
if(type){
|
2020-10-11 09:56:33 +00:00
|
|
|
if((direction || ThePaths.m_carPathLinks[node].pathNodeIndex == vehicle->AutoPilot.m_nPrevRouteNode) &&
|
|
|
|
(!direction || ThePaths.m_carPathLinks[node].pathNodeIndex != vehicle->AutoPilot.m_nPrevRouteNode))
|
2020-04-10 16:36:39 +00:00
|
|
|
if(alwaysStop ||
|
2020-10-11 09:56:33 +00:00
|
|
|
type == 1 && LightForCars1() != CAR_LIGHTS_GREEN ||
|
|
|
|
type == 2 && LightForCars2() != CAR_LIGHTS_GREEN){
|
2020-05-03 13:57:57 +00:00
|
|
|
float dist = DotProduct2D(CVector2D(vehicle->GetPosition()) - ThePaths.m_carPathLinks[node].GetPosition(),
|
|
|
|
ThePaths.m_carPathLinks[node].GetDirection());
|
2020-04-10 16:36:39 +00:00
|
|
|
if(vehicle->AutoPilot.m_nPreviousDirection == -1){
|
|
|
|
if(dist > 0.0f && dist < 6.0f)
|
|
|
|
return true;
|
|
|
|
}else{
|
|
|
|
if(dist < 0.0f && dist > -6.0f)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
CTrafficLights::ShouldCarStopForBridge(CVehicle *vehicle)
|
|
|
|
{
|
2020-05-03 15:28:54 +00:00
|
|
|
#ifdef GTA_BRIDGE
|
2020-04-10 16:36:39 +00:00
|
|
|
return ThePaths.m_carPathLinks[vehicle->AutoPilot.m_nNextPathNodeInfo].bBridgeLights &&
|
|
|
|
!ThePaths.m_carPathLinks[vehicle->AutoPilot.m_nCurrentPathNodeInfo].bBridgeLights;
|
2020-05-03 15:28:54 +00:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2020-04-10 16:36:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
CTrafficLights::FindTrafficLightType(CEntity *light)
|
|
|
|
{
|
|
|
|
float orientation = RADTODEG(CGeneral::GetATanOfXY(light->GetForward().x, light->GetForward().y));
|
|
|
|
if((orientation > 60.0f && orientation < 60.0f + 90.0f) ||
|
|
|
|
(orientation > 240.0f && orientation < 240.0f + 90.0f))
|
|
|
|
return 1;
|
|
|
|
return 2;
|
|
|
|
}
|
2019-08-08 00:21:38 +00:00
|
|
|
|
|
|
|
uint8
|
|
|
|
CTrafficLights::LightForPeds(void)
|
|
|
|
{
|
2020-04-10 16:36:39 +00:00
|
|
|
uint32 period = CTimer::GetTimeInMilliseconds() % 16384;
|
2019-08-08 00:21:38 +00:00
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
if(period < 12000)
|
|
|
|
return PED_LIGHTS_DONT_WALK;
|
|
|
|
else if(period < 16384 - 1000)
|
2019-08-08 00:21:38 +00:00
|
|
|
return PED_LIGHTS_WALK;
|
|
|
|
else
|
2020-04-10 16:36:39 +00:00
|
|
|
return PED_LIGHTS_WALK_BLINK;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8
|
|
|
|
CTrafficLights::LightForCars1(void)
|
|
|
|
{
|
2020-08-30 20:47:12 +00:00
|
|
|
if (CWeather::Wind > 1.1f)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
|
|
|
|
if (bGreenLightsCheat)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
uint32 period = CTimer::GetTimeInMilliseconds() % 16384;
|
|
|
|
|
|
|
|
if(period < 5000)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
else if(period < 5000 + 1000)
|
|
|
|
return CAR_LIGHTS_YELLOW;
|
|
|
|
else
|
|
|
|
return CAR_LIGHTS_RED;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8
|
|
|
|
CTrafficLights::LightForCars2(void)
|
|
|
|
{
|
2020-08-30 20:47:12 +00:00
|
|
|
if (CWeather::Wind > 1.1f)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
|
|
|
|
if (bGreenLightsCheat)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
|
2020-04-10 16:36:39 +00:00
|
|
|
uint32 period = CTimer::GetTimeInMilliseconds() % 16384;
|
|
|
|
|
|
|
|
if(period < 6000)
|
|
|
|
return CAR_LIGHTS_RED;
|
|
|
|
else if(period < 12000 - 1000)
|
|
|
|
return CAR_LIGHTS_GREEN;
|
|
|
|
else if(period < 12000)
|
|
|
|
return CAR_LIGHTS_YELLOW;
|
|
|
|
else
|
|
|
|
return CAR_LIGHTS_RED;
|
|
|
|
}
|
2020-10-11 09:56:33 +00:00
|
|
|
|
|
|
|
uint8
|
|
|
|
CTrafficLights::LightForCars1_Visual(void)
|
|
|
|
{
|
|
|
|
if (CWeather::Wind <= 1.1f)
|
|
|
|
return LightForCars1();
|
|
|
|
return (CTimer::GetTimeInMilliseconds() & 0x400 ? CAR_LIGHTS_NONE : CAR_LIGHTS_YELLOW);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8
|
|
|
|
CTrafficLights::LightForCars2_Visual(void)
|
|
|
|
{
|
|
|
|
if (CWeather::Wind <= 1.1f)
|
|
|
|
return LightForCars2();
|
|
|
|
return (CTimer::GetTimeInMilliseconds() & 0x400 ? CAR_LIGHTS_NONE : CAR_LIGHTS_YELLOW);
|
|
|
|
}
|