Blame clutter-gtk/gtk-clutter-offscreen.c

Packit 1069cd
/*
Packit 1069cd
 * gtkclutteroffscreen.c
Packit 1069cd
 */
Packit 1069cd
Packit 1069cd
#include "config.h"
Packit 1069cd
Packit 1069cd
#include <gtk/gtk.h>
Packit 1069cd
Packit 1069cd
#include "gtk-clutter-embed.h"
Packit 1069cd
#include "gtk-clutter-offscreen.h"
Packit 1069cd
#include "gtk-clutter-actor-internal.h"
Packit 1069cd
Packit 1069cd
G_DEFINE_TYPE (GtkClutterOffscreen, _gtk_clutter_offscreen, GTK_TYPE_BIN);
Packit 1069cd
Packit 1069cd
void _gtk_clutter_embed_set_child_active (GtkClutterEmbed *embed,
Packit 1069cd
					  GtkWidget *child,
Packit 1069cd
					  gboolean active);
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_add (GtkContainer *container,
Packit 1069cd
                           GtkWidget    *child)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (container);
Packit 1069cd
Packit 1069cd
  GTK_CONTAINER_CLASS (_gtk_clutter_offscreen_parent_class)->add (container, child);
Packit 1069cd
Packit 1069cd
  if (offscreen->actor != NULL &&
Packit 1069cd
      clutter_actor_is_visible (offscreen->actor))
Packit 1069cd
    {
Packit 1069cd
      /* force a relayout */
Packit 1069cd
      clutter_actor_queue_relayout (offscreen->actor);
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_remove (GtkContainer *container,
Packit 1069cd
                              GtkWidget    *child)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (container);
Packit 1069cd
Packit 1069cd
  GTK_CONTAINER_CLASS (_gtk_clutter_offscreen_parent_class)->remove (container, child);
Packit 1069cd
Packit 1069cd
  if (offscreen->actor != NULL &&
Packit 1069cd
      clutter_actor_is_visible (offscreen->actor))
Packit 1069cd
    {
Packit 1069cd
      /* force a relayout */
Packit 1069cd
      clutter_actor_queue_relayout (offscreen->actor);
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_check_resize (GtkContainer *container)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (container);
Packit 1069cd
Packit 1069cd
  /* queue a relayout only if we're not in the middle of an
Packit 1069cd
   * allocation
Packit 1069cd
   */
Packit 1069cd
  if (offscreen->actor != NULL && !offscreen->in_allocation)
Packit 1069cd
    clutter_actor_queue_relayout (offscreen->actor);
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
offscreen_window_to_parent (GdkWindow           *offscreen_window,
Packit 1069cd
                            double               offscreen_x,
Packit 1069cd
                            double               offscreen_y,
Packit 1069cd
                            double              *parent_x,
Packit 1069cd
                            double              *parent_y,
Packit 1069cd
                            GtkClutterOffscreen *offscreen)
Packit 1069cd
{
Packit 1069cd
  ClutterVertex point, vertex;
Packit 1069cd
Packit 1069cd
  point.x = offscreen_x;
Packit 1069cd
  point.y = offscreen_y;
Packit 1069cd
  point.z = 0;
Packit 1069cd
  clutter_actor_apply_transform_to_point (offscreen->actor, &point, &vertex);
Packit 1069cd
  *parent_x = vertex.x;
Packit 1069cd
  *parent_y = vertex.y;
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
offscreen_window_from_parent (GdkWindow           *window,
Packit 1069cd
                              double               parent_x,
Packit 1069cd
                              double               parent_y,
Packit 1069cd
                              double              *offscreen_x,
Packit 1069cd
                              double              *offscreen_y,
Packit 1069cd
                              GtkClutterOffscreen *offscreen)
Packit 1069cd
{
Packit 1069cd
  gfloat x, y;
Packit 1069cd
Packit 1069cd
  if (clutter_actor_transform_stage_point (offscreen->actor,
Packit 1069cd
                                           parent_x,
Packit 1069cd
                                           parent_y,
Packit 1069cd
                                           &x, &y))
Packit 1069cd
    {
Packit 1069cd
      *offscreen_x = x;
Packit 1069cd
      *offscreen_y = y;
Packit 1069cd
    }
Packit 1069cd
  else
Packit 1069cd
    {
Packit 1069cd
      /* Could't transform. What the heck do we do now? */
Packit 1069cd
      *offscreen_x = parent_x;
Packit 1069cd
      *offscreen_y = parent_y;
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_realize (GtkWidget *widget)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (widget);
Packit 1069cd
  GtkAllocation allocation;
Packit 1069cd
  GtkStyleContext *style_context;
Packit 1069cd
  GdkWindow *window;
Packit 1069cd
  GdkWindowAttr attributes;
Packit 1069cd
  gint attributes_mask;
Packit 1069cd
  guint border_width;
Packit 1069cd
  GtkWidget *parent, *child;
Packit 1069cd
  GdkScreen *screen;
Packit 1069cd
Packit 1069cd
  gtk_widget_set_realized (widget, TRUE);
Packit 1069cd
Packit 1069cd
  border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
Packit 1069cd
Packit 1069cd
  gtk_widget_get_allocation (widget, &allocation);
Packit 1069cd
Packit 1069cd
  attributes.x = allocation.x + border_width;
Packit 1069cd
  attributes.y = allocation.y + border_width;
Packit 1069cd
  attributes.width = allocation.width - 2 * border_width;
Packit 1069cd
  attributes.height = allocation.height - 2 * border_width;
Packit 1069cd
  attributes.window_type = GDK_WINDOW_OFFSCREEN;
Packit 1069cd
  attributes.event_mask = gtk_widget_get_events (widget)
Packit 1069cd
                        | GDK_EXPOSURE_MASK;
Packit 1069cd
  attributes.visual = gtk_widget_get_visual (widget);
Packit 1069cd
  attributes.wclass = GDK_INPUT_OUTPUT;
Packit 1069cd
Packit 1069cd
  attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL;
Packit 1069cd
Packit 1069cd
  parent = gtk_widget_get_parent (widget);
Packit 1069cd
  screen = gtk_widget_get_screen (widget);
Packit 1069cd
Packit 1069cd
  window = gdk_window_new (gdk_screen_get_root_window (screen),
Packit 1069cd
			   &attributes,
Packit 1069cd
                           attributes_mask);
Packit 1069cd
  gtk_widget_set_window (widget, window);
Packit 1069cd
  gdk_window_set_user_data (window, widget);
Packit 1069cd
Packit 1069cd
  gdk_offscreen_window_set_embedder (gtk_widget_get_window (parent),
Packit 1069cd
                                     window);
Packit 1069cd
  g_signal_connect (window, "to-embedder",
Packit 1069cd
		    G_CALLBACK (offscreen_window_to_parent),
Packit 1069cd
                    widget);
Packit 1069cd
  g_signal_connect (window, "from-embedder",
Packit 1069cd
		    G_CALLBACK (offscreen_window_from_parent),
Packit 1069cd
                    widget);
Packit 1069cd
Packit 1069cd
  child = gtk_bin_get_child (GTK_BIN (widget));
Packit 1069cd
  if (child != NULL)
Packit 1069cd
    gtk_widget_set_parent_window (child, window);
Packit 1069cd
Packit 1069cd
  style_context = gtk_widget_get_style_context (widget);
Packit 1069cd
  gtk_style_context_set_background (style_context, window);
Packit 1069cd
Packit 1069cd
  if (offscreen->active)
Packit 1069cd
    _gtk_clutter_embed_set_child_active (GTK_CLUTTER_EMBED (parent),
Packit 1069cd
					 widget, TRUE);
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_unrealize (GtkWidget *widget)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (widget);
Packit 1069cd
Packit 1069cd
  if (offscreen->active)
Packit 1069cd
    _gtk_clutter_embed_set_child_active (GTK_CLUTTER_EMBED (gtk_widget_get_parent (widget)),
Packit 1069cd
					 widget, FALSE);
Packit 1069cd
Packit 1069cd
  GTK_WIDGET_CLASS (_gtk_clutter_offscreen_parent_class)->unrealize (widget);
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_get_preferred_width (GtkWidget *widget,
Packit 1069cd
                                           gint      *minimum,
Packit 1069cd
                                           gint      *natural)
Packit 1069cd
{
Packit 1069cd
  GtkBin *bin = GTK_BIN (widget);
Packit 1069cd
  GtkWidget *child;
Packit 1069cd
  gint border_width;
Packit 1069cd
Packit 1069cd
  border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
Packit 1069cd
Packit 1069cd
  *minimum = border_width * 2;
Packit 1069cd
  *natural = border_width * 2;
Packit 1069cd
Packit 1069cd
  child = gtk_bin_get_child (bin);
Packit 1069cd
Packit 1069cd
  if (child != NULL && gtk_widget_get_visible (child))
Packit 1069cd
    {
Packit 1069cd
      gint child_min, child_nat;
Packit 1069cd
Packit 1069cd
      gtk_widget_get_preferred_width (child, &child_min, &child_nat);
Packit 1069cd
Packit 1069cd
      *minimum += child_min;
Packit 1069cd
      *natural += child_nat;
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_get_preferred_height (GtkWidget *widget,
Packit 1069cd
                                            gint      *minimum,
Packit 1069cd
                                            gint      *natural)
Packit 1069cd
{
Packit 1069cd
  GtkBin *bin = GTK_BIN (widget);
Packit 1069cd
  GtkWidget *child;
Packit 1069cd
  gint border_width;
Packit 1069cd
Packit 1069cd
  border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
Packit 1069cd
Packit 1069cd
  *minimum = border_width * 2;
Packit 1069cd
  *natural = border_width * 2;
Packit 1069cd
Packit 1069cd
  child = gtk_bin_get_child (bin);
Packit 1069cd
Packit 1069cd
  if (child != NULL && gtk_widget_get_visible (child))
Packit 1069cd
    {
Packit 1069cd
      gint child_min, child_nat;
Packit 1069cd
Packit 1069cd
      gtk_widget_get_preferred_height (child, &child_min, &child_nat);
Packit 1069cd
Packit 1069cd
      *minimum += child_min;
Packit 1069cd
      *natural += child_nat;
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
gtk_clutter_offscreen_size_allocate (GtkWidget     *widget,
Packit 1069cd
				     GtkAllocation *allocation)
Packit 1069cd
{
Packit 1069cd
  GtkAllocation old_allocation;
Packit 1069cd
  GtkBin *bin = GTK_BIN (widget);
Packit 1069cd
  GtkWidget *child;
Packit 1069cd
  gint border_width;
Packit 1069cd
Packit 1069cd
  gtk_widget_get_allocation (widget, &old_allocation);
Packit 1069cd
Packit 1069cd
  /* some widgets call gtk_widget_queue_resize() which triggers a
Packit 1069cd
   * size request/allocate cycle.
Packit 1069cd
   *
Packit 1069cd
   * Calling gdk_window_move_resize() triggers an expose-event of the entire
Packit 1069cd
   * widget tree, so we only want to do it if the allocation has changed in
Packit 1069cd
   * some way, otherwise we can just ignore it.
Packit 1069cd
   */
Packit 1069cd
  if (gtk_widget_get_realized (widget) &&
Packit 1069cd
      (allocation->x != old_allocation.x ||
Packit 1069cd
       allocation->y != old_allocation.y ||
Packit 1069cd
       allocation->width != old_allocation.width ||
Packit 1069cd
       allocation->height != old_allocation.height))
Packit 1069cd
    {
Packit 1069cd
      gdk_window_move_resize (gtk_widget_get_window (widget),
Packit 1069cd
                              0, 0,
Packit 1069cd
                              allocation->width,
Packit 1069cd
                              allocation->height);
Packit 1069cd
    }
Packit 1069cd
Packit 1069cd
  gtk_widget_set_allocation (widget, allocation);
Packit 1069cd
Packit 1069cd
  border_width = gtk_container_get_border_width (GTK_CONTAINER (widget));
Packit 1069cd
Packit 1069cd
  child = gtk_bin_get_child (bin);
Packit 1069cd
Packit 1069cd
  if (child != NULL && gtk_widget_get_visible (child))
Packit 1069cd
    {
Packit 1069cd
      GtkAllocation  child_alloc;
Packit 1069cd
Packit 1069cd
      child_alloc.x = border_width;
Packit 1069cd
      child_alloc.y = border_width;
Packit 1069cd
      child_alloc.width = allocation->width - 2 * border_width;
Packit 1069cd
      child_alloc.height = allocation->height - 2 * border_width;
Packit 1069cd
Packit 1069cd
      gtk_widget_size_allocate (child, &child_alloc);
Packit 1069cd
    }
Packit 1069cd
Packit 1069cd
  gtk_widget_queue_draw (widget);
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static gboolean
Packit 1069cd
gtk_clutter_offscreen_damage (GtkWidget      *widget,
Packit 1069cd
			      GdkEventExpose *event)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen = GTK_CLUTTER_OFFSCREEN (widget);
Packit 1069cd
Packit 1069cd
  _gtk_clutter_actor_update (GTK_CLUTTER_ACTOR (offscreen->actor),
Packit 1069cd
                             event->area.x,
Packit 1069cd
                             event->area.y,
Packit 1069cd
                             event->area.width,
Packit 1069cd
                             event->area.height);
Packit 1069cd
Packit 1069cd
  return TRUE;
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
_gtk_clutter_offscreen_class_init (GtkClutterOffscreenClass *klass)
Packit 1069cd
{
Packit 1069cd
  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
Packit 1069cd
  GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
Packit 1069cd
Packit 1069cd
  widget_class->realize = gtk_clutter_offscreen_realize;
Packit 1069cd
  widget_class->unrealize = gtk_clutter_offscreen_unrealize;
Packit 1069cd
  widget_class->get_preferred_width = gtk_clutter_offscreen_get_preferred_width;
Packit 1069cd
  widget_class->get_preferred_height = gtk_clutter_offscreen_get_preferred_height;
Packit 1069cd
  widget_class->size_allocate = gtk_clutter_offscreen_size_allocate;
Packit 1069cd
Packit 1069cd
  container_class->add = gtk_clutter_offscreen_add;
Packit 1069cd
  container_class->remove = gtk_clutter_offscreen_remove;
Packit 1069cd
  container_class->check_resize = gtk_clutter_offscreen_check_resize;
Packit 1069cd
Packit 1069cd
  g_signal_override_class_handler ("damage-event",
Packit 1069cd
				   GTK_CLUTTER_TYPE_OFFSCREEN,
Packit 1069cd
				   G_CALLBACK (gtk_clutter_offscreen_damage));
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
static void
Packit 1069cd
_gtk_clutter_offscreen_init (GtkClutterOffscreen *offscreen)
Packit 1069cd
{
Packit 1069cd
  gtk_widget_set_has_window (GTK_WIDGET (offscreen), TRUE);
Packit 1069cd
Packit 1069cd
  G_GNUC_BEGIN_IGNORE_DEPRECATIONS
Packit 1069cd
  gtk_container_set_resize_mode (GTK_CONTAINER (offscreen), GTK_RESIZE_IMMEDIATE);
Packit 1069cd
  G_GNUC_END_IGNORE_DEPRECATIONS
Packit 1069cd
Packit 1069cd
  offscreen->active = TRUE;
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
GtkWidget *
Packit 1069cd
_gtk_clutter_offscreen_new (ClutterActor *actor)
Packit 1069cd
{
Packit 1069cd
  GtkClutterOffscreen *offscreen;
Packit 1069cd
Packit 1069cd
  offscreen = g_object_new (GTK_CLUTTER_TYPE_OFFSCREEN, NULL);
Packit 1069cd
  offscreen->actor = actor; /* Back pointer, actor owns widget */
Packit 1069cd
Packit 1069cd
  return GTK_WIDGET (offscreen);
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
void
Packit 1069cd
_gtk_clutter_offscreen_set_active (GtkClutterOffscreen *offscreen,
Packit 1069cd
                                   gboolean             active)
Packit 1069cd
{
Packit 1069cd
  GtkWidget *parent;
Packit 1069cd
Packit 1069cd
  active = !!active;
Packit 1069cd
Packit 1069cd
  if (offscreen->active != active)
Packit 1069cd
    {
Packit 1069cd
      offscreen->active = active;
Packit 1069cd
      parent = gtk_widget_get_parent (GTK_WIDGET (offscreen));
Packit 1069cd
      if (parent != NULL)
Packit 1069cd
	_gtk_clutter_embed_set_child_active (GTK_CLUTTER_EMBED (parent),
Packit 1069cd
					     GTK_WIDGET (offscreen),
Packit 1069cd
					     active);
Packit 1069cd
    }
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
void
Packit 1069cd
_gtk_clutter_offscreen_set_in_allocation (GtkClutterOffscreen *offscreen,
Packit 1069cd
                                          gboolean             in_allocation)
Packit 1069cd
{
Packit 1069cd
  in_allocation = !!in_allocation;
Packit 1069cd
Packit 1069cd
  offscreen->in_allocation = in_allocation;
Packit 1069cd
}
Packit 1069cd
Packit 1069cd
cairo_surface_t *
Packit 1069cd
_gtk_clutter_offscreen_get_surface (GtkClutterOffscreen *offscreen)
Packit 1069cd
{
Packit 1069cd
  return gdk_offscreen_window_get_surface (gtk_widget_get_window (GTK_WIDGET (offscreen)));
Packit 1069cd
}