using OWML.Common; using QSB.ItemSync.WorldObjects.Items; using QSB.Player.TransformSync; using QSB.Tools.FlashlightTool; using QSB.Tools.ProbeTool; using QSB.Utility; using System; using System.Collections; using System.Collections.Generic; using System.Linq; using UnityEngine; namespace QSB.Player; public static class QSBPlayerManager { public static PlayerInfo LocalPlayer { get { var localInstance = PlayerTransformSync.LocalInstance; if (localInstance == null) { DebugLog.ToConsole("Error - Trying to get LocalPlayer when the local PlayerTransformSync instance is null." + $"{Environment.NewLine} Stacktrace : {Environment.StackTrace} ", MessageType.Error); return null; } return localInstance.Player; } } public static uint LocalPlayerId => LocalPlayer?.PlayerId ?? uint.MaxValue; /// /// called right after player is added /// public static Action OnAddPlayer; /// /// called right before player is removed /// public static Action OnRemovePlayer; public static readonly List PlayerList = new(); public static PlayerInfo GetPlayer(uint id) { if (id is uint.MaxValue or 0) { return default; } var player = PlayerList.FirstOrDefault(x => x.PlayerId == id); if (player == null) { DebugLog.ToConsole($"Error - Player with id {id} does not exist! Stacktrace : {Environment.StackTrace}", MessageType.Error); return default; } return player; } public static bool PlayerExists(uint id) => id is not (uint.MaxValue or 0) && PlayerList.Any(x => x.PlayerId == id); public static List GetPlayersWithCameras(bool includeLocalCamera = true) { var cameraList = PlayerList.Where(x => x.Camera != null && x.PlayerId != LocalPlayerId).ToList(); if (includeLocalCamera && LocalPlayer != default && LocalPlayer.Camera != null) { cameraList.Add(LocalPlayer); } else if (includeLocalCamera && (LocalPlayer == default || LocalPlayer.Camera == null)) { if (LocalPlayer == default) { DebugLog.ToConsole($"Error - LocalPlayer is null.", MessageType.Error); return cameraList; } DebugLog.ToConsole($"Error - LocalPlayer.Camera is null.", MessageType.Error); LocalPlayer.Camera = Locator.GetPlayerCamera(); } return cameraList; } public static (Flashlight LocalFlashlight, IEnumerable RemoteFlashlights) GetPlayerFlashlights() => (Locator.GetFlashlight(), PlayerList.Where(x => x.FlashLight != null).Select(x => x.FlashLight)); public static (SurveyorProbe LocalProbe, IEnumerable RemoteProbes) GetPlayerProbes() => (Locator.GetProbe(), PlayerList.Where(x => x.Probe != null).Select(x => x.Probe)); public static IEnumerable GetThrusterLightTrackers() => PlayerList.Select(x => x.ThrusterLightTracker).Where(x => x != null); public static void ShowAllPlayers(float time = 2f) => PlayerList.ForEach(x => x.SetVisible(true, time)); public static void HideAllPlayers(float time = 2f) => PlayerList.ForEach(x => x.SetVisible(false, time)); public static PlayerInfo GetClosestPlayerToWorldPoint(Vector3 worldPoint, bool includeLocalPlayer) => includeLocalPlayer ? GetClosestPlayerToWorldPoint(PlayerList, worldPoint) : GetClosestPlayerToWorldPoint(PlayerList.Where(x => x != LocalPlayer).ToList(), worldPoint); public static PlayerInfo GetClosestPlayerToWorldPoint(List playerList, Vector3 worldPoint) { if (playerList == null) { DebugLog.ToConsole($"Error - Cannot get closest player from null player list.", MessageType.Error); return null; } playerList = playerList.Where(x => x.IsReady && x.Body != null).ToList(); if (playerList.Count == 0) { DebugLog.ToConsole($"Error - Cannot get closest player from empty (ready) player list.", MessageType.Error); return null; } return playerList.MinBy(x => Vector3.Distance(x.Body.transform.position, worldPoint)); } public static IEnumerable<(PlayerInfo Player, IQSBItem HeldItem)> GetPlayerCarryItems() => PlayerList.Select(x => (x, x.HeldItem)); private static Dictionary _connectionIdToPlayer = new(); public static IEnumerator ValidatePlayers() { while (true) { if (QSBCore.IsInMultiplayer && QSBCore.IsHost) { _connectionIdToPlayer.Clear(); var playersToRemove = new List(); foreach (var player in PlayerList) { var transformSync = player.TransformSync; if (transformSync == null) { DebugLog.ToConsole($"Error - {player.PlayerId}'s TransformSync is null.", MessageType.Error); playersToRemove.Add(player); continue; } var networkIdentity = transformSync.netIdentity; if (networkIdentity == null) { DebugLog.ToConsole($"Error - {player.PlayerId}'s TransformSync's NetworkIdentity is null.", MessageType.Error); playersToRemove.Add(player); continue; } var connectionToClient = networkIdentity.connectionToClient; if (_connectionIdToPlayer.ContainsKey(connectionToClient.connectionId)) { // oh god oh fuck DebugLog.ToConsole($"Error - {player.PlayerId}'s connectionToClient.connectionId is already being used?!?", MessageType.Error); playersToRemove.Add(player); continue; } _connectionIdToPlayer.Add(connectionToClient.connectionId, player); } if (playersToRemove.Count != 0) { DebugLog.DebugWrite($"Removing {playersToRemove.Count} invalid players.", MessageType.Success); foreach (var player in playersToRemove) { OnRemovePlayer?.Invoke(player); player.HudMarker?.Remove(); PlayerList.Remove(player); DebugLog.DebugWrite($"Remove Invalid Player : {player}", MessageType.Info); } } } yield return new WaitForSecondsRealtime(5); } } }