mirror of
https://github.com/misternebula/quantum-space-buddies.git
synced 2025-01-25 15:35:22 +00:00
728 lines
24 KiB
C#
728 lines
24 KiB
C#
using QuantumUNET.Components;
|
|
using QuantumUNET.Messages;
|
|
using QuantumUNET.Transport;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
using UnityEngine.Networking;
|
|
|
|
namespace QuantumUNET
|
|
{
|
|
public class QSBClientScene
|
|
{
|
|
internal static void SetNotReady() => ready = false;
|
|
|
|
public static List<QSBPlayerController> localPlayers { get; private set; } = new List<QSBPlayerController>();
|
|
|
|
public static bool ready { get; private set; }
|
|
|
|
public static QSBNetworkConnection readyConnection { get; private set; }
|
|
|
|
public static int reconnectId { get; private set; } = -1;
|
|
|
|
public static Dictionary<NetworkInstanceId, QSBNetworkIdentity> Objects => s_NetworkScene.localObjects;
|
|
|
|
public static Dictionary<NetworkHash128, GameObject> Prefabs => QSBNetworkScene.guidToPrefab;
|
|
|
|
public static Dictionary<NetworkSceneId, QSBNetworkIdentity> SpawnableObjects { get; private set; }
|
|
|
|
internal static void Shutdown()
|
|
{
|
|
s_NetworkScene.Shutdown();
|
|
localPlayers = new List<QSBPlayerController>();
|
|
s_PendingOwnerIds = new List<PendingOwner>();
|
|
SpawnableObjects = null;
|
|
readyConnection = null;
|
|
ready = false;
|
|
s_IsSpawnFinished = false;
|
|
reconnectId = -1;
|
|
NetworkTransport.Shutdown();
|
|
NetworkTransport.Init();
|
|
}
|
|
|
|
internal static bool GetPlayerController(short playerControllerId, out QSBPlayerController player)
|
|
{
|
|
player = null;
|
|
bool result;
|
|
if (playerControllerId >= localPlayers.Count)
|
|
{
|
|
Debug.Log("ClientScene::GetPlayer: no local player found for: " + playerControllerId);
|
|
result = false;
|
|
}
|
|
else if (localPlayers[playerControllerId] == null)
|
|
{
|
|
Debug.LogWarning("ClientScene::GetPlayer: local player is null for: " + playerControllerId);
|
|
result = false;
|
|
}
|
|
else
|
|
{
|
|
player = localPlayers[playerControllerId];
|
|
result = (player.Gameobject != null);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
internal static void InternalAddPlayer(QSBNetworkIdentity view, short playerControllerId)
|
|
{
|
|
Debug.LogWarning("ClientScene::InternalAddPlayer: playerControllerId : " + playerControllerId);
|
|
if (playerControllerId >= localPlayers.Count)
|
|
{
|
|
Debug.LogWarning("ClientScene::InternalAddPlayer: playerControllerId higher than expected: " + playerControllerId);
|
|
while (playerControllerId >= localPlayers.Count)
|
|
{
|
|
localPlayers.Add(new QSBPlayerController());
|
|
}
|
|
}
|
|
var playerController = new QSBPlayerController
|
|
{
|
|
Gameobject = view.gameObject,
|
|
PlayerControllerId = playerControllerId,
|
|
UnetView = view
|
|
};
|
|
localPlayers[playerControllerId] = playerController;
|
|
readyConnection.SetPlayerController(playerController);
|
|
}
|
|
|
|
public static bool AddPlayer(short playerControllerId) => AddPlayer(null, playerControllerId);
|
|
|
|
public static bool AddPlayer(QSBNetworkConnection readyConn, short playerControllerId) => AddPlayer(readyConn, playerControllerId, null);
|
|
|
|
public static bool AddPlayer(QSBNetworkConnection readyConn, short playerControllerId, QSBMessageBase extraMessage)
|
|
{
|
|
bool result;
|
|
if (playerControllerId < 0)
|
|
{
|
|
Debug.LogError("ClientScene::AddPlayer: playerControllerId of " + playerControllerId + " is negative");
|
|
result = false;
|
|
}
|
|
else if (playerControllerId > 32)
|
|
{
|
|
Debug.LogError(string.Concat(new object[]
|
|
{
|
|
"ClientScene::AddPlayer: playerControllerId of ",
|
|
playerControllerId,
|
|
" is too high, max is ",
|
|
32
|
|
}));
|
|
result = false;
|
|
}
|
|
else
|
|
{
|
|
if (playerControllerId > 16)
|
|
{
|
|
Debug.LogWarning("ClientScene::AddPlayer: playerControllerId of " + playerControllerId + " is unusually high");
|
|
}
|
|
while (playerControllerId >= localPlayers.Count)
|
|
{
|
|
localPlayers.Add(new QSBPlayerController());
|
|
}
|
|
if (readyConn == null)
|
|
{
|
|
if (!ready)
|
|
{
|
|
Debug.LogError("Must call AddPlayer() with a connection the first time to become ready.");
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ready = true;
|
|
readyConnection = readyConn;
|
|
}
|
|
if (readyConnection.GetPlayerController(playerControllerId, out var playerController))
|
|
{
|
|
if (playerController.IsValid && playerController.Gameobject != null)
|
|
{
|
|
Debug.LogError("ClientScene::AddPlayer: playerControllerId of " + playerControllerId + " already in use.");
|
|
return false;
|
|
}
|
|
}
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"ClientScene::AddPlayer() for ID ",
|
|
playerControllerId,
|
|
" called with connection [",
|
|
readyConnection,
|
|
"]"
|
|
}));
|
|
var addPlayerMessage = new QSBAddPlayerMessage
|
|
{
|
|
playerControllerId = playerControllerId
|
|
};
|
|
if (extraMessage != null)
|
|
{
|
|
var networkWriter = new QSBNetworkWriter();
|
|
extraMessage.Serialize(networkWriter);
|
|
addPlayerMessage.msgData = networkWriter.ToArray();
|
|
addPlayerMessage.msgSize = networkWriter.Position;
|
|
}
|
|
readyConnection.Send(37, addPlayerMessage);
|
|
result = true;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public static bool RemovePlayer(short playerControllerId)
|
|
{
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"ClientScene::RemovePlayer() for ID ",
|
|
playerControllerId,
|
|
" called with connection [",
|
|
readyConnection,
|
|
"]"
|
|
}));
|
|
bool result;
|
|
if (readyConnection.GetPlayerController(playerControllerId, out var playerController))
|
|
{
|
|
var removePlayerMessage = new QSBRemovePlayerMessage
|
|
{
|
|
PlayerControllerId = playerControllerId
|
|
};
|
|
readyConnection.Send(38, removePlayerMessage);
|
|
readyConnection.RemovePlayerController(playerControllerId);
|
|
localPlayers[playerControllerId] = new QSBPlayerController();
|
|
Object.Destroy(playerController.Gameobject);
|
|
result = true;
|
|
}
|
|
else
|
|
{
|
|
Debug.LogError("Failed to find player ID " + playerControllerId);
|
|
result = false;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public static bool Ready(QSBNetworkConnection conn)
|
|
{
|
|
bool result;
|
|
if (ready)
|
|
{
|
|
Debug.LogError("A connection has already been set as ready. There can only be one.");
|
|
result = false;
|
|
}
|
|
else
|
|
{
|
|
Debug.Log("ClientScene::Ready() called with connection [" + conn + "]");
|
|
if (conn != null)
|
|
{
|
|
var msg = new QSBReadyMessage();
|
|
conn.Send(35, msg);
|
|
ready = true;
|
|
readyConnection = conn;
|
|
readyConnection.isReady = true;
|
|
result = true;
|
|
}
|
|
else
|
|
{
|
|
Debug.LogError("Ready() called with invalid connection object: conn=null");
|
|
result = false;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public static QSBNetworkClient ConnectLocalServer()
|
|
{
|
|
var localClient = new QSBLocalClient();
|
|
QSBNetworkServer.instance.ActivateLocalClientScene();
|
|
localClient.InternalConnectLocalServer(true);
|
|
return localClient;
|
|
}
|
|
|
|
internal static QSBNetworkClient ReconnectLocalServer()
|
|
{
|
|
var localClient = new QSBLocalClient();
|
|
QSBNetworkServer.instance.ActivateLocalClientScene();
|
|
localClient.InternalConnectLocalServer(false);
|
|
return localClient;
|
|
}
|
|
|
|
internal static void ClearLocalPlayers() => localPlayers.Clear();
|
|
|
|
internal static void HandleClientDisconnect(QSBNetworkConnection conn)
|
|
{
|
|
if (readyConnection == conn && ready)
|
|
{
|
|
ready = false;
|
|
readyConnection = null;
|
|
}
|
|
}
|
|
|
|
internal static void PrepareToSpawnSceneObjects()
|
|
{
|
|
SpawnableObjects = new Dictionary<NetworkSceneId, QSBNetworkIdentity>();
|
|
foreach (var networkIdentity in Resources.FindObjectsOfTypeAll<QSBNetworkIdentity>())
|
|
{
|
|
if (!networkIdentity.gameObject.activeSelf)
|
|
{
|
|
if (networkIdentity.gameObject.hideFlags != HideFlags.NotEditable && networkIdentity.gameObject.hideFlags != HideFlags.HideAndDontSave)
|
|
{
|
|
if (!networkIdentity.SceneId.IsEmpty())
|
|
{
|
|
SpawnableObjects[networkIdentity.SceneId] = networkIdentity;
|
|
Debug.Log("ClientScene::PrepareSpawnObjects sceneId:" + networkIdentity.SceneId);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal static QSBNetworkIdentity SpawnSceneObject(NetworkSceneId sceneId)
|
|
{
|
|
QSBNetworkIdentity result;
|
|
if (SpawnableObjects.ContainsKey(sceneId))
|
|
{
|
|
var networkIdentity = SpawnableObjects[sceneId];
|
|
SpawnableObjects.Remove(sceneId);
|
|
result = networkIdentity;
|
|
}
|
|
else
|
|
{
|
|
result = null;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
internal static void RegisterSystemHandlers(QSBNetworkClient client, bool localClient)
|
|
{
|
|
if (localClient)
|
|
{
|
|
client.RegisterHandlerSafe(1, new QSBNetworkMessageDelegate(OnLocalClientObjectDestroy));
|
|
client.RegisterHandlerSafe(13, new QSBNetworkMessageDelegate(OnLocalClientObjectHide));
|
|
client.RegisterHandlerSafe(3, new QSBNetworkMessageDelegate(OnLocalClientObjectSpawn));
|
|
client.RegisterHandlerSafe(10, new QSBNetworkMessageDelegate(OnLocalClientObjectSpawnScene));
|
|
client.RegisterHandlerSafe(15, new QSBNetworkMessageDelegate(OnClientAuthority));
|
|
}
|
|
else
|
|
{
|
|
client.RegisterHandlerSafe(3, new QSBNetworkMessageDelegate(OnObjectSpawn));
|
|
client.RegisterHandlerSafe(10, new QSBNetworkMessageDelegate(OnObjectSpawnScene));
|
|
client.RegisterHandlerSafe(12, new QSBNetworkMessageDelegate(OnObjectSpawnFinished));
|
|
client.RegisterHandlerSafe(1, new QSBNetworkMessageDelegate(OnObjectDestroy));
|
|
client.RegisterHandlerSafe(13, new QSBNetworkMessageDelegate(OnObjectDestroy));
|
|
client.RegisterHandlerSafe(8, new QSBNetworkMessageDelegate(OnUpdateVarsMessage));
|
|
client.RegisterHandlerSafe(4, new QSBNetworkMessageDelegate(OnOwnerMessage));
|
|
client.RegisterHandlerSafe(9, new QSBNetworkMessageDelegate(OnSyncListMessage));
|
|
client.RegisterHandlerSafe(40, new QSBNetworkMessageDelegate(QSBNetworkAnimator.OnAnimationClientMessage));
|
|
client.RegisterHandlerSafe(41, new QSBNetworkMessageDelegate(QSBNetworkAnimator.OnAnimationParametersClientMessage));
|
|
client.RegisterHandlerSafe(15, new QSBNetworkMessageDelegate(OnClientAuthority));
|
|
}
|
|
client.RegisterHandlerSafe(2, new QSBNetworkMessageDelegate(OnRPCMessage));
|
|
client.RegisterHandlerSafe(7, new QSBNetworkMessageDelegate(OnSyncEventMessage));
|
|
client.RegisterHandlerSafe(42, new QSBNetworkMessageDelegate(QSBNetworkAnimator.OnAnimationTriggerClientMessage));
|
|
}
|
|
|
|
internal static string GetStringForAssetId(NetworkHash128 assetId)
|
|
{
|
|
string result;
|
|
if (QSBNetworkScene.GetPrefab(assetId, out var gameObject))
|
|
{
|
|
result = gameObject.name;
|
|
}
|
|
else if (QSBNetworkScene.GetSpawnHandler(assetId, out var func))
|
|
{
|
|
result = func.GetMethodName();
|
|
}
|
|
else
|
|
{
|
|
result = "unknown";
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public static void RegisterPrefab(GameObject prefab, NetworkHash128 newAssetId) => QSBNetworkScene.RegisterPrefab(prefab, newAssetId);
|
|
|
|
public static void RegisterPrefab(GameObject prefab) => QSBNetworkScene.RegisterPrefab(prefab);
|
|
|
|
public static void RegisterPrefab(GameObject prefab, QSBSpawnDelegate spawnHandler, UnSpawnDelegate unspawnHandler) => QSBNetworkScene.RegisterPrefab(prefab, spawnHandler, unspawnHandler);
|
|
|
|
public static void UnregisterPrefab(GameObject prefab) => QSBNetworkScene.UnregisterPrefab(prefab);
|
|
|
|
public static void RegisterSpawnHandler(NetworkHash128 assetId, QSBSpawnDelegate spawnHandler, UnSpawnDelegate unspawnHandler) => QSBNetworkScene.RegisterSpawnHandler(assetId, spawnHandler, unspawnHandler);
|
|
|
|
public static void UnregisterSpawnHandler(NetworkHash128 assetId) => QSBNetworkScene.UnregisterSpawnHandler(assetId);
|
|
|
|
public static void ClearSpawners() => QSBNetworkScene.ClearSpawners();
|
|
|
|
public static void DestroyAllClientObjects() => s_NetworkScene.DestroyAllClientObjects();
|
|
|
|
public static void SetLocalObject(NetworkInstanceId netId, GameObject obj) => s_NetworkScene.SetLocalObject(netId, obj, s_IsSpawnFinished, false);
|
|
|
|
public static GameObject FindLocalObject(NetworkInstanceId netId) => s_NetworkScene.FindLocalObject(netId);
|
|
|
|
private static void ApplySpawnPayload(QSBNetworkIdentity uv, Vector3 position, byte[] payload, NetworkInstanceId netId, GameObject newGameObject)
|
|
{
|
|
if (!uv.gameObject.activeSelf)
|
|
{
|
|
uv.gameObject.SetActive(true);
|
|
}
|
|
uv.transform.position = position;
|
|
if (payload != null && payload.Length > 0)
|
|
{
|
|
var reader = new QSBNetworkReader(payload);
|
|
uv.OnUpdateVars(reader, true);
|
|
}
|
|
if (!(newGameObject == null))
|
|
{
|
|
newGameObject.SetActive(true);
|
|
uv.SetNetworkInstanceId(netId);
|
|
SetLocalObject(netId, newGameObject);
|
|
if (s_IsSpawnFinished)
|
|
{
|
|
uv.OnStartClient();
|
|
CheckForOwner(uv);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void OnObjectSpawn(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectSpawnMessage>(s_ObjectSpawnMessage);
|
|
if (!s_ObjectSpawnMessage.assetId.IsValid())
|
|
{
|
|
Debug.LogError($"OnObjSpawn netId: {s_ObjectSpawnMessage.NetId} has invalid asset Id. {s_ObjectSpawnMessage.assetId}");
|
|
}
|
|
else
|
|
{
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectSpawnMessage.NetId, out var component))
|
|
{
|
|
ApplySpawnPayload(component, s_ObjectSpawnMessage.Position, s_ObjectSpawnMessage.Payload, s_ObjectSpawnMessage.NetId, null);
|
|
}
|
|
else if (QSBNetworkScene.GetPrefab(s_ObjectSpawnMessage.assetId, out var original))
|
|
{
|
|
var gameObject = Object.Instantiate<GameObject>(original, s_ObjectSpawnMessage.Position, s_ObjectSpawnMessage.Rotation);
|
|
component = gameObject.GetComponent<QSBNetworkIdentity>();
|
|
if (component == null)
|
|
{
|
|
Debug.LogError("Client object spawned for " + s_ObjectSpawnMessage.assetId + " does not have a NetworkIdentity");
|
|
}
|
|
else
|
|
{
|
|
component.Reset();
|
|
ApplySpawnPayload(component, s_ObjectSpawnMessage.Position, s_ObjectSpawnMessage.Payload, s_ObjectSpawnMessage.NetId, gameObject);
|
|
}
|
|
}
|
|
else if (QSBNetworkScene.GetSpawnHandler(s_ObjectSpawnMessage.assetId, out var spawnDelegate))
|
|
{
|
|
var gameObject2 = spawnDelegate(s_ObjectSpawnMessage.Position, s_ObjectSpawnMessage.assetId);
|
|
if (gameObject2 == null)
|
|
{
|
|
Debug.LogWarning("Client spawn handler for " + s_ObjectSpawnMessage.assetId + " returned null");
|
|
}
|
|
else
|
|
{
|
|
component = gameObject2.GetComponent<QSBNetworkIdentity>();
|
|
if (component == null)
|
|
{
|
|
Debug.LogError("Client object spawned for " + s_ObjectSpawnMessage.assetId + " does not have a network identity");
|
|
}
|
|
else
|
|
{
|
|
component.Reset();
|
|
component.SetDynamicAssetId(s_ObjectSpawnMessage.assetId);
|
|
ApplySpawnPayload(component, s_ObjectSpawnMessage.Position, s_ObjectSpawnMessage.Payload, s_ObjectSpawnMessage.NetId, gameObject2);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Debug.LogError(string.Concat(new object[]
|
|
{
|
|
"Failed to spawn server object, did you forget to add it to the QSBNetworkManager? assetId=",
|
|
s_ObjectSpawnMessage.assetId,
|
|
" netId=",
|
|
s_ObjectSpawnMessage.NetId
|
|
}));
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void OnObjectSpawnScene(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectSpawnSceneMessage>(s_ObjectSpawnSceneMessage);
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectSpawnSceneMessage.NetId, out var networkIdentity))
|
|
{
|
|
ApplySpawnPayload(networkIdentity, s_ObjectSpawnSceneMessage.Position, s_ObjectSpawnSceneMessage.Payload, s_ObjectSpawnSceneMessage.NetId, networkIdentity.gameObject);
|
|
}
|
|
else
|
|
{
|
|
var networkIdentity2 = SpawnSceneObject(s_ObjectSpawnSceneMessage.SceneId);
|
|
if (networkIdentity2 == null)
|
|
{
|
|
Debug.LogError("Spawn scene object not found for " + s_ObjectSpawnSceneMessage.SceneId);
|
|
}
|
|
else
|
|
{
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"Client spawn for [netId:",
|
|
s_ObjectSpawnSceneMessage.NetId,
|
|
"] [sceneId:",
|
|
s_ObjectSpawnSceneMessage.SceneId,
|
|
"] obj:",
|
|
networkIdentity2.gameObject.name
|
|
}));
|
|
ApplySpawnPayload(networkIdentity2, s_ObjectSpawnSceneMessage.Position, s_ObjectSpawnSceneMessage.Payload, s_ObjectSpawnSceneMessage.NetId, networkIdentity2.gameObject);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void OnObjectSpawnFinished(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectSpawnFinishedMessage>(s_ObjectSpawnFinishedMessage);
|
|
Debug.Log("SpawnFinished:" + s_ObjectSpawnFinishedMessage.State);
|
|
if (s_ObjectSpawnFinishedMessage.State == 0U)
|
|
{
|
|
PrepareToSpawnSceneObjects();
|
|
s_IsSpawnFinished = false;
|
|
}
|
|
else
|
|
{
|
|
foreach (var networkIdentity in Objects.Values)
|
|
{
|
|
if (!networkIdentity.IsClient)
|
|
{
|
|
networkIdentity.OnStartClient();
|
|
CheckForOwner(networkIdentity);
|
|
}
|
|
}
|
|
s_IsSpawnFinished = true;
|
|
}
|
|
}
|
|
|
|
private static void OnObjectDestroy(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectDestroyMessage>(s_ObjectDestroyMessage);
|
|
Debug.Log("ClientScene::OnObjDestroy netId:" + s_ObjectDestroyMessage.NetId);
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectDestroyMessage.NetId, out var networkIdentity))
|
|
{
|
|
networkIdentity.OnNetworkDestroy();
|
|
if (!QSBNetworkScene.InvokeUnSpawnHandler(networkIdentity.AssetId, networkIdentity.gameObject))
|
|
{
|
|
if (networkIdentity.SceneId.IsEmpty())
|
|
{
|
|
Object.Destroy(networkIdentity.gameObject);
|
|
}
|
|
else
|
|
{
|
|
networkIdentity.gameObject.SetActive(false);
|
|
SpawnableObjects[networkIdentity.SceneId] = networkIdentity;
|
|
}
|
|
}
|
|
s_NetworkScene.RemoveLocalObject(s_ObjectDestroyMessage.NetId);
|
|
networkIdentity.MarkForReset();
|
|
}
|
|
else
|
|
{
|
|
Debug.LogWarning("Did not find target for destroy message for " + s_ObjectDestroyMessage.NetId);
|
|
}
|
|
}
|
|
|
|
private static void OnLocalClientObjectDestroy(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectDestroyMessage>(s_ObjectDestroyMessage);
|
|
Debug.Log("ClientScene::OnLocalObjectObjDestroy netId:" + s_ObjectDestroyMessage.NetId);
|
|
s_NetworkScene.RemoveLocalObject(s_ObjectDestroyMessage.NetId);
|
|
}
|
|
|
|
private static void OnLocalClientObjectHide(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectDestroyMessage>(s_ObjectDestroyMessage);
|
|
Debug.Log("ClientScene::OnLocalObjectObjHide netId:" + s_ObjectDestroyMessage.NetId);
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectDestroyMessage.NetId, out var networkIdentity))
|
|
{
|
|
networkIdentity.OnSetLocalVisibility(false);
|
|
}
|
|
}
|
|
|
|
private static void OnLocalClientObjectSpawn(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectSpawnMessage>(s_ObjectSpawnMessage);
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectSpawnMessage.NetId, out var networkIdentity))
|
|
{
|
|
networkIdentity.OnSetLocalVisibility(true);
|
|
}
|
|
}
|
|
|
|
private static void OnLocalClientObjectSpawnScene(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBObjectSpawnSceneMessage>(s_ObjectSpawnSceneMessage);
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ObjectSpawnSceneMessage.NetId, out var networkIdentity))
|
|
{
|
|
networkIdentity.OnSetLocalVisibility(true);
|
|
}
|
|
}
|
|
|
|
private static void OnUpdateVarsMessage(QSBNetworkMessage netMsg)
|
|
{
|
|
var networkInstanceId = netMsg.Reader.ReadNetworkId();
|
|
if (s_NetworkScene.GetNetworkIdentity(networkInstanceId, out var networkIdentity))
|
|
{
|
|
networkIdentity.OnUpdateVars(netMsg.Reader, false);
|
|
}
|
|
else
|
|
{
|
|
Debug.LogWarning("Did not find target for sync message for " + networkInstanceId);
|
|
}
|
|
}
|
|
|
|
private static void OnRPCMessage(QSBNetworkMessage netMsg)
|
|
{
|
|
var num = (int)netMsg.Reader.ReadPackedUInt32();
|
|
var networkInstanceId = netMsg.Reader.ReadNetworkId();
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"ClientScene::OnRPCMessage hash:",
|
|
num,
|
|
" netId:",
|
|
networkInstanceId
|
|
}));
|
|
if (s_NetworkScene.GetNetworkIdentity(networkInstanceId, out var networkIdentity))
|
|
{
|
|
networkIdentity.HandleRPC(num, netMsg.Reader);
|
|
}
|
|
else
|
|
{
|
|
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(num);
|
|
Debug.LogWarningFormat("Could not find target object with netId:{0} for RPC call {1}", new object[]
|
|
{
|
|
networkInstanceId,
|
|
cmdHashHandlerName
|
|
});
|
|
}
|
|
}
|
|
|
|
private static void OnSyncEventMessage(QSBNetworkMessage netMsg)
|
|
{
|
|
var cmdHash = (int)netMsg.Reader.ReadPackedUInt32();
|
|
var networkInstanceId = netMsg.Reader.ReadNetworkId();
|
|
Debug.Log("ClientScene::OnSyncEventMessage " + networkInstanceId);
|
|
if (s_NetworkScene.GetNetworkIdentity(networkInstanceId, out var networkIdentity))
|
|
{
|
|
networkIdentity.HandleSyncEvent(cmdHash, netMsg.Reader);
|
|
}
|
|
else
|
|
{
|
|
Debug.LogWarning("Did not find target for SyncEvent message for " + networkInstanceId);
|
|
}
|
|
}
|
|
|
|
private static void OnSyncListMessage(QSBNetworkMessage netMsg)
|
|
{
|
|
var networkInstanceId = netMsg.Reader.ReadNetworkId();
|
|
var cmdHash = (int)netMsg.Reader.ReadPackedUInt32();
|
|
Debug.Log("ClientScene::OnSyncListMessage " + networkInstanceId);
|
|
if (s_NetworkScene.GetNetworkIdentity(networkInstanceId, out var networkIdentity))
|
|
{
|
|
networkIdentity.HandleSyncList(cmdHash, netMsg.Reader);
|
|
}
|
|
else
|
|
{
|
|
Debug.LogWarning("Did not find target for SyncList message for " + networkInstanceId);
|
|
}
|
|
}
|
|
|
|
private static void OnClientAuthority(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBClientAuthorityMessage>(s_ClientAuthorityMessage);
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"ClientScene::OnClientAuthority for connectionId=",
|
|
netMsg.Connection.connectionId,
|
|
" netId: ",
|
|
s_ClientAuthorityMessage.netId
|
|
}));
|
|
if (s_NetworkScene.GetNetworkIdentity(s_ClientAuthorityMessage.netId, out var networkIdentity))
|
|
{
|
|
networkIdentity.HandleClientAuthority(s_ClientAuthorityMessage.authority);
|
|
}
|
|
}
|
|
|
|
private static void OnOwnerMessage(QSBNetworkMessage netMsg)
|
|
{
|
|
netMsg.ReadMessage<QSBOwnerMessage>(s_OwnerMessage);
|
|
Debug.Log(string.Concat(new object[]
|
|
{
|
|
"ClientScene::OnOwnerMessage - connectionId=",
|
|
netMsg.Connection.connectionId,
|
|
" netId: ",
|
|
s_OwnerMessage.NetId
|
|
}));
|
|
if (netMsg.Connection.GetPlayerController(s_OwnerMessage.PlayerControllerId, out var playerController))
|
|
{
|
|
playerController.UnetView.SetNotLocalPlayer();
|
|
}
|
|
if (s_NetworkScene.GetNetworkIdentity(s_OwnerMessage.NetId, out var networkIdentity))
|
|
{
|
|
networkIdentity.SetConnectionToServer(netMsg.Connection);
|
|
networkIdentity.SetLocalPlayer(s_OwnerMessage.PlayerControllerId);
|
|
InternalAddPlayer(networkIdentity, s_OwnerMessage.PlayerControllerId);
|
|
}
|
|
else
|
|
{
|
|
var item = new PendingOwner
|
|
{
|
|
netId = s_OwnerMessage.NetId,
|
|
playerControllerId = s_OwnerMessage.PlayerControllerId
|
|
};
|
|
s_PendingOwnerIds.Add(item);
|
|
}
|
|
}
|
|
|
|
private static void CheckForOwner(QSBNetworkIdentity uv)
|
|
{
|
|
var i = 0;
|
|
while (i < s_PendingOwnerIds.Count)
|
|
{
|
|
var pendingOwner = s_PendingOwnerIds[i];
|
|
if (pendingOwner.netId == uv.NetId)
|
|
{
|
|
uv.SetConnectionToServer(readyConnection);
|
|
uv.SetLocalPlayer(pendingOwner.playerControllerId);
|
|
Debug.Log("ClientScene::OnOwnerMessage - player=" + uv.gameObject.name);
|
|
if (readyConnection.connectionId < 0)
|
|
{
|
|
Debug.LogError("Owner message received on a local client.");
|
|
break;
|
|
}
|
|
InternalAddPlayer(uv, pendingOwner.playerControllerId);
|
|
s_PendingOwnerIds.RemoveAt(i);
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
private static bool s_IsSpawnFinished;
|
|
|
|
private static readonly QSBNetworkScene s_NetworkScene = new QSBNetworkScene();
|
|
|
|
private static readonly QSBObjectSpawnSceneMessage s_ObjectSpawnSceneMessage = new QSBObjectSpawnSceneMessage();
|
|
|
|
private static readonly QSBObjectSpawnFinishedMessage s_ObjectSpawnFinishedMessage = new QSBObjectSpawnFinishedMessage();
|
|
|
|
private static readonly QSBObjectDestroyMessage s_ObjectDestroyMessage = new QSBObjectDestroyMessage();
|
|
|
|
private static readonly QSBObjectSpawnMessage s_ObjectSpawnMessage = new QSBObjectSpawnMessage();
|
|
|
|
private static readonly QSBOwnerMessage s_OwnerMessage = new QSBOwnerMessage();
|
|
|
|
private static readonly QSBClientAuthorityMessage s_ClientAuthorityMessage = new QSBClientAuthorityMessage();
|
|
|
|
public const int ReconnectIdInvalid = -1;
|
|
|
|
public const int ReconnectIdHost = 0;
|
|
|
|
private static List<PendingOwner> s_PendingOwnerIds = new List<PendingOwner>();
|
|
|
|
private struct PendingOwner
|
|
{
|
|
public NetworkInstanceId netId;
|
|
|
|
public short playerControllerId;
|
|
}
|
|
}
|
|
} |