2019-08-03 16:01:13 +03:00
|
|
|
#include "common.h"
|
2020-04-17 16:31:11 +03:00
|
|
|
|
2019-08-03 16:01:13 +03:00
|
|
|
#include "Restart.h"
|
2019-10-10 12:20:53 +03:00
|
|
|
#include "Zones.h"
|
|
|
|
#include "PathFind.h"
|
2019-08-03 16:01:13 +03:00
|
|
|
|
2020-04-17 08:54:14 +03:00
|
|
|
uint8 CRestart::OverrideHospitalLevel;
|
|
|
|
uint8 CRestart::OverridePoliceStationLevel;
|
|
|
|
bool CRestart::bFadeInAfterNextArrest;
|
|
|
|
bool CRestart::bFadeInAfterNextDeath;
|
2019-10-08 00:29:30 +03:00
|
|
|
|
2020-04-17 08:54:14 +03:00
|
|
|
bool CRestart::bOverrideRestart;
|
|
|
|
CVector CRestart::OverridePosition;
|
|
|
|
float CRestart::OverrideHeading;
|
2019-10-10 12:20:53 +03:00
|
|
|
|
2020-04-17 08:54:14 +03:00
|
|
|
CVector CRestart::HospitalRestartPoints[NUM_RESTART_POINTS];
|
|
|
|
float CRestart::HospitalRestartHeadings[NUM_RESTART_POINTS];
|
|
|
|
uint16 CRestart::NumberOfHospitalRestarts;
|
2019-10-10 12:20:53 +03:00
|
|
|
|
2020-04-17 08:54:14 +03:00
|
|
|
CVector CRestart::PoliceRestartPoints[NUM_RESTART_POINTS];
|
|
|
|
float CRestart::PoliceRestartHeadings[NUM_RESTART_POINTS];
|
|
|
|
uint16 CRestart::NumberOfPoliceRestarts;
|
2019-10-10 12:20:53 +03:00
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::Initialise()
|
|
|
|
{
|
2020-07-13 17:43:09 +03:00
|
|
|
OverridePoliceStationLevel = LEVEL_GENERIC;
|
|
|
|
OverrideHospitalLevel = LEVEL_GENERIC;
|
2020-04-09 20:50:24 +02:00
|
|
|
bFadeInAfterNextArrest = true;
|
|
|
|
bFadeInAfterNextDeath = true;
|
|
|
|
OverrideHeading = 0.0f;
|
|
|
|
OverridePosition = CVector(0.0f, 0.0f, 0.0f);
|
|
|
|
bOverrideRestart = false;
|
|
|
|
NumberOfPoliceRestarts = 0;
|
|
|
|
NumberOfHospitalRestarts = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_RESTART_POINTS; i++) {
|
|
|
|
HospitalRestartPoints[i] = CVector(0.0f, 0.0f, 0.0f);
|
|
|
|
HospitalRestartHeadings[i] = 0.0f;
|
|
|
|
PoliceRestartPoints[i] = CVector(0.0f, 0.0f, 0.0f);
|
|
|
|
PoliceRestartHeadings[i] = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CRestart::AddHospitalRestartPoint(const CVector &pos, float heading)
|
|
|
|
{
|
|
|
|
HospitalRestartPoints[NumberOfHospitalRestarts] = pos;
|
|
|
|
HospitalRestartHeadings[NumberOfHospitalRestarts++] = heading;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CRestart::AddPoliceRestartPoint(const CVector &pos, float heading)
|
|
|
|
{
|
|
|
|
PoliceRestartPoints[NumberOfPoliceRestarts] = pos;
|
|
|
|
PoliceRestartHeadings[NumberOfPoliceRestarts++] = heading;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CRestart::OverrideNextRestart(const CVector &pos, float heading)
|
|
|
|
{
|
|
|
|
bOverrideRestart = true;
|
|
|
|
OverridePosition = pos;
|
|
|
|
OverrideHeading = heading;
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::CancelOverrideRestart()
|
|
|
|
{
|
|
|
|
bOverrideRestart = false;
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::FindClosestHospitalRestartPoint(const CVector &pos, CVector *outPos, float *outHeading)
|
|
|
|
{
|
|
|
|
if (bOverrideRestart) {
|
|
|
|
*outPos = OverridePosition;
|
|
|
|
*outHeading = OverrideHeading;
|
|
|
|
CancelOverrideRestart();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-06 12:23:57 +02:00
|
|
|
eLevelName curlevel = CTheZones::GetLevelFromPosition(&pos);
|
2020-04-09 20:50:24 +02:00
|
|
|
float fMinDist = 16000000.0f;
|
|
|
|
int closestPoint = NUM_RESTART_POINTS;
|
|
|
|
|
|
|
|
// find closest point on this level
|
|
|
|
for (int i = 0; i < NumberOfHospitalRestarts; i++) {
|
2020-07-13 18:10:17 +03:00
|
|
|
if (CTheZones::GetLevelFromPosition(&HospitalRestartPoints[i]) == (OverrideHospitalLevel != LEVEL_GENERIC ? OverrideHospitalLevel : curlevel)) {
|
2020-04-09 20:50:24 +02:00
|
|
|
float dist = (pos - HospitalRestartPoints[i]).MagnitudeSqr();
|
|
|
|
if (fMinDist >= dist) {
|
|
|
|
fMinDist = dist;
|
|
|
|
closestPoint = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we didn't find anything, find closest point on any level
|
|
|
|
if (closestPoint == NUM_RESTART_POINTS) {
|
|
|
|
for (int i = 0; i < NumberOfHospitalRestarts; i++) {
|
|
|
|
float dist = (pos - HospitalRestartPoints[i]).MagnitudeSqr();
|
|
|
|
if (fMinDist >= dist) {
|
|
|
|
fMinDist = dist;
|
|
|
|
closestPoint = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we still didn't find anything, find closest path node
|
|
|
|
if (closestPoint == NUM_RESTART_POINTS) {
|
2020-05-03 15:57:57 +02:00
|
|
|
*outPos = ThePaths.m_pathNodes[ThePaths.FindNodeClosestToCoors(pos, PATH_PED, 999999.9f)].GetPosition();
|
2020-04-09 20:50:24 +02:00
|
|
|
*outHeading = 0.0f;
|
|
|
|
printf("Couldn't find a hospital restart zone near the player %f %f %f->%f %f %f\n", pos.x, pos.y, pos.z, outPos->x, outPos->y, outPos->z);
|
|
|
|
} else {
|
|
|
|
*outPos = HospitalRestartPoints[closestPoint];
|
|
|
|
*outHeading = HospitalRestartHeadings[closestPoint];
|
|
|
|
}
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::FindClosestPoliceRestartPoint(const CVector &pos, CVector *outPos, float *outHeading)
|
|
|
|
{
|
|
|
|
if (bOverrideRestart) {
|
|
|
|
*outPos = OverridePosition;
|
|
|
|
*outHeading = OverrideHeading;
|
|
|
|
CancelOverrideRestart();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-06 12:23:57 +02:00
|
|
|
eLevelName curlevel = CTheZones::GetLevelFromPosition(&pos);
|
2020-04-09 20:50:24 +02:00
|
|
|
float fMinDist = 16000000.0f;
|
|
|
|
int closestPoint = NUM_RESTART_POINTS;
|
|
|
|
|
|
|
|
// find closest point on this level
|
|
|
|
for (int i = 0; i < NumberOfPoliceRestarts; i++) {
|
2020-07-13 18:10:17 +03:00
|
|
|
if (CTheZones::GetLevelFromPosition(&PoliceRestartPoints[i]) == (OverridePoliceStationLevel != LEVEL_GENERIC ? OverridePoliceStationLevel : curlevel)) {
|
2020-04-09 20:50:24 +02:00
|
|
|
float dist = (pos - PoliceRestartPoints[i]).MagnitudeSqr();
|
|
|
|
if (fMinDist >= dist) {
|
|
|
|
fMinDist = dist;
|
|
|
|
closestPoint = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we didn't find anything, find closest point on any level
|
|
|
|
if (closestPoint == NUM_RESTART_POINTS) {
|
|
|
|
for (int i = 0; i < NumberOfPoliceRestarts; i++) {
|
|
|
|
float dist = (pos - PoliceRestartPoints[i]).MagnitudeSqr();
|
|
|
|
if (fMinDist >= dist) {
|
|
|
|
fMinDist = dist;
|
|
|
|
closestPoint = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we still didn't find anything, find closest path node
|
|
|
|
if (closestPoint == NUM_RESTART_POINTS) {
|
|
|
|
printf("Couldn't find a police restart zone near the player\n");
|
2020-05-03 15:57:57 +02:00
|
|
|
*outPos = ThePaths.m_pathNodes[ThePaths.FindNodeClosestToCoors(pos, PATH_PED, 999999.9f)].GetPosition();
|
2020-04-09 20:50:24 +02:00
|
|
|
*outHeading = 0.0f;
|
|
|
|
} else {
|
|
|
|
*outPos = PoliceRestartPoints[closestPoint];
|
|
|
|
*outHeading = PoliceRestartHeadings[closestPoint];
|
|
|
|
}
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::LoadAllRestartPoints(uint8 *buf, uint32 size)
|
|
|
|
{
|
|
|
|
Initialise();
|
|
|
|
|
|
|
|
INITSAVEBUF
|
|
|
|
CheckSaveHeader(buf, 'R','S','T','\0', size - SAVE_HEADER_SIZE);
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_RESTART_POINTS; i++) {
|
|
|
|
HospitalRestartPoints[i] = ReadSaveBuf<CVector>(buf);
|
|
|
|
HospitalRestartHeadings[i] = ReadSaveBuf<float>(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_RESTART_POINTS; i++) {
|
|
|
|
PoliceRestartPoints[i] = ReadSaveBuf<CVector>(buf);
|
|
|
|
PoliceRestartHeadings[i] = ReadSaveBuf<float>(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
NumberOfHospitalRestarts = ReadSaveBuf<uint16>(buf);
|
|
|
|
NumberOfPoliceRestarts = ReadSaveBuf<uint16>(buf);
|
|
|
|
bOverrideRestart = ReadSaveBuf<bool>(buf);
|
|
|
|
|
|
|
|
// skip something unused
|
|
|
|
ReadSaveBuf<uint8>(buf);
|
|
|
|
ReadSaveBuf<uint16>(buf);
|
|
|
|
|
|
|
|
OverridePosition = ReadSaveBuf<CVector>(buf);
|
|
|
|
OverrideHeading = ReadSaveBuf<float>(buf);
|
|
|
|
bFadeInAfterNextDeath = ReadSaveBuf<bool>(buf);
|
|
|
|
bFadeInAfterNextArrest = ReadSaveBuf<bool>(buf);
|
|
|
|
OverrideHospitalLevel = ReadSaveBuf<uint8>(buf);
|
|
|
|
OverridePoliceStationLevel = ReadSaveBuf<uint8>(buf);
|
|
|
|
VALIDATESAVEBUF(size);
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|
|
|
|
|
2020-04-09 20:50:24 +02:00
|
|
|
void
|
|
|
|
CRestart::SaveAllRestartPoints(uint8 *buf, uint32 *size)
|
|
|
|
{
|
|
|
|
*size = SAVE_HEADER_SIZE
|
|
|
|
+ sizeof(HospitalRestartPoints)
|
|
|
|
+ sizeof(HospitalRestartHeadings)
|
|
|
|
+ sizeof(PoliceRestartPoints)
|
|
|
|
+ sizeof(PoliceRestartHeadings)
|
|
|
|
+ sizeof(NumberOfHospitalRestarts)
|
|
|
|
+ sizeof(NumberOfPoliceRestarts)
|
|
|
|
+ sizeof(bOverrideRestart)
|
|
|
|
+ sizeof(uint8)
|
|
|
|
+ sizeof(uint16)
|
|
|
|
+ sizeof(OverridePosition)
|
|
|
|
+ sizeof(OverrideHeading)
|
|
|
|
+ sizeof(bFadeInAfterNextDeath)
|
|
|
|
+ sizeof(bFadeInAfterNextArrest)
|
|
|
|
+ sizeof(OverrideHospitalLevel)
|
|
|
|
+ sizeof(OverridePoliceStationLevel); // == 292
|
|
|
|
|
|
|
|
INITSAVEBUF
|
|
|
|
WriteSaveHeader(buf, 'R','S','T','\0', *size - SAVE_HEADER_SIZE);
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_RESTART_POINTS; i++) {
|
|
|
|
WriteSaveBuf(buf, HospitalRestartPoints[i]);
|
|
|
|
WriteSaveBuf(buf, HospitalRestartHeadings[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_RESTART_POINTS; i++) {
|
|
|
|
WriteSaveBuf(buf, PoliceRestartPoints[i]);
|
|
|
|
WriteSaveBuf(buf, PoliceRestartHeadings[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteSaveBuf(buf, NumberOfHospitalRestarts);
|
|
|
|
WriteSaveBuf(buf, NumberOfPoliceRestarts);
|
|
|
|
WriteSaveBuf(buf, bOverrideRestart);
|
|
|
|
|
|
|
|
WriteSaveBuf(buf, (uint8)0);
|
|
|
|
WriteSaveBuf(buf, (uint16)0);
|
|
|
|
|
|
|
|
WriteSaveBuf(buf, OverridePosition);
|
|
|
|
WriteSaveBuf(buf, OverrideHeading);
|
|
|
|
WriteSaveBuf(buf, bFadeInAfterNextDeath);
|
|
|
|
WriteSaveBuf(buf, bFadeInAfterNextArrest);
|
|
|
|
WriteSaveBuf(buf, OverrideHospitalLevel);
|
|
|
|
WriteSaveBuf(buf, OverridePoliceStationLevel);
|
|
|
|
VALIDATESAVEBUF(*size);
|
2019-10-10 12:20:53 +03:00
|
|
|
}
|