Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/toastisme/openbehaviourtree

Design and monitor behaviour trees visually in Unity.
https://github.com/toastisme/openbehaviourtree

Last synced: 2 months ago
JSON representation

Design and monitor behaviour trees visually in Unity.

Awesome Lists containing this project

README

        

# OpenBehaviourTree

OpenBehaviourTree is an open source tool to design and monitor behaviour trees visually in Unity.

![OpenBehaviourTree](https://media.githubusercontent.com/media/toastisme/OpenBehaviourTree/main/Runtime/Resources/Screenshots/ExampleTree.PNG?token=AOQPGDSRNMAR3CIUOX5I4B3CDARAK)

## Features
- Custom editor window to design trees, and watch nodes being evaluated in real time during Play Mode
- Priority, sequence, and probability selector nodes
- Action nodes that can run user-defined class coroutines
- A blackboard to share variables between nodes
- Boolean, timeout, and cooldown decorators
- User-defined probability weights for all children of probability selector nodes
- Decorators and probability weights can use constant values or values from the blackboard

## Installation

Tested on Windows 10 using Unity 2021.2.11f1
- Install Git (e.g. for Windows https://gitforwindows.org/)
- In Unity open the package manager (`Window` --> `Package Manager`)
- Use the plus button to add a new package, and choose `Add package from git URL`
- Add the URL `https://github.com/toastisme/OpenBehaviourTree.git`

## Getting Started

### Creating a new tree
OpenBehaviourTree works as a custom editor. To create a behaviour tree right-click in a given folder in your Project view, and choose `Create` --> `BehaviourTree`.
Most behaviour trees will also need a blackboard (`Create` --> `BehaviourTreeBlackboard`).
Double-clicking on the newly created BehaviourTree will then open an editor window with a single Root node. This is the base of your tree.

### Navigation

To move around the tree left-click and drag the mouse. Zoom is controlled with the mouse wheel, and nodes can be moved by left-clicking and dragging them.

### Adding nodes

On the right-hand side is the Details panel. Clicking on the Blackboard tab will allow you to add a given blackboard. Here you can also add blackboard keys that can be used in the behaviour tree.

To begin adding nodes left-click the bottom connection point of the Root node. This will begin to draw a connection from the node to your cursor.
Left-clicking anywhere on the grid will then give a series of options for creating a node. Right-clicking while drawing a connection cancels the connection.
When left-clicking on a given node, its details are shown in the Details panel when looking at the Selected tab, along with any properties that can be modified by the user.
Decorators can be added to a node by right-clicking on the nodes.

### Creating custom Actions

Action nodes can be created from anything that derives from `BehaviourTreeTask`. A new action class just needs to implement two things:
- `public override void Setup(MonoBehaviour monoBehaviour)`, which passes in the `MonoBehaviour` of the `GameObject` running the behaviour tree. Here you can cache any references to components needed for the action,
such as a `RigidBody` or `NavMeshAgent`.
- `public override IEnumerator ExecuteTask(System.Action CurrentState)`. This is the coroutine that carries out the action. `CurrentState` is the state
of the `ActionNode`, and it is assumed that at the start of the coroutine this will be set to `Behaviour.NodeState.Running`, and when the coroutine finishes it will be set to
`Behaviour.Nodestate.Succeeded`.
All classes derived from `BehaviourTreeTask` have access to the blackboard, and can get and set blackboard keys using the methods:

```
GetInt(string keyName)
GetBool(string keyName)
GetFloat(string keyName)
GetGameObject(string keyName)
GetString(string keyName)
GetVector3(string keyName)
GetVector2(string keyName)

SetInt(string keyName, int val)
SetBool(string keyName, bool val)
SetFloat(string keyName, float val)
SetGameObject(string keyName, GameObject val)
SetString(string keyName, string val)
SetVector3(string keyName, Vector3 val)
SetVector2(string keyName, Vector2 val)
```

So, for example, an action that added force to the `GameObject` can be written like this

```
using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class ExampleTask : Behaviour.BehaviourTreeTask
{
Rigidbody rb;

public override void Setup(MonoBehaviour monoBehaviour){
rb = monoBehaviour.GetComponent();
}

public override IEnumerator ExecuteTask(System.Action currentState){

// Put the node in the Running state
currentState(Behaviour.NodeState.Running);

// Get values from the blackboard
Vector3 force = blackboard.GetVector3("currentForce");
float timeDelay = blackboard.GetFloat("timeDelay");

// Carry out the task
rb.AddForce(force);
yield return new WaitForSeconds(timeDelay);

// Put the node in the Succeeded state
currentState(Behaviour.NodeState.Succeeded);
}
}
```

### Attaching a behaviour tree to a GameObject
- Behaviour trees are attached to a `GameObject` by attaching the `BehaviourTreeController` script, and then adding the `BehaviourTree` to the `BehaviourTreeController`.
- The tree can then be started and stopped using the `BehaviourTreeController` `RunTree()` and `StopTree()` methods, respectively.
- In `Play Mode`, when selecting a `GameObject` that has a `BehaviourTreeController`, the `BehaviourTree` will be shown in the editor window, with active/disabled nodes highlighted.

### Core Node Types

![OpenBehaviourTree](https://media.githubusercontent.com/media/toastisme/OpenBehaviourTree/main/Runtime/Resources/Screenshots/NodeTypes.PNG?token=AOQPGDXCHQSWSHT5XVMS3I3CDARCW)

- **Priority Selector**: Calls child nodes in the order they were connected. If any return `Succeeded` when evaluated the remaining child nodes are not called and the node returns `Succeeded` to its parent.
- **Probability Selector**: Selects a single child node to call at random, based on probabilities defined by probability weights. Returns the result of the selected node to its parent.
- **Sequence Selector**: Calls child nodes from left to right. If any return `Failed` the node returns `Failed` to its parent. Only returns `Succeeded` if all child nodes return `Succeeded` when called.
- **Action**: Executes a coroutine, the inner workings of which determine the state of the node. This type cannot have any children.

### Nodes that can be added to core node types
- **Decorator**: The node it is attached to is only evaluated if the boolean condition is met (or the opposite if `invert condition` is selected)
- **Timeout**: If the node it is attached to remains in the `Running` state longer than the timeout, the node is cancelled and returns `Failed`.
- **Cooldown**: Stops the node it is attached to from being run while the cooldown is active. Can be set to start when the node fails, succeeds, or both.

## Running Tests
- In your project packages folder, open the manifest file and add `"com.davidmcdonagh-openbehaviourtree"` to `"testables"` (see the Enabling tests for a package section [here](https://docs.unity3d.com/Manual/cus-tests.html#tests)).
- In the Editor open `Window -> General -> Test Runner
- Under `PlayMode` you should now see tests for OpenBehaviourTree

## TODO
- Undo/Redo

## Further Info
- [Full documentation](https://toastisme.github.io/OpenBehaviourTree/)
- See also [OpenGOAP](https://github.com/toastisme/OpenGOAP)