StationObscurum/Assets/FishNet/Runtime/Object/Synchronizing/SyncDictionary.cs

629 lines
22 KiB
C#
Raw Normal View History

2023-05-31 17:32:21 +02:00
using FishNet.Documenting;
using FishNet.Managing.Logging;
using FishNet.Object.Synchronizing.Internal;
using FishNet.Serializing;
using FishNet.Utility.Extension;
using JetBrains.Annotations;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Runtime.CompilerServices;
using UnityEngine;
namespace FishNet.Object.Synchronizing
{
public class SyncIDictionary<TKey, TValue> : SyncBase, IDictionary<TKey, TValue>, IReadOnlyDictionary<TKey, TValue>
{
#region Types.
/// <summary>
/// Information needed to invoke a callback.
/// </summary>
private struct CachedOnChange
{
internal readonly SyncDictionaryOperation Operation;
internal readonly TKey Key;
internal readonly TValue Value;
public CachedOnChange(SyncDictionaryOperation operation, TKey key, TValue value)
{
Operation = operation;
Key = key;
Value = value;
}
}
/// <summary>
/// Information about how the collection has changed.
/// </summary>
private struct ChangeData
{
internal readonly SyncDictionaryOperation Operation;
internal readonly TKey Key;
internal readonly TValue Value;
public ChangeData(SyncDictionaryOperation operation, TKey key, TValue value)
{
this.Operation = operation;
this.Key = key;
this.Value = value;
}
}
#endregion
#region Public.
/// <summary>
/// Implementation from Dictionary<TKey,TValue>. Not used.
/// </summary>
[APIExclude]
public bool IsReadOnly => false;
/// <summary>
/// Delegate signature for when SyncDictionary changes.
/// </summary>
/// <param name="op">Operation being completed, such as Add, Set, Remove.</param>
/// <param name="key">Key being modified.</param>
/// <param name="value">Value of operation.</param>
/// <param name="asServer">True if callback is on the server side. False is on the client side.</param>
[APIExclude]
public delegate void SyncDictionaryChanged(SyncDictionaryOperation op, TKey key, TValue value, bool asServer);
/// <summary>
/// Called when the SyncDictionary changes.
/// </summary>
public event SyncDictionaryChanged OnChange;
/// <summary>
/// Collection of objects.
/// </summary>
public readonly IDictionary<TKey, TValue> Collection;
/// <summary>
/// Copy of objects on client portion when acting as a host.
/// </summary>
public readonly IDictionary<TKey, TValue> ClientHostCollection = new Dictionary<TKey, TValue>();
/// <summary>
/// Number of objects in the collection.
/// </summary>
public int Count => Collection.Count;
/// <summary>
/// Keys within the collection.
/// </summary>
public ICollection<TKey> Keys => Collection.Keys;
[APIExclude]
IEnumerable<TKey> IReadOnlyDictionary<TKey, TValue>.Keys => Collection.Keys;
/// <summary>
/// Values within the collection.
/// </summary>
public ICollection<TValue> Values => Collection.Values;
[APIExclude]
IEnumerable<TValue> IReadOnlyDictionary<TKey, TValue>.Values => Collection.Values;
#endregion
#region Private.
/// <summary>
/// Initial values for the dictionary.
/// </summary>
private IDictionary<TKey, TValue> _initialValues = new Dictionary<TKey, TValue>();
/// <summary>
/// Changed data which will be sent next tick.
/// </summary>
private readonly List<ChangeData> _changed = new List<ChangeData>();
/// <summary>
/// Server OnChange events waiting for start callbacks.
/// </summary>
private readonly List<CachedOnChange> _serverOnChanges = new List<CachedOnChange>();
/// <summary>
/// Client OnChange events waiting for start callbacks.
/// </summary>
private readonly List<CachedOnChange> _clientOnChanges = new List<CachedOnChange>();
/// <summary>
/// True if values have changed since initialization.
/// The only reasonable way to reset this during a Reset call is by duplicating the original list and setting all values to it on reset.
/// </summary>
private bool _valuesChanged;
/// <summary>
/// True to send all values in the next WriteDelta.
/// </summary>
private bool _sendAll;
#endregion
[APIExclude]
public SyncIDictionary(IDictionary<TKey, TValue> objects)
{
this.Collection = objects;
//Add to clienthostcollection.
foreach (KeyValuePair<TKey, TValue> item in objects)
this.ClientHostCollection[item.Key] = item.Value;
}
/// <summary>
/// Gets the collection being used within this SyncList.
/// </summary>
/// <param name="asServer">True if returning the server value, false if client value. The values will only differ when running as host. While asServer is true the most current values on server will be returned, and while false the latest values received by client will be returned.</param>
/// <returns>The used collection.</returns>
public Dictionary<TKey, TValue> GetCollection(bool asServer)
{
bool asClientAndHost = (!asServer && base.NetworkManager.IsServer);
IDictionary<TKey, TValue> collection = (asClientAndHost) ? ClientHostCollection : Collection;
return (collection as Dictionary<TKey, TValue>);
}
/// <summary>
/// Called when the SyncType has been registered, but not yet initialized over the network.
/// </summary>
protected override void Registered()
{
base.Registered();
foreach (KeyValuePair<TKey, TValue> item in Collection)
_initialValues[item.Key] = item.Value;
}
/// <summary>
/// Adds an operation and invokes callback locally.
/// Internal use.
/// May be used for custom SyncObjects.
/// </summary>
/// <param name="operation"></param>
/// <param name="key"></param>
/// <param name="value"></param>
[APIExclude]
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void AddOperation(SyncDictionaryOperation operation, TKey key, TValue value)
{
if (!base.IsRegistered)
return;
/* asServer might be true if the client is setting the value
* through user code. Typically synctypes can only be set
* by the server, that's why it is assumed asServer via user code.
* However, when excluding owner for the synctype the client should
* have permission to update the value locally for use with
* prediction. */
bool asServerInvoke = (!base.IsNetworkInitialized || base.NetworkBehaviour.IsServer);
if (asServerInvoke)
{
_valuesChanged = true;
if (base.Dirty())
{
ChangeData change = new ChangeData(operation, key, value);
_changed.Add(change);
}
}
InvokeOnChange(operation, key, value, asServerInvoke);
}
/// <summary>
/// Called after OnStartXXXX has occurred.
/// </summary>
/// <param name="asServer">True if OnStartServer was called, false if OnStartClient.</param>
public override void OnStartCallback(bool asServer)
{
base.OnStartCallback(asServer);
List<CachedOnChange> collection = (asServer) ? _serverOnChanges : _clientOnChanges;
if (OnChange != null)
{
foreach (CachedOnChange item in collection)
OnChange.Invoke(item.Operation, item.Key, item.Value, asServer);
}
collection.Clear();
}
/// <summary>
/// Writes all changed values.
/// Internal use.
/// May be used for custom SyncObjects.
/// </summary>
/// <param name="writer"></param>
///<param name="resetSyncTick">True to set the next time data may sync.</param>
[APIExclude]
public override void WriteDelta(PooledWriter writer, bool resetSyncTick = true)
{
base.WriteDelta(writer, resetSyncTick);
//If sending all then clear changed and write full.
if (_sendAll)
{
_sendAll = false;
_changed.Clear();
WriteFull(writer);
}
else
{
//False for not full write.
writer.WriteBoolean(false);
writer.WriteInt32(_changed.Count);
for (int i = 0; i < _changed.Count; i++)
{
ChangeData change = _changed[i];
writer.WriteByte((byte)change.Operation);
//Clear does not need to write anymore data so it is not included in checks.
if (change.Operation == SyncDictionaryOperation.Add ||
change.Operation == SyncDictionaryOperation.Set)
{
writer.Write(change.Key);
writer.Write(change.Value);
}
else if (change.Operation == SyncDictionaryOperation.Remove)
{
writer.Write(change.Key);
}
}
_changed.Clear();
}
}
/// <summary>
/// Writers all values if not initial values.
/// Internal use.
/// May be used for custom SyncObjects.
/// </summary>
/// <param name="writer"></param>
[APIExclude]
public override void WriteFull(PooledWriter writer)
{
if (!_valuesChanged)
return;
base.WriteHeader(writer, false);
//True for full write.
writer.WriteBoolean(true);
writer.WriteInt32(Collection.Count);
foreach (KeyValuePair<TKey, TValue> item in Collection)
{
writer.WriteByte((byte)SyncDictionaryOperation.Add);
writer.Write(item.Key);
writer.Write(item.Value);
}
}
/// <summary>
/// Reads and sets the current values for server or client.
/// </summary>
[APIExclude]
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public override void Read(PooledReader reader, bool asServer)
{
/* When !asServer don't make changes if server is running.
* This is because changes would have already been made on
* the server side and doing so again would result in duplicates
* and potentially overwrite data not yet sent. */
bool asClientAndHost = (!asServer && base.NetworkBehaviour.IsServer);
IDictionary<TKey, TValue> collection = (asClientAndHost) ? ClientHostCollection : Collection;
//Clear collection since it's a full write.
bool fullWrite = reader.ReadBoolean();
if (fullWrite)
collection.Clear();
int changes = reader.ReadInt32();
for (int i = 0; i < changes; i++)
{
SyncDictionaryOperation operation = (SyncDictionaryOperation)reader.ReadByte();
TKey key = default;
TValue value = default;
/* Add, Set.
* Use the Set code for add and set,
* especially so collection doesn't throw
* if entry has already been added. */
if (operation == SyncDictionaryOperation.Add || operation == SyncDictionaryOperation.Set)
{
key = reader.Read<TKey>();
value = reader.Read<TValue>();
collection[key] = value;
}
//Clear.
else if (operation == SyncDictionaryOperation.Clear)
{
collection.Clear();
}
//Remove.
else if (operation == SyncDictionaryOperation.Remove)
{
key = reader.Read<TKey>();
collection.Remove(key);
}
InvokeOnChange(operation, key, value, false);
}
//If changes were made invoke complete after all have been read.
if (changes > 0)
InvokeOnChange(SyncDictionaryOperation.Complete, default, default, false);
}
/// <summary>
/// Invokes OnChanged callback.
/// </summary>
private void InvokeOnChange(SyncDictionaryOperation operation, TKey key, TValue value, bool asServer)
{
if (asServer)
{
if (base.NetworkBehaviour.OnStartServerCalled)
OnChange?.Invoke(operation, key, value, asServer);
else
_serverOnChanges.Add(new CachedOnChange(operation, key, value));
}
else
{
if (base.NetworkBehaviour.OnStartClientCalled)
OnChange?.Invoke(operation, key, value, asServer);
else
_clientOnChanges.Add(new CachedOnChange(operation, key, value));
}
}
/// <summary>
/// Resets to initialized values.
/// </summary>
[APIExclude]
public override void Reset()
{
base.Reset();
_sendAll = false;
_changed.Clear();
Collection.Clear();
ClientHostCollection.Clear();
_valuesChanged = false;
foreach (KeyValuePair<TKey, TValue> item in _initialValues)
{
Collection[item.Key] = item.Value;
ClientHostCollection[item.Key] = item.Value;
}
}
/// <summary>
/// Adds item.
/// </summary>
/// <param name="item">Item to add.</param>
public void Add(KeyValuePair<TKey, TValue> item)
{
Add(item.Key, item.Value);
}
/// <summary>
/// Adds key and value.
/// </summary>
/// <param name="key">Key to add.</param>
/// <param name="value">Value for key.</param>
public void Add(TKey key, TValue value)
{
Add(key, value, true);
}
private void Add(TKey key, TValue value, bool asServer)
{
if (!base.CanNetworkSetValues(true))
return;
Collection.Add(key, value);
if (asServer)
AddOperation(SyncDictionaryOperation.Add, key, value);
}
/// <summary>
/// Clears all values.
/// </summary>
public void Clear()
{
Clear(true);
}
private void Clear(bool asServer)
{
if (!base.CanNetworkSetValues(true))
return;
Collection.Clear();
if (asServer)
AddOperation(SyncDictionaryOperation.Clear, default, default);
}
/// <summary>
/// Returns if key exist.
/// </summary>
/// <param name="key">Key to use.</param>
/// <returns>True if found.</returns>
public bool ContainsKey(TKey key)
{
return Collection.ContainsKey(key);
}
/// <summary>
/// Returns if item exist.
/// </summary>
/// <param name="item">Item to use.</param>
/// <returns>True if found.</returns>
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool Contains(KeyValuePair<TKey, TValue> item)
{
return TryGetValue(item.Key, out TValue value) && EqualityComparer<TValue>.Default.Equals(value, item.Value);
}
/// <summary>
/// Copies collection to an array.
/// </summary>
/// <param name="array">Array to copy to.</param>
/// <param name="offset">Offset of array data is copied to.</param>
public void CopyTo([NotNull] KeyValuePair<TKey, TValue>[] array, int offset)
{
if (offset <= -1 || offset >= array.Length)
{
base.NetworkManager.LogError($"Index is out of range.");
return;
}
int remaining = array.Length - offset;
if (remaining < Count)
{
base.NetworkManager.LogError($"Array is not large enough to copy data. Array is of length {array.Length}, index is {offset}, and number of values to be copied is {Count.ToString()}.");
return;
}
int i = offset;
foreach (KeyValuePair<TKey, TValue> item in Collection)
{
array[i] = item;
i++;
}
}
/// <summary>
/// Removes a key.
/// </summary>
/// <param name="key">Key to remove.</param>
/// <returns>True if removed.</returns>
public bool Remove(TKey key)
{
if (!base.CanNetworkSetValues(true))
return false;
if (Collection.Remove(key))
{
AddOperation(SyncDictionaryOperation.Remove, key, default);
return true;
}
return false;
}
/// <summary>
/// Removes an item.
/// </summary>
/// <param name="item">Item to remove.</param>
/// <returns>True if removed.</returns>
public bool Remove(KeyValuePair<TKey, TValue> item)
{
return Remove(item.Key);
}
/// <summary>
/// Tries to get value from key.
/// </summary>
/// <param name="key">Key to use.</param>
/// <param name="value">Variable to output to.</param>
/// <returns>True if able to output value.</returns>
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool TryGetValue(TKey key, out TValue value)
{
return Collection.TryGetValueIL2CPP(key, out value);
}
/// <summary>
/// Gets or sets value for a key.
/// </summary>
/// <param name="key">Key to use.</param>
/// <returns>Value when using as Get.</returns>
public TValue this[TKey key]
{
get => Collection[key];
set
{
if (!base.CanNetworkSetValues(true))
return;
Collection[key] = value;
AddOperation(SyncDictionaryOperation.Set, key, value);
}
}
/// <summary>
/// Dirties the entire collection forcing a full send.
/// </summary>
public void DirtyAll()
{
if (!base.IsRegistered)
return;
if (!base.CanNetworkSetValues(true))
return;
if (base.Dirty())
_sendAll = true;
}
/// <summary>
/// Dirties an entry by key.
/// </summary>
/// <param name="key">Key to dirty.</param>
public void Dirty(TKey key)
{
if (!base.IsRegistered)
return;
if (!base.CanNetworkSetValues(true))
return;
if (Collection.TryGetValueIL2CPP(key, out TValue value))
AddOperation(SyncDictionaryOperation.Set, key, value);
}
/// <summary>
/// Dirties an entry by value.
/// This operation can be very expensive, will cause allocations, and may fail if your value cannot be compared.
/// </summary>
/// <param name="value">Value to dirty.</param>
/// <returns>True if value was found and marked dirty.</returns>
public bool Dirty(TValue value, EqualityComparer<TValue> comparer = null)
{
if (!base.IsRegistered)
return false;
if (!base.CanNetworkSetValues(true))
return false;
if (comparer == null)
comparer = EqualityComparer<TValue>.Default;
foreach (KeyValuePair<TKey, TValue> item in Collection)
{
if (comparer.Equals(item.Value, value))
{
AddOperation(SyncDictionaryOperation.Set, item.Key, value);
return true;
}
}
//Not found.
return false;
}
/// <summary>
/// Gets the IEnumerator for the collection.
/// </summary>
/// <returns></returns>
public IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator() => Collection.GetEnumerator();
/// <summary>
/// Gets the IEnumerator for the collection.
/// </summary>
/// <returns></returns>
IEnumerator IEnumerable.GetEnumerator() => Collection.GetEnumerator();
}
[APIExclude]
public class SyncDictionary<TKey, TValue> : SyncIDictionary<TKey, TValue>
{
[APIExclude]
public SyncDictionary() : base(new Dictionary<TKey, TValue>()) { }
[APIExclude]
public SyncDictionary(IEqualityComparer<TKey> eq) : base(new Dictionary<TKey, TValue>(eq)) { }
[APIExclude]
public new Dictionary<TKey, TValue>.ValueCollection Values => ((Dictionary<TKey, TValue>)Collection).Values;
[APIExclude]
public new Dictionary<TKey, TValue>.KeyCollection Keys => ((Dictionary<TKey, TValue>)Collection).Keys;
[APIExclude]
public new Dictionary<TKey, TValue>.Enumerator GetEnumerator() => ((Dictionary<TKey, TValue>)Collection).GetEnumerator();
}
}