Make simple animations

Luxor.jl can help you build simple animations, by assembling a series of PNG images into an animated movie.

Note

To make richer and more complex animations, use Javis.jl, which is designed specifically for the purpose. Makie.jl is also a good choice.

1: A Julia spinner

The first thing to do is to create a Movie object. This acts as a useful way to pass information from function to function.

using Luxor
mymovie = Movie(400, 400, "mymovie")

The resulting animation will be 400 × 400 pixels.

To make the graphics, define a function called frame() (it doesn't have to be called that, but it's a good name) which accepts two arguments, a Scene object, and a framenumber (which will be an integer).

A movie consists of one or more scenes. A scene determines how many Luxor drawings should be made into a sequence and what function should be used to make them. The framenumber lets you keep track of where you are in a scene.

Here's a simple frame function which creates a drawing.

function frame(scene::Scene, framenumber::Int64)
    background("white")
    norm_framenumber = rescale(framenumber,
        scene.framerange.start,
        scene.framerange.stop,
        0, 1)
    rotate(norm_framenumber * 2π)
    juliacircles(100)
end

This function is responsible for drawing all the graphics for a single frame. The incoming frame number is converted (normalized) to lie between 0 and 1 - ie. between the first frame and the last frame of the scene. It's multiplied by 2π and used as input to rotate. So, as the framenumber goes from 1 to the last frame in the scene, each drawing will be rotated by an increasing angle from 0 to 2π. For example, for a scene with 60 frames, framenumber 30 will set a rotation value of about 2π * 0.5.

The Scene object has details about the number of frames for this scene, including the number of times the frame function is called.

To actually build the animation, the animate function takes a movie and an array of one or more scenes and creates all the drawings required. It can also build a GIF or movie file.

animate(mymovie,
        [
            Scene(mymovie, frame, 1:60)
        ],
    creategif=true,
    pathname="juliaspinner.gif")

julia spinner

Obviously, if you increase the range from 1:60 to, say, 1:300, you'll generate 300 drawings rather than 60, and the rotation will take longer and will be much smoother. Of course, you could change the framerate to be something other than the default 30.

Use createmovie = true instead of creategif = true if you want to make a video file. Specify the location and format for the video file in pathname:

...
    createmovie = true,
    pathname = "/tmp/juliaspinner.mkv"
...

2: Combining scenes

In the next example, we'll construct an animation that uses different scenes.

Consider this animation, showing the sun’s position for each hour of a 24 hour day. (It’s only a model...)

sun24 animation

Again, start by creating a movie, a useful handle that we can pass from function to function. We'll specify 24 frames for the entire animation.

sun24demo = Movie(400, 400, "sun24", 0:23)

We'll define a simple backgroundfunction function that draws a background that will be used for all frames (since animated GIFs like constant backgrounds):

function backgroundfunction(scene::Scene, framenumber)
    background("black")
end

A nightskyfunction draws the night sky, covering the entire drawing:

function nightskyfunction(scene::Scene, framenumber)
    sethue("midnightblue")
    box(O, 400, 400, action = :fill)
end

A dayskyfunction draws the daytime sky:

function dayskyfunction(scene::Scene, framenumber)
    sethue("skyblue")
    box(O, 400, 400, action = :fill)
end

The sunfunction draws a sun at 24 positions during the day. Since the framenumber will be a number between 0 and 23, this can be easily converted to lie between 0 and 2π.

function sunfunction(scene::Scene, framenumber)
    t = rescale(framenumber, 0, 23, 2pi, 0)
    gsave()
    sethue("yellow")
    circle(polar(150, t), 20, action = :fill)
    grestore()
end

And finally, tere's a groundfunction that draws the ground, the lower half of the drawing:

function groundfunction(scene::Scene, framenumber)
    gsave()
    sethue("brown")
    box(Point(O.x, O.y + 100), 400, 200, action = :fill)
    grestore()
    sethue("white")
end

To combine these together, we'll define a group of Scenes that make up the movie. The scenes specify which functions are to be used, and for which frames:

backdrop  = Scene(sun24demo, backgroundfunction, 0:23)   # every frame
nightsky  = Scene(sun24demo, nightskyfunction, 0:6)      # midnight to 06:00
nightsky1 = Scene(sun24demo, nightskyfunction, 17:23)    # 17:00 to 23:00
daysky    = Scene(sun24demo, dayskyfunction, 5:19)       # 05:00 to 19:00
sun       = Scene(sun24demo, sunfunction, 6:18)          # 06:00 to 18:00
ground    = Scene(sun24demo, groundfunction, 0:23)       # every frame

Finally, the animate function scans all the scenes in the scenelist for the movie, and calls the specified functions for each frame to build the animation:

animate(sun24demo, [
       backdrop, nightsky, nightsky1, daysky, sun, ground
   ],
   framerate=5,
   creategif=true)

sun24 animation

Notice that, for some frames, such as frame 0, 1, or 23, three of the functions are called: for others, such as 7 and 8, four or more functions are called.

An alternative

As this is a very simple example, there is of course an easier way to make this particular animation.

We can use the incoming framenumber, rescaled, as the master parameter that determines the position and appearance of all the graphics.

function frame(scene, framenumber)
    background("black")
    n   = rescale(framenumber, scene.framerange.start, scene.framerange.stop, 0, 1)
    n2π = rescale(n, 0, 1, 0, 2π)
    sethue(n, 0.5, 0.5)
    box(BoundingBox(), action = :fill)
    if 0.25 < n < 0.75
        sethue("yellow")
        circle(polar(150, n2π + π/2), 20, action = :fill)
    end
    if n < 0.25 || n > 0.75
        sethue("white")
        circle(polar(150, n2π + π/2), 20, action = :fill)
    end
end

Live graphics

Although Luxor is designed primarily for creating static graphics, it's possible to run it with an external buffer to display moving graphics. This example finds an approximation to π, updating the window continually with the latest estimate:

using Luxor
using MiniFB
include(dirname(pathof(Luxor)) * "/play.jl")

function run()
    within_circle = 0
    total_points = 0
    @play 400 400 begin
        pt = Point(rand(), rand())
        total_points += 1
        d = distance(O, pt)
        if d <= 1
            randomhue()
            circle(200pt, 1, :fill)
            within_circle += 1
        end
        pi_estimate = 4.0within_circle / total_points
        # show estimate
        sethue("black")
        box(boxtopleft(), boxmiddleright(), :fill)
        sethue("white")
        fontsize(30)
        text(string(pi_estimate), boxtopleft() + (10, 50), halign=:left)
        fontsize(20)
        text(string(total_points), boxtopleft() + (10, 100), halign=:left)
        sleep(0.05)
    end
end

run()

pi live drawing a few moments later...

For more information, see Interactive graphics and Threads.