This manual documents TIPS (The Inside Place Storymaker), a tool for creating interactive stories on the Playdate. You'll work primarily in Articy to build your story, and the system will handle converting it to the proper JSON format for the game engine.
In Articy, you'll create a Flow container that holds different types of story nodes. Each node determines how the player progresses through your game.
Flow nodes can be configured as one of several types by adding specific text to their "Text" field:
Type | Keyword | Description |
---|---|---|
Map | map |
A collection of locations the player can visit |
Video | video |
Plays a video file (intro, cutscene, etc.) |
Cutscene | cutscene |
A linear dialogue sequence that plays automatically |
Endgame | endgame |
Defines multiple endings based on player score |
To create a map with visitable locations:
map
to its Text fieldmap, backgroundImage=map1.png, bgm=birdsong
Each child node becomes a location the player can visit on the map screen.
video
in its Text fieldvideo, videoPath=intro.pdv, audioPath=intro.wav
cutscene
in its Text fieldCutscenes are linear scenes that play without requiring the player to select a map location. They're perfect for story moments that happen between exploration segments.
For endgame nodes with multiple endings, use this format:
endgame
type=endgame
endings:
Good Ending=100,ending_good.pdv,ending_good.wav
Bad Ending=50,ending_bad.pdv,ending_bad.wav
Terrible Ending=0,ending_terrible.pdv,ending_terrible.wav
Where each ending line follows the format: Name=RequiredScore,VideoPath,AudioPath
In Articy, you'll build scenes by connecting dialogue nodes to create a flow of conversation and events.
A typical scene consists of dialogue nodes connected in sequence, with possible branches for choices. When exported, these scenes are converted to JSON files for the game engine.
Stage directions control what happens in your scenes. Add them as comma-separated key-value pairs like key=value
.
Command | Description | Example |
---|---|---|
image |
Sets the background image | image=background1.png |
clearPortrait |
Removes the character portrait | clearPortrait=true or just clearPortrait |
characterPortrait |
Sets the character portrait | characterPortrait=Logan |
characterMood |
Sets the character's mood/expression | characterMood=happy |
Command | Description | Example |
---|---|---|
bgm |
Sets background music | bgm=mysterious.wav |
sfx |
Plays a sound effect | sfx=bloop |
Command | Description | Example |
---|---|---|
animation |
Adds animation to the text box | animation=shake:500:10 |
portraitAnimation |
Adds animation to the character portrait | portraitAnimation=slideUp:200:10 |
textSpeed |
Controls the speed of text display | textSpeed=20 |
imageAnimation |
Animates the background | imageAnimation=handheld:5:1 |
Command | Description | Example |
---|---|---|
addInventoryItem |
Adds an item to inventory | addInventoryItem=item_a_sandwich |
removeInventoryItem |
Removes an item from inventory | removeInventoryItem=item_a_sandwich |
unlockLocation |
Unlocks a location on the map | unlockLocation=FFr_01863B14 |
addScore |
Adds points to player score | addScore=10 |
failState |
Triggers game over with specified type | failState=FALL |
For animation
and portraitAnimation
, the format is: type:duration:intensity
shake
- Shakes the element randomly in different directionsalarm
- Creates a flashing/alert effectwoozy
- Creates a wavy, disorienting animationbounce
- Makes the element bounce up and downslideUp
- Slides the element up into viewslideDown
- Slides the element down into view500
)10
)Apply effects to background images using imageAnimation
:
handheld
- Slight camera shake effect
imageAnimation=handheld:intensity:speed
imageAnimation=handheld:5:1
stop
- Stops any active background animation
imageAnimation=stop
Dialogue nodes are the basic building blocks of a scene and can include text, speaker information, and stage directions.
{
"id": "DFr_12345678",
"type": "dialogue",
"text": "Hello, world!",
"speaker": "Character Name",
"next": ["NextNodeId"],
"animation": {
"type": "slideUp",
"duration": 500,
"intensity": 10
}
}
id
: Unique identifier for the node (usually starts with DFr_
)type
: Must be "dialogue"
text
: The text content to displaynext
: Array of node IDs to proceed to (usually a single ID)speaker
: Character name to display (empty for narrator)menuText
: Text to display in choice menus referencing this nodeYou can trigger a game over or fail state from any dialogue node using the failState
stage direction:
{
"id": "DFr_DEADEND",
"type": "dialogue",
"text": "You feel a strange sensation as the world fades to black...",
"failState": {
"type": "DEATH"
}
}
When a fail state is triggered:
To create a fail state, add failState=TYPE
to your stage directions, where TYPE can be:
DEATH
- Generic death/failureFALL
- Death by fallingDROWN
- Death by drowningATTACK
- Death by attackThe type affects which game over screen and sound effects are played.
Choice nodes present the player with multiple options to select from.
Choices are created automatically when you connect one dialogue node to multiple subsequent dialogue nodes:
For each dialogue node that represents a choice option:
requiredItem=item_key, fallbackText=You need a key...
Available choice properties:
Property | Description | Where to Enter |
---|---|---|
menuText |
Text displayed for the option | Menu Text field in Articy |
requiredItem |
Item needed to select this option | Stage Directions field |
fallbackText |
Text shown when required item is missing | Stage Directions field |
consumeItem |
Override automatic consumption | Stage Directions field |
If no Menu Text is provided, the system will use the first few words of the dialogue node's text as the menu option.
TIPS supports hub nodes (choice nodes that you can loop back to repeatedly) for creating interactive dialogues with multiple selectable options. By default, previously selected options can be hidden to encourage players to explore all dialogue paths.
When a player returns to a hub node, previously selected options are automatically filtered out until all options have been explored. This creates a natural progression through dialogue trees and ensures players see all content.
To create an effective hub node, ensure the text in your hub node works regardless of which option was previously chosen. You can use this pattern to create conversation trees where players explore topics in any order.
Hub Node: "What would you like to know about?"
↓
Option 1: "Tell me about your background." → Dialogue → Return to Hub
↓
Option 2: "What are you working on?" → Dialogue → Return to Hub
↓
Option 3: "I need to go now." → Exit dialogue
Previously selected options will not appear in the list until all options have been explored, allowing for clean dialogue progression.
TIPS supports two types of inventory items that serve different gameplay purposes.
Narrative items advance the story when examined in the player's inventory. When selected, they trigger their own scene.
ItemType = 0
(for narrative)isConsumable = false
(narrative items are typically not consumed)ItemName
and ItemDescription
ItemImage
to the filename (e.g., "laptop.png")For each narrative item, create a corresponding Flow fragment in your main Flow:
item
to its Text fieldThe system will automatically link narrative items to their scenes based on matching names.
Utility items are used to solve puzzles and unlock choices in dialogue. They can be consumable or reusable.
ItemType = 1
(for utility)isConsumable = true/false
depending on whether it should disappear after useItemName
and ItemDescription
ItemImage
to the filenameItemHint
that describes what the item does - this will be shown if the player tries to use a choice that requires this itemTo create a choice that requires an item, add these stage directions to your dialogue node:
requiredItem=item_a_sandwich, fallbackText=You need something to eat...
Utility items that are marked as consumable in Articy are consumed automatically when used unless you specifically flag them otherwise with consumeItem=false
in the stage directions.
TIPS includes a smart system that automatically discards utility items that are no longer needed:
To make this work, each utility item tracks which scenes it can be used in via the usableIn
array in the generated JSON.
When converted, items appear in inventory_items.json
like this:
// Narrative item example:
{
"key": "item_logans_laptop",
"name": "Logan's Laptop",
"description": "The laptop from Logan's house.",
"image": "assets/images/items/no_image.png",
"type": "narrative",
"consumable": false,
"useScript": "FFr_EB82AA1A.json"
}
// Utility item example:
{
"key": "item_a_sandwich",
"name": "Sandwich",
"description": "A consumable item.",
"image": "assets/images/items/no_image.png",
"type": "utility",
"consumable": true,
"usableIn": ["FFr_D09B2739", "FFr_A5F93E81", "FFr_EB82AA1A"],
"hint": "Is soft and wet and crispy all at once."
}
You can add various animations to enhance your scenes directly through stage directions in Articy.
To animate the text box containing dialogue, add an animation
stage direction:
animation=shake:500:10
Format: animation=type:duration:intensity
Animation Type | Effect | Example Usage |
---|---|---|
shake |
Shakes the text box randomly | For earthquakes, explosions, strong emotions |
alarm |
Flashes/pulses the text box | For alerts, warnings, sudden realization |
woozy |
Wavy, disorienting motion | For dizziness, intoxication, confusion |
bounce |
Text box bounces up and down | For excitement, happiness, emphasis |
slideUp |
Slides the text box up into place | For smooth scene transitions |
slideDown |
Slides the text box down into place | For smooth scene transitions |
To animate the character portrait, use the portraitAnimation
stage direction:
portraitAnimation=slideUp:200:10
The same animation types apply to portraits. Common uses:
portraitAnimation=shake:500:8
- Character is angry or surprisedportraitAnimation=bounce:500:10
- Character is excited or energeticportraitAnimation=slideUp:300:0
- Character smoothly enters the sceneportraitAnimation=slideDown:300:0
- Character smoothly exits the sceneApply effects to background images using imageAnimation
:
imageAnimation=handheld:5:1
handheld
- Creates a subtle camera shake effect
imageAnimation=handheld:intensity:speed
stop
- Stops any active background animation
imageAnimation=stop
The game world is organized into maps containing locations that can be visited.
story.json
{
"type": "map",
"mapId": 1,
"backgroundImage": "map1.png",
"bgm": "birdsong",
"unlocked": false,
"locations": [
{
"name": "Location Name",
"locationId": "FFr_12345678",
"unlocked": true,
"optional": false,
"conditionalUnlock": ["FFr_PreviousLocationId"],
"sprite": "point.png",
"nextEvent": {
"type": "text",
"script": "FFr_12345678.json"
},
"coordinates": {
"x": 100,
"y": 120
}
}
]
}
Each location on a map is a child Flow fragment inside the map node. Configure locations using stage directions in the Text field:
name=Secret Cave, unlocked=true, optional=false, coordinates=150:65
Available location properties:
Property | Description | Example |
---|---|---|
name |
Display name of the location | name=Lockers |
unlocked |
Whether location starts unlocked | unlocked=true |
optional |
Whether completion is optional | optional=false |
coordinates |
Position on map screen (x:y) | coordinates=100:120 |
Locations can also be unlocked by completing other locations. This happens automatically based on the connections between Flow fragments.
TIPS supports multiple media formats.
{
"type": "video",
"videoPath": "intro2.pdv",
"audioPath": "intro2.wav"
}
TIPS supports both background music and sound effects to enhance the player experience.
Background music can be set for entire scenes or maps using the bgm
directive:
bgm=mysterious.wav
This will play the music file until another BGM is set or the current one is stopped. To stop music, use:
bgm=
Place music files in assets/sounds/
folder with supported formats:
Trigger sound effects at specific points in dialogue with the sfx
directive:
sfx=bloop
Sound effects are ideal for:
ADPCM is the preferred audio format for Playdate. You can convert to ADPCM using:
ffmpeg -i input.mp3 -acodec adpcm_ima_wav output.wav
TIPS tracks a player score that determines which ending they'll get.
addScore
stage directionaddScore=10
In story.json
, endings can be defined based on score thresholds:
{
"type": "endgame",
"id": "FFr_65111FF5",
"endings": [
{
"name": "Terrible Ending",
"requiredScore": 0,
"videoPath": "ending1.pdv",
"audioPath": "ending1.wav"
},
{
"name": "Nailed It!",
"requiredScore": 100,
"videoPath": "ending3.pdv",
"audioPath": "ending3.wav"
}
]
}
After building your story in Articy, you'll need to convert it to the proper JSON format for the game engine. This is done using the provided Python conversion scripts.
Place the scripts scene_convert.py
and story_convert.py
in the same folder as your exported data, then run:
# First convert individual scenes
python scene_convert.py
# Then convert the full story structure and inventory items
python story_convert.py
This creates a /story/
subfolder containing:
story.json
: Overall story structure and flowinventory_items.json
: All inventory items and their propertiesFFr_*.json
and DFr_*.json
)Copy all files from the /story/
folder to the assets/story
folder in your game project before building.
cp -r story/* /path/to/game/assets/story/
This covers the core functionality of TIPS. Using these commands and structures, you can make branching stories with inventory puzzles and dynamic storytelling.