Graphviz 13.0.0~dev.20250210.0415
Loading...
Searching...
No Matches
gvloadimage_gdk.c
Go to the documentation of this file.
1/*************************************************************************
2 * Copyright (c) 2011 AT&T Intellectual Property
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * https://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors: Details at https://graphviz.org
9 *************************************************************************/
10
11#include "config.h"
12
13#include <assert.h>
14#include <stdbool.h>
15#include <stdio.h>
16#include <stdlib.h>
17#include <util/alloc.h>
18#include <util/agxbuf.h>
19
21#include <gvc/gvio.h>
22
23#ifdef HAVE_PANGOCAIRO
24#include <cairo.h>
25#include <gdk-pixbuf/gdk-pixbuf.h>
26#include <gdk/gdk.h>
27
28enum {
29 FORMAT_BMP_CAIRO,
30 FORMAT_JPEG_CAIRO,
32 FORMAT_ICO_CAIRO,
33};
34
35static void gdk_set_mimedata_from_file (cairo_surface_t *image, const char *mime_type, const char *file)
36{
37 FILE *fp;
38 unsigned char *data = NULL;
39 long len;
40 const char *id_prefix = "gvloadimage_gdk-";
41
42 fp = fopen (file, "rb");
43 if (fp == NULL)
44 return;
45 fseek (fp, 0, SEEK_END);
46 len = ftell(fp);
47 rewind(fp);
48 if (len > 0)
49 data = malloc ((size_t)len);
50 if (data) {
51 if (fread(data, (size_t)len, 1, fp) != 1) {
52 free (data);
53 data = NULL;
54 }
55 }
56 fclose(fp);
57
58 if (data) {
59 cairo_surface_set_mime_data (image, mime_type, data, (unsigned long)len, free, data);
60 agxbuf id = {0};
61 agxbprint(&id, "%s%s", id_prefix, file);
62 char *unique_id = agxbdisown(&id);
63 cairo_surface_set_mime_data(image, CAIRO_MIME_TYPE_UNIQUE_ID,
64 (unsigned char*)unique_id,
65 strlen(unique_id), free, unique_id);
66 }
67}
68
69static void gdk_set_mimedata(cairo_surface_t *image, usershape_t *us)
70{
71 switch (us->type) {
72 case FT_PNG:
73 gdk_set_mimedata_from_file (image, CAIRO_MIME_TYPE_PNG, us->name);
74 break;
75 case FT_JPEG:
76 gdk_set_mimedata_from_file (image, CAIRO_MIME_TYPE_JPEG, us->name);
77 break;
78 default:
79 break;
80 }
81}
82
84typedef struct {
85 cairo_surface_t *surface;
86 GdkPixbuf *origin;
87} data_t;
88
89static void gdk_freeimage(usershape_t *us)
90{
91 assert(us->data != NULL);
92 data_t *const data = us->data;
93 const bool is_last = cairo_surface_get_reference_count(data->surface) == 1;
94 cairo_surface_destroy(data->surface);
95 if (is_last) {
96 free(data->origin);
97 free(data);
98 }
99}
100
101static cairo_surface_t* gdk_loadimage(GVJ_t * job, usershape_t *us)
102{
103 cairo_t *cr = job->context; /* target context */
104 GdkPixbuf *image = NULL;
105 cairo_surface_t *cairo_image = NULL;
106 cairo_pattern_t *pattern;
107
108 assert(job);
109 assert(us);
110 assert(us->name);
111
112 if (us->data) {
113 if (us->datafree == gdk_freeimage) {
114 // use cached data
115 data_t *const data = us->data;
116 cairo_image = cairo_surface_reference(data->surface);
117 } else {
118 us->datafree(us); /* free incompatible cache data */
119 us->datafree = NULL;
120 us->data = NULL;
121 }
122 }
123 if (!cairo_image) { /* read file into cache */
125 return NULL;
126 switch (us->type) {
127 case FT_PNG:
128 case FT_JPEG:
129 case FT_BMP:
130 case FT_ICO:
131 case FT_TIFF:
132 // FIXME - should be using a stream reader
133 image = gdk_pixbuf_new_from_file(us->name, NULL);
134 break;
135 default:
136 image = NULL;
137 }
138 if (image) {
139 cairo_save (cr);
140 gdk_cairo_set_source_pixbuf (cr, image, 0, 0);
141 pattern = cairo_get_source (cr);
142 assert(cairo_pattern_get_type (pattern) == CAIRO_PATTERN_TYPE_SURFACE);
143 cairo_pattern_get_surface (pattern, &cairo_image);
144 cairo_image = cairo_surface_reference (cairo_image);
145 cairo_restore (cr);
146 gdk_set_mimedata (cairo_image, us);
147 data_t *const data = gv_alloc(sizeof(data_t));
148 *data = (data_t){.surface = cairo_surface_reference(cairo_image),
149 .origin = image};
150 us->data = data;
151 us->datafree = gdk_freeimage;
152 }
154 }
155 return cairo_image;
156}
157
158static void gdk_loadimage_cairo(GVJ_t * job, usershape_t *us, boxf b, bool filled)
159{
160 (void)filled;
161
162 cairo_t *cr = job->context; /* target context */
163 cairo_surface_t *image;
164
165 image = gdk_loadimage(job, us);
166 if (image) {
167 cairo_save(cr);
168 cairo_translate(cr, b.LL.x, -b.UR.y);
169 cairo_scale(cr, (b.UR.x - b.LL.x) / us->w, (b.UR.y - b.LL.y) / us->h);
170 cairo_set_source_surface (cr, image, 0, 0);
171 cairo_paint (cr);
172 cairo_restore(cr);
173 cairo_surface_destroy (image);
174 }
175}
176
177static gvloadimage_engine_t engine_gdk = {
178 gdk_loadimage_cairo
179};
180
181#endif
182
184#ifdef HAVE_PANGOCAIRO
185 {FORMAT_BMP_CAIRO, "bmp:cairo", 1, &engine_gdk, NULL},
186 {FORMAT_JPEG_CAIRO, "jpe:cairo", 2, &engine_gdk, NULL},
187 {FORMAT_JPEG_CAIRO, "jpg:cairo", 2, &engine_gdk, NULL},
188 {FORMAT_JPEG_CAIRO, "jpeg:cairo", 2, &engine_gdk, NULL},
189 {FORMAT_PNG_CAIRO, "png:cairo", -1, &engine_gdk, NULL},
190 {FORMAT_ICO_CAIRO, "ico:cairo", 1, &engine_gdk, NULL},
191#endif
192 {0, NULL, 0, NULL, NULL}
193};
static int agxbprint(agxbuf *xb, const char *fmt,...)
Printf-style output to an agxbuf.
Definition agxbuf.h:234
static char * agxbdisown(agxbuf *xb)
Definition agxbuf.h:327
Memory allocation wrappers that exit on failure.
static void * gv_alloc(size_t size)
Definition alloc.h:47
static double len(glCompPoint p)
Definition glutils.c:150
void * malloc(YYSIZE_T)
void free(void *)
node NULL
Definition grammar.y:163
void gvusershape_file_release(usershape_t *us)
bool gvusershape_file_access(usershape_t *us)
static const char * mime_type(imagetype_t file_type)
@ FORMAT_PNG_CAIRO
gvplugin_installed_t gvloadimage_gdk_types[]
T_cell image
Definition htmlparse.y:340
void * context
Definition gvcjob.h:295
Definition geom.h:41
pointf UR
Definition geom.h:41
pointf LL
Definition geom.h:41
Definition legal.c:50
ingroup plugin_api
Definition gvplugin.h:35
double x
Definition geom.h:29
double y
Definition geom.h:29
const char * name
Definition usershape.h:54
void(* datafree)(usershape_t *us)
Definition usershape.h:65
void * data
Definition usershape.h:63
imagetype_t type
Definition usershape.h:59
double h
Definition usershape.h:61
double w
Definition usershape.h:61
@ FT_BMP
Definition usershape.h:25
@ FT_ICO
Definition usershape.h:27
@ FT_TIFF
Definition usershape.h:27
@ FT_JPEG
Definition usershape.h:25
@ FT_PNG
Definition usershape.h:25