This Jupyter notebook can be found in this collection of Clawpack apps as the file $CLAW/apps/notebooks/visclaw/animation_tools_demo.ipynb.
To run this notebook, install Clawpack, and download it from the link above, or clone the entire apps repository.
A static view of this and other notebooks can be found in the Clawpack Gallery of Jupyter notebooks.
This notebook demonstrates the tools in the clawpack.visclaw.animation_tools module (new in Version 5.4.0 and substantially updated in v5.7.0), which facilitates creating a list of figures or images and viewing them as an animation.
Several approaches are supported:
Using the ipywidgets.interact allows generating an interactive widget for sweeping through the frames. This may be easiest to work with when running a notebook interactively, including when using http://mybinder.org.
Creating an animation as javascript in the notebook. This can now be done directly using the to_jshtml method defined for an animation created using matplotlib.animation.
Historical note: this approach was first pioneered in Clawpack and developed further by Jake Vanderplas in his JSAnimation package, later incorporated into matplotlib.
Alternatively the to_html5_video method can be used to create an mp4 video.
The to_jshtml versions can also be embedded in another webpage or in Sphinx documentation, using the animation_tools.make_html and animation_tools.make_rst functions.
_plots directory to animations:¶The animation_tools.make_anim_outputs_from_plotdir function can be used to turn the png files in a Clawpack _plots directory into stand-alone .html, .mp4, or .rst files.
All of these possibilities are illustrated below.
%pylab inline
from __future__ import print_function
from ipywidgets import interact, interact_manual
import ipywidgets
import os
import io
import base64
from IPython.display import display, FileLink, FileLinks, HTML
from clawpack.visclaw import animation_tools
You can make figures however you want. As they are created, append to a list called figs:
figs = []
x = linspace(0,1,1000)
for k in range(11):
fig = figure(figsize=(5,3))
plot(x, sin(2*pi*k*x), 'b')
ylim(-1.1,1.1)
title('$\sin(2\pi kx)$ for k = %s' % k)
figs.append(fig)
close(fig)
IPywidgets are sometimes a nice way to animate a set of images. But note that the widgets only work if you are running the notebook. If you save it as an html file (e.g. to post on the web) then the widgets don't work. So if you are viewing a rendered version of this notebook, you won't see output for several of the cells below unless you download and run the notebook.
A better approach is generally to embed the animation as javascript or an mp4 file, as discussed in the next section.
The tool animation_tools.interact_animate_figs can be used to create an interact that loops over all the frames contained in a list of figures.
animation_tools.interact_animate_figs(figs)
Passing in the argument manual=True will use the widget interact_manual instead of interact. This refrains from updating the image as you move the slider bar. Instead you move the slider as desired and then click on the Run button to re-display the image. This is useful if there are many frames and you want to be able to jump to around without all the intermediate frames being displayed, which can slow down the response significantly.
animation_tools.interact_animate_figs(figs, manual=True)
The argument TextInput=True can be specified to produce a text input cell rather than a slider bar:
animation_tools.interact_animate_figs(figs, manual=True, TextInput=True)
The tool animation_tools.interact_animate_images can be used to create an interact that loops over all the frames contained in a list of images rather than figures. The images can be generated from a list of figures, as illustrated in the next cell. Or they can be read in from a directory of png files, for example, as illustrated later.
This function also takes the arguments manual and TextInput as described above, with default values False.
images = animation_tools.make_images(figs)
animation_tools.interact_animate_images(images, figsize=(6,3))
A list of images can also be turned into a animation inline plot. This is often preferable to using widgets. In particular, these movies are persistent if you export the notebook as an html file and can be played by anyone viewing the html file.
images = animation_tools.make_images(figs)
anim = animation_tools.animate_images(images, figsize=(6,3))
HTML(anim.to_jshtml())
HTML(anim.to_html5_video())
Alternatively, a separate stand-alone html page can be created with the JSAnimation. This can be posted on the web for others to view, for example.
file_name = 'SineFunctions.html'
animation_tools.make_html(anim, file_name=file_name, title="Sine Functions",
raw_html="Illustration of html file created by <tt>make_html</tt>")
FileLink(file_name)
file_name = 'SineFunctions.mp4'
animation_tools.make_mp4(anim, file_name=file_name)
You can also produce a .rst file containing the animation, which can be incorporated into Sphinx documentation:
file_name = 'SineFunctions.rst'
animation_tools.make_rst(anim, file_name=file_name)
Sometimes it is useful to create a directory containing png files for each figure, or you might have such a directory generated by other means (e.g. from VisClaw).
Here we create a directory named _plots to store png files for each frame:
plotdir = '_plots'
animation_tools.make_plotdir(plotdir, clobber=True)
You can create frames any way you wish and then use animation_tools.save_frame to save each one. You can also explicitly call savefig, but then you should construct file names such that glob.glob can be used to return a list of filenames in the directory that are in the correct order for plotting as frames in an animation. The animation_tools.save_frame creates names such as frame00000.png etc. as shown below. The optional argument fname_base can be used to change frame to something else.
x = linspace(0,1,1000)
for k in range(5):
fig = figure(figsize=(6,3))
plot(x, cos(2*pi*k*x), 'b')
ylim(-1.1,1.1)
title('$\cos(2\pi kx)$ for k = %s' % k)
animation_tools.save_frame(k, verbose=True)
close(fig)
FileLinks(plotdir)
Given such a directory of png files, we can read them in using animation_tools.read_images to produce a list of images:
cosine_images = animation_tools.read_images(plotdir='_plots', fname_pattern='frame*.png')
print("Found %i images" % len(cosine_images))
The resulting list of images new_images can now be animated using any of the tools illustrated earlier, e.g.
#animation_tools.interact_animate_images(cosine_images)
anim = animation_tools.animate_images(cosine_images, figsize=(6,3))
HTML(anim.to_jshtml())
If you already have a list of figures or images and wish to create a directory containing them, the utilities animation_tools.save_figs or animation_tools.save_images can be used. For example:
animation_tools.save_images(cosine_images, plotdir='cosine_images', fname_base='wavenumber',
format='png', verbose=True, kwargs={'dpi':150})
FileLinks('cosine_images')
reload_images = animation_tools.read_images(plotdir='cosine_images', fname_pattern='wavenumber*.png')
animation_tools.interact_animate_images(reload_images)
Finally we illustrate how plots in an existing _plots directory created using a Fortran version of Clawpack Classic, AMRClaw, or GeoClaw can be animated in the notebook.
Here we assume the plotdir referenced below has already been created e.g. via make .plots at the command line.
Also take a look at the clawutil.nbtools module for a set of tools that can be used to compile and run Clawpack and create plots from within a notebook. These are illustrated in some of the other notebooks in this apps repository, in particular in $CLAW/apps/notebooks/classic/advection_1d/advection_1d.ipynb
plotdir = '../classic/advection_1d/_plots'
anim = animation_tools.animate_from_plotdir(plotdir, figno=1, figsize=(10,6))
HTML(anim.to_jshtml())