Blame examples/threads.c

Packit Service bf98b9
#include <stdio.h>
Packit Service bf98b9
#include <stdlib.h>
Packit Service bf98b9
#include <clutter/clutter.h>
Packit Service bf98b9
Packit Service bf98b9
/* our thread-specific data */
Packit Service bf98b9
typedef struct
Packit Service bf98b9
{
Packit Service bf98b9
  ClutterActor *stage;
Packit Service bf98b9
  ClutterActor *label;
Packit Service bf98b9
  ClutterActor *progress;
Packit Service bf98b9
  ClutterActor *rect;
Packit Service bf98b9
Packit Service bf98b9
  ClutterTransition *flip;
Packit Service bf98b9
  ClutterTransition *bounce;
Packit Service bf98b9
} TestThreadData;
Packit Service bf98b9
Packit Service bf98b9
static TestThreadData *
Packit Service bf98b9
test_thread_data_new (void)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data;
Packit Service bf98b9
Packit Service bf98b9
  data = g_new0 (TestThreadData, 1);
Packit Service bf98b9
Packit Service bf98b9
  return data;
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static void
Packit Service bf98b9
test_thread_data_free (gpointer _data)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data = _data;
Packit Service bf98b9
Packit Service bf98b9
  if (data == NULL)
Packit Service bf98b9
    return;
Packit Service bf98b9
Packit Service bf98b9
  g_print ("Removing thread data [%p]\n", _data);
Packit Service bf98b9
Packit Service bf98b9
  g_clear_object (&data->progress);
Packit Service bf98b9
  g_clear_object (&data->label);
Packit Service bf98b9
  g_clear_object (&data->stage);
Packit Service bf98b9
  g_clear_object (&data->rect);
Packit Service bf98b9
  g_clear_object (&data->flip);
Packit Service bf98b9
  g_clear_object (&data->bounce);
Packit Service bf98b9
Packit Service bf98b9
  g_free (data);
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static gboolean
Packit Service bf98b9
test_thread_done_idle (gpointer user_data)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data = user_data;
Packit Service bf98b9
Packit Service bf98b9
  g_print ("Last update [%p]\n", data);
Packit Service bf98b9
Packit Service bf98b9
  clutter_text_set_text (CLUTTER_TEXT (data->label), "Completed");
Packit Service bf98b9
Packit Service bf98b9
  clutter_actor_remove_transition (data->rect, "bounce");
Packit Service bf98b9
  clutter_actor_remove_transition (data->rect, "flip");
Packit Service bf98b9
Packit Service bf98b9
  return G_SOURCE_REMOVE;
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static void
Packit Service bf98b9
test_thread_data_done (gpointer _data)
Packit Service bf98b9
{
Packit Service bf98b9
  if (_data == NULL)
Packit Service bf98b9
    return;
Packit Service bf98b9
Packit Service bf98b9
  g_print ("Thread completed\n");
Packit Service bf98b9
Packit Service bf98b9
  /* since the TestThreadData structure references Clutter data structures
Packit Service bf98b9
   * we need to free it from within the same thread that called clutter_main()
Packit Service bf98b9
   * which means using an idle handler in the main loop.
Packit Service bf98b9
   *
Packit Service bf98b9
   * clutter_threads_add_idle() is guaranteed to run the callback passed to
Packit Service bf98b9
   * to it under the Big Clutter Lock.
Packit Service bf98b9
   */
Packit Service bf98b9
  clutter_threads_add_idle_full (G_PRIORITY_DEFAULT,
Packit Service bf98b9
                                 test_thread_done_idle,
Packit Service bf98b9
                                 _data,
Packit Service bf98b9
                                 test_thread_data_free);
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
/* thread local storage */
Packit Service bf98b9
static GPrivate test_thread_data = G_PRIVATE_INIT (test_thread_data_done);
Packit Service bf98b9
Packit Service bf98b9
typedef struct {
Packit Service bf98b9
  gint count;
Packit Service bf98b9
  TestThreadData *thread_data;
Packit Service bf98b9
} TestUpdate;
Packit Service bf98b9
Packit Service bf98b9
static gboolean
Packit Service bf98b9
update_label_idle (gpointer data)
Packit Service bf98b9
{
Packit Service bf98b9
  TestUpdate *update = data;
Packit Service bf98b9
  guint width;
Packit Service bf98b9
  gchar *text;
Packit Service bf98b9
Packit Service bf98b9
  if (update->thread_data->label == NULL)
Packit Service bf98b9
    return G_SOURCE_REMOVE;
Packit Service bf98b9
Packit Service bf98b9
  text = g_strdup_printf ("Count to %d", update->count);
Packit Service bf98b9
  clutter_text_set_text (CLUTTER_TEXT (update->thread_data->label), text);
Packit Service bf98b9
Packit Service bf98b9
  clutter_actor_set_width (update->thread_data->label, -1);
Packit Service bf98b9
Packit Service bf98b9
  if (update->count == 0)
Packit Service bf98b9
    width = 0;
Packit Service bf98b9
  else if (update->count == 100)
Packit Service bf98b9
    width = 350;
Packit Service bf98b9
  else
Packit Service bf98b9
    width = (guint) (update->count / 100.0 * 350.0);
Packit Service bf98b9
Packit Service bf98b9
  clutter_actor_save_easing_state (update->thread_data->progress);
Packit Service bf98b9
  clutter_actor_set_width (update->thread_data->progress, width);
Packit Service bf98b9
  clutter_actor_restore_easing_state (update->thread_data->progress);
Packit Service bf98b9
Packit Service bf98b9
  g_free (text);
Packit Service bf98b9
  g_free (update);
Packit Service bf98b9
Packit Service bf98b9
  return G_SOURCE_REMOVE;
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static void
Packit Service bf98b9
do_something_very_slow (void)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data;
Packit Service bf98b9
  gint i;
Packit Service bf98b9
Packit Service bf98b9
  data = g_private_get (&test_thread_data);
Packit Service bf98b9
Packit Service bf98b9
  for (i = 0; i <= 100; i++)
Packit Service bf98b9
    {
Packit Service bf98b9
      gint msecs;
Packit Service bf98b9
Packit Service bf98b9
      msecs = 1 + (int) (100.0 * rand () / ((RAND_MAX + 1.0) / 3));
Packit Service bf98b9
Packit Service bf98b9
      /* sleep for a while, to emulate some work being done */
Packit Service bf98b9
      g_usleep (msecs * 1000);
Packit Service bf98b9
Packit Service bf98b9
      if ((i % 10) == 0)
Packit Service bf98b9
        {
Packit Service bf98b9
          TestUpdate *update;
Packit Service bf98b9
Packit Service bf98b9
          /* update the UI from within the main loop, making sure that the
Packit Service bf98b9
           * Big Clutter Lock is held; only one thread at a time can call
Packit Service bf98b9
           * Clutter API, and it's mandatory to do this from the same thread
Packit Service bf98b9
           * that called clutter_init()/clutter_main().
Packit Service bf98b9
           */
Packit Service bf98b9
          update = g_new (TestUpdate, 1);
Packit Service bf98b9
          update->count = i;
Packit Service bf98b9
          update->thread_data = data;
Packit Service bf98b9
Packit Service bf98b9
          clutter_threads_add_idle_full (G_PRIORITY_HIGH,
Packit Service bf98b9
                                         update_label_idle,
Packit Service bf98b9
                                         update, NULL);
Packit Service bf98b9
        }
Packit Service bf98b9
    }
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static gpointer
Packit Service bf98b9
test_thread_func (gpointer user_data)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data = user_data;
Packit Service bf98b9
Packit Service bf98b9
  g_private_set (&test_thread_data, data);
Packit Service bf98b9
Packit Service bf98b9
  /* this function will block */
Packit Service bf98b9
  do_something_very_slow ();
Packit Service bf98b9
Packit Service bf98b9
  return NULL;
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
static ClutterActor *count_label   = NULL;
Packit Service bf98b9
static ClutterActor *help_label    = NULL;
Packit Service bf98b9
static ClutterActor *progress_rect = NULL;
Packit Service bf98b9
static ClutterActor *rect          = NULL;
Packit Service bf98b9
static ClutterTransition *flip     = NULL;
Packit Service bf98b9
static ClutterTransition *bounce   = NULL;
Packit Service bf98b9
Packit Service bf98b9
static gboolean
Packit Service bf98b9
on_key_press_event (ClutterStage *stage,
Packit Service bf98b9
                    ClutterEvent *event,
Packit Service bf98b9
                    gpointer      user_data)
Packit Service bf98b9
{
Packit Service bf98b9
  TestThreadData *data;
Packit Service bf98b9
Packit Service bf98b9
  switch (clutter_event_get_key_symbol (event))
Packit Service bf98b9
    {
Packit Service bf98b9
    case CLUTTER_KEY_s:
Packit Service bf98b9
      clutter_text_set_text (CLUTTER_TEXT (help_label), "Press 'q' to quit");
Packit Service bf98b9
Packit Service bf98b9
      /* start the animations */
Packit Service bf98b9
      clutter_actor_add_transition (rect, "flip", flip);
Packit Service bf98b9
      clutter_actor_add_transition (rect, "bounce", bounce);
Packit Service bf98b9
Packit Service bf98b9
      /* the data structure holding all our objects */
Packit Service bf98b9
      data = test_thread_data_new ();
Packit Service bf98b9
      data->stage = g_object_ref (stage);
Packit Service bf98b9
      data->label = g_object_ref (count_label);
Packit Service bf98b9
      data->progress = g_object_ref (progress_rect);
Packit Service bf98b9
      data->rect = g_object_ref (rect);
Packit Service bf98b9
      data->flip = g_object_ref (flip);
Packit Service bf98b9
      data->bounce = g_object_ref (bounce);
Packit Service bf98b9
Packit Service bf98b9
      /* start the thread that updates the counter and the progress bar */
Packit Service bf98b9
      g_thread_new ("counter", test_thread_func, data);
Packit Service bf98b9
Packit Service bf98b9
      return CLUTTER_EVENT_STOP;
Packit Service bf98b9
Packit Service bf98b9
    case CLUTTER_KEY_q:
Packit Service bf98b9
      clutter_main_quit ();
Packit Service bf98b9
Packit Service bf98b9
      return CLUTTER_EVENT_STOP;
Packit Service bf98b9
Packit Service bf98b9
    default:
Packit Service bf98b9
      break;
Packit Service bf98b9
    }
Packit Service bf98b9
Packit Service bf98b9
  return CLUTTER_EVENT_PROPAGATE;
Packit Service bf98b9
}
Packit Service bf98b9
Packit Service bf98b9
int
Packit Service bf98b9
main (int argc, char *argv[])
Packit Service bf98b9
{
Packit Service bf98b9
  ClutterTransition *transition;
Packit Service bf98b9
  ClutterActor *stage;
Packit Service bf98b9
  ClutterPoint start = CLUTTER_POINT_INIT (75.f, 150.f);
Packit Service bf98b9
  ClutterPoint end = CLUTTER_POINT_INIT (400.f, 150.f);
Packit Service bf98b9
Packit Service bf98b9
  if (clutter_init (&argc, &argv) != CLUTTER_INIT_SUCCESS)
Packit Service bf98b9
    return 1;
Packit Service bf98b9
Packit Service bf98b9
  stage = clutter_stage_new ();
Packit Service bf98b9
  clutter_stage_set_title (CLUTTER_STAGE (stage), "Threading");
Packit Service bf98b9
  clutter_actor_set_background_color (stage, CLUTTER_COLOR_Aluminium3);
Packit Service bf98b9
  clutter_actor_set_size (stage, 600, 300);
Packit Service bf98b9
  g_signal_connect (stage, "destroy", G_CALLBACK (clutter_main_quit), NULL);
Packit Service bf98b9
  
Packit Service bf98b9
  count_label = clutter_text_new_with_text ("Mono 12", "Counter");
Packit Service bf98b9
  clutter_actor_set_position (count_label, 350, 50);
Packit Service bf98b9
  clutter_actor_add_child (stage, count_label);
Packit Service bf98b9
Packit Service bf98b9
  help_label = clutter_text_new_with_text ("Mono 12", "Press 's' to start");
Packit Service bf98b9
  clutter_actor_set_position (help_label, 50, 50);
Packit Service bf98b9
  clutter_actor_add_child (stage, help_label);
Packit Service bf98b9
Packit Service bf98b9
  /* a progress bar */
Packit Service bf98b9
  progress_rect = clutter_actor_new ();
Packit Service bf98b9
  clutter_actor_set_background_color (progress_rect, CLUTTER_COLOR_DarkChameleon);
Packit Service bf98b9
  clutter_actor_set_position (progress_rect, 50, 225);
Packit Service bf98b9
  clutter_actor_set_size (progress_rect, 350, 50);
Packit Service bf98b9
  clutter_actor_add_child (stage, progress_rect);
Packit Service bf98b9
Packit Service bf98b9
  /* an actor we bounce around */
Packit Service bf98b9
  rect = clutter_actor_new ();
Packit Service bf98b9
  clutter_actor_set_background_color (rect, CLUTTER_COLOR_LightScarletRed);
Packit Service bf98b9
  clutter_actor_set_position (rect, 75, 150);
Packit Service bf98b9
  clutter_actor_set_size (rect, 50, 50);
Packit Service bf98b9
  clutter_actor_set_pivot_point (rect, .5f, .5f);
Packit Service bf98b9
  clutter_actor_set_opacity (rect, 224);
Packit Service bf98b9
  clutter_actor_add_child (stage, rect);
Packit Service bf98b9
Packit Service bf98b9
  /* two transitions we use to bounce rect around */
Packit Service bf98b9
  transition = clutter_property_transition_new ("rotation-angle-z");
Packit Service bf98b9
  clutter_transition_set_from (transition, G_TYPE_DOUBLE, 0.0);
Packit Service bf98b9
  clutter_transition_set_to (transition, G_TYPE_DOUBLE, 360.0);
Packit Service bf98b9
  clutter_timeline_set_repeat_count (CLUTTER_TIMELINE (transition), -1);
Packit Service bf98b9
  clutter_timeline_set_auto_reverse (CLUTTER_TIMELINE (transition), TRUE);
Packit Service bf98b9
  clutter_timeline_set_duration (CLUTTER_TIMELINE (transition), 3000);
Packit Service bf98b9
  flip = transition;
Packit Service bf98b9
Packit Service bf98b9
  transition = clutter_property_transition_new ("position");
Packit Service bf98b9
  clutter_transition_set_from (transition, CLUTTER_TYPE_POINT, &start;;
Packit Service bf98b9
  clutter_transition_set_to (transition, CLUTTER_TYPE_POINT, &end;;
Packit Service bf98b9
  clutter_timeline_set_repeat_count (CLUTTER_TIMELINE (transition), -1);
Packit Service bf98b9
  clutter_timeline_set_auto_reverse (CLUTTER_TIMELINE (transition), TRUE);
Packit Service bf98b9
  clutter_timeline_set_duration (CLUTTER_TIMELINE (transition), 3000);
Packit Service bf98b9
  bounce = transition;
Packit Service bf98b9
Packit Service bf98b9
  g_signal_connect (stage,
Packit Service bf98b9
                    "button-press-event", G_CALLBACK (clutter_main_quit),
Packit Service bf98b9
                    NULL);
Packit Service bf98b9
  g_signal_connect (stage,
Packit Service bf98b9
                    "key-press-event", G_CALLBACK (on_key_press_event),
Packit Service bf98b9
                    NULL);
Packit Service bf98b9
Packit Service bf98b9
  clutter_actor_show (stage);
Packit Service bf98b9
Packit Service bf98b9
  clutter_main ();
Packit Service bf98b9
Packit Service bf98b9
  return EXIT_SUCCESS;
Packit Service bf98b9
}