329 lines
11 KiB
C#
329 lines
11 KiB
C#
using MLAPI.Data;
|
|
using MLAPI.NetworkingManagerComponents.Binary;
|
|
using MLAPI.NetworkingManagerComponents.Core;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
|
|
namespace MLAPI.MonoBehaviours.Core
|
|
{
|
|
/// <summary>
|
|
/// A component used to identify that a GameObject is networked
|
|
/// </summary>
|
|
[AddComponentMenu("MLAPI/NetworkedObject", -99)]
|
|
public sealed class NetworkedObject : MonoBehaviour
|
|
{
|
|
private void OnValidate()
|
|
{
|
|
if(string.IsNullOrEmpty(NetworkedPrefabName))
|
|
{
|
|
Debug.LogWarning("MLAPI: The NetworkedObject " + gameObject.name + " does not have a NetworkedPrefabName. It has been set to the gameObject name");
|
|
NetworkedPrefabName = gameObject.name;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the unique ID of this object that is synced across the network
|
|
/// </summary>
|
|
public uint NetworkId
|
|
{
|
|
get
|
|
{
|
|
return networkId;
|
|
}
|
|
}
|
|
internal uint networkId;
|
|
/// <summary>
|
|
/// Gets the clientId of the owner of this NetworkedObject
|
|
/// </summary>
|
|
public uint OwnerClientId
|
|
{
|
|
get
|
|
{
|
|
return ownerClientId;
|
|
}
|
|
}
|
|
private uint? _ownerClientId = null;
|
|
//internal uint ownerClientId = new NetId(0, 0, false, true).GetClientId();
|
|
internal uint ownerClientId
|
|
{
|
|
get
|
|
{
|
|
if (_ownerClientId == null)
|
|
return NetworkingManager.singleton.NetworkConfig.NetworkTransport.InvalidDummyId;
|
|
else
|
|
return _ownerClientId.Value;
|
|
}
|
|
set
|
|
{
|
|
if (value == NetworkingManager.singleton.NetworkConfig.NetworkTransport.InvalidDummyId)
|
|
_ownerClientId = null;
|
|
else
|
|
_ownerClientId = value;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// The name of the NetworkedPrefab
|
|
/// </summary>
|
|
public string NetworkedPrefabName = string.Empty;
|
|
/// <summary>
|
|
/// Gets if this object is a player object
|
|
/// </summary>
|
|
public bool isPlayerObject
|
|
{
|
|
get
|
|
{
|
|
return _isPlayerObject;
|
|
}
|
|
}
|
|
internal bool _isPlayerObject = false;
|
|
/// <summary>
|
|
/// Gets if this object is part of a pool
|
|
/// </summary>
|
|
public bool isPooledObject
|
|
{
|
|
get
|
|
{
|
|
return _isPooledObject;
|
|
}
|
|
}
|
|
internal bool _isPooledObject = false;
|
|
/// <summary>
|
|
/// Gets the poolId this object is part of
|
|
/// </summary>
|
|
public ushort PoolId
|
|
{
|
|
get
|
|
{
|
|
return poolId;
|
|
}
|
|
}
|
|
internal ushort poolId;
|
|
/// <summary>
|
|
/// Gets if the object is the the personal clients player object
|
|
/// </summary>
|
|
public bool isLocalPlayer
|
|
{
|
|
get
|
|
{
|
|
return isPlayerObject && (OwnerClientId == NetworkingManager.singleton.MyClientId || (ownerClientId == NetworkingManager.singleton.NetworkConfig.NetworkTransport.HostDummyId && NetworkingManager.singleton.isHost));
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Gets if the object is owned by the local player
|
|
/// </summary>
|
|
public bool isOwner
|
|
{
|
|
get
|
|
{
|
|
return !isPlayerObject && (OwnerClientId == NetworkingManager.singleton.MyClientId || (ownerClientId == NetworkingManager.singleton.NetworkConfig.NetworkTransport.HostDummyId && NetworkingManager.singleton.isHost));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets if the object has yet been spawned across the network
|
|
/// </summary>
|
|
public bool isSpawned
|
|
{
|
|
get
|
|
{
|
|
return _isSpawned;
|
|
}
|
|
}
|
|
internal bool _isSpawned = false;
|
|
internal bool? sceneObject = null;
|
|
|
|
public HashSet<uint> observers = new HashSet<uint>();
|
|
|
|
internal void RebuildObservers(uint? clientId = null)
|
|
{
|
|
bool initial = clientId != null;
|
|
if (initial)
|
|
{
|
|
bool shouldBeAdded = true;
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
bool state = childNetworkedBehaviours[i].OnCheckObserver(clientId.Value);
|
|
if (state == false)
|
|
{
|
|
shouldBeAdded = false;
|
|
break;
|
|
}
|
|
}
|
|
if (shouldBeAdded)
|
|
observers.Add(clientId.Value);
|
|
}
|
|
else
|
|
{
|
|
HashSet<uint> previousObservers = new HashSet<uint>(observers);
|
|
HashSet<uint> newObservers = new HashSet<uint>();
|
|
bool update = false;
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
bool changed = childNetworkedBehaviours[i].OnRebuildObservers(newObservers);
|
|
if (changed)
|
|
{
|
|
observers = newObservers;
|
|
update = true;
|
|
break;
|
|
}
|
|
}
|
|
if (update)
|
|
{
|
|
foreach (KeyValuePair<uint, NetworkedClient> pair in NetworkingManager.singleton.connectedClients)
|
|
{
|
|
if (pair.Key == NetworkingManager.singleton.NetworkConfig.NetworkTransport.HostDummyId)
|
|
continue;
|
|
if ((previousObservers.Contains(pair.Key) && !newObservers.Contains(pair.Key)) ||
|
|
(!previousObservers.Contains(pair.Key) && newObservers.Contains(pair.Key)))
|
|
{
|
|
//Something changed for this client.
|
|
using (BitWriter writer = new BitWriter())
|
|
{
|
|
writer.WriteUInt(networkId);
|
|
writer.WriteBool(observers.Contains(pair.Key));
|
|
|
|
InternalMessageHandler.Send(pair.Key, "MLAPI_SET_VISIBILITY", "MLAPI_INTERNAL", writer.Finalize(), null);
|
|
}
|
|
FlushToClient(pair.Key);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void SetLocalVisibility(bool visibility)
|
|
{
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
childNetworkedBehaviours[i].OnSetLocalVisibility(visibility);
|
|
}
|
|
}
|
|
|
|
private void OnDestroy()
|
|
{
|
|
if (NetworkingManager.singleton != null)
|
|
SpawnManager.OnDestroyObject(NetworkId, false);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Spawns this GameObject across the network. Can only be called from the Server
|
|
/// </summary>
|
|
public void Spawn()
|
|
{
|
|
if (NetworkingManager.singleton != null)
|
|
SpawnManager.SpawnPrefabIndexServer(this);
|
|
}
|
|
/// <summary>
|
|
/// Spawns an object across the network with a given owner. Can only be called from server
|
|
/// </summary>
|
|
/// <param name="clientId">The clientId to own the object</param>
|
|
public void SpawnWithOwnership(uint clientId)
|
|
{
|
|
if (NetworkingManager.singleton != null)
|
|
SpawnManager.SpawnPrefabIndexServer(this, clientId);
|
|
}
|
|
/// <summary>
|
|
/// Removes all ownership of an object from any client. Can only be called from server
|
|
/// </summary>
|
|
public void RemoveOwnership()
|
|
{
|
|
SpawnManager.RemoveOwnership(NetworkId);
|
|
}
|
|
/// <summary>
|
|
/// Changes the owner of the object. Can only be called from server
|
|
/// </summary>
|
|
/// <param name="newOwnerClientId">The new owner clientId</param>
|
|
public void ChangeOwnership(uint newOwnerClientId)
|
|
{
|
|
SpawnManager.ChangeOwnership(NetworkId, newOwnerClientId);
|
|
}
|
|
|
|
internal void InvokeBehaviourOnLostOwnership()
|
|
{
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
childNetworkedBehaviours[i].OnLostOwnership();
|
|
}
|
|
}
|
|
|
|
internal void InvokeBehaviourOnGainedOwnership()
|
|
{
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
childNetworkedBehaviours[i].OnGainedOwnership();
|
|
}
|
|
}
|
|
|
|
internal void InvokeBehaviourNetworkSpawn()
|
|
{
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
//We check if we are it's networkedObject owner incase a networkedObject exists as a child of our networkedObject.
|
|
if(!childNetworkedBehaviours[i].networkedStartInvoked)
|
|
{
|
|
childNetworkedBehaviours[i].NetworkStart();
|
|
childNetworkedBehaviours[i].SyncVarInit();
|
|
childNetworkedBehaviours[i].networkedStartInvoked = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
private List<NetworkedBehaviour> _childNetworkedBehaviours;
|
|
internal List<NetworkedBehaviour> childNetworkedBehaviours
|
|
{
|
|
get
|
|
{
|
|
if(_childNetworkedBehaviours == null)
|
|
{
|
|
_childNetworkedBehaviours = new List<NetworkedBehaviour>();
|
|
NetworkedBehaviour[] behaviours = GetComponentsInChildren<NetworkedBehaviour>();
|
|
for (int i = 0; i < behaviours.Length; i++)
|
|
{
|
|
if (behaviours[i].networkedObject == this)
|
|
_childNetworkedBehaviours.Add(behaviours[i]);
|
|
}
|
|
}
|
|
return _childNetworkedBehaviours;
|
|
}
|
|
}
|
|
|
|
internal static List<NetworkedBehaviour> NetworkedBehaviours = new List<NetworkedBehaviour>();
|
|
internal static void InvokeSyncvarUpdate()
|
|
{
|
|
for (int i = 0; i < NetworkedBehaviours.Count; i++)
|
|
{
|
|
NetworkedBehaviours[i].SyncVarUpdate();
|
|
}
|
|
}
|
|
|
|
//Flushes all syncVars to client
|
|
internal void FlushToClient(uint clientId)
|
|
{
|
|
for (int i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
childNetworkedBehaviours[i].FlushToClient(clientId);
|
|
}
|
|
}
|
|
|
|
internal ushort GetOrderIndex(NetworkedBehaviour instance)
|
|
{
|
|
for (ushort i = 0; i < childNetworkedBehaviours.Count; i++)
|
|
{
|
|
if (childNetworkedBehaviours[i] == instance)
|
|
return i;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
internal NetworkedBehaviour GetBehaviourAtOrderIndex(ushort index)
|
|
{
|
|
//TODO index out of bounds
|
|
return childNetworkedBehaviours[index];
|
|
}
|
|
|
|
//Key: behaviourOrderId, value key: messageType, value value callback
|
|
internal Dictionary<ushort, Dictionary<ushort, Action<uint, byte[]>>> targetMessageActions = new Dictionary<ushort, Dictionary<ushort, Action<uint, byte[]>>>();
|
|
}
|
|
}
|