unity-atoms/Packages/Core/Runtime/VariableInstancers/AtomVariableInstancer.cs
Adam Ramberg c3bc59259f More WIP
2020-03-02 02:26:06 +01:00

142 lines
5.4 KiB
C#

using System;
using UnityEngine;
using UnityEngine.Assertions;
namespace UnityAtoms
{
/// <summary>
/// A Variable Instancer is a MonoBehaviour that takes a variable as a base and creates an in memory copy of it OnEnable.
/// This is handy when you want to use atoms for prefabs that are instantiated at runtime. Use together with AtomCollection to
/// react accordingly when a prefab with an assoicated atom is added or deleted to the scene.
/// </summary>
/// <typeparam name="V">Variable of type T.</typeparam>
/// <typeparam name="P">IPair of type `T`.</typeparam>
/// <typeparam name="T">The value type.</typeparam>
/// <typeparam name="E1">Event of type T.</typeparam>
/// <typeparam name="E2">Event x 2 of type T.</typeparam>
/// <typeparam name="F">Function of type T => T</typeparam>
[EditorIcon("atom-icon-hotpink")]
[DefaultExecutionOrder(Runtime.ExecutionOrder.VARIABLE_INSTANCER)]
public abstract class AtomVariableInstancer<V, P, T, E1, E2, F, CO, L> : MonoBehaviour, IGetEvent, ISetEvent
where V : AtomVariable<T, P, E1, E2, F>
where P : struct, IPair<T>
where E1 : AtomEvent<T>
where E2 : AtomEvent<P>
where F : AtomFunction<T, T>
where CO : IGetValue<IAtomCollection>
where L : IGetValue<IAtomList>
{
/// <summary>
/// Getter for retrieving the in memory runtime variable.
/// </summary>
public V Variable { get => _inMemoryCopy; }
/// <summary>
/// Getter for retrieving the value of the in memory runtime variable.
/// </summary>
public T Value { get => _inMemoryCopy.Value; set => _inMemoryCopy.Value = value; }
[SerializeField]
[ReadOnly]
private V _inMemoryCopy;
/// <summary>
/// The variable that the in memory copy will be based on when created at runtime.
/// </summary>
[SerializeField]
private V _base = null;
/// <summary>
/// If assigned the in memory copy variable will be added to the collection on Start using the gameObject's instance id as key. The value will also be removed from the collection OnDestroy.
/// </summary>
[SerializeField]
private CO _syncToCollection = default(CO);
/// <summary>
/// If assigned the in memory copy variable will be added to the list on Start. The value will also be removed from the list OnDestroy.
/// </summary>
[SerializeField]
private L _syncToList = default(L);
private void OnEnable()
{
Assert.IsNotNull(_base);
_inMemoryCopy = Instantiate(_base);
if (_base.Changed != null)
{
_inMemoryCopy.Changed = Instantiate(_base.Changed);
}
if (_base.ChangedWithHistory != null)
{
_inMemoryCopy.ChangedWithHistory = Instantiate(_base.ChangedWithHistory);
}
}
void Start()
{
// Adding to the collection on Start instead of OnEnable because of timing issues that otherwise occurs when listeners register themselves OnEnable.
// This is an issue when a Game Object has a Variable Instancer attached to it when the scene starts and at the same time their is an AtomBaseListener listening to the associated Added event to a Collection.
if (_syncToCollection != null)
{
_syncToCollection.GetValue().Add(GetInstanceID().ToString(), _inMemoryCopy);
}
if (_syncToList != null)
{
_syncToList.GetValue().Add(_inMemoryCopy);
}
}
private void OnDestroy()
{
if (_syncToCollection != null)
{
_syncToCollection.GetValue().Remove(GetInstanceID().ToString());
}
if (_syncToList != null)
{
_syncToList.GetValue().Remove(_inMemoryCopy);
}
}
/// <summary>
/// Get event by type. Don't use directly! Used only so that we don't need two implementations of Event Instancer and Listeners (one for `T` and one for `IPair&lt;T&gt;`)
/// </summary>
/// <typeparam name="E"></typeparam>
/// <returns>The event.</returns>
public E GetEvent<E>() where E : AtomEventBase
{
if (typeof(E) == typeof(E1))
return (_inMemoryCopy.Changed as E);
if (typeof(E) == typeof(E2))
return (_inMemoryCopy.ChangedWithHistory as E);
throw new Exception($"Event type {typeof(E)} not supported! Use {typeof(E1)} or {typeof(E2)}.");
}
/// <summary>
/// Set event by type. Don't use directly! Used only so that we don't need two implementations of Event Instancer and Listeners (one for `T` and one for `IPair&lt;T&gt;`)
/// </summary>
/// <param name="e">The new event value.</param>
/// <typeparam name="E"></typeparam>
public void SetEvent<E>(E e) where E : AtomEventBase
{
if (typeof(E) == typeof(E1))
{
_inMemoryCopy.Changed = (e as E1);
return;
}
if (typeof(E) == typeof(E2))
{
_inMemoryCopy.ChangedWithHistory = (e as E2);
return;
}
throw new Exception($"Event type {typeof(E)} not supported! Use {typeof(E1)} or {typeof(E2)}.");
}
}
}