Available as an interactive Colab notebook here

There are lots of examples of somewhat complex animation in the examples/animations directory, but here are some simpler (and shorter) ones, that demonstrate the fundamentals of how animations are built in coldtype.

A circle moving

from coldtype import *

tl = Timeline(30)

@animation((250, 250), timeline=tl, bg=1)
def circle(f):
    return PS([
                .offset(f.e("qeio", 0, rng=(-f.a.r.w, f.a.r.w)), 0))
            .f(hsl(f.e("qeio", 0), s=0.75)))])
circle_contactsheet = circle.contactsheet(6)

Save that code in a file called, then run it ala coldtype — when the render preview runs and a window pops up, try hitting the left and right arrow keys on your keyboard to go backward and forward in time.

If you hold down the arrow keys, you can preview the animation; depending on how complex your animation is, “previewing” might be exactly the same as viewing the final animation, if the frames render as fast as real-time. (Even if they don’t, holding down the right arrow key should still give you a good sense of the mechanics of the animation in time.)

To see the animation come to life, hit the space bar. If your animation can render faster than real-time (i.e. faster than the frame-rate defined in the Timeline associated with the @animation decorator), when you hit the spacebar, the animation will play back at the defined frame rate (which defaults to 30fps). If not, not a big deal, it’ll just be a little slower than the intended frame rate.

One cool thing to note — once you hit the space bar, if you make changes to the source file and hit save, the animation will update and keeping playing back.

If you want to see the full animation played back at its true frame rate regardless of how intensive the rendering is, hit the key “a” on your keyboard, and you should see some text appear in your terminal, noting that the frames of the animation are being saved to disk. When it finishes (it’ll say something like rendered), the animation will automatically start playing back. You can hit shift+space in the viewer to stop that playback.

Here’re all the frames of that animation (a circle moving across the frame, and changing colors):


A letter flying

from coldtype import *

@animation((250, 250), bg=0, timeline=24)
def flyinga(f):
    return PS([
            .f(hsl(f.e("qeio", 0)))),
        (StSt("A", Font.MutatorSans(),
            50, wght=0.2)
            .scale(f.e("eei", 0, rng=(1, 51)))
            .rotate(f.e("qeio", 0, rng=(0, 360)))
flyinga_contact = flyinga.contactsheet(4, slice(0, None, 1))

And here’s a little bit of code to generate gifs, using ffmpeg, which will need to be installed on your computer independently of coldtype (via something like brew install ffmpeg) (or you can, as described below, import the pngs as an image sequence into something like Premiere).

To get this code to run, you want to trigger the Release KeyboardShortcut, by hitting R in the viewing app.

def release(passes):
    FFMPEGExport(circle, passes).gif().write()
    FFMPEGExport(flyinga, passes).gif().write()
../_images/circle.gif ../_images/flyinga.gif

Jumping to an NLE

While it is possible to complete an animation using nothing but code (and in the near future this process will get easier), I usually employ a program like Premiere, After Effects, or DaVinci Resolve to both view and finish animations that I start in code, either to apply effects in After Effects, or to match the animation with music in Premiere or Resolve. Because animations done in Coldtype are rendered to PNGs, you can import those easily in any video editing program. (In Premiere, just make sure to select the 0000 image, then select Options > Image Sequence when importing.)

To generate a full set of frames for a coldtype animation, hit the a key in the viewer app — once you do, you should see the command line prompt printing out a bunch of information about frames being rendered. (Also, once you do that, you can hit shift+space to preview the animation in real time at the correct frame rate, using the cached frames.)