unity-atoms/Packages/FSM/Runtime/FiniteStateMachine/FiniteStateMachine.cs

429 lines
15 KiB
C#
Raw Normal View History

2020-03-09 18:51:14 -04:00
using System;
2020-03-08 07:32:41 -04:00
using UnityEngine;
2020-03-15 18:18:29 -04:00
using UnityEngine.Events;
using UnityEngine.SceneManagement;
2020-03-08 07:32:41 -04:00
using UnityAtoms.BaseAtoms;
2020-03-11 20:30:25 -04:00
using UnityAtoms;
2020-03-08 07:32:41 -04:00
namespace UnityAtoms.FSM
{
2020-03-21 17:45:39 -04:00
/// <summary>
/// This is an implementation of an FSM in Unity Atoms. It is build using a set of states and a set of transitions. A set can only change through dispatching commands defined by the transitions.
/// </summary>
2020-03-08 15:41:22 -04:00
[EditorIcon("atom-icon-delicate")]
[CreateAssetMenu(menuName = "Unity Atoms/FSM/Finite State Machine", fileName = "FiniteStateMachine")]
2020-03-08 07:32:41 -04:00
public class FiniteStateMachine : StringVariable
{
2020-03-11 16:11:27 -04:00
/// <summary>
/// Get or set current value of this FSM. If a sub FSM is having the current state, then its state will be returned. Using the setter is the same thing as calling `Dispatch`.
/// </summary>
/// <value>The command to issue.</value>
2020-03-09 18:51:14 -04:00
public override string Value
{
get
{
var state = GetState(_value);
2020-03-11 20:30:25 -04:00
return state != null && state.SubMachine != null ? state.SubMachine.Value : _value;
2020-03-09 18:51:14 -04:00
}
set => Dispatch(value);
}
2020-03-08 08:24:46 -04:00
public FSMTransitionDataEvent TransitionStarted { get => _transitionStarted; set => _transitionStarted = value; }
public BoolEvent CompleteCurrentTransition { get => _completeCurrentTransition; set => _completeCurrentTransition = value; }
2020-03-11 20:30:25 -04:00
public override string InitialValue { get => _states?.List != null && _states.List.Count > 0 ? _states.List[0].Id : ""; }
2020-03-08 07:32:41 -04:00
/// <summary>
/// Gets a boolean value indicating if the state machine is currently transitioning.
/// </summary>
public bool IsTransitioning { get => _currentTransition != null; }
[SerializeField]
2020-03-08 08:24:46 -04:00
private FSMTransitionDataEvent _transitionStarted = default(FSMTransitionDataEvent);
2020-03-08 07:32:41 -04:00
[SerializeField]
private BoolEvent _completeCurrentTransition = default(BoolEvent);
2020-03-09 18:51:14 -04:00
[SerializeField]
2020-03-11 20:30:25 -04:00
[AtomList]
private FSMStateListWrapper _states;
2020-03-08 07:32:41 -04:00
[SerializeField]
2020-03-11 20:30:25 -04:00
[AtomList]
private TransitionListWrapper _transitions;
2020-03-08 07:32:41 -04:00
private bool _isUpdatingState = false;
private Transition _currentTransition = null;
private bool _resetOnNextTransitionCompleted = false;
2020-03-11 16:11:27 -04:00
private bool _triggerEventsOnNextReset = false;
2020-03-09 18:51:14 -04:00
private event Action<float, string> _onUpdate;
2020-03-11 16:11:27 -04:00
private event Action<float, string> _onFixedUpdate;
2020-03-09 18:51:14 -04:00
private event Action<string> _dispatchWhen;
2020-03-11 16:11:27 -04:00
private event Action<string> _onStateCooldown;
2020-03-09 18:51:14 -04:00
/// <summary>
/// The value in this state machine, disregarding a "deeper" values in a sub machine.
/// </summary>
private string _currentFlatValue;
2020-03-15 18:18:29 -04:00
private void OnEnable()
2020-03-11 16:11:27 -04:00
{
2020-03-11 16:42:59 -04:00
if (CompleteCurrentTransition != null && CompleteCurrentTransition.ReplayBufferSize > 0)
{
Debug.LogWarning("The Complete Current Transition event had a replay buffer size great than 0, which would cause unwanted behaviour. Setting it to 0 in order to avoid unexpected behaviour.");
CompleteCurrentTransition.ReplayBufferSize = 0;
}
2020-03-15 18:18:29 -04:00
_isUpdatingState = false;
_currentTransition = null;
_resetOnNextTransitionCompleted = false;
_triggerEventsOnNextReset = false;
2020-03-17 21:00:18 -04:00
_currentFlatValue = Value;
2020-03-11 16:11:27 -04:00
2020-03-15 18:18:29 -04:00
// Make sure application is playing
if (Application.isPlaying)
{
FiniteStateMachineMonoHook.GetInstance(createIfNotExist: true).OnUpdate -= OnUpdate;
FiniteStateMachineMonoHook.GetInstance().OnUpdate += OnUpdate;
FiniteStateMachineMonoHook.GetInstance().OnStart -= OnStart;
FiniteStateMachineMonoHook.GetInstance().OnStart += OnStart;
}
else
{
UnityAction<Scene, LoadSceneMode> handler = null;
handler = (scene, mode) =>
{
SceneManager.sceneLoaded -= handler;
FiniteStateMachineMonoHook.GetInstance(createIfNotExist: true).OnUpdate -= OnUpdate;
FiniteStateMachineMonoHook.GetInstance().OnUpdate += OnUpdate;
Reset();
};
SceneManager.sceneLoaded += handler;
}
2020-03-11 16:11:27 -04:00
}
2020-03-09 18:51:14 -04:00
private void OnDisable()
{
2020-03-11 16:11:27 -04:00
if (FiniteStateMachineMonoHook.GetInstance() != null)
2020-03-09 18:51:14 -04:00
{
2020-03-11 16:11:27 -04:00
FiniteStateMachineMonoHook.GetInstance().OnUpdate -= OnUpdate;
FiniteStateMachineMonoHook.GetInstance().OnStart -= OnStart;
2020-03-09 18:51:14 -04:00
}
_onUpdate = null;
2020-03-11 16:11:27 -04:00
_onFixedUpdate = null;
2020-03-09 18:51:14 -04:00
_dispatchWhen = null;
2020-03-11 16:11:27 -04:00
_onStateCooldown = null;
2020-03-09 18:51:14 -04:00
}
2020-03-21 17:59:30 -04:00
/// <summary>
/// Calls the handler on every Update.
/// </summary>
/// <param name="handler">The handler to called.</param>
/// <param name="gameObject">The gameObject where this handler is setup.</param>
2020-03-11 16:11:27 -04:00
public void OnUpdate(Action<float, string> handler, GameObject gameObject)
2020-03-09 18:51:14 -04:00
{
2020-03-11 16:11:27 -04:00
Action<float, string> extendedHandler = null;
extendedHandler = (deltaTime, state) =>
{
// Unregister created handler if original handler doesn't or if the GameObject has been destroyed
2020-03-15 19:11:18 -04:00
if (handler == null || gameObject == null)
2020-03-11 16:11:27 -04:00
{
_onUpdate -= extendedHandler;
return;
}
handler(deltaTime, state);
};
_onUpdate += extendedHandler;
2020-03-09 18:51:14 -04:00
}
2020-03-21 17:59:30 -04:00
/// <summary>
/// Calls the handler on every FixedUpdate.
/// </summary>
/// <param name="handler">The handler to called.</param>
/// <param name="gameObject">The gameObject where this hook is setup.</param>
2020-03-11 16:11:27 -04:00
public void OnFixedUpdate(Action<float, string> handler, GameObject gameObject)
2020-03-09 18:51:14 -04:00
{
2020-03-11 16:11:27 -04:00
Action<float, string> extendedHandler = null;
extendedHandler = (deltaTime, state) =>
2020-03-09 18:51:14 -04:00
{
2020-03-11 16:11:27 -04:00
// Unregister created handler if original handler doesn't or if the GameObject has been destroyed
2020-03-15 19:11:18 -04:00
if (handler == null || gameObject == null)
2020-03-11 16:11:27 -04:00
{
_onFixedUpdate -= extendedHandler;
return;
}
handler(deltaTime, state);
};
_onFixedUpdate += extendedHandler;
}
2020-03-21 17:59:30 -04:00
/// <summary>
/// Define a command that is going to automatically be dispatched when the condition provided is met.
/// </summary>
2020-03-11 16:11:27 -04:00
public void DispatchWhen(string command, Func<string, bool> func, GameObject gameObject)
{
Action<string> extendedHandler = null;
extendedHandler = (value) =>
{
2020-03-15 19:11:18 -04:00
// Unregister created handler if original handler doesn't exists or if the GameObject has been destroyed
if (func == null || gameObject == null)
2020-03-11 16:11:27 -04:00
{
_dispatchWhen -= extendedHandler;
return;
}
2020-03-09 18:51:14 -04:00
if (func(value))
{
Dispatch(command);
}
};
2020-03-11 16:11:27 -04:00
_dispatchWhen += extendedHandler;
2020-03-09 18:51:14 -04:00
}
2020-03-08 07:32:41 -04:00
2020-03-21 17:59:30 -04:00
/// <summary>
/// Called on every state cooldown.
/// </summary>
/// <param name="state">The state where we want to do something on the cool down.</param>
/// <param name="handler">Handler to be called on cooldown.</param>
/// <param name="gameObject">The gameObject where this hook is setup.</param>
2020-03-11 16:11:27 -04:00
public void OnStateCooldown(string state, Action<string> handler, GameObject gameObject)
2020-03-08 07:32:41 -04:00
{
2020-03-11 16:11:27 -04:00
Action<string> extendedHandler = null;
extendedHandler = (value) =>
{
// Unregister created handler if original handler doesn't or if the GameObject has been destroyed
2020-03-15 19:11:18 -04:00
if (handler == null || gameObject == null)
2020-03-11 16:11:27 -04:00
{
_onStateCooldown -= extendedHandler;
return;
}
if (value == state)
{
handler(value);
}
};
_onStateCooldown += extendedHandler;
}
2020-03-21 17:59:30 -04:00
/// <summary>
/// Reset
/// </summary>
/// <param name="shouldTriggerEvents">Should we trigger Change Events.</param>
2020-03-11 16:11:27 -04:00
public override void Reset(bool shouldTriggerEvents = false)
{
2020-03-11 20:30:25 -04:00
Validate();
2020-03-11 16:11:27 -04:00
// Set all timers to the same as the cooldown
2020-03-11 20:30:25 -04:00
for (var i = 0; i < _states.List.Count; ++i)
2020-03-11 16:11:27 -04:00
{
2020-03-11 20:30:25 -04:00
_states.List[i].Timer = _states.List[i].Cooldown;
2020-03-11 16:11:27 -04:00
}
2020-03-09 18:51:14 -04:00
2020-03-08 07:32:41 -04:00
if (!_resetOnNextTransitionCompleted && !IsTransitioning)
{
2020-03-08 15:41:22 -04:00
_resetOnNextTransitionCompleted = false;
2020-03-09 18:51:14 -04:00
ResetAllSubMachines();
2020-03-11 16:11:27 -04:00
base.Reset(shouldTriggerEvents);
2020-03-09 18:51:14 -04:00
_currentFlatValue = _value;
2020-03-08 07:32:41 -04:00
}
else
{
2020-03-11 16:11:27 -04:00
_triggerEventsOnNextReset = shouldTriggerEvents;
2020-03-08 07:32:41 -04:00
_resetOnNextTransitionCompleted = true;
}
}
/// <summary>
/// Dispatches a new command to the FiniteStateMachine, invoking any necessary transitions.
/// </summary>
/// <param name="command"></param>
public void Dispatch(string command)
{
// Commands dispatched during transition will be ignored
if (IsTransitioning)
return;
var transition = FindTransitionForCurrentState(command);
if (transition != null)
{
// Commands should not be dispatched in events listening to state changes of this machine
if (_isUpdatingState)
{
Debug.LogWarning("Do not call Dispatch from handlers listening to Changed and ChangedWithHistory events.");
return;
}
if (transition.TestCondition())
{
_currentTransition = transition;
if (_transitionStarted != null)
{
_transitionStarted.Raise(
2020-03-08 08:24:46 -04:00
new FSMTransitionData()
2020-03-08 07:32:41 -04:00
{
FromState = transition.FromState,
ToState = transition.ToState,
Command = transition.Command,
CompleteTransition = _completeCurrentTransition,
}
);
}
2020-03-11 16:11:27 -04:00
transition.Begin(this, EndCurrentTransition);
2020-03-08 07:32:41 -04:00
}
}
2020-03-09 18:51:14 -04:00
else
{
2020-03-11 16:11:27 -04:00
// State doesn't exist in this FSM, propagate down to all sub FSMs.
2020-03-09 18:51:14 -04:00
var state = GetState(_currentFlatValue);
2020-03-15 18:18:29 -04:00
state?.SubMachine?.Dispatch(command);
2020-03-09 18:51:14 -04:00
}
2020-03-08 07:32:41 -04:00
}
2020-03-11 16:11:27 -04:00
protected override bool ValueEquals(string other) => false; // Always trigger events even if changing to the same state as previous
2020-03-09 18:51:14 -04:00
2020-03-11 20:30:25 -04:00
private void Validate()
{
for (var i = 0; i < _transitions.List.Count; ++i)
{
var transition = _transitions.List[i];
if (!_states.List.Exists((s) => s.Id == transition.FromState))
{
Debug.LogError($"Transition with From State {transition.FromState} can't be found in the defined states.");
}
if (!_states.List.Exists((s) => s.Id == transition.ToState))
{
Debug.LogError($"Transition with To State {transition.ToState} can't be found in the defined states.");
}
}
}
2020-03-11 16:11:27 -04:00
private void EndCurrentTransition()
2020-03-08 07:32:41 -04:00
{
if (_resetOnNextTransitionCompleted)
{
2020-03-08 15:41:22 -04:00
_resetOnNextTransitionCompleted = false;
2020-03-11 16:11:27 -04:00
Reset(_triggerEventsOnNextReset);
2020-03-08 07:32:41 -04:00
return;
}
_isUpdatingState = true;
2020-03-09 18:51:14 -04:00
var toState = GetState(_currentTransition.ToState);
if (toState.SubMachine != null)
{
// Reset sub machines in to state
2020-03-11 16:11:27 -04:00
toState.SubMachine.Reset();
2020-03-09 18:51:14 -04:00
base.Value = toState.SubMachine.Value;
}
else
{
base.Value = _currentTransition.ToState;
}
_currentFlatValue = _currentTransition.ToState;
2020-03-08 15:41:22 -04:00
_currentTransition = null;
2020-03-09 18:51:14 -04:00
2020-03-08 07:32:41 -04:00
_isUpdatingState = false;
}
2020-03-11 16:11:27 -04:00
private void ResetAllSubMachines()
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
for (var i = 0; i < _states.List.Count; ++i)
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
if (_states.List[i].SubMachine != null)
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
_states.List[i].SubMachine.Reset();
2020-03-09 18:51:14 -04:00
}
}
}
2020-03-11 16:11:27 -04:00
private void OnStart()
2020-03-09 18:51:14 -04:00
{
2020-03-11 16:11:27 -04:00
Reset();
}
private void OnUpdate(float deltaTime)
{
// Update timers and call OnStateCooldown handlers if applicable
2020-03-09 18:51:14 -04:00
var currentValue = Value;
2020-03-11 20:30:25 -04:00
for (var i = 0; i < _states.List.Count; ++i)
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
var state = _states.List[i];
2020-03-11 16:11:27 -04:00
if (state.Cooldown > 0f)
2020-03-09 18:51:14 -04:00
{
var isCurrent = state.Id == currentValue;
var isOngoing = !isCurrent && state.Timer > 0f;
if (isCurrent || isOngoing)
{
2020-03-11 16:11:27 -04:00
if (isCurrent && state.Timer <= 0f && _onStateCooldown != null)
{
_onStateCooldown.Invoke(currentValue);
}
2020-03-09 18:51:14 -04:00
state.Timer += deltaTime;
2020-03-11 16:11:27 -04:00
if (state.Timer >= state.Cooldown)
2020-03-09 18:51:14 -04:00
{
state.Timer = 0f;
}
}
}
}
2020-03-11 16:11:27 -04:00
// Call OnUpdate handlers
2020-03-09 18:51:14 -04:00
if (_onUpdate != null)
{
_onUpdate.Invoke(deltaTime, currentValue);
}
2020-03-11 16:11:27 -04:00
// Call DispatchWhen handlers
2020-03-09 18:51:14 -04:00
if (_dispatchWhen != null)
{
_dispatchWhen.Invoke(currentValue);
}
}
2020-03-11 16:11:27 -04:00
private void OnFixedUpdate(float deltaTime)
{
// Call OnFixedUpdate handlers
if (_onFixedUpdate != null)
{
_onFixedUpdate.Invoke(deltaTime, Value);
}
}
2020-03-08 07:32:41 -04:00
private Transition FindTransitionForCurrentState(string command)
{
Transition ret = null;
2020-03-11 20:30:25 -04:00
for (var i = 0; i < _transitions.List.Count; ++i)
2020-03-08 07:32:41 -04:00
{
2020-03-11 20:30:25 -04:00
var transition = _transitions.List[i];
2020-03-09 18:51:14 -04:00
if (command == transition.Command && _currentFlatValue == transition.FromState)
2020-03-08 07:32:41 -04:00
{
2020-03-08 15:41:22 -04:00
return transition;
2020-03-08 07:32:41 -04:00
}
}
return ret;
}
2020-03-09 18:51:14 -04:00
private FSMState GetState(string id)
{
2020-03-11 20:30:25 -04:00
for (var i = 0; i < _states.List.Count; ++i)
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
if (_states.List[i].Id == id)
2020-03-09 18:51:14 -04:00
{
2020-03-11 20:30:25 -04:00
return _states.List[i];
2020-03-09 18:51:14 -04:00
}
}
2020-03-11 20:30:25 -04:00
return null;
2020-03-09 18:51:14 -04:00
}
2020-03-08 07:32:41 -04:00
}
}