mdif
v0.0.5
Published
dialog/interactive fiction engine that lets you use markdown to describe conversations and games
Downloads
5
Maintainers
Readme
Markdown Interactive Fiction
This is a dialog/interactive fiction engine (for nodejs) that lets you describe conversations and games. It uses a subset of markdown to define your dialogs, and mustache to express light logic/formatting. It can be used in any sort of game/app that has dialog with (or without) logic.
usage
In your own project:
npm i mdif
import { promises as fs } from 'node:fs'
import { runDialog, getASTInfo } from 'mdif'
const md = await fs.readFile('example.md')
// read some info
console.log(getASTInfo(md).info)
// get the first line of dialog from "start"
const variables = {
player: { name: 'Peter' },
konsumer: { scared: false }
}
let screen = runDialog(md, 'start', variables)
// increment to the next page
screen = runDialog(md, 'start', variables, 1)
console.log(screen)
try it out here
git clone https://github.com/konsumer/mdif.git
cd mdif
npm i # install deps & tools
npm start # run examples/cli demo
npm run raylib # run examples/raylib demo
npm test # run unit-tests in test.js, that also have some usage info
story reference
Essentially, you can use mustache for all logic, and markdown for the dialogs, which are made up of conversation
, options
, and code
.
markdown
- An h2 (
##
) defines a new dialog. - a blockquote (
>
) defines something someone is saying. Optionally, wrap "who" in italic:*Feindish Guy* Oh hey, my fellow feind
- a list of links is options available at the end. It looks like this:
- [hmm?](#start)
- [yes](#thats_my_name)
- [no](#lie_about_name)
- [wait, how do you know my name?](#lie_about_name)
The URLs should be #id
(to link to other dialogs) or file#id
(to load a different conversation-collection.) This library doesn't manage that at all, so you will have to parse the url, in your code:
let md = await fs.readFile('example.md')
// user has progressed to page 1, which in this case has a menu (it's an id not in conversation lines)
let dialog = runDialog(md, 'start', variables, 1)
// is this a menu or a line of text
if (Array.isArray(dialog)) {
// ... show options and get user-selection here
const [file, hash] = dialog[SELECTION].url.split('#')
// it's another file, so set that to be the main md
if (file) {
md = await fs.readFile(file)
}
// load next dialog
dialog = runDialog(md, hash, variables)
}
// do other stuff in loop, like show current dialog, allow user to progress, etc
code
Code is pulled out of codeblocks, like this (surround with 3 backticks, and set language):
player.happy = true
It is run when the individual dialog first loads. Currently, the only supported langage is js
, but we may support more later.
frontmatter
You can use frontmatter to define any meta-information for the whole file. It goes at the top, and can be used externally with getASTInfo(md).info
. We support TOML & YAML.
YAML:
---
name: My Cool World
friendly: true
---
TOML:
+++
preferred_language = 'TOML'
name = 'My Cool World'
+++
mustache
Go read the docs for more about how to use it, but essentially mustache is parsed before the dialog section is run, so you can insert logic based on your variables, like substitution, loops, condtionals, etc.