2021-12-15 03:31:57 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
using GTA;
|
|
|
|
|
using GTA.Native;
|
|
|
|
|
using GTA.Math;
|
|
|
|
|
|
|
|
|
|
namespace CoopClient.Entities
|
|
|
|
|
{
|
|
|
|
|
public partial class EntitiesPed
|
|
|
|
|
{
|
|
|
|
|
#region -- VARIABLES --
|
|
|
|
|
private ulong VehicleStopTime { get; set; }
|
|
|
|
|
|
|
|
|
|
internal bool IsInVehicle { get; set; }
|
|
|
|
|
private int LastVehicleModelHash = 0;
|
|
|
|
|
private int CurrentVehicleModelHash = 0;
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The latest vehicle model hash (may not have been applied yet)
|
|
|
|
|
/// </summary>
|
|
|
|
|
public int VehicleModelHash
|
|
|
|
|
{
|
|
|
|
|
get => CurrentVehicleModelHash;
|
|
|
|
|
internal set
|
|
|
|
|
{
|
|
|
|
|
LastVehicleModelHash = CurrentVehicleModelHash == 0 ? value : CurrentVehicleModelHash;
|
|
|
|
|
CurrentVehicleModelHash = value;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-16 22:05:40 +01:00
|
|
|
|
private byte[] LastVehicleColors = new byte[] { 0, 0 };
|
|
|
|
|
internal byte[] VehicleColors { get; set; }
|
2021-12-15 03:31:57 +01:00
|
|
|
|
private Dictionary<int, int> LastVehicleMods = new Dictionary<int, int>();
|
|
|
|
|
internal Dictionary<int, int> VehicleMods { get; set; }
|
|
|
|
|
internal bool VehicleDead { get; set; }
|
|
|
|
|
internal float VehicleEngineHealth { get; set; }
|
2021-12-16 22:05:40 +01:00
|
|
|
|
internal short VehicleSeatIndex { get; set; }
|
2021-12-15 03:31:57 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// ?
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Vehicle MainVehicle { get; internal set; }
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The latest vehicle rotation (may not have been applied yet)
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Quaternion VehicleRotation { get; internal set; }
|
|
|
|
|
internal Vector3 VehicleVelocity { get; set; }
|
|
|
|
|
private float LastVehicleSpeed { get; set; }
|
|
|
|
|
private float CurrentVehicleSpeed { get; set; }
|
|
|
|
|
internal float VehicleSpeed
|
|
|
|
|
{
|
|
|
|
|
get => CurrentVehicleSpeed;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
LastVehicleSpeed = CurrentVehicleSpeed;
|
|
|
|
|
CurrentVehicleSpeed = value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
internal float VehicleSteeringAngle { get; set; }
|
|
|
|
|
private int LastVehicleAim;
|
|
|
|
|
internal bool VehIsEngineRunning { get; set; }
|
|
|
|
|
internal float VehRPM { get; set; }
|
|
|
|
|
private bool LastTransformed = false;
|
|
|
|
|
internal bool Transformed { get; set; }
|
|
|
|
|
private bool LastHornActive = false;
|
|
|
|
|
internal bool IsHornActive { get; set; }
|
|
|
|
|
internal bool VehAreLightsOn { get; set; }
|
|
|
|
|
internal bool VehAreHighBeamsOn { get; set; }
|
|
|
|
|
internal byte VehLandingGear { get; set; }
|
2021-12-18 09:46:42 +01:00
|
|
|
|
internal bool VehRoofOpened { get; set; }
|
2021-12-15 03:31:57 +01:00
|
|
|
|
internal bool VehIsSireneActive { get; set; }
|
2021-12-23 22:03:01 +01:00
|
|
|
|
internal VehicleDamageModel VehDamageModel { get; set; }
|
2021-12-15 03:31:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
private void DisplayInVehicle()
|
|
|
|
|
{
|
|
|
|
|
if (MainVehicle == null || !MainVehicle.Exists() || MainVehicle.Model.Hash != CurrentVehicleModelHash)
|
|
|
|
|
{
|
|
|
|
|
bool vehFound = false;
|
|
|
|
|
|
|
|
|
|
if (NPCVehHandle != 0)
|
|
|
|
|
{
|
|
|
|
|
lock (Main.NPCsVehicles)
|
|
|
|
|
{
|
|
|
|
|
if (Main.NPCsVehicles.ContainsKey(NPCVehHandle))
|
|
|
|
|
{
|
|
|
|
|
Vehicle targetVehicle = World.GetAllVehicles().First(x => x.Handle == Main.NPCsVehicles[NPCVehHandle]);
|
|
|
|
|
if (targetVehicle == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MainVehicle = targetVehicle;
|
|
|
|
|
vehFound = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Vehicle targetVehicle = World.GetClosestVehicle(Position, 7f, new Model[] { CurrentVehicleModelHash });
|
|
|
|
|
|
|
|
|
|
if (targetVehicle != null)
|
|
|
|
|
{
|
|
|
|
|
if (targetVehicle.IsSeatFree((VehicleSeat)VehicleSeatIndex))
|
|
|
|
|
{
|
|
|
|
|
MainVehicle = targetVehicle;
|
|
|
|
|
vehFound = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!vehFound)
|
|
|
|
|
{
|
|
|
|
|
Model vehicleModel = CurrentVehicleModelHash.ModelRequest();
|
|
|
|
|
if (vehicleModel == null)
|
|
|
|
|
{
|
|
|
|
|
//GTA.UI.Notification.Show($"~r~(Vehicle)Model ({CurrentVehicleModelHash}) cannot be loaded!");
|
|
|
|
|
ModelNotFound = 2;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-23 22:03:01 +01:00
|
|
|
|
MainVehicle = World.CreateVehicle(vehicleModel, Position);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
vehicleModel.MarkAsNoLongerNeeded();
|
|
|
|
|
if (NPCVehHandle != 0)
|
|
|
|
|
{
|
|
|
|
|
Main.NPCsVehicles.Add(NPCVehHandle, MainVehicle.Handle);
|
|
|
|
|
}
|
|
|
|
|
MainVehicle.Quaternion = VehicleRotation;
|
2021-12-19 00:53:03 +01:00
|
|
|
|
|
|
|
|
|
if (MainVehicle.HasRoof)
|
|
|
|
|
{
|
|
|
|
|
bool roofOpened = MainVehicle.RoofState == VehicleRoofState.Opened || MainVehicle.RoofState == VehicleRoofState.Opening;
|
|
|
|
|
if (roofOpened != VehRoofOpened)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.RoofState = VehRoofOpened ? VehicleRoofState.Opened : VehicleRoofState.Closed;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-15 03:31:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!Character.IsInVehicle() || (int)Character.SeatIndex != VehicleSeatIndex || Character.CurrentVehicle.Handle != MainVehicle.Handle)
|
|
|
|
|
{
|
|
|
|
|
if (VehicleSeatIndex == -1 &&
|
|
|
|
|
Game.Player.Character.IsInVehicle() &&
|
|
|
|
|
(int)Game.Player.Character.SeatIndex == -1 &&
|
|
|
|
|
Game.Player.Character.CurrentVehicle.Handle == MainVehicle.Handle)
|
|
|
|
|
{
|
|
|
|
|
Game.Player.Character.Task.WarpOutOfVehicle(MainVehicle);
|
|
|
|
|
GTA.UI.Notification.Show("~r~Car jacked!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Character.SetIntoVehicle(MainVehicle, (VehicleSeat)VehicleSeatIndex);
|
|
|
|
|
Character.IsVisible = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region -- VEHICLE SYNC --
|
|
|
|
|
if (AimCoords != default)
|
|
|
|
|
{
|
|
|
|
|
if (MainVehicle.IsTurretSeat(VehicleSeatIndex))
|
|
|
|
|
{
|
|
|
|
|
int gameTime = Game.GameTime;
|
|
|
|
|
if (gameTime - LastVehicleAim > 30)
|
|
|
|
|
{
|
|
|
|
|
Function.Call(Hash.TASK_VEHICLE_AIM_AT_COORD, Character.Handle, AimCoords.X, AimCoords.Y, AimCoords.Z);
|
|
|
|
|
LastVehicleAim = gameTime;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (MainVehicle.GetResponsiblePedHandle() != Character.Handle)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (VehicleColors != null && VehicleColors != LastVehicleColors)
|
|
|
|
|
{
|
|
|
|
|
Function.Call(Hash.SET_VEHICLE_COLOURS, MainVehicle, VehicleColors[0], VehicleColors[1]);
|
|
|
|
|
|
|
|
|
|
LastVehicleColors = VehicleColors;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Character.IsOnBike && MainVehicle.ClassType == VehicleClass.Cycles)
|
|
|
|
|
{
|
|
|
|
|
bool isFastPedaling = Function.Call<bool>(Hash.IS_ENTITY_PLAYING_ANIM, Character.Handle, PedalingAnimDict(), "fast_pedal_char", 3);
|
|
|
|
|
if (CurrentVehicleSpeed < 0.2f)
|
|
|
|
|
{
|
|
|
|
|
StopPedalingAnim(isFastPedaling);
|
|
|
|
|
}
|
|
|
|
|
else if (CurrentVehicleSpeed < 11f && !Function.Call<bool>(Hash.IS_ENTITY_PLAYING_ANIM, Character.Handle, PedalingAnimDict(), "cruise_pedal_char", 3))
|
|
|
|
|
{
|
|
|
|
|
StartPedalingAnim(false);
|
|
|
|
|
}
|
|
|
|
|
else if (CurrentVehicleSpeed >= 11f && !isFastPedaling)
|
|
|
|
|
{
|
|
|
|
|
StartPedalingAnim(true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-12-16 22:26:45 +01:00
|
|
|
|
if (VehicleMods != null && !VehicleMods.Compare(LastVehicleMods))
|
2021-12-15 03:31:57 +01:00
|
|
|
|
{
|
|
|
|
|
Function.Call(Hash.SET_VEHICLE_MOD_KIT, MainVehicle, 0);
|
|
|
|
|
|
|
|
|
|
foreach (KeyValuePair<int, int> mod in VehicleMods)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.Mods[(VehicleModType)mod.Key].Index = mod.Value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LastVehicleMods = VehicleMods;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MainVehicle.EngineHealth = VehicleEngineHealth;
|
|
|
|
|
|
|
|
|
|
if (VehicleDead && !MainVehicle.IsDead)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.Explode();
|
|
|
|
|
}
|
|
|
|
|
else if (!VehicleDead && MainVehicle.IsDead)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.Repair();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (VehIsEngineRunning != MainVehicle.IsEngineRunning)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.IsEngineRunning = VehIsEngineRunning;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MainVehicle.CurrentRPM = VehRPM;
|
|
|
|
|
|
|
|
|
|
if (VehAreLightsOn != MainVehicle.AreLightsOn)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.AreLightsOn = VehAreLightsOn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (VehAreHighBeamsOn != MainVehicle.AreHighBeamsOn)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.AreHighBeamsOn = VehAreHighBeamsOn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (MainVehicle.IsSubmarineCar)
|
|
|
|
|
{
|
|
|
|
|
if (Transformed && !LastTransformed)
|
|
|
|
|
{
|
|
|
|
|
LastTransformed = true;
|
2021-12-19 00:53:03 +01:00
|
|
|
|
Function.Call(Hash._TRANSFORM_VEHICLE_TO_SUBMARINE, MainVehicle.Handle, false);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
}
|
|
|
|
|
else if (!Transformed && LastTransformed)
|
|
|
|
|
{
|
|
|
|
|
LastTransformed = false;
|
2021-12-19 00:53:03 +01:00
|
|
|
|
Function.Call(Hash._TRANSFORM_SUBMARINE_TO_VEHICLE, MainVehicle.Handle, false);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (MainVehicle.IsPlane)
|
|
|
|
|
{
|
|
|
|
|
if (VehLandingGear != (byte)MainVehicle.LandingGearState)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.LandingGearState = (VehicleLandingGearState)VehLandingGear;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (MainVehicle.HasSiren && VehIsSireneActive != MainVehicle.IsSirenActive)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.IsSirenActive = VehIsSireneActive;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (IsHornActive && !LastHornActive)
|
|
|
|
|
{
|
|
|
|
|
LastHornActive = true;
|
|
|
|
|
MainVehicle.SoundHorn(99999);
|
|
|
|
|
}
|
|
|
|
|
else if (!IsHornActive && LastHornActive)
|
|
|
|
|
{
|
|
|
|
|
LastHornActive = false;
|
|
|
|
|
MainVehicle.SoundHorn(1);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-18 09:46:42 +01:00
|
|
|
|
if (MainVehicle.HasRoof)
|
|
|
|
|
{
|
|
|
|
|
bool roofOpened = MainVehicle.RoofState == VehicleRoofState.Opened || MainVehicle.RoofState == VehicleRoofState.Opening;
|
|
|
|
|
if (roofOpened != VehRoofOpened)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.RoofState = VehRoofOpened ? VehicleRoofState.Opening : VehicleRoofState.Closing;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-15 03:31:57 +01:00
|
|
|
|
Function.Call(Hash.SET_VEHICLE_BRAKE_LIGHTS, MainVehicle, CurrentVehicleSpeed > 0.2f && LastVehicleSpeed > CurrentVehicleSpeed);
|
|
|
|
|
|
|
|
|
|
if (LastSyncWasFull)
|
|
|
|
|
{
|
2021-12-23 22:03:01 +01:00
|
|
|
|
MainVehicle.SetVehicleDamageModel(VehDamageModel);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (VehicleSteeringAngle != MainVehicle.SteeringAngle)
|
|
|
|
|
{
|
|
|
|
|
MainVehicle.CustomSteeringAngle((float)(Math.PI / 180) * VehicleSteeringAngle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Good enough for now, but we need to create a better sync
|
2021-12-26 23:19:12 +01:00
|
|
|
|
if (CurrentVehicleSpeed > 0.05f && MainVehicle.IsInRange(Position, 7.0f))
|
2021-12-15 03:31:57 +01:00
|
|
|
|
{
|
|
|
|
|
int forceMultiplier = (Game.Player.Character.IsInVehicle() && MainVehicle.IsTouching(Game.Player.Character.CurrentVehicle)) ? 1 : 3;
|
|
|
|
|
|
2021-12-26 23:19:12 +01:00
|
|
|
|
MainVehicle.Velocity = VehicleVelocity + forceMultiplier * (Position - MainVehicle.Position);
|
|
|
|
|
MainVehicle.Quaternion = Quaternion.Slerp(MainVehicle.Quaternion, VehicleRotation, 0.5f);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
|
|
|
|
|
VehicleStopTime = Util.GetTickCount64();
|
|
|
|
|
}
|
|
|
|
|
else if ((Util.GetTickCount64() - VehicleStopTime) <= 1000)
|
|
|
|
|
{
|
2021-12-26 23:19:12 +01:00
|
|
|
|
Vector3 posTarget = Util.LinearVectorLerp(MainVehicle.Position, Position + (Position - MainVehicle.Position), Util.GetTickCount64() - VehicleStopTime, 1000);
|
|
|
|
|
|
|
|
|
|
MainVehicle.PositionNoOffset = posTarget;
|
|
|
|
|
MainVehicle.Quaternion = Quaternion.Slerp(MainVehicle.Quaternion, VehicleRotation, 0.5f);
|
2021-12-15 03:31:57 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-12-23 22:03:01 +01:00
|
|
|
|
MainVehicle.PositionNoOffset = Position;
|
2021-12-15 03:31:57 +01:00
|
|
|
|
MainVehicle.Quaternion = VehicleRotation;
|
|
|
|
|
}
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region -- PEDALING --
|
2021-12-26 23:19:12 +01:00
|
|
|
|
/*
|
|
|
|
|
* Thanks to @oldnapalm.
|
|
|
|
|
*/
|
2021-12-15 03:31:57 +01:00
|
|
|
|
|
|
|
|
|
private string PedalingAnimDict()
|
|
|
|
|
{
|
|
|
|
|
switch ((VehicleHash)CurrentVehicleModelHash)
|
|
|
|
|
{
|
|
|
|
|
case VehicleHash.Bmx:
|
|
|
|
|
return "veh@bicycle@bmx@front@base";
|
|
|
|
|
case VehicleHash.Cruiser:
|
|
|
|
|
return "veh@bicycle@cruiserfront@base";
|
|
|
|
|
case VehicleHash.Scorcher:
|
|
|
|
|
return "veh@bicycle@mountainfront@base";
|
|
|
|
|
default:
|
|
|
|
|
return "veh@bicycle@roadfront@base";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string PedalingAnimName(bool fast)
|
|
|
|
|
{
|
|
|
|
|
return fast ? "fast_pedal_char" : "cruise_pedal_char";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StartPedalingAnim(bool fast)
|
|
|
|
|
{
|
|
|
|
|
Character.Task.PlayAnimation(PedalingAnimDict(), PedalingAnimName(fast), 8.0f, -8.0f, -1, AnimationFlags.Loop | AnimationFlags.AllowRotation, 1.0f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StopPedalingAnim(bool fast)
|
|
|
|
|
{
|
|
|
|
|
Character.Task.ClearAnimation(PedalingAnimDict(), PedalingAnimName(fast));
|
|
|
|
|
}
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
|
|
|
|
}
|