Dialogs
Structure
All in-game dialog is encoded as a state machine (a single state machine can represent the whole game dialog). States of this machine correspond to NPC text, while transitions are (usually) PC replies.
On a given state (when presented when a text), the PC can choose which reply to select, i.e. which transition to follow to a target state. Some transitions are final (they close the dialog instead of having a target state), and some transitions also happen to not have PC text associated.
Since each state has a NPC speaker, states are indexed by pairs (actor, key)
, where the actor is the identity of the speaker; this is also how dialog is encoded in the game files: all dialog belonging to a single actor (states of this actor, and transitions from these states) are stored in a single game resource. Accordingly, the Actor
data structure corresponds to all text said by a given NPC speaker, and all PC replies to this text.
Displaying dialogs
Reading an actor is as simple as loading it and letting the REPL display it: just try invoking actor("imoen")
and look at the result.
InfinityEngine.actor
— Functionactor([game], "name")
Loads the named actor from game files, or creates an empty actor if none exists with this name.
Creating new dialogs
Several functions enable creating and editing game dialogs. The easiest feature is creation of entirely new dialogs. For this, it is enough to give a list of states connected by transitions.
The currently active actor is selected by the actor
function. It is legal to call this on a non-existing actor name; it simply creates an empty actor, to which states and transitions will then be attached.
The say
function creates states, while reply
creates transitions. The module maintains a “last added state” variable. This variable is updated by calls to say
to the last state added, and used by calls to reply
to determine from which state a transition must be created.
InfinityEngine.say
— Functionsay({text | (label => text)}*; priority, trigger)
Introduces states of dialog for the current actor. If the label is omitted, a default (numeric, increasing) label will be inserted (although inserting an explicit label makes the state easier to reach).
A single say
call is equivalent to several successive say
calls for the same current actor.
Special cases
- implicit, text-less transitions:
say(text1) say(text2)
; - multi-say:
say(text1, text2, ...)
— actually equivalent to the previous form; - chain with actor change:
actor(name1) say(text1) actor(name2) say(text2)...
;
InfinityEngine.reply
— Functionreply(text => label)
Introduces a state transition (player reply) pointing to the given label. The label may be one of:
("actor", state)
(equivalently"actor" => state
);state
(uses current target actor);exit
(creates a final transition).
State may be either numeric (referring to the base game's states) or string. In the latter case, if it does not contain a slash, it will be prefixed by the current namespace : "namespace/state"
. This prevents states from different namespaces from interfering.
Special forms:
reply(exit)
creates a text-less, final transition;reply(text)
creates a pending transition: this will be connected to the next state inserted (viasay
).
Examples:
# chain to other actor:
reply("Say Hi to Hull" => "hull" => 0)
# connect pending transition:
say("How do you do?") reply("Fine!") say("Let'sa go!") reply(exit)
State labels
Numeric state keys correspond in principle to original game dialog. New labels should use strings.
Label strings are namespaced to avoid collisions.
Internally, a numeric label is produced by hashing the strings (to 64-bit integers). This allows adding approximately 2³² states (4 billion) to any actor before risking a key collision, and approximately 2⁵⁰ states before hitting any target lower than 16000 (where original game labels presumably reside).
Final transitions
Chaining and implicit transitions
Pending transitions
The implicit transitions created by chaining are all text-less transitions. If PC comments are needed, this can be done via pending transitions.
A pending transition is a transition with no target state indicated. The transition will be connected to the next state to be added. Inserting such transitions in the middle of a say
chain has the effect of inserting PC text while maintaining the structure of the chain: for example, say(A); reply(a); say(B)
is equivalent to say(A); reply(a => labelB); say(labelB => B)
, without the need to give an explicit label to the target state.
The current pending transition must be connected (by calling say
or interject
, both of which always resolve existing pending transitions) before any other transition is created (by calling reply
).
Extending existing dialogs
The from
function allows changing the value of the “last added state” variable to any actor and any state.
InfinityEngine.from
— Functionfrom([game], [actor], label)
Sets current state to actor
, label
. The state must exist.
Note that this is not always the same actor as the speaking actor selected by actor
: namely, from
selects a source actor (i.e. already existing states), whereas actor
selects a target actor, for which states will be inserted.
With a combination of from
, actor
and reply
it is possible to extend existing states for an actor (by adding new transitions) and to add new states.
from("imoen", 0)
reply("Oh, hi Imoen!" => "new state")
say("new state" => "Hi you! Now we go on to our normal conversation.")
Inserting into existing dialogs
New states can also be inserted into an existing transition. Say that A
is the last created (source) state, with transitions tᵢ
to states Bᵢ
: A ——tᵢ——→ Bᵢ
. The function interject
can insert a new state X
at the tail end of all the arrows tᵢ
: A ——→X——tᵢ——→Bᵢ
. Namely, there is now a single text-less transition from A
to X
, and all the original transitions from A
now start from state X
.
InfinityEngine.interject
— Functioninterject({text | (label => text)}*; priority, trigger)
Inserts text inside existing dialog. The new state(s) are inserted just after the current state, using tail insertions.
interject
moves the “source state” pointer to the newly created state X
, so that it is possible to chain calls. For example, after from(A); interject(X); interject(Y)
, the result will be something like A——→X——→Y——tᵢ——→Bᵢ
.
interject
and pending transitions
If the source state A
has a pending transition when interject
is called, then instead of creating a new text-less transition A→X
, this pending transition is used (and connected) instead. This allows replacing the default text-less transition A→X
by a transition with text A——x—→X
in the following way: from(A); reply(x); interject(X)
.
Deleting existing dialogs
Not currently possible (and not a very high priority).
Attaching data to dialogs
TODO.
State priority
Triggers
Actions and journal
InfinityEngine.trigger
— Functiontrigger(string)
Attaches a trigger to the next transition or state.
InfinityEngine.action
— Functionaction(string)
Attaches an action to the latest transition.
InfinityEngine.journal
— Functionjournal(string)
Attaches a journal entry to the latest transition.