Skip to main content

Ports & Triggers

Ports and triggers belong to entities and are arguably the most important part of Warudo scripting. They are used to pass data between entities, trigger actions, and provide user interaction in the editor.

Data Input Ports

Data input ports are used to provide data to an entity by either the user (using the editor) or another entity. Data inputs can be of various types, such as strings, numbers, booleans, or even complex types like structured data or arrays.

A data input is defined as a public field in an entity subclass, decorated with the [DataInput] attribute. In the Getting Started example, we saw a DataInput that defines a number slider:

[DataInput]
[IntegerSlider(1, 100)]
public int LuckyNumber = 42;

Here are a few more examples:

  • String Input:
    [DataInput]
    public string MyName = "Alice";
  • Enum Input: (Shown as a dropdown in the editor)
    public enum Color {
    Red,
    Green,
    Blue
    }

    [DataInput]
    public Color MyColor = Color.Red;
  • Array Input: (Shown as an editable list in the editor)
    [DataInput]
    public float[] MyFavoriteNumbers = new float[] { 3.14f, 2.718f };

This is how they look in the editor (we use a node here, but these data inputs work the same way in assets and plugins):

Note how they are initialized with the default values we specified in the code. These default values will be assigned to the data input again when the user clicks the "Reset" button next to the data input label.

A data input typically has a serializable type ("serializable" here means that the data value can be saved and restored when Warudo is closed and reopened). The following types are serializable by default:

  • Primitive types: int, float, bool, string, any Enum type
  • Unity types: Vector2, Vector3, Vector4, Color
  • Structured data types
  • Asset references
  • Arrays of serializable types

For nodes, it is possible to define non-serializable data inputs that cannot be edited in the editor but instead processed by the node itself. For example, the following code calls the ToString() method on the generic object data input:

[NodeType(Id = "dc28819e-5149-4573-945e-40e81e2874c4", Title = "ToString()", Category = "CATEGORY_ARITHMETIC")]
public class ToStringNode : Node {

[DataInput]
public object A; // Not serialized

[DataOutput]
[Label("OUTPUT_STRING")]
public string Result() {
return A?.ToString();
}

}

Other common data input types that are not serializable but can be passed between nodes include Dictionary, GameObject and Quaternion.

Attributes

Data input ports can be annotated with attributes to provide additional context to the editor. For example, the [IntegerSlider] attribute we saw earlier specifies that the data input should be displayed as a slider with a range of 1 to 100. Here are the supported attributes:

  • [Label(string label)]: Specifies the label of the data input.
  • [HideLabel]: Specifies that the label of the data input should be hidden.
  • [Description(string description)]: Specifies the description of the data input.
  • [HiddenIf(string methodName)]: Specifies that the data input should be hidden if the specified method returns true. The method must be a public or protected method in the entity class that returns a bool.
    [DataInput]
    public int MyNumber = 0;

    [DataInput]
    [HiddenIf(nameof(IsSecretDataInputHidden))]
    public string SecretDataInput = "I am hidden unless MyNumber is 42!";

    public bool IsSecretDataInputHidden() => MyNumber != 42;
  • [HiddenIf(string dataInputPortName, If @if, object value)]: Specifies that the data input should be hidden if the specified data input port satisfies the @if condition. The value must be a constant.
    [DataInput]
    public int MyNumber = 0;

    [DataInput]
    [HiddenIf(nameof(MyNumber), If.NotEqual, 42)]
    public string SecretDataInput = "I am hidden unless MyNumber is 42!";
  • [HiddenIf(string dataInputPortName, Is @is)]: Specifies that the data input should be hidden if the specified data input port satisfies the @is condition.
    [DataInput]
    public CharacterAsset MyCharacter;

    [DataInput]
    [HiddenIf(nameof(MyCharacter), Is.NullOrInactive)]
    public string SecretDataInput = "I am hidden unless MyCharacter is selected and active!";
  • [DisabledIf(...)]: Similar to [HiddenIf(...)], but the data input is disabled instead of hidden.
  • [Hidden]: Specifies that the data input should be always hidden. This is useful when you want to use a data input in code but not expose it to the user.
  • [Disabled]: Specifies that the data input should be always disabled (non-editable). This is useful for array data inputs that have a constant length, or for visualizing data inputs that are always programmatically set, etc.
  • [Section(string title)]: Specifies that the data input and all subsequent data inputs should be displayed in a new section with the specified title, unless another section is specified.
  • [SectionHiddenIf(string methodName)]: Requires attribute [Section]. Specifies that the section should be hidden if the specified method returns true. The method must be a public or protected method in the entity class that returns a bool. The @if and @is conditions are also supported.
  • [Markdown(bool primary = false)]: Specifies that the data input should be displayed as a Markdown text and cannot be edited. The data input must be of type string. If primary is true, the text will be displayed in a larger font size without a color background.
info

[HiddenIf] and [DisabledIf] attributes are evaluated every frame when the asset or node is visible in the editor. Therefore, you should avoid using expensive operations in these methods.

Some attributes are specific to certain data input types:

  • [IntegerSlider(int min, int max, int step = 1)]: Requires data type int or int[]. Specifies that the data input should be displayed as an integer slider with the specified range.
  • [FloatSlider(float min, float max, float step = 0.01f)]: Requires data type float or float[]. Specifies that the data input should be displayed as a float slider with the specified range.
  • [AutoCompleteResource(string resourceType, string defaultLabel = null)]: Requires data type string. Specifies that the data input should be displayed as an auto-complete list of resources of the specified type. For example, the "Character → Default Idle Animation" data input is defined as [AutoCompleteResource("CharacterAnimation")]. Please refer to the Resource Providers & Resolvers page for more information.
  • [AutoCompleteList(string methodName, bool forceSelection = false, string defaultLabel = null)]: Requires data type string. Specifies that the data input should be displayed as a dropdown menu generated by the specified method. The method must be a public or protected method in the entity class that returns a UniTask<AutoCompleteList>. Note that the method can be asynchronous. If forceSelection is true, the user can only select a value from the dropdown list, or the value is assigned null.
    using System.Linq;

    [DataInput]
    [AutoComplete(nameof(AutoCompleteVipName), forceSelection: true)]
    public string VipName = "Alice";

    protected async UniTask<AutoCompleteList> AutoCompleteVipName() {
    return AutoCompleteList.Single(vipNames.Select(name => new AutoCompleteEntry {
    label = name, // This is what the user sees
    value = name // This is what the field stores
    }).ToList());
    }

    private List<string> vipNames; // Entity-controlled runtime data

    // Some other code should update the vipNames list
    tip

    Autocomplete lists are useful when you want to provide a list of options that are not known at compile time. For example, you can use an autocomplete list to provide a list of files from a directory, a list of emotes from the remote server, etc. They are used very frequently in Warudo's internal nodes and assets!

  • [MultilineInput]: Requires data type string. Specifies that the data input should be displayed as a multiline text input field.
  • [CardSelect]: Requires an enum data type. Specifies that the data input should be displayed as a card selection list (similar to "Camera → Control Mode").
    public enum Color {
    [Label("#00FF00")]
    [Description("I am so hot!")]
    Red,
    [Label("#00FF00")]
    [Description("I am so natural!")]
    Green,
    [Label("#0000FF")]
    [Description("I am so cool!")]
    Blue
    }

    [DataInput]
    [CardSelect]
    public Color MyColor = Color.Red;

Enums

You can customize the editor labels of enum values by using the [Label(string label)] attribute in your enum type. For example:

public enum Color {
[Label("#FF0000")]
Red,
[Label("#00FF00")]
Green,
[Label("#0000FF")]
Blue
}

As mentioned in the Attributes section, you can also use the [Description(string description)] and [Icon(string icon)] attributes to show the enum input as a list of cards.

tip

icon should be a single SVG element (e.g., <svg>...</svg>). For example:

<svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 512 512">
<path>...</path>
</svg>

Asset References

warning

Asset references are only available in assets, nodes, and non-plugin structured data.

A data input can be used to reference another asset in the current scene. For example, the following code defines a data input that references a CharacterAsset:

[DataInput]
public CharacterAsset MyCharacter;

In the editor, the user can select a character asset from the dropdown list:

You can then access ports and triggers of the referenced asset:

[FlowInput]
public Continuation Enter() {
if (MyCharacter.IsNonNullAndActive()) {
MyCharacter.EnterExpression("Joy", transient: true); // Make the character smile!
}
return Exit;
}
tip

You can check if the asset is null or inactive by using asset.IsNullOrInactive(), and vice versa, asset.IsNonNullAndActive().

What if you want to filter the list of assets shown in the dropdown? You can use the [AssetFilter(string methodName)] attribute to specify a method that is used to filter the assets in the scene. The method must be a public or protected method that receives a parameter of the asset type and returns a bool. For example:

[DataInput]
[AssetFilter(nameof(FilterCharacterAsset))]
public CharacterAsset MyCharacter;

protected bool FilterCharacterAsset(CharacterAsset character) {
return character.Active; // Only show active characters
}

Accessing Data Inputs Programmatically

Let's say you have an entity. There are two ways to read its data inputs:

  1. Access the data input field directly. For example, if you have a node with a public data input field public int MyNumber = 42;, you can read the value of MyNumber directly by using node.MyNumber.
  2. Use the T GetDataInput<T>(string key) or object GetDataInput(string key) method. This method is available in all entities and returns the value of the data input with the specified name. For example, if you have a node with a data input named MyNumber, you can read the value of MyNumber by using node.GetDataInput<int>("MyNumber") (or node.GetDataInput<int>(nameof(node.MyNumber)) which is stylistically preferred).
tip

The key of a port is always the name of the field, unless the port is dynamically added (see Dynamic Ports)

The second method is useful when you need to access data inputs dynamically, for example, when you need to access a data input based on a string variable. (Also see Dynamic Ports.) Otherwise, the two methods do not have practical differences.

Similarly, to write to a data input, you can either assign a value to the data input field directly or use the void SetDataInput<T>(string key, T value) method. For example, to set the value of a data input named MyNumber, you can use node.MyNumber = 42 or node.SetDataInput("MyNumber", 42, broadcast: true) (or node.SetDataInput(nameof(node.MyNumber), 42, broadcast: true) which is stylistically preferred).

tip

Another alternative to SetDataInput(nameof(MyNumber), 42, broadcast: true) is to write:

MyNumber = 42;
BroadcastDataInput(nameof(MyNumber));

However, in this case, the second method is strongly recommended due to two reasons:

  1. It ensures watchers of this data input are notified of the change.
  2. By setting the broadcast parameter to true, the change is sent to the editor. Otherwise, you need to use BroadcastDataInput(string key) to manually send the change to the editor.

You should use the first method only if:

  1. You are updating the data input extremely frequently, and you do not need to send every change to the editor, i.e., you will call BroadcastDataInput(string key) sporadically. This saves performance.
  2. You explicitly do not want to notify watchers of this data input. This is rare.

Data Output Ports

Data output ports are node-specific and are used to provide data to other nodes. A data output is defined as a public method in a node subclass, decorated with the [DataOutput] attribute. The method can return any non-void type. Here is an example:

[DataOutput]
public int RandomNumber() {
return Random.Range(1, 100); // Return a random number between 1 and 100
}

Data outputs support a subset of data input attributes: [Label], [HideLabel], [Description], [HiddenIf], and [DisabledIf].

Flow Input Ports

Flow input ports are node-specific and are used to receive flow signals from other nodes to trigger certain actions. A flow input is defined as a public method in a node subclass, decorated with the [FlowInput] attribute. The method must return a flow output Continuation. Here is an example:

[DataInput]
public bool FlowToA = true;

[FlowInput]
public Continuation Enter() {
return FlowToA ? ExitA : ExitB; // If FlowToA is true, trigger ExitA; otherwise, trigger ExitB
}

[FlowOutput]
public Continuation ExitA;

[FlowOutput]
public Continuation ExitB;

Flow inputs support a subset of data input attributes: [Label], [HideLabel], and [Description]. Note that if the method is named Enter() and without a [Label] attribute, the label will be automatically set to the word "Enter" localized in the editor's language.

Flow Output Ports

Flow output ports are node-specific and are used to send flow signals to other nodes. A flow output is defined as a public field in a node subclass, decorated with the [FlowOutput] attribute. The field must be of type Continuation. See Flow Inputs for an example.

Flow outputs support a subset of data input attributes: [Label], [HideLabel], and [Description]. Note that if the field is named Exit and without a [Label] attribute, the label will be automatically set to the word "Exit" localized in the editor's language.

Triggers

Triggers are, simply put, buttons that can be clicked in the editor to trigger certain actions. A trigger is defined as a public method in an entity subclass, decorated with the [Trigger] attribute. Here is an example:

[Trigger]
public void ShowPopupMessage() {
Context.Service.PromptMessage("Title of the message", "Content of the message");
}

Which is rendered in the editor as:

When the user clicks the button, the ShowPopupMessage method is called.

Triggers support a subset of data input attributes: [Label], [HideLabel], [Description], [HiddenIf], [DisabledIf], [Section], and [SectionHiddenIf].

Asynchronous Triggers

Trigger methods can be asynchronous. For example, if you want to show a message after a delay, you can use UniTask:

[Trigger]
public async void ShowPopupMessageAfterDelay() { // Note the async keyword
await UniTask.Delay(TimeSpan.FromSeconds(1)); // Wait for 1 second
Context.Service.PromptMessage("Title of the message", "Content of the message");
}

A more practical example is to show a confirmation dialog before proceeding:

[Trigger]
public async void ShowConfirmationDialog() {
bool confirmed = await Context.Service.PromptConfirmation("Are you sure?", "Do you want to proceed?");
if (confirmed) {
// Proceed
}
}

Invoking Triggers Programmatically

Similar to accessing data inputs, you can invoke triggers programmatically by calling the method directly or using the void InvokeTrigger(string key) method on the entity. For example, to invoke a trigger named ShowPopupMessage, you can use entity.ShowPopupMessage() or entity.InvokeTrigger("ShowPopupMessage").

Port Order

By default, ports are automatically ordered based on their declaration order in the entity class. However, you can manually specify the order of ports by using the order parameter in the [DataInput], [DataOutput], [FlowInput], [FlowOutput], and [Trigger] attributes. For example:

[DataInput(order = 1)]
public int MyNumber = 42;

[DataInput(order = -1)]
public string MyString = "Hello, World!"; // This will be displayed before MyNumber

Dynamic Ports

Sometimes, you want to dynamically add or remove ports based on certain conditions. For example, the built-in Multi Gate node has a dynamic number of flow outputs (exits) based on the "Exit Count" data input.

To achieve this, you can access the underlying port collections at runtime:

FlowOutputPortCollection.GetPorts().Clear(); // Clear all flow output ports
for (var i = 1; i <= ExitCount; i++) {
AddFlowOutputPort("Exit" + i, new FlowOutputProperties {
label = "EXIT".Localized() + " " + i
}); // Create a new flow output port for each exit
}
Broadcast(); // Notify the editor that the ports have changed
tip

You can find the full source code of the Multi Gate node here.

info

You should not add or remove ports frequently (i.e., on every frame), as it may cause performance issues.

Dynamic Port Properties

You can also dynamically change the properties of a port, such as label, description, or type-specific properties. For example, consider the following:

[DataInput]
[IntegerSlider(1, 10)]
public int CurrentItem = 1;

private int itemCount = 10; // Assume we have 10 items initially

The range of the [IntegerSlider] is determined at compile time. But if itemCount changes, we will want to update the range of the slider. To do this, you can access the port properties directly:

var properties = GetDataInputPort(nameof(CurrentItem)).Properties;
properties.description = $"Select from {itemCount} items."; // Change the description

var typeProperties = (IntegerDataInputTypeProperties) properties.typeProperties; // Get type-specific properties
typeProperties.min = 1;
typeProperties.max = itemCount; // Change the slider range

BroadcastDataInputProperties(nameof(CurrentItem)); // Notify the editor that the properties have changed

Contributions

Edit on Github

Authors

Last updated on 2024.06.22