No longer under active development for now. If I need to home-spin a graph editor in the future, I may return to this.
A flavor of Behavior Trees for Godot 4 featuring a GraphEdit-based editor. Latest tested Godot version: 4.0.3 stable.
I like being able to edit and author trees visually. I couldn't find a Behavior Tree solution for Godot 4 that used GraphEdit.
| Beehave is very good. And more stable. Maybe use it instead of this right now?
- Please evaluate in isolation first. Tool has only recently stabilized and may have dragons lurking.
- BehTrees contain BehNodes. Both are Resources.
- tick() BehTrees.
fn tick(dt: float, bb: Dictionary) -> BehConst.Status
- It takes a delta-time "dt" and arbitrary-purpose blackboard "bb". Returns Busy, Resolved, or Failed.
- e.g.: Call tick() in a Node's _process.
- To create custom behavior, define your own BehNode implementations. BehNode is a Resource type. New BehNodes MUST be
@tool
!- There are a handful of useful built-in impls, and a template txt for new BehNodes in the addon folder.
- New BehNodes MUST be
@tool
! This is due to editor limitations and the decision to reduce boilerplate,BehNode
features overrideable methods to define how it looks in the BehTree Editor.
- Do not share tree references across multiple Node runners and expect sensible behavior.
bb
is supposed to be the sole source of state, but this is not true in reality currently, due to Select / Sequence node impls.
- Similarly to the above, BehNodes should be kept inside BehTrees and not in the file system.
- More specifically, a BehNode should be unique to a given BehTree, because the BehNode stores tree-dependent information (currently). This can be fixed by preventing BehNodes from storing their own children, and enforcing
bb
as the source of all data, which would allow BehNodes to be safely re-used in different trees. None of that is the case currently.
- More specifically, a BehNode should be unique to a given BehTree, because the BehNode stores tree-dependent information (currently). This can be fixed by preventing BehNodes from storing their own children, and enforcing
- Performance has NOT been evaluated. Production use is currently just beginning. 1.0 indicates the tool has been stable through testing, but in-game performance is still to be determined, and the current intended use is for smaller projects that are unlikely to encounter performance issues.
- After defining a new BehNode @tool, you likely need to Reload the project (Project -> Reload Current Project) to avoid issues with the add-node resource picker.
- Inspect a BehTree to open the BehTree Editor in the bottom dock.
- Right-click on the graph to add a new BehNode.
- Double-click a BehNode in a BehTree to inspect it and edit any @exported variables.
- Delete nodes by selecting or drag-selecting and hitting the Delete key.
- Undo/redo is supported. If you encounter errors please file an Issue!
- Customize your BehNodes' editor appearances by overriding
editor_
methods defined in BehNode (see:tree/beh_node.gd
).
Sequence
: Supports N children, ticking one at a time, in order. Once a child is ticked, the sequence waits until re-ticking (this behavior might change in a major version revision).Set
: Supports N children, ticking all of them each tick().Select
: Supports N children. Ticks children in order until one of them returns Resolved or Busy. A Busy child is re-ticked on the next call.Select Random
: Supports N children. Randomly ticks one of its children. If a child returns Busy, that child is re-ticked on the next call.Condition
: Supports up to 2 children. The first child is ticked if the condition returns true, otherwise the second child is ticked, if it exists. This behavior can be inverted.
Select, Select Random, and Condition exhibit "sticky" child-ticking behaviors. When a child returns Busy, it gets ticked again when the parent selector gets its next tick(), ignoring its original selector condition as long as that child is still Busy. Some selectors can disable this behavior as an inspector option (a mode referred to as "rude").
A base class BehNodeXMultiChildren
is used by nodes that can have children. Sequence and Condition are good examples of correct implemenations. In particular:
get_can_add_child()
must always return true or always return false. To limit adding children only sometimes, return false intry_add_child()
instead.get_children()
must return the mutable backing array containing a BehNode's children. This allows the editor to sort those children according to editor node positions.
Incorrectly implementing node children can break the tree. Fun!
I'm a sucker for alphabetical sorting. I hope this naming scheme doesn't drive you mad.
BehNodeAFooBar
- class_name pattern for top-level, general-purpose, built-in BehNode implementors.BehNodeBFooBar
- class_name pattern for built-in BehNodes that are more use-case specific.BehNodeCFooBar
- class_name pattern that is unused, but that in theory, would be used for game-specific implementors.BehNodeXFooBar
- class_name pattern for base classes intended for inheritance but not direct usage.
I find this pattern useful when managing a large collection of Resource implemenations in a resource picker.
VERY verbose, but useful, log printing can be enabled in beh_editor.gd
.dprintd(s: String)
. (What can I say; Godot's editor tool debugging support is somewhat lackluster).
Dual-licensed under MIT & Apache 2.0.
- Consider beehave, which offers
Node
-based (rather thanResource
+Editor-based) behavior trees.
<3