Machine-generated memes in Perl

The Cat's Meow

Author(s):

It started off harmlessly enough with a few funny pictures of cats, but eventually it became the Internet phenomenon par excellence. It's no joke: Perl gives you some great tools for building and customizing memes yourself.

Summer time is intern time: As always during the summer months, my employer has taken on some college students, while we old guys scratch our balding pates and wonder how weird young academics can get. This year, the interns' sense of humor was on full display: Every presentation was adorned with image macros [1] for the purpose of amusement, either as static pictures or animated GIFs in infinite loops.

What started with "I Can Has Cheezburger?" [2] with cuddly kittens – the Lolcats – and cheeky sayings has morphed into an established part of culture known as the "meme." Take an expressive image and put an orthographically or a grammatically challenged saying (Lolspeak) in the header and footer using the Impact font – and you have a ready-made joke (Figure 1).

Figure 1: Modern classic – a Lolcat with orthographically questionable Lolspeak. © Niccolò Capanti.

The word "meme" originates from ancient Greek, where "mimema" means something imitated, and evolutionary biology uses the term to describe the process of social transmission of cultural values. On the Internet, memes are a mass phenomenon that spreads virally by email, chat, or on social networks (Figure 2).

Figure 2: A cartilaginous fish with a house – a typical example of an image macro meme.© Sarah Charlesworth.

Machine vs. Manual

Using a graphics program like GIMP, image macros are quickly made in the conventional manner, but a Perl script gives you a fast command-line-based approach. Listing 1 [3] shows the simplest version that hard codes the coordinates measured previously for the text strings. The CPAN Imager module reads the original file, turtle.jpg, which is a photo that I personally took on vacation in Hawaii of a giant turtle swimming.

Listing 1

meme-first

 

I found the Impact font in my Ubuntu distribution as a .ttf file in the path shown in Listing 1 (lines 12 and 13). I call the string() method twice, once with the footer and once with the header, set the color to white, and specify a font size of 60. Then, I turn on anti-aliasing for less powerful displays and write the output file with write() – the result is the fantastic joke that you can see in Figure 3.

Figure 3: At the command line, meme-first generates a zeitgeist joke.

Variable Comfort Levels

For variable text strings, the script needs to position the strings dynamically in the center. Listing 2 expects three parameters – the image to be modified, the header, and the footer – and turns them into an image macro. The following line

Listing 2

meme-simple

 

meme-simple turtle.jpg "ARRIVING FIRST" "SO NOT WORTH IT."

generates a file by the name of turtle-meme.jpg. To achieve this, the script defines a vertical distance $margin_y from the top edge of the image to the header and from the bottom edge of the image to the footer.

The function defined in lines 71-86 (dimensions) calculates the width and height of the string produced using Impact font size 60. To do this, it passes in the desired string to the bounding_box() method of an Imager::Font type object. The result is the width and height in pixels, after the function has subtracted the potentially negative coordinates of the left edge of the first glyph in the string ($neg_width) from the position at the right end of the string ($pos_width).

It uses the same approach for the position at the top ($asc) and bottom ($desc) edges of the string. The values $global_asc and $global_desc don't matter here, because they do not relate to the current string, but to the maximum possible extent of any glyphs.

Lines 35 and 39 center the footer and header, respectively, relative to the photo by dividing the overall width of the picture returned by getwidth() by two and subtracting half the string width. The results are the required starting coordinates of the centered string as an x-y pair of values, with x running from left to right and y from top to bottom.

As in Listing 1, the string() method in Listing 2 now draws the two text strings in the Impact font on the image, and the write() method writes the result to a file on disk, adding a -meme suffix to the file name.

Running Gag

A short movie is even funnier. The browser loads an animated GIF image in one fell swoop and plays back the frames it contains until the end of time, if the infinite flag is set. This method dates back to 1987 and is still very popular, despite HTML5, especially because it also works in very old browsers. Video sequences without sound can easily be embedded into HTML; Wikipedia pages, for example, use this technique to visualize algorithms or the interaction between the moving parts of mechanical equipment.

Comedians in the software development business add GIFs to PowerPoint presentations and comment fields for pull requests on GitHub. The often jerky frames are reminiscent of slapstick scenes from the early days of cinema or clumsy "Candid Camera" material. Incidentally, the argument that has been raging for decades about whether GIF is pronounced "giff" or "jiff" remains unresolved to the present day. Only the front between both righteous parties has hardened [4].

Strategically extracting individual frames from a video file sounds like a job for MPlayer:

mplayer -vf screenshot <video>.avi

While the movie is playing, you need to press the S key to save the next frame displayed as a PNG file. When you are done, the current directory will contain files numbered shot0000.png to shot<XXXX>.png with the frames you grabbed (Figure 4).

Figure 4: Frames can be copied from the video with MPlayer and displayed with the Eye of Gnome viewer tool.

To avoid bloating the .gif file, the total number of frames should be no more than about 20. Also, rapid-motion scenes need a faster sequence of frames for the viewer to keep up. To do this, simply press S twice as fast as usual during the rapid-motion scenes.

Motivation: Annoying People

As a demo, I used a sequence I shot during another vacation from my hotel room on the beach boardwalk at Venice Beach near Los Angeles [5]. It shows a tourist desperately trying to tow back a stubborn Segway to the rental company. Schadenfreude is the only true joy, says the cynic.

After extracting the photos from the video as described earlier, Listing 3 converts the individual frames into an animated GIF movie using the Imager module:

Listing 3

anigif

 

anigif shot*.png

The for loop in line 7 iterates over all file names passed in at the command line. The animation size of 300x200 pixels in lines 14 and 15 and the 26 total frames selected result in an animated GIF of about 1MB named anim.gif.

The write_multi() method writes the frames read previously with the new() function to the hard drive as a .gif file. The necessary image format conversions are done automatically behind the scenes. The make_colors option uses mediancut to normalize the color table between the frames, thereby ensuring faster conversion. It is important to set the gif_loop option to  , causing the browser to run the image sequence forever after loading the image (Figure  5).

Figure 5: The GIF now runs in a browser in an endless loop.

For a perfect finishing touch, I want to add a funny header to all the screenshots, while leaving the footer empty. The script in Listing 1 will do just fine for the single frames:

for i in *.png
do
  meme-simple $i "SEGWAY FAIL" ""
done

The command anigif shot*-meme.png reads all files with the -meme.png extension and generates the animated GIF. Figure 6 shows a scene from the masterpiece. Throughout the entire movie, the heading with the title stays motionless and perched 100 pixels below the top edge of the (originally sized) image because the string is built into every frame in the same way. It is white and uses the Impact joke font. If you like, you can check out the animated image online  [6].

Figure 6: The animated GIF contains a header in Impact font in each frame.

Thoughts

At the cutting table, I muse about Internet humor and other design options: Should I perhaps generate jokes automatically from random text? Maybe using quotes from the IMDB movie portal? My interns will be really surprised next summer.

Infos

  1. Image Macros: http://en.wikipedia.org/wiki/Image_macro
  2. "I Can Has Cheezburger?": http://en.wikipedia.org/wiki/I_Can_Has_Cheezburger
  3. Listings for this article: ftp://ftp.linux-magazin.de/pub/listings/magazine/157
  4. "Battle Over 'GIF' Pronunciation Erupts" by Amy O'Leary, New York Times: http://bits.blogs.nytimes.com/2013/05/23/battle-over-gif-pronunciation-erupts/
  5. Michael Schilli, "Segway FAIL" (Video): http://www.youtube.com/watch?v=8_EbnF9xl-g
  6. Animated GIF for "Segway FAIL" movie: http://perlmeister.com/anim.gif

The Author

Mike Schilli works as a software engineer with Yahoo! in Sunnyvale, California. He can be contacted at mailto:mschilli@perlmeister.com. Mike's homepage can be found at http://perlmeister.com.