2023-10-22 17:08:53 +08:00
|
|
|
|
|
|
|
#include "../main.h"
|
2024-04-23 22:44:38 +08:00
|
|
|
#include "../../raknet/SocketDataEncryptor.h"
|
2024-06-06 23:09:13 +08:00
|
|
|
#include "../mathutils.h"
|
2023-10-22 17:08:53 +08:00
|
|
|
|
2024-05-17 22:48:32 +08:00
|
|
|
#define NETGAME_VERSION 4057
|
|
|
|
|
2024-04-25 22:10:37 +08:00
|
|
|
char szGameModeFile[256];
|
|
|
|
|
2024-04-25 22:33:18 +08:00
|
|
|
#define PLAYER_STATE_NONE 0
|
|
|
|
#define PLAYER_STATE_ONFOOT 1
|
|
|
|
#define PLAYER_STATE_DRIVER 2
|
|
|
|
#define PLAYER_STATE_PASSENGER 3
|
|
|
|
#define PLAYER_STATE_WASTED 7
|
|
|
|
#define PLAYER_STATE_SPAWNED 8
|
|
|
|
#define PLAYER_STATE_SPECTATING 9
|
|
|
|
|
2024-04-25 22:19:11 +08:00
|
|
|
#pragma pack(1)
|
|
|
|
typedef struct _AIM_SYNC_DATA // size: 31
|
|
|
|
{
|
|
|
|
char _gap0[31];
|
|
|
|
} AIM_SYNC_DATA;
|
|
|
|
|
2024-04-25 22:20:29 +08:00
|
|
|
#pragma pack(1)
|
|
|
|
typedef struct _TRAILER_SYNC_DATA // size: 54
|
|
|
|
{
|
|
|
|
char _gap0[54];
|
|
|
|
} TRAILER_SYNC_DATA;
|
|
|
|
|
2023-11-18 00:00:52 +08:00
|
|
|
char unnamed_2[63];
|
2024-04-25 22:33:18 +08:00
|
|
|
PASSENGER_SYNC_DATA unnamed_5[MAX_PLAYERS];
|
2024-06-08 21:48:47 +08:00
|
|
|
BOOL bPlayerSlotState[MAX_PLAYERS];
|
2024-07-07 22:35:03 +08:00
|
|
|
BYTE byteState;
|
2024-06-06 23:09:13 +08:00
|
|
|
ONFOOT_SYNC_DATA ofSync;
|
2024-07-01 22:59:50 +08:00
|
|
|
ONFOOT_SYNC_DATA unnamed_3[MAX_PLAYERS];
|
2024-06-29 22:53:05 +08:00
|
|
|
BYTE bytePlayerState[MAX_PLAYERS];
|
2024-06-26 23:08:18 +08:00
|
|
|
BOOL bVehicleSlotState[MAX_VEHICLES];
|
2024-07-01 22:59:50 +08:00
|
|
|
INCAR_SYNC_DATA unnamed_4[MAX_PLAYERS];
|
2024-06-08 21:34:42 +08:00
|
|
|
BYTE byteMySeatID;
|
2023-11-18 00:00:52 +08:00
|
|
|
|
2024-06-03 22:53:22 +08:00
|
|
|
bool bSpawned = false;
|
|
|
|
|
2024-06-29 22:55:26 +08:00
|
|
|
void CNetGame::SetPlayerAdded(PLAYERID playerId, BOOL a2)
|
|
|
|
{
|
|
|
|
if(playerId < MAX_PLAYERS)
|
|
|
|
{
|
|
|
|
bPlayerSlotState[playerId] = a2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CNetGame::SetVehicleAdded(VEHICLEID VehicleID, BOOL a2)
|
|
|
|
{
|
|
|
|
if(VehicleID < MAX_VEHICLES)
|
|
|
|
{
|
|
|
|
bVehicleSlotState[VehicleID] = a2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-29 23:24:41 +08:00
|
|
|
void CNetGame::SetPlayerState(PLAYERID playerId, BYTE byteState)
|
|
|
|
{
|
|
|
|
if(playerId < MAX_PLAYERS)
|
|
|
|
{
|
|
|
|
bytePlayerState[playerId] = byteState;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-29 22:53:05 +08:00
|
|
|
BYTE CNetGame::GetPlayerState(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return PLAYER_STATE_NONE;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return PLAYER_STATE_NONE;
|
|
|
|
|
|
|
|
return bytePlayerState[playerId];
|
|
|
|
}
|
|
|
|
|
2024-07-01 22:59:50 +08:00
|
|
|
BOOL CNetGame::GetPlayerPos(PLAYERID playerId, PVECTOR Vector)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS || !bPlayerSlotState[playerId])
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
switch(bytePlayerState[playerId])
|
|
|
|
{
|
|
|
|
case PLAYER_STATE_ONFOOT:
|
|
|
|
Vector->X = unnamed_3[playerId].vecPos.X;
|
|
|
|
Vector->Y = unnamed_3[playerId].vecPos.Y;
|
|
|
|
Vector->Z = unnamed_3[playerId].vecPos.Z;
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
case PLAYER_STATE_DRIVER:
|
|
|
|
Vector->X = unnamed_4[playerId].vecPos.X;
|
|
|
|
Vector->Y = unnamed_4[playerId].vecPos.Y;
|
|
|
|
Vector->Z = unnamed_4[playerId].vecPos.Z;
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
case PLAYER_STATE_PASSENGER:
|
|
|
|
Vector->X = unnamed_5[playerId].vecPos.X;
|
|
|
|
Vector->Y = unnamed_5[playerId].vecPos.Y;
|
|
|
|
Vector->Z = unnamed_5[playerId].vecPos.Z;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2024-07-01 23:01:21 +08:00
|
|
|
VEHICLEID CNetGame::GetPlayerVehicleID(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return INVALID_VEHICLE_ID;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return INVALID_VEHICLE_ID;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
return unnamed_4[playerId].VehicleID;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
|
|
|
|
{
|
|
|
|
return unnamed_5[playerId].VehicleID;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return INVALID_VEHICLE_ID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-01 23:03:22 +08:00
|
|
|
BYTE CNetGame::GetPlayerArmedWeapon(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return 0;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return 0;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
return unnamed_3[playerId].byteCurrentWeapon;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
return unnamed_4[playerId].byteCurrentWeapon;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
|
|
|
|
{
|
|
|
|
return unnamed_5[playerId].byteCurrentWeapon;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-03 23:23:15 +08:00
|
|
|
BYTE CNetGame::GetPlayerHealth(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return 0;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return 0;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
return unnamed_3[playerId].byteHealth;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
return unnamed_4[playerId].bytePlayerHealth;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
|
|
|
|
{
|
|
|
|
return unnamed_5[playerId].bytePlayerHealth;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-03 23:24:36 +08:00
|
|
|
BYTE CNetGame::GetPlayerArmour(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return 0;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return 0;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
return unnamed_3[playerId].byteArmour;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
return unnamed_4[playerId].bytePlayerArmour;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
|
|
|
|
{
|
|
|
|
return unnamed_5[playerId].bytePlayerArmour;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-06 22:45:04 +08:00
|
|
|
BOOL CNetGame::GetPlayerKeys(PLAYERID playerId, WORD *udAnalog, WORD *lrAnalog, WORD *wKeys)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return FALSE;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return FALSE;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
*udAnalog = unnamed_3[playerId].udAnalog;
|
|
|
|
*lrAnalog = unnamed_3[playerId].lrAnalog;
|
|
|
|
*wKeys = unnamed_3[playerId].wKeys;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
*udAnalog = unnamed_4[playerId].udAnalog;
|
|
|
|
*lrAnalog = unnamed_4[playerId].lrAnalog;
|
|
|
|
*wKeys = unnamed_4[playerId].wKeys;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
|
|
|
|
{
|
|
|
|
*udAnalog = unnamed_5[playerId].udAnalog;
|
|
|
|
*lrAnalog = unnamed_5[playerId].lrAnalog;
|
|
|
|
*wKeys = unnamed_5[playerId].wKeys;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-08 18:01:38 +08:00
|
|
|
BYTE CNetGame::GetPlayerSpecialAction(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return SPECIAL_ACTION_NONE;
|
|
|
|
if(bPlayerSlotState[playerId] == FALSE) return SPECIAL_ACTION_NONE;
|
|
|
|
|
|
|
|
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
return unnamed_3[playerId].byteSpecialAction;
|
|
|
|
}
|
|
|
|
return SPECIAL_ACTION_NONE;
|
|
|
|
}
|
|
|
|
|
2024-06-08 21:52:14 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
// MATCH
|
|
|
|
BOOL CNetGame::IsPlayerAdded(PLAYERID playerId)
|
|
|
|
{
|
|
|
|
if(playerId >= MAX_PLAYERS) return FALSE;
|
|
|
|
|
|
|
|
return bPlayerSlotState[playerId] != FALSE;
|
|
|
|
}
|
|
|
|
|
2024-07-05 22:43:08 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
BOOL CNetGame::IsVehicleAdded(VEHICLEID VehicleID)
|
|
|
|
{
|
|
|
|
if(VehicleID >= MAX_VEHICLES) return FALSE;
|
|
|
|
|
|
|
|
return bVehicleSlotState[VehicleID] != FALSE;
|
|
|
|
}
|
|
|
|
|
2024-07-07 22:37:45 +08:00
|
|
|
//----------------------------------------------------
|
2024-07-08 18:03:56 +08:00
|
|
|
float CNetGame::GetDistanceFromMeToPoint(PVECTOR vecPos)
|
|
|
|
{
|
|
|
|
VECTOR vecMyPos;
|
|
|
|
|
|
|
|
if(GetMyPos(&vecMyPos))
|
|
|
|
{
|
|
|
|
float fX = vecMyPos.X - vecPos->X;
|
|
|
|
float fY = vecMyPos.Y - vecPos->Y;
|
|
|
|
float fZ = vecMyPos.Z - vecPos->Z;
|
|
|
|
|
|
|
|
return (float)sqrt(fX * fX + fY * fY + fZ * fZ);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
}
|
2024-07-07 22:40:49 +08:00
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
PVECTOR CNetGame::GetMyPos(PVECTOR Vector)
|
|
|
|
{
|
|
|
|
if(byteState == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
Vector->X = ofSync.vecPos.X;
|
|
|
|
Vector->Y = ofSync.vecPos.Y;
|
|
|
|
Vector->Z = ofSync.vecPos.Z;
|
|
|
|
return Vector;
|
|
|
|
}
|
|
|
|
else if(byteState == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
Vector->X = icSync.vecPos.X;
|
|
|
|
Vector->Y = icSync.vecPos.Y;
|
|
|
|
Vector->Z = icSync.vecPos.Z;
|
|
|
|
return Vector;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-07 22:37:45 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
void CNetGame::SetMyPos(PVECTOR Vector)
|
|
|
|
{
|
|
|
|
if(byteState == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
ofSync.vecPos.X = Vector->X;
|
|
|
|
ofSync.vecPos.Y = Vector->Y;
|
|
|
|
ofSync.vecPos.Z = Vector->Z;
|
|
|
|
}
|
|
|
|
else if (byteState == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
icSync.vecPos.X = Vector->X;
|
|
|
|
icSync.vecPos.Y = Vector->Y;
|
|
|
|
icSync.vecPos.Z = Vector->Z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-07 22:35:03 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
float CNetGame::GetMyZAngle()
|
|
|
|
{
|
|
|
|
MATRIX4X4 mat;
|
|
|
|
|
|
|
|
if(byteState == PLAYER_STATE_ONFOOT)
|
|
|
|
{
|
|
|
|
QuaternionToMatrix(&ofSync.quatRotation, &mat);
|
|
|
|
|
|
|
|
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f / PI;
|
|
|
|
// Bound it to [0, 360)
|
|
|
|
if ( fZAngle < 0.0f )
|
|
|
|
fZAngle += 360.0f;
|
|
|
|
else if ( fZAngle >= 360.0f )
|
|
|
|
fZAngle -= 360.0f;
|
|
|
|
return fZAngle;
|
|
|
|
}
|
|
|
|
else if(byteState == PLAYER_STATE_DRIVER)
|
|
|
|
{
|
|
|
|
QuaternionToMatrix(&icSync.quatRotation, &mat);
|
|
|
|
|
|
|
|
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f/PI;
|
|
|
|
|
|
|
|
// Bound it to [0, 360)
|
|
|
|
if ( fZAngle < 0.0f )
|
|
|
|
fZAngle += 360.0f;
|
|
|
|
else if ( fZAngle >= 360.0f )
|
|
|
|
fZAngle -= 360.0f;
|
|
|
|
return fZAngle;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
2024-06-06 23:09:13 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
// MATCH
|
|
|
|
void CNetGame::SetMyZAngle(float fAngle)
|
|
|
|
{
|
|
|
|
//logprintf("CNetGame::SetMyZAngle(%f)", fAngle);
|
|
|
|
|
|
|
|
float fRadians = fAngle * 0.017453292f; // (PI/180.0f)
|
|
|
|
|
|
|
|
MATRIX4X4 mat;
|
|
|
|
memset(&mat,0,sizeof(MATRIX4X4));
|
|
|
|
mat.right.Z = 0.0f;
|
|
|
|
mat.up.Z = 0.0f;
|
|
|
|
mat.at.X = 0.0f;
|
|
|
|
mat.at.Y = 0.0f;
|
|
|
|
mat.at.Z = 1.0f;
|
|
|
|
|
|
|
|
float fCos = cos(fRadians);
|
|
|
|
float fSin = sin(fRadians);
|
|
|
|
|
|
|
|
mat.right.X = fCos;
|
|
|
|
mat.right.Y = fSin;
|
|
|
|
mat.up.X = -fSin;
|
|
|
|
mat.up.Y = fCos;
|
|
|
|
|
|
|
|
MatrixToQuaternion(&mat, &ofSync.quatRotation);
|
|
|
|
}
|
|
|
|
|
2023-10-22 17:08:53 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-24 22:36:46 +08:00
|
|
|
// TODO: length assert has to be at line 397
|
|
|
|
// what the fuck is on top of the netgame.cpp that makes GetPacketID is at line 397 ????
|
|
|
|
BYTE GetPacketID(Packet *p)
|
2023-10-22 17:08:53 +08:00
|
|
|
{
|
2024-04-24 22:36:46 +08:00
|
|
|
if (p==0) return 255;
|
|
|
|
|
|
|
|
if ((unsigned char)p->data[0] == ID_TIMESTAMP) {
|
|
|
|
assert(p->length > sizeof(unsigned char) + sizeof(unsigned long));
|
|
|
|
return (unsigned char) p->data[sizeof(unsigned char) + sizeof(unsigned long)];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return (unsigned char) p->data[0];
|
|
|
|
}
|
|
|
|
}
|
2023-10-22 17:08:53 +08:00
|
|
|
|
2024-04-24 22:36:46 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
CNetGame::CNetGame()
|
|
|
|
{
|
|
|
|
// nothing
|
2023-10-22 17:08:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
CNetGame::~CNetGame()
|
|
|
|
{
|
2024-01-29 19:36:08 +08:00
|
|
|
m_pRakClient->Disconnect(0);
|
|
|
|
UnRegisterRPCs(m_pRakClient);
|
|
|
|
UnRegisterScriptRPCs(m_pRakClient); // Unregister server-side scripting RPCs.
|
|
|
|
RakNetworkFactory::DestroyRakClientInterface(m_pRakClient);
|
|
|
|
SAFE_DELETE(m_pGameMode);
|
2024-03-14 23:24:26 +08:00
|
|
|
SAFE_DELETE(m_pScriptTimers);
|
|
|
|
SAFE_DELETE(m_pPlayerPool);
|
|
|
|
SAFE_DELETE(m_pVehiclePool);
|
2023-10-22 17:08:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-01 18:30:44 +08:00
|
|
|
void CNetGame::ShutdownForGameModeRestart()
|
|
|
|
{
|
|
|
|
m_byteWorldTime = 12;
|
|
|
|
m_byteWorldMinute = 0;
|
|
|
|
m_byteWeather = 10;
|
|
|
|
m_byteHoldTime = 1;
|
|
|
|
m_bUseCJWalk = FALSE;
|
|
|
|
m_fGravity = (float)0.008000000;
|
|
|
|
m_iDeathDropMoney = 0;
|
|
|
|
|
|
|
|
m_iGameState = GAMESTATE_RESTARTING;
|
|
|
|
|
|
|
|
// Process the pool one last time
|
|
|
|
m_pPlayerPool->Process();
|
|
|
|
|
|
|
|
ResetVehiclePool();
|
|
|
|
|
|
|
|
StopRecordingPlayback();
|
|
|
|
|
|
|
|
memset(unnamed_2,0,sizeof(unnamed_2));
|
|
|
|
memset(unnamed_3,0,sizeof(unnamed_3));
|
|
|
|
memset(unnamed_4,0,sizeof(unnamed_4));
|
|
|
|
memset(unnamed_5,0,sizeof(unnamed_5));
|
2024-06-26 23:08:18 +08:00
|
|
|
memset(&bVehicleSlotState[0],0,sizeof(BOOL)*MAX_VEHICLES);
|
2024-06-06 23:09:13 +08:00
|
|
|
memset(&ofSync,0,sizeof(ONFOOT_SYNC_DATA));
|
2024-06-08 21:48:47 +08:00
|
|
|
memset(&bPlayerSlotState[0],0,sizeof(BOOL)*MAX_PLAYERS);
|
2024-06-29 22:53:05 +08:00
|
|
|
memset(&bytePlayerState[0],0,sizeof(BYTE)*MAX_PLAYERS);
|
2024-06-01 18:30:44 +08:00
|
|
|
|
|
|
|
m_bZoneNames = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2023-11-18 00:00:52 +08:00
|
|
|
void CNetGame::Init(PCHAR szHostOrIp, int iPort,
|
2024-01-15 23:53:36 +08:00
|
|
|
PCHAR szPlayerName, PCHAR szPass,
|
|
|
|
PCHAR szNpcMode)
|
2023-10-22 17:08:53 +08:00
|
|
|
{
|
2023-11-18 00:00:52 +08:00
|
|
|
strcpy(m_szHostName, "San Andreas Multiplayer");
|
|
|
|
strncpy(m_szHostOrIp, szHostOrIp, sizeof(m_szHostOrIp));
|
|
|
|
m_iPort = iPort;
|
|
|
|
|
2024-01-19 23:19:26 +08:00
|
|
|
m_pGameMode = new CGameMode();
|
2023-11-18 00:00:52 +08:00
|
|
|
m_pScriptTimers = new CScriptTimers();
|
|
|
|
|
|
|
|
// Setup player pool
|
|
|
|
m_pPlayerPool = new CPlayerPool();
|
|
|
|
m_pPlayerPool->SetLocalPlayerName(szPlayerName);
|
|
|
|
m_pVehiclePool = new CVehiclePool();
|
|
|
|
|
2024-01-15 23:53:36 +08:00
|
|
|
m_pRakClient = RakNetworkFactory::GetRakClientInterface();
|
|
|
|
|
2024-04-23 22:44:38 +08:00
|
|
|
SocketDataEncryptor::SetKey(iPort);
|
|
|
|
|
2024-01-15 23:53:36 +08:00
|
|
|
RegisterRPCs(m_pRakClient);
|
2024-04-25 22:10:37 +08:00
|
|
|
RegisterScriptRPCs(m_pRakClient); // Register server-side scripting RPCs.
|
2024-01-15 23:53:36 +08:00
|
|
|
|
2024-04-25 22:10:37 +08:00
|
|
|
sprintf(szGameModeFile, "npcmodes/%s.amx", szNpcMode);
|
|
|
|
if (!m_pGameMode->Load(szGameModeFile))
|
2024-06-30 22:38:49 +08:00
|
|
|
{
|
|
|
|
//logprintf("NPC: I can't load %s so I'm quiting.", szGameModeFile);
|
2024-04-25 22:10:37 +08:00
|
|
|
exit(1);
|
2024-06-30 22:38:49 +08:00
|
|
|
}
|
|
|
|
|
2024-04-25 22:10:37 +08:00
|
|
|
m_pRakClient->SetPassword(szPass);
|
|
|
|
m_pRakClient->Connect(szHostOrIp,iPort,0,0,10);
|
2024-01-15 23:53:36 +08:00
|
|
|
|
2024-04-25 22:10:37 +08:00
|
|
|
m_iGameState = GAMESTATE_CONNECTING;
|
2023-10-22 17:08:53 +08:00
|
|
|
|
2024-06-30 22:38:49 +08:00
|
|
|
//logprintf("NPC(%s): connecting to %s:%d...",szPlayerName,szHostOrIp,iPort);
|
2024-04-25 22:10:37 +08:00
|
|
|
|
|
|
|
m_iSpawnsAvailable = 0;
|
|
|
|
m_byteWorldTime = 12;
|
|
|
|
m_byteWorldMinute = 0;
|
|
|
|
m_byteWeather = 10;
|
|
|
|
m_fGravity = (float)0.008000000;
|
|
|
|
m_iDeathDropMoney = 0;
|
|
|
|
m_bLanMode = FALSE;
|
|
|
|
m_byteHoldTime = 1;
|
|
|
|
m_bUseCJWalk = FALSE;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 100; i++) m_dwMapIcon[i] = NULL;
|
|
|
|
|
|
|
|
m_byteFriendlyFire = 1;
|
|
|
|
m_bZoneNames = FALSE;
|
|
|
|
m_bInstagib = FALSE;
|
|
|
|
|
2023-11-18 00:00:52 +08:00
|
|
|
memset(unnamed_2,0,sizeof(unnamed_2));
|
2024-06-06 23:09:13 +08:00
|
|
|
memset(&ofSync,0,sizeof(ONFOOT_SYNC_DATA));
|
2023-11-18 00:00:52 +08:00
|
|
|
memset(unnamed_3,0,sizeof(unnamed_3));
|
|
|
|
memset(unnamed_4,0,sizeof(unnamed_4));
|
|
|
|
memset(unnamed_5,0,sizeof(unnamed_5));
|
2024-06-26 23:08:18 +08:00
|
|
|
memset(&bVehicleSlotState[0],0,sizeof(BOOL)*MAX_VEHICLES);
|
2024-06-08 21:48:47 +08:00
|
|
|
memset(&bPlayerSlotState[0],0,sizeof(BOOL)*MAX_PLAYERS);
|
2024-06-29 22:53:05 +08:00
|
|
|
memset(&bytePlayerState[0],0,sizeof(BYTE)*MAX_PLAYERS);
|
2023-11-18 00:00:52 +08:00
|
|
|
field_1DE = 0;
|
|
|
|
field_1E2 = 0;
|
2023-12-18 23:27:45 +08:00
|
|
|
field_1F2 = GetTickCount();
|
2024-07-07 22:35:03 +08:00
|
|
|
byteState = PLAYER_STATE_NONE;
|
2023-11-18 00:00:52 +08:00
|
|
|
field_1FA = -1;
|
|
|
|
field_1FE = -1;
|
2023-10-22 17:08:53 +08:00
|
|
|
srand(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-24 22:15:44 +08:00
|
|
|
#ifdef WIN32
|
|
|
|
|
|
|
|
#pragma comment(lib, "winmm.lib")
|
|
|
|
float GetElapsedTime()
|
|
|
|
{
|
|
|
|
static BOOL bTimerInit = false;
|
|
|
|
static BOOL bUsingOPF = false;
|
|
|
|
static LONGLONG nTicksPerSec = 0;
|
|
|
|
|
|
|
|
if (!bTimerInit)
|
|
|
|
{
|
|
|
|
bTimerInit = true;
|
|
|
|
LARGE_INTEGER qwTicksPerSec;
|
|
|
|
bUsingOPF = QueryPerformanceFrequency(&qwTicksPerSec);
|
|
|
|
if (bUsingOPF) nTicksPerSec = qwTicksPerSec.QuadPart;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bUsingOPF)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER qwTime;
|
|
|
|
QueryPerformanceCounter(&qwTime);
|
|
|
|
static LONGLONG llLastTime = qwTime.QuadPart;
|
|
|
|
double fElapsedTime = (double)(qwTime.QuadPart - llLastTime) / (double) nTicksPerSec;
|
|
|
|
llLastTime = qwTime.QuadPart;
|
|
|
|
return (float)fElapsedTime;
|
|
|
|
} else {
|
|
|
|
double fTime = timeGetTime() * 0.001;
|
|
|
|
static double fLastTime = fTime;
|
|
|
|
double fElapsedTime = (double)(fTime - fLastTime);
|
|
|
|
fLastTime = fTime;
|
|
|
|
return (float)fElapsedTime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
float GetElapsedTime()
|
|
|
|
{
|
|
|
|
static timeval lasttv;
|
|
|
|
timeval tv;
|
|
|
|
float fRet;
|
|
|
|
|
|
|
|
gettimeofday(&tv, NULL);
|
|
|
|
|
|
|
|
if (!timerisset(&lasttv)) memcpy(&lasttv, &tv, sizeof(timeval));
|
|
|
|
|
|
|
|
fRet = (float)((tv.tv_sec - lasttv.tv_sec) * 1000000) + (tv.tv_usec - lasttv.tv_usec);
|
|
|
|
fRet /= 1000000.0f;
|
|
|
|
|
|
|
|
memcpy(&lasttv,&tv,sizeof(timeval));
|
|
|
|
|
|
|
|
return fRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // WIN32
|
|
|
|
|
2023-10-22 17:08:53 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
void CNetGame::Process()
|
|
|
|
{
|
2024-04-25 22:14:30 +08:00
|
|
|
float fElapsedTime = GetElapsedTime();
|
|
|
|
|
|
|
|
UpdateNetwork();
|
|
|
|
|
|
|
|
|
|
|
|
|
2023-10-22 17:08:53 +08:00
|
|
|
// TODO: CNetGame::Process (W: 00418370 L: 080AD6A4)
|
|
|
|
/*
|
|
|
|
if ( this->field_C == 2 )
|
|
|
|
{
|
|
|
|
if ( this->field_386 )
|
|
|
|
sub_80A969E(this->field_386, v2);
|
|
|
|
if ( this->field_38A )
|
|
|
|
sub_80B8610(this->field_38A, (signed __int64)(v2 * 1000.0));
|
|
|
|
if ( byte_810A708 )
|
|
|
|
{
|
|
|
|
if ( this->field_1DE )
|
|
|
|
{
|
|
|
|
sub_80ADFDE(this);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sub_80AD770(this);
|
|
|
|
byte_80E16B7 = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
}
|
|
|
|
|
2024-04-25 22:14:30 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
// UPDATE NETWORK
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
void CNetGame::UpdateNetwork()
|
|
|
|
{
|
|
|
|
Packet* pkt=NULL;
|
|
|
|
unsigned char packetIdentifier;
|
|
|
|
|
|
|
|
while((pkt = m_pRakClient->Receive()))
|
|
|
|
{
|
|
|
|
packetIdentifier = GetPacketID(pkt);
|
|
|
|
|
|
|
|
switch(packetIdentifier)
|
|
|
|
{
|
2024-06-01 19:59:47 +08:00
|
|
|
case ID_RSA_PUBLIC_KEY_MISMATCH:
|
|
|
|
Packet_RSAPublicKeyMismatch(pkt);
|
|
|
|
break;
|
2024-05-10 22:20:05 +08:00
|
|
|
case ID_CONNECTION_BANNED:
|
|
|
|
Packet_ConnectionBanned(pkt);
|
|
|
|
break;
|
2024-06-01 20:10:45 +08:00
|
|
|
case ID_NO_FREE_INCOMING_CONNECTIONS:
|
|
|
|
Packet_NoFreeIncomingConnections(pkt);
|
|
|
|
break;
|
2024-05-10 22:20:46 +08:00
|
|
|
case ID_DISCONNECTION_NOTIFICATION:
|
|
|
|
Packet_DisconnectionNotification(pkt);
|
|
|
|
break;
|
2024-06-01 20:34:10 +08:00
|
|
|
case ID_CONNECTION_LOST:
|
|
|
|
Packet_ConnectionLost(pkt);
|
|
|
|
break;
|
2024-06-01 20:35:12 +08:00
|
|
|
case ID_INVALID_PASSWORD:
|
|
|
|
Packet_InvalidPassword(pkt);
|
|
|
|
break;
|
2024-04-25 22:14:30 +08:00
|
|
|
case ID_MODIFIED_PACKET:
|
|
|
|
Packet_ModifiedPacket(pkt);
|
|
|
|
break;
|
2024-06-01 18:44:36 +08:00
|
|
|
case ID_CONNECTION_ATTEMPT_FAILED:
|
|
|
|
Packet_ConnectAttemptFailed(pkt);
|
|
|
|
break;
|
2024-05-17 22:48:32 +08:00
|
|
|
case ID_CONNECTION_REQUEST_ACCEPTED:
|
|
|
|
Packet_ConnectionSucceeded(pkt);
|
|
|
|
break;
|
2024-04-25 22:33:18 +08:00
|
|
|
case ID_PASSENGER_SYNC:
|
|
|
|
Packet_PassengerSync(pkt);
|
|
|
|
break;
|
2024-04-25 22:19:11 +08:00
|
|
|
case ID_AIM_SYNC:
|
|
|
|
Packet_AimSync(pkt);
|
|
|
|
break;
|
2024-04-25 22:20:29 +08:00
|
|
|
case ID_TRAILER_SYNC:
|
|
|
|
Packet_TrailerSync(pkt);
|
|
|
|
break;
|
2024-04-25 22:14:30 +08:00
|
|
|
}
|
2024-06-08 21:57:35 +08:00
|
|
|
|
|
|
|
m_pRakClient->DeallocatePacket(pkt);
|
2024-04-25 22:14:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
// PACKET HANDLERS INTERNAL
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-25 22:19:11 +08:00
|
|
|
void CNetGame::Packet_AimSync(Packet *p)
|
|
|
|
{
|
|
|
|
RakNet::BitStream bsAimSync((PCHAR)p->data, p->length, false);
|
|
|
|
AIM_SYNC_DATA aimSync;
|
|
|
|
BYTE bytePacketID=0;
|
|
|
|
BYTE bytePlayerID=0;
|
|
|
|
|
|
|
|
if(GetGameState() != GAMESTATE_CONNECTED) return;
|
|
|
|
|
|
|
|
bsAimSync.Read(bytePacketID);
|
|
|
|
bsAimSync.Read(bytePlayerID);
|
|
|
|
bsAimSync.Read((PCHAR)&aimSync,sizeof(AIM_SYNC_DATA));
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-25 22:33:18 +08:00
|
|
|
void CNetGame::Packet_PassengerSync(Packet *p)
|
|
|
|
{
|
|
|
|
RakNet::BitStream bsPassengerSync((PCHAR)p->data, p->length, false);
|
|
|
|
BYTE bytePacketID=0;
|
|
|
|
PLAYERID playerId=0;
|
|
|
|
PASSENGER_SYNC_DATA psSync;
|
|
|
|
|
|
|
|
if(GetGameState() != GAMESTATE_CONNECTED) return;
|
|
|
|
|
|
|
|
bsPassengerSync.Read(bytePacketID);
|
|
|
|
bsPassengerSync.Read(playerId);
|
|
|
|
bsPassengerSync.Read((PCHAR)&psSync,sizeof(PASSENGER_SYNC_DATA));
|
|
|
|
|
|
|
|
if (playerId < MAX_PLAYERS)
|
|
|
|
{
|
|
|
|
memcpy(&unnamed_5[playerId],&psSync,sizeof(PASSENGER_SYNC_DATA));
|
2024-06-29 22:53:05 +08:00
|
|
|
bytePlayerState[playerId] = PLAYER_STATE_PASSENGER;
|
2024-04-25 22:33:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-25 22:20:29 +08:00
|
|
|
void CNetGame::Packet_TrailerSync(Packet *p)
|
|
|
|
{
|
|
|
|
RakNet::BitStream bsTrailerSync((PCHAR)p->data, p->length, false);
|
|
|
|
if(GetGameState() != GAMESTATE_CONNECTED) return;
|
|
|
|
|
|
|
|
BYTE bytePacketID=0;
|
|
|
|
BYTE bytePlayerID=0;
|
|
|
|
TRAILER_SYNC_DATA trSync;
|
|
|
|
|
|
|
|
bsTrailerSync.Read(bytePacketID);
|
|
|
|
bsTrailerSync.Read(bytePlayerID);
|
|
|
|
bsTrailerSync.Read((PCHAR)&trSync, sizeof(TRAILER_SYNC_DATA));
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-01 19:59:47 +08:00
|
|
|
void CNetGame::Packet_RSAPublicKeyMismatch(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: Failed to initialize encryption.");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-05-10 22:20:05 +08:00
|
|
|
void CNetGame::Packet_ConnectionBanned(Packet* packet)
|
|
|
|
{
|
2024-06-01 20:04:15 +08:00
|
|
|
//logprintf("NPC: You're banned from the server.");
|
2024-05-10 22:20:05 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-01 20:09:10 +08:00
|
|
|
void CNetGame::Packet_ConnectionRequestAccepted(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: Server has accepted the connection.");
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-01 20:10:45 +08:00
|
|
|
void CNetGame::Packet_NoFreeIncomingConnections(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: The server is full.");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-05-10 22:20:46 +08:00
|
|
|
void CNetGame::Packet_DisconnectionNotification(Packet* packet)
|
|
|
|
{
|
2024-06-01 20:32:29 +08:00
|
|
|
//logprintf("NPC: Disconnected.");
|
2024-05-10 22:20:46 +08:00
|
|
|
m_pRakClient->Disconnect(0);
|
2024-06-01 20:34:10 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
void CNetGame::Packet_ConnectionLost(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: Lost connection to the server.");
|
|
|
|
m_pRakClient->Disconnect(0);
|
2024-06-01 20:35:12 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
|
|
|
void CNetGame::Packet_InvalidPassword(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: Wrong server password.");
|
|
|
|
m_pRakClient->Disconnect(0);
|
2024-05-10 22:20:46 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-04-25 22:14:30 +08:00
|
|
|
void CNetGame::Packet_ModifiedPacket(Packet* packet)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-06-01 18:44:36 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
// RST
|
|
|
|
|
|
|
|
void CNetGame::Packet_ConnectAttemptFailed(Packet* packet)
|
|
|
|
{
|
|
|
|
//logprintf("NPC: Connection attempt failed.");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2024-05-17 22:48:32 +08:00
|
|
|
//----------------------------------------------------
|
|
|
|
// Connection Success
|
|
|
|
|
|
|
|
void CNetGame::Packet_ConnectionSucceeded(Packet *p)
|
|
|
|
{
|
|
|
|
m_iGameState = GAMESTATE_AWAIT_JOIN;
|
|
|
|
|
|
|
|
RakNet::BitStream bsReturnParams((PCHAR)p->data, p->length, true);
|
|
|
|
|
|
|
|
BYTE bytePacketID=0;
|
|
|
|
unsigned int binaryAddr=0;
|
|
|
|
unsigned short port=0;
|
|
|
|
unsigned short playerId=0;
|
|
|
|
unsigned int uiChallenge=0;
|
|
|
|
|
|
|
|
bsReturnParams.Read(bytePacketID);
|
|
|
|
bsReturnParams.Read(binaryAddr);
|
|
|
|
bsReturnParams.Read(port);
|
|
|
|
bsReturnParams.Read(playerId);
|
|
|
|
bsReturnParams.Read(uiChallenge);
|
|
|
|
|
|
|
|
uiChallenge ^= NETGAME_VERSION;
|
|
|
|
|
2024-06-30 22:38:03 +08:00
|
|
|
//logprintf("NPC: Connection Succeeded");
|
|
|
|
|
2024-05-17 22:48:32 +08:00
|
|
|
int iVersion = NETGAME_VERSION;
|
|
|
|
BYTE byteMod = 1;
|
|
|
|
BYTE byteNameLen = (BYTE)strlen(m_pPlayerPool->GetLocalPlayerName());
|
|
|
|
|
|
|
|
RakNet::BitStream bsSend;
|
|
|
|
bsSend.Write(iVersion);
|
|
|
|
bsSend.Write(byteMod);
|
|
|
|
bsSend.Write(byteNameLen);
|
|
|
|
bsSend.Write(m_pPlayerPool->GetLocalPlayerName(),byteNameLen);
|
|
|
|
bsSend.Write(uiChallenge);
|
|
|
|
|
|
|
|
m_pRakClient->RPC(RPC_NPCJoin,&bsSend,HIGH_PRIORITY,RELIABLE,0,FALSE);
|
|
|
|
}
|
|
|
|
|
2023-10-22 17:08:53 +08:00
|
|
|
//----------------------------------------------------
|
2024-03-14 23:24:26 +08:00
|
|
|
|
2024-06-01 18:30:44 +08:00
|
|
|
void CNetGame::ResetVehiclePool()
|
|
|
|
{
|
|
|
|
if(m_pVehiclePool) {
|
|
|
|
delete m_pVehiclePool;
|
|
|
|
}
|
|
|
|
m_pVehiclePool = new CVehiclePool();
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-01 18:35:59 +08:00
|
|
|
void CNetGame::ResetPlayerPool()
|
|
|
|
{
|
|
|
|
if(m_pPlayerPool) {
|
|
|
|
delete m_pPlayerPool;
|
|
|
|
}
|
|
|
|
m_pPlayerPool = new CPlayerPool();
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------
|
|
|
|
|
2024-06-29 22:45:28 +08:00
|
|
|
void CNetGame::SendChat(char *szText)
|
|
|
|
{
|
|
|
|
BYTE byteTextLen = strlen(szText);
|
|
|
|
|
|
|
|
RakNet::BitStream bsSend;
|
|
|
|
bsSend.Write(byteTextLen);
|
|
|
|
bsSend.Write(szText,byteTextLen);
|
|
|
|
|
|
|
|
GetRakClient()->RPC(RPC_Chat,&bsSend,HIGH_PRIORITY,RELIABLE,0,false);
|
|
|
|
}
|
|
|
|
|
2024-06-29 22:47:25 +08:00
|
|
|
void CNetGame::SendCommand(char *szCommand)
|
|
|
|
{
|
|
|
|
RakNet::BitStream bsParams;
|
|
|
|
int iStrlen = strlen(szCommand);
|
|
|
|
|
|
|
|
bsParams.Write(iStrlen);
|
|
|
|
bsParams.Write(szCommand, iStrlen);
|
|
|
|
GetRakClient()->RPC(RPC_ServerCommand,&bsParams,HIGH_PRIORITY,RELIABLE,0,false);
|
|
|
|
}
|
|
|
|
|
2024-06-01 18:30:44 +08:00
|
|
|
void CNetGame::StopRecordingPlayback()
|
|
|
|
{
|
|
|
|
field_1DE = 0;
|
|
|
|
field_1DA = 0;
|
|
|
|
}
|
|
|
|
|
2024-06-29 22:38:46 +08:00
|
|
|
void CNetGame::PauseRecordingPlayback()
|
|
|
|
{
|
|
|
|
if(field_1DE == 1 || field_1DE == 2)
|
|
|
|
field_1FA = 11;
|
|
|
|
}
|
2024-06-01 18:30:44 +08:00
|
|
|
|
2024-06-29 22:41:19 +08:00
|
|
|
void CNetGame::ResumeRecordingPlayback()
|
|
|
|
{
|
|
|
|
field_1FE = 9;
|
|
|
|
field_1DE = 1;
|
|
|
|
if(field_1FA == -1)
|
|
|
|
field_1EA += GetTickCount() - field_1F6 + 100;
|
|
|
|
else
|
|
|
|
field_1FA = -1;
|
|
|
|
}
|