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

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