This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
user:smeas:portfolio:hpc0project2 [2013/05/05 21:33] – [Reflection] smeas | user:smeas:portfolio:hpc0project2 [2013/05/05 21:40] (current) – [Code] smeas | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ======Creating Animated gifs with the GD Image Library====== | ||
+ | So, you see all these funny animated images people have made, and you wonder to yourself, "How do they do that? Is it magic? Sorcery? Wizardry?" | ||
+ | |||
+ | In all actuality, it's actually pretty simple! And this tutorial is going to show you how to use the GD library, and a really handy and simple Linux tool to create an animated gif of your very own! | ||
+ | =====Objectives===== | ||
+ | The purpose of this project is to create a procedurally generated set of images using the C programming language and the GD image library. The user will write a program which generates a set number of images, and then runs a Linux tool called " | ||
+ | =====Prerequisites===== | ||
+ | There are three prerequisites to this project. | ||
+ | |||
+ | * Intermediate knowledge of C programming (and preferably the implementation of the GD image library) | ||
+ | * A Linux machine with the GD Image Library installed | ||
+ | * A Linux machine with the imagemagick package installed. | ||
+ | |||
+ | These packages are already installed on Lab46. If you plan on doing this on a different machine, however, you can find them in your distribution' | ||
+ | |||
+ | To install these on Ubuntu, or any other Debian based distribution, | ||
+ | |||
+ | GD Image Library | ||
+ | <cli> | ||
+ | sudo apt-get install libgd2-xpm libgd2-xpm-dev imagemagick | ||
+ | </ | ||
+ | |||
+ | =====Background===== | ||
+ | For the EOCE for C Programming, | ||
+ | |||
+ | {{: | ||
+ | |||
+ | It wasn't pretty, as each image was randomly generated. But I thought it was pretty fun to do. So, I decided to pass on my knowledge to others so they may also partake in creating gifs using the GD library (hopefully prettier ones, and not just entropic foliage). | ||
+ | |||
+ | =====Scope===== | ||
+ | Upon completion, you should have an animated gif created with the GD Image Library. If you follow the tutorial correctly, it will be a spiral made up of multi-colored circles. | ||
+ | |||
+ | =====Code===== | ||
+ | <code c> | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | #define BLACK 0 | ||
+ | #define GRAY 1 | ||
+ | #define VIOLET | ||
+ | #define INDIGO | ||
+ | #define BLUE 4 | ||
+ | #define GREEN 5 | ||
+ | #define YELLOW | ||
+ | #define ORANGE | ||
+ | #define RED 8 | ||
+ | #define WHITE 9 | ||
+ | |||
+ | #define PI 3.1415926535897 | ||
+ | </ | ||
+ | |||
+ | The first step is to set up our header files, and to define names for our colors to make things a little more readable. | ||
+ | |||
+ | <code c> | ||
+ | int main(int argc, char* argv[]) | ||
+ | { | ||
+ | // Creation of file pointers for our output files and GD image | ||
+ | FILE *out; | ||
+ | gdImagePtr img; | ||
+ | |||
+ | // Setting of the output files name formatting | ||
+ | char format[] = " | ||
+ | char outfile[sizeof format+100]; | ||
+ | |||
+ | // Instantiation of the array of usable colors | ||
+ | unsigned int color[10]; | ||
+ | |||
+ | // Instantiation of the image dimensions | ||
+ | int wide, high; | ||
+ | |||
+ | // Instantiation of variables needed for the spiral. | ||
+ | int degree, x, y; | ||
+ | float radian; | ||
+ | |||
+ | // Counter variables | ||
+ | int imageNum, i, j, k; | ||
+ | </ | ||
+ | |||
+ | The declaration of the main function, and declaration of the variables used for the program. A breakdown of important bits relevant to this tutorial is as follows. | ||
+ | * <code c>FILE *out; | ||
+ | gdImagePtr img;</ | ||
+ | * Creates pointers for our output file and to our GD image. | ||
+ | * <code c>char format[] = " | ||
+ | char outfile[sizeof format+100];</ | ||
+ | * This is where we define how the image files will be dynamically named. First, if you're writing to another folder (like we are, writing to the subfolder output), be sure that folder exists before you run your program. | ||
+ | * <code c> | ||
+ | * This is an array made to hold all the color values we plan on creating for the images. You can create many more, but for this purpose of this tutorial, we will be using 10. | ||
+ | |||
+ | <code c> | ||
+ | /* Conditional statement for size of images based on arguments, defaulting to correct syntax in | ||
+ | * case of improper usage | ||
+ | */ | ||
+ | if(argc == 1) | ||
+ | { | ||
+ | wide = 800; | ||
+ | high = 600; | ||
+ | printf(" | ||
+ | printf(" | ||
+ | } | ||
+ | else if(argc == 3) | ||
+ | { | ||
+ | wide = atoi(argv[1]); | ||
+ | high = atoi(argv[2]); | ||
+ | printf(" | ||
+ | printf(" | ||
+ | } | ||
+ | else | ||
+ | { | ||
+ | printf(" | ||
+ | printf(" | ||
+ | exit(1); | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | This if statement determines what to set the size of the images to, based on the arguments given at runtime. If the program is run with no additional arguments, it will set the images to a resolution of 800x600 by default. If run with two additional arguments, it will set the resolution to the first argument x the second argument. If run with any other number of arguments, it will print the correct syntax of the program, and then exit. | ||
+ | |||
+ | <code c> | ||
+ | // Main loop of the program. Set to run as many times as the number of images you wish to create | ||
+ | for(imageNum = 0; imageNum < 100; imageNum++) | ||
+ | { | ||
+ | // Creation of a new image | ||
+ | img = gdImageCreate(wide, | ||
+ | |||
+ | // Setting of the color values | ||
+ | color[BLACK] | ||
+ | color[GRAY] | ||
+ | color[VIOLET] | ||
+ | color[INDIGO] | ||
+ | color[BLUE] | ||
+ | color[GREEN] | ||
+ | color[YELLOW] | ||
+ | color[ORANGE] | ||
+ | color[RED] | ||
+ | color[WHITE] | ||
+ | |||
+ | /* Set the output file's name to the specified format, and name it based on the number of | ||
+ | * times the main loop has run. | ||
+ | */ | ||
+ | sprintf(outfile, | ||
+ | |||
+ | /* Loop to generate a number of circles based on the number of images created with | ||
+ | * positions and colors set relevant to the number of times through the loop | ||
+ | */ | ||
+ | for(i = 0; i <= imageNum; i++) | ||
+ | { | ||
+ | degree = i * 10; | ||
+ | radian = degree * (PI / 180); | ||
+ | x = (wide / 2) + (i * 2) * sin(radian); | ||
+ | y = (high / 2) + (i * 2) * cos(radian); | ||
+ | gdImageFilledArc(img, | ||
+ | } | ||
+ | |||
+ | // Write the output file, and then write the image to the output file | ||
+ | out = fopen(outfile, | ||
+ | gdImagePngEx(img, | ||
+ | |||
+ | // Print the file being written | ||
+ | printf(" | ||
+ | |||
+ | // Clean up | ||
+ | fclose(out); | ||
+ | gdImageDestroy(img); | ||
+ | } | ||
+ | |||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Now we're starting to get into the real meat of the program. Here is where we begin the main loop that this program will run. It is set to execute equal to how many images we want to create (As stated by the " | ||
+ | |||
+ | The image creation loop runs a variable amount of times, depending on which image is currently being generated (one time for the first image, two times for the second, etc.). The first thing done in this loop are a few calculations to get the x and y values for the center of the circle being generated this time through the loop. After the program has these values, we finally draw something. We use the gdImageFilledArc function to draw a filled arc! This function accepts 9 arguments. First, like the gdImageColorAllocate function, we pass it the pointer of the image we're working on. We then tell it the center x and y values of the circle that were calculated in the previous four lines. We tell it the width and height of the circle (in this case, (i + 3) for both, to increase the size of the circle as the loop executes more times), the starting and stopping point of the arc ((0, 360) to make a full circle), the color to be used (color[i % 10], to give us a different color depending on how many times the loop has executed), and finally which type of arc to create (gdArc, in this case). | ||
+ | |||
+ | After the image has been completely generated, the next step is to write the image out. We first set the out pointer we used earlier to open a file with the filename calculated earlier in the loop. Next, we use the gdImagePngEx function to write the image we've been working with (img) to the file we just opened (out). As a bit of a debug message, we then have a printf statement which tells us which current file is being written. And finally, the last step is clean up before the loop starts again. We use fclose to close the file we opened, and gdImageDestroy to get rid of the image we've been working with so we can make a new one. | ||
+ | |||
+ | |||
+ | =====Execution===== | ||
+ | Once we have our program all written, the first step is to compile it (being sure to link to the GD library with -lgd). | ||
+ | |||
+ | <cli> | ||
+ | gcc -o gdgif gdgif.c -lgd | ||
+ | </ | ||
+ | |||
+ | Next, run the program. For demonstration' | ||
+ | |||
+ | <cli> | ||
+ | ./gdgif 800 600 | ||
+ | </ | ||
+ | |||
+ | If everything worked right, you should see a bunch of messages with the output of the program saying that a total of 100 images were written to the output folder. Now that we have our images, the next step is to actually create a gif out of them. So first, cd into your output folder. Once there, do a quick ls to make sure you have 100 .png files, titled from 00.png to 99.png. And as long as they' | ||
+ | |||
+ | <cli> | ||
+ | convert -loop 0 *.png loop.gif | ||
+ | </ | ||
+ | |||
+ | This tells it to create an animated file, set to loop with a delay of 0 between loops, of all the .png files in this directory and save it as loop.gif. Once that is done, verify to be sure the loop.gif file is there. And if it is, you're on the final step. Enjoyment. :) If you followed this tutorial correctly, your loop.png file should look like this. | ||
+ | |||
+ | {{: | ||
+ | |||
+ | Now that you have your image, maybe you want to go back and make it your own. Maybe you already know the GD Image Library, or you're looking to experiment. Fortunately, | ||
+ | |||
+ | |||
+ | |||
+ | |||
+ | =====References===== | ||
+ | While written for the PHP GD Library, this manual was very helpful in looking up the syntax for functions. All it takes is converting the function name from it's name in the PHP library to it's name in the C library (imagefilledarc for example in the PHP library is named gdImageFilledArc in the C library). | ||
+ | * http:// |