228 lines
5.4 KiB
C
228 lines
5.4 KiB
C
|
/* -*- mode: C; c-basic-offset: 2; indent-tabs-mode: nil; -*- */
|
||
|
|
||
|
#include <gtk/gtk.h>
|
||
|
#include <math.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "frame-stats.h"
|
||
|
|
||
|
#define RADIUS 64
|
||
|
#define DIAMETER (2*RADIUS)
|
||
|
#define WIDTH 600
|
||
|
#define HEIGHT 600
|
||
|
#define WINDOW_SIZE_JITTER 200
|
||
|
#define CYCLE_TIME 5.
|
||
|
|
||
|
static GtkWidget *window;
|
||
|
static int window_width = WIDTH, window_height = HEIGHT;
|
||
|
|
||
|
gint64 start_frame_time;
|
||
|
static double angle;
|
||
|
|
||
|
static double load_factor = 1.0;
|
||
|
static double cb_no_resize = FALSE;
|
||
|
|
||
|
static cairo_surface_t *source_surface;
|
||
|
|
||
|
static void
|
||
|
ensure_resources(cairo_surface_t *target)
|
||
|
{
|
||
|
cairo_t *cr;
|
||
|
int i, j;
|
||
|
|
||
|
if (source_surface != NULL)
|
||
|
return;
|
||
|
|
||
|
source_surface = cairo_surface_create_similar (target, CAIRO_CONTENT_COLOR_ALPHA,
|
||
|
16 * DIAMETER, 16 * DIAMETER);
|
||
|
cr = cairo_create(source_surface);
|
||
|
|
||
|
cairo_save(cr);
|
||
|
cairo_set_source_rgba(cr, 0, 0, 0, 0);
|
||
|
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
|
||
|
cairo_paint(cr);
|
||
|
cairo_restore(cr);
|
||
|
|
||
|
cairo_set_line_width(cr, 1.0);
|
||
|
|
||
|
for (j = 0; j < 16; j++)
|
||
|
for (i = 0; i < 16; i++)
|
||
|
{
|
||
|
cairo_set_source_rgba(cr,
|
||
|
((i * 41) % 16) / 15.,
|
||
|
((i * 31) % 16) / 15.,
|
||
|
((i * 23) % 16) / 15.,
|
||
|
0.25);
|
||
|
cairo_arc(cr,
|
||
|
i * DIAMETER + RADIUS, j * DIAMETER + RADIUS,
|
||
|
RADIUS - 0.5, 0, 2 * M_PI);
|
||
|
cairo_fill_preserve(cr);
|
||
|
cairo_set_source_rgba(cr,
|
||
|
((i * 41) % 16) / 15.,
|
||
|
((i * 31) % 16) / 15.,
|
||
|
((i * 23) % 16) / 15.,
|
||
|
1.0);
|
||
|
cairo_stroke(cr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_draw (GtkDrawingArea *da,
|
||
|
cairo_t *cr,
|
||
|
int width,
|
||
|
int height,
|
||
|
gpointer data)
|
||
|
|
||
|
{
|
||
|
GRand *rand = g_rand_new_with_seed(0);
|
||
|
int i;
|
||
|
|
||
|
ensure_resources (cairo_get_target (cr));
|
||
|
|
||
|
cairo_set_source_rgb(cr, 1, 1, 1);
|
||
|
cairo_paint(cr);
|
||
|
|
||
|
cairo_set_source_rgb(cr, 0, 0, 0);
|
||
|
cairo_set_line_width(cr, 1.0);
|
||
|
cairo_rectangle (cr, 0.5, 0.5, width - 1, height - 1);
|
||
|
cairo_stroke (cr);
|
||
|
|
||
|
for(i = 0; i < load_factor * 150; i++)
|
||
|
{
|
||
|
int source = g_rand_int_range(rand, 0, 255);
|
||
|
double phi = g_rand_double_range(rand, 0, 2 * M_PI) + angle;
|
||
|
double r = g_rand_double_range(rand, 0, width / 2 - RADIUS);
|
||
|
int x, y;
|
||
|
|
||
|
int source_x = (source % 16) * DIAMETER;
|
||
|
int source_y = (source / 16) * DIAMETER;
|
||
|
|
||
|
x = round(width / 2 + r * cos(phi) - RADIUS);
|
||
|
y = round(height / 2 - r * sin(phi) - RADIUS);
|
||
|
|
||
|
cairo_set_source_surface(cr, source_surface,
|
||
|
x - source_x, y - source_y);
|
||
|
cairo_rectangle(cr, x, y, DIAMETER, DIAMETER);
|
||
|
cairo_fill(cr);
|
||
|
}
|
||
|
|
||
|
g_rand_free(rand);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_frame (double progress)
|
||
|
{
|
||
|
int jitter;
|
||
|
|
||
|
angle = 2 * M_PI * progress;
|
||
|
jitter = WINDOW_SIZE_JITTER * sin(angle);
|
||
|
|
||
|
if (!cb_no_resize)
|
||
|
{
|
||
|
window_width = WIDTH + jitter;
|
||
|
window_height = HEIGHT + jitter;
|
||
|
}
|
||
|
|
||
|
gtk_window_set_default_size (GTK_WINDOW (window),
|
||
|
window_width, window_height);
|
||
|
|
||
|
gtk_widget_queue_draw (window);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
resize_idle (gpointer user_data)
|
||
|
{
|
||
|
GdkFrameClock *frame_clock = user_data;
|
||
|
gint64 frame_time = gdk_frame_clock_get_frame_time (frame_clock);
|
||
|
double scaled_time;
|
||
|
|
||
|
if (start_frame_time == 0)
|
||
|
start_frame_time = frame_time;
|
||
|
|
||
|
scaled_time = (frame_time - start_frame_time) / (CYCLE_TIME * 1000000);
|
||
|
on_frame (scaled_time - floor (scaled_time));
|
||
|
|
||
|
return G_SOURCE_REMOVE;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
tick_callback (GtkWidget *widget,
|
||
|
GdkFrameClock *frame_clock,
|
||
|
gpointer user_data)
|
||
|
{
|
||
|
g_idle_add (resize_idle, frame_clock);
|
||
|
|
||
|
return G_SOURCE_CONTINUE;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
on_map (GtkWidget *widget)
|
||
|
{
|
||
|
gtk_widget_add_tick_callback (window, tick_callback, NULL, NULL);
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
static GOptionEntry options[] = {
|
||
|
{ "factor", 'f', 0, G_OPTION_ARG_DOUBLE, &load_factor, "Load factor", "FACTOR" },
|
||
|
{ "no-resize", 'n', 0, G_OPTION_ARG_NONE, &cb_no_resize, "No Resize", NULL },
|
||
|
{ NULL }
|
||
|
};
|
||
|
|
||
|
static void
|
||
|
quit_cb (GtkWidget *widget,
|
||
|
gpointer data)
|
||
|
{
|
||
|
gboolean *done = data;
|
||
|
|
||
|
*done = TRUE;
|
||
|
|
||
|
g_main_context_wakeup (NULL);
|
||
|
}
|
||
|
|
||
|
int
|
||
|
main(int argc, char **argv)
|
||
|
{
|
||
|
GError *error = NULL;
|
||
|
GtkWidget *da;
|
||
|
gboolean done = FALSE;
|
||
|
|
||
|
GOptionContext *context = g_option_context_new (NULL);
|
||
|
g_option_context_add_main_entries (context, options, NULL);
|
||
|
frame_stats_add_options (g_option_context_get_main_group (context));
|
||
|
|
||
|
if (!g_option_context_parse (context, &argc, &argv, &error))
|
||
|
{
|
||
|
g_printerr ("Option parsing failed: %s\n", error->message);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
gtk_init ();
|
||
|
|
||
|
g_print ("# Load factor: %g\n",
|
||
|
load_factor);
|
||
|
g_print ("# Resizing?: %s\n",
|
||
|
cb_no_resize ? "no" : "yes");
|
||
|
|
||
|
window = gtk_window_new ();
|
||
|
frame_stats_ensure (GTK_WINDOW (window));
|
||
|
|
||
|
da = gtk_drawing_area_new ();
|
||
|
gtk_drawing_area_set_draw_func (GTK_DRAWING_AREA (da), on_draw, NULL, NULL);
|
||
|
gtk_window_set_child (GTK_WINDOW (window), da);
|
||
|
|
||
|
g_signal_connect (window, "destroy",
|
||
|
G_CALLBACK (quit_cb), NULL);
|
||
|
|
||
|
g_signal_connect (window, "map",
|
||
|
G_CALLBACK (on_map), NULL);
|
||
|
on_frame (0.);
|
||
|
|
||
|
gtk_window_present (GTK_WINDOW (window));
|
||
|
|
||
|
while (!done)
|
||
|
g_main_context_iteration (NULL, TRUE);
|
||
|
|
||
|
return 0;
|
||
|
}
|