392 lines
13 KiB
C#
Raw Normal View History

2022-05-23 09:15:50 +08:00
using System;
2022-05-22 15:55:26 +08:00
using System.IO;
2021-07-07 13:36:25 +02:00
using System.Linq;
using System.Windows.Forms;
using System.Collections.Generic;
2021-08-18 11:47:59 +02:00
using System.Drawing;
2022-05-22 15:55:26 +08:00
using System.Threading;
using System.Diagnostics;
using RageCoop.Client.Menus;
using RageCoop.Core;
2021-07-07 13:36:25 +02:00
using GTA;
using GTA.Native;
2022-05-22 15:55:26 +08:00
using GTA.Math;
2021-07-07 13:36:25 +02:00
2022-05-22 15:55:26 +08:00
namespace RageCoop.Client
2021-07-07 13:36:25 +02:00
{
2021-12-03 20:30:00 +01:00
/// <summary>
/// Don't use it!
/// </summary>
internal class Main : Script
2021-07-07 13:36:25 +02:00
{
private bool _gameLoaded = false;
internal static readonly string CurrentVersion = "V0_4_1";
2021-07-07 13:36:25 +02:00
internal static int LocalPlayerID=0;
internal static RelationshipGroup SyncedPedsGroup;
2021-07-07 13:36:25 +02:00
internal static new Settings Settings = null;
2021-08-18 11:47:59 +02:00
#if !NON_INTERACTIVE
#endif
internal static Chat MainChat = null;
internal static Stopwatch Counter = new Stopwatch();
internal static Core.Logging.Logger Logger = null;
internal static ulong Ticked = 0;
2022-06-12 17:11:14 +08:00
internal static Scripting.Resources Resources=null;
2022-05-22 15:55:26 +08:00
private static List<Func<bool>> QueuedActions = new List<Func<bool>>();
2021-12-03 20:30:00 +01:00
/// <summary>
/// Don't use it!
/// </summary>
2021-07-07 13:36:25 +02:00
public Main()
{
2022-06-04 12:04:02 +08:00
Settings = Util.ReadSettings();
2022-05-31 19:35:01 -08:00
Logger=new Core.Logging.Logger()
{
LogPath=$"RageCoop\\RageCoop.Client.log",
2022-05-31 19:35:01 -08:00
UseConsole=false,
#if DEBUG
LogLevel = 0,
#else
2022-06-02 08:53:01 +08:00
LogLevel=Settings.LogLevel,
2022-05-31 19:35:01 -08:00
#endif
};
2022-06-12 17:11:14 +08:00
Resources = new Scripting.Resources();
// Required for some synchronization!
2022-05-22 15:55:26 +08:00
/*if (Game.Version < GameVersion.v1_0_1290_1_Steam)
{
Tick += (object sender, EventArgs e) =>
{
if (Game.IsLoading)
{
return;
}
2022-05-31 19:35:01 -08:00
if (!_gameLoaded)
{
GTA.UI.Notification.Show("~r~Please update your GTA5 to v1.0.1290 or newer!", true);
_gameLoaded = true;
}
};
return;
2022-05-22 15:55:26 +08:00
}*/
SyncedPedsGroup=World.AddRelationshipGroup("SYNCPED");
Game.Player.Character.RelationshipGroup.SetRelationshipBetweenGroups(SyncedPedsGroup, Relationship.Neutral, true);
#if !NON_INTERACTIVE
#endif
MainChat = new Chat();
2022-05-23 09:15:50 +08:00
2021-07-07 13:36:25 +02:00
Tick += OnTick;
Tick += (s,e) => { Scripting.API.Events.InvokeTick(); };
2021-07-07 13:36:25 +02:00
KeyDown += OnKeyDown;
2021-08-21 16:52:17 +02:00
Aborted += (object sender, EventArgs e) => CleanUp();
2022-05-22 15:55:26 +08:00
2021-07-13 16:32:45 +02:00
Util.NativeMemory();
2022-05-22 15:55:26 +08:00
Counter.Restart();
2021-07-07 13:36:25 +02:00
}
2022-05-22 15:55:26 +08:00
#if DEBUG
private ulong _lastDebugData;
private int _debugBytesSend;
private int _debugBytesReceived;
#endif
2021-07-07 13:36:25 +02:00
private void OnTick(object sender, EventArgs e)
{
if (Game.IsLoading)
{
return;
}
else if (!_gameLoaded && (_gameLoaded = true))
2021-07-07 13:36:25 +02:00
{
#if !NON_INTERACTIVE
GTA.UI.Notification.Show(GTA.UI.NotificationIcon.AllPlayersConf, "RAGECOOP","Welcome!", $"Press ~g~{Main.Settings.MenuKey}~s~ to open the menu.");
#endif
2021-07-07 13:36:25 +02:00
}
#if !NON_INTERACTIVE
CoopMenu.MenuPool.Process();
#endif
2022-05-22 15:55:26 +08:00
DoQueuedActions();
if (!Networking.IsOnServer)
2021-07-07 13:36:25 +02:00
{
return;
}
2022-05-22 15:55:26 +08:00
if (Game.TimeScale!=1)
2022-04-11 15:10:27 +02:00
{
2022-05-22 15:55:26 +08:00
Game.TimeScale=1;
2022-04-11 15:10:27 +02:00
}
2022-05-22 15:55:26 +08:00
try
{
EntityPool.DoSync();
2022-05-22 15:55:26 +08:00
}
catch (Exception ex)
{
2022-05-31 19:35:01 -08:00
Main.Logger.Error(ex);
2022-05-22 15:55:26 +08:00
}
2022-05-22 15:55:26 +08:00
MapLoader.LoadAll();
2021-08-18 11:47:59 +02:00
#if DEBUG
if (Networking.ShowNetworkInfo)
2021-08-18 11:47:59 +02:00
{
ulong time = Util.GetTickCount64();
if (time - _lastDebugData > 1000)
{
_lastDebugData = time;
_debugBytesReceived = Networking.BytesReceived;
Networking.BytesReceived = 0;
_debugBytesSend = Networking.BytesSend;
Networking.BytesSend = 0;
}
new LemonUI.Elements.ScaledText(new PointF(Screen.PrimaryScreen.Bounds.Width / 2, 0), $"L: {Networking.Latency * 1000:N0}ms", 0.5f) { Alignment = GTA.UI.Alignment.Center }.Draw();
new LemonUI.Elements.ScaledText(new PointF(Screen.PrimaryScreen.Bounds.Width / 2, 30), $"R: {Lidgren.Network.NetUtility.ToHumanReadable(_debugBytesReceived)}/s", 0.5f) { Alignment = GTA.UI.Alignment.Center }.Draw();
new LemonUI.Elements.ScaledText(new PointF(Screen.PrimaryScreen.Bounds.Width / 2, 60), $"S: {Lidgren.Network.NetUtility.ToHumanReadable(_debugBytesSend)}/s", 0.5f) { Alignment = GTA.UI.Alignment.Center }.Draw();
2021-08-18 11:47:59 +02:00
}
#endif
2022-05-22 15:55:26 +08:00
2021-07-07 13:36:25 +02:00
MainChat.Tick();
PlayerList.Tick();
if (!Scripting.API.Config.EnableAutoRespawn)
{
Function.Call(Hash.PAUSE_DEATH_ARREST_RESTART, true);
Function.Call(Hash.FORCE_GAME_STATE_PLAYING);
var P = Game.Player.Character;
if (P.IsDead)
{
Function.Call(Hash.TERMINATE_ALL_SCRIPTS_WITH_THIS_NAME, "respawn_controller");
Function.Call(Hash.SET_FADE_OUT_AFTER_DEATH, false);
if (P.Health!=1)
{
P.Health=1;
Game.Player.WantedLevel=0;
Main.Logger.Debug("Player died.");
}
GTA.UI.Screen.StopEffects();
}
else
{
2021-07-07 13:36:25 +02:00
Function.Call(Hash.DISPLAY_HUD, true);
}
2021-07-07 13:36:25 +02:00
}
2022-05-22 15:55:26 +08:00
Ticked++;
2021-07-07 13:36:25 +02:00
}
private void OnKeyDown(object sender, KeyEventArgs e)
{
if (MainChat.Focused)
{
MainChat.OnKeyDown(e.KeyCode);
return;
}
if (Networking.IsOnServer)
{
if (Game.IsControlPressed(GTA.Control.FrontendPause))
{
Function.Call(Hash.ACTIVATE_FRONTEND_MENU, Function.Call<int>(Hash.GET_HASH_KEY, "FE_MENU_VERSION_SP_PAUSE"), false, 0);
return;
}
if (Game.IsControlPressed(GTA.Control.FrontendPauseAlternate)&&Settings.DisableAlternatePause)
{
Function.Call(Hash.ACTIVATE_FRONTEND_MENU, Function.Call<int>(Hash.GET_HASH_KEY, "FE_MENU_VERSION_SP_PAUSE"), false, 0);
return;
}
}
if (e.KeyCode == Settings.MenuKey)
2022-05-22 15:55:26 +08:00
{
if (CoopMenu.MenuPool.AreAnyVisible)
{
CoopMenu.MenuPool.ForEach<LemonUI.Menus.NativeMenu>(x =>
{
if (x.Visible)
{
CoopMenu.LastMenu=x;
x.Visible=false;
}
});
}
else
{
CoopMenu.LastMenu.Visible = true;
}
2022-05-22 15:55:26 +08:00
}
else if (Game.IsControlJustPressed(GTA.Control.MultiplayerInfo))
2022-05-22 15:55:26 +08:00
{
if (Networking.IsOnServer)
{
ulong currentTimestamp = Util.GetTickCount64();
PlayerList.Pressed = (currentTimestamp - PlayerList.Pressed) < 5000 ? (currentTimestamp - 6000) : currentTimestamp;
}
2022-05-22 15:55:26 +08:00
}
else if (Game.IsControlJustPressed(GTA.Control.MpTextChatAll))
2022-05-22 15:55:26 +08:00
{
if (Networking.IsOnServer)
{
MainChat.Focused = true;
}
2022-05-22 15:55:26 +08:00
}
else if (e.KeyCode==Settings.PassengerKey)
2022-05-22 15:55:26 +08:00
{
var P = Game.Player.Character;
if (!P.IsInVehicle())
2022-05-22 15:55:26 +08:00
{
2022-05-23 13:02:28 +08:00
if (P.IsTaskActive(TaskType.CTaskEnterVehicle))
2022-05-22 15:55:26 +08:00
{
P.Task.ClearAll();
2022-05-22 15:55:26 +08:00
}
else
2022-05-22 15:55:26 +08:00
{
var V = World.GetClosestVehicle(P.Position, 50);
if (V!=null)
{
var seat = P.GetNearestSeat(V);
P.Task.EnterVehicle(V, seat);
}
2022-05-22 15:55:26 +08:00
}
}
}
2021-07-07 13:36:25 +02:00
}
2022-05-22 15:55:26 +08:00
public static void CleanUp()
{
2022-05-22 15:55:26 +08:00
MainChat.Clear();
EntityPool.Cleanup();
PlayerList.Cleanup();
Main.LocalPlayerID=default;
}
internal static readonly Dictionary<ulong, byte> CheckNativeHash = new Dictionary<ulong, byte>()
{
{ 0xD49F9B0955C367DE, 1 }, // Entities
{ 0xEF29A16337FACADB, 1 }, //
{ 0xB4AC7D0CF06BFE8F, 1 }, //
{ 0x9B62392B474F44A0, 1 }, //
{ 0x7DD959874C1FD534, 1 }, //
{ 0xAF35D0D2583051B0, 2 }, // Vehicles
{ 0x63C6CCA8E68AE8C8, 2 }, //
{ 0x509D5878EB39E842, 3 }, // Props
{ 0x9A294B2138ABB884, 3 }, //
{ 0x46818D79B1F7499A, 4 }, // Blips
{ 0x5CDE92C702A8FCE7, 4 }, //
{ 0xBE339365C863BD36, 4 }, //
{ 0x5A039BB0BCA604B6, 4 }, //
{ 0x0134F0835AB6BFCB, 5 } // Checkpoints
};
internal static Dictionary<int, byte> ServerItems = new Dictionary<int, byte>();
internal static void CleanUpWorld()
{
if (ServerItems.Count == 0)
{
return;
}
lock (ServerItems)
{
foreach (KeyValuePair<int, byte> item in ServerItems)
{
try
{
switch (item.Value)
{
case 1:
World.GetAllEntities().FirstOrDefault(x => x.Handle == item.Key)?.Delete();
break;
case 2:
World.GetAllVehicles().FirstOrDefault(x => x.Handle == item.Key)?.Delete();
break;
case 3:
World.GetAllProps().FirstOrDefault(x => x.Handle == item.Key)?.Delete();
break;
case 4:
Blip blip = new Blip(item.Key);
if (blip.Exists())
{
blip.Delete();
}
break;
case 5:
Checkpoint checkpoint = new Checkpoint(item.Key);
if (checkpoint.Exists())
{
checkpoint.Delete();
}
break;
}
}
catch
{
2022-03-29 00:28:57 +02:00
GTA.UI.Notification.Show("~r~~h~CleanUpWorld() Error");
2022-05-31 19:35:01 -08:00
Main.Logger.Error($"CleanUpWorld(): ~r~Item {item.Value} cannot be deleted!");
}
}
ServerItems.Clear();
}
}
2022-05-22 15:55:26 +08:00
private static void DoQueuedActions()
{
lock (QueuedActions)
{
foreach (var action in QueuedActions.ToArray())
{
try
{
if (action())
{
QueuedActions.Remove(action);
}
}
catch(Exception ex)
{
GTA.UI.Screen.ShowSubtitle(ex.ToString());
QueuedActions.Remove(action);
}
}
}
}
/// <summary>
/// Queue an action to be executed on next tick, allowing you to call scripting API from another thread.
/// </summary>
/// <param name="a"> The action to be executed, must return a bool indicating whether the action cane be removed after execution.</param>
internal static void QueueAction(Func<bool> a)
2022-05-22 15:55:26 +08:00
{
lock (QueuedActions)
{
QueuedActions.Add(a);
}
}
internal static void QueueAction(Action a)
2022-05-22 15:55:26 +08:00
{
lock (QueuedActions)
{
QueuedActions.Add(() => { a(); return true; }) ;
}
}
/// <summary>
/// Clears all queued actions
/// </summary>
internal static void ClearQueuedActions()
2022-05-22 15:55:26 +08:00
{
lock (QueuedActions) { QueuedActions.Clear(); }
}
2021-07-07 13:36:25 +02:00
}
}