2019-04-11 13:52:13 +02:00
|
|
|
using System;
|
2020-03-08 13:24:46 +01:00
|
|
|
using System.Linq;
|
2020-02-23 21:44:04 +01:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using UnityEngine;
|
2018-10-30 20:05:06 +01:00
|
|
|
|
|
|
|
namespace UnityAtoms
|
|
|
|
{
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
Added Variable Instancer, Event Reference, Atom Collection and Atom List (old Atom List renamed to Atom Value List) (#110)
AtomVariableInstancer
- Added AtomVariableInstancer as an option to AtomReference.
- Added AtomVariableInstancer to generator.
- Added editor icon for AtomVariableInstancer.
AtomEventReference
- Added an AtomEventReference class (and AtomEventX2Reference). It’s similar to an AtomReference, but for Events. Let’s you pick between an Event, Variable (will select the Changed event) and a VariableInstancer (see above).
- Added AtomEventReference and AtomEventX2Reference to generator.
- Added a drawer for AtomEventReference.
- Listeners are now using AtomEventReference instead of AtomEvent.
- Refactoring of VoidHooks since Listeners are now using AtomEventReference.
AtomCollection
- Created an AtomCollection - a collection of Atoms associated with key strings (AtomReferences).
- Added new editor icon for collections.
- Created a SerializableDictionary class, which AtomCollection is using.
- Custom property drawer for SerializableDictionary.
- SerializableDictionary supports nested structures meaning that a AtomCollection can have a KVP that is pointing to another AtomCollection.
- AtomCollections have 3 events: Added, Removed, Cleared.
- Added an option to sync an InstanceVariable to collection - adding it to the collection when created (using gameObject’s instance id as key) and removing it from the collection when destroyed.
AtomList
- Renamed old AtomList to AtomValueList
- Added AtomList, like Collection, but a list
- Added new icon for AtomList
- Created a AtomBaseVariableList class, which AtomList is using.
- Custom property drawer for AtomBaseVariableList.
- AtomLists have 3 events: Added, Removed, Cleared.
- Added an option to sync an InstanceVariable to list - adding it to the list when created and removing it from the list when destroyed.
2020-02-23 02:39:43 +01:00
|
|
|
/// Generic base class for Events. Inherits from `AtomEventBase`.
|
2019-10-15 20:44:25 +02:00
|
|
|
/// </summary>
|
|
|
|
/// <typeparam name="T">The type for this Event.</typeparam>
|
2019-10-14 16:51:54 +02:00
|
|
|
[EditorIcon("atom-icon-cherry")]
|
2020-03-05 00:48:39 +01:00
|
|
|
public class AtomEvent<T> : AtomEventBase
|
2018-10-30 20:05:06 +01:00
|
|
|
{
|
2020-03-01 21:32:52 +01:00
|
|
|
public T InspectorRaiseValue { get => _inspectorRaiseValue; }
|
|
|
|
|
2020-03-08 13:24:46 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Retrieve Replay Buffer as a List. This call will allocate memory so use sparsely.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns></returns>
|
|
|
|
public List<T> ReplayBuffer { get => _replayBuffer.ToList(); }
|
|
|
|
|
2020-03-11 21:42:59 +01:00
|
|
|
public int ReplayBufferSize { get => _replayBufferSize; set => _replayBufferSize = value; }
|
|
|
|
|
2020-02-23 21:44:04 +01:00
|
|
|
[SerializeField]
|
2020-03-09 00:37:52 +01:00
|
|
|
protected event Action<T> _onEvent;
|
2020-02-23 21:44:04 +01:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// The event replays the specified number of old values to new subscribers. Works like a ReplaySubject in Rx.
|
|
|
|
/// </summary>
|
|
|
|
[SerializeField]
|
|
|
|
[Range(0, 10)]
|
|
|
|
[Tooltip("The number of old values (between 0-10) being replayed when someone subscribes to this Event.")]
|
|
|
|
private int _replayBufferSize = 1;
|
|
|
|
|
|
|
|
private Queue<T> _replayBuffer = new Queue<T>();
|
|
|
|
|
|
|
|
private void OnDisable()
|
|
|
|
{
|
|
|
|
// Clear all delegates when exiting play mode
|
|
|
|
if (_onEvent != null)
|
|
|
|
{
|
|
|
|
var invocationList = _onEvent.GetInvocationList();
|
|
|
|
foreach (var d in invocationList)
|
|
|
|
{
|
|
|
|
_onEvent -= (Action<T>)d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
2020-02-23 21:44:04 +01:00
|
|
|
/// Used when raising values from the inspector for debugging purposes.
|
2019-10-15 20:44:25 +02:00
|
|
|
/// </summary>
|
2020-02-23 21:44:04 +01:00
|
|
|
[SerializeField]
|
|
|
|
[Tooltip("Value that will be used when using the Raise button in the editor inspector.")]
|
2020-03-02 20:27:38 +01:00
|
|
|
private T _inspectorRaiseValue = default(T);
|
2018-10-30 20:05:06 +01:00
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Raise the Event.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="item">The value associated with the Event.</param>
|
2018-10-30 20:05:06 +01:00
|
|
|
public void Raise(T item)
|
|
|
|
{
|
2020-03-05 00:48:39 +01:00
|
|
|
base.Raise();
|
2020-02-23 21:44:04 +01:00
|
|
|
_onEvent?.Invoke(item);
|
|
|
|
AddToReplayBuffer(item);
|
2018-10-30 20:05:06 +01:00
|
|
|
}
|
|
|
|
|
2020-03-12 01:30:25 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Used in editor scipts since Raise is ambigious when using reflection to get method.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="item"></param>
|
|
|
|
public void RaiseEditor(T item) => Raise(item);
|
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Register handler to be called when the Event triggers.
|
|
|
|
/// </summary>
|
2020-02-23 21:44:04 +01:00
|
|
|
/// <param name="action">The handler.</param>
|
|
|
|
public void Register(Action<T> action)
|
2019-04-16 22:32:17 +02:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
_onEvent += action;
|
2020-03-08 13:24:46 +01:00
|
|
|
ReplayBufferToSubscriber(action);
|
2019-04-16 22:32:17 +02:00
|
|
|
}
|
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Unregister handler that was registered using the `Register` method.
|
|
|
|
/// </summary>
|
2020-02-23 21:44:04 +01:00
|
|
|
/// <param name="action">The handler.</param>
|
|
|
|
public void Unregister(Action<T> action)
|
2019-04-16 22:32:17 +02:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
_onEvent -= action;
|
2019-04-16 22:32:17 +02:00
|
|
|
}
|
|
|
|
|
2020-03-09 23:51:14 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Unregister all handlers that were registered using the `Register` method.
|
|
|
|
/// </summary>
|
|
|
|
public void UnregisterAll()
|
|
|
|
{
|
|
|
|
_onEvent = null;
|
|
|
|
}
|
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Register a Listener that in turn trigger all its associated handlers when the Event triggers.
|
|
|
|
/// </summary>
|
2019-11-26 20:12:54 +01:00
|
|
|
/// <param name="listener">The Listener to register.</param>
|
2019-09-25 21:05:06 +02:00
|
|
|
public void RegisterListener(IAtomListener<T> listener)
|
2018-10-30 20:05:06 +01:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
_onEvent += listener.OnEventRaised;
|
2020-03-08 13:24:46 +01:00
|
|
|
ReplayBufferToSubscriber(listener.OnEventRaised);
|
2018-10-30 20:05:06 +01:00
|
|
|
}
|
|
|
|
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Unregister a listener that was registered using the `RegisterListener` method.
|
|
|
|
/// </summary>
|
2019-11-26 20:12:54 +01:00
|
|
|
/// <param name="listener">The Listener to unregister.</param>
|
2019-09-25 21:05:06 +02:00
|
|
|
public void UnregisterListener(IAtomListener<T> listener)
|
2018-10-30 20:05:06 +01:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
_onEvent -= listener.OnEventRaised;
|
2018-10-30 20:05:06 +01:00
|
|
|
}
|
2019-04-16 22:32:17 +02:00
|
|
|
|
|
|
|
#region Observable
|
2019-10-15 20:44:25 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Turn the Event into an `IObservable<T>`. Makes Events compatible with for example UniRx.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>The Event as an `IObservable<T>`.</returns>
|
2019-04-16 22:32:17 +02:00
|
|
|
public IObservable<T> Observe()
|
|
|
|
{
|
|
|
|
return new ObservableEvent<T>(Register, Unregister);
|
|
|
|
}
|
|
|
|
#endregion // Observable
|
|
|
|
|
2020-03-09 00:37:52 +01:00
|
|
|
protected void AddToReplayBuffer(T item)
|
2019-04-16 22:32:17 +02:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
if (_replayBufferSize > 0)
|
|
|
|
{
|
|
|
|
while (_replayBuffer.Count >= _replayBufferSize) { _replayBuffer.Dequeue(); }
|
|
|
|
_replayBuffer.Enqueue(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-08 13:24:46 +01:00
|
|
|
private void ReplayBufferToSubscriber(Action<T> action)
|
2020-02-23 21:44:04 +01:00
|
|
|
{
|
|
|
|
if (_replayBufferSize > 0 && _replayBuffer.Count > 0)
|
2019-04-16 22:32:17 +02:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
var enumerator = _replayBuffer.GetEnumerator();
|
|
|
|
try
|
2019-04-16 22:32:17 +02:00
|
|
|
{
|
2020-02-23 21:44:04 +01:00
|
|
|
while (enumerator.MoveNext())
|
|
|
|
{
|
|
|
|
action(enumerator.Current);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
finally
|
|
|
|
{
|
|
|
|
enumerator.Dispose();
|
2019-04-16 22:32:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-30 20:05:06 +01:00
|
|
|
}
|
2019-03-17 23:43:20 +01:00
|
|
|
}
|