Skip to content

Yarn Syntax Reference

Faulty edited this page Feb 24, 2021 · 1 revision

This is a quick reference guide for the various syntax you can write in your Yarn files. It is not intended to be a tutorial - see here for that! This document is instead a helpful reference if you’re wondering how a line of Yarn code needs to be structured to work.

 

The Basics

Yarn is a programming language designed for dialogue. This means it has to follow some rules, so a computer can make sense of it.

Yarn files are broken up into nodes. Each node has multiple different statements that make it up. In general, Yarn Spinner treats each new line in the Yarn file as a separate element. It needs to work out what that may be. Nodes

Everything in Yarn is a node, and all your dialogue is contained within a node. Your Yarn files might contain multiple nodes or only a single node, but all Yarn has to be part of a node. All nodes have the same basic structure:

header
---
body
===

The header is separated from the body by --- (three - symbols on a line by themselves). The node is terminated by === (three = symbols on a line by themselves).

If you are using a visual editor, these nodes make up the little boxes you see on the canvas.

 

Header

The header contains metadata for the node, and can have as many pieces of node metadata as you wish. Metadata is created as key-value pairs. Each metadata piece has the same layout:

key:value

Every metadata piece has to go one a line by itself.

An example of this: myKey:myValue has a key of myKey and a value of myValue.

 

Title

The title header defines the name of the node. All nodes must have a title.

This header will look something like title:nodename. The name of the node, in this case nodename, needs to be unique across the entire project, and can be any combination of upper or lower case letters and numbers. Node titles can’t have spaces.

The value of this is case sensitive, so nodename is not the same as nodeName.

 

Tags

Nodes can have additional metadata about them, which you can define in the tags header. This is a space-separated list of words.

Unlike the title metadata, the values don’t have to be unique.

 

Body

The body, which are the parts of the node in-between the --- and === lines, are where the bulk of your time will be spent and is where all of your dialogue will go.

 

Dialogue

The majority of your Yarn files will normally be dialogue - lines spoken by your characters in the game. Each line of dialogue is treated as a new dialogue statement by Yarn Spinner. Almost any text can go inside your dialogue.

Most lines of dialogue are written like the following:

Player: Hey, Sally.
Sally: Oh! Hi.
Sally: You snuck up on me.

In this case, we have the characters names preceding the text we want them to say. This is optional, and you can just have the dialogue if you prefer:

Hey, Sally.
Oh! Hi.
You snuck up on me.

If you don’t include character names, your game will need some other way of working out which line is for which character, and we leave that up to you.

 

Inline Expressions

Lines of dialogue can have expressions embedded inside them. These expressions will be expanded when the dialogue runner encounters them. Any expression inside { and } will be evaluated and rendered in-line with the rest of the text.

For example, imagine you have a variable called $gold. You can display the contents of this variable to the player like this:

You have {$gold} gold pieces.

More complex expressions can be used as well. For example, you can do maths with your values:

If you had twice as much gold, you'd have {$gold * 2} gold pieces!

Inline expressions can be used with format functions, especially when you need to localise your text. Options

Options are how you can move between different nodes.

[[Text to show to the user|NodeName]]

Options are broken up into two parts seperated by a vertical bar symbol (|). The first part is any text that is to be shown to the user, and the second part after the bar is the name of the node. The node name must be the name of another node somewhere else in your Yarn files, or else the program will not work. After the player selects an option the node linked, in this case NodeName, will be loaded and the story will continue from there.

Inline expressions can be used in options. Learn more about options in Controlling Dialogue. Jumps

A jump tells Yarn Spinner to start running a different node. They’re not shown to the player.

[[NodeName]]

When Yarn Spinner encounters this line, it will immediately start running the contents of the NodeName node. Learn more about jumps in Controlling Dialogue.

 

Shortcut Options

Shortcut options let you create simple branching dialogue without needing to create new nodes. Each shortcut starts with the -> arrow and then the line of text that should be presented to the player.

Where should we go next?
-> North
-> South
-> East
-> West

This would show the line Where should we go next? and then the options of North, South, East, West. Learn more about shortcut options in Controlling Dialogue.

 

Shortcut scope

Shortcuts allow you add additional statements below the shortcut to run additional lines, and in many cases is used for setting or updating variables. The new statements have to go below the shortcut option they are a part of and need to be indented relative to the shortcut.

-> North
    It is cold to the north.
-> South
    You've heard that the weather to the south is sunny.
-> East
    You're not sure if the rumours of bandits to the east are true.
-> West

You continue your journey.

Different lines will be shown to the player depending on what options they select:

  • In the above example if the player selected the North option the dialogue line It is cold to the north would be shown after and then You continue your journey.
  • If they select South or East, they’ll see different lines before You continue your journey
  • If they select West, they’ll only see You continue your journey.

The indentation must be kept consistent, or else Yarn Spinner won’t be able to work out your intent.

 

Shortcut conditionals

Shortcuts can also be limited to only be presented if needed. You do this by adding an if statement to the end of the shortcut.

-> Locked Trapdoor <<if $hasKey is true>>
-> North
-> South
-> East
-> West

In the above example the shortcut Locked Trapdoor will only appear as an option your players can pick if the variable $hasKey is true. If $hasKey is false only the four directions will appear.

 

Variables

You can store information inside variables.

Variable names are any combination of letters and numbers preceded by the $ symbol. Variable names are case sensitive, so $varName and $varname are two different variables as far as Yarn Spinner is concerned.

Variables are global in scope, so they exist across all nodes across all files. This means you can declare a variable in one file and then later use that variable in another file.

 

Values

Variables can hold numbers, text, true and false, or null.

 

Setting values

You can set and update values inside of variables using the set operation:

<<set $var to 1>>

The set operation takes the form of the command opening symbol <<, followed immediately by the set keyword, then the variable, $var in our case, then the to keyword, and finally the value you want.

You can also use the = in place of the to keyword if you wish.

 

Numbers

Numbers are always decimals (technically, floating point) regardless of what values you give them. This means if you give a variable the value of 1 when you get it back out from Yarn Spinner it will be 1.0. Text

Text inside variables can be anything you want but must be contained with quotation marks. This means that while <<set $var to "hello">> is valid, <<set $var to hello>> is not. True and False

Variables can be set to be true or false using the keywords true or TRUE, and false and FALSE. These are case sensitive, so <<set $var to true>> will work, but <<set $var to True>> will not. Conditionals

Conditionals are how you can create different branching dialogue and events based on logical statements. All conditionals take the same basic structure, an if statement, then zero or more elseif statements, then zero or one else statements, and finally the endif.

Yarn Spinner will work its way down the conditional until it meets a piece of it it can run and presents the statements for that piece.

<<if $money > 1>>
	I would like a horse please.
	<<set $money to $money - 2>>
	<<set $hasHorse to true>>
<<elseif $money eq 1 >>
	Just a drink thanks.
	<<set $money to $money - 1>>
<<else>>
	Drat, I can't afford anything.
<<endif>>

Take the above example, if $money happens to be 1 then the line Just a drink thanks. will be shown, but if $money was 5 then the line I would like a horse please. would be shown.

It can be more readable to indent lines inside the if, but it’s not required. Learn more about conditionals in Controlling Dialogue. if, elseif, else and endif

The if statement opens a conditional and is comprised of the << command opening keyword followed immediately by the if keyword, then an expression that controls the if and finally the command close keyword >>.

Any lines that go between the if and the next part of the conditional, so either an elseif, an else, or an endif, is shown if the expression is ultimately true.

 

elseif

An elseif is an optional part of the conditional and works similar to the if but instead uses the elseif keyword. Each elseif is identical to its if counterpart in structure and you can have as many elseif's as you want, including none.

<<elseif $money eq 1>>

else

The else is the fallback of the conditional and is run only if the if and all its elseif's evaluate to false. It is optional.

<>

endif

The <> finishes the conditional statement and is required. It is needed so Yarn Spinner knows when the conditional has concluded. Expression

An expression is a mathematical or logical operation and work and looks like a line of maths. For the expression to be useful in the conditional it needs to eventually evaluate to true or false. If the expression results in true it will be the part of the conditional that gets run.

All expressions follow the same pattern of a subexpression followed by an operator and then another subexpression. The subexpressions can further broken up into other expression if needed:

<<if ($counter + 1) >= ($max - 2)>>

Logical operators

Yarn Spinner supports the following logical operators and most have multiple ways being written:

Equality: eq or is or ==
Inequality: neq or !
Greater than: gt or >
Less than: lt or <
Less than or equal to: lte or <=
Greater than or equal to: gte or >=
Boolean OR: or or ||
Boolean XOR: xor or ^
Boolean Negation: !
Boolean AND: and or &&

Maths operators

Addition: +
Subtraction: -
Multiplication: *
Division: /
Truncating Remainder Division: %
Brackets: ( to open the brackets and ) to close them.

Order of operations

Yarn Spinner follows a fairly standard order of operations, and falling back to using left to right when operators are of equivalent priority. The order of operations is as follows:

Brackets
Boolean Negation
Multiplication, Division, and Truncating Remainder Division
Addition, Subtraction
Less than or equals, Greater than or equals, Less than, Greater than
Equality, Inequality
Boolean AND, Boolean OR, Boolean XOR

Commands

Commands are a way of Yarn Spinner communicating back to the game that events have happened that need to be handled. These are often used to trigger achievements and to move characters and cameras around to where they need to be.

Commands start by having the command opening symbol << then any text you want sent over to the game, and finish with the command close symbol >>. As an example:

<> <>

Commands by themselves do nothing, you need to handle these messages yourself.

Inline expressions can be used in options. To learn more about how to define your own commands for your game, see Working With Commands. Localisation Tags

Localisation tags are a way of marking lines of dialogue to the localisation system. If you aren’t localising your game you don’t need them and will not encounter them. Each tag starts with a # symbol and then have a line keyword and an autogenerated value.

#line:a8e70c

The tags always go on the end of the line and should never be edited or created manually. You will find localisation tags at the end of dialogue lines, shortcuts, and options.

Player: Hey. #line:a8e70c Sally: Oh! Hi. #line:2dc39b

See you later. #line:0facf7

Format functions

Format functions allow you to select content based on a value. Format functions are useful for adapting a line to be grammatically valid based on the value of a variable.

For example, if you want to say the sentence “I have 1 apple”, the word “apple” in English needs to change depending on whether you have 1 apple or 2 apples. Format functions allow you to switch the word out.

Format functions are extremely useful for localisation, because they can be different in each of the string tables that you’re working with.

Format functions start and end with [ and ]. Inside these braces, you specify which specific format function you want to use, and then a list of categories and replacements to use. The format function syntax is based on similar implementations in Unreal, Unity, and Unicode’s MessageFormat.

There are three format functions available: select, plural and ordinal. select

The select format function allows you to use a variable’s value to select between a fixed set of options.

[select {$value} option1="replacement1" option2="replacement2"]

Based on the value of the variable you provide, different replacement will be selected. In the above example, if the value of $value is "option1", then the text "replacement1" will be inserted; if $value is "option2", then "replacement2" will be inserted. There’s no limit on the number of options. If $value is neither, an error value will be inserted. The select format function does not look for specific values; rather, it’s a generic text replacement tool.

This is most useful for when you want to use different pronouns based on a variable storing the gender of a character.

For example, imagine that the variable $gender contains the gender of a character, and may contain the values “m” for male, “f” for female, or “nb” for nonbinary.

I haven't seen [select {$gender} m="him" f="her" nb="them"].

plural and ordinal

The plural and ordinal format functions maps a number value to its plural class, based on the current language.

Different languages use different rules for pluralising numbers. In English (and many other languages), there are two plural classes numbers: if an integer is 1, it is singular, and otherwise, it’s plural; other rules apply to different languages.

These format functions let you pass in a variable, and select the appropriate text to display based on the plural class:

The plural format function selects a number’s cardinal plural class, used for counting values: “1 apple”, “2 apples”, and so on.
The ordinal format function selects a number’s ordinal plural class, used for ranking values: “1st”, “2nd”, and so on.

The available plural classes for use in plural and ordinal are: zero, one, two, few, many, and other. Different plural classes exist in different languages; when localising a line that contains the plural or ordinal format functions, you will need to use the plural classes that apply to your target language.

The rules used to select the plural class for the number are derived from Unicode’s CLDR. The specific rules used at run-time are determined by the text language property on the Dialogue Runner.

 

plural

The plural format function selects text based on the input value’s cardinal plural class.

For example, imagine you have a variable called $apples, which represents how many apples the player has, and you want to show this to the player. You can use the plural format function for this, like so:

You have {$apples} [plural {$apples} one="apple" other="apples"]

When the language text of the dialogue runner is set to English, this will render in the following ways:

$apple == 1: “You have 1 apple”
$apple == 2: “You have 2 apples”
$apple == 0: “You have 0 apples”

 

ordinal

The plural format function selects text based on the input value’s ordinal plural class.

For example, imagine you have a variable called $race_position, which contains the position that the player came in in a race. You can use the ordinal format function for this, like so:

I came in {$race_position}[ordinal {$race_position} one="st" two="nd" few="rd" other="th"] place!

When the language text of the dialogue runner is set to English, this will render in the following ways:

$race_position == 1: “I came in 1st place!”
$race_position == 2: “I came in 2nd place!”
$race_position == 3: “I came in 3rd place!”
$race_position == 6: “I came in 6th place!”

 

Value insertion

In a format function’s replacement value, the % character is replaced with the original value that was used to select the appropriate text to use. For example, going back to our “number of apples” example, you can write the line in a way that only causes numbers to appear in the plural case:

You have [plural {$apples} one="an apple" other="% apples"]

In English, this will be rendered as:

$apple == 1: “You have an apple”
$apple == 2: “You have 2 apples”