1 /* Pixbufs
2 *
3 * A GdkPixbuf represents an image, normally in RGB or RGBA format.
4 * Pixbufs are normally used to load files from disk and perform
5 * image scaling.
6 *
7 * This demo is not all that educational, but looks cool. It was written
8 * by Extreme Pixbuf Hacker Federico Mena Quintero. It also shows
9 * off how to use GtkDrawingArea to do a simple animation.
10 *
11 * Look at the Image demo for additional pixbuf usage examples.
12 *
13 */
14
15 #include <stdlib.h>
16 #include <gtk/gtk.h>
17 #include <math.h>
18
19 #include "demo-common.h"
20
21 #define FRAME_DELAY 50
22
23 #define BACKGROUND_NAME "background.jpg"
24
25 static const char *image_names[] = {
26 "apple-red.png",
27 "gnome-applets.png",
28 "gnome-calendar.png",
29 "gnome-foot.png",
30 "gnome-gmush.png",
31 "gnome-gimp.png",
32 "gnome-gsame.png",
33 "gnu-keys.png"
34 };
35
36 #define N_IMAGES G_N_ELEMENTS (image_names)
37
38 /* demo window */
39 static GtkWidget *window = NULL;
40
41 /* Current frame */
42 static GdkPixbuf *frame;
43
44 /* Background image */
45 static GdkPixbuf *background;
46 static gint back_width, back_height;
47
48 /* Images */
49 static GdkPixbuf *images[N_IMAGES];
50
51 /* Widgets */
52 static GtkWidget *da;
53
54 /* Loads the images for the demo and returns whether the operation succeeded */
55 static gboolean
load_pixbufs(GError ** error)56 load_pixbufs (GError **error)
57 {
58 gint i;
59 char *filename;
60
61 if (background)
62 return TRUE; /* already loaded earlier */
63
64 /* demo_find_file() looks in the current directory first,
65 * so you can run gtk-demo without installing GTK, then looks
66 * in the location where the file is installed.
67 */
68 filename = demo_find_file (BACKGROUND_NAME, error);
69 if (!filename)
70 return FALSE; /* note that "error" was filled in and returned */
71
72 background = gdk_pixbuf_new_from_file (filename, error);
73 g_free (filename);
74
75 if (!background)
76 return FALSE; /* Note that "error" was filled with a GError */
77
78 back_width = gdk_pixbuf_get_width (background);
79 back_height = gdk_pixbuf_get_height (background);
80
81 for (i = 0; i < N_IMAGES; i++)
82 {
83 filename = demo_find_file (image_names[i], error);
84 if (!filename)
85 return FALSE; /* Note that "error" was filled with a GError */
86
87 images[i] = gdk_pixbuf_new_from_file (filename, error);
88 g_free (filename);
89
90 if (!images[i])
91 return FALSE; /* Note that "error" was filled with a GError */
92 }
93
94 return TRUE;
95 }
96
97 /* Expose callback for the drawing area */
98 static gint
expose_cb(GtkWidget * widget,GdkEventExpose * event,gpointer data)99 expose_cb (GtkWidget *widget,
100 GdkEventExpose *event,
101 gpointer data)
102 {
103 cairo_t *cr;
104
105 cr = gdk_cairo_create (event->window);
106
107 gdk_cairo_set_source_pixbuf (cr, frame, 0, 0);
108 gdk_cairo_rectangle (cr, &event->area);
109 cairo_fill (cr);
110
111 cairo_destroy (cr);
112
113 return TRUE;
114 }
115
116 #define CYCLE_LEN 60
117
118 static int frame_num;
119
120 /* Timeout handler to regenerate the frame */
121 static gint
timeout(gpointer data)122 timeout (gpointer data)
123 {
124 double f;
125 int i;
126 double xmid, ymid;
127 double radius;
128
129 gdk_pixbuf_copy_area (background, 0, 0, back_width, back_height,
130 frame, 0, 0);
131
132 f = (double) (frame_num % CYCLE_LEN) / CYCLE_LEN;
133
134 xmid = back_width / 2.0;
135 ymid = back_height / 2.0;
136
137 radius = MIN (xmid, ymid) / 2.0;
138
139 for (i = 0; i < N_IMAGES; i++)
140 {
141 double ang;
142 int xpos, ypos;
143 int iw, ih;
144 double r;
145 GdkRectangle r1, r2, dest;
146 double k;
147
148 ang = 2.0 * G_PI * (double) i / N_IMAGES - f * 2.0 * G_PI;
149
150 iw = gdk_pixbuf_get_width (images[i]);
151 ih = gdk_pixbuf_get_height (images[i]);
152
153 r = radius + (radius / 3.0) * sin (f * 2.0 * G_PI);
154
155 xpos = floor (xmid + r * cos (ang) - iw / 2.0 + 0.5);
156 ypos = floor (ymid + r * sin (ang) - ih / 2.0 + 0.5);
157
158 k = (i & 1) ? sin (f * 2.0 * G_PI) : cos (f * 2.0 * G_PI);
159 k = 2.0 * k * k;
160 k = MAX (0.25, k);
161
162 r1.x = xpos;
163 r1.y = ypos;
164 r1.width = iw * k;
165 r1.height = ih * k;
166
167 r2.x = 0;
168 r2.y = 0;
169 r2.width = back_width;
170 r2.height = back_height;
171
172 if (gdk_rectangle_intersect (&r1, &r2, &dest))
173 gdk_pixbuf_composite (images[i],
174 frame,
175 dest.x, dest.y,
176 dest.width, dest.height,
177 xpos, ypos,
178 k, k,
179 GDK_INTERP_NEAREST,
180 ((i & 1)
181 ? MAX (127, fabs (255 * sin (f * 2.0 * G_PI)))
182 : MAX (127, fabs (255 * cos (f * 2.0 * G_PI)))));
183 }
184
185 GDK_THREADS_ENTER ();
186 gtk_widget_queue_draw (da);
187 GDK_THREADS_LEAVE ();
188
189 frame_num++;
190 return TRUE;
191 }
192
193 static guint timeout_id;
194
195 static void
cleanup_callback(GtkObject * object,gpointer data)196 cleanup_callback (GtkObject *object,
197 gpointer data)
198 {
199 g_source_remove (timeout_id);
200 timeout_id = 0;
201 }
202
203 GtkWidget *
do_pixbufs(GtkWidget * do_widget)204 do_pixbufs (GtkWidget *do_widget)
205 {
206 if (!window)
207 {
208 GError *error;
209
210 window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
211 gtk_window_set_screen (GTK_WINDOW (window),
212 gtk_widget_get_screen (do_widget));
213 gtk_window_set_title (GTK_WINDOW (window), "Pixbufs");
214 gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
215
216 g_signal_connect (window, "destroy",
217 G_CALLBACK (gtk_widget_destroyed), &window);
218 g_signal_connect (window, "destroy",
219 G_CALLBACK (cleanup_callback), NULL);
220
221
222 error = NULL;
223 if (!load_pixbufs (&error))
224 {
225 GtkWidget *dialog;
226
227 dialog = gtk_message_dialog_new (GTK_WINDOW (window),
228 GTK_DIALOG_DESTROY_WITH_PARENT,
229 GTK_MESSAGE_ERROR,
230 GTK_BUTTONS_CLOSE,
231 "Failed to load an image: %s",
232 error->message);
233
234 g_error_free (error);
235
236 g_signal_connect (dialog, "response",
237 G_CALLBACK (gtk_widget_destroy), NULL);
238
239 gtk_widget_show (dialog);
240 }
241 else
242 {
243 gtk_widget_set_size_request (window, back_width, back_height);
244
245 frame = gdk_pixbuf_new (GDK_COLORSPACE_RGB, FALSE, 8, back_width, back_height);
246
247 da = gtk_drawing_area_new ();
248
249 g_signal_connect (da, "expose-event",
250 G_CALLBACK (expose_cb), NULL);
251
252 gtk_container_add (GTK_CONTAINER (window), da);
253
254 timeout_id = g_timeout_add (FRAME_DELAY, timeout, NULL);
255 }
256 }
257
258 if (!gtk_widget_get_visible (window))
259 {
260 gtk_widget_show_all (window);
261 }
262 else
263 {
264 gtk_widget_destroy (window);
265 window = NULL;
266 g_object_unref (frame);
267 }
268
269 return window;
270 }
271