| 1 | /* $Id$ $Revision$ */ |
| 2 | /* vim:set shiftwidth=4 ts=8: */ |
| 3 | |
| 4 | /************************************************************************* |
| 5 | * Copyright (c) 2011 AT&T Intellectual Property |
| 6 | * All rights reserved. This program and the accompanying materials |
| 7 | * are made available under the terms of the Eclipse Public License v1.0 |
| 8 | * which accompanies this distribution, and is available at |
| 9 | * http://www.eclipse.org/legal/epl-v10.html |
| 10 | * |
| 11 | * Contributors: See CVS logs. Details at http://www.graphviz.org/ |
| 12 | *************************************************************************/ |
| 13 | |
| 14 | #include "config.h" |
| 15 | |
| 16 | #include <stdlib.h> |
| 17 | #include <stddef.h> |
| 18 | #include <string.h> |
| 19 | |
| 20 | #ifdef HAVE_PANGOCAIRO |
| 21 | #include <cairo.h> |
| 22 | #endif |
| 23 | |
| 24 | #include "gvplugin_loadimage.h" |
| 25 | #include "gvio.h" |
| 26 | #include "gd.h" |
| 27 | |
| 28 | typedef enum { |
| 29 | FORMAT_PNG_GD, FORMAT_GIF_GD, FORMAT_JPG_GD, FORMAT_GD_GD, FORMAT_GD2_GD, FORMAT_XPM_GD, FORMAT_WBMP_GD, FORMAT_XBM_GD, |
| 30 | FORMAT_PNG_PS, FORMAT_GIF_PS, FORMAT_JPG_PS, FORMAT_GD_PS, FORMAT_GD2_PS, FORMAT_XPM_PS, FORMAT_WBMP_PS, FORMAT_XBM_PS, |
| 31 | FORMAT_PNG_CAIRO, FORMAT_GIF_CAIRO, FORMAT_JPG_CAIRO, FORMAT_GD_CAIRO, FORMAT_GD2_CAIRO, FORMAT_XPM_CAIRO, FORMAT_WBMP_CAIRO, FORMAT_XBM_CAIRO, |
| 32 | } format_type; |
| 33 | |
| 34 | |
| 35 | static void gd_freeimage(usershape_t *us) |
| 36 | { |
| 37 | gdImageDestroy((gdImagePtr)us->data); |
| 38 | } |
| 39 | |
| 40 | static gdImagePtr gd_loadimage(GVJ_t * job, usershape_t *us) |
| 41 | { |
| 42 | assert(job); |
| 43 | assert(us); |
| 44 | assert(us->name); |
| 45 | |
| 46 | if (us->data) { |
| 47 | if (us->datafree != gd_freeimage) { |
| 48 | us->datafree(us); /* free incompatible cache data */ |
| 49 | us->data = NULL; |
| 50 | us->datafree = NULL; |
| 51 | } |
| 52 | } |
| 53 | if (!us->data) { /* read file into cache */ |
| 54 | if (!gvusershape_file_access(us)) |
| 55 | return NULL; |
| 56 | switch (us->type) { |
| 57 | #if 0 |
| 58 | case FT_GD: |
| 59 | im = gdImageCreateFromGd(us->f); |
| 60 | break; |
| 61 | case FT_GD2: |
| 62 | im = gdImageCreateFromGd2(us->f); |
| 63 | break; |
| 64 | #endif |
| 65 | #ifdef HAVE_GD_PNG |
| 66 | case FT_PNG: |
| 67 | us->data = (void*)gdImageCreateFromPng(us->f); |
| 68 | break; |
| 69 | #endif |
| 70 | #ifdef HAVE_GD_GIF |
| 71 | case FT_GIF: |
| 72 | us->data = (void*)gdImageCreateFromGif(us->f); |
| 73 | break; |
| 74 | #endif |
| 75 | #ifdef HAVE_GD_JPEG |
| 76 | case FT_JPEG: |
| 77 | us->data = (void*)gdImageCreateFromJpeg(us->f); |
| 78 | break; |
| 79 | #endif |
| 80 | #if 0 |
| 81 | #ifdef HAVE_GD_XPM |
| 82 | case FT_XPM: |
| 83 | us->data = (void*)gdImageCreateFromXpm(us->f); |
| 84 | break; |
| 85 | #endif |
| 86 | #ifdef HAVE_GD_WBMP |
| 87 | case FT_WBMP: |
| 88 | us->data = (void*)gdImageCreateFromWbmp(us->f); |
| 89 | break; |
| 90 | #endif |
| 91 | #endif |
| 92 | default: |
| 93 | break; |
| 94 | } |
| 95 | if (us->data) |
| 96 | us->datafree = gd_freeimage; |
| 97 | |
| 98 | gvusershape_file_release(us); |
| 99 | } |
| 100 | return (gdImagePtr)(us->data); |
| 101 | } |
| 102 | |
| 103 | static gdImagePtr gd_rotateimage(gdImagePtr im, int rotation) |
| 104 | { |
| 105 | gdImagePtr im2 = gdImageCreate(im->sy, im->sx); |
| 106 | |
| 107 | gdImageCopyRotated(im2, im, im2->sx / 2., im2->sy / 2., |
| 108 | 0, 0, im->sx, im->sy, rotation); |
| 109 | gdImageDestroy(im); |
| 110 | return im2; |
| 111 | } |
| 112 | |
| 113 | static void gd_loadimage_gd(GVJ_t * job, usershape_t *us, boxf b, boolean filled) |
| 114 | { |
| 115 | gdImagePtr im2, im = (gdImagePtr) job->context; |
| 116 | |
| 117 | if ((im2 = gd_loadimage(job, us))) { |
| 118 | if (job->rotation) |
| 119 | im2 = gd_rotateimage(im2, job->rotation); |
| 120 | gdImageCopyResized(im, im2, ROUND(b.LL.x), ROUND(b.LL.y), 0, 0, |
| 121 | ROUND(b.UR.x - b.LL.x), ROUND(b.UR.y - b.LL.y), im2->sx, im2->sy); |
| 122 | } |
| 123 | } |
| 124 | |
| 125 | #ifdef HAVE_PANGOCAIRO |
| 126 | static void gd_loadimage_cairo(GVJ_t * job, usershape_t *us, boxf b, boolean filled) |
| 127 | { |
| 128 | cairo_t *cr = (cairo_t *) job->context; /* target context */ |
| 129 | unsigned int x, y, stride, width, height, px; |
| 130 | unsigned char *data; |
| 131 | cairo_surface_t *surface; /* source surface */ |
| 132 | gdImagePtr im; |
| 133 | |
| 134 | if ((im = gd_loadimage(job, us))) { |
| 135 | width = im->sx; |
| 136 | height = im->sy; |
| 137 | // cairo_format_stride_for_width() not available prior to cairo-1.6.4 or so (fc9) |
| 138 | //stride = cairo_format_stride_for_width (CAIRO_FORMAT_ARGB32, width); |
| 139 | stride = width*4; |
| 140 | data = malloc (stride * height); |
| 141 | surface = cairo_image_surface_create_for_data (data, CAIRO_FORMAT_ARGB32, |
| 142 | width, height, stride); |
| 143 | |
| 144 | if (im->trueColor) { |
| 145 | if (im->saveAlphaFlag) { |
| 146 | for (y = 0; y < height; y++) { |
| 147 | for (x = 0; x < width; x++) { |
| 148 | px = gdImageTrueColorPixel(im, x, y); |
| 149 | *data++ = gdTrueColorGetBlue(px); |
| 150 | *data++ = gdTrueColorGetGreen(px); |
| 151 | *data++ = gdTrueColorGetRed(px); |
| 152 | *data++ = (0x7F-gdTrueColorGetAlpha(px)) << 1; |
| 153 | } |
| 154 | } |
| 155 | } |
| 156 | else { |
| 157 | for (y = 0; y < height; y++) { |
| 158 | for (x = 0; x < width; x++) { |
| 159 | px = gdImageTrueColorPixel(im, x, y); |
| 160 | *data++ = gdTrueColorGetBlue(px); |
| 161 | *data++ = gdTrueColorGetGreen(px); |
| 162 | *data++ = gdTrueColorGetRed(px); |
| 163 | *data++ = 0xFF; |
| 164 | } |
| 165 | } |
| 166 | } |
| 167 | } |
| 168 | else { |
| 169 | for (y = 0; y < height; y++) { |
| 170 | for (x = 0; x < width; x++) { |
| 171 | px = gdImagePalettePixel(im, x, y); |
| 172 | *data++ = im->blue[px]; |
| 173 | *data++ = im->green[px]; |
| 174 | *data++ = im->red[px]; |
| 175 | *data++ = (px==im->transparent)?0x00:0xff; |
| 176 | } |
| 177 | } |
| 178 | } |
| 179 | |
| 180 | cairo_save(cr); |
| 181 | cairo_translate(cr, b.LL.x, -b.UR.y); |
| 182 | cairo_scale(cr, (b.UR.x - b.LL.x)/(us->w), (b.UR.y - b.LL.y)/(us->h)); |
| 183 | cairo_set_source_surface (cr, surface, 0, 0); |
| 184 | cairo_paint (cr); |
| 185 | cairo_restore(cr); |
| 186 | |
| 187 | cairo_surface_destroy(surface); |
| 188 | } |
| 189 | } |
| 190 | #endif |
| 191 | |
| 192 | static void gd_loadimage_ps(GVJ_t * job, usershape_t *us, boxf b, boolean filled) |
| 193 | { |
| 194 | gdImagePtr im = NULL; |
| 195 | int X, Y, x, y, px; |
| 196 | |
| 197 | if ((im = gd_loadimage(job, us))) { |
| 198 | X = im->sx; |
| 199 | Y = im->sy; |
| 200 | |
| 201 | gvputs(job, "save\n" ); |
| 202 | |
| 203 | /* define image data as string array (one per raster line) */ |
| 204 | gvputs(job, "/myctr 0 def\n" ); |
| 205 | gvputs(job, "/myarray [\n" ); |
| 206 | if (im->trueColor) { |
| 207 | for (y = 0; y < Y; y++) { |
| 208 | gvputs(job, "<" ); |
| 209 | for (x = 0; x < X; x++) { |
| 210 | px = gdImageTrueColorPixel(im, x, y); |
| 211 | gvprintf(job, "%02x%02x%02x" , |
| 212 | gdTrueColorGetRed(px), |
| 213 | gdTrueColorGetGreen(px), |
| 214 | gdTrueColorGetBlue(px)); |
| 215 | } |
| 216 | gvputs(job, ">\n" ); |
| 217 | } |
| 218 | } |
| 219 | else { |
| 220 | for (y = 0; y < Y; y++) { |
| 221 | gvputs(job, "<" ); |
| 222 | for (x = 0; x < X; x++) { |
| 223 | px = gdImagePalettePixel(im, x, y); |
| 224 | gvprintf(job, "%02x%02x%02x" , |
| 225 | im->red[px], |
| 226 | im->green[px], |
| 227 | im->blue[px]); |
| 228 | } |
| 229 | gvputs(job, ">\n" ); |
| 230 | } |
| 231 | } |
| 232 | gvputs(job, "] def\n" ); |
| 233 | gvputs(job,"/myproc { myarray myctr get /myctr myctr 1 add def } def\n" ); |
| 234 | |
| 235 | /* this sets the position of the image */ |
| 236 | gvprintf(job, "%g %g translate\n" , |
| 237 | (b.LL.x + (b.UR.x - b.LL.x) * (1. - (job->dpi.x) / 96.) / 2.), |
| 238 | (b.LL.y + (b.UR.y - b.LL.y) * (1. - (job->dpi.y) / 96.) / 2.)); |
| 239 | |
| 240 | /* this sets the rendered size to fit the box */ |
| 241 | gvprintf(job,"%g %g scale\n" , |
| 242 | ((b.UR.x - b.LL.x) * (job->dpi.x) / 96.), |
| 243 | ((b.UR.y - b.LL.y) * (job->dpi.y) / 96.)); |
| 244 | |
| 245 | /* xsize ysize bits-per-sample [matrix] */ |
| 246 | gvprintf(job, "%d %d 8 [%d 0 0 %d 0 %d]\n" , X, Y, X, -Y, Y); |
| 247 | |
| 248 | gvputs(job, "{myproc} false 3 colorimage\n" ); |
| 249 | |
| 250 | gvputs(job, "restore\n" ); |
| 251 | } |
| 252 | } |
| 253 | |
| 254 | static gvloadimage_engine_t engine = { |
| 255 | gd_loadimage_gd |
| 256 | }; |
| 257 | |
| 258 | static gvloadimage_engine_t engine_ps = { |
| 259 | gd_loadimage_ps |
| 260 | }; |
| 261 | |
| 262 | #ifdef HAVE_PANGOCAIRO |
| 263 | static gvloadimage_engine_t engine_cairo = { |
| 264 | gd_loadimage_cairo |
| 265 | }; |
| 266 | #endif |
| 267 | |
| 268 | gvplugin_installed_t gvloadimage_gd_types[] = { |
| 269 | {FORMAT_GD_GD, "gd:gd" , 1, &engine, NULL}, |
| 270 | {FORMAT_GD2_GD, "gd2:gd" , 1, &engine, NULL}, |
| 271 | #ifdef HAVE_GD_GIF |
| 272 | {FORMAT_GIF_GD, "gif:gd" , 1, &engine, NULL}, |
| 273 | #endif |
| 274 | #ifdef HAVE_GD_JPEG |
| 275 | {FORMAT_JPG_GD, "jpeg:gd" , 1, &engine, NULL}, |
| 276 | {FORMAT_JPG_GD, "jpe:gd" , 1, &engine, NULL}, |
| 277 | {FORMAT_JPG_GD, "jpg:gd" , 1, &engine, NULL}, |
| 278 | #endif |
| 279 | #ifdef HAVE_GD_PNG |
| 280 | {FORMAT_PNG_GD, "png:gd" , 1, &engine, NULL}, |
| 281 | #endif |
| 282 | #ifdef HAVE_GD_WBMP |
| 283 | {FORMAT_WBMP_GD, "wbmp:gd" , 1, &engine, NULL}, |
| 284 | #endif |
| 285 | #ifdef HAVE_GD_XPM |
| 286 | {FORMAT_XBM_GD, "xbm:gd" , 1, &engine, NULL}, |
| 287 | #endif |
| 288 | |
| 289 | {FORMAT_GD_PS, "gd:ps" , 1, &engine_ps, NULL}, |
| 290 | {FORMAT_GD_PS, "gd:lasi" , 1, &engine_ps, NULL}, |
| 291 | {FORMAT_GD2_PS, "gd2:ps" , 1, &engine_ps, NULL}, |
| 292 | {FORMAT_GD2_PS, "gd2:lasi" , 1, &engine_ps, NULL}, |
| 293 | #ifdef HAVE_GD_GIF |
| 294 | {FORMAT_GIF_PS, "gif:ps" , 1, &engine_ps, NULL}, |
| 295 | {FORMAT_GIF_PS, "gif:lasi" , 1, &engine_ps, NULL}, |
| 296 | #endif |
| 297 | #ifdef HAVE_GD_JPEG |
| 298 | {FORMAT_JPG_PS, "jpeg:ps" , 1, &engine_ps, NULL}, |
| 299 | {FORMAT_JPG_PS, "jpg:ps" , 1, &engine_ps, NULL}, |
| 300 | {FORMAT_JPG_PS, "jpe:ps" , 1, &engine_ps, NULL}, |
| 301 | {FORMAT_JPG_PS, "jpeg:lasi" , 1, &engine_ps, NULL}, |
| 302 | {FORMAT_JPG_PS, "jpg:lasi" , 1, &engine_ps, NULL}, |
| 303 | {FORMAT_JPG_PS, "jpe:lasi" , 1, &engine_ps, NULL}, |
| 304 | #endif |
| 305 | #ifdef HAVE_GD_PNG |
| 306 | {FORMAT_PNG_PS, "png:ps" , 1, &engine_ps, NULL}, |
| 307 | {FORMAT_PNG_PS, "png:lasi" , 1, &engine_ps, NULL}, |
| 308 | #endif |
| 309 | #ifdef HAVE_GD_WBMP |
| 310 | {FORMAT_WBMP_PS, "wbmp:ps" , 1, &engine_ps, NULL}, |
| 311 | {FORMAT_WBMP_PS, "wbmp:lasi" , 1, &engine_ps, NULL}, |
| 312 | #endif |
| 313 | #ifdef HAVE_GD_XPM |
| 314 | {FORMAT_XBM_PS, "xbm:ps" , 1, &engine_ps, NULL}, |
| 315 | {FORMAT_XBM_PS, "xbm:lasi" , 1, &engine_ps, NULL}, |
| 316 | #endif |
| 317 | |
| 318 | #ifdef HAVE_PANGOCAIRO |
| 319 | {FORMAT_GD_CAIRO, "gd:cairo" , 1, &engine_cairo, NULL}, |
| 320 | {FORMAT_GD2_CAIRO, "gd2:cairo" , 1, &engine_cairo, NULL}, |
| 321 | #ifdef HAVE_GD_GIF |
| 322 | {FORMAT_GIF_CAIRO, "gif:cairo" , 1, &engine_cairo, NULL}, |
| 323 | #endif |
| 324 | #ifdef HAVE_GD_JPEG |
| 325 | {FORMAT_JPG_CAIRO, "jpeg:cairo" , 1, &engine_cairo, NULL}, |
| 326 | {FORMAT_JPG_CAIRO, "jpg:cairo" , 1, &engine_cairo, NULL}, |
| 327 | {FORMAT_JPG_CAIRO, "jpe:cairo" , 1, &engine_cairo, NULL}, |
| 328 | #endif |
| 329 | #ifdef HAVE_GD_PNG |
| 330 | {FORMAT_PNG_CAIRO, "png:cairo" , -1, &engine_cairo, NULL}, |
| 331 | #endif |
| 332 | #ifdef HAVE_GD_WBMP |
| 333 | {FORMAT_WBMP_CAIRO, "wbmp:cairo" , 1, &engine_cairo, NULL}, |
| 334 | #endif |
| 335 | #ifdef HAVE_GD_XPM |
| 336 | {FORMAT_XBM_CAIRO, "xbm:cairo" , 1, &engine_cairo, NULL}, |
| 337 | #endif |
| 338 | #endif /* HAVE_PANGOCAIRO */ |
| 339 | {0, NULL, 0, NULL, NULL} |
| 340 | }; |
| 341 | |