]> Pileus Git - ~andy/gtk/blob - demos/gtk-demo/pixbufs.c
add default icon
[~andy/gtk] / demos / gtk-demo / pixbufs.c
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 <config.h>
16 #include <stdlib.h>
17 #include <gtk/gtk.h>
18 #include <math.h>
19
20 #define FRAME_DELAY 50
21
22 #define RELATIVE_BACKGROUND_NAME "background.jpg"
23 #define INSTALLED_BACKGROUND_NAME DEMOCODEDIR"/background.jpg"
24
25 static const char *relative_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 static const char *installed_image_names[] = {
37   DEMOCODEDIR"/apple-red.png",
38   DEMOCODEDIR"/gnome-applets.png",
39   DEMOCODEDIR"/gnome-calendar.png",
40   DEMOCODEDIR"/gnome-foot.png",
41   DEMOCODEDIR"/gnome-gmush.png",
42   DEMOCODEDIR"/gnome-gimp.png",
43   DEMOCODEDIR"/gnome-gsame.png",
44   DEMOCODEDIR"/gnu-keys.png"
45 };
46
47 #define N_IMAGES G_N_ELEMENTS (relative_image_names)
48
49 /* demo window */
50 static GtkWidget *window = NULL;
51
52 /* Current frame */
53 static GdkPixbuf *frame;
54
55 /* Background image */
56 static GdkPixbuf *background;
57 static gint back_width, back_height;
58
59 /* Images */
60 static GdkPixbuf *images[N_IMAGES];
61
62 /* Widgets */
63 static GtkWidget *da;
64
65 /* Loads the images for the demo and returns whether the operation succeeded */
66 static gboolean
67 load_pixbufs (GError **error)
68 {
69   gint i;
70   const gchar **image_names;
71
72   if (background)
73     return TRUE; /* already loaded earlier */
74
75   background = gdk_pixbuf_new_from_file (RELATIVE_BACKGROUND_NAME, NULL);
76
77   if (!background)
78     background = gdk_pixbuf_new_from_file (INSTALLED_BACKGROUND_NAME, error);
79
80   if (!background)
81     return FALSE; /* note that "error" was filled in and returned */
82
83   back_width = gdk_pixbuf_get_width (background);
84   back_height = gdk_pixbuf_get_height (background);
85
86   if (g_file_test (relative_image_names[0], G_FILE_TEST_EXISTS))
87     image_names = relative_image_names;
88   else
89     image_names = installed_image_names;
90
91   for (i = 0; i < N_IMAGES; i++)
92     {
93       images[i] = gdk_pixbuf_new_from_file (image_names[i], error);
94       if (!images[i])
95         return FALSE; /* Note that "error" was filled with a GError */
96     }
97
98   return TRUE;
99 }
100
101 /* Expose callback for the drawing area */
102 static gint
103 expose_cb (GtkWidget      *widget,
104            GdkEventExpose *event,
105            gpointer        data)
106 {
107   guchar *pixels;
108   int rowstride;
109
110   rowstride = gdk_pixbuf_get_rowstride (frame);
111
112   pixels = gdk_pixbuf_get_pixels (frame) + rowstride * event->area.y + event->area.x * 3;
113
114   gdk_draw_rgb_image_dithalign (widget->window,
115                                 widget->style->black_gc,
116                                 event->area.x, event->area.y,
117                                 event->area.width, event->area.height,
118                                 GDK_RGB_DITHER_NORMAL,
119                                 pixels, rowstride,
120                                 event->area.x, event->area.y);
121
122   return TRUE;
123 }
124
125 #define CYCLE_LEN 60
126
127 static int frame_num;
128
129 /* Timeout handler to regenerate the frame */
130 static gint
131 timeout (gpointer data)
132 {
133   double f;
134   int i;
135   double xmid, ymid;
136   double radius;
137
138   gdk_pixbuf_copy_area (background, 0, 0, back_width, back_height,
139                         frame, 0, 0);
140
141   f = (double) (frame_num % CYCLE_LEN) / CYCLE_LEN;
142
143   xmid = back_width / 2.0;
144   ymid = back_height / 2.0;
145
146   radius = MIN (xmid, ymid) / 2.0;
147
148   for (i = 0; i < N_IMAGES; i++)
149     {
150       double ang;
151       int xpos, ypos;
152       int iw, ih;
153       double r;
154       GdkRectangle r1, r2, dest;
155       double k;
156
157       ang = 2.0 * M_PI * (double) i / N_IMAGES - f * 2.0 * M_PI;
158
159       iw = gdk_pixbuf_get_width (images[i]);
160       ih = gdk_pixbuf_get_height (images[i]);
161
162       r = radius + (radius / 3.0) * sin (f * 2.0 * M_PI);
163
164       xpos = floor (xmid + r * cos (ang) - iw / 2.0 + 0.5);
165       ypos = floor (ymid + r * sin (ang) - ih / 2.0 + 0.5);
166
167       k = (i & 1) ? sin (f * 2.0 * M_PI) : cos (f * 2.0 * M_PI);
168       k = 2.0 * k * k;
169       k = MAX (0.25, k);
170
171       r1.x = xpos;
172       r1.y = ypos;
173       r1.width = iw * k;
174       r1.height = ih * k;
175
176       r2.x = 0;
177       r2.y = 0;
178       r2.width = back_width;
179       r2.height = back_height;
180
181       if (gdk_rectangle_intersect (&r1, &r2, &dest))
182         gdk_pixbuf_composite (images[i],
183                               frame,
184                               dest.x, dest.y,
185                               dest.width, dest.height,
186                               xpos, ypos,
187                               k, k,
188                               GDK_INTERP_NEAREST,
189                               ((i & 1)
190                                ? MAX (127, fabs (255 * sin (f * 2.0 * M_PI)))
191                                : MAX (127, fabs (255 * cos (f * 2.0 * M_PI)))));
192     }
193
194   gtk_widget_queue_draw (da);
195
196   frame_num++;
197   return TRUE;
198 }
199
200 static guint timeout_id;
201
202 static void
203 cleanup_callback (GtkObject *object,
204                   gpointer   data)
205 {
206   g_source_remove (timeout_id);
207   timeout_id = 0;
208 }
209
210 GtkWidget *
211 do_pixbufs (void)
212 {
213   if (!window)
214     {
215       GError *error;
216
217
218       window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
219       gtk_window_set_title (GTK_WINDOW (window), "Pixbufs");
220       gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
221
222       gtk_signal_connect (GTK_OBJECT (window), "destroy", GTK_SIGNAL_FUNC (gtk_widget_destroyed), &window);
223       gtk_signal_connect (GTK_OBJECT (window), "destroy", GTK_SIGNAL_FUNC (cleanup_callback), NULL);
224
225
226       error = NULL;
227       if (!load_pixbufs (&error))
228         {
229           GtkWidget *dialog;
230
231           dialog = gtk_message_dialog_new (GTK_WINDOW (window),
232                                            GTK_DIALOG_DESTROY_WITH_PARENT,
233                                            GTK_MESSAGE_ERROR,
234                                            GTK_BUTTONS_CLOSE,
235                                            "Failed to load an image: %s",
236                                            error->message);
237
238           g_error_free (error);
239
240           gtk_signal_connect (GTK_OBJECT (dialog),
241                               "response",
242                               GTK_SIGNAL_FUNC (gtk_widget_destroy),
243                               NULL);
244
245           gtk_widget_show (dialog);
246         }
247       else
248         {
249           gtk_widget_set_size_request (window, back_width, back_height);
250
251           frame = gdk_pixbuf_new (GDK_COLORSPACE_RGB, FALSE, 8, back_width, back_height);
252
253           da = gtk_drawing_area_new ();
254
255           gtk_signal_connect (GTK_OBJECT (da), "expose_event",
256                               GTK_SIGNAL_FUNC (expose_cb), NULL);
257
258           gtk_container_add (GTK_CONTAINER (window), da);
259
260           timeout_id = gtk_timeout_add (FRAME_DELAY, timeout, NULL);
261         }
262     }
263
264   if (!GTK_WIDGET_VISIBLE (window))
265     {
266       gtk_widget_show_all (window);
267     }
268   else
269     {
270       gtk_widget_destroy (window);
271       window = NULL;
272     }
273
274   return window;
275 }