Rhubarb Lip Sync is a command-line tool that automatically creates 2D mouth animation from voice recordings. You can use it for characters in computer games, in animated cartoons, or in any other project that requires animating mouths based on existing recordings.
= Rhubarb Lip Sync
:toc:
:icons: font
:A: Ⓐ
:B: Ⓑ
:C: Ⓒ
😄: Ⓓ
:E: Ⓔ
:F: Ⓕ
:G: Ⓖ
:H: Ⓗ
:X: Ⓧ
image:https://img.shields.io/twitter/follow/RhubarbLipSync.svg?style=social&label=Follow[“Twitter”, link=“https://twitter.com/RhubarbLipSync”]
image:https://github.com/DanielSWolf/rhubarb-lip-sync/actions/workflows/ci.yml/badge.svg[“Build status”, link=“https://github.com/DanielSWolf/rhubarb-lip-sync/actions/workflows/ci.yml”]
image:img/logo.png[align=“center”]
Rhubarb Lip Sync allows you to quickly create 2D mouth animation from voice recordings. It analyzes your audio files, recognizes what is being said, then automatically generates lip sync information. You can use it for animating speech in computer games, animated cartoons, or any similar project.
Rhubarb Lip Sync integrates with the following applications:
In addition, you can use Rhubarb Lip Sync’s command line interface (CLI) to generate files in various <<outputFormats,output formats>> (<<tsv,TSV>>/<<xml,XML>>/<<json,JSON>>).
== Demo video
Click the image for a demo video.
https://www.youtube.com/watch?v=zzdPSFJRlEo[image:http://img.youtube.com/vi/zzdPSFJRlEo/0.jpg[]]
== Integrations
[[afterEffects]]
=== Adobe After Effects
You can use Rhubarb Lip Sync to animate dialog right from Adobe After Effects. For more information, <<extras/AdobeAfterEffects/README.adoc#,follow this link>> or see the directory extras/AdobeAfterEffects
.
image:img/after-effects.png[]
[[moho]]
=== Moho and OpenToonz
Rhubarb Lip Sync can create .dat switch data files, which are understood by Moho and OpenToonz. You can set the frame rate using the --datFrameRate
option; to control the shape names, use the --datUsePrestonBlair
flag. For more details, see <
image:img/moho.png[]
[[spine]]
=== Spine by Esoteric Software
Rhubarb Lip Sync for Spine is a graphical tool that allows you to import a Spine project, perform automatic lip sync, then re-import the result into Spine. For more information, <<extras/EsotericSoftwareSpine/README.adoc#,follow this link>> or see the directory extras/EsotericSoftwareSpine
of the download.
image:img/spine.png[]
[[vegas]]
=== Vegas Pro by Magix
Rhubarb Lip Sync also comes with two plugin scripts for Vegas Pro (previously Sony Vegas). For more information, <<extras/MagixVegas/README.adoc#,follow this link>> or see the directory extras/MagixVegas
of the download.
image:img/vegas.png[]
[[mouth-shapes]]
== Mouth shapes
Rhubarb Lip Sync can use between six and nine different mouth positions. The first six mouth shapes ({A}-{F}) are the basic mouth shapes and the absolute minimum you have to draw for your character. These six mouth shapes were invented at the Hanna-Barbera studios for shows such as Scooby-Doo and The Flintstones. Since then, they have evolved into a de-facto standard for 2D animation, and have been widely used by studios like Disney and Warner Bros.
In addition to the six basic mouth shapes, there are three extended mouth shapes: {G}, {H}, and {X}. These are optional. You may choose to draw all three of them, pick just one or two, or leave them out entirely.
[cols=“1h,2,6”]
|===
| {A} | image:img/lisa-A.png[]
| Closed mouth for the “P
”, “B
”, and “M
” sounds. This is almost identical to the {X} shape, but there is ever-so-slight pressure between the lips.
| {B} | image:img/lisa-B.png[]
| Slightly open mouth with clenched teeth. This mouth shape is used for most consonants (“K
”, “S
”, “T
”, etc.). It’s also used for some vowels such as the “EE
” sound in bee.
| {C} | image:img/lisa-C.png[]
| Open mouth. This mouth shape is used for vowels like “EH
” as in men and “AE
” as in bat. It’s also used for some consonants, depending on context.
This shape is also used as an in-between when animating from {A} or {B} to {D}. So make sure the animations {A}{C}{D} and {B}{C}{D} look smooth!
| {D} | image:img/lisa-D.png[]
| Wide open mouth. This mouth shapes is used for vowels like “AA
” as in father.
| {E} | image:img/lisa-E.png[]
| Slightly rounded mouth. This mouth shape is used for vowels like “AO
” as in off and “ER
” as in bird.
This shape is also used as an in-between when animating from {C} or {D} to {F}. Make sure the mouth isn’t wider open than for {C}. Both {C}{E}{F} and {D}{E}{F} should result in smooth animation.
| {F} | image:img/lisa-F.png[]
| Puckered lips. This mouth shape is used for “UW
” as in you, “OW
” as in show, and “W
” as in way.
| {G} | image:img/lisa-G.png[]
| Upper teeth touching the lower lip for “F
” as in for and “V
” as in very.
This extended mouth shape is optional. If your art style is detailed enough, it greatly improves the overall look of the animation. If you decide not to use it, you can specify so using the <<extendedShapes,extendedShapes
>> option.
| {H} | image:img/lisa-H.png[]
| This shape is used for long “L
” sounds, with the tongue raised behind the upper teeth. The mouth should be at least far open as in {C}, but not quite as far as in {D}.
This extended mouth shape is optional. Depending on your art style and the angle of the head, the tongue may not be visible at all. In this case, there is no point in drawing this extra shape. If you decide not to use it, you can specify so using the <<extendedShapes,extendedShapes
>> option.
| {X} | image:img/lisa-X.png[]
| Idle position. This mouth shape is used for pauses in speech. This should be the same mouth drawing you use when your character is walking around without talking. It is almost identical to {A}, but with slightly less pressure between the lips: For {X}, the lips should be closed but relaxed.
This extended mouth shape is optional. Whether there should be any visible difference between the rest position {X} and the closed talking mouth {A} depends on your art style and personal taste. If you decide not to use it, you can specify so using the <<extendedShapes,extendedShapes
>> option.
|===
== How to run Rhubarb Lip Sync
=== General usage ===
Rhubarb Lip Sync is a command-line tool that is currently available for Windows, macOS, and Linux.
rhubarb
, passing it an audio file as argument and telling it where to create the output file. In its simplest form, this might look like this: rhubarb -o output.txt my-recording.wav
. There are additional <<options,command-line options>> you can specify in order to get better results.stderr
and exit with a non-zero exit code.[[options]]
=== Command-line options ===
==== Basic command-line options ====
The following command-line options are the most common:
[cols=“2,5a”]
|===
| Option | Description
|
| The audio file to be analyzed. This must be the last command-line argument. Supported file formats are WAVE (.wav) and Ogg Vorbis (.ogg).
| -r
--recognizer
| Specifies how Rhubarb Lip Sync recognizes speech within the recording. Options: pocketSphinx
(use for English recordings), phonetic
(use for non-English recordings). For details, see <
Default value: pocketSphinx
| -f
--exportFormat
| The export format. Options: tsv
(tab-separated values, see <<tsv,details>>), xml
(see <<xml,details>>), json
(see <<json,details>>), dat
(see <
Default value: tsv
| -d
--dialogFile
| With this option, you can provide Rhubarb Lip Sync with the dialog text to get more reliable results. Specify the path to a plain-text file (in ASCII or UTF-8 format) containing the dialog contained in the audio file. Rhubarb Lip Sync will still perform word recognition internally, but it will prefer words and phrases that occur in the dialog file. This leads to better recognition results and thus more reliable animation.
For instance, let’s say you’re recording dialog for a computer game. The script says: “That's all gobbledygook to me.
” But actually, the voice artist ends up saying “That's _just_ gobbledygook to me,
” deviating from the dialog. If you specify a dialog file with the original line (“That's all gobbledygook to me
”), this will still allow Rhubarb Lip Sync to produce better results, because it will watch out for the uncommon word “gobbledygook
”. Rhubarb Lip Sync will ignore the dialog file where it audibly differs from the recording, and benefit from it where it matches.
It is always a good idea to specify the dialog text. This will usually lead to more reliable mouth animation, even if the text is not completely accurate.
[[extendedShapes]]
| --extendedShapes
| As described in <GX
; to use only the six basic mouth shapes, specify an empty string: ""
.
Default value: GHX
| -o
, --output
| The name of the output file to create. If the file already exists, it will be overwritten. If you don’t specify an output file, the result will be written to stdout
.
| --version
| Displays version information and exits.
| -h
, --help
| Displays usage information and exits.
| --datFrameRate
number
| Only valid when using the dat
export format. Controls the frame rate for the output file.
Default value: 24
| --datUsePrestonBlair
| Only valid when using the dat
export format. Uses Preston Blair mouth shapes names instead of the default alphabetical ones. This applies the following mapping:
!===
! Alphabetic name ! Preston Blair name
! A ! MBP
! B ! etc
! C ! E
! D ! AI
! E ! O
! F ! U
! G ! FV
! H ! L
! X ! rest
!===
Caution: This mapping is only applied when exporting, after the recording has been animated. To control which mouth shapes to use, use the <<extendedShapes,extendedShapes
>> option with the alphabetic names.
Tip: For optimal results, make sure your mouth drawings follow the guidelines in the <
|===
==== Advanced command-line options ====
The following command-line options can be helpful in special situations, especially when automating Rhubarb Lip Sync.
[cols=“2,5”]
|===
| Option | Description
[[quiet]]
| -q
, --quiet
| By default, Rhubarb Lip Sync writes a number of progress messages to stderr
. If you’re using it as part of a batch process, this may clutter your console. If you specify the --quiet
flag, there won’t be any output to stderr
unless an error occurred.
You can combine this option with the <<consoleLevel,consoleLevel
>> option to change the minimum event level that is printed to stderr
.
| --machineReadable
a| This option is useful if you want to integrate Rhubarb Lip Sync with another (possibly graphical) application. All status messages to stderr
will be in structured JSON format, allowing your program to parse them and display a graphical progress bar or something similar. For details, see <<machineReadable,Machine-readable status messages>>.
[[consoleLevel]]
| --consoleLevel
| Sets the log level for reporting to the console (stderr
). Options: trace
, debug
, info
, warning
, error
, fatal
.
If <<quiet,--quiet
>> is also specified, only events with the specified level or higher will be printed. Otherwise, a small number of essential events (startup, progress, etc.) will be printed even if their levels are below the specified value.
Default value: error
| --logFile
| Creates a log file with diagnostic information at the specified path.
|--logLevel
| Sets the log level for the log file. Only events with the specified level or higher will be logged. Options: trace
, debug
, info
, warning
, error
, fatal
.
Default value: debug
| --threads
| Rhubarb Lip Sync uses multithreading to speed up processing. By default, it creates as many worker threads as there are cores on your CPU, which results in optimal processing speed. You may choose to specify a lower number if you feel that Rhubarb Lip Sync is slowing down other applications. Specifying a higher number is not recommended, as it won’t result in any additional speed-up.
Note that for short audio files, Rhubarb Lip Sync may choose to use fewer threads than specified.
Default value: as many threads as your CPU has cores
|===
[[recognizers]]
== Recognizers
The first step in processing an audio file is determining what is being said. More specifically, Rhubarb Lip Sync uses speech recognition to figure out what sound is being said at what point in time. You can choose between two recognizers:
=== PocketSphinx
PocketSphinx is an open-source speech recognition library that generally gives good results. This is the default recognizer. The downside is that PocketSphinx only recognizes English dialog. So if your recordings are in a language other than English, this is not a good choice.
=== Phonetic
Rhubarb Lip Sync also comes with a phonetic recognizer. Phonetic means that this recognizer won’t try to understand entire (English) words and phrases. Instead, it will recognize individual sounds and syllables. The results are usually less precise than those from the PocketSphinx recognizer. The advantage is that this recognizer is language-independent. Use it if your recordings are not in English.
[[outputFormats]]
== Output formats
The output of Rhubarb Lip Sync is a file that tells you which mouth shape to display at what time within the recording. You can choose between three file formats – TSV, XML, and JSON. The following paragraphs show you what each of these formats looks like.
[[tsv]]
=== Tab-separated values (tsv
)
TSV is the simplest and most compact export format supported by Rhubarb Lip Sync. Each line starts with a timestamp (in seconds), followed by a tab, followed by the name of the mouth shape. The following is the output for a recording of a person saying ‘Hi.’
Here’s how to read it:
HH
” sound, anticipating the “AY
” sound that will follow.AY
” diphtong (0.31s into the recording) requires clenched teeth (shape {B}). Before that, shape {C} is inserted as an in-between at 0.27s. This allows for a smoother animation from {D} to {B}.extendedShapes
>> settings).[[xml]]
=== XML format (xml
)
XML format is rather verbose. The following is the output for a person saying ‘Hi,’ the same recording as above.
The file starts with a metadata
block containing the full path of the original recording and its duration (truncated to a multiple of 0.01s). After that, each mouthCue
element indicates the start and end of a certain mouth shape, as explained for <<tsv,TSV format>>. Note that the end of each mouth cue is identical with the start of the following one. This is a bit redundant, but it means that we don’t need a special final element like in TSV format.
[[json]]
=== JSON format (json
)
JSON format is very similar to <<xml,XML format>>. The choice mainly depends on the programming language you use, which may have built-in support for one format but not the other. The following is the output for a person saying ‘Hi,’ the same recording as above.
There is nothing surprising here; everything said about XML format applies to JSON, too.
[[machineReadable]]
== Machine-readable status messages
Use the --machineReadable
command-line option to enable machine-readable status messages. In this mode, each line printed to stderr
will be an object in JSON format. Every object contains the following:
type
: The type of the event. Currently, one of "start"
(application start), "progress"
(numeric progress), "success"
(successful termination), "failure"
(unsuccessful termination), and "log"
(a log message without structured information)."progress"
event contains the property value
with a numeric value between 0.0 and 1.0.log
: A log message describing the event, plus severity information. If you aren’t interested in the structured data, you can display this as a fallback. For instance, a "progress"
event with the structured information "value": 0.69
may contain the following redundant log message: "Progress: 69%"
.You can combine this option with the <<consoleLevel,consoleLevel
>> option. Note, however, that this only affects unstructured events of type "log"
(not to be confused with the log
property each event contains).
The following is an example output to stderr
from a successful run:
The following is an example output to stderr
from a failed run:
Note that the output format <<Versioning,adheres to SemVer>>. That means that the JSON output created after a minor upgrade will still be compatible. Note, however, that the following kinds of changes may occur at any time, because I consider them non-breaking:
log
property."log"
or changes in the wording of log messages[[versioning]]
== Versioning (SemVer)
Rhubarb Lip Sync uses Semantic Versioning (SemVer) for its command-line interface. For general information on Semantic Versioning, have a look at the http://semver.org/[official SemVer website].
As a rule of thumb, everything you can use through the command-line interface adheres to SemVer. Everything else (i.e., the source code, integrations with third-party software, etc.) does not.
[[building-from-source]]
== Building from source
To use Rhubarb Lip Sync on Windows, macOS, or Linux, you can just download the binary release for your operating system. If you want to modify the code or use Rhubarb on a less-common operating system, this section describes how to build it yourself.
You’ll need the following software installed:
Then, follow these steps:
. Create an empty directory /build
within the Rhubarb repository
. Move to the new /build
directory
. Configure CMake by running cmake ..
+
Optionally, pass flags for setting the generator, compiler etc… For working examples, see .github\workflows\ci.yml
.
. Build Rhubarb Lip Sync by running cmake --build . --config Release
== I’d love to hear from you!
Have you created something great using Rhubarb Lip Sync? – https://twitter.com/RhubarbLipSync[Let me know on Twitter] or send me an email at +[email protected]+!
Do you need help? Have you spotted a bug? Do you have a suggestion? – https://github.com/DanielSWolf/rhubarb-lip-sync/issues[Create an issue!]