Abstract
CL-GD is a library for Common Lisp which provides an interface to the GD Graphics Library for the dynamic creation of images. It is based on UFFI and should thus be portable to all CL implementations supported by UFFI.The focus of CL-GD is convenience and correctness, not necessarily speed. If you think CL-GD is too slow and you're concerned about speed, contact me before you start coding in C... :)
CL-GD comes with a BSD-style license so you can basically do with it whatever you want. Please send bug reports to the mailing list mentioned below if you encounter any problems with CL-GD. (I'm glad to fix CL-GD but I can't do much about GD, of course. So if CL-GD basically works for you but you encounter seemingly strange behaviour when drawing please try if and how you can achieve the intended result with GD directly. That would help me a lot. Thanks.)
(with-image* (200 200) ; create 200x200 pixel image (allocate-color 68 70 85) ; background color (let ((beige (allocate-color 222 200 81)) (brown (allocate-color 206 150 75)) (green (allocate-color 104 156 84)) (red (allocate-color 163 83 84)) (white (allocate-color 255 255 255)) (two-pi (* 2 pi))) ;; move origin to center of image (with-transformation (:x1 -100 :x2 100 :y1 -100 :y2 100 :radians t) ;; draw some 'pie slices' (draw-arc 0 0 130 130 0 (* .6 two-pi) :center-connect t :filled t :color beige) (draw-arc 0 0 130 130 (* .6 two-pi) (* .8 two-pi) :center-connect t :filled t :color brown) (draw-arc 0 0 130 130 (* .8 two-pi) (* .95 two-pi) :center-connect t :filled t :color green) (draw-arc 0 0 130 130 (* .95 two-pi) two-pi :center-connect t :filled t :color red) (with-default-color (white) (with-default-font (:small) (draw-string -8 -30 "60%") (draw-string -20 40 "20%") (draw-string 20 30 "15%")) (draw-freetype-string -90 75 "Global Revenue" ;; this assumes that 'DEFAULT_FONTPATH' ;; is set correctly :font-name "verdanab")))) (write-image-to-file "chart.png" :compression-level 6 :if-exists :supersede))
See below for more examples.
 
create-image
    create-image-from-file
    create-image-from-gd2-part
    destroy-image
    with-image
    with-image-from-file
    with-image-from-gd2-part
    *default-image*
    with-default-image
    with-image*
    with-image-from-file*
    with-image-from-gd2-part*
    write-jpeg-to-stream
    write-png-to-stream
    write-wbmp-to-stream
    write-gif-to-stream
    write-gd-to-stream
    write-gd2-to-stream
    write-image-to-stream
    write-image-to-file
    image-width
    image-height
    image-size
  WITH-CAST-POINTER and DEF-FOREIGN-VAR which haven't yet been ported to all UFFI platforms.
cl-gd.tgz, unzip and untar the file and put the resulting directory wherever you want, then cd into this directory.
cl-gd-glue.c into a shared library for your platform. On Linux this would be
gcc -fPIC -c cl-gd-glue.c ld -lgd -lz -lpng -ljpeg -lfreetype -lm -liconv -shared cl-gd-glue.o -o cl-gd-glue.so rm cl-gd-glue.o
cl-gd.asd can be seen from asdf (this is usually achieved by a symbolic link), start your favorite Lisp, and compile CL-GD:
(asdf:oos 'asdf:compile-op :cl-gd)
(asdf:oos 'asdf:load-op :cl-gd)
(asdf:oos 'asdf:load-op :cl-gd-test) (cl-gd-test:test)If you have the
georgiab.ttf
TrueType font from Microsoft you can also check the FreeType
support of CL-GD with
(cl-gd-test:test #p"/usr/X11R6/lib/X11/fonts/truetype/georgiab.ttf")where you should obviously replace the path above with the pull path to the font on your machine.
Note that CL-GD might work correctly even if some of the tests fail (as long as you don't get error messages). The exact results of the tests seem to depend on the versions of the C libraries which are used.
It is recommended that you at least skim over the original GD documentation before you start using CL-GD.
Note: If you're on Windows you might want to try this:
cl-gd-glue.dll from cl-gd-glue.dll and put it into the CL-GD folder. You don't need to download and install GD itself because it's already integrated into cl-gd-glue.dll.
WITH-IMAGE- macros instead of the
CREATE-IMAGE- functions so you can be sure that images
will always be destroyed no matter what happens.
[Function]
create-image width height &optional true-color => image
Allocates and returns an image with sizewidthxheight(in pixels). Creates a true color image iftrue-coloris true - the default isNIL. You are responsible for destroying the image after you're done with it. It is advisable to useWITH-IMAGEinstead.
[Function]
create-image-from-file file-name &optional type => image
Creates an image from the file specified byfile-name(which is either a pathname or a string). The type of the image can be provided astype(one of the keywords:JPG,:JPEG,:GIF,:PNG,:GD,:GD2,:XBM, or:XPM), or otherwise it will be guessed from thePATHNAME-TYPEoffile-name. You are responsible for destroying the image after you're done with it. It is advisable to useWITH-IMAGE-FROM-FILEinstead.
[Function]
create-image-from-gd2-part file-name src-x src-y width height => image
Creates an image from the part of the GD2 file specified byfile-name(which is either a pathname or a string) specified bysrc-x,src-y,width, andheight. You are responsible for destroying the image after you're done with it. It is advisable to useWITH-IMAGE-FROM-GD2-PARTinstead.
[Function]
destroy-image image => result
Destroys (deallocates)imagewhich has been created byCREATE-IMAGE,CREATE-IMAGE-FROM-FILE, orCREATE-IMAGE-FROM-GD2-PART.resultis alwaysNIL.
[Macro]
with-image (name width height &optional true-color) form* => results
Creates an image as withCREATE-IMAGEand executesform*with the image bound toname. The image is guaranteed to be destroyed before this macro exits.
[Macro]
with-image-from-file (name file-name &optional type) form* => results
Creates an image as withCREATE-IMAGE-FROM-FILEand executesform*with the image bound toname. The image is guaranteed to be destroyed before this macro exits.
(with-image-from-file (old "zappa.jpg")(multiple-value-bind (width height) (image-size old) (with-image (new width height) (allocate-color 0 255 0 :image new) ; green background (copy-image old new 0 0 0 0 width height :merge 50) (write-image-to-file "zappa-green.jpg" :image new :if-exists :supersede))))
[Macro]
with-image-from-gd2-part (name file-name src-x src-y width height) form* => results
Creates an image as withCREATE-IMAGE-FROM-GD2-PARTand executesform*with the image bound toname. The image is guaranteed to be destroyed before this macro exits.
[Special variable]
*default-image*
Whenever a CL-GD function or macro has an optional or keyword argument called image the default is to use*default-image*. The idea behind this is that you'll never have to provide these arguments as long as you work with one image at a time (which should be the usual case). See the example at the top of the page.
[Macro]
with-default-image (image) form* => results
This is just a convenience macro which will executeform*with*DEFAULT-IMAGE*bound toimage.
[Macro]
with-image* (width height &optional true-color) form* => results
[Macro]
with-image-from-file* (file-name &optional type) form* => results
[Macro]
with-image-from-gd2-part* (file-name src-x src-y width height) form* => results
These are essentially like their asterisk-less counterparts but bind the image to*DEFAULT-IMAGE*instead.
For the rest of this document, whenever a function expects an image as
one of its arguments you must pass a value which was created
with one of the functions or macros above. An image should be
considered an opaque object which you can pass to CL-GD functions but
should otherwise leave alone. (Internally it is a foreign pointer
wrapped in a CL-GD::IMAGE structure in order to enable
type checking.)
[Function]
write-jpeg-to-stream stream &key quality image => image
Writes imageimageto the streamstreamas a JPEG file. Ifqualityis not specified, the default IJG JPEG quality value is used. Otherwise,qualitymust be an integer in the range 0-100.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-png-to-stream stream &key compression-level image => image
Writes imageimageto the streamstreamas a PNG file. Ifcompression-levelis not specified, the default compression level at the time zlib was compiled on your system will be used. Otherwise, a compression level of 0 means 'no compression', a compression level of 1 means 'compressed, but as quickly as possible', a compression level of 9 means 'compressed as much as possible to produce the smallest possible file.'streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-wbmp-to-stream stream &key foreground image => image
Writes imageimageto the streamstreamas a WBMP (wireless bitmap) file. WBMP file support is black and white only. The color specified by theforegroundargument is the "foreground," and only pixels of this color will be set in the WBMP file.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-gd-to-stream stream &key image => image
Writes imageimageto the streamstreamas a GD file.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-gif-to-stream stream &key image => image
Writes imageimageto the streamstreamas a GIF file.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-gd2-to-stream stream &key image => image
Writes imageimageto the streamstreamas a GD2 file.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-image-to-stream stream type &key &allow-other-keys => image
Writes imageimageto the streamstream. The type of the image is determined bytypewhich must be one of the keywords:JPG,:JPEG,:GIF,:PNG,:WBMP,:GD, or:GD2. The rest of the keyword arguments are handed over to the correspondingWRITE-XXX-TO-STREAMfunction.streammust be a character stream or a binary stream of element type(UNSIGNED-BYTE 8). If STREAM is a character stream, the user of this function has to make sure the external format yields faithful output of all 8-bit characters. CL-GD knows about AllegroCL's simple streams and the bivalent streams of LispWorks 4.3 and acts accordingly, i.e. it usesWRITE-BYTEinstead ofWRITE-CHARwhenever possible.
[Function]
write-image-to-file file-name type if-exists &key &allow-other-keys => image
Writes imageimageto the file specified byfile-name(which is either a pathname or a string). Thetypeargument is interpreted as inWRITE-IMAGE-TO-STREAM. If it is not provided it will be guessed from thePATHNAME-TYPEoffile-name. Theif-existskeyword argument is given toOPEN, the rest of the keyword arguments are handed over to the correspondingWRITE-XXX-TO-STREAMfunction.
[Function]
image-width &optional image => width
Returns the width of the imageimage. The result of this function is affected byWITH-TRANSFORMATION.
[Function]
image-height &optional image => height
Returns the height of the imageimage. The result of this function is affected byWITH-TRANSFORMATION.
[Function]
image-size &optional image => width, height
Returns the width and height of the imageimageas two values. The results of this function are affected byWITH-TRANSFORMATION.
Colors are determined by specifying values for their red, green, blue, and optionally alpha components. The first three have to be integer values in the range 0-255 while the last one has to be in the range 0-127. For a palette-based image the first color you allocate will be its background color. Note that colors are allocated per image, i.e. you can't allocate a color in one image and then use it to draw something in another image.
See also the next section for some 'special colors.'
[Special variable]
*default-color*
Whenever a CL-GD function or macro has an optional or keyword argument called color the default is to use*default-color*. SeeWITH-DEFAULT-COLORbelow.
[Macro]
with-default-color (color) form* => results
This is just a convenience macro which will executeform*with*DEFAULT-COLOR*bound tocolor.
[Function]
allocate-color red green blue &key alpha errorp image => color
Finds the first available color index in the imageimagespecified, sets its RGB values to those requested (255 is the maximum for each), and returns the index of the new color table entry, or an RGBA value in the case of a true color image. In either case you can then use the returned value as a color parameter to drawing functions. When creating a new palette-based image, the first time you invoke this function you are setting the background color for that image. Ifalpha(not greater than 127) is provided, an RGBA color will always be allocated. If all+MAX-COLORS+have already been allocated this function will, depending on the value oferrorp, either raise an error or returnNIL. The default is to raise an error.
[Function]
find-color red green blue &key alpha exact hwb resolve image => color
Tries to find and/or allocate a color fromimage's color palette. Ifexactis true, the color will only be returned if it is already allocated. Ifexactis false, a color which is 'close' to the color specified byred,green, andblue(and probablyalpha) might be returned (unless there aren't any colors allocated in the image yet). Ifhwbis true, the 'closeness' will be determined by hue, whiteness, and blackness, otherwise by the Euclidian distance of the RGB values. Ifresolveis true a color (probably a new one) will always be returned, otherwise the result of this function might beNIL. Ifalpha(not greater than 127) is provided, an RGBA color (orNIL) will be returned.alpha,exact, andhwbare mutually exclusive.resolvecan't be used together withexactorhwb.
[Function]
find-color-from-image color source-image &key alpha exact hwb resolve image => color
Tries to find and/or allocate a color fromimage's color palette that corresponds tocolorinsource-image.find-color-from-imagecallsfind-colorwith the color components ofcolor. Refer tofind-colorfor a description of the keyword arguments.
[Function]
color-component color component &key image => component
Returns the specified color component ofcolor.componentcan be one of:RED,:GREEN,:BLUE, and:ALPHA.
[Function]
color-components color &key image => components
Returns the color components ofcoloras a list. The components are in the order red, green, blue, alpha.
* (defun foo ()
    (with-image* (10 10)
      (loop for i below +max-colors+ do
            ;; allocate enough colors (all gray) to fill the palette
            (allocate-color i i i))
      (format t "Number of colors allocated: ~A~%" (number-of-colors))
      (format t "Maximal number of colors: ~A~%" +max-colors+)
      (format t "Exact match for red: ~A~%" (find-color 255 0 0 :exact t))
      (format t "Red, green, and blue components of 'closest' match for red: ~A~%"
              (let ((match (find-color 255 0 0)))
                (if match
                  (list (color-component :red match)
                        (color-component :green match)
                        (color-component :blue match))))))
    (values))
FOO
* (foo)
Number of colors allocated: 256
Maximal number of colors: 256
Exact match for red: NIL
Red, green, and blue components of 'closest' match for red: (64 64 64)
[Function]
deallocate-color color &optional image => color
Marks the specified colorcoloras being available for reuse. No attempt will be made to determine whether the color index is still in use in the imageimage.
[Function]
true-color-p &optional image => result
Returns true iffimageis a true color image.
[Function]
number-of-colors &optional image => result
Returns the number of colors allocated inimageor NIL ifimageis a true color image.
[Constant]
+max-colors+
Maximum number of colors for palette-based images.
[Accessor]
transparent-color &optional image => color
(setf (transparent-color &optional image) color)
Gets and sets the transparent color ofimage. IfcolorisNILthere is no transparent color.
[Accessor]
alpha-blending-p &optional image => blending
(setf (alpha-blending-p &optional image) blending)
Gets and set whether pixels drawn onimagewill be copied literally including alpha channel information (ifblendingis false) or if their alpha channel information will determine how much of the underlying color will shine through (ifblendingis true). This is only meaningful for true color images.
[Accessor]
save-alpha-p &optional image => save
(setf (save-alpha-p &optional image) save)
Gets and sets whether PNG images will be saved with full alpha channel information.
(with-image* (200 100)(allocate-color 255 165 0) ; orange background (with-image (brush 6 6) (let* ((black (allocate-color 0 0 0 :image brush)) ; black background (red (allocate-color 255 0 0 :image brush)) (blue (allocate-color 0 0 255 :image brush))) (setf (transparent-color brush) black) ; make background transparent ;; now set the pixels in the brush (set-pixels '(2 2 2 3 3 2 3 3) :color blue :image brush) (set-pixels '(3 2 3 3 1 2 1 3 4 2 4 3 2 1 3 1 2 4 3 4) :color red :image brush) ;; then use it to draw an arc (draw-arc 100 50 180 80 180 300 ;; convert BRUSH to brush :color (make-brush brush))) (write-image-to-file "brushed-arc.png" :compression-level 7 :if-exists :supersede)))
DRAW-FREETYPE-STRING
being the only exception) will, when expecting a color, also accept other types of arguments. The
full range of allowed types which can be used for
color keyword arguments is listed below:
NIL for transparent colors. When a style is used as the
  color, the colors of the pixels are drawn successively from the
  sequence provided. If the corresponding element of the sequence is
  NIL, that pixel is not altered.
  MAKE-BRUSH for drawing lines. A
  brush is itself an image created as described
  above. When a brush is used as the color, the brush image is drawn
  in place of each pixel of the line drawn. (The brush is usually
  larger than one pixel, creating the effect of a wide paintbrush.)
  MAKE-TILE for filling regions. A
  tile is itself an image created as described
  above. When a tile is used as the color, a pixel from the tile image
  is selected in such a way as to ensure that the filled area will be
  tiled with copies of the tile image.
  CONS where the CAR is a brush and
  the CDR is a list or a vector. This is called a
  styled brush. When a styled brush is used as the color, the
  brush image is drawn at each pixel of the line, provided that the
  corresponding element of the style sequence is true.
  (Pixels are drawn successively from the style as the line is drawn,
  returning to the beginning when the available pixels in the style
  are exhausted.) Note that the semantics described here differ
  slightly from the styles described above.
  MAKE-ANTI-ALIASED for
  drawing lines. When an anti-aliased color is used, the line is drawn
  with anti-aliasing mechanisms to minimize any "jagged"
  appearance.
  
In GD itself, if you, say, change a brush after you've 'set' it with
gdImageSetBrush
but before you actually use it for drawing these changes won't be
visible, i.e. the brush is 'frozen' once it's 'set.' The same applies
to tiles and styles. CL-GD's behaviour differs in this regard,
i.e. brushes, tiles, and styles are 'set' at the very moment they're
used. This introduces a little bit of overhead but feels more 'Lisp-y'
and intuitive to me.
[Function]
make-brush image => brush
Creates a brush from the image image. Note that the new
brush is still 'linked' to image, i.e. changes you
make to image will also be visible in the
brush - the brush is just a kind of 'tagged' image.
[Function]
make-tile image => tile
Creates a tile from the image image. Note that the new
tile is still 'linked' to image, i.e. changes you
make to image will also be visible in the
tile - the tile is just a kind of 'tagged' image.
[Function]
make-anti-aliased color &optionaldo-not-blend => color'
Creates an anti-aliased color from the
color
color. do-not-blend (if provided) is the
color anti-aliased lines stand out against clearly.
(with-image* (150 50)
 255 0 0)))
    ;; white background rectangle in the middle third
    (draw-rectangle* 50 0 99 49
                    :filled t
                    :color white)
    (with-thickness (2)
      ;; just a red line
      (draw-line 5 10 145 10 :color red)
      ;; anti-aliased red line
      (draw-line 5 25 145 25 :color (make-anti-aliased red))
      ;; anti-aliased red line which should stand out against
      ;; orange background
      (draw-line 5 40 145 40 :color (make-anti-aliased red orange))))
  (write-image-to-file "anti-aliased-lines.png"
                       :compression-level 3
                       :if-exists :supersede))
 (let ((orange (allocate-color 255 165 0)) ; orange background
        (white (allocate-color 255 255 255))
        (red ( href="#allocate-color">allocate-color
  (let ((orange (allocate-color 255 165 0)) ; orange background
        (white (allocate-color 255 255 255))
        (red ( href="#allocate-color">allocate-color
 Transformations
Usually, CL-GD coordinates and dimensions (width and height) have to be integers. The origin (0,0) of an image is its upper left corner and all other points like (X,Y) have positive X and Y values. Angles are also provided as integers (in the range 0-360) meaning degrees. A transformation provides a way to change this. 
Executes form* such that all points and width/height data are
subject to a simple affine transformation defined by the keyword
parameters. The new x-axis of image will start at x1 and end at x2 and
have length width. The new y-axis of image will start at y1 and end at
y2 and have length height. In both cases it suffices to provide two of
the three values - if you provide all three they have to match. If
reverse-x is false the x-axis will be oriented as usual in Cartesian
coordinates, otherwise its direction will be reversed. The same
applies to reverse-y, of course. If radians is true angles inside of
the macro's body will be assumed to be provided in radians, otherwise in degrees. The previous transformation (if any) will be restored before this macro exits.
with-transformation macros can be nested but they always transform the original coordinates of the image, i.e. you shouldn't expect that, say, two succesive applications of reverse-x will neutralize each other. There's a little bit of overhead involved with this macro, so it is recommended to wrap it around everything you do with an image instead of calling it repeatedly. Note that transformations are always bound to one particular image.
[Macro]
without-transformations form* => results
Executes form* without any transformations applied.
 Drawing and filling
This section (and the next one about strings) finally describes how you can actually change the visual appearance of an image. You can set single pixels, draw lines or geometric figures, and fill regions. Note that the current transformation (if any) applies to the input and output of these functions.
[Function]
set-pixel x y &key color image => x, y
Sets the pixel specified byxandyto the color specified bycolor.
[Generic function]
set-pixels points &key color image => points
Sets the pixels specified bypointswhich can be a list(X1 Y1 X2 Y2 ...)or a vector#(X1 Y1 X2 Y2 ...)to the color specified bycolor.
[Function]
draw-line x1 y1 x2 y2 &key color image => x1, y1, x2, y2
Draws a line with colorcolorfrom point(x1,y1)to point(x2,y2).
[Function]
draw-rectangle rectangle &key filled color image => rectangle
Draws a rectangle with upper left corner(X1,Y1)and lower right corner(X2,Y2)whererectangleis the list(X1 Y2 X2 Y2). Iffilledis true the rectangle will be filled withcolor, otherwise it will be outlined.
[Function]
draw-rectangle* x1 y1 x2 y2 &key filled color image => x1, y1, x2, y2
Draws a rectangle with upper left corner(x1,y1)and lower right corner(x2,y2). Iffilledis true the rectangle will be filled withcolor, otherwise it will be outlined.
[Generic function]
draw-polygon vertices &key filled start end color image => vertices
Draws a polygon with the vertices (at least three) specified as a list(X1 Y1 X2 Y2 ...)or as a vector#\(X1 Y1 X2 Y2 ...). Iffilledis true the polygon will be filled with the colorcolor, otherwise it will be outlined. Ifstartand/orendare specified then only the corresponding part ofverticesis used as input.
(with-image* (100 100)(allocate-color 255 255 255) ; white background (let ((red (allocate-color 255 0 0)) (yellow (allocate-color 255 255 0)) (orange (allocate-color 255 165 0))) ;; thin black border (draw-rectangle* 0 0 99 99 :color (allocate-color 0 0 0)) ;; line thickness is five pixels (with-thickness (5) ;; triangle (draw-polygon (list 10 10 90 50 50 90) ;; styled color :color (list red red red yellow yellow yellow nil nil nil orange orange orange)) (write-image-to-file "triangle.png" :compression-level 8 :if-exists :supersede))))
[Function]
draw-filled-circle center-x center-y radius &key color image => center-x center-y radius
Draws a filled circle with center(center-x,center-y)and radiusradius.
Draws a filled ellipse with center(center-x,center-y), widthwidth, and heightheight.
(with-image* (250 150) (with-image-from-file (zappa "smallzappa.png")(setf (transparent-color) (allocate-color 255 255 255)) (draw-filled-ellipse 125 75 250 150 :color (make-tile zappa))) (write-image-to-file "zappa-ellipse.png" :if-exists :supersede))
Draws a partial ellipse centered at(center-x,center-y)with widthwidthand heightheight. The arc begins at anglestartand ends at angleend. Ifstraight-lineis true the start and end points are just connected with a straight line. Ifcenter-connectis true, they are connected to the center (which is useful to create 'pie slices' - see example at the top of the page.). Iffilledis true the arc will be filled with the colorcolor, otherwise it will be outlined.
[Function]
fill-image x y &key border color image => x, y
Floods a portion of the imageimagewith the colorcolorbeginning at point(x,y)and extending into the surrounding region. Ifborderis true it must be a color and the filling will stop at the specified border color. (You can't use 'special colors' for the border color.) Otherwise only points with the same color as the starting point will be colored. Ifcoloris a tile the tile must not have a transparent color.
Gets and sets the clipping rectangle ofimagewhererectangleshould be a list(X1 Y1 X2 Y2)describing the upper left and lower right corner of the rectangle. Once a clipping rectangle has been set, all future drawing operations onimagewill remain within the specified clipping area, until a new clipping rectangle is established. For instance, if a clipping rectangle(25 25 75 75)has been set within a 100x100 image, a diagonal line from(0,0)to(99,99)will appear only between(25,25)and(75,75). See alsoCLIPPING-RECTANGLE*andSET-CLIPPING-RECTANGLE*.
[Function]
clipping-rectangle* &optional image => x1, y1, x2, y2
Returns the clipping rectangle ofimageas four values.
[Function]
set-clipping-rectangle* x1 y1 x2 y2 &optional image => x1, y1, x2, y2
Sets the clipping rectangle ofimageas if set with(SETF (.CLIPPING-RECTANGLEIMAGE) (LIST X1 Y1 X2 Y2))
[Macro]
with-clipping-rectangle (rectangle &key image) form* => results
Executesform*with the clipping rectangle ofimageset torectanglewhich should be a list as inCLIPPING-RECTANGLE. The previous clipping rectangle is guaranteed to be restored before the macro exits.
[Macro]
with-clipping-rectangle* (x1 y1 x2 y2 &key image) form* => results
Executesform*with the clipping rectangle ofimageset as if set with(SETF (. The previous clipping rectangle is guaranteed to be restored before the macro exits.CLIPPING-RECTANGLEIMAGE) (LIST X1 Y1 X2 Y2))
(with-image* (150 150)(allocate-color 255 255 255) ; white background ;; transform such that x axis ranges from (- PI) to PI and y ;; axis ranges from -3 to 3 (with-transformation (:x1 (- pi) :width (* 2 pi) :y1 -3 :y2 3) (let ((black (allocate-color 0 0 0)) (red (allocate-color 255 0 0)) (rectangle (list (- .4 pi) 2.5 (- pi .4) -2.5))) (with-default-color (black) ;; draw axes (draw-line 0 -3 0 3 :color black) (draw-line (- pi) 0 pi 0)) ;; show clipping rectangle (styled) (draw-rectangle rectangle :color (list black black black nil black nil)) (with-clipping-rectangle (rectangle) ;; draw tangent function (loop for x from (- pi) below (* 2 pi) by (/ pi 75) do (set-pixel x (tan x) :color red))))) (write-image-to-file "clipped-tangent.png" :if-exists :supersede))
[Accessor]
current-thickness &optional image => thickness
(setf (current-thickness &optional image) thickness)
Get and sets the current thickness ofimagein pixels. This determines the width of lines drawn with the drawing functions.thicknesshas to be an integer. See alsoWITH-THICKNESS.
[Macro]
with-thickness (thickness &key image) form* => results
Executesform*with the current thickness ofimageset tothickness. The image's previous thickness is guaranteed to be restored before the macro exits.
:TINY, :SMALL, :MEDIUM, :MEDIUM-BOLD (a synonym for :MEDIUM), :LARGE, and :GIANT and used with  DRAW-STRING and DRAW-CHAR. Using these fonts will make your application portable to all platforms supported by CL-GD (and thus GD). You can also invoke the FreeType library to draw (anti-aliased) strings with arbitrary TrueType fonts, sizes, and angles. This is, however, subject to the availability and location of the corresponding fonts on your target platform.
[Special variable]
*default-font*
Whenever a CL-GD string or character function has an optional or keyword argument called font or font-name the default is to use*default-font*. SeeWITH-DEFAULT-FONTbelow.
[Macro]
with-default-font (font) form* => results
This is just a convenience macro which will executeform*with*DEFAULT-FONT*bound tofont. But note that the fonts used forDRAW-STRING/DRAW-CHARandDRAW-FREETYPE-STRINGare incompatible
[Function]
draw-character x y char &key up font color image => char
Draws the charactercharfrom fontfontin colorcolorat position(x,y). Ifupis true the character will be drawn from bottom to top (rotated 90 degrees).fontmust be one of the keywords listed above.
[Function]
draw-string x y string &key up font color image => string
Draws the stringstringin colorcolorat position(y,y). Ifupis true the string will be drawn from bottom to top (rotated 90 degrees).fontmust be one of the keywords listed above.
Draws the stringstringin colorcolorat position(x,y)using the FreeType library.font-nameis the full path (a pathname or a string) to a TrueType font file, or a font face name if theGDFONTPATHenvironment variable or FreeType'sDEFAULT_FONTPATHvariable have been set intelligently. The string may be arbitrarily scaled (point-size) and rotated (anglein radians). The direction of rotation is counter-clockwise, with 0 radians (0 degrees) at 3 o'clock and(/ PI 2)radians (90 degrees) at 12 o'clock. Note that theangleargument is purposefully not affected byWITH-TRANSFORMATION. Ifanti-aliasedif false, anti-aliasing is disabled. It is enabled by default. To output multiline text with a specific line spacing, provide a value forline-spacing, expressed as a multiple of the font height. The default is to use 1.05. The string may contain XML character entity references like "À". Ifconvert-charsis true (which is the default) characters ofstringwithCHAR-CODEgreater than 127 are converted accordingly. This of course pre-supposes that your Lisp'sCHAR-CODEfunction returns ISO/IEC 10646 (Unicode) character codes.The return value is an array containing 8 elements representing the 4 corner coordinates (lower left, lower right, upper right, upper left) of the bounding rectangle around the string that was drawn. The points are relative to the text regardless of the angle, so "upper left" means in the top left-hand corner seeing the text horizontally. Set
do-not-drawto true to get the bounding rectangle without rendering. This is a relatively cheap operation if followed by a rendering of the same string, because of the caching of the partial rendering during bounding rectangle calculation.
(with-image* (200 200);; set background (white) and make it transparent (setf (transparent-color) (allocate-color 255 255 255)) (loop for angle from 0 to (* 2 pi) by (/ pi 6) for blue downfrom 255 by 20 do (draw-freetype-string 100 100 "Common Lisp" :font-name "/usr/X11R6/lib/X11/fonts/truetype/georgia.ttf" :angle angle ;; note that ALLOCATE-COLOR won't work ;; here because the anti-aliasing uses ;; up too much colors :color (find-color 0 0 blue :resolve t))) (write-image-to-file "strings.png" :if-exists :supersede))
[Macro]
do-rows (y-var &optional image) declaration* form* => results
This macro loops through all rows (from top to bottom) in turn and executesform*for each row withy-varbound to the vertical index of the current row (starting with0). It is not affected byWITH-TRANSFORMATION.
[Local macro]
do-pixels-in-row (x-var) declaration* form* => results
This macro is only available within the body of aDO-ROWSform. It loops through all pixels (from left to right) in turn and executesform*for each pixel withx-varbound to the horizontal index of the current pixel (starting with0). It is not affected byWITH-TRANSFORMATION.
[Macro]
do-pixels (&optional image) declaration* form* => results
This is a shortcut for the previous two macros. It loops through all pixels and executesform*for each pixel. Obviously it only makes sense when used together withRAW-PIXEL.
[Accessor]
raw-pixel => pixel
(setf (raw-pixel) pixel)
This accessor is only available within the body of aDO-PIXELS-IN-ROWform (and thus also withinDO-PIXELSforms). It provides access to the "raw" pixel the loop is currently at, i.e. for true color images you access an element of theim->tpixelsarray, for palette-based images it'sim->pixels. Read the original GD documentation for details. Make sure you know what you're doing if you change these values...
* (with-image* (3 3 t) ; true-color image with 3x3 pixels
    (draw-rectangle* 0 0 2 2 :color (allocate-color 0 0 0)) ; black background
    (draw-line 0 0 2 2 :color (allocate-color 255 255 255)) ; white line
    (do-pixels ()
      ;; loop through all pixels and change those which arent't black
      (unless (zerop (raw-pixel))
        (decf (raw-pixel) #xff)))
    (do-rows (y)
      ;; loop through all rows
      (format t "Starting with row ~A~%" y)
      (do-pixels-in-row (x)
        ;; loop through all pixels in row
        (format t "  Pixel <~A,~A> has value ~X~%" x y (raw-pixel)))
      (format t "Done with row ~A~%" y)))
Starting with row 0
  Pixel <0,0> has value FFFF00 ; the line is yellow now
  Pixel <1,0> has value 0
  Pixel <2,0> has value 0
Done with row 0
Starting with row 1
  Pixel <0,1> has value 0
  Pixel <1,1> has value FFFF00
  Pixel <2,1> has value 0
Done with row 1
Starting with row 2
  Pixel <0,2> has value 0
  Pixel <1,2> has value 0
  Pixel <2,2> has value FFFF00
Done with row 2
NIL
[Accessor]
interlacedp &optional image => interlaced
(setf (interlacedp &optional image) interlaced)
Gets or sets whetherimagewill be stored in an interlaced fashion.
[Function]
differentp image1 image2 => different
Returns false if the two images won't appear different when
displayed. Otherwise the return value is a list of keywords describing
the differences between the images.
Copies (a part of) the image source into the image destination. Copies the
rectangle with the upper left corner (source-x,source-y) and size
width x height to the rectangle with the upper left corner (dest-x,dest-y).
If resample is true pixel colors will be
smoothly interpolated. If resize is true
the copied rectangle will be strechted or shrunk so that its size is
dest-width x
dest-height. If rotate is true
the image will be rotated by angle. In this
particular case dest-x and
dest-y specify the center of the copied
image rather than its upper left corner! If merge
is true then it has to be an integer in the range 0-100 and the
two images will be 'merged' by the amount specified. If
merge is 100 then the source image will simply be
copied. If instead merge-gray is true the hue of
the source image is preserved by converting the destination area to
gray pixels before merging.
The keyword arguments resample, rotate, resize, merge, and merge-gray
are mutually exclusive (with the exception of resample and
resize). angle is assumed to be specified in degrees if it's an
integer, and in radians otherwise. This function is not affected by WITH-TRANSFORMATION.
[Function]
copy-palette source destination  => destination
Copies the palette of the image source to the image destination attempting to
match the colors in the target image to the colors in the source palette.
[Function]
true-color-to-palette &key dither colors-wanted image => image
Converts the true color image image to a palette-based image using
a high-quality two-pass quantization routine. If dither is true, the
image will be dithered to approximate colors better, at the expense of
some obvious "speckling." colors-wanted can be any positive integer
up to 256 (which is the default). If the original source image
includes photographic information or anything that came out of a JPEG,
256 is strongly recommended. 100% transparency of a single transparent
color in the original true color image will be preserved. There is no
other support for preservation of alpha channel or transparency in the
destination image.
(with-image* ((+ 256 384) 384 t)
  (let ((white (allocate-color 255 255 255))
        (red (allocate-color 255 0 0))
        (green (allocate-color 0 255 0))
        (blue (allocate-color 0 0 255))
        (vertices (list 64 0 0 128 128 128))
        (image-width (image-width))
        (image-height (image-height)))
    (setf (transparent-color) white)
    (draw-rectangle* 0 0 image-width image-height :color white)
    ;; "demoin.png" is part of the GD distribution
    (with-image-from-file (in-file "demoin.png")
      (copy-image in-file *default-image*
                  0 0 32 32 192 192
                  :resize t
                  :dest-width 255
                  :dest-height 255
                  :resample t)
      (multiple-value-bind (in-width in-height)
          (image-size in-file)
        (loop for a below 360 by 45 do
              (copy-image in-file *default-image*
                          0 0
                          (+ 256 192 (* 128 (cos (* a .0174532925))))
                          (- 192 (* 128 (sin (* a .0174532925))))
                          in-width in-height
                          :rotate t
                          :angle a))
        (with-default-color (green)
          (with-thickness (4)
            (draw-line 16 16 240 16)
            (draw-line 240 16 240 240)
            (draw-line 240 240 16 240)
            (draw-line 16 240 16 16))
          (draw-polygon vertices :filled t))
        (dotimes (i 3)
          (incf (nth (* 2 i) vertices) 128))
        (draw-polygon vertices
                      :color (make-anti-aliased green)
                      :filled t)
        (with-default-color (blue)
          (draw-arc 128 128 60 20 0 720)
          (draw-arc 128 128 40 40 90 270)
          (fill-image 8 8))
        (with-image (brush 16 16 t)
          (copy-image in-file brush
                      0 0 0 0
                      in-width in-height
                      :resize t
                      :dest-width (image-width brush)
                      :dest-height (image-height brush))
          (draw-line 0 255 255 0
                     :color (cons (make-brush brush)
                                  (list nil nil nil nil nil nil nil t))))))
    (with-default-color (red)
      (draw-string 32 32 "hi" :font :giant)
      (draw-string 64 64 "hi" :font :small))
    (with-clipping-rectangle* (0 (- image-height 100) 100 image-height)
      (with-default-color ((make-anti-aliased white))
        (dotimes (i 100)
          (draw-line (random image-width)
                     (random image-height)
                     (random image-width)
                     (random image-height))))))
  (setf (interlacedp) t)
  (write-image-to-file "demoout.png"
                       :if-exists :supersede)
  (true-color-to-palette)
  (write-image-to-file "demooutp.png"
                       :if-exists :supersede))
This last example is the demo which comes with GD. The equivalent C code is here.
 
 
$Header: /usr/local/cvsrep/gd/doc/index.html,v 1.35 2004/11/26 16:05:49 edi Exp $