Create animated GIFs from screen recording

Are you developing a new game? You are proud of your new creation and you want to demonstrate it without further delay on Discord or on our website? You will probably want to create a Animated GIF to showcase your masterpiece to the entire community. So, learn to master META Screen Recorder: a new tool of capture for META.

Advanced

1h

Academy
Create animated GIFs from screen recording

Overview

Here you will learn how to configure and use META Screen Recorder for all your Gamebuino projects. This utility complements the capture tool provided with the META (accessible by the Home button), to offer you more flexibility, simplicity and features:

  • You will be able to start and stop recording the screen whenever you want. Indeed, you will no longer have to interrupt the progress of your application to start a recording. A simple and prolonged press of the Menu button is all it takes. What about stopping the recording? Briefly press the Menu button again. It's that simple!

  • You can also start and stop the recording directly in your code! This feature will allow you to record a game scene more accurately, or when specific events occur (don't miss the moment you knock out the boss anymore).

  • You can also take advantage of META's full capabilities and record your applications running in High Resolution!

To apply what you will learn here, you must of course have a Gamebuino META, which is essential to be able to make a screen recording.

Demonstration

Here is a demonstration of the recording process performed with META Screen Recorder:

Architecture

META Screen Recorder is based on two components:

  • ScreenRecorder: a C++ class that you will have to integrate into your project. This class sends all display data, during the execution of your application, to the serial port when your console is connected to your computer via a USB cable.

  • screenrecord: a standalone PHP script that you will need to install and run on your computer to receive all the data sent by META on the serial port. This script decodes all the data it receives and rebuilds the screenshots that follow one another to save them in PNG files.

All you have to do is compile all these PNG files with a specialized tool (ImageMagick) to generate a beautiful animated GIF that accurately reflects what happened on the console screen.

Software environment required

To be able to use META Screen Recorder and generate animated GIFs, you will need to install the following free software on your computer:

  • PHP 7.x in version CLI (Command Line Interface), i. e. accessible from your shell.

  • The GD extension for PHP, which allows you to create images in a wide range of formats.

  • ImageMagick, which includes a library and a set of command line utilities to create, convert, edit and display images in a wide variety of formats.

We will not detail here the procedures for installing these software.
Nevertheless, you can find these procedures on the detailed documentation of META Screen Recorder.

Before continuing, make sure you have installed these different software.
To check it, open a shell and execute the following commands:

$ php -v              # displays the installed version of PHP
$ php -m              # displays the list of extensions installed for PHP
$ convert -version    # displays the installed version of ImageMagick

Downloading the ScreenRecorder class

We will now integrate the ScreenRecorder class into your C++ project.
First download the following two files (right-click + Save Link as...):

  • ScreenRecorder.h
  • ScreenRecorder.cpp

Once downloaded, move these two files in the root directory of your Gamebuino project.

Manual recording configuration

Remember that META Screen Recorder allows you to take screenshots in two ways:

  • Manually, by using the Menu* button to start and stop recording.
  • Automatically, by inserting instructions in the code to start and stop recording.

* You are free to modify the code of the ScreenRecorder class if you want to redefine a combination of buttons of your choice instead of the Menu button.

First, let's look at the manual recording procedure.

Remember, ScreenRecorder is able to take screenshots in the standard resolution (80x64), but also in high resolution (160x128). We will distinguish these two cases here.

For the standard resolution

Open your main sketch and insert the following lines:

#include <Gamebuino-Meta.h>
#include "ScreenRecorder.h" // <-- insert this line

void setup() {
    gb.begin();
    // your initialization
    // instructions
    ScreenRecorder::init();          // <-- insert this line
    ScreenRecorder::setForWindows(); // <-- also insert this one if you are on Windows
}

void loop() {
    while (!gb.update());
    // your instructions to
    // make on-screen plots
    // using gb.display
    ScreenRecorder::monitor(gb.display._buffer); // <-- insert this line
}

That's all you have to do!... Simple, right?

Important note about Windows

The implementation of the PHP fread function in Windows is buggy: indeed, the read buffer only reveals the accumulated data in 8K packets. Therefore, we are forced to use a trick to get rid of this bug. So it is imperative to add the following line to specify that the data reception will be done on Windows:

ScreenRecorder::setForWindows();

If you are on macOS or Linux, do not add this line.

For high resolution

In the case of an application developed for high resolution, things are a little different. Indeed, you will not be able to use the traditional display methods provided by gb.display... I encourage you to read Andy's excellent article on the subject: High Resolution without gb.display, which briefly explains why this is not possible and provides you with a workaround method that consists of using gb.tft directly.

I wrote a very complete tutorial on how to deepen this technique. You can read it if you are interested (beginners may have some difficulty assimilating everything): Shading Effect in High Resolution. It will give you an in-depth understanding of how to apply this technique in your applications.

Let's take the example from Andy's article, which I slightly modified to adapt it to a division of the screen into horizontal slices. Download the SketchExampleForHD.ino code (right-click + Save Link as...), create a new directory and move the downloaded source code to it.

Then open the SketchExampleForHD.ino sketch and check that the following lines appear:

#include <Gamebuino-Meta.h>
#include "ScreenRecorder.h" // <-- loading the ScreenRecorder class

void setup() {
    gb.begin();
    gb.display.init(0, 0, ColorMode::rgb565);
    gb.setFrameRate(32);
    ScreenRecorder::init(SLICE_HEIGHT); // <-- initializes ScreenRecorder
    ScreenRecorder::setForWindows();    // <-- !!! insert this line if you are on Windows !!!
}

void loop() {
    while (!gb.update());
    for (
        uint8_t sliceIndex = 0;
        sliceIndex < SCREEN_HEIGHT / SLICE_HEIGHT;
        sliceIndex++
    ) {
        // 
        // at the end of the loop...
        // 
        customDrawBuffer(0, sliceY, buffer, SCREEN_WIDTH, SLICE_HEIGHT);
        ScreenRecorder::monitor(buffer, sliceIndex); // <-- screen monitoring
    }
    waitForPreviousDraw();
}

You will notice that, unlike the configuration for the standard resolution:

  • we initialize ScreenRecorder by setting the height of the screen slices.

  • and we also specify the index of the current slice during the monitoring of the screen: it is necessary for it to start the transmission to the PHP script as soon as it detects the first slice (the one of zero index).

In addition, you can see that the ScreenRecorder::monitor() is only invoked after the current slice buffer has been filled. Indeed, at that time, all the plots on the current slice that are intended to be sent to the screen through the DMA controller have been completed.

Important note

When initializing the ScreenRecorder, note that the height of the slices must necessarily be a non-zero power of 2, whose maximum value is 16… so only the values 2, 4, 8 and 16 will be taken into account. If you use a value greater than 16, the recording will simply be disabled. And if you use a value lower than 16 which is not a power of 2, you will get a rather strange recording...

I advise you to use the value 8 which only requires 2 x 2.5 = 5 KB in RAM.

Configuring automatic recording

Instead of triggering and stopping the recording using the Menu button, you can also do it automatically by inserting the corresponding instructions directly into your code. This can be very useful when you want to start or stop recording at specific times, or when certain events occur. To do this, simply insert each of the following two instructions in the appropriate places:

ScreenRecorder::startRecording();
ScreenRecorder::stopRecording();

Downloading the PHP screenrecord script

All we have to do is download and install the PHP screenrecord script on your computer to receive the data sent on the serial port by the ScreenRecorder class, and convert this data into a series of PNG files.

First download the script (right-click + Save Link as...): screenrecord
And move it to your project folder.

Note, by the way, that rather than systematically duplicating the screenrecord script in each of your project folders, it is quite possible to install it globally in your user space or within your system. This procedure is detailed in the META Screen Recorder documentation.

Then, open a shell, go to your project folder, and check that you are able to execute the script:

macOS & Linux

$ cd ~/path/to/your/project/folder # we move to the right folder
$ chmod u+x screenrecord           # we allow the execution of the script
$ ./screenrecord -h                # we launch the script with the help option

Windows

$ cd C:\path\to\your\project\folder # we move to the right folder
$ php screenrecord -h               # we launch the script with the help option

You should get the following output:

+---------------------------------------+
|    Gamebuino META Screen Recorder     |
| © 2019 Stéphane Calderoni (aka Steph) |
|     https://gamebuino.com/@steph      |
+---------------------------------------+

Usage: screenrecord [options]

DESCRIPTION

	This utility records the Gamebuino META screen in Standard & High Definition
	using communication through a serial port

OPTIONS

	-h  displays the synopsis
	-p  sets the serial port (default: /dev/cu.usbmodem141401)
	-d  sets the output directory (default: current directory)
	-b  sets the image files basename (default: frame)
	-n  sets the number of digits to number the image files (default: 4)

ANIMATED GIF CREATION

	You must have imagemagick installed to be able to generate the animated GIF file.
	For example, to generate an animation with a resolution of 160x128 pixels at 25 fps,
	simply type the following command:

	convert -delay 4 -loop 0 png_dir_path/*.png -scale 160x128 screenrecording.gif

The `-h' option tells you how to use the script.
We will focus here mainly on the following options:

  • -p: which allows you to specify the identifier of the serial port to which the META is connected
  • -d: which allows you to specify the directory in which the PNG files will be saved

We will immediately create a directory to save the PNG files and avoid putting the box in the directory of your project. We will name this directory frames:

$ mkdir frames

Identify the serial port

To determine which serial port your META is connected to, launch the Arduino IDE and go to the Tools menu:

Identify the serial port

Note about Windows

You will first have to go to the Device Manager to identify the right port:

Device Manager

Start the recording

Compile and upload your project to your Gamebuino. Wait for the application to launch, then run the screenrecord script to listen on the serial port (specifying the correct port):

$ ./screenrecord -d frames -p /dev/cu.usbmodem141401  # <-- macOS & Linux
$ php screenrecord -d frames -p COM13                 # <-- Windows 10

Then, press and hold (at least 1 second) the Menu button on the META to start recording. You will notice that the LEDs on the console emit red flashes at regular intervals. This means that it is actually sending the display data to the serial port. At the same time, you will see on your shell that the script receives this data and records it in the order of arrival of the captures. You should observe something like that:

Start listening to the serial port /dev/cu.usbmodem141401

Waiting for data... Start screen recording in high resolution

Decoding frame 0001... saved
Decoding frame 0002... saved
Decoding frame 0003... saved
Decoding frame 0004... saved
Decoding frame 0005... saved
Decoding frame 0006... saved
Decoding frame 0007... saved
Decoding frame 0008... saved
Decoding frame 0009... saved
Decoding frame 0010... saved
.
.
.

To stop recording, briefly press the Menu... button again. The LEDs go out, the data flow stops and the script stops also:

.
.
.
Decoding frame 0100... saved
Decoding frame 0101... saved
Decoding frame 0102... saved
Decoding frame 0103... saved
Decoding frame 0104... saved
Decoding frame 0105... saved
Decoding frame 0106... saved

106 PNG files have been recorded in 160x128 (high resolution).

You must have imagemagick installed to convert these PNG files into an animated GIF file.
For a 25 fps animation, simply type the following command:

convert -delay 4 -loop 0 frames/*.png -scale 160x128 screenrecording.gif

You can already check that the script has saved a sequence of PNG files in the frames directory:

$ ls -l frames  # <-- macOS & Linux
$ dir frames    # <-- Windows

The command to create an animated GIF from the series of captures is indicated:

$ convert -delay 4 -loop 0 frames/*.png -scale 160x128 screenrecording.gif

Note on the delay option

The value N expected by the delay option is an integer that corresponds to an animation frequency of 100/N fps. So, if N=4, your animated GIF will have a frequency of 100/4 = 25 fps.

In the case of our example SketchExampleForHD.ino for high resolution, remember that we had set a framerate of 32 fps :

gb.setFrameRate(32);

So for our GIF to roughly respect this frequency, we will have to set N=3 instead. In addition, it is possible to create an image with dimensions different from those of the original capture. For example, let's go from (160x128) to (320x256):

$ convert -delay 3 -loop 0 frames/*.png -scale 320x256 screenrecording.gif

And we get the screenrecording.gif file whose beautiful content is as follows:

animated GIF

Note

I cheated a little... I deleted the extra frames so that the loop animation is perfectly connected... (it is enough here to keep only the first 64 frames). It is still possible to correct your recording, by deleting frames, or reordering them if necessary. Practical, isn't it?

The final word

That's it, we've reached the end of this little tutorial. You should now be able to capture animations of all your projects with META Screen Recorder. I hope you will find this tool useful. I had a great need for it myself, and I am very happy to be able to share it today. If you notice any dysfunctions, if you see possible improvements, or if you want to show me the captures you are most proud of, don't hesitate to leave me a little comment on the page of this creation!

The code of the ScreenRecorder class and the screenrecord script are available on my GitHub repository. You can modify them freely, and if you improve them, we count on you to share them with us!

The author

Steph

https://gamebuino.m1cr0lab.com

See profile