using System; using System.IO; using System.Linq; using System.Collections.Generic; using Microsoft.ClearScript.V8; using GTA; using GTA.Native; using GTA.Math; namespace CoopClient { /// /// Don't use this! /// public class JavascriptHook : Script { private static readonly List ScriptEngines = new List(); /// /// Don't use this! /// public JavascriptHook() { Tick += Ontick; } private void Ontick(object sender, EventArgs e) { if (!Main.MainNetworking.IsOnServer() || ScriptEngines.Count == 0) { return; } lock (ScriptEngines) { ScriptEngines.ForEach(engine => engine.Script.API.InvokeTick()); } } internal static void LoadAll() { string serverAddress = Main.MainSettings.LastServerAddress.Replace(":", "."); if (!Directory.Exists("scripts\\resources\\" + serverAddress)) { try { Directory.CreateDirectory("scripts\\resources\\" + serverAddress); } catch (Exception ex) { GTA.UI.Notification.Show("~r~~h~Javascript Error"); Logger.Write(ex.Message, Logger.LogLevel.Server); // Without the directory we can't do the other stuff return; } } lock (ScriptEngines) { foreach (string script in Directory.GetFiles("scripts\\resources\\" + serverAddress, "*.js")) { V8ScriptEngine engine = new V8ScriptEngine(); engine.AccessContext = typeof(ScriptContext); engine.AddHostObject("API", new ScriptContext()); engine.AddHostType("Dictionary", typeof(Dictionary<,>)); engine.AddHostType("Vector3", typeof(Vector3)); try { engine.Execute(File.ReadAllText(script)); } catch (Exception ex) { GTA.UI.Notification.Show("~r~~h~Javascript Error"); Logger.Write(ex.Message, Logger.LogLevel.Server); } finally { engine.Script.API.InvokeStart(); ScriptEngines.Add(engine); } } } } internal static void StopAll() { lock (ScriptEngines) { ScriptEngines.ForEach(engine => engine.Script.API.InvokeStop()); ScriptEngines.Clear(); } } internal static void InvokePlayerConnect(string username, long nethandle) { lock (ScriptEngines) { ScriptEngines.ForEach(engine => engine.Script.API.InvokePlayerConnect(username, nethandle)); } } internal static void InvokePlayerDisonnect(string username, long nethandle, string reason = null) { lock (ScriptEngines) { ScriptEngines.ForEach(engine => engine.Script.API.InvokePlayerDisonnect(username, nethandle, reason)); } } internal static void InvokeChatMessage(string from, string message) { lock (ScriptEngines) { ScriptEngines.ForEach(engine => engine.Script.API.InvokeChatMessage(from, message)); } } } internal class ScriptContext { #region DELEGATES public delegate void EmptyEvent(); public delegate void PlayerConnectEvent(string username, long nethandle, string reason); public delegate void ChatMessageEvent(string from, string message); #endregion #region EVENTS public event EmptyEvent OnStart, OnStop, OnTick; public event PlayerConnectEvent OnPlayerConnect, OnPlayerDisconnect; public event ChatMessageEvent OnChatMessage; internal void InvokeStart() { OnStart?.Invoke(); } internal void InvokeStop() { OnStop?.Invoke(); } internal void InvokeTick() { OnTick?.Invoke(); } internal void InvokePlayerConnect(string username, long nethandle) { OnPlayerConnect?.Invoke(username, nethandle, null); } internal void InvokePlayerDisonnect(string username, long nethandle, string reason) { OnPlayerDisconnect?.Invoke(username, nethandle, reason); } internal void InvokeChatMessage(string from, string message) { OnChatMessage?.Invoke(from, message); } #endregion /* ===== PLAYER STUFF ===== */ public void SendLocalMessage(string message) { Main.MainChat.AddMessage("JAVASCRIPT", message); } public string GetLocalUsername() { return Main.MainSettings.Username; } public long GetLocalNetHandle() { return Main.LocalNetHandle; } // This only applies to server-side created objects public void CleanUpWorld() { Main.CleanUpWorld(); } // This create an object to delete it with CleanUpWorld() or on disconnect public void CreateObject(string hash, params object[] args) { if (!Hash.TryParse(hash, out Hash ourHash) || !Main.CheckNativeHash.ContainsKey((ulong)ourHash)) { GTA.UI.Notification.Show("~r~~h~Javascript Error"); Logger.Write($"Hash \"{ourHash}\" has not been found!", Logger.LogLevel.Server); return; } int result = Function.Call(ourHash, args.Select(o => new InputArgument(o)).ToArray()); foreach (KeyValuePair checkHash in Main.CheckNativeHash) { if (checkHash.Key == (ulong)ourHash) { lock (Main.ServerItems) { Main.ServerItems.Add(result, checkHash.Value); } break; } } } public void SendNotification(string message) { GTA.UI.Notification.Show(message); } public void SendNotification(string[] messages) { SendNotification(string.Concat(messages)); } public Vector3 GetPlayerPosition() { return Game.Player.Character.Position; } public Vector3 GetPlayerRotation() { return Game.Player.Character.Rotation; } public int GetCurrentCharachterHandle() { return Game.Player.Character?.Handle ?? 0; } public int GetCurrentVehicleHandle() { return Game.Player.Character?.CurrentVehicle?.Handle ?? 0; } public int GetCurrentVehicleSeatIndex() { return Game.Player.Character?.CurrentVehicle != null ? (int)Game.Player.Character?.SeatIndex : 0; } /* ===== OTHER PLAYER STUFF ===== */ public Vector3 GetPlayerPosition(long nethandle) { lock (Main.Players) { return Main.Players.ContainsKey(nethandle) ? Main.Players.First(x => x.Key == nethandle).Value.Position : default; } } public Vector3 GetPlayerRotation(long nethandle) { lock (Main.Players) { return Main.Players.ContainsKey(nethandle) ? Main.Players.First(x => x.Key == nethandle).Value.Rotation : default; } } // Get nethandle and charachter handle public Dictionary GetAllNearbyPlayers(float distance) { Dictionary result = new Dictionary(); lock (Main.Players) { Vector3 localPosition = Game.Player.Character.Position; foreach (KeyValuePair player in Main.Players) { if (player.Value.Position.DistanceTo2D(localPosition) < distance) { // Character handle = 0 (if no character exists) result.Add(player.Key, player.Value.Character?.Handle ?? 0); } } } return result; } public long GetNetHandleByUsername(string username) { lock (Main.Players) { return Main.Players.FirstOrDefault(x => x.Value.Username == username).Key; } } } }