Utils for working with screenplays and the Fountain and FDX scriptwriting formats.
Project description
screenplay-tools
screenplay-tools is a set of libraries for representing a format-agnostic screenplay format, with support for import and export to the Fountain script-writing language and Final Draft FDX files. These libraries are written in C++, Javascript, Python, and C#.
Fountain
The Fountain module provides methods for writing and reading Fountain files.
INT. DAVE'S APARTMENT - DAY
Dave is standing in the open window looking out at the pouring rain.
DAVE
(cheerfully)
Nice day for it!
CUT TO:
Fountain is a simple plaintext format for movie and TV scripts. It's also used as an export and import format by multiple script-writing apps, such as Fade In, Highland, Final Draft, Slugline and others.
The main Parser class turns the raw Fountain files into a Script object representing the screenplay, which you can do what you want with. The Writer class turns a Script into a Fountain file. This is all UTF-8 compatible!
FDX (Final Draft)
The FDX module provides methods for reading and writing Final Draft (.fdx) files.
The FDX.Parser class turns the FDX XML files into a Script object. The FDX.Writer class turns a Script into an FDX file.
Contents
The Basics
The Fountain.Parser supplied in the tools will take this:
INT. DAVE'S APARTMENT - DAY
Dave is standing in the open window looking out at the pouring rain.
DAVE
(cheerfully)
Nice day for it!
CUT TO:
INT. SPACE STATION - EARTHDAWN #1a#
Jennifer is upside down, looking out through a round porthole of a window, at the sun rising over Earth.
JENNIFER
(bitter)
Nice day for it.
COLIN (O.S.)
Oh no. Not again.
and break it into objects like this:
HEADING: { text: "INT. DAVE'S APARTMENT - DAY" }
ACTION: { text: "Dave is standing in the open window looking out at the pouring rain." }
CHARACTER: { name: "DAVE" }
PARENTHETICAL: { text: "cheerfully" }
DIALOGUE: { text: "Nice day for it!" }
TRANSITION: { text: "CUT TO:" }
HEADING: { text: "INT. SPACE STATION - EARTHDAWN", sceneNumber: "1a"}
ACTION: { text: "Jennifer is upside down, looking out through a round porthole of a window, at the sun rising over Earth." }
CHARACTER: { name: "JENNIFER" }
PARENTHETICAL: { text: "bitter" }
DIALOGUE: { text: "Nice day for it." }
CHARACTER: { name: "COLIN", extension: "O.S." }
DIALOGUE: { text: "Oh no. Not again." }
You can then do what you like with the objects. The same is achieveable from Final Draft (FDX) format, through FDX.Parser.
An alternative parser, Fountain.CallbackParser, will gather up material and call back using your supplied functions during parsing. In particular, it gathers dialogue lines together with characters, which is often more useful if you want to display dialogue on-screen for some reason.
onSceneHeading: { text: "INT. DAVE'S APARTMENT - DAY" }
onAction: { text: "Dave is standing in the open window looking out at the pouring rain." }
onDialogue: { character: "DAVE", parenthetical: "cheerfully", line: "Nice day for it!" }
onTransition: { text: "CUT TO:" }
onSceneHeading: { text: "INT. SPACE STATION - EARTHDAWN", sceneNumber: "1a"}
onAction: { text: "Jennifer is upside down, looking out through a round porthole of a window, at the sun rising over Earth." }
onDialogue: { character: "JENNIFER" , parenthetical: "bitter", line: "Nice day for it." }
onDialogue: { character: "COLIN", extension: "O.S." line: "Oh no. Not again." }
Source Code
The source can be found on Github, and is available under the MIT license.
Releases
Releases are available in the releases area in Github and are available for multiple platforms:
- Javascript - a JS file for use in ESM modules, and a minified JS file for use in a browser.
- Python - a Python package for import into other Python files.
- C# - a DotNET DLL for use in any C# project
- C++ - a set of source files for you to compile yourself
Usage
Overview
- The
ScreenplayTools.Fountain.Parserwill parse a Fountain file incremementally - using methods likeaddText(). - As the parsing continues, the
scriptmember of the parser will contain the parsed information in aScreenplayTools.Scriptobject. - A
Scriptconsists of:- A set of
titleEntrieswhich are information from the script's title page. - A set of
elementswhich are line-by-line parts of the script e.g. anActionline, aCharacterline or a line ofDialogue. - Some
NotesandBoneyardmaterial which you can probably ignore.
- A set of
ScreenplayTools.Fountain.Writercan write the script out again in a standard Fountain format.ScreenplayTools.Fountain.FormatHelpercan split Fountain bold/italic/underline markup into standard HTML markup.ScreenplayTools.Fountain.CallbackParseris a different version of the parser which calls you back as the script is parsed, and aggregates things like character and dialogue lines together. You can use it to call you with a line of dialogue with the right character and directions attached to it.
Here are some examples in different languages:
Parsing a file
// Javascript
import { FountainParser } from "screenplayTools.js";
const filePath = '../examples/Test.fountain';
const fileContent = readFileSync(filePath, 'utf-8');
let fp = new FountainParser();
// Expects a file full of UTF8 script
fp.addText(fileContent);
// Dump a debug version of the script
console.log(fp.script.dump());
Parsing from a set of lines
// C#
using System;
using System.Collections.Generic;
using ScreenplayTools;
using ScreenplayTools.Fountain;
class Program
{
static void Main(string[] args)
{
Parser fp = new Parser();
// Example lines from the script
List<string> lines = new List<string>
{
"EXT. MY BASEMENT",
"DAVE",
"",
"(Shouting)",
"Hey, anyone home?"
};
// Add a whole set of lines to the parser
fp.AddLines(lines);
// Dump a debug version of the script
Console.WriteLine(fp.Script.Dump());
}
}
Parsing a line at a time
# Python
from screenplay_tools.fountain.parser import Parser
# Create an instance of Parser
fp = Parser()
# Add individual lines to the parser
fp.add_line("EXT. MY BASEMENT")
fp.add_line("")
fp.add_line("DAVE")
fp.add_line("(Shouting)")
fp.add_line("Hey, anyone home?")
# Need to use finalize() after individual lines
# as some lines care about what gets added next.
fp.finalize()
# Dump the parsed script
print(fp.script.dump())
Tags
I've extended the definition of Fountain files to introduce the concept of tagging a line: that is, being able to attach a number of data items to a line. This concept is stolen from Inkle's Ink markup language, and is useful for embedding information in a script.
An example is like so
INT. DAVE'S APARTMENT - DAY #slow_load
Dave is standing in the open window looking out at the pouring rain.
DAVE
(cheerfully)
Nice day for it! #color:blue #useAnim
In here, some of the lines have tags attached. A tag is always at the end of a normal line, starts with the #character, and then can contain any other character until it hits whitespace. You can have more than one tag.
(If the first character on a line is a # character it will be treated as a Section as per the Fountain spec, not as a tag.)
To parse this, you can pass useTags=true to your Parser class. Then each Element's tags property will be populated with the tags for that particular script element. The parse works like so:
HEADING: { text: "INT. DAVE'S APARTMENT - DAY", tags:["slow_load"]}
ACTION: { text: "Dave is standing in the open window looking out at the pouring rain." }
CHARACTER: { name: "DAVE" }
PARENTHETICAL: { text: "cheerfully" }
DIALOGUE: { text: "Nice day for it!", tags:["color:blue", "useAnim"]}
Extended Sections
BirdCatcherGames has extended the code to allow up to six levels of Fountain's section format i.e.
### This is a section
###### And this is a much lower section.
Rather than Fountain's usual 3.
Javascript as an ES6 module
import { FountainParser } from './screenplayTools.js';
const parser = new FountainParser();
parser.addText("INT. SCENE HEADER\n\nDAVE\nHello, fellow kids!");
console.log(parser.script.dump());
Javascript in a browser
Either you can use the same module / ESM format (screenplayTools.js):
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Screenplay Tools</title>
</head>
<body>
<script type="module">
import { FountainParser } from './screenplayTools.js';
const parser = new FountainParser();
parser.addText("INT. SCENE HEADER\n\nDAVE\nHello, fellow kids!");
console.log(parser.script.dump());
</script>
</body>
</html>
Or you can use a minified IIFE version (screenplayTools.min.js):
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Screenplay Tools</title>
<script src="screenplayTools.min.js"></script>
</head>
<body>
<script>
// Access the global ScreenplayTools object
const parser = new ScreenplayTools.Fountain.FountainParser();
parser.addText("INT. SCENE HEADER\n\nDAVE\nHello, fellow kids!");
// Use the imported function
console.log(parser.script.dump());
</script>
</body>
</html>
Python
from screenplay_tools.fountain.parser import Parser
from screenplay_tools.fountain.writer import Writer
script_text = """
INT. ROOM - DAY
DAVE
(loudly)
Hello, fellow kids!
"""
# Initialize parser and writer
parser = Parser()
writer = Writer()
# Parse the script
parser.add_text(script_text)
# Write the script back to text
formatted_script = writer.write(parser.script)
print(formatted_script)
C
Install the DLL in your project, and use it like so:
using System;
using ScreenplayTools;
using ScreenplayTools.Fountain;
class Program
{
static void Main(string[] args)
{
Parser parser = new Parser();
parser.AddText("INT. SCENE HEADER\n\nDAVE\nHello, fellow kids!");
Console.WriteLine(parser.Script.Dump());
}
}
C++
I haven't supplied any built libs (because building multiplatform libs is outside my scope right now). Instead I have supplied source code in the zip - you should be able to build and use it with your project.
#include "screenplay_tools/fountain/parser.h"
#include "screenplay_tools/screenplay.h"
#include <iostream>
#include <string>
// Use the namespace
using namespace ScreenplayTools;
using namespace ScreenplayTools::Fountain;
int main() {
// Create an instance of Parser
Parser parser;
// Example Fountain script text
std::string scriptText = R"(Title: Example Script
Author: Test Author
INT. ROOM - DAY
A description of the scene.
CHARACTER
Dialogue line.)";
// Add text to the parser
parser.addText(scriptText);
// Assuming the parser has a way to get the parsed script (e.g., via a 'getScript()' method)
const auto& script = parser.getScript();
// Dump the parsed script to the console
std::cout << script->dump() << std::endl;
return 0;
}
API
Parser
JS: FountainParser
Python: screenplay_tools.fountain.parser.Parser
C#: ScreenplayTools.Fountain.Parser
C++: ScreenplayTools::Fountain::Parser
The normal incremental parser. It stores the parsed script in the script property. Consider CallbackParser if you want something which bundles up the dialogue in a more useful way.
mergeActions:bool / mergeDialogue:bool
Default: True
These variables control whether multiple dialogue or action lines get merged together into one script element, or if you get called with a list of separate elements instead.
Merging can be unhelpful if using an incremental parse.
useTags:bool
Default: False
If True, extracts and parses tags from the Fountain file. See Tags above.
addText(text:string)
Split UTF-8 text into lines and parse them.
addLines(lines:list)
Parse an array of UTF-8 text lines.
addLine(line:string)
Parse a single UTF-8 text line.
script / getScript()
Parsed script. Grows as more lines are parsed!
Script
JS: Script
Python: screenplay_tools.screenplay.Script
C#: ScreenplayTools.Script
C++: ScreenplayTools::Script
The parsed script.
titleEntries / getTitleEntries()
Return a list of the information from the title page of the script, as TitleEntry objects.
elements / getElements()
Returns a list of the parsed elements in the script as Element objects. Determine the element type using element.getType()
notes / getNotes()
Returns a list of embedded notes as Note objects. This isn't much use in parsing, and is merely to preserve info read from the original Fountain file.
boneyards / getBoneyards()
Returns a list of commented-out chunks of text as Boneyard objects. This isn't much use in parsing, and is merely to preserve info read from the original Fountain file.
Element
JS: ScreenplayElement
Python: screenplay_tools.screenplay.Element
C#: ScreenplayTools.Element
C++: ScreenplayTools::Element
Superclass of all the elements in the script. Use type/getType() to figure out what type it can be downcast to.
type / getType()
Returns an ElementType enum which will give the type of script element.
text / getText()
Gives the main text element of the asset but doesn't include other parsed information.
Elements
Take a look at the Fountain syntax to understand what these all are.
TitleEntry
JS: ScreenplayTitleEntry
Python: screenplay_tools.screenplay.TitleEntry
C#: ScreenplayTools.TitleEntry
C++: ScreenplayTools::TitleEntry
https://fountain.io/syntax/#title-page
Entry on the title page. Consists of a key and text. This element will only be found inside the getTitleEntries() section of the script.
Author: Dave Smith
Action
JS: ScreenplayAction
Python: screenplay_tools.screenplay.Action
C#: ScreenplayTools.Action
C++: ScreenplayTools::Action
https://fountain.io/syntax/#action
Single line of action in text. If centered is true, the text is intended to be center-justified.
He runs upstairs and eviscerates the wombat.
> Centered <
Scene Heading
JS: ScreenplaySceneHeading
Python: screenplay_tools.screenplay.SceneHeading
C#: ScreenplayTools.SceneHeading
C++: ScreenplayTools::SceneHeading
https://fountain.io/syntax/#scene-headings
Heading for a scene as text. Optional sceneNumber.
INT. BATHROOM - DAY - text
INT. BATHROOM - DAY #1-a# - text #sceneNumber#
Character
JS: ScreenplayCharacter
Python: screenplay_tools.screenplay.Character
C#: ScreenplayTools.Character
C++: ScreenplayTools::Character
https://fountain.io/syntax/#charater
Character header. Consists of name, optional extension, and isDualDialogue can be true.
DAVE - name
DAVE (V.O.) - name, (extension)
DAVE (V.O.) ^ - name, (extension), isDualDialogue=true
Dialogue
JS: ScreenplayDialogue
Python: screenplay_tools.screenplay.Dialogue
C#: ScreenplayTools.Dialogue
C++: ScreenplayTools::Dialogue
https://fountain.io/syntax/#dialogue
Line of dialogue as text.
Hello everyone!
Parenthetical
JS: ScreenplayParenthetical
Python: screenplay_tools.screenplay.Parenthetical
C#: ScreenplayTools.Parenthetical
C++: ScreenplayTools::Parenthetical
https://fountain.io/syntax/#parenthetical
Direction in parenthesis before a line of dialogue. Doesn't include the parenthesis in the text.
(underhandedly)
Lyrics
JS: ScreenplayLyric
Python: screenplay_tools.screenplay.Lyric
C#: ScreenplayTools.Lyric
C++: ScreenplayTools::Lyric
https://fountain.io/syntax/#lyrics
Line of lyrics, as text.
~ These are some song lyrics.
Transition
JS: ScreenplayTransition
Python: screenplay_tools.screenplay.Transition
C#: ScreenplayTools.Transition
C++: ScreenplayTools::Transition
https://fountain.io/syntax/#transition
Transition line, as text.
CUT TO:
Page Break
JS: ScreenplayPageBreak
Python: screenplay_tools.screenplay.PageBreak
C#: ScreenplayTools.PageBreak
C++: ScreenplayTools::PageBreak
https://fountain.io/syntax/#page-breaks
Uh, a page break.
===
Section
JS: ScreenplaySection
Python: screenplay_tools.screenplay.Section
C#: ScreenplayTools.Section
C++: ScreenplayTools::Section
https://fountain.io/syntax/#sections-synopses
A section header as text and a number level.
# This is a section
## This is a level 2 section
Synopsis
JS: ScreenplaySynopsis
Python: screenplay_tools.screenplay.Synopsis
C#: ScreenplayTools.Synopsis
C++: ScreenplayTools::Synopsis
https://fountain.io/syntax/#sections-synopses
A synopsis in text.
= Synopsis of this section
Note
JS: ScreenplayNote
Python: screenplay_tools.screenplay.Note
C#: ScreenplayTools.Note
C++: ScreenplayTools::Note
https://fountain.io/syntax/#notes
The content of a parsed note, as text.
This element will only be found inside the getNotes() section of the script.
Here is a normal line [[This is a note though]].
Boneyard
JS: ScreenplayBoneyard
Python: screenplay_tools.screenplay.Boneyard
C#: ScreenplayTools.Boneyard
C++: ScreenplayTools::Boneyard
https://fountain.io/syntax/#boneyard
The content of a boneyard, also known as a commented-out section, as text.
This element will only be found inside the getBoneyards() section of the script.
Here is a normal line /*This is a boneyard*/.
CallbackParser
JS: FountainCallbackParser
Python: screenplay_tools.fountain.callback_parser.CallbackParser
C#: ScreenplayTools.Fountain.CallbackParser
C++: ScreenplayTools::Fountain::CallbackParser
A version of the parser which lets you set up callbacks which will be called as lines are parsed.
onTitlePage
entries- a list of key/values for the title page.
onDialogue
character- the character name in the scriptextension- optional bracketed exension e.g. DAVE (V.O.)parenthetical- optional parenthetical before the dialogue line e.g. (loudly) or (angrily)line- line of dialogue,isDualDialogue- True if the caret ^ is present indicating dual dialogue in the script
onAction
text
onSceneHeading
textsceneNumber- optional
onLyrics
text
onTransition
text
onSection
textlevel- number
onSynopsis
text
onPageBreak
ignoreBlanks: bool
Set to false if you want empty dialogue and actions.
Writer
JS: FountainWriter
Python: screenplay_tools.fountain.writer.Writer
C#: ScreenplayTools.Fountain.Writer
C++: ScreenplayTools::Fountain::Writer
A simpler writer to write a script as UTF-8 text.
prettyPrint
Set to false if you don't want indents in the output.
write(script)
Pass in a Script, get back a UTF-8 string.
FormatHelper
JS: FountainFormatHelper
Python: screenplay_tools.fountain.formatHelper.FormatHelper
C#: ScreenplayTools.Fountain.FormatHelper
C++: ScreenplayTools::Fountain::FormatHelper
fountainToHtml
Convert Fountain markup (italic, bold, bolditalic underline) to HTML.
FDX Parser
C#: ScreenplayTools.FDX.Parser
The FDX parser reads a Final Draft XML file.
Parse(xml:string)
Parses the FDX XML string and returns a Script object.
FDX Writer
C#: ScreenplayTools.FDX.Writer
Write(script:Script)
Takes a Script object and returns a string containing the FDX XML.
Contributors
- wildwinter - original author
- BirdCatcherGames - extending Fountain spec to allow 6-deep sections
License
MIT License
Copyright (c) 2024 Ian Thomas
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file screenplay_tools-0.0.10.tar.gz.
File metadata
- Download URL: screenplay_tools-0.0.10.tar.gz
- Upload date:
- Size: 25.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1a9d6126af3eae5c260bdadaa1e69001c0cc43882fad15c698d7a876a8a117b2
|
|
| MD5 |
b76adfa5a2c41c1e03a5a0f502a5be3a
|
|
| BLAKE2b-256 |
7debdb57082721d609d851879a851b47f83b031b309605e52e2e84711136a336
|
File details
Details for the file screenplay_tools-0.0.10-py3-none-any.whl.
File metadata
- Download URL: screenplay_tools-0.0.10-py3-none-any.whl
- Upload date:
- Size: 20.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1b05d6ed4033bf6598ed6fe151c834f9e53ba9400b821005d430995706f83f46
|
|
| MD5 |
9d961d759d136be53403018489e2e2aa
|
|
| BLAKE2b-256 |
1a02b7f028f0d4fd26d6977cd38e3d7fa233b7eb5e776e68168f6dd2e3fcfad2
|