/* * Copyright (C) 2005 Ray Strode <rstrode@redhat.com>, * Matthias Clasen <mclasen@redhat.com>, * Søren Sandmann <sandmann@redhat.com> * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301, USA. * * Originally written by: Ray Strode <rstrode@redhat.com> * * Later contributions by: Matthias Clasen <mclasen@redhat.com> * Søren Sandmann <sandmann@redhat.com> */ #include "config.h" #include <math.h> #include <stdlib.h> #include <sysexits.h> #include <time.h> #include <glib.h> #include <glib/gi18n.h> #include <gdk/gdk.h> #include <gdk/gdkx.h> #include <gtk/gtk.h> #include "gs-theme-window.h" #ifndef trunc #define trunc(x) (((x) > 0.0) ? floor((x)) : -floor(-(x))) #endif #ifndef OPTIMAL_FRAME_RATE #define OPTIMAL_FRAME_RATE (25.0) #endif #ifndef STAT_PRINT_FREQUENCY #define STAT_PRINT_FREQUENCY (2000) #endif #ifndef FLOATER_MAX_SIZE #define FLOATER_MAX_SIZE (128.0) #endif #ifndef FLOATER_MIN_SIZE #define FLOATER_MIN_SIZE (16.0) #endif #ifndef FLOATER_DEFAULT_COUNT #define FLOATER_DEFAULT_COUNT (5) #endif #ifndef SMALL_ANGLE #define SMALL_ANGLE (0.025 * G_PI) #endif #ifndef BIG_ANGLE #define BIG_ANGLE (0.125 * G_PI) #endif #ifndef GAMMA #define GAMMA 2.2 #endif static gboolean should_show_paths = FALSE; static gboolean should_do_rotations = FALSE; static gboolean should_print_stats = FALSE; static gint max_floater_count = FLOATER_DEFAULT_COUNT; static gchar *geometry = NULL; static gchar **filenames = NULL; static GOptionEntry options[] = { { "show-paths", 'p', 0, G_OPTION_ARG_NONE, &should_show_paths, N_("Show paths that images follow"), NULL }, { "do-rotations", 'r', 0, G_OPTION_ARG_NONE, &should_do_rotations, N_("Occasionally rotate images as they move"), NULL }, { "print-stats", 's', 0, G_OPTION_ARG_NONE, &should_print_stats, N_("Print out frame rate and other statistics"), NULL }, { "number-of-images", 'n', 0, G_OPTION_ARG_INT, &max_floater_count, N_("The maximum number of images to keep on screen"), N_("MAX_IMAGES") }, { "geometry", 0, 0, G_OPTION_ARG_STRING, &geometry, N_("The initial size and position of window"), N_("WIDTHxHEIGHT+X+Y") }, { G_OPTION_REMAINING, 0, 0, G_OPTION_ARG_FILENAME_ARRAY, &filenames, N_("The source image to use"), NULL }, {NULL} }; typedef struct _Point Point; typedef struct _Path Path; typedef struct _Rectangle Rectangle; typedef struct _ScreenSaverFloater ScreenSaverFloater; typedef struct _CachedSource CachedSource; typedef struct _ScreenSaver ScreenSaver; struct _Point { gdouble x, y; }; struct _Path { Point start_point; Point start_control_point; Point end_control_point; Point end_point; gdouble x_linear_coefficient, y_linear_coefficient; gdouble x_quadratic_coefficient, y_quadratic_coefficient; gdouble x_cubic_coefficient, y_cubic_coefficient; gdouble duration; }; struct _CachedSource { cairo_pattern_t *pattern; gint width, height; }; struct _Rectangle { Point top_left_point; Point bottom_right_point; }; struct _ScreenSaverFloater { GdkRectangle bounds; Point start_position; Point position; gdouble scale; gdouble opacity; Path *path; gdouble path_start_time; gdouble path_start_scale; gdouble path_end_scale; gdouble angle; gdouble angle_increment; }; struct _ScreenSaver { GtkWidget *drawing_area; Rectangle canvas_rectangle; GHashTable *cached_sources; char *filename; gdouble first_update_time; gdouble last_calculated_stats_time, current_calculated_stats_time; gint update_count, frame_count; gdouble updates_per_second; gdouble frames_per_second; guint state_update_timeout_id; guint stats_update_timeout_id; GList *floaters; gint max_floater_count; guint should_do_rotations: 1; guint should_show_paths : 1; guint draw_ops_pending : 1; }; static Path *path_new (Point *start_point, Point *start_control_point, Point *end_control_point, Point *end_point, gdouble duration); static void path_free (Path *path); static ScreenSaverFloater *screen_saver_floater_new (ScreenSaver *screen_saver, Point *position, gdouble scale); static void screen_saver_floater_free (ScreenSaver *screen_saver, ScreenSaverFloater *floater); static gboolean screen_saver_floater_is_off_canvas (ScreenSaver *screen_saver, ScreenSaverFloater *floater); static gboolean screen_saver_floater_should_bubble_up (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble performance_ratio, gdouble *duration); static gboolean screen_saver_floater_should_come_on_screen (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble performance_ratio, gdouble *duration); static Point screen_saver_floater_get_position_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time); static gdouble screen_saver_floater_get_scale_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time); static gdouble screen_saver_floater_get_angle_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time); static Path *screen_saver_floater_create_path_to_on_screen (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration); static Path *screen_saver_floater_create_path_to_bubble_up (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration); static Path *screen_saver_floater_create_path_to_random_point (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration); static Path *screen_saver_floater_create_path (ScreenSaver *screen_saver, ScreenSaverFloater *floater); static void screen_saver_floater_update_state (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time); static gboolean screen_saver_floater_do_draw (ScreenSaver *screen_saver, ScreenSaverFloater *floater, cairo_t *context); static CachedSource *cached_source_new (cairo_pattern_t *pattern, gint width, gint height); static void cached_source_free (CachedSource *source); static ScreenSaver *screen_saver_new (GtkDrawingArea *drawing_area, const gchar *filename, gint max_floater_count, gboolean should_do_rotations, gboolean should_show_paths); static void screen_saver_free (ScreenSaver *screen_saver); static gdouble screen_saver_get_timestamp (ScreenSaver *screen_saver); static void screen_saver_get_initial_state (ScreenSaver *screen_saver); static void screen_saver_update_state (ScreenSaver *screen_saver, gdouble time); static gboolean screen_saver_do_update_state (ScreenSaver *screen_saver); static gboolean screen_saver_do_update_stats (ScreenSaver *screen_saver); static gdouble screen_saver_get_updates_per_second (ScreenSaver *screen_saver); static gdouble screen_saver_get_frames_per_second (ScreenSaver *screen_saver); static gdouble screen_saver_get_image_cache_usage (ScreenSaver *screen_saver); static void screen_saver_create_floaters (ScreenSaver *screen_saver); static void screen_saver_destroy_floaters (ScreenSaver *screen_saver); static void screen_saver_on_size_allocate (ScreenSaver *screen_saver, GtkAllocation *allocation); #if GTK_CHECK_VERSION (3, 0, 0) static void screen_saver_on_draw (ScreenSaver *screen_saver, cairo_t *context); #else static void screen_saver_on_expose_event (ScreenSaver *screen_saver, GdkEventExpose *event); #endif static gboolean do_print_screen_saver_stats (ScreenSaver *screen_saver); static GdkPixbuf *gamma_correct (const GdkPixbuf *input_pixbuf); static CachedSource* cached_source_new (cairo_pattern_t *pattern, gint width, gint height) { CachedSource *source; source = g_new (CachedSource, 1); source->pattern = cairo_pattern_reference (pattern); source->width = width; source->height = height; return source; } static void cached_source_free (CachedSource *source) { if (source == NULL) return; cairo_pattern_destroy (source->pattern); g_free (source); } static Path * path_new (Point *start_point, Point *start_control_point, Point *end_control_point, Point *end_point, gdouble duration) { Path *path; path = g_new (Path, 1); path->start_point = *start_point; path->start_control_point = *start_control_point; path->end_control_point = *end_control_point; path->end_point = *end_point; path->duration = duration; /* we precompute the coefficients to the cubic bezier curve here * so that we don't have to do it repeatedly later The equation is: * * B(t) = A * t^3 + B * t^2 + C * t + start_point */ path->x_linear_coefficient = 3 * (start_control_point->x - start_point->x); path->x_quadratic_coefficient = 3 * (end_control_point->x - start_control_point->x) - path->x_linear_coefficient; path->x_cubic_coefficient = end_point->x - start_point->x - path->x_linear_coefficient - path->x_quadratic_coefficient; path->y_linear_coefficient = 3 * (start_control_point->y - start_point->y); path->y_quadratic_coefficient = 3 * (end_control_point->y - start_control_point->y) - path->y_linear_coefficient; path->y_cubic_coefficient = end_point->y - start_point->y - path->y_linear_coefficient - path->y_quadratic_coefficient; return path; } static void path_free (Path *path) { g_free (path); } static ScreenSaverFloater* screen_saver_floater_new (ScreenSaver *screen_saver, Point *position, gdouble scale) { ScreenSaverFloater *floater; floater = g_new (ScreenSaverFloater, 1); floater->bounds.width = 0; floater->start_position = *position; floater->position = *position; floater->scale = scale; floater->opacity = pow (scale, 1.0 / GAMMA); floater->path = NULL; floater->path_start_time = 0.0; floater->path_start_scale = 1.0; floater->path_end_scale = 0.0; floater->angle = 0.0; floater->angle_increment = 0.0; return floater; } void screen_saver_floater_free (ScreenSaver *screen_saver, ScreenSaverFloater *floater) { if (floater == NULL) return; path_free (floater->path); g_free (floater); } static gboolean screen_saver_floater_is_off_canvas (ScreenSaver *screen_saver, ScreenSaverFloater *floater) { if ((floater->position.x < screen_saver->canvas_rectangle.top_left_point.x) || (floater->position.x > screen_saver->canvas_rectangle.bottom_right_point.x) || (floater->position.y < screen_saver->canvas_rectangle.top_left_point.y) || (floater->position.y > screen_saver->canvas_rectangle.bottom_right_point.y)) return TRUE; return FALSE; } static gboolean screen_saver_floater_should_come_on_screen (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble performance_ratio, gdouble *duration) { if (!screen_saver_floater_is_off_canvas (screen_saver, floater)) return FALSE; if ((abs (performance_ratio - .5) >= G_MINDOUBLE) && (g_random_double () > .5)) { if (duration) *duration = g_random_double_range (3.0, 7.0); return TRUE; } return FALSE; } static gboolean screen_saver_floater_should_bubble_up (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble performance_ratio, gdouble *duration) { if ((performance_ratio < .5) && (g_random_double () > .5)) { if (duration) *duration = performance_ratio * 30.0; return TRUE; } if ((floater->scale < .3) && (g_random_double () > .6)) { if (duration) *duration = 30.0; return TRUE; } return FALSE; } static Point screen_saver_floater_get_position_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time) { Point point; time = time / floater->path->duration; point.x = floater->path->x_cubic_coefficient * (time * time * time) + floater->path->x_quadratic_coefficient * (time * time) + floater->path->x_linear_coefficient * (time) + floater->path->start_point.x; point.y = floater->path->y_cubic_coefficient * (time * time * time) + floater->path->y_quadratic_coefficient * (time * time) + floater->path->y_linear_coefficient * (time) + floater->path->start_point.y; return point; } static gdouble screen_saver_floater_get_scale_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time) { gdouble completion_ratio, total_scale_growth, new_scale; completion_ratio = time / floater->path->duration; total_scale_growth = (floater->path_end_scale - floater->path_start_scale); new_scale = floater->path_start_scale + total_scale_growth * completion_ratio; return CLAMP (new_scale, 0.0, 1.0); } static gdouble screen_saver_floater_get_angle_from_time (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time) { gdouble completion_ratio; gdouble total_rotation; completion_ratio = time / floater->path->duration; total_rotation = floater->angle_increment * floater->path->duration; return floater->angle + total_rotation * completion_ratio; } static Path * screen_saver_floater_create_path_to_on_screen (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration) { Point start_position, end_position, start_control_point, end_control_point; start_position = floater->position; end_position.x = g_random_double_range (.25, .75) * (screen_saver->canvas_rectangle.top_left_point.x + screen_saver->canvas_rectangle.bottom_right_point.x); end_position.y = g_random_double_range (.25, .75) * (screen_saver->canvas_rectangle.top_left_point.y + screen_saver->canvas_rectangle.bottom_right_point.y); start_control_point.x = start_position.x + .9 * (end_position.x - start_position.x); start_control_point.y = start_position.y + .9 * (end_position.y - start_position.y); end_control_point.x = start_position.x + 1.0 * (end_position.x - start_position.x); end_control_point.y = start_position.y + 1.0 * (end_position.y - start_position.y); return path_new (&start_position, &start_control_point, &end_control_point, &end_position, duration); } static Path * screen_saver_floater_create_path_to_bubble_up (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration) { Point start_position, end_position, start_control_point, end_control_point; start_position = floater->position; end_position.x = start_position.x; end_position.y = screen_saver->canvas_rectangle.top_left_point.y - FLOATER_MAX_SIZE; start_control_point.x = .5 * start_position.x; start_control_point.y = .5 * start_position.y; end_control_point.x = 1.5 * end_position.x; end_control_point.y = .5 * end_position.y; return path_new (&start_position, &start_control_point, &end_control_point, &end_position, duration); } static Path * screen_saver_floater_create_path_to_random_point (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble duration) { Point start_position, end_position, start_control_point, end_control_point; start_position = floater->position; end_position.x = start_position.x + (g_random_double_range (-.5, .5) * 4 * FLOATER_MAX_SIZE); end_position.y = start_position.y + (g_random_double_range (-.5, .5) * 4 * FLOATER_MAX_SIZE); start_control_point.x = start_position.x + .95 * (end_position.x - start_position.x); start_control_point.y = start_position.y + .95 * (end_position.y - start_position.y); end_control_point.x = start_position.x + 1.0 * (end_position.x - start_position.x); end_control_point.y = start_position.y + 1.0 * (end_position.y - start_position.y); return path_new (&start_position, &start_control_point, &end_control_point, &end_position, duration); } static Path * screen_saver_floater_create_path (ScreenSaver *screen_saver, ScreenSaverFloater *floater) { gdouble performance_ratio; gdouble duration; performance_ratio = screen_saver_get_frames_per_second (screen_saver) / OPTIMAL_FRAME_RATE; if (abs (performance_ratio) <= G_MINDOUBLE) performance_ratio = 1.0; if (screen_saver_floater_should_bubble_up (screen_saver, floater, performance_ratio, &duration)) return screen_saver_floater_create_path_to_bubble_up (screen_saver, floater, duration); if (screen_saver_floater_should_come_on_screen (screen_saver, floater, performance_ratio, &duration)) return screen_saver_floater_create_path_to_on_screen (screen_saver, floater, duration); return screen_saver_floater_create_path_to_random_point (screen_saver, floater, g_random_double_range (3.0, 7.0)); } static void screen_saver_floater_update_state (ScreenSaver *screen_saver, ScreenSaverFloater *floater, gdouble time) { gdouble performance_ratio; performance_ratio = screen_saver_get_frames_per_second (screen_saver) / OPTIMAL_FRAME_RATE; if (floater->path == NULL) { floater->path = screen_saver_floater_create_path (screen_saver, floater); floater->path_start_time = time; floater->path_start_scale = floater->scale; if (g_random_double () > .5) floater->path_end_scale = g_random_double_range (0.10, performance_ratio); /* poor man's distribution */ if (screen_saver->should_do_rotations && (g_random_double () < .75 * performance_ratio)) { gint r; r = g_random_int_range (0, 100); if (r < 80) floater->angle_increment = 0.0; else if (r < 95) floater->angle_increment = g_random_double_range (-SMALL_ANGLE, SMALL_ANGLE); else floater->angle_increment = g_random_double_range (-BIG_ANGLE, BIG_ANGLE); } } if (time < (floater->path_start_time + floater->path->duration)) { gdouble path_time; path_time = time - floater->path_start_time; floater->position = screen_saver_floater_get_position_from_time (screen_saver, floater, path_time); floater->scale = screen_saver_floater_get_scale_from_time (screen_saver, floater, path_time); floater->angle = screen_saver_floater_get_angle_from_time (screen_saver, floater, path_time); floater->opacity = pow (floater->scale, 1.0 / GAMMA); } else { path_free (floater->path); floater->path = NULL; floater->path_start_time = 0.0; } } static GdkPixbuf * gamma_correct (const GdkPixbuf *input_pixbuf) { gint x, y, width, height, rowstride; GdkPixbuf *output_pixbuf; guchar *pixels; output_pixbuf = gdk_pixbuf_copy (input_pixbuf); pixels = gdk_pixbuf_get_pixels (output_pixbuf); width = gdk_pixbuf_get_width (output_pixbuf); height = gdk_pixbuf_get_height (output_pixbuf); rowstride = gdk_pixbuf_get_rowstride (output_pixbuf); for (y = 0; y < height; y++) for (x = 0; x < width; x++) { guchar *alpha_channel; guchar opacity; alpha_channel = pixels + y * (rowstride / 4) + x + 3; opacity = (guchar) (255 * pow ((*alpha_channel / 255.0), 1.0 / GAMMA)); *alpha_channel = opacity; } return output_pixbuf; } static gboolean screen_saver_floater_do_draw (ScreenSaver *screen_saver, ScreenSaverFloater *floater, cairo_t *context) { gint size; CachedSource *source; size = CLAMP ((int) (FLOATER_MAX_SIZE * floater->scale), FLOATER_MIN_SIZE, FLOATER_MAX_SIZE); source = g_hash_table_lookup (screen_saver->cached_sources, GINT_TO_POINTER (size)); if (source == NULL) { GdkPixbuf *pixbuf; GError *error; pixbuf = NULL; error = NULL; pixbuf = gdk_pixbuf_new_from_file_at_size (screen_saver->filename, size, -1, &error); if (pixbuf == NULL) { g_assert (error != NULL); g_printerr ("%s", _(error->message)); g_error_free (error); return FALSE; } if (gdk_pixbuf_get_has_alpha (pixbuf)) gamma_correct (pixbuf); gdk_cairo_set_source_pixbuf (context, pixbuf, 0.0, 0.0); source = cached_source_new (cairo_get_source (context), gdk_pixbuf_get_width (pixbuf), gdk_pixbuf_get_height (pixbuf)); g_object_unref (pixbuf); g_hash_table_insert (screen_saver->cached_sources, GINT_TO_POINTER (size), source); } cairo_save (context); if (screen_saver->should_do_rotations && (abs (floater->angle) > G_MINDOUBLE)) { floater->bounds.width = G_SQRT2 * source->width + 2; floater->bounds.height = G_SQRT2 * source->height + 2; floater->bounds.x = (int) (floater->position.x - .5 * G_SQRT2 * source->width) - 1; floater->bounds.y = (int) (floater->position.y - .5 * G_SQRT2 * source->height) - 1; cairo_translate (context, trunc (floater->position.x), trunc (floater->position.y)); cairo_rotate (context, floater->angle); cairo_translate (context, -trunc (floater->position.x), -trunc (floater->position.y)); } else { floater->bounds.width = source->width + 2; floater->bounds.height = source->height + 2; floater->bounds.x = (int) (floater->position.x - .5 * source->width) - 1; floater->bounds.y = (int) (floater->position.y - .5 * source->height) - 1; } cairo_translate (context, trunc (floater->position.x - .5 * source->width), trunc (floater->position.y - .5 * source->height)); cairo_set_source (context, source->pattern); cairo_rectangle (context, trunc (.5 * (source->width - floater->bounds.width)), trunc (.5 * (source->height - floater->bounds.height)), floater->bounds.width, floater->bounds.height); cairo_clip (context); cairo_paint_with_alpha (context, floater->opacity); cairo_restore (context); if (screen_saver->should_show_paths && (floater->path != NULL)) { gdouble dash_pattern[] = { 5.0 }; gint size; size = CLAMP ((int) (FLOATER_MAX_SIZE * floater->path_start_scale), FLOATER_MIN_SIZE, FLOATER_MAX_SIZE); cairo_save (context); cairo_set_source_rgba (context, 1.0, 1.0, 1.0, .2 * floater->opacity); cairo_move_to (context, floater->path->start_point.x, floater->path->start_point.y); cairo_curve_to (context, floater->path->start_control_point.x, floater->path->start_control_point.y, floater->path->end_control_point.x, floater->path->end_control_point.y, floater->path->end_point.x, floater->path->end_point.y); cairo_set_line_cap (context, CAIRO_LINE_CAP_ROUND); cairo_stroke (context); cairo_set_source_rgba (context, 1.0, 0.0, 0.0, .5 * floater->opacity); cairo_rectangle (context, floater->path->start_point.x - 3, floater->path->start_point.y - 3, 6, 6); cairo_fill (context); cairo_set_source_rgba (context, 0.0, 0.5, 0.0, .5 * floater->opacity); cairo_arc (context, floater->path->start_control_point.x, floater->path->start_control_point.y, 3, 0.0, 2.0 * G_PI); cairo_stroke (context); cairo_set_source_rgba (context, 0.5, 0.0, 0.5, .5 * floater->opacity); cairo_arc (context, floater->path->end_control_point.x, floater->path->end_control_point.y, 3, 0.0, 2.0 * G_PI); cairo_stroke (context); cairo_set_source_rgba (context, 0.0, 0.0, 1.0, .5 * floater->opacity); cairo_rectangle (context, floater->path->end_point.x - 3, floater->path->end_point.y - 3, 6, 6); cairo_fill (context); cairo_set_dash (context, dash_pattern, G_N_ELEMENTS (dash_pattern), 0); cairo_set_source_rgba (context, .5, .5, .5, .2 * floater->scale); cairo_move_to (context, floater->path->start_point.x, floater->path->start_point.y); cairo_line_to (context, floater->path->start_control_point.x, floater->path->start_control_point.y); cairo_stroke (context); cairo_move_to (context, floater->path->end_point.x, floater->path->end_point.y); cairo_line_to (context, floater->path->end_control_point.x, floater->path->end_control_point.y); cairo_stroke (context); cairo_restore (context); } return TRUE; } static ScreenSaver * screen_saver_new (GtkDrawingArea *drawing_area, const gchar *filename, gint max_floater_count, gboolean should_do_rotations, gboolean should_show_paths) { ScreenSaver *screen_saver; screen_saver = g_new (ScreenSaver, 1); screen_saver->filename = g_strdup (filename); screen_saver->drawing_area = GTK_WIDGET (drawing_area); screen_saver->cached_sources = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify) cached_source_free); g_signal_connect_swapped (G_OBJECT (drawing_area), "size-allocate", G_CALLBACK (screen_saver_on_size_allocate), screen_saver); #if GTK_CHECK_VERSION (3, 0, 0) g_signal_connect_swapped (G_OBJECT (drawing_area), "draw", G_CALLBACK (screen_saver_on_draw), screen_saver); #else g_signal_connect_swapped (G_OBJECT (drawing_area), "expose-event", G_CALLBACK (screen_saver_on_expose_event), screen_saver); #endif screen_saver->first_update_time = 0.0; screen_saver->current_calculated_stats_time = 0.0; screen_saver->last_calculated_stats_time = 0.0; screen_saver->update_count = 0; screen_saver->frame_count = 0; screen_saver->updates_per_second = 0.0; screen_saver->frames_per_second = 0.0; screen_saver->floaters = NULL; screen_saver->max_floater_count = max_floater_count; screen_saver->should_show_paths = should_show_paths; screen_saver->should_do_rotations = should_do_rotations; screen_saver_get_initial_state (screen_saver); screen_saver->state_update_timeout_id = g_timeout_add (1000 / (2.0 * OPTIMAL_FRAME_RATE), (GSourceFunc) screen_saver_do_update_state, screen_saver); screen_saver->stats_update_timeout_id = g_timeout_add (1000, (GSourceFunc) screen_saver_do_update_stats, screen_saver); return screen_saver; } static void screen_saver_free (ScreenSaver *screen_saver) { if (screen_saver == NULL) return; g_free (screen_saver->filename); g_hash_table_destroy (screen_saver->cached_sources); if (screen_saver->state_update_timeout_id != 0) { g_source_remove (screen_saver->state_update_timeout_id); screen_saver->state_update_timeout_id = 0; } if (screen_saver->stats_update_timeout_id != 0) { g_source_remove (screen_saver->stats_update_timeout_id); screen_saver->stats_update_timeout_id = 0; } screen_saver_destroy_floaters (screen_saver); g_free (screen_saver); } static gdouble screen_saver_get_timestamp (ScreenSaver *screen_saver) { const gdouble microseconds_per_second = (gdouble ) G_USEC_PER_SEC; gdouble timestamp; GTimeVal now = { 0L, /* zero-filled */ }; g_get_current_time (&now); timestamp = ((microseconds_per_second * now.tv_sec) + now.tv_usec) / microseconds_per_second; return timestamp; } static void screen_saver_create_floaters (ScreenSaver *screen_saver) { gint i; for (i = 0; i < screen_saver->max_floater_count; i++) { ScreenSaverFloater *floater; Point position; gdouble scale; position.x = g_random_double_range (screen_saver->canvas_rectangle.top_left_point.x, screen_saver->canvas_rectangle.bottom_right_point.x); position.y = g_random_double_range (screen_saver->canvas_rectangle.top_left_point.y, screen_saver->canvas_rectangle.bottom_right_point.y); scale = g_random_double (); floater = screen_saver_floater_new (screen_saver, &position, scale); screen_saver->floaters = g_list_prepend (screen_saver->floaters, floater); } } static gdouble screen_saver_get_updates_per_second (ScreenSaver *screen_saver) { return screen_saver->updates_per_second; } static gdouble screen_saver_get_frames_per_second (ScreenSaver *screen_saver) { return screen_saver->frames_per_second; } static gdouble screen_saver_get_image_cache_usage (ScreenSaver *screen_saver) { static const gdouble cache_capacity = (FLOATER_MAX_SIZE - FLOATER_MIN_SIZE + 1); return g_hash_table_size (screen_saver->cached_sources) / cache_capacity; } static void screen_saver_destroy_floaters (ScreenSaver *screen_saver) { if (screen_saver->floaters == NULL) return; g_list_foreach (screen_saver->floaters, (GFunc) screen_saver_floater_free, NULL); g_list_free (screen_saver->floaters); screen_saver->floaters = NULL; } static void screen_saver_on_size_allocate (ScreenSaver *screen_saver, GtkAllocation *allocation) { Rectangle canvas_rectangle; canvas_rectangle.top_left_point.x = allocation->x - .1 * allocation->width; canvas_rectangle.top_left_point.y = allocation->y - .1 * allocation->height; canvas_rectangle.bottom_right_point.x = allocation->x + (1.1 * allocation->width); canvas_rectangle.bottom_right_point.y = allocation->y + (1.1 * allocation->height); screen_saver->canvas_rectangle = canvas_rectangle; } static gint compare_floaters (ScreenSaverFloater *a, ScreenSaverFloater *b) { if (a->scale > b->scale) return 1; else if (abs (a->scale - b->scale) <= G_MINDOUBLE) return 0; else return -1; } static void #if GTK_CHECK_VERSION (3, 0, 0) screen_saver_on_draw (ScreenSaver *screen_saver, cairo_t *context) #else screen_saver_on_expose_event (ScreenSaver *screen_saver, GdkEventExpose *event) #endif { GList *tmp; #if !GTK_CHECK_VERSION (3, 0, 0) cairo_t *context; #endif if (screen_saver->floaters == NULL) screen_saver_create_floaters (screen_saver); #if !GTK_CHECK_VERSION (3, 0, 0) context = gdk_cairo_create (gtk_widget_get_window (screen_saver->drawing_area)); cairo_rectangle (context, (double) event->area.x, (double) event->area.y, (double) event->area.width, (double) event->area.height); cairo_clip (context); #endif screen_saver->floaters = g_list_sort (screen_saver->floaters, (GCompareFunc)compare_floaters); for (tmp = screen_saver->floaters; tmp != NULL; tmp = tmp->next) { ScreenSaverFloater *floater; #if !GTK_CHECK_VERSION (3, 0, 0) GdkRectangle rect; gint size; #endif floater = (ScreenSaverFloater *) tmp->data; #if !GTK_CHECK_VERSION (3, 0, 0) size = CLAMP ((int) (FLOATER_MAX_SIZE * floater->scale), FLOATER_MIN_SIZE, FLOATER_MAX_SIZE); rect.x = (int) (floater->position.x - .5 * G_SQRT2 * size); rect.y = (int) (floater->position.y - .5 * G_SQRT2 * size); rect.width = G_SQRT2 * size; rect.height = G_SQRT2 * size; if (!gdk_region_rect_in (event->region, &rect)) continue; #endif if (!screen_saver_floater_do_draw (screen_saver, floater, context)) { gtk_main_quit (); break; } } #if !GTK_CHECK_VERSION (3, 0, 0) cairo_destroy (context); #endif screen_saver->draw_ops_pending = TRUE; screen_saver->frame_count++; } static void screen_saver_update_state (ScreenSaver *screen_saver, gdouble time) { GList *tmp; tmp = screen_saver->floaters; while (tmp != NULL) { ScreenSaverFloater *floater; floater = (ScreenSaverFloater *) tmp->data; screen_saver_floater_update_state (screen_saver, floater, time); if (gtk_widget_get_realized (screen_saver->drawing_area) && (floater->bounds.width > 0) && (floater->bounds.height > 0)) { gint size; size = CLAMP ((int) (FLOATER_MAX_SIZE * floater->scale), FLOATER_MIN_SIZE, FLOATER_MAX_SIZE); gtk_widget_queue_draw_area (screen_saver->drawing_area, floater->bounds.x, floater->bounds.y, floater->bounds.width, floater->bounds.height); /* the edges could concievably be spread across two * pixels so we add +2 to invalidated region */ if (screen_saver->should_do_rotations) gtk_widget_queue_draw_area (screen_saver->drawing_area, (int) (floater->position.x - .5 * G_SQRT2 * size), (int) (floater->position.y - .5 * G_SQRT2 * size), G_SQRT2 * size + 2, G_SQRT2 * size + 2); else gtk_widget_queue_draw_area (screen_saver->drawing_area, (int) (floater->position.x - .5 * size), (int) (floater->position.y - .5 * size), size + 2, size + 2); if (screen_saver->should_show_paths) gtk_widget_queue_draw (screen_saver->drawing_area); } tmp = tmp->next; } } static void screen_saver_get_initial_state (ScreenSaver *screen_saver) { screen_saver->first_update_time = screen_saver_get_timestamp (screen_saver); screen_saver_update_state (screen_saver, 0.0); } static gboolean screen_saver_do_update_state (ScreenSaver *screen_saver) { gdouble current_update_time; /* flush pending requests to the X server and block for * replies before proceeding to help prevent the X server from * getting overrun with requests */ if (screen_saver->draw_ops_pending) { gdk_flush (); screen_saver->draw_ops_pending = FALSE; } current_update_time = screen_saver_get_timestamp (screen_saver); screen_saver_update_state (screen_saver, current_update_time - screen_saver->first_update_time); screen_saver->update_count++; return TRUE; } static gboolean screen_saver_do_update_stats (ScreenSaver *screen_saver) { gdouble last_calculated_stats_time, seconds_since_last_stats_update; last_calculated_stats_time = screen_saver->current_calculated_stats_time; screen_saver->current_calculated_stats_time = screen_saver_get_timestamp (screen_saver); screen_saver->last_calculated_stats_time = last_calculated_stats_time; if (abs (last_calculated_stats_time) <= G_MINDOUBLE) return TRUE; seconds_since_last_stats_update = screen_saver->current_calculated_stats_time - last_calculated_stats_time; screen_saver->updates_per_second = screen_saver->update_count / seconds_since_last_stats_update; screen_saver->frames_per_second = screen_saver->frame_count / seconds_since_last_stats_update; screen_saver->update_count = 0; screen_saver->frame_count = 0; return TRUE; } static gboolean do_print_screen_saver_stats (ScreenSaver *screen_saver) { g_print ("updates per second: %.2f, frames per second: %.2f, " "image cache %.0f%% full\n", screen_saver_get_updates_per_second (screen_saver), screen_saver_get_frames_per_second (screen_saver), screen_saver_get_image_cache_usage (screen_saver) * 100.0); return TRUE; } int main (int argc, char *argv[]) { ScreenSaver *screen_saver; GtkWidget *window; GtkWidget *drawing_area; #if GTK_CHECK_VERSION (3, 0, 0) GdkRGBA bg; GdkRGBA fg; #else GtkStyle *style; GtkStateType state; #endif GError *error; error = NULL; bindtextdomain (GETTEXT_PACKAGE, MATELOCALEDIR); bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8"); textdomain (GETTEXT_PACKAGE); gtk_init_with_args (&argc, &argv, /* translators: the word "image" here * represents a command line argument */ _("image - floats images around the screen"), options, GETTEXT_PACKAGE, &error); if (error != NULL) { g_printerr (_("%s. See --help for usage information.\n"), _(error->message)); g_error_free (error); return EX_SOFTWARE; } if ((filenames == NULL) || (filenames[0] == NULL) || (filenames[1] != NULL)) { g_printerr (_("You must specify one image. See --help for usage " "information.\n")); return EX_USAGE; } window = gs_theme_window_new (); g_signal_connect (G_OBJECT (window), "delete-event", G_CALLBACK (gtk_main_quit), NULL); drawing_area = gtk_drawing_area_new (); #if GTK_CHECK_VERSION (3, 0, 0) bg.red = 0; bg.green = 0; bg.blue = 0; bg.alpha = 1.0; fg.red = 0.8; fg.green = 0.8; fg.blue = 0.8; fg.alpha = 1.0; gtk_widget_override_background_color (drawing_area, 0, &bg); gtk_widget_override_color (drawing_area, 0, &fg); #else style = gtk_widget_get_style (drawing_area); state = (GtkStateType) 0; while (state < (GtkStateType) G_N_ELEMENTS (style->bg)) { gtk_widget_modify_bg (drawing_area, state, &style->mid[state]); state++; } #endif gtk_widget_show (drawing_area); gtk_container_add (GTK_CONTAINER (window), drawing_area); screen_saver = screen_saver_new (GTK_DRAWING_AREA (drawing_area), filenames[0], max_floater_count, should_do_rotations, should_show_paths); g_strfreev (filenames); if (should_print_stats) g_timeout_add (STAT_PRINT_FREQUENCY, (GSourceFunc) do_print_screen_saver_stats, screen_saver); if ((geometry == NULL) || !gtk_window_parse_geometry (GTK_WINDOW (window), geometry)) gtk_window_set_default_size (GTK_WINDOW (window), 640, 480); gtk_widget_show (window); gtk_main (); screen_saver_free (screen_saver); return EX_OK; }