Blob Blame History Raw
/* dzl-dock-overlay.c
 *
 * Copyright (C) 2016 Christian Hergert <chergert@redhat.com>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

#define G_LOG_DOMAIN "dzl-dock-overlay"

#include "config.h"

#include "animation/dzl-animation.h"
#include "panel/dzl-dock-overlay-edge.h"
#include "panel/dzl-dock-item.h"
#include "panel/dzl-dock-overlay.h"
#include "panel/dzl-tab.h"
#include "panel/dzl-tab-strip.h"
#include "util/dzl-util-private.h"

#define MNEMONIC_REVEAL_DURATION 200

typedef struct
{
  GtkOverlay         *overlay;
  DzlDockOverlayEdge *edges [4];
  GtkAdjustment      *edge_adj [4];
  GtkAdjustment      *edge_handle_adj [4];
  GtkAllocation       hover_borders [4];
  guint               child_reveal : 4;
  guint               child_revealed : 4;
  guint               child_transient : 4;
} DzlDockOverlayPrivate;

static void dzl_dock_overlay_init_dock_iface      (DzlDockInterface     *iface);
static void dzl_dock_overlay_init_dock_item_iface (DzlDockItemInterface *iface);
static void dzl_dock_overlay_init_buildable_iface (GtkBuildableIface    *iface);
static void dzl_dock_overlay_set_child_reveal     (DzlDockOverlay       *self,
                                                   GtkWidget            *child,
                                                   gboolean              reveal);

G_DEFINE_TYPE_EXTENDED (DzlDockOverlay, dzl_dock_overlay, GTK_TYPE_EVENT_BOX, 0,
                        G_ADD_PRIVATE (DzlDockOverlay)
                        G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE, dzl_dock_overlay_init_buildable_iface)
                        G_IMPLEMENT_INTERFACE (DZL_TYPE_DOCK_ITEM, dzl_dock_overlay_init_dock_item_iface)
                        G_IMPLEMENT_INTERFACE (DZL_TYPE_DOCK, dzl_dock_overlay_init_dock_iface))

enum {
  PROP_0,
  PROP_MANAGER,
  N_PROPS
};

enum {
  CHILD_PROP_0,
  CHILD_PROP_REVEAL,
  CHILD_PROP_REVEALED,
  N_CHILD_PROPS
};

enum {
  HIDE_EDGES,
  N_SIGNALS
};

static GParamSpec *child_properties [N_CHILD_PROPS];
static guint signals [N_SIGNALS];

static void
dzl_dock_overlay_update_focus_chain (DzlDockOverlay *self)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  GList *focus_chain = NULL;
  GtkWidget *child;
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));

  for (i = G_N_ELEMENTS (priv->edges); i > 0; i--)
    {
      DzlDockOverlayEdge *edge = priv->edges [i - 1];

      if (edge != NULL)
        focus_chain = g_list_prepend (focus_chain, edge);
    }

  child = gtk_bin_get_child (GTK_BIN (self));

  if (child != NULL)
    focus_chain = g_list_prepend (focus_chain, child);

  if (focus_chain != NULL)
    {
      gtk_container_set_focus_chain (GTK_CONTAINER (self), focus_chain);
      g_list_free (focus_chain);
    }
}

static void
dzl_dock_overlay_get_edge_position (DzlDockOverlay     *self,
                                    DzlDockOverlayEdge *edge,
                                    GtkAllocation      *allocation)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  GtkPositionType type;
  gdouble value;
  gdouble handle_value;
  gdouble flipped_value;
  gint nat_width;
  gint nat_height;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (DZL_IS_DOCK_OVERLAY_EDGE (edge));
  g_assert (allocation != NULL);

  gtk_widget_get_allocation (GTK_WIDGET (self), allocation);

  allocation->x = 0;
  allocation->y = 0;

  type = dzl_dock_overlay_edge_get_edge (edge);

  if (type == GTK_POS_LEFT || type == GTK_POS_RIGHT)
    {
      nat_height = MAX (allocation->height, 1);
      gtk_widget_get_preferred_width_for_height (GTK_WIDGET (edge), nat_height, NULL, &nat_width);
    }
  else if (type == GTK_POS_TOP || type == GTK_POS_BOTTOM)
    {
      nat_width = MAX (allocation->width, 1);
      gtk_widget_get_preferred_height_for_width (GTK_WIDGET (edge), nat_width, NULL, &nat_height);
    }
  else
    {
      g_assert_not_reached ();
      return;
    }

  value = gtk_adjustment_get_value (priv->edge_adj [type]);
  flipped_value = 1.0 - value;

  handle_value = gtk_adjustment_get_value (priv->edge_handle_adj [type]);

  switch (type)
    {
    case GTK_POS_LEFT:
      allocation->width = nat_width;
      allocation->x -= nat_width * value;
      if (flipped_value * nat_width <= handle_value)
        allocation->x += (handle_value - (flipped_value * nat_width));
      break;

    case GTK_POS_RIGHT:
      allocation->x = allocation->x + allocation->width - nat_width;
      allocation->width = nat_width;
      allocation->x += nat_width * value;
      if (flipped_value * nat_width <= handle_value)
        allocation->x -= (handle_value - (flipped_value * nat_width));
      break;

    case GTK_POS_BOTTOM:
      allocation->y = allocation->y + allocation->height - nat_height;
      allocation->height = nat_height;
      allocation->y += nat_height * value;
      if (flipped_value * nat_height <= handle_value)
        allocation->y -= (handle_value - (flipped_value * nat_height));
      break;

    case GTK_POS_TOP:
      allocation->height = nat_height;
      allocation->y -= nat_height * value;
      if (flipped_value * nat_height <= handle_value)
        allocation->y += (handle_value - (flipped_value * nat_height));
      break;

    default:
      g_assert_not_reached ();
    }
}

static gboolean
dzl_dock_overlay_get_child_position (DzlDockOverlay *self,
                                     GtkWidget      *widget,
                                     GtkAllocation  *allocation)
{
  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (widget));
  g_assert (allocation != NULL);

  if (DZL_IS_DOCK_OVERLAY_EDGE (widget))
    {
      dzl_dock_overlay_get_edge_position (self, DZL_DOCK_OVERLAY_EDGE (widget), allocation);
      return TRUE;
    }

  return FALSE;
}

static void
dzl_dock_overlay_add (GtkContainer *container,
                      GtkWidget    *widget)
{
  DzlDockOverlay *self = (DzlDockOverlay *)container;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (widget));

  gtk_container_add (GTK_CONTAINER (priv->overlay), widget);

  dzl_dock_overlay_update_focus_chain (self);

  if (DZL_IS_DOCK_ITEM (widget))
    {
      dzl_dock_item_adopt (DZL_DOCK_ITEM (self), DZL_DOCK_ITEM (widget));
      dzl_dock_item_update_visibility (DZL_DOCK_ITEM (widget));
    }
}

static void
dzl_dock_overlay_toplevel_mnemonics (DzlDockOverlay *self,
                                     GParamSpec     *pspec,
                                     GtkWindow      *toplevel)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  const gchar *style_prop;
  gboolean mnemonics_visible;
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (pspec != NULL);
  g_assert (GTK_IS_WINDOW (toplevel));

  mnemonics_visible = gtk_window_get_mnemonics_visible (toplevel);
  style_prop = mnemonics_visible ? "mnemonic-overlap-size" : "overlap-size";

  for (i = 0; i < G_N_ELEMENTS (priv->edges); i++)
    {
      DzlDockOverlayEdge *edge = priv->edges [i];
      GtkAdjustment *handle_adj = priv->edge_handle_adj [i];
      gint overlap = 0;

      gtk_widget_style_get (GTK_WIDGET (edge), style_prop, &overlap, NULL);

      dzl_object_animate (handle_adj,
                          DZL_ANIMATION_EASE_IN_OUT_CUBIC,
                          MNEMONIC_REVEAL_DURATION,
                          gtk_widget_get_frame_clock (GTK_WIDGET (self)),
                          "value", (gdouble)overlap,
                          NULL);
    }

  gtk_widget_queue_allocate (GTK_WIDGET (self));
}

typedef struct
{
  DzlDockOverlay     *self;
  DzlDockOverlayEdge *edge;
  GtkWidget          *current_grab;
  gboolean            result;
} ForallState;

/* Same as gtk_widget_is_ancestor but take care of
 * following the popovers relative-to links.
 */
static gboolean
dzl_overlay_dock_widget_is_ancestor (GtkWidget *widget,
                                     GtkWidget *ancestor)
{
  GtkWidget *parent;

  g_assert (GTK_IS_WIDGET (widget));
  g_assert (GTK_IS_WIDGET (ancestor));

  while (widget != NULL)
    {
      if (GTK_IS_POPOVER (widget))
        {
          if (NULL == (widget = gtk_popover_get_relative_to (GTK_POPOVER (widget))))
            return FALSE;

          if (widget == ancestor)
            return TRUE;
        }

      parent = gtk_widget_get_parent (widget);
      if (parent == ancestor)
        return TRUE;

      widget = parent;
    }

  return FALSE;
}

static void
dzl_overlay_container_forall_cb (GtkWidget *widget,
                                 gpointer   user_data)
{
  ForallState *state = (ForallState *)user_data;

  if (state->result == TRUE)
    return;

  if (GTK_IS_POPOVER (widget) &&
      gtk_widget_is_visible (widget) &&
      state->current_grab == widget &&
      dzl_overlay_dock_widget_is_ancestor (widget, GTK_WIDGET (state->edge)))
    state->result = TRUE;
}

static gboolean
dzl_dock_overlay_edge_need_to_close (DzlDockOverlay     *self,
                                     DzlDockOverlayEdge *edge,
                                     GtkWidget          *focus)
{
  GtkWidget *toplevel;
  GtkWidget *current_grab;
  GtkWidget *current_focus;
  gboolean result = TRUE;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (DZL_IS_DOCK_OVERLAY_EDGE (edge));
  g_assert (focus == NULL || GTK_IS_WIDGET (focus));

  if (focus != NULL)
    return !dzl_overlay_dock_widget_is_ancestor (focus, GTK_WIDGET (edge));

  toplevel = gtk_widget_get_toplevel (GTK_WIDGET (edge));
  current_grab = gtk_grab_get_current ();
  if (current_grab != NULL)
    {
      if (GTK_IS_WINDOW (toplevel))
        {
          ForallState state = {self, edge, current_grab, FALSE};

          gtk_container_forall (GTK_CONTAINER (toplevel), dzl_overlay_container_forall_cb, &state);
          result = !state.result;
        }
    }
  else
    {
      if (GTK_IS_WINDOW (toplevel) &&
          NULL != (current_focus = gtk_window_get_focus (GTK_WINDOW (toplevel))))
        result = !dzl_overlay_dock_widget_is_ancestor (current_focus, GTK_WIDGET (edge));
    }

  return result;
}

static void
dzl_dock_overlay_toplevel_set_focus (DzlDockOverlay *self,
                                     GtkWidget      *widget,
                                     GtkWindow      *toplevel)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (!widget || GTK_IS_WIDGET (widget));
  g_assert (GTK_IS_WINDOW (toplevel));

  /*
   * TODO: If the overlay obscurs the new focus widget,
   *       hide immediately. Otherwise, use a short timeout.
   */
  for (i = 0; i < G_N_ELEMENTS (priv->edges); i++)
    {
      DzlDockOverlayEdge *edge = priv->edges [i];

      if (!!(priv->child_reveal & (1 << i)) &&
          dzl_dock_overlay_edge_need_to_close (self, edge, widget))
        {
          gtk_container_child_set (GTK_CONTAINER (self), GTK_WIDGET (edge),
                                   "reveal", FALSE,
                                   NULL);
        }
    }
}

static void
dzl_dock_overlay_hide_edges (DzlDockOverlay *self)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  GtkWidget *child;
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));

  for (i = 0; i < G_N_ELEMENTS (priv->edges); i++)
    {
      DzlDockOverlayEdge *edge = priv->edges [i];

      gtk_container_child_set (GTK_CONTAINER (self), GTK_WIDGET (edge),
                               "reveal", FALSE,
                               NULL);
    }

  child = gtk_bin_get_child (GTK_BIN (self));

  if (child != NULL)
    gtk_widget_grab_focus (child);
}

static void
dzl_dock_overlay_destroy (GtkWidget *widget)
{
  DzlDockOverlay *self = (DzlDockOverlay *)widget;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  guint i;

  g_assert (GTK_IS_WIDGET (widget));

  for (i = 0; i < G_N_ELEMENTS (priv->edge_adj); i++)
    g_clear_object (&priv->edge_adj [i]);

  GTK_WIDGET_CLASS (dzl_dock_overlay_parent_class)->destroy (widget);
}

static void
dzl_dock_overlay_hierarchy_changed (GtkWidget *widget,
                                    GtkWidget *old_toplevel)
{
  DzlDockOverlay *self = (DzlDockOverlay *)widget;
  GtkWidget *toplevel;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (!old_toplevel || GTK_IS_WIDGET (old_toplevel));

  if (old_toplevel != NULL)
    {
      g_signal_handlers_disconnect_by_func (old_toplevel,
                                            G_CALLBACK (dzl_dock_overlay_toplevel_mnemonics),
                                            self);
      g_signal_handlers_disconnect_by_func (old_toplevel,
                                            G_CALLBACK (dzl_dock_overlay_toplevel_set_focus),
                                            self);
    }

  toplevel = gtk_widget_get_toplevel (GTK_WIDGET (self));

  if (GTK_IS_WINDOW (toplevel))
    {
      g_signal_connect_object (toplevel,
                               "notify::mnemonics-visible",
                               G_CALLBACK (dzl_dock_overlay_toplevel_mnemonics),
                               self,
                               G_CONNECT_SWAPPED);

      g_signal_connect_object (toplevel,
                               "set-focus",
                               G_CALLBACK (dzl_dock_overlay_toplevel_set_focus),
                               self,
                               G_CONNECT_SWAPPED);
    }
}

static gboolean
dzl_dock_overlay_get_child_reveal (DzlDockOverlay *self,
                                   GtkWidget      *child)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (child));

  if (DZL_IS_DOCK_OVERLAY_EDGE (child))
    {
      GtkPositionType edge;

      edge = dzl_dock_overlay_edge_get_edge (DZL_DOCK_OVERLAY_EDGE (child));

      return !!(priv->child_reveal & (1 << edge));
    }

  return FALSE;
}

static gboolean
dzl_dock_overlay_get_child_revealed (DzlDockOverlay *self,
                                     GtkWidget      *child)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (child));

  if (DZL_IS_DOCK_OVERLAY_EDGE (child))
    {
      GtkPositionType edge;

      edge = dzl_dock_overlay_edge_get_edge (DZL_DOCK_OVERLAY_EDGE (child));

      return !!(priv->child_revealed & (1 << edge));
    }

  return FALSE;
}

typedef struct
{
  DzlDockOverlay  *self;
  GtkWidget       *child;
  GtkPositionType  edge : 2;
  guint            revealing : 1;
} ChildRevealState;

static void
child_reveal_state_free (gpointer data)
{
  ChildRevealState *state = (ChildRevealState *)data;

  g_object_unref (state->self);
  g_object_unref (state->child);

  g_slice_free (ChildRevealState, state);
}

static void
dzl_dock_overlay_child_reveal_done (gpointer user_data)
{
  ChildRevealState *state = (ChildRevealState *)user_data;
  DzlDockOverlay *self = state->self;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (state->child));

  if (state->revealing)
    priv->child_revealed = priv->child_revealed | (1 << state->edge);
  else
    priv->child_revealed = priv->child_revealed & ~(1 << state->edge);

  gtk_container_child_notify_by_pspec (GTK_CONTAINER (self),
                                       state->child,
                                       child_properties [CHILD_PROP_REVEALED]);

  child_reveal_state_free (state);
}

static void
dzl_dock_overlay_set_child_reveal (DzlDockOverlay *self,
                                   GtkWidget      *child,
                                   gboolean        reveal)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  ChildRevealState *state;
  GtkPositionType edge;
  guint child_reveal;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (GTK_IS_WIDGET (child));

  if (!DZL_IS_DOCK_OVERLAY_EDGE (child))
    return;

  edge = dzl_dock_overlay_edge_get_edge (DZL_DOCK_OVERLAY_EDGE (child));

  if (reveal)
    child_reveal = priv->child_reveal | (1 << edge);
  else
    child_reveal = priv->child_reveal & ~(1 << edge);

  if (priv->child_reveal != child_reveal)
    {
      GtkAllocation alloc;
      GdkMonitor *monitor;
      GdkWindow *window;
      guint duration = 0;

      state = g_slice_new0 (ChildRevealState);
      state->self = g_object_ref (self);
      state->child = g_object_ref (child);
      state->edge = edge;
      state->revealing = !!reveal;

      priv->child_reveal = child_reveal;

      window = gtk_widget_get_window (GTK_WIDGET (self));

      if (window != NULL)
        {
          GdkDisplay *display = gtk_widget_get_display (child);

          monitor = gdk_display_get_monitor_at_window (display, window);

          gtk_widget_get_allocation (child, &alloc);

          if (edge == GTK_POS_LEFT || edge == GTK_POS_RIGHT)
            duration = dzl_animation_calculate_duration (monitor, 0, alloc.width);
          else
            duration = dzl_animation_calculate_duration (monitor, 0, alloc.height);
        }

#if 0
      g_print ("Animating %s %d msec (currently at value=%lf)\n",
               reveal ? "in" : "out",
               duration,
               gtk_adjustment_get_value (priv->edge_adj [edge]));
#endif

      dzl_object_animate_full (priv->edge_adj [edge],
                               DZL_ANIMATION_EASE_IN_OUT_CUBIC,
                               duration,
                               gtk_widget_get_frame_clock (child),
                               dzl_dock_overlay_child_reveal_done,
                               state,
                               "value", reveal ? 0.0 : 1.0,
                               NULL);

      gtk_container_child_notify_by_pspec (GTK_CONTAINER (self),
                                           child,
                                           child_properties [CHILD_PROP_REVEAL]);
    }
}

static gboolean
widget_descendant_contains_focus (GtkWidget *widget)
{
  GtkWidget *toplevel;

  g_assert (GTK_IS_WIDGET (widget));

  toplevel = gtk_widget_get_toplevel (widget);

  if (GTK_IS_WINDOW (toplevel))
    {
      GtkWidget *focus = gtk_window_get_focus (GTK_WINDOW (toplevel));

      if (focus != NULL)
        return gtk_widget_is_ancestor (focus, widget);
    }

  return FALSE;
}

static inline gboolean
rectangle_contains_point (const GdkRectangle *a, gint x, gint y)
{
  return x >= a->x &&
         x <= (a->x + a->width) &&
         y >= a->y &&
         y <= (a->y + a->height);
}

static gboolean
dzl_dock_overlay_motion_notify_event (GtkWidget      *widget,
                                      GdkEventMotion *event)
{
  DzlDockOverlay *self = (DzlDockOverlay *)widget;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  GdkWindow *iter;
  GdkWindow *window;
  gdouble x, y;
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (event != NULL);

  window = gtk_widget_get_window (widget);

  x = event->x;
  y = event->y;

  for (iter = event->window; iter != window; iter = gdk_window_get_parent (iter))
    gdk_window_coords_to_parent (iter, x, y, &x, &y);

  for (i = 0; i < G_N_ELEMENTS (priv->hover_borders); i++)
    {
      DzlDockOverlayEdge *edge = priv->edges [i];
      GtkPositionType edge_type = dzl_dock_overlay_edge_get_position (edge);
      GtkAllocation *hover_border = &priv->hover_borders [i];
      guint mask = 1 << edge_type;

      if (rectangle_contains_point (hover_border, x, y))
        {
          /* Ignore this edge if it is already revealing */
          if (dzl_dock_overlay_get_child_reveal (self, GTK_WIDGET (edge)) ||
              dzl_dock_overlay_get_child_revealed (self, GTK_WIDGET (edge)))
            continue;

          dzl_dock_overlay_set_child_reveal (self, GTK_WIDGET (edge), TRUE);

          priv->child_transient |= mask;
        }
      else if ((priv->child_transient & mask) != 0)
        {
          GtkWidget *event_widget = NULL;
          GtkAllocation alloc;
          gint rel_x;
          gint rel_y;

          gdk_window_get_user_data (event->window, (gpointer *)&event_widget);
          gtk_widget_get_allocation (GTK_WIDGET (edge), &alloc);
          gtk_widget_translate_coordinates (event_widget,
                                            GTK_WIDGET (edge),
                                            event->x,
                                            event->y,
                                            &rel_x,
                                            &rel_y);

          /*
           * If this edge is transient, and the event window is not a
           * descendant of the edges window, then we should dismiss the
           * transient state.
           */
          if (dzl_dock_overlay_get_child_revealed (self, GTK_WIDGET (edge)) &&
              !rectangle_contains_point (&alloc, rel_x, rel_y) &&
              !widget_descendant_contains_focus (GTK_WIDGET (edge)))
            {
              dzl_dock_overlay_set_child_reveal (self, GTK_WIDGET (edge), FALSE);
              priv->child_transient &= ~mask;
            }
        }
    }

  return GTK_WIDGET_CLASS (dzl_dock_overlay_parent_class)->motion_notify_event (widget, event);
}

static void
dzl_dock_overlay_size_allocate (GtkWidget     *widget,
                                GtkAllocation *alloc)
{
  DzlDockOverlay *self = (DzlDockOverlay *)widget;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  GtkAllocation copy;
  GtkAllocation *edge;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (alloc != NULL);

  copy = *alloc;
  copy.x = 0;
  copy.y = 0;

#define GRAB_AREA 15

  edge = &priv->hover_borders [GTK_POS_TOP];
  edge->x = copy.x;
  edge->y = copy.y;
  edge->width = copy.width;
  edge->height = GRAB_AREA;

  edge = &priv->hover_borders [GTK_POS_LEFT];
  edge->x = copy.x;
  edge->y = copy.y;
  edge->width = GRAB_AREA;
  edge->height = copy.height;

  edge = &priv->hover_borders [GTK_POS_RIGHT];
  edge->x = copy.x + copy.width - GRAB_AREA;
  edge->y = copy.y;
  edge->width = GRAB_AREA;
  edge->height = copy.height;

  edge = &priv->hover_borders [GTK_POS_BOTTOM];
  edge->x = copy.x;
  edge->y = copy.y + copy.height - GRAB_AREA;
  edge->width = copy.width;
  edge->height = GRAB_AREA;

#undef GRAB_AREA

  GTK_WIDGET_CLASS (dzl_dock_overlay_parent_class)->size_allocate (widget, alloc);
}

static void
dzl_dock_overlay_get_property (GObject    *object,
                               guint       prop_id,
                               GValue     *value,
                               GParamSpec *pspec)
{
  DzlDockOverlay *self = DZL_DOCK_OVERLAY (object);

  switch (prop_id)
    {
    case PROP_MANAGER:
      g_value_set_object (value, dzl_dock_item_get_manager (DZL_DOCK_ITEM (self)));
      break;

    default:
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
    }
}

static void
dzl_dock_overlay_set_property (GObject      *object,
                               guint         prop_id,
                               const GValue *value,
                               GParamSpec   *pspec)
{
  DzlDockOverlay *self = DZL_DOCK_OVERLAY (object);

  switch (prop_id)
    {
    case PROP_MANAGER:
      dzl_dock_item_set_manager (DZL_DOCK_ITEM (self), g_value_get_object (value));
      break;

    default:
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
    }
}

static void
dzl_dock_overlay_get_child_property (GtkContainer *container,
                                     GtkWidget    *widget,
                                     guint         prop_id,
                                     GValue       *value,
                                     GParamSpec   *pspec)
{
  DzlDockOverlay *self = DZL_DOCK_OVERLAY (container);

  switch (prop_id)
    {
    case CHILD_PROP_REVEAL:
      g_value_set_boolean (value, dzl_dock_overlay_get_child_reveal (self, widget));
      break;

    case CHILD_PROP_REVEALED:
      g_value_set_boolean (value, dzl_dock_overlay_get_child_revealed (self, widget));
      break;

    default:
      GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, prop_id, pspec);
    }
}

static void
dzl_dock_overlay_set_child_property (GtkContainer *container,
                                     GtkWidget    *widget,
                                     guint         prop_id,
                                     const GValue *value,
                                     GParamSpec   *pspec)
{
  DzlDockOverlay *self = DZL_DOCK_OVERLAY (container);

  switch (prop_id)
    {
    case CHILD_PROP_REVEAL:
      dzl_dock_overlay_set_child_reveal (self, widget, g_value_get_boolean (value));
      break;

    default:
      GTK_CONTAINER_WARN_INVALID_CHILD_PROPERTY_ID (container, prop_id, pspec);
    }
}

static void
dzl_dock_overlay_class_init (DzlDockOverlayClass *klass)
{
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
  GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
  GtkBindingSet *binding_set;

  object_class->get_property = dzl_dock_overlay_get_property;
  object_class->set_property = dzl_dock_overlay_set_property;

  widget_class->destroy = dzl_dock_overlay_destroy;
  widget_class->hierarchy_changed = dzl_dock_overlay_hierarchy_changed;
  widget_class->motion_notify_event = dzl_dock_overlay_motion_notify_event;
  widget_class->size_allocate = dzl_dock_overlay_size_allocate;

  container_class->add = dzl_dock_overlay_add;
  container_class->get_child_property = dzl_dock_overlay_get_child_property;
  container_class->set_child_property = dzl_dock_overlay_set_child_property;

  klass->hide_edges = dzl_dock_overlay_hide_edges;

  g_object_class_override_property (object_class, PROP_MANAGER, "manager");

  /* The difference between those two is:
   * CHILD_PROP_REVEAL change its state at the animation start and can
   * trigger a state change (so read/write capabilities)
   *
   * CHILD_PROP_REVEALED change its state at the animation end
   * but is only readable.
   */
  child_properties [CHILD_PROP_REVEAL] =
    g_param_spec_boolean ("reveal",
                          "Reveal",
                          "If the panel edge should be revealed",
                          FALSE,
                          (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));

  child_properties [CHILD_PROP_REVEALED] =
    g_param_spec_boolean ("revealed",
                          "Revealed",
                          "If the panel edge is revealed",
                          FALSE,
                          (G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));

  gtk_container_class_install_child_properties (container_class, N_CHILD_PROPS, child_properties);

  gtk_widget_class_set_css_name (widget_class, "dzldockoverlay");

  signals [HIDE_EDGES] =
    g_signal_new ("hide-edges",
                  G_TYPE_FROM_CLASS (klass),
                  G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
                  G_STRUCT_OFFSET (DzlDockOverlayClass, hide_edges),
                  NULL, NULL, NULL,
                  G_TYPE_NONE, 0);

  binding_set = gtk_binding_set_by_class (klass);

  gtk_binding_entry_add_signal (binding_set,
                                GDK_KEY_Escape,
                                0,
                                "hide-edges",
                                0);
}

static void
dzl_dock_overlay_init (DzlDockOverlay *self)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  guint i;

  gtk_widget_add_events (GTK_WIDGET (self), GDK_POINTER_MOTION_MASK);

  priv->overlay = g_object_new (GTK_TYPE_OVERLAY,
                                "visible", TRUE,
                                NULL);

  GTK_CONTAINER_CLASS (dzl_dock_overlay_parent_class)->add (GTK_CONTAINER (self),
                                                            GTK_WIDGET (priv->overlay));

  g_signal_connect_object (priv->overlay,
                           "get-child-position",
                           G_CALLBACK (dzl_dock_overlay_get_child_position),
                           self,
                           G_CONNECT_SWAPPED);

  for (i = 0; i <= GTK_POS_BOTTOM; i++)
    {
      DzlDockOverlayEdge *edge;

      edge = g_object_new (DZL_TYPE_DOCK_OVERLAY_EDGE,
                           "edge", (GtkPositionType)i,
                           "visible", TRUE,
                           NULL);

      dzl_set_weak_pointer (&priv->edges[i], edge);

      gtk_overlay_add_overlay (priv->overlay, GTK_WIDGET (priv->edges [i]));

      priv->edge_adj [i] = gtk_adjustment_new (1, 0, 1, 0, 0, 0);

      g_signal_connect_object (priv->edge_adj [i],
                               "value-changed",
                               G_CALLBACK (gtk_widget_queue_allocate),
                               priv->overlay,
                               G_CONNECT_SWAPPED);

      priv->edge_handle_adj [i] = gtk_adjustment_new (0, 0, 1000, 0, 0, 0);

      g_signal_connect_object (priv->edge_handle_adj [i],
                               "value-changed",
                               G_CALLBACK (gtk_widget_queue_allocate),
                               priv->overlay,
                               G_CONNECT_SWAPPED);
    }
}

static void
dzl_dock_overlay_init_dock_iface (DzlDockInterface *iface)
{
}

GtkWidget *
dzl_dock_overlay_new (void)
{
  return g_object_new (DZL_TYPE_DOCK_OVERLAY, NULL);
}

static void
dzl_dock_overlay_real_add_child (GtkBuildable *buildable,
                                 GtkBuilder   *builder,
                                 GObject      *child,
                                 const gchar  *type)
{
  DzlDockOverlay *self = (DzlDockOverlay *)buildable;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  DzlDockOverlayEdge *parent = NULL;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (builder == NULL || GTK_IS_BUILDER (builder));
  g_assert (G_IS_OBJECT (child));

  if (!GTK_IS_WIDGET (child))
    {
      g_warning ("Attempt to add a child of type \"%s\" to a \"%s\"",
                 G_OBJECT_TYPE_NAME (child), G_OBJECT_TYPE_NAME (self));
      return;
    }

  if ((type == NULL) || (g_strcmp0 ("center", type) == 0))
    {
      gtk_container_add (GTK_CONTAINER (priv->overlay), GTK_WIDGET (child));
      goto adopt;
    }

  if (g_strcmp0 ("top", type) == 0)
    parent = priv->edges [GTK_POS_TOP];
  else if (g_strcmp0 ("bottom", type) == 0)
    parent = priv->edges [GTK_POS_BOTTOM];
  else if (g_strcmp0 ("right", type) == 0)
    parent = priv->edges [GTK_POS_RIGHT];
  else
    parent = priv->edges [GTK_POS_LEFT];

  gtk_container_add (GTK_CONTAINER (parent), GTK_WIDGET (child));

adopt:
  if (DZL_IS_DOCK_ITEM (child))
    dzl_dock_item_adopt (DZL_DOCK_ITEM (self), DZL_DOCK_ITEM (child));
}

static void
dzl_dock_overlay_init_buildable_iface (GtkBuildableIface *iface)
{
  iface->add_child = dzl_dock_overlay_real_add_child;
}

/**
 * dzl_dock_overlay_add_child:
 * @self: a #DzlDockOverlay.
 * @child: a #GtkWidget.
 * @type: the type of the child to add (center, left, right, top, bottom).
 *
 */
void
dzl_overlay_add_child (DzlDockOverlay *self,
                       GtkWidget      *child,
                       const gchar    *type)
{
  g_assert (DZL_IS_DOCK_OVERLAY (self));

  dzl_dock_overlay_real_add_child (GTK_BUILDABLE (self), NULL, G_OBJECT (child), type);
}

static void
dzl_dock_overlay_present_child (DzlDockItem *item,
                                DzlDockItem *child)
{
  DzlDockOverlay *self = (DzlDockOverlay *)item;

  g_assert (DZL_IS_DOCK_OVERLAY (self));
  g_assert (DZL_IS_DOCK_ITEM (child));

  gtk_container_child_set (GTK_CONTAINER (self), GTK_WIDGET (child),
                           "reveal", TRUE,
                           NULL);
}

static void
dzl_dock_overlay_update_visibility (DzlDockItem *item)
{
  DzlDockOverlay *self = (DzlDockOverlay *)item;
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);
  guint i;

  g_assert (DZL_IS_DOCK_OVERLAY (self));

  for (i = 0; i < G_N_ELEMENTS (priv->edges); i++)
    {
      DzlDockOverlayEdge *edge = priv->edges [i];
      gboolean has_widgets;

      if (edge == NULL)
        continue;

      has_widgets = dzl_dock_item_has_widgets (DZL_DOCK_ITEM (edge));

      gtk_widget_set_child_visible (GTK_WIDGET (edge), has_widgets);
    }

  gtk_widget_queue_resize (GTK_WIDGET (self));
}

static void
dzl_dock_overlay_init_dock_item_iface (DzlDockItemInterface *iface)
{
  iface->present_child = dzl_dock_overlay_present_child;
  iface->update_visibility = dzl_dock_overlay_update_visibility;
}

/**
 * dzl_dock_overlay_get_edge:
 * @self: An #DzlDockOverlay.
 * @position: the edge position.
 *
 * Returns: (transfer none): The corresponding #DzlDockOverlayEdge.
 */
DzlDockOverlayEdge *
dzl_dock_overlay_get_edge (DzlDockOverlay  *self,
                           GtkPositionType  position)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_return_val_if_fail (DZL_IS_DOCK_OVERLAY (self), NULL);

  return priv->edges [position];
}

/**
 * dzl_dock_overlay_get_edge_adjustment:
 * @self: An #DzlDockOverlay.
 * @position: the edge position.
 *
 * Returns: (transfer none): The corresponding #GtkAdjustment.
 */
GtkAdjustment *
dzl_dock_overlay_get_edge_adjustment (DzlDockOverlay  *self,
                                      GtkPositionType  position)
{
  DzlDockOverlayPrivate *priv = dzl_dock_overlay_get_instance_private (self);

  g_return_val_if_fail (DZL_IS_DOCK_OVERLAY (self), NULL);

  return priv->edge_adj [position];
}