forked from AuroraMiddleware/gtk
f37a0627fc
2001-08-07 Havoc Pennington <hp@pobox.com> * gtk/gtkfilesel.c (open_ref_dir): fix a typo. * gtk/gtkplug.c (gtk_plug_init): remove setting of auto_shrink; some fixage is needed here, but nothing simple. Owen understands it. ;-) * gtk/gtkwindow.h, gtk/gtkwindow.c: Rework code and API for window sizing and positioning. Also, fix bug in compute_geometry_hints (width/height confusion for setting min size). (gtk_window_move): new function (gtk_window_resize): new function (gtk_window_get_size): new function (gtk_window_get_position): new function (gtk_window_parse_geometry): new function * gtk/gtkwidget.c (gtk_widget_set_size_request): new function (gtk_widget_get_size_request): new function (gtk_widget_get_usize): delete, that was a short-lived function ;-) (gtk_widget_set_usize): deprecate (gtk_widget_set_uposition): deprecate, make it a trivial gtk_window_move() wrapper (gtk_widget_class_init): remove x/y/width/height properties, add width_request height_request * demos/*: update to avoid deprecated functions * gtk/gtklayout.c: add x/y child properties * gtk/gtkfixed.c: add x/y child properties, and get rid of uses of "gint16" * tests/testgtk.c (create_window_sizing): lots of tweaks to window sizing test * gdk/x11/gdkevents-x11.c (gdk_event_translate): Ensure that configure events on toplevel windows are always in root window coordinates, following ICCCM spec that all synthetic events are in root window coords already, while real events are in parent window coords. Previously the code assumed that coords of 0,0 were parent window coords, which was really broken. * gtk/gtkcontainer.c (gtk_container_get_focus_chain): fix warning * gdk/gdkwindow.h (GdkWindowHints): add GDK_HINT_USER_POS and GDK_HINT_USER_SIZE so we can set USSize and USPosition hints in gtk_window_parse_geometry() * gdk/x11/gdkwindow-x11.c (gdk_window_set_geometry_hints): support new USER_POS USER_SIZE hints
276 lines
6.3 KiB
C
276 lines
6.3 KiB
C
/* Pixbufs
|
|
*
|
|
* A GdkPixbuf represents an image, normally in RGB or RGBA format.
|
|
* Pixbufs are normally used to load files from disk and perform
|
|
* image scaling.
|
|
*
|
|
* This demo is not all that educational, but looks cool. It was written
|
|
* by Extreme Pixbuf Hacker Federico Mena Quintero. It also shows
|
|
* off how to use GtkDrawingArea to do a simple animation.
|
|
*
|
|
* Look at the Image demo for additional pixbuf usage examples.
|
|
*
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <stdlib.h>
|
|
#include <gtk/gtk.h>
|
|
#include <math.h>
|
|
|
|
#define FRAME_DELAY 50
|
|
|
|
#define RELATIVE_BACKGROUND_NAME "background.jpg"
|
|
#define INSTALLED_BACKGROUND_NAME DEMOCODEDIR"/background.jpg"
|
|
|
|
static const char *relative_image_names[] = {
|
|
"apple-red.png",
|
|
"gnome-applets.png",
|
|
"gnome-calendar.png",
|
|
"gnome-foot.png",
|
|
"gnome-gmush.png",
|
|
"gnome-gimp.png",
|
|
"gnome-gsame.png",
|
|
"gnu-keys.png"
|
|
};
|
|
|
|
static const char *installed_image_names[] = {
|
|
DEMOCODEDIR"/apple-red.png",
|
|
DEMOCODEDIR"/gnome-applets.png",
|
|
DEMOCODEDIR"/gnome-calendar.png",
|
|
DEMOCODEDIR"/gnome-foot.png",
|
|
DEMOCODEDIR"/gnome-gmush.png",
|
|
DEMOCODEDIR"/gnome-gimp.png",
|
|
DEMOCODEDIR"/gnome-gsame.png",
|
|
DEMOCODEDIR"/gnu-keys.png"
|
|
};
|
|
|
|
#define N_IMAGES G_N_ELEMENTS (relative_image_names)
|
|
|
|
/* demo window */
|
|
static GtkWidget *window = NULL;
|
|
|
|
/* Current frame */
|
|
static GdkPixbuf *frame;
|
|
|
|
/* Background image */
|
|
static GdkPixbuf *background;
|
|
static gint back_width, back_height;
|
|
|
|
/* Images */
|
|
static GdkPixbuf *images[N_IMAGES];
|
|
|
|
/* Widgets */
|
|
static GtkWidget *da;
|
|
|
|
/* Loads the images for the demo and returns whether the operation succeeded */
|
|
static gboolean
|
|
load_pixbufs (GError **error)
|
|
{
|
|
gint i;
|
|
const gchar **image_names;
|
|
|
|
if (background)
|
|
return TRUE; /* already loaded earlier */
|
|
|
|
background = gdk_pixbuf_new_from_file (RELATIVE_BACKGROUND_NAME, NULL);
|
|
|
|
if (!background)
|
|
background = gdk_pixbuf_new_from_file (INSTALLED_BACKGROUND_NAME, error);
|
|
|
|
if (!background)
|
|
return FALSE; /* note that "error" was filled in and returned */
|
|
|
|
back_width = gdk_pixbuf_get_width (background);
|
|
back_height = gdk_pixbuf_get_height (background);
|
|
|
|
if (g_file_test (relative_image_names[0], G_FILE_TEST_EXISTS))
|
|
image_names = relative_image_names;
|
|
else
|
|
image_names = installed_image_names;
|
|
|
|
for (i = 0; i < N_IMAGES; i++)
|
|
{
|
|
images[i] = gdk_pixbuf_new_from_file (image_names[i], error);
|
|
if (!images[i])
|
|
return FALSE; /* Note that "error" was filled with a GError */
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Expose callback for the drawing area */
|
|
static gint
|
|
expose_cb (GtkWidget *widget,
|
|
GdkEventExpose *event,
|
|
gpointer data)
|
|
{
|
|
guchar *pixels;
|
|
int rowstride;
|
|
|
|
rowstride = gdk_pixbuf_get_rowstride (frame);
|
|
|
|
pixels = gdk_pixbuf_get_pixels (frame) + rowstride * event->area.y + event->area.x * 3;
|
|
|
|
gdk_draw_rgb_image_dithalign (widget->window,
|
|
widget->style->black_gc,
|
|
event->area.x, event->area.y,
|
|
event->area.width, event->area.height,
|
|
GDK_RGB_DITHER_NORMAL,
|
|
pixels, rowstride,
|
|
event->area.x, event->area.y);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define CYCLE_LEN 60
|
|
|
|
static int frame_num;
|
|
|
|
/* Timeout handler to regenerate the frame */
|
|
static gint
|
|
timeout (gpointer data)
|
|
{
|
|
double f;
|
|
int i;
|
|
double xmid, ymid;
|
|
double radius;
|
|
|
|
gdk_pixbuf_copy_area (background, 0, 0, back_width, back_height,
|
|
frame, 0, 0);
|
|
|
|
f = (double) (frame_num % CYCLE_LEN) / CYCLE_LEN;
|
|
|
|
xmid = back_width / 2.0;
|
|
ymid = back_height / 2.0;
|
|
|
|
radius = MIN (xmid, ymid) / 2.0;
|
|
|
|
for (i = 0; i < N_IMAGES; i++)
|
|
{
|
|
double ang;
|
|
int xpos, ypos;
|
|
int iw, ih;
|
|
double r;
|
|
GdkRectangle r1, r2, dest;
|
|
double k;
|
|
|
|
ang = 2.0 * M_PI * (double) i / N_IMAGES - f * 2.0 * M_PI;
|
|
|
|
iw = gdk_pixbuf_get_width (images[i]);
|
|
ih = gdk_pixbuf_get_height (images[i]);
|
|
|
|
r = radius + (radius / 3.0) * sin (f * 2.0 * M_PI);
|
|
|
|
xpos = floor (xmid + r * cos (ang) - iw / 2.0 + 0.5);
|
|
ypos = floor (ymid + r * sin (ang) - ih / 2.0 + 0.5);
|
|
|
|
k = (i & 1) ? sin (f * 2.0 * M_PI) : cos (f * 2.0 * M_PI);
|
|
k = 2.0 * k * k;
|
|
k = MAX (0.25, k);
|
|
|
|
r1.x = xpos;
|
|
r1.y = ypos;
|
|
r1.width = iw * k;
|
|
r1.height = ih * k;
|
|
|
|
r2.x = 0;
|
|
r2.y = 0;
|
|
r2.width = back_width;
|
|
r2.height = back_height;
|
|
|
|
if (gdk_rectangle_intersect (&r1, &r2, &dest))
|
|
gdk_pixbuf_composite (images[i],
|
|
frame,
|
|
dest.x, dest.y,
|
|
dest.width, dest.height,
|
|
xpos, ypos,
|
|
k, k,
|
|
GDK_INTERP_NEAREST,
|
|
((i & 1)
|
|
? MAX (127, fabs (255 * sin (f * 2.0 * M_PI)))
|
|
: MAX (127, fabs (255 * cos (f * 2.0 * M_PI)))));
|
|
}
|
|
|
|
gtk_widget_queue_draw (da);
|
|
|
|
frame_num++;
|
|
return TRUE;
|
|
}
|
|
|
|
static guint timeout_id;
|
|
|
|
static void
|
|
cleanup_callback (GtkObject *object,
|
|
gpointer data)
|
|
{
|
|
g_source_remove (timeout_id);
|
|
timeout_id = 0;
|
|
}
|
|
|
|
GtkWidget *
|
|
do_pixbufs (void)
|
|
{
|
|
if (!window)
|
|
{
|
|
GError *error;
|
|
|
|
|
|
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_window_set_title (GTK_WINDOW (window), "Pixbufs");
|
|
gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (window), "destroy", GTK_SIGNAL_FUNC (gtk_widget_destroyed), &window);
|
|
gtk_signal_connect (GTK_OBJECT (window), "destroy", GTK_SIGNAL_FUNC (cleanup_callback), NULL);
|
|
|
|
|
|
error = NULL;
|
|
if (!load_pixbufs (&error))
|
|
{
|
|
GtkWidget *dialog;
|
|
|
|
dialog = gtk_message_dialog_new (GTK_WINDOW (window),
|
|
GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
GTK_MESSAGE_ERROR,
|
|
GTK_BUTTONS_CLOSE,
|
|
"Failed to load an image: %s",
|
|
error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (dialog),
|
|
"response",
|
|
GTK_SIGNAL_FUNC (gtk_widget_destroy),
|
|
NULL);
|
|
|
|
gtk_widget_show (dialog);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_set_size_request (window, back_width, back_height);
|
|
|
|
frame = gdk_pixbuf_new (GDK_COLORSPACE_RGB, FALSE, 8, back_width, back_height);
|
|
|
|
da = gtk_drawing_area_new ();
|
|
|
|
gtk_signal_connect (GTK_OBJECT (da), "expose_event",
|
|
GTK_SIGNAL_FUNC (expose_cb), NULL);
|
|
|
|
gtk_container_add (GTK_CONTAINER (window), da);
|
|
|
|
timeout_id = gtk_timeout_add (FRAME_DELAY, timeout, NULL);
|
|
}
|
|
}
|
|
|
|
if (!GTK_WIDGET_VISIBLE (window))
|
|
{
|
|
gtk_widget_show_all (window);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_destroy (window);
|
|
window = NULL;
|
|
}
|
|
|
|
return window;
|
|
}
|