Skip to content

Visualising State Machines with Animator Graphs

Inspiaaa edited this page Apr 1, 2025 · 2 revisions

Visualising State Machines with Animator Graphs

Once you have written a state machine in code using UnityHFSM, you can automatically generate an animator controller for it. This allows you to explore its structure visually in the Unity editor, helping other team-members understand the code quickly, or help you debug the behaviour using the live preview feature.

The following tutorial shows you how you can use this feature. The example code is taken from the Guard AI tutorial.

Creating the Animator Graph

  1. Import UnityHFSM's visualisation tools:

    using UnityEngine;
    using UnityHFSM;  // Import UnityHFSM.
    using UnityHFSM.Visualization;  // Import the animator graph feature.
    
    public class GuardAI : MonoBehaviour
    {
        // ...
    }
  2. Generate the animator controller after initialising the state machine:

    public class GuardAI : MonoBehaviour
    {
        // Declare the finite state machine
        private StateMachine fsm;
    
        // ...
    
        void Start()
        {
            fsm = new StateMachine();
    
            // Initialise the state machine here ...
    
            #if UNITY_EDITOR
            HfsmAnimatorGraph.CreateAnimatorFromStateMachine(
                fsm,
                outputFolderPath: "Assets/DebugAnimators",
                animatorName: "StateMachineAnimatorGraph.controller"
            );
            #endif
        }
    
        // ...
    }

    Note that if you only intend to use this code in the editor and are going to remove it before building your project, you do not need the #if ... #endif preprocessor directives.

  3. Run the game: Once you start the game, the code will generate the animator controller and place it in your Assets folder at the desired location. By opening it in the editor, you can now view your state machine.

    AnimatorController in the Assets folder

    The initial layout is most likely not ideal, so you may have to re-arrange the states. If you run your project again, which will regenerate the animator file, it will preserve your custom layout.

    Initial Layout

    Manual Layout

    By double-clicking on the hexagonal state (Fight in this example), you can go down a layer in the hierarchy:

    As you can see, UnityHFSM also visualises the entry state, the exit transitions, and transitions from any for you.

    Nested State Machine

    By clicking on (Up) Base Layer or using the navigation controls at the top of the window, you can return to the root state machine:

    Navigating to the root state machine

Live Preview

  1. In order to view the active state of the state machine in real time, we have to add an animator component to the scene. This animator can be attached to the current game object, or to an independent one. The Controller property of the animator should be set to the generated instance from the Assets folder.

    In this case, I decided to add it to a separate game object:

    Animator in a separate game object

    Controller property of the Animator

  2. Next, we have to make this animator available to our code:

    public class GuardAI : MonoBehaviour
    {
        // ...
    
        [SerializeField] private Animator fsmAnimator;
    
        // ...
    }

    In the inspector we can now attach the new animator component:

    Inspector

  3. Finally, we can setup the live preview from code:

    public class GuardAI2 : MonoBehaviour
    {
        // ...
    
        void Update()
        {
            fsm.OnLogic();
    
            #if UNITY_EDITOR
            HfsmAnimatorGraph.PreviewStateMachineInAnimator(fsm, fsmAnimator);
            #endif
        }
    }

    By starting the game, selecting the game object with the animator and opening the animator controller from the assets, we can see what the state machine is doing.

    Live Preview Animation


Using HfsmAnimatorGraph.PreviewStateMachineInAnimator(...); works as long as we don't have any naming collisions (states in different places in the hierarchy with the same name). To get an accurate preview when we have a naming collision, we can use the dedicated IPreviewer object returned by HfsmAnimatorGraph.CreateAnimatorFromStateMachine(...) which stores additional information:

public class GuardAI : MonoBehaviour
{
    #if UNITY_EDITOR
    [SerializeField] private Animator fsmAnimator;
    private HfsmAnimatorGraph.IPreviewer animatorPreviewer;  // Keep a reference to the previewer.
    #endif

    void Start()
    {
        #if UNITY_EDITOR
        // The first result component is the AnimatorController which we don't need => ignore.
        (_, animatorPreviewer) = HfsmAnimatorGraph.CreateAnimatorFromStateMachine(fsm);
        #endif
    }

    void Update()
    {
        fsm.OnLogic();

        #if UNITY_EDITOR
        animatorPreviewer.PreviewStateMachineInAnimator(fsmAnimator);
        #endif
    }
}