quantum-space-buddies/QSB/Player/QSBPlayerManager.cs
2022-08-14 20:23:02 +01:00

192 lines
5.8 KiB
C#

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;
/// <summary>
/// called right after player is added
/// </summary>
public static Action<PlayerInfo> OnAddPlayer;
/// <summary>
/// called right before player is removed
/// </summary>
public static Action<PlayerInfo> OnRemovePlayer;
public static readonly List<PlayerInfo> 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<PlayerInfo> 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<QSBFlashlight> RemoteFlashlights) GetPlayerFlashlights()
=> (Locator.GetFlashlight(), PlayerList.Where(x => x.FlashLight != null).Select(x => x.FlashLight));
public static (SurveyorProbe LocalProbe, IEnumerable<QSBProbe> RemoteProbes) GetPlayerProbes()
=> (Locator.GetProbe(), PlayerList.Where(x => x.Probe != null).Select(x => x.Probe));
public static IEnumerable<ThrusterLightTracker> 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<PlayerInfo> 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<int, PlayerInfo> _connectionIdToPlayer = new();
public static IEnumerator ValidatePlayers()
{
while (true)
{
if (QSBCore.IsInMultiplayer && QSBCore.IsHost)
{
_connectionIdToPlayer.Clear();
var playersToRemove = new List<PlayerInfo>();
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);
}
}
}