[libhandy/wip/haecker-felix/flap-widget: 1/2] Add HdyFlap




commit a1dba333d65e0c9b374db1225fc8e7b2123170b2
Author: Alexander Mikhaylenko <alexm gnome org>
Date:   Wed Nov 25 15:28:06 2020 +0500

    Add HdyFlap
    
    Co-authored-by: Felix Häcker <haeckerfelix gnome org>

 debian/libhandy-1-0.symbols        |   31 +
 doc/handy-docs.xml                 |    1 +
 glade/glade-hdy-flap.c             |  131 ++
 glade/glade-hdy-flap.h             |   27 +
 glade/libhandy.xml                 |   34 +
 glade/meson.build                  |    1 +
 src/handy.h                        |    1 +
 src/hdy-enums.c.in                 |    1 +
 src/hdy-flap.c                     | 2477 ++++++++++++++++++++++++++++++++++++
 src/hdy-flap.h                     |  122 ++
 src/meson.build                    |    3 +
 src/themes/Adwaita-dark.css        |   16 +-
 src/themes/Adwaita.css             |   16 +-
 src/themes/HighContrast.css        |   16 +-
 src/themes/HighContrastInverse.css |   16 +-
 src/themes/_fallback-base.scss     |    1 +
 src/themes/fallback.css            |   16 +-
 tests/meson.build                  |    1 +
 tests/test-flap.c                  |  425 +++++++
 19 files changed, 3296 insertions(+), 40 deletions(-)
---
diff --git a/debian/libhandy-1-0.symbols b/debian/libhandy-1-0.symbols
index 75901fae..39f2f1dc 100644
--- a/debian/libhandy-1-0.symbols
+++ b/debian/libhandy-1-0.symbols
@@ -120,6 +120,37 @@ libhandy-1.so.0 libhandy-1-0 #MINVER#
  hdy_expander_row_set_show_enable_switch@LIBHANDY_1_0 0.0.6
  hdy_expander_row_set_subtitle@LIBHANDY_1_0 0.80.0
  hdy_expander_row_set_use_underline@LIBHANDY_1_0 0.80.0
+ hdy_flap_fold_policy_get_type@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_content@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_flap@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_flap_position@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_fold_duration@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_fold_policy@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_folded@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_locked@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_modal@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_reveal_duration@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_reveal_flap@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_reveal_progress@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_separator@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_swipe_to_close@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_swipe_to_open@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_transition_type@LIBHANDY_1_0 1.1.0
+ hdy_flap_get_type@LIBHANDY_1_0 1.1.0
+ hdy_flap_new@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_flap@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_flap_position@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_fold_duration@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_fold_policy@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_locked@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_modal@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_reveal_duration@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_reveal_flap@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_separator@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_swipe_to_close@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_swipe_to_open@LIBHANDY_1_0 1.1.0
+ hdy_flap_set_transition_type@LIBHANDY_1_0 1.1.0
+ hdy_flap_transition_type_get_type@LIBHANDY_1_0 1.1.0
  hdy_get_enable_animations@LIBHANDY_1_0 0.0.11
  hdy_header_bar_get_centering_policy@LIBHANDY_1_0 0.0.10
  hdy_header_bar_get_custom_title@LIBHANDY_1_0 0.0.10
diff --git a/doc/handy-docs.xml b/doc/handy-docs.xml
index e60dc514..3d1531c8 100644
--- a/doc/handy-docs.xml
+++ b/doc/handy-docs.xml
@@ -47,6 +47,7 @@
     <xi:include href="xml/hdy-deck.xml"/>
     <xi:include href="xml/hdy-enum-value-object.xml"/>
     <xi:include href="xml/hdy-expander-row.xml"/>
+    <xi:include href="xml/hdy-flap.xml"/>
     <xi:include href="xml/hdy-header-bar.xml"/>
     <xi:include href="xml/hdy-header-group.xml"/>
     <xi:include href="xml/hdy-keypad.xml"/>
diff --git a/glade/glade-hdy-flap.c b/glade/glade-hdy-flap.c
new file mode 100644
index 00000000..85115afb
--- /dev/null
+++ b/glade/glade-hdy-flap.c
@@ -0,0 +1,131 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Based on
+ * glade-gtk-header-bar.c - GladeWidgetAdaptor for GtkHeaderBar
+ */
+
+#include <config.h>
+#include <glib/gi18n-lib.h>
+
+#include "glade-hdy-flap.h"
+
+#include <gladeui/glade.h>
+#include "glade-hdy-utils.h"
+
+void
+glade_hdy_flap_post_create (GladeWidgetAdaptor *adaptor,
+                            GObject            *container,
+                            GladeCreateReason   reason)
+{
+  GtkWidget *child;
+
+  if (!hdy_flap_get_flap (HDY_FLAP (container))) {
+    child = glade_placeholder_new ();
+    g_object_set_data (G_OBJECT (child), "special-child-type", "flap");
+    hdy_flap_set_flap (HDY_FLAP (container), child);
+  }
+
+  if (!hdy_flap_get_separator (HDY_FLAP (container))) {
+    child = glade_placeholder_new ();
+    g_object_set_data (G_OBJECT (child), "special-child-type", "separator");
+    hdy_flap_set_separator (HDY_FLAP (container), child);
+  }
+
+  if (!hdy_flap_get_content (HDY_FLAP (container))) {
+    child = glade_placeholder_new ();
+    gtk_container_add (GTK_CONTAINER (container), child);
+  }
+}
+
+void
+glade_hdy_flap_add_child (GladeWidgetAdaptor *adaptor,
+                          GObject            *parent,
+                          GObject            *child)
+{
+  gchar *special_child_type = g_object_get_data (child, "special-child-type");
+  GtkWidget *content;
+
+  if (special_child_type && !strcmp (special_child_type, "flap")) {
+    hdy_flap_set_flap (HDY_FLAP (parent), GTK_WIDGET (child));
+
+    return;
+  }
+
+  if (special_child_type && !strcmp (special_child_type, "separator")) {
+    hdy_flap_set_separator (HDY_FLAP (parent), GTK_WIDGET (child));
+
+    return;
+  }
+
+  /* Get a placeholder out of the way before adding the child */
+  content = hdy_flap_get_content (HDY_FLAP (parent));
+
+  if (content) {
+    if (GLADE_IS_PLACEHOLDER (content))
+      gtk_container_remove (GTK_CONTAINER (parent), content);
+    else {
+      g_critical ("Can't add more than one content widget to a HdyFlap");
+
+      return;
+    }
+  }
+
+  gtk_container_add (GTK_CONTAINER (parent), GTK_WIDGET (child));
+}
+
+void
+glade_hdy_flap_remove_child (GladeWidgetAdaptor *adaptor,
+                             GObject            *object,
+                             GObject            *child)
+{
+  gchar *special_child_type = g_object_get_data (child, "special-child-type");
+  GtkWidget *replacement = glade_placeholder_new ();
+
+  if (special_child_type && !strcmp (special_child_type, "flap")) {
+    g_object_set_data (G_OBJECT (replacement), "special-child-type", "flap");
+    hdy_flap_set_flap (HDY_FLAP (object), replacement);
+
+    return;
+  }
+
+  if (special_child_type && !strcmp (special_child_type, "separator")) {
+    g_object_set_data (G_OBJECT (replacement), "special-child-type", "separator");
+    hdy_flap_set_separator (HDY_FLAP (object), replacement);
+
+    return;
+  }
+
+  gtk_container_remove (GTK_CONTAINER (object), GTK_WIDGET (child));
+  gtk_container_add (GTK_CONTAINER (object), replacement);
+}
+
+void
+glade_hdy_flap_replace_child (GladeWidgetAdaptor *adaptor,
+                              GObject            *container,
+                              GObject            *current,
+                              GObject            *new_widget)
+{
+  gchar *special_child_type =
+    g_object_get_data (G_OBJECT (current), "special-child-type");
+
+  if (special_child_type && !strcmp (special_child_type, "flap")) {
+    g_object_set_data (G_OBJECT (new_widget), "special-child-type", "flap");
+    hdy_flap_set_flap (HDY_FLAP (container), GTK_WIDGET (new_widget));
+
+    return;
+  }
+
+  if (special_child_type && !strcmp (special_child_type, "separator")) {
+    g_object_set_data (G_OBJECT (new_widget), "special-child-type", "separator");
+    hdy_flap_set_separator (HDY_FLAP (container), GTK_WIDGET (new_widget));
+
+    return;
+  }
+
+  g_object_set_data (G_OBJECT (new_widget), "special-child-type", NULL);
+  gtk_container_remove (GTK_CONTAINER (container), GTK_WIDGET (current));
+  gtk_container_add (GTK_CONTAINER (container), GTK_WIDGET (new_widget));
+}
diff --git a/glade/glade-hdy-flap.h b/glade/glade-hdy-flap.h
new file mode 100644
index 00000000..4782e845
--- /dev/null
+++ b/glade/glade-hdy-flap.h
@@ -0,0 +1,27 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ */
+
+#pragma once
+
+#include <gladeui/glade.h>
+
+#include <handy.h>
+
+
+void glade_hdy_flap_post_create (GladeWidgetAdaptor *adaptor,
+                                 GObject            *container,
+                                 GladeCreateReason   reason);
+
+void glade_hdy_flap_add_child (GladeWidgetAdaptor *adaptor,
+                               GObject            *parent,
+                               GObject            *child);
+void glade_hdy_flap_remove_child (GladeWidgetAdaptor *adaptor,
+                                  GObject            *object,
+                                  GObject            *child);
+void glade_hdy_flap_replace_child (GladeWidgetAdaptor *adaptor,
+                                   GObject            *container,
+                                   GObject            *current,
+                                   GObject            *new_widget);
diff --git a/glade/libhandy.xml b/glade/libhandy.xml
index 544c5268..c2247e3b 100644
--- a/glade/libhandy.xml
+++ b/glade/libhandy.xml
@@ -143,6 +143,39 @@
         <property id="enable-expansion" save="True" ignore="True"/>
       </properties>
     </glade-widget-class>
+    <glade-widget-class name="HdyFlap" generic-name="flap" title="Flap" since="1.1" use-placeholders="False">
+      <post-create-function>glade_hdy_flap_post_create</post-create-function>
+      <add-child-function>glade_hdy_flap_add_child</add-child-function>
+      <remove-child-function>glade_hdy_flap_remove_child</remove-child-function>
+      <replace-child-function>glade_hdy_flap_replace_child</replace-child-function>
+      <special-child-type>type</special-child-type>
+      <properties>
+        <property id="flap" disabled="True"/>
+        <property id="separator" disabled="True"/>
+        <property id="fold-policy">
+          <displayable-values>
+            <!-- HdyFlapFoldPolicy enumeration value -->
+            <value id="HDY_FLAP_FOLD_POLICY_NEVER" name="Never"/>
+            <!-- HdyFlapFoldPolicy enumeration value -->
+            <value id="HDY_FLAP_FOLD_POLICY_ALWAYS" name="Always"/>
+            <!-- HdyFlapFoldPolicy enumeration value -->
+            <value id="HDY_FLAP_FOLD_POLICY_AUTO" name="Auto"/>
+          </displayable-values>
+        </property>
+        <property id="transition-type">
+          <displayable-values>
+            <!-- HdyFlapTransitionType enumeration value -->
+            <value id="HDY_FLAP_TRANSITION_TYPE_OVER" name="Over"/>
+            <!-- HdyFlapTransitionType enumeration value -->
+            <value id="HDY_FLAP_TRANSITION_TYPE_UNDER" name="Under"/>
+            <!-- HdyFlapTransitionType enumeration value -->
+            <value id="HDY_FLAP_TRANSITION_TYPE_SLIDE" name="Slide"/>
+            <!-- HdyFlapTransitionType enumeration value -->
+            <value id="HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT" name="Over (transparent)"/>
+          </displayable-values>
+        </property>
+      </properties>
+    </glade-widget-class>
     <glade-widget-class name="HdyHeaderBar" generic-name="headerbar" title="Header Bar" since="0.0.10">
       <post-create-function>glade_hdy_header_bar_post_create</post-create-function>
       <add-child-function>glade_hdy_header_bar_add_child</add-child-function>
@@ -401,6 +434,7 @@
     <glade-widget-class-ref name="HdyComboRow"/>
     <glade-widget-class-ref name="HdyDeck"/>
     <glade-widget-class-ref name="HdyExpanderRow"/>
+    <glade-widget-class-ref name="HdyFlap"/>
     <glade-widget-class-ref name="HdyHeaderBar"/>
     <glade-widget-class-ref name="HdyHeaderGroup"/>
     <glade-widget-class-ref name="HdyKeypad"/>
diff --git a/glade/meson.build b/glade/meson.build
index cba9a364..946cd76a 100644
--- a/glade/meson.build
+++ b/glade/meson.build
@@ -10,6 +10,7 @@ glade_catalogdir   = gladeui_dep.get_pkgconfig_variable('catalogdir',
 libglade_hdy_sources = [
   'glade-hdy-carousel.c',
   'glade-hdy-expander-row.c',
+  'glade-hdy-flap.c',
   'glade-hdy-header-bar.c',
   'glade-hdy-header-group.c',
   'glade-hdy-leaflet.c',
diff --git a/src/handy.h b/src/handy.h
index 1ea48a7e..c02fa52a 100644
--- a/src/handy.h
+++ b/src/handy.h
@@ -34,6 +34,7 @@ G_BEGIN_DECLS
 #include "hdy-deprecation-macros.h"
 #include "hdy-enum-value-object.h"
 #include "hdy-expander-row.h"
+#include "hdy-flap.h"
 #include "hdy-header-bar.h"
 #include "hdy-header-group.h"
 #include "hdy-keypad.h"
diff --git a/src/hdy-enums.c.in b/src/hdy-enums.c.in
index a6305554..ff368a55 100644
--- a/src/hdy-enums.c.in
+++ b/src/hdy-enums.c.in
@@ -3,6 +3,7 @@
 #include "config.h"
 #include "hdy-deck.h"
 #include "hdy-enums.h"
+#include "hdy-flap.h"
 #include "hdy-header-bar.h"
 #include "hdy-header-group.h"
 #include "hdy-leaflet.h"
diff --git a/src/hdy-flap.c b/src/hdy-flap.c
new file mode 100644
index 00000000..2e97bc0b
--- /dev/null
+++ b/src/hdy-flap.c
@@ -0,0 +1,2477 @@
+/*
+ * Copyright (C) 2020 Felix Häcker <haeckerfelix gnome org>
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ */
+
+#include "config.h"
+#include "hdy-flap.h"
+
+#include <glib/gi18n-lib.h>
+#include <math.h>
+
+#include "hdy-animation-private.h"
+#include "hdy-shadow-helper-private.h"
+#include "hdy-swipeable.h"
+#include "hdy-swipe-tracker.h"
+
+/**
+ * SECTION:hdy-flap
+ * @Title: HdyFlap
+ *
+ * Since: 1.1
+ */
+
+/**
+ * HdyFlapFoldPolicy:
+ * @HDY_FLAP_FOLD_POLICY_NEVER: Disable folding
+ * @HDY_FLAP_FOLD_POLICY_ALWAYS: Keep the flap always folded
+ * @HDY_FLAP_FOLD_POLICY_AUTO: Fold and unfold the flap based on available space
+ *
+ * These enumeration values describe the possible folding behavior in a #HdyFlap
+ * widget.
+ *
+ * Since: 1.1
+ */
+
+/**
+ * HdyFlapTransitionType:
+ * @HDY_FLAP_TRANSITION_TYPE_OVER: TBD
+ * @HDY_FLAP_TRANSITION_TYPE_UNDER: TBD
+ * @HDY_FLAP_TRANSITION_TYPE_SLIDE: TBD
+ * @HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT: TBD
+ *
+ * These enumeration values describe the possible transitions between children
+ * in a #HdyFlap widget.
+ *
+ * New values may be added to this enum over time.
+ *
+ * Since: 1.1
+ */
+
+#define HDY_SWIPE_BORDER 16
+
+typedef struct {
+  GtkWidget *widget;
+  GdkWindow *window;
+  GtkAllocation allocation;
+} ChildInfo;
+
+struct _HdyFlap
+{
+  GtkContainer parent_instance;
+
+  ChildInfo content;
+  ChildInfo flap;
+  ChildInfo separator;
+
+  HdyFlapFoldPolicy fold_policy;
+  HdyFlapTransitionType transition_type;
+  GtkPackType flap_position;
+  gboolean reveal_flap;
+  gboolean locked;
+  gboolean folded;
+
+  guint fold_duration;
+  gdouble fold_progress;
+  HdyAnimation *fold_animation;
+
+  guint reveal_duration;
+  gdouble reveal_progress;
+  HdyAnimation *reveal_animation;
+
+  gboolean schedule_fold;
+
+  GtkOrientation orientation;
+
+  HdyShadowHelper *shadow_helper;
+
+  gboolean swipe_to_open;
+  gboolean swipe_to_close;
+  HdySwipeTracker *tracker;
+  gboolean swipe_active;
+
+  gboolean modal;
+  GtkGesture *click_gesture;
+  GtkEventController *key_controller;
+};
+
+static void hdy_flap_buildable_init (GtkBuildableIface *iface);
+static void hdy_flap_swipeable_init (HdySwipeableInterface *iface);
+
+G_DEFINE_TYPE_WITH_CODE (HdyFlap, hdy_flap, GTK_TYPE_CONTAINER,
+                         G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL)
+                         G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE, hdy_flap_buildable_init)
+                         G_IMPLEMENT_INTERFACE (HDY_TYPE_SWIPEABLE, hdy_flap_swipeable_init))
+
+enum {
+  PROP_0,
+  PROP_FLAP,
+  PROP_SEPARATOR,
+  PROP_FLAP_POSITION,
+  PROP_REVEAL_FLAP,
+  PROP_REVEAL_DURATION,
+  PROP_REVEAL_PROGRESS,
+  PROP_FOLD_POLICY,
+  PROP_FOLD_DURATION,
+  PROP_FOLDED,
+  PROP_LOCKED,
+  PROP_TRANSITION_TYPE,
+  PROP_MODAL,
+  PROP_SWIPE_TO_OPEN,
+  PROP_SWIPE_TO_CLOSE,
+
+  /* Overridden properties */
+  PROP_ORIENTATION,
+
+  LAST_PROP = PROP_ORIENTATION,
+};
+
+static GParamSpec *props[LAST_PROP];
+
+static void
+update_swipe_tracker (HdyFlap *self)
+{
+  gboolean reverse = self->flap_position == GTK_PACK_START;
+
+  if (!self->tracker)
+    return;
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL &&
+      gtk_widget_get_direction (GTK_WIDGET (self)) == GTK_TEXT_DIR_RTL)
+    reverse = !reverse;
+
+  hdy_swipe_tracker_set_enabled (self->tracker, self->flap.widget &&
+                                 (self->swipe_to_open || self->swipe_to_close));
+  hdy_swipe_tracker_set_reversed (self->tracker, reverse);
+  gtk_orientable_set_orientation (GTK_ORIENTABLE (self->tracker),
+                                  self->orientation);
+}
+
+static void
+set_orientation (HdyFlap        *self,
+                 GtkOrientation  orientation)
+{
+  if (self->orientation == orientation)
+    return;
+
+  self->orientation = orientation;
+
+  gtk_widget_queue_resize (GTK_WIDGET (self));
+  update_swipe_tracker (self);
+
+  g_object_notify (G_OBJECT (self), "orientation");
+}
+
+static void
+update_child_visibility (HdyFlap *self)
+{
+  gboolean visible = self->reveal_progress > 0;
+
+  if (self->flap.widget)
+    gtk_widget_set_child_visible (self->flap.widget, visible);
+
+  if (self->separator.widget)
+    gtk_widget_set_child_visible (self->separator.widget, visible);
+
+  if (!gtk_widget_get_realized (GTK_WIDGET (self)))
+    return;
+
+  if (self->flap.widget) {
+    if (visible)
+      gdk_window_show (self->flap.window);
+    else
+      gdk_window_hide (self->flap.window);
+  }
+
+  if (self->separator.widget) {
+    if (visible)
+      gdk_window_show (self->separator.window);
+    else
+      gdk_window_hide (self->separator.window);
+  }
+
+  gtk_widget_queue_resize (GTK_WIDGET (self));
+}
+
+static void
+set_reveal_progress (HdyFlap *self,
+                     gdouble  progress)
+{
+  self->reveal_progress = progress;
+
+  update_child_visibility (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_REVEAL_PROGRESS]);
+}
+
+static void
+fold_animation_value_cb (gdouble  value,
+                         HdyFlap *self)
+{
+  self->fold_progress = value;
+
+  gtk_widget_queue_resize (GTK_WIDGET (self));
+}
+
+static void
+fold_animation_done_cb (HdyFlap *self)
+{
+  g_clear_pointer (&self->fold_animation, hdy_animation_unref);
+}
+
+static void
+animate_fold (HdyFlap *self)
+{
+  if (self->fold_animation)
+    hdy_animation_stop (self->fold_animation);
+
+  self->fold_animation =
+    hdy_animation_new (GTK_WIDGET (self),
+                       self->fold_progress,
+                       self->folded ? 1 : 0,
+                       /* When the flap is completely hidden, we can skip animation */
+                       (self->reveal_progress > 0) ? self->fold_duration : 0,
+                       hdy_ease_out_cubic,
+                       (HdyAnimationValueCallback) fold_animation_value_cb,
+                       (HdyAnimationDoneCallback) fold_animation_done_cb,
+                       self);
+
+  hdy_animation_start (self->fold_animation);
+}
+
+static void
+reveal_animation_value_cb (gdouble  value,
+                           HdyFlap *self)
+{
+  set_reveal_progress (self, value);
+}
+
+static void
+reveal_animation_done_cb (HdyFlap *self)
+{
+  g_clear_pointer (&self->reveal_animation, hdy_animation_unref);
+
+  if (self->reveal_progress <= 0)
+    hdy_shadow_helper_clear_cache (self->shadow_helper);
+
+  if (self->schedule_fold) {
+    self->schedule_fold = FALSE;
+
+    animate_fold (self);
+  }
+
+  gtk_widget_queue_allocate (GTK_WIDGET (self));
+}
+
+static void
+animate_reveal (HdyFlap *self,
+                gdouble  to,
+                gint64   duration)
+{
+  if (self->reveal_animation)
+    hdy_animation_stop (self->reveal_animation);
+
+  self->reveal_animation =
+    hdy_animation_new (GTK_WIDGET (self),
+                       self->reveal_progress,
+                       to,
+                       duration,
+                       hdy_ease_out_cubic,
+                       (HdyAnimationValueCallback) reveal_animation_value_cb,
+                       (HdyAnimationDoneCallback) reveal_animation_done_cb,
+                       self);
+
+  hdy_animation_start (self->reveal_animation);
+}
+
+static void
+set_reveal_flap (HdyFlap  *self,
+                 gboolean  reveal_flap,
+                 guint64   duration,
+                 gboolean  emit_child_switched)
+{
+  reveal_flap = !!reveal_flap;
+
+  if (self->reveal_flap == reveal_flap)
+    return;
+
+  self->reveal_flap = reveal_flap;
+
+  if (!self->swipe_active) {
+    animate_reveal (self, reveal_flap ? 1 : 0, duration);
+
+    if (emit_child_switched)
+      hdy_swipeable_emit_child_switched (HDY_SWIPEABLE (self), reveal_flap ? 1 : 0, duration);
+  }
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_REVEAL_FLAP]);
+}
+
+static void
+set_folded (HdyFlap  *self,
+            gboolean  folded)
+{
+  folded = !!folded;
+
+  if (self->folded == folded)
+    return;
+
+  self->folded = folded;
+
+  gtk_widget_queue_allocate (GTK_WIDGET (self));
+
+   /* When unlocked, folding should also hide flap. We don't want two concurrent
+    * animations in this case, instead only animate reveal and schedule a fold
+    * after it finishes, which will be skipped because the flap is fuly hidden.
+    * Meanwhile if it's unfolding, animate folding immediately. */
+  if (!self->locked && folded)
+    self->schedule_fold = TRUE;
+  else
+    animate_fold (self);
+
+  if (!self->locked)
+    set_reveal_flap (self, !self->folded, self->fold_duration, TRUE);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_FOLDED]);
+}
+
+static inline GtkPackType
+get_start_or_end (HdyFlap *self)
+{
+  GtkTextDirection direction = gtk_widget_get_direction (GTK_WIDGET (self));
+  gboolean is_rtl = direction == GTK_TEXT_DIR_RTL;
+  gboolean is_horiz = self->orientation == GTK_ORIENTATION_HORIZONTAL;
+
+  return (is_rtl && is_horiz) ? GTK_PACK_END : GTK_PACK_START;
+}
+
+static void
+begin_swipe_cb (HdySwipeTracker        *tracker,
+                HdyNavigationDirection  direction,
+                gboolean                direct,
+                HdyFlap                *self)
+{
+  if (self->reveal_progress <= 0 && !self->swipe_to_open)
+    return;
+
+  if (self->reveal_progress >= 1 && !self->swipe_to_close)
+    return;
+
+  if (self->reveal_animation)
+    hdy_animation_stop (self->reveal_animation);
+
+  self->swipe_active = TRUE;
+}
+
+static void
+update_swipe_cb (HdySwipeTracker *tracker,
+                 gdouble          progress,
+                 HdyFlap         *self)
+{
+  if (!self->swipe_active)
+    return;
+
+  set_reveal_progress (self, progress);
+}
+
+static void
+end_swipe_cb (HdySwipeTracker *tracker,
+              gint64           duration,
+              gdouble          to,
+              HdyFlap         *self)
+{
+  if (!self->swipe_active)
+    return;
+
+  self->swipe_active = FALSE;
+
+  if ((to > 0) == self->reveal_flap)
+    animate_reveal (self, to, duration);
+  else
+    set_reveal_flap (self, to > 0, duration, FALSE);
+}
+
+static void
+pressed_cb (GtkGestureMultiPress *gesture,
+            gint                  n_press,
+            gdouble               x,
+            gdouble               y,
+            HdyFlap              *self)
+{
+  if (self->reveal_progress <= 0 || self->fold_progress <= 0) {
+    gtk_gesture_set_state (GTK_GESTURE (gesture), GTK_EVENT_SEQUENCE_DENIED);
+
+    return;
+  }
+
+  if (x >= self->flap.allocation.x &&
+      x <= self->flap.allocation.x + self->flap.allocation.width &&
+      y >= self->flap.allocation.y &&
+      y <= self->flap.allocation.y + self->flap.allocation.height) {
+    gtk_gesture_set_state (GTK_GESTURE (gesture), GTK_EVENT_SEQUENCE_DENIED);
+
+    return;
+  }
+
+  hdy_flap_set_reveal_flap (self, FALSE);
+}
+
+static gboolean
+key_pressed_cb (GtkEventControllerKey *controller,
+                guint                  keyval,
+                guint                  keycode,
+                GdkModifierType        modifiers,
+                HdyFlap               *self)
+{
+  if (keyval == GDK_KEY_Escape &&
+      self->reveal_progress > 0 &&
+      self->fold_progress > 0) {
+    hdy_flap_set_reveal_flap (self, FALSE);
+
+    return GDK_EVENT_STOP;
+  }
+
+  return GDK_EVENT_PROPAGATE;
+}
+
+static void
+register_window (HdyFlap   *self,
+                 ChildInfo *info)
+{
+  GdkWindowAttr attributes = { 0 };
+  GdkWindowAttributesType attributes_mask;
+
+  if (!info->widget)
+    return;
+
+  attributes.x = info->allocation.x;
+  attributes.y = info->allocation.y;
+  attributes.width = info->allocation.width;
+  attributes.height = info->allocation.height;
+  attributes.window_type = GDK_WINDOW_CHILD;
+  attributes.wclass = GDK_INPUT_OUTPUT;
+  attributes.visual = gtk_widget_get_visual (info->widget);
+  attributes.event_mask = gtk_widget_get_events (info->widget);
+  attributes_mask = (GDK_WA_X | GDK_WA_Y) | GDK_WA_VISUAL;
+
+  attributes.event_mask = gtk_widget_get_events (GTK_WIDGET (self)) |
+                          gtk_widget_get_events (info->widget);
+
+  info->window = gdk_window_new (gtk_widget_get_window (GTK_WIDGET (self)),
+                                 &attributes, attributes_mask);
+  gtk_widget_register_window (GTK_WIDGET (self), info->window);
+
+  gtk_widget_set_parent_window (info->widget, info->window);
+
+  gdk_window_show (info->window);
+}
+
+static void
+unregister_window (HdyFlap   *self,
+                   ChildInfo *info)
+{
+  if (!info->window)
+    return;
+
+  gtk_widget_unregister_window (GTK_WIDGET (self), info->window);
+  gdk_window_destroy (info->window);
+  info->window = NULL;
+}
+
+static gboolean
+transition_is_content_above_flap (HdyFlap *self)
+{
+  switch (self->transition_type) {
+  case HDY_FLAP_TRANSITION_TYPE_OVER:
+  case HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT:
+    return FALSE;
+
+  case HDY_FLAP_TRANSITION_TYPE_UNDER:
+  case HDY_FLAP_TRANSITION_TYPE_SLIDE:
+    return TRUE;
+
+  default:
+    g_assert_not_reached ();
+  }
+}
+
+static gboolean
+transition_should_clip (HdyFlap *self)
+{
+  switch (self->transition_type) {
+  case HDY_FLAP_TRANSITION_TYPE_OVER:
+  case HDY_FLAP_TRANSITION_TYPE_UNDER:
+    return TRUE;
+
+  case HDY_FLAP_TRANSITION_TYPE_SLIDE:
+  case HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT:
+    return FALSE;
+
+  default:
+    g_assert_not_reached ();
+  }
+}
+
+static gdouble
+transition_get_content_motion_factor (HdyFlap *self)
+{
+  switch (self->transition_type) {
+  case HDY_FLAP_TRANSITION_TYPE_OVER:
+  case HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT:
+    return 0;
+
+  case HDY_FLAP_TRANSITION_TYPE_UNDER:
+  case HDY_FLAP_TRANSITION_TYPE_SLIDE:
+    return 1;
+
+  default:
+    g_assert_not_reached ();
+  }
+}
+
+static gdouble
+transition_get_flap_motion_factor (HdyFlap *self)
+{
+  switch (self->transition_type) {
+  case HDY_FLAP_TRANSITION_TYPE_OVER:
+  case HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT:
+  case HDY_FLAP_TRANSITION_TYPE_SLIDE:
+    return 1;
+
+  case HDY_FLAP_TRANSITION_TYPE_UNDER:
+    return 0;
+
+  default:
+    g_assert_not_reached ();
+  }
+}
+
+static void
+restack_windows (HdyFlap *self)
+{
+  gboolean content_above_flap = transition_is_content_above_flap (self);
+
+  if (!content_above_flap) {
+    if (self->content.window)
+      gdk_window_raise (self->content.window);
+
+    if (self->separator.window)
+      gdk_window_raise (self->separator.window);
+  }
+
+  if (self->flap.window)
+    gdk_window_raise (self->flap.window);
+
+  if (content_above_flap) {
+    if (self->separator.window)
+      gdk_window_raise (self->separator.window);
+
+    if (self->content.window)
+      gdk_window_raise (self->content.window);
+  }
+}
+
+static void
+add_child (HdyFlap   *self,
+           ChildInfo *info)
+{
+  if (gtk_widget_get_realized (GTK_WIDGET (self))) {
+    register_window (self, info);
+    restack_windows (self);
+  }
+
+  gtk_widget_set_parent (info->widget, GTK_WIDGET (self));
+}
+
+static void
+remove_child (HdyFlap   *self,
+              ChildInfo *info)
+{
+  if (gtk_widget_get_realized (GTK_WIDGET (self)))
+    unregister_window (self, info);
+
+  gtk_widget_unparent (info->widget);
+}
+
+static inline void
+get_preferred_size (GtkWidget      *widget,
+                    GtkOrientation  orientation,
+                    gint           *min,
+                    gint           *nat)
+{
+  if (orientation == GTK_ORIENTATION_HORIZONTAL)
+    gtk_widget_get_preferred_width (widget, min, nat);
+  else
+    gtk_widget_get_preferred_height (widget, min, nat);
+}
+
+static void
+compute_sizes (HdyFlap       *self,
+               GtkAllocation *alloc,
+               gboolean       folded,
+               gboolean       revealed,
+               gint          *flap_size,
+               gint          *content_size,
+               gint          *separator_size)
+{
+  gboolean flap_expand, content_expand;
+  gint total, extra;
+  gint flap_nat, content_nat;
+
+  if (!self->flap.widget && !self->content.widget)
+    return;
+
+  if (self->separator.widget)
+    get_preferred_size (self->separator.widget, self->orientation, separator_size, NULL);
+  else
+    *separator_size = 0;
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL)
+    total = alloc->width;
+  else
+    total = alloc->height;
+
+  if (!self->flap.widget) {
+    *content_size = total;
+    *flap_size = 0;
+
+    return;
+  }
+
+  if (!self->content.widget) {
+    *content_size = 0;
+    *flap_size = total;
+
+    return;
+  }
+
+  get_preferred_size (self->flap.widget, self->orientation, flap_size, &flap_nat);
+  get_preferred_size (self->content.widget, self->orientation, content_size, &content_nat);
+
+  flap_expand = gtk_widget_compute_expand (self->flap.widget, self->orientation);
+  content_expand = gtk_widget_compute_expand (self->content.widget, self->orientation);
+
+  if (folded) {
+    *content_size = total;
+
+    if (flap_expand) {
+      *flap_size = total;
+    } else {
+      get_preferred_size (self->flap.widget, self->orientation, NULL, flap_size);
+      *flap_size = MIN (*flap_size, total);
+    }
+
+    return;
+  }
+
+  if (revealed)
+    total -= *separator_size;
+
+  if (flap_expand && content_expand) {
+    *flap_size = MAX (total / 2, *flap_size);
+
+    if (!revealed)
+      *content_size = total;
+    else
+      *content_size = total - *flap_size;
+
+    return;
+  }
+
+  extra = total - *content_size - *flap_size;
+
+  if (flap_expand) {
+    *flap_size += extra;
+    extra = 0;
+
+    if (!revealed)
+      *content_size = total;
+
+    return;
+  }
+
+  if (content_expand) {
+    *content_size += extra;
+    extra = 0;
+  }
+
+  if (extra > 0) {
+    GtkRequestedSize sizes[2];
+
+    sizes[0].data = self->flap.widget;
+    sizes[0].minimum_size = *flap_size;
+    sizes[0].natural_size = flap_nat;
+
+    sizes[1].data = self->content.widget;
+    sizes[1].minimum_size = *content_size;
+    sizes[1].natural_size = content_nat;
+
+    extra = gtk_distribute_natural_allocation (extra, 2, sizes);
+
+    *flap_size = sizes[0].minimum_size;
+    *content_size = sizes[1].minimum_size + extra;
+  }
+
+  if (!revealed)
+    *content_size = total;
+}
+
+static inline void
+interpolate_reveal (HdyFlap       *self,
+                    GtkAllocation *alloc,
+                    gboolean       folded,
+                    gint          *flap_size,
+                    gint          *content_size,
+                    gint          *separator_size)
+{
+  if (self->reveal_progress <= 0) {
+    compute_sizes (self, alloc, folded, FALSE, flap_size, content_size, separator_size);
+  } else if (self->reveal_progress >= 1) {
+    compute_sizes (self, alloc, folded, TRUE, flap_size, content_size, separator_size);
+  } else {
+    gint flap_revealed, content_revealed, separator_revealed;
+    gint flap_hidden, content_hidden, separator_hidden;
+
+    compute_sizes (self, alloc, folded, TRUE, &flap_revealed, &content_revealed, &separator_revealed);
+    compute_sizes (self, alloc, folded, FALSE, &flap_hidden, &content_hidden, &separator_hidden);
+
+    *flap_size =
+      (gint) round (hdy_lerp (flap_hidden, flap_revealed,
+                              self->reveal_progress));
+    *content_size =
+      (gint) round (hdy_lerp (content_hidden, content_revealed,
+                              self->reveal_progress));
+    *separator_size =
+      (gint) round (hdy_lerp (separator_hidden, separator_revealed,
+                              self->reveal_progress));
+  }
+}
+
+static inline void
+interpolate_fold (HdyFlap       *self,
+                  GtkAllocation *alloc,
+                  gint          *flap_size,
+                  gint          *content_size,
+                  gint          *separator_size)
+{
+  if (self->fold_progress <= 0) {
+    interpolate_reveal (self, alloc, FALSE, flap_size, content_size, separator_size);
+  } else if (self->fold_progress >= 1) {
+    interpolate_reveal (self, alloc, TRUE, flap_size, content_size, separator_size);
+  } else {
+    gint flap_folded, content_folded, separator_folded;
+    gint flap_unfolded, content_unfolded, separator_unfolded;
+
+    interpolate_reveal (self, alloc, TRUE, &flap_folded, &content_folded, &separator_folded);
+    interpolate_reveal (self, alloc, FALSE, &flap_unfolded, &content_unfolded, &separator_unfolded);
+
+    *flap_size =
+      (gint) round (hdy_lerp (flap_unfolded, flap_folded,
+                              self->fold_progress));
+    *content_size =
+      (gint) round (hdy_lerp (content_unfolded, content_folded,
+                              self->fold_progress));
+    *separator_size =
+      (gint) round (hdy_lerp (separator_unfolded, separator_folded,
+                              self->fold_progress));
+  }
+}
+
+static void
+compute_allocation (HdyFlap       *self,
+                    GtkAllocation *alloc,
+                    GtkAllocation *flap_alloc,
+                    GtkAllocation *content_alloc,
+                    GtkAllocation *separator_alloc)
+{
+  gdouble distance;
+  gint content_size, flap_size, separator_size;
+  gint total, content_pos, flap_pos, separator_pos;
+  gboolean content_above_flap = transition_is_content_above_flap (self);
+
+  if (!self->flap.widget && !self->content.widget && !self->separator.widget)
+    return;
+
+  content_alloc->x = 0;
+  content_alloc->y = 0;
+  flap_alloc->x = 0;
+  flap_alloc->y = 0;
+  separator_alloc->x = 0;
+  separator_alloc->y = 0;
+
+  interpolate_fold (self, alloc, &flap_size, &content_size, &separator_size);
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL) {
+    flap_alloc->width = flap_size;
+    content_alloc->width = content_size;
+    separator_alloc->width = separator_size;
+    flap_alloc->height = content_alloc->height = separator_alloc->height = alloc->height;
+    total = alloc->width;
+  } else {
+    flap_alloc->height = flap_size;
+    content_alloc->height = content_size;
+    separator_alloc->height = separator_size;
+    flap_alloc->width = content_alloc->width = separator_alloc->width = alloc->width;
+    total = alloc->height;
+  }
+
+  if (!self->flap.widget)
+    return;
+
+  if (content_above_flap)
+    distance = flap_size + separator_size;
+  else
+    distance = flap_size + separator_size * (1 - self->fold_progress);
+
+  flap_pos = -(gint) round ((1 - self->reveal_progress) * transition_get_flap_motion_factor (self) * 
distance);
+
+  if (content_above_flap) {
+    content_pos = (gint) round (self->reveal_progress * transition_get_content_motion_factor (self) * 
distance);
+    separator_pos = flap_pos + flap_size;
+  } else {
+    content_pos = total - content_size + (gint) round (self->reveal_progress * self->fold_progress * 
transition_get_content_motion_factor (self) * distance);
+    separator_pos = content_pos - separator_size;
+  }
+
+  if (self->flap_position != get_start_or_end (self)) {
+    flap_pos = total - flap_pos - flap_size;
+    separator_pos = total - separator_pos - separator_size;
+    content_pos = total - content_pos - content_size;
+  }
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL) {
+    content_alloc->x = content_pos;
+    flap_alloc->x = flap_pos;
+    separator_alloc->x = separator_pos;
+  } else {
+    content_alloc->y = content_pos;
+    flap_alloc->y = flap_pos;
+    separator_alloc->y = separator_pos;
+  }
+}
+
+static inline void
+allocate_child (HdyFlap   *self,
+                ChildInfo *info,
+                gboolean   expand_window)
+{
+  GtkAllocation child_alloc;
+
+  if (!info->widget)
+    return;
+
+  if (gtk_widget_get_realized (GTK_WIDGET (self))) {
+    if (expand_window)
+      gdk_window_move_resize (info->window,
+                              0, 0,
+                              gtk_widget_get_allocated_width (GTK_WIDGET (self)),
+                              gtk_widget_get_allocated_height (GTK_WIDGET (self)));
+    else
+      gdk_window_move_resize (info->window,
+                              info->allocation.x,
+                              info->allocation.y,
+                              info->allocation.width,
+                              info->allocation.height);
+  }
+
+  child_alloc.x = expand_window ? info->allocation.x : 0;
+  child_alloc.y = expand_window ? info->allocation.y : 0;
+  child_alloc.width = info->allocation.width;
+  child_alloc.height = info->allocation.height;
+
+  gtk_widget_size_allocate (info->widget, &child_alloc);
+}
+
+static void
+hdy_flap_size_allocate (GtkWidget     *widget,
+                        GtkAllocation *alloc)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+
+  gtk_widget_set_allocation (widget, alloc);
+
+  if (gtk_widget_get_realized (widget))
+    gdk_window_move_resize (gtk_widget_get_window (widget),
+                            alloc->x, alloc->y, alloc->width, alloc->height);
+
+  if (self->fold_policy == HDY_FLAP_FOLD_POLICY_AUTO) {
+    GtkRequisition flap_min = { 0, 0 };
+    GtkRequisition content_min = { 0, 0 };
+    GtkRequisition separator_min = { 0, 0 };
+
+    if (self->flap.widget)
+      gtk_widget_get_preferred_size (self->flap.widget, &flap_min, NULL);
+    if (self->content.widget)
+      gtk_widget_get_preferred_size (self->content.widget, &content_min, NULL);
+    if (self->separator.widget)
+      gtk_widget_get_preferred_size (self->separator.widget, &separator_min, NULL);
+
+    if (self->orientation == GTK_ORIENTATION_HORIZONTAL)
+      set_folded (self, alloc->width < content_min.width + flap_min.width + separator_min.width);
+    else
+      set_folded (self, alloc->height < content_min.height + flap_min.height + separator_min.height);
+  }
+
+  compute_allocation (self,
+                      alloc,
+                      &self->flap.allocation,
+                      &self->content.allocation,
+                      &self->separator.allocation);
+
+  allocate_child (self, &self->content, FALSE);
+  allocate_child (self, &self->separator, FALSE);
+  allocate_child (self, &self->flap,
+                  self->modal &&
+                  self->reveal_progress > 0 &&
+                  self->fold_progress > 0);
+
+  gtk_widget_set_clip (widget, alloc);
+}
+
+/* This private method is prefixed by the call name because it will be a virtual
+ * method in GTK 4.
+ */
+static void
+hdy_flap_measure (GtkWidget      *widget,
+                  GtkOrientation  orientation,
+                  gint            for_size,
+                  gint           *minimum,
+                  gint           *natural,
+                  gint           *minimum_baseline,
+                  gint           *natural_baseline)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+
+  gint content_min = 0, content_nat = 0;
+  gint flap_min = 0, flap_nat = 0;
+  gint separator_min = 0, separator_nat = 0;
+  gint min, nat;
+
+  if (self->content.widget)
+    get_preferred_size (self->content.widget, orientation, &content_min, &content_nat);
+
+  if (self->flap.widget)
+    get_preferred_size (self->flap.widget, orientation, &flap_min, &flap_nat);
+
+  if (self->separator.widget)
+    get_preferred_size (self->separator.widget, orientation, &separator_min, &separator_nat);
+
+  if (self->orientation == orientation &&
+      self->fold_policy != HDY_FLAP_FOLD_POLICY_AUTO) {
+    gdouble progress = (1 - self->fold_progress) * self->reveal_progress;
+
+    min = MAX (content_min + (gint) round ((flap_min + separator_min) * progress), flap_min);
+    nat = MAX (content_nat + (gint) round ((flap_nat + separator_nat) * progress), flap_nat);
+  } else {
+    min = MAX (MAX (content_min, flap_min), separator_min);
+    nat = MAX (MAX (content_nat, flap_nat), separator_nat);
+  }
+
+  if (minimum)
+    *minimum = min;
+  if (natural)
+    *natural = nat;
+  if (minimum_baseline)
+    *minimum_baseline = -1;
+  if (natural_baseline)
+    *natural_baseline = -1;
+}
+
+static void
+hdy_flap_get_preferred_width_for_height (GtkWidget *widget,
+                                         gint       height,
+                                         gint      *minimum,
+                                         gint      *natural)
+{
+  hdy_flap_measure (widget, GTK_ORIENTATION_HORIZONTAL, height,
+                    minimum, natural, NULL, NULL);
+}
+
+static void
+hdy_flap_get_preferred_width (GtkWidget *widget,
+                              gint      *minimum,
+                              gint      *natural)
+{
+  hdy_flap_measure (widget, GTK_ORIENTATION_HORIZONTAL, -1,
+                    minimum, natural, NULL, NULL);
+}
+
+
+static void
+hdy_flap_get_preferred_height_for_width (GtkWidget *widget,
+                                         gint       width,
+                                         gint      *minimum,
+                                         gint      *natural)
+{
+  hdy_flap_measure (widget, GTK_ORIENTATION_VERTICAL, width,
+                    minimum, natural, NULL, NULL);
+}
+
+static void
+hdy_flap_get_preferred_height (GtkWidget *widget,
+                               gint      *minimum,
+                               gint      *natural)
+{
+  hdy_flap_measure (widget, GTK_ORIENTATION_VERTICAL, -1,
+                    minimum, natural, NULL, NULL);
+}
+
+static gboolean
+hdy_flap_draw (GtkWidget *widget,
+               cairo_t   *cr)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+  gint width, height;
+  gint shadow_x = 0, shadow_y = 0;
+  gdouble shadow_progress;
+  GtkPanDirection shadow_direction;
+  gboolean content_above_flap = transition_is_content_above_flap (self);
+  GtkAllocation *shadow_alloc;
+  gboolean should_clip;
+
+  shadow_alloc = content_above_flap ? &self->content.allocation : &self->flap.allocation;
+
+  width = gtk_widget_get_allocated_width (widget);
+  height = gtk_widget_get_allocated_height (widget);
+
+  if (self->orientation == GTK_ORIENTATION_VERTICAL) {
+    if ((self->flap_position == GTK_PACK_START) != content_above_flap) {
+      shadow_direction = GTK_PAN_DIRECTION_UP;
+      shadow_y = shadow_alloc->y + shadow_alloc->height;
+    } else {
+      shadow_direction = GTK_PAN_DIRECTION_DOWN;
+      shadow_y = shadow_alloc->y - height;
+    }
+  } else {
+    if ((self->flap_position == get_start_or_end (self)) != content_above_flap) {
+      shadow_direction = GTK_PAN_DIRECTION_LEFT;
+      shadow_x = shadow_alloc->x + shadow_alloc->width;
+    } else {
+      shadow_direction = GTK_PAN_DIRECTION_RIGHT;
+      shadow_x = shadow_alloc->x - width;
+    }
+  }
+
+  switch (self->transition_type) {
+  case HDY_FLAP_TRANSITION_TYPE_OVER:
+  case HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT:
+    shadow_progress = 1 - MIN (self->reveal_progress, self->fold_progress);
+    break;
+
+  case HDY_FLAP_TRANSITION_TYPE_UNDER:
+    shadow_progress = self->reveal_progress;
+    break;
+
+  case HDY_FLAP_TRANSITION_TYPE_SLIDE:
+    shadow_progress = 1;
+    break;
+
+  default:
+    g_assert_not_reached ();
+  }
+
+  should_clip = transition_should_clip (self) &&
+                shadow_progress < 1 &&
+                self->reveal_progress > 0;
+
+  if (should_clip) {
+    cairo_save (cr);
+    cairo_rectangle (cr, shadow_x, shadow_y, width, height);
+    cairo_clip (cr);
+  }
+
+  if (!content_above_flap) {
+    if (self->content.widget)
+      gtk_container_propagate_draw (GTK_CONTAINER (self),
+                                    self->content.widget,
+                                    cr);
+
+    if (self->separator.widget)
+      gtk_container_propagate_draw (GTK_CONTAINER (self),
+                                    self->separator.widget,
+                                    cr);
+
+    if (should_clip)
+      cairo_restore (cr);
+  }
+
+  if (self->flap.widget)
+    gtk_container_propagate_draw (GTK_CONTAINER (self),
+                                  self->flap.widget,
+                                  cr);
+
+  if (content_above_flap) {
+    if (self->separator.widget)
+      gtk_container_propagate_draw (GTK_CONTAINER (self),
+                                    self->separator.widget,
+                                    cr);
+
+    if (should_clip)
+      cairo_restore (cr);
+
+    if (self->content.widget)
+      gtk_container_propagate_draw (GTK_CONTAINER (self),
+                                    self->content.widget,
+                                    cr);
+  }
+
+  if (!self->flap.widget)
+    return GDK_EVENT_PROPAGATE;
+
+  if (shadow_progress < 1 && gtk_widget_get_mapped (self->flap.widget)) {
+    cairo_save (cr);
+    cairo_translate (cr, shadow_x, shadow_y);
+    hdy_shadow_helper_draw_shadow (self->shadow_helper, cr, width, height,
+                                   shadow_progress, shadow_direction);
+    cairo_restore (cr);
+  }
+
+  return GDK_EVENT_PROPAGATE;
+}
+
+static void
+hdy_flap_realize (GtkWidget *widget)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+  GtkAllocation allocation;
+  GdkWindowAttr attributes;
+  gint attributes_mask;
+  GdkWindow *window;
+
+  gtk_widget_get_allocation (widget, &allocation);
+  gtk_widget_set_realized (widget, TRUE);
+
+  attributes.x = allocation.x;
+  attributes.y = allocation.y;
+  attributes.width = allocation.width;
+  attributes.height = allocation.height;
+  attributes.window_type = GDK_WINDOW_CHILD;
+  attributes.event_mask = gtk_widget_get_events (widget);
+  attributes.visual = gtk_widget_get_visual (widget);
+  attributes.wclass = GDK_INPUT_OUTPUT;
+  attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL;
+
+  window = gdk_window_new (gtk_widget_get_parent_window (widget),
+                           &attributes,
+                           attributes_mask);
+  gtk_widget_set_window (widget, window);
+  gtk_widget_register_window (widget, window);
+
+  register_window (self, &self->content);
+  register_window (self, &self->separator);
+  register_window (self, &self->flap);
+
+  restack_windows (self);
+}
+
+static void
+hdy_flap_unrealize (GtkWidget *widget)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+
+  unregister_window (self, &self->content);
+  unregister_window (self, &self->separator);
+  unregister_window (self, &self->flap);
+
+  GTK_WIDGET_CLASS (hdy_flap_parent_class)->unrealize (widget);
+}
+
+static void
+hdy_flap_direction_changed (GtkWidget        *widget,
+                            GtkTextDirection  previous_direction)
+{
+  HdyFlap *self = HDY_FLAP (widget);
+
+  update_swipe_tracker (self);
+
+  GTK_WIDGET_CLASS (hdy_flap_parent_class)->direction_changed (widget,
+                                                               previous_direction);
+}
+
+static void
+hdy_flap_forall (GtkContainer *container,
+                 gboolean      include_internals,
+                 GtkCallback   callback,
+                 gpointer      callback_data)
+{
+  HdyFlap *self = HDY_FLAP (container);
+
+  if (self->content.widget)
+    callback (self->content.widget, callback_data);
+
+  if (self->separator.widget)
+    callback (self->separator.widget, callback_data);
+
+  if (self->flap.widget)
+    callback (self->flap.widget, callback_data);
+}
+
+static void
+hdy_flap_add (GtkContainer *container,
+              GtkWidget    *widget)
+{
+  HdyFlap *self = HDY_FLAP (container);
+
+  if (self->content.widget) {
+    g_warning ("Attempting to add a widget with type %s to a %s, "
+               "but %s can only contain one widget at a time; "
+               "it already contains a widget of type %s",
+               g_type_name (G_OBJECT_TYPE (widget)),
+               g_type_name (G_OBJECT_TYPE (self)),
+               g_type_name (G_OBJECT_TYPE (self)),
+               g_type_name (G_OBJECT_TYPE (self->content.widget)));
+
+    return;
+  }
+
+  self->content.widget = widget;
+
+  add_child (self, &self->content);
+}
+
+static void
+hdy_flap_remove (GtkContainer *container,
+                 GtkWidget    *widget)
+{
+  HdyFlap *self = HDY_FLAP (container);
+
+  if (widget == self->flap.widget) {
+    hdy_flap_set_flap (self, NULL);
+
+    return;
+  }
+
+  if (widget == self->separator.widget) {
+    hdy_flap_set_separator (self, NULL);
+
+    return;
+  }
+
+  if (widget == self->content.widget) {
+    remove_child (self, &self->content);
+    self->content.widget = NULL;
+
+    return;
+  }
+
+  g_return_if_reached ();
+}
+
+static void
+hdy_flap_get_property (GObject    *object,
+                       guint       prop_id,
+                       GValue     *value,
+                       GParamSpec *pspec)
+{
+  HdyFlap *self = HDY_FLAP (object);
+
+  switch (prop_id) {
+  case PROP_FLAP:
+    g_value_set_object (value, hdy_flap_get_flap (self));
+    break;
+  case PROP_SEPARATOR:
+    g_value_set_object (value, hdy_flap_get_separator (self));
+    break;
+  case PROP_FLAP_POSITION:
+    g_value_set_enum (value, hdy_flap_get_flap_position (self));
+    break;
+  case PROP_REVEAL_FLAP:
+    g_value_set_boolean (value, hdy_flap_get_reveal_flap (self));
+    break;
+  case PROP_REVEAL_DURATION:
+    g_value_set_uint (value, hdy_flap_get_reveal_duration (self));
+    break;
+  case PROP_REVEAL_PROGRESS:
+    g_value_set_double (value, hdy_flap_get_reveal_progress (self));
+    break;
+  case PROP_FOLD_POLICY:
+    g_value_set_enum (value, hdy_flap_get_fold_policy (self));
+    break;
+  case PROP_FOLD_DURATION:
+    g_value_set_uint (value, hdy_flap_get_fold_duration (self));
+    break;
+  case PROP_FOLDED:
+    g_value_set_boolean (value, hdy_flap_get_folded (self));
+    break;
+  case PROP_LOCKED:
+    g_value_set_boolean (value, hdy_flap_get_locked (self));
+    break;
+  case PROP_TRANSITION_TYPE:
+    g_value_set_enum (value, hdy_flap_get_transition_type (self));
+    break;
+  case PROP_MODAL:
+    g_value_set_boolean (value, hdy_flap_get_modal (self));
+    break;
+  case PROP_SWIPE_TO_OPEN:
+    g_value_set_boolean (value, hdy_flap_get_swipe_to_open (self));
+    break;
+  case PROP_SWIPE_TO_CLOSE:
+    g_value_set_boolean (value, hdy_flap_get_swipe_to_close (self));
+    break;
+  case PROP_ORIENTATION:
+    g_value_set_enum (value, self->orientation);
+    break;
+  default:
+    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+  }
+}
+
+static void
+hdy_flap_set_property (GObject      *object,
+                       guint         prop_id,
+                       const GValue *value,
+                       GParamSpec   *pspec)
+{
+  HdyFlap *self = HDY_FLAP (object);
+
+  switch (prop_id) {
+  case PROP_FLAP:
+    hdy_flap_set_flap (self, g_value_get_object (value));
+    break;
+  case PROP_SEPARATOR:
+    hdy_flap_set_separator (self, g_value_get_object (value));
+    break;
+  case PROP_FLAP_POSITION:
+    hdy_flap_set_flap_position (self, g_value_get_enum (value));
+    break;
+  case PROP_REVEAL_FLAP:
+    hdy_flap_set_reveal_flap (self, g_value_get_boolean (value));
+    break;
+  case PROP_REVEAL_DURATION:
+    hdy_flap_set_reveal_duration (self, g_value_get_uint (value));
+    break;
+  case PROP_FOLD_POLICY:
+    hdy_flap_set_fold_policy (self, g_value_get_enum (value));
+    break;
+  case PROP_FOLD_DURATION:
+    hdy_flap_set_fold_duration (self, g_value_get_uint (value));
+    break;
+  case PROP_LOCKED:
+    hdy_flap_set_locked (self, g_value_get_boolean (value));
+    break;
+  case PROP_TRANSITION_TYPE:
+    hdy_flap_set_transition_type (self, g_value_get_enum (value));
+    break;
+  case PROP_MODAL:
+    hdy_flap_set_modal (self, g_value_get_boolean (value));
+    break;
+  case PROP_SWIPE_TO_OPEN:
+    hdy_flap_set_swipe_to_open (self, g_value_get_boolean (value));
+    break;
+  case PROP_SWIPE_TO_CLOSE:
+    hdy_flap_set_swipe_to_close (self, g_value_get_boolean (value));
+    break;
+  case PROP_ORIENTATION:
+    set_orientation (self, g_value_get_enum (value));
+    break;
+  default:
+    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+  }
+}
+
+static void
+hdy_flap_dispose (GObject *object)
+{
+  HdyFlap *self = HDY_FLAP (object);
+
+  g_clear_object (&self->shadow_helper);
+  g_clear_object (&self->tracker);
+  g_clear_object (&self->click_gesture);
+  g_clear_object (&self->key_controller);
+
+  G_OBJECT_CLASS (hdy_flap_parent_class)->dispose (object);
+}
+
+static void
+hdy_flap_class_init (HdyFlapClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+  GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
+
+  object_class->get_property = hdy_flap_get_property;
+  object_class->set_property = hdy_flap_set_property;
+  object_class->dispose = hdy_flap_dispose;
+
+  widget_class->get_preferred_width = hdy_flap_get_preferred_width;
+  widget_class->get_preferred_width_for_height = hdy_flap_get_preferred_width_for_height;
+  widget_class->get_preferred_height = hdy_flap_get_preferred_height;
+  widget_class->get_preferred_height_for_width = hdy_flap_get_preferred_height_for_width;
+  widget_class->size_allocate = hdy_flap_size_allocate;
+  widget_class->draw = hdy_flap_draw;
+  widget_class->realize = hdy_flap_realize;
+  widget_class->unrealize = hdy_flap_unrealize;
+  widget_class->direction_changed = hdy_flap_direction_changed;
+
+  container_class->remove = hdy_flap_remove;
+  container_class->add = hdy_flap_add;
+  container_class->forall = hdy_flap_forall;
+
+  /**
+   * HdyFlap:flap:
+   *
+   * Since: 1.1
+   */
+  props[PROP_FLAP] =
+    g_param_spec_object ("flap",
+                         _("Flap widget"),
+                         _("Flap Widget"),
+                         GTK_TYPE_WIDGET,
+                         G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:separator:
+   *
+   * Since: 1.1
+   */
+  props[PROP_SEPARATOR] =
+    g_param_spec_object ("separator",
+                         _("Separator widget"),
+                         _("Separator Widget"),
+                         GTK_TYPE_WIDGET,
+                         G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:flap-position:
+   *
+   * Since: 1.1
+   */
+  props[PROP_FLAP_POSITION] =
+    g_param_spec_enum ("flap-position",
+                       _("Flap Position"),
+                       _("Flap Position"),
+                       GTK_TYPE_PACK_TYPE,
+                       GTK_PACK_START,
+                       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:reveal-flap:
+   *
+   * Since: 1.1
+   */
+  props[PROP_REVEAL_FLAP] =
+    g_param_spec_boolean ("reveal-flap",
+                          _("Reveal Flap"),
+                          _("Reveal Flap"),
+                          TRUE,
+                          G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:reveal-duration:
+   *
+   * The reveal transition animation duration, in milliseconds.
+   *
+   * Since: 1.1
+   */
+  props[PROP_REVEAL_DURATION] =
+    g_param_spec_uint ("reveal-duration",
+                       _("Reveal Duration"),
+                       _("The reveal transition animation duration, in milliseconds"),
+                       0, G_MAXINT,
+                       250,
+                       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:reveal-progress:
+   *
+   * Since: 1.1
+   */
+  props[PROP_REVEAL_PROGRESS] =
+    g_param_spec_double ("reveal-progress",
+                          _("Reveal progress"),
+                          _("Reveal progress"),
+                          0.0, 1.0, 1.0,
+                          G_PARAM_READABLE);
+
+  /**
+   * HdyFlap:fold-policy:
+   *
+   * Since: 1.1
+   */
+  props[PROP_FOLD_POLICY] =
+    g_param_spec_enum ("fold-policy",
+                       _("Fold Policy"),
+                       _("Fold Policy"),
+                       HDY_TYPE_FLAP_FOLD_POLICY,
+                       HDY_FLAP_FOLD_POLICY_AUTO,
+                       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:fold-duration:
+   *
+   * The fold transition animation duration, in milliseconds.
+   *
+   * Since: 1.1
+   */
+  props[PROP_FOLD_DURATION] =
+    g_param_spec_uint ("fold-duration",
+                       _("Fold Duration"),
+                       _("The fold transition animation duration, in milliseconds"),
+                       0, G_MAXINT,
+                       250,
+                       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:folded:
+   *
+   * Whether the flap is currently folded.
+   *
+   * See #HdyFlap:fold-policy.
+   *
+   * Since: 1.1
+   */
+  props[PROP_FOLDED] =
+    g_param_spec_boolean ("folded",
+                          _("Folded"),
+                          _("Whether the flap is currently folded"),
+                          FALSE,
+                          G_PARAM_READABLE);
+
+  /**
+   * HdyFlap:locked:
+   *
+   * Whether the flap is locked.
+   *
+   * If %FALSE, folding when the flap is revealed automatically closes it, and
+   * unfolding it when the flap is not revealed opens it. If %TRUE,
+   * #HdyFlap:reveal-flap value never changes on its own.
+   *
+   * Since: 1.1
+   */
+  props[PROP_LOCKED] =
+    g_param_spec_boolean ("locked",
+                          _("Locked"),
+                          _("Whether the flap is locked"),
+                          FALSE,
+                          G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:transition-type:
+   *
+   * The type of animation that will be used for reveal and fold transitions
+   * in @self.
+   *
+   * Since: 1.1
+   */
+  props[PROP_TRANSITION_TYPE] =
+    g_param_spec_enum ("transition-type",
+                       _("Transition Type"),
+                       _("The type of animation used for reveal and fold transitions"),
+                       HDY_TYPE_FLAP_TRANSITION_TYPE,
+                       HDY_FLAP_TRANSITION_TYPE_OVER,
+                       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:modal:
+   *
+   * Whether the flap is modal.
+   *
+   * If %TRUE, clicking the content widget while flap is revealed, as well as
+   * pressing Escape key, will close the flap. If %FALSE, clicks are passed
+   * through to the content widget.
+   *
+   * Since: 1.1
+   */
+  props[PROP_MODAL] =
+    g_param_spec_boolean ("modal",
+                          _("Modal"),
+                          _("Whether the flap is modal"),
+                          TRUE,
+                          G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:swipe-to-open:
+   *
+   * Whether the flap can be opened with a swipe gesture.
+   *
+   * The area that can be swiped depends on the #HdyFlap:transition-type value.
+   *
+   * Since: 1.1
+   */
+  props[PROP_SWIPE_TO_OPEN] =
+    g_param_spec_boolean ("swipe-to-open",
+                          _("Swipe to Open"),
+                          _("Whether the flap can be opened with a swipe gesture"),
+                          TRUE,
+                          G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  /**
+   * HdyFlap:swipe-to-close:
+   *
+   * Whether the flap can be closed with a swipe gesture.
+   *
+   * The area that can be swiped depends on the #HdyFlap:transition-type value.
+   *
+   * Since: 1.1
+   */
+  props[PROP_SWIPE_TO_CLOSE] =
+    g_param_spec_boolean ("swipe-to-close",
+                          _("Swipe to Close"),
+                          _("Whether the flap can be closed with a swipe gesture"),
+                          TRUE,
+                          G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+  g_object_class_install_properties (object_class, LAST_PROP, props);
+
+  g_object_class_override_property (object_class,
+                                    PROP_ORIENTATION,
+                                    "orientation");
+
+  gtk_widget_class_set_css_name (widget_class, "flap");
+}
+
+static void
+hdy_flap_init (HdyFlap *self)
+{
+  gtk_widget_add_events (GTK_WIDGET (self), GDK_KEY_PRESS_MASK);
+
+  self->orientation = GTK_ORIENTATION_HORIZONTAL;
+  self->flap_position = GTK_PACK_START;
+  self->fold_policy = HDY_FLAP_FOLD_POLICY_AUTO;
+  self->transition_type = HDY_FLAP_TRANSITION_TYPE_OVER;
+  self->reveal_flap = TRUE;
+  self->locked = FALSE;
+  self->reveal_progress = 1;
+  self->folded = FALSE;
+  self->fold_progress = 0;
+  self->fold_duration = 250;
+  self->reveal_duration = 250;
+  self->modal = TRUE;
+  self->swipe_to_open = TRUE;
+  self->swipe_to_close = TRUE;
+
+  self->shadow_helper = hdy_shadow_helper_new (GTK_WIDGET (self));
+  self->tracker = hdy_swipe_tracker_new (HDY_SWIPEABLE (self));
+  hdy_swipe_tracker_set_enabled (self->tracker, FALSE);
+
+  g_signal_connect_object (self->tracker, "begin-swipe", G_CALLBACK (begin_swipe_cb), self, 0);
+  g_signal_connect_object (self->tracker, "update-swipe", G_CALLBACK (update_swipe_cb), self, 0);
+  g_signal_connect_object (self->tracker, "end-swipe", G_CALLBACK (end_swipe_cb), self, 0);
+
+  update_swipe_tracker (self);
+
+  self->click_gesture = gtk_gesture_multi_press_new (GTK_WIDGET (self));
+  gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (self->click_gesture), TRUE);
+  gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (self->click_gesture), GDK_BUTTON_PRIMARY);
+  gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (self->click_gesture),
+                                              GTK_PHASE_CAPTURE);
+  g_signal_connect_object (self->click_gesture, "pressed", G_CALLBACK (pressed_cb), self, 0);
+
+  self->key_controller = gtk_event_controller_key_new (GTK_WIDGET (self));
+  gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (self->key_controller),
+                                              GTK_PHASE_BUBBLE);
+  g_signal_connect_object (self->key_controller, "key-pressed", G_CALLBACK (key_pressed_cb), self, 0);
+}
+
+static void
+hdy_flap_add_child (GtkBuildable *buildable,
+                    GtkBuilder   *builder,
+                    GObject      *child,
+                    const gchar  *type)
+{
+  if (!type)
+    gtk_container_add (GTK_CONTAINER (buildable), GTK_WIDGET (child));
+  else if (!g_strcmp0 (type, "flap"))
+    hdy_flap_set_flap (HDY_FLAP (buildable), GTK_WIDGET (child));
+  else if (!g_strcmp0 (type, "separator"))
+    hdy_flap_set_separator (HDY_FLAP (buildable), GTK_WIDGET (child));
+  else
+    GTK_BUILDER_WARN_INVALID_CHILD_TYPE (HDY_FLAP (buildable), type);
+}
+
+static void
+hdy_flap_buildable_init (GtkBuildableIface *iface)
+{
+  iface->add_child = hdy_flap_add_child;
+}
+
+static void
+hdy_flap_switch_child (HdySwipeable *swipeable,
+                       guint         index,
+                       gint64        duration)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+
+  set_reveal_flap (self, index > 0, duration, FALSE);
+}
+
+static HdySwipeTracker *
+hdy_flap_get_swipe_tracker (HdySwipeable *swipeable)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+
+  return self->tracker;
+}
+
+static gdouble
+hdy_flap_get_distance (HdySwipeable *swipeable)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+  gint flap, separator;
+
+  if (!self->flap.widget)
+    return 0;
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL) {
+    flap = self->flap.allocation.width;
+    separator = self->separator.allocation.width;
+  } else {
+    flap = self->flap.allocation.height;
+    separator = self->separator.allocation.height;
+  }
+
+  if (transition_is_content_above_flap (self))
+    return flap + separator;
+
+  return flap + separator * (1 - self->fold_progress);
+}
+
+static gdouble *
+hdy_flap_get_snap_points (HdySwipeable *swipeable,
+                          gint         *n_snap_points)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+  gboolean can_open = self->reveal_progress > 0 || self->swipe_to_open || self->swipe_active;
+  gboolean can_close = self->reveal_progress < 1 || self->swipe_to_close || self->swipe_active;
+  gdouble *points;
+
+  if (!can_open && !can_close)
+    return NULL;
+
+  if (can_open && can_close) {
+    points = g_new0 (gdouble, 2);
+
+    if (n_snap_points)
+      *n_snap_points = 2;
+
+    points[0] = 0;
+    points[1] = 1;
+
+    return points;
+  }
+
+  points = g_new0 (gdouble, 1);
+
+  if (n_snap_points)
+    *n_snap_points = 1;
+
+  points[0] = can_open ? 1 : 0;
+
+  return points;
+}
+
+static gdouble
+hdy_flap_get_progress (HdySwipeable *swipeable)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+
+  return self->reveal_progress;
+}
+
+static gdouble
+hdy_flap_get_cancel_progress (HdySwipeable *swipeable)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+
+  return round (self->reveal_progress);
+}
+
+static void
+hdy_flap_get_swipe_area (HdySwipeable           *swipeable,
+                         HdyNavigationDirection  navigation_direction,
+                         gboolean                is_drag,
+                         GdkRectangle           *rect)
+{
+  HdyFlap *self = HDY_FLAP (swipeable);
+  GtkAllocation *alloc;
+  gint width, height;
+  gdouble flap_factor, content_factor;
+  gboolean content_above_flap;
+
+  if (!self->flap.widget) {
+    rect->x = 0;
+    rect->y = 0;
+    rect->width = 0;
+    rect->height = 0;
+
+    return;
+  }
+
+  width = gtk_widget_get_allocated_width (GTK_WIDGET (self));
+  height = gtk_widget_get_allocated_height (GTK_WIDGET (self));
+
+  content_above_flap = transition_is_content_above_flap (self);
+  flap_factor = transition_get_flap_motion_factor (self);
+  content_factor = transition_get_content_motion_factor (self);
+
+  if (!is_drag ||
+      (flap_factor >= 1 && content_factor >= 1) ||
+      (self->fold_progress < 1 && flap_factor > 0)) {
+    rect->x = 0;
+    rect->y = 0;
+    rect->width = width;
+    rect->height = height;
+
+    return;
+  }
+
+  alloc = content_above_flap
+    ? &self->content.allocation
+    : &self->flap.allocation;
+
+  if (self->orientation == GTK_ORIENTATION_HORIZONTAL) {
+    if (alloc->x <= 0) {
+      rect->x = 0;
+      rect->width = MAX (alloc->width + alloc->x, HDY_SWIPE_BORDER);
+    } else if (alloc->x + alloc->width >= width) {
+      rect->width = MAX (width - alloc->x, HDY_SWIPE_BORDER);
+      rect->x = width - rect->width;
+    } else {
+      g_assert_not_reached ();
+    }
+
+    rect->y = alloc->y;
+    rect->height = alloc->height;
+  } else {
+    if (alloc->y <= 0) {
+      rect->y = 0;
+      rect->height = MAX (alloc->height + alloc->y, HDY_SWIPE_BORDER);
+    } else if (alloc->y + alloc->height >= height) {
+      rect->height = MAX (height - alloc->y, HDY_SWIPE_BORDER);
+      rect->y = height - rect->height;
+    } else {
+      g_assert_not_reached ();
+    }
+
+    rect->x = alloc->x;
+    rect->width = alloc->width;
+  }
+}
+
+static void
+hdy_flap_swipeable_init (HdySwipeableInterface *iface)
+{
+  iface->switch_child = hdy_flap_switch_child;
+  iface->get_swipe_tracker = hdy_flap_get_swipe_tracker;
+  iface->get_distance = hdy_flap_get_distance;
+  iface->get_snap_points = hdy_flap_get_snap_points;
+  iface->get_progress = hdy_flap_get_progress;
+  iface->get_cancel_progress = hdy_flap_get_cancel_progress;
+  iface->get_swipe_area = hdy_flap_get_swipe_area;
+}
+
+/**
+ * hdy_flap_new:
+ *
+ * Creates a new #HdyFlap.
+ *
+ * Returns: a new #HdyFlap
+ *
+ * Since: 1.1
+ */
+GtkWidget *
+hdy_flap_new (void)
+{
+  return g_object_new (HDY_TYPE_FLAP, NULL);
+}
+
+/**
+ * hdy_flap_get_content:
+ * @self: a #HdyFlap
+ *
+ * Returns: (transfer none) (nullable): Flap child.
+ *
+ * Since: 1.1
+ */
+GtkWidget *
+hdy_flap_get_content (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), NULL);
+
+  return self->content.widget;
+}
+
+/**
+ * hdy_flap_get_flap:
+ * @self: a #HdyFlap
+ *
+ * Returns: (transfer none) (nullable): Flap child.
+ *
+ * Since: 1.1
+ */
+GtkWidget *
+hdy_flap_get_flap (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), NULL);
+
+  return self->flap.widget;
+}
+
+/**
+ * hdy_flap_set_flap:
+ * @self: a #HdyFlap
+ * @flap: (nullable): flap widget
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_flap (HdyFlap   *self,
+                   GtkWidget *flap)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+  g_return_if_fail (GTK_IS_WIDGET (flap) || flap == NULL);
+
+  if (self->flap.widget == flap)
+    return;
+
+  if (self->flap.widget)
+    remove_child (self, &self->flap);
+
+  self->flap.widget = flap;
+
+  if (self->flap.widget)
+    add_child (self, &self->flap);
+
+  update_swipe_tracker (self);
+  update_child_visibility (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_FLAP]);
+}
+
+/**
+ * hdy_flap_get_separator:
+ * @self: a #HdyFlap
+ *
+ * Returns: (transfer none) (nullable): Flap child.
+ *
+ * Since: 1.1
+ */
+GtkWidget *
+hdy_flap_get_separator (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), NULL);
+
+  return self->separator.widget;
+}
+
+/**
+ * hdy_flap_set_seoarator:
+ * @self: a #HdyFlap
+ * @separator: (nullable): flap widget
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_separator (HdyFlap   *self,
+                        GtkWidget *separator)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+  g_return_if_fail (GTK_IS_WIDGET (separator) || separator == NULL);
+
+  if (self->separator.widget == separator)
+    return;
+
+  if (self->separator.widget)
+    remove_child (self, &self->separator);
+
+  self->separator.widget = separator;
+
+  if (self->separator.widget)
+    add_child (self, &self->separator);
+
+  update_child_visibility (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SEPARATOR]);
+}
+
+/**
+ * hdy_flap_get_flap_position:
+ * @self: a #HdyFlap
+ *
+ * Returns: current flap position.
+ *
+ * Since: 1.1
+ */
+GtkPackType
+hdy_flap_get_flap_position (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), GTK_PACK_START);
+
+  return self->flap_position;
+}
+
+/**
+ * hdy_flap_set_flap_position:
+ * @self: a #HdyFlap
+ * @position: Flap position
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_flap_position (HdyFlap     *self,
+                            GtkPackType  position)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  if (self->flap_position == position)
+    return;
+
+  self->flap_position = position;
+
+  gtk_widget_queue_allocate (GTK_WIDGET (self));
+  hdy_shadow_helper_clear_cache (self->shadow_helper);
+  update_swipe_tracker (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_FLAP_POSITION]);
+}
+
+/**
+ * hdy_flap_get_reveal_flap:
+ * @self: a #HdyFlap
+ *
+ * Returns: %TRUE if the flap widget is revealed, %FALSE otherwise.
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_reveal_flap (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->reveal_flap;
+}
+
+/**
+ * hdy_flap_set_reveal_flap:
+ * @self: a #HdyFlap
+ * @reveal_flap: Reveal flap
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_reveal_flap (HdyFlap  *self,
+                          gboolean  reveal_flap)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  set_reveal_flap (self, reveal_flap, self->reveal_duration, TRUE);
+}
+
+/**
+ * hdy_flap_get_reveal_duration:
+ * @self: a #HdyFlap
+ *
+ * Returns the amount of time (in milliseconds) that reveal transitions in @self
+ * will take.
+ *
+ * Returns: the reveal transition duration
+ *
+ * Since: 1.1
+ */
+guint
+hdy_flap_get_reveal_duration (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), 0);
+
+  return self->reveal_duration;
+}
+
+/**
+ * hdy_flap_set_reveal_duration:
+ * @self: a #HdyFlap
+ * @duration: the new duration, in milliseconds
+ *
+ * Sets the duration that reveal transitions in @self will take.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_reveal_duration (HdyFlap *self,
+                              guint    duration)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  if (self->reveal_duration == duration)
+    return;
+
+  self->reveal_duration = duration;
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_REVEAL_DURATION]);
+}
+
+/**
+ * hdy_flap_get_reveal_progress:
+ * @self: a #HdyFlap
+ *
+ * Returns: %TRUE if flap widget is click to close.
+ *
+ * Since: 1.1
+ */
+gdouble
+hdy_flap_get_reveal_progress (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), 0.0);
+
+  return self->reveal_progress;
+}
+
+/**
+ * hdy_flap_get_fold_policy:
+ * @self: a #HdyFlap
+ *
+ * Returns: current fold policy.
+ *
+ * Since: 1.1
+ */
+HdyFlapFoldPolicy
+hdy_flap_get_fold_policy (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), HDY_FLAP_FOLD_POLICY_NEVER);
+
+  return self->fold_policy;
+}
+
+/**
+ * hdy_flap_set_fold_policy:
+ * @self: a #HdyFlap
+ * @policy: Flap fold policy
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_fold_policy (HdyFlap           *self,
+                          HdyFlapFoldPolicy  policy)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  if (self->fold_policy == policy)
+    return;
+
+  self->fold_policy = policy;
+
+  switch (self->fold_policy) {
+  case HDY_FLAP_FOLD_POLICY_NEVER:
+    set_folded (self, FALSE);
+    break;
+
+  case HDY_FLAP_FOLD_POLICY_ALWAYS:
+    set_folded (self, TRUE);
+    break;
+
+  case HDY_FLAP_FOLD_POLICY_AUTO:
+    gtk_widget_queue_allocate (GTK_WIDGET (self));
+    break;
+
+  default:
+    g_assert_not_reached ();
+  }
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_FOLD_POLICY]);
+}
+
+/**
+ * hdy_flap_get_fold_duration:
+ * @self: a #HdyFlap
+ *
+ * Returns the amount of time (in milliseconds) that fold transitions in @self
+ * will take.
+ *
+ * Returns: the fold transition duration
+ *
+ * Since: 1.1
+ */
+guint
+hdy_flap_get_fold_duration (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), 0);
+
+  return self->fold_duration;
+}
+
+/**
+ * hdy_flap_set_fold_duration:
+ * @self: a #HdyFlap
+ * @duration: the new duration, in milliseconds
+ *
+ * Sets the duration that fold transitions in @self will take.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_fold_duration (HdyFlap *self,
+                            guint    duration)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  if (self->fold_duration == duration)
+    return;
+
+  self->fold_duration = duration;
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_FOLD_DURATION]);
+}
+
+/**
+ * hdy_flap_get_folded:
+ * @self: a #HdyFlap
+ *
+ * Gets whether @self is currently folded.
+ *
+ * See #HdyFlap:fold-policy.
+ *
+ * Returns: %TRUE if @self is currently folded, %FALSE otherwise
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_folded (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->folded;
+}
+
+/**
+ * hdy_flap_get_locked:
+ * @self: a #HdyFlap
+ *
+ * Gets whether @self is locked.
+ *
+ * Returns: %TRUE if @self is locked, %FALSE otherwise
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_locked (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->locked;
+}
+
+/**
+ * hdy_flap_set_locked:
+ * @self: a #HdyFlap
+ * @locked: the new value
+ *
+ * Sets whether @self is locked.
+ *
+ * If %FALSE, folding @self when the flap is revealed automatically closes it,
+ * and unfolding it when the flap is not revealed opens it. If %TRUE,
+ * #HdyFlap:reveal-flap value never changes on its own.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_locked (HdyFlap  *self,
+                     gboolean  locked)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  locked = !!locked;
+
+  if (self->locked == locked)
+    return;
+
+  self->locked = locked;
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_LOCKED]);
+}
+
+/**
+ * hdy_flap_get_transition_type:
+ * @self: a #HdyFlap
+ *
+ * Gets the type of animation that will be used for reveal and fold transitions
+ * in @self.
+ *
+ * Returns: the current transition type of @self
+ *
+ * Since: 1.1
+ */
+HdyFlapTransitionType
+hdy_flap_get_transition_type (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), HDY_FLAP_TRANSITION_TYPE_OVER);
+
+  return self->transition_type;
+}
+
+/**
+ * hdy_flap_set_transition_type:
+ * @self: a #HdyFlap
+ * @transition_type: the new transition type
+ *
+ * Sets the type of animation that will be used for reveal and fold transitions
+ * in @self.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_transition_type (HdyFlap               *self,
+                              HdyFlapTransitionType  transition_type)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  if (self->transition_type == transition_type)
+    return;
+
+  self->transition_type = transition_type;
+
+  restack_windows (self);
+
+  if (self->reveal_progress > 0 || (self->fold_progress > 0 && self->fold_progress < 1))
+    gtk_widget_queue_allocate (GTK_WIDGET (self));
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_TRANSITION_TYPE]);
+}
+
+/**
+ * hdy_flap_get_modal:
+ * @self: a #HdyFlap
+ *
+ * Gets whether the @self is modal. See hdy_flap_set_modal().
+ *
+ * Returns: %TRUE if @self is modal
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_modal (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->modal;
+}
+
+/**
+ * hdy_flap_set_modal:
+ * @self: a #HdyFlap
+ * @modal: Whether @self can be closed with a click
+ *
+ * Sets whether the @self can be closed with a click.
+ *
+ * If @modal %TRUE, clicking the content widget while flap is revealed, as well
+ * as pressing Escape key, will close the flap. If %FALSE, clicks are passed
+ * through to the content widget.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_modal (HdyFlap  *self,
+                    gboolean  modal)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  modal = !!modal;
+
+  if (self->modal == modal)
+    return;
+
+  self->modal = modal;
+
+  gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (self->click_gesture),
+                                              modal ? GTK_PHASE_CAPTURE : GTK_PHASE_NONE);
+  gtk_event_controller_set_propagation_phase (self->key_controller,
+                                              modal ? GTK_PHASE_BUBBLE : GTK_PHASE_NONE);
+
+  gtk_widget_queue_allocate (GTK_WIDGET (self));
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_MODAL]);
+}
+
+/**
+ * hdy_flap_get_swipe_to_open:
+ * @self: a #HdyFlap
+ *
+ * Gets whether @self can be opened with a swipe gesture.
+ *
+ * Returns: %TRUE if @self can be opened with a swipe gesture
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_swipe_to_open (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->swipe_to_open;
+}
+
+/**
+ * hdy_flap_set_swipe_to_open:
+ * @self: a #HdyFlap
+ * @swipe_to_open: Whether @self can be opened with a swipe gesture
+ *
+ * Sets whether @self can be opened with a swipe gesture.
+ *
+ * The area that can be swiped depends on the #HdyFlap:transition-type value.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_swipe_to_open (HdyFlap  *self,
+                            gboolean  swipe_to_open)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  swipe_to_open = !!swipe_to_open;
+
+  if (self->swipe_to_open == swipe_to_open)
+    return;
+
+  self->swipe_to_open = swipe_to_open;
+
+  update_swipe_tracker (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SWIPE_TO_OPEN]);
+}
+
+/**
+ * hdy_flap_get_swipe_to_close:
+ * @self: a #HdyFlap
+ *
+ * Gets whether @self can be closed with a swipe gesture.
+ *
+ * Returns: %TRUE if @self can be closed with a swipe gesture
+ *
+ * Since: 1.1
+ */
+gboolean
+hdy_flap_get_swipe_to_close (HdyFlap *self)
+{
+  g_return_val_if_fail (HDY_IS_FLAP (self), FALSE);
+
+  return self->swipe_to_close;
+}
+
+/**
+ * hdy_flap_set_swipe_to_close:
+ * @self: a #HdyFlap
+ * @swipe_to_close: Whether @self can be closed with a swipe gesture
+ *
+ * Sets whether @self can be closed with a swipe gesture.
+ *
+ * The area that can be swiped depends on the #HdyFlap:transition-type value.
+ *
+ * Since: 1.1
+ */
+void
+hdy_flap_set_swipe_to_close (HdyFlap  *self,
+                             gboolean  swipe_to_close)
+{
+  g_return_if_fail (HDY_IS_FLAP (self));
+
+  swipe_to_close = !!swipe_to_close;
+
+  if (self->swipe_to_close == swipe_to_close)
+    return;
+
+  self->swipe_to_close = swipe_to_close;
+
+  update_swipe_tracker (self);
+
+  g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SWIPE_TO_CLOSE]);
+}
diff --git a/src/hdy-flap.h b/src/hdy-flap.h
new file mode 100644
index 00000000..d11ae6e1
--- /dev/null
+++ b/src/hdy-flap.h
@@ -0,0 +1,122 @@
+/*
+ * Copyright (C) 2020 Felix Häcker <haeckerfelix gnome org>
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ */
+
+#pragma once
+
+#if !defined(_HANDY_INSIDE) && !defined(HANDY_COMPILATION)
+#error "Only <handy.h> can be included directly."
+#endif
+
+#include "hdy-version.h"
+
+#include <gtk/gtk.h>
+#include "hdy-enums.h"
+
+G_BEGIN_DECLS
+
+#define HDY_TYPE_FLAP (hdy_flap_get_type ())
+
+HDY_AVAILABLE_IN_1_1
+G_DECLARE_FINAL_TYPE (HdyFlap, hdy_flap, HDY, FLAP, GtkContainer)
+
+typedef enum {
+  HDY_FLAP_FOLD_POLICY_NEVER,
+  HDY_FLAP_FOLD_POLICY_ALWAYS,
+  HDY_FLAP_FOLD_POLICY_AUTO,
+} HdyFlapFoldPolicy;
+
+typedef enum {
+  HDY_FLAP_TRANSITION_TYPE_OVER,
+  HDY_FLAP_TRANSITION_TYPE_UNDER,
+  HDY_FLAP_TRANSITION_TYPE_SLIDE,
+  HDY_FLAP_TRANSITION_TYPE_OVER_TRANSPARENT,
+} HdyFlapTransitionType;
+
+HDY_AVAILABLE_IN_1_1
+GtkWidget *hdy_flap_new (void);
+
+HDY_AVAILABLE_IN_1_1
+GtkWidget *hdy_flap_get_content (HdyFlap *self);
+
+HDY_AVAILABLE_IN_1_1
+GtkWidget *hdy_flap_get_flap (HdyFlap   *self);
+HDY_AVAILABLE_IN_1_1
+void       hdy_flap_set_flap (HdyFlap   *self,
+                              GtkWidget *flap);
+
+HDY_AVAILABLE_IN_1_1
+GtkWidget *hdy_flap_get_separator (HdyFlap   *self);
+HDY_AVAILABLE_IN_1_1
+void       hdy_flap_set_separator (HdyFlap   *self,
+                                   GtkWidget *separator);
+
+HDY_AVAILABLE_IN_1_1
+GtkPackType hdy_flap_get_flap_position (HdyFlap *self);
+HDY_AVAILABLE_IN_1_1
+void        hdy_flap_set_flap_position (HdyFlap     *self,
+                                        GtkPackType  position);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_reveal_flap (HdyFlap *self);
+HDY_AVAILABLE_IN_1_1
+void     hdy_flap_set_reveal_flap (HdyFlap  *self,
+                                   gboolean  reveal_flap);
+
+HDY_AVAILABLE_IN_1_1
+guint hdy_flap_get_reveal_duration (HdyFlap *self);
+HDY_AVAILABLE_IN_1_1
+void  hdy_flap_set_reveal_duration (HdyFlap *self,
+                                    guint    duration);
+
+HDY_AVAILABLE_IN_1_1
+gdouble hdy_flap_get_reveal_progress (HdyFlap *self);
+
+HDY_AVAILABLE_IN_1_1
+HdyFlapFoldPolicy hdy_flap_get_fold_policy (HdyFlap           *self);
+HDY_AVAILABLE_IN_1_1
+void              hdy_flap_set_fold_policy (HdyFlap           *self,
+                                            HdyFlapFoldPolicy  policy);
+
+HDY_AVAILABLE_IN_1_1
+guint hdy_flap_get_fold_duration (HdyFlap *self);
+HDY_AVAILABLE_IN_1_1
+void  hdy_flap_set_fold_duration (HdyFlap *self,
+                                  guint    duration);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_folded (HdyFlap *self);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_locked (HdyFlap *self);
+HDY_AVAILABLE_IN_1_1
+void     hdy_flap_set_locked (HdyFlap  *self,
+                              gboolean  locked);
+
+HDY_AVAILABLE_IN_1_1
+HdyFlapTransitionType hdy_flap_get_transition_type (HdyFlap               *self);
+HDY_AVAILABLE_IN_1_1
+void                  hdy_flap_set_transition_type (HdyFlap               *self,
+                                                    HdyFlapTransitionType  transition_type);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_modal (HdyFlap  *self);
+HDY_AVAILABLE_IN_1_1
+void     hdy_flap_set_modal (HdyFlap  *self,
+                             gboolean  modal);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_swipe_to_open (HdyFlap  *self);
+HDY_AVAILABLE_IN_1_1
+void     hdy_flap_set_swipe_to_open (HdyFlap  *self,
+                                     gboolean  swipe_to_open);
+
+HDY_AVAILABLE_IN_1_1
+gboolean hdy_flap_get_swipe_to_close (HdyFlap  *self);
+HDY_AVAILABLE_IN_1_1
+void     hdy_flap_set_swipe_to_close (HdyFlap  *self,
+                                      gboolean  swipe_to_close);
+
+G_END_DECLS
diff --git a/src/meson.build b/src/meson.build
index 3e1c181c..af850b0d 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -9,6 +9,7 @@ libhandy_resources = gnome.compile_resources(
 
 hdy_public_enum_headers = [
   'hdy-deck.h',
+  'hdy-flap.h',
   'hdy-header-bar.h',
   'hdy-header-group.h',
   'hdy-leaflet.h',
@@ -77,6 +78,7 @@ src_headers = [
   'hdy-deprecation-macros.h',
   'hdy-enum-value-object.h',
   'hdy-expander-row.h',
+  'hdy-flap.h',
   'hdy-header-bar.h',
   'hdy-header-group.h',
   'hdy-keypad.h',
@@ -129,6 +131,7 @@ src_sources = [
   'hdy-deck.c',
   'hdy-enum-value-object.c',
   'hdy-expander-row.c',
+  'hdy-flap.c',
   'hdy-header-bar.c',
   'hdy-header-group.c',
   'hdy-keypad-button.c',
diff --git a/src/themes/Adwaita-dark.css b/src/themes/Adwaita-dark.css
index 74fe5885..f4db9595 100644
--- a/src/themes/Adwaita-dark.css
+++ b/src/themes/Adwaita-dark.css
@@ -23,21 +23,21 @@ row.expander:checked image.expander-row-arrow:not(:disabled) { color: #15539e; }
 
 row.expander image.expander-row-arrow:disabled { color: #919190; }
 
-deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.24); }
+flap > dimming, deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.24); }
 
-deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 0.2); }
+flap > border, deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 
0.2); }
 
-deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
+flap > shadow, deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
 
-deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 0.06), rgba(0, 0, 
0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.left, deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 
0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to left, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 0, 0, 0.06), rgba(0, 0, 0, 
0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.right, deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to 
left, rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 
0, 0, 0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 0, 0.06), rgba(0, 0, 
0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.up, deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 
0, 0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 0.06), rgba(0, 0, 0, 
0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.down, deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 
0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 255, 255, 0.05); }
+flap > outline, deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 
255, 255, 0.05); }
 
 avatar { border-radius: 9999px; -gtk-outline-radius: 9999px; font-weight: bold; }
 
diff --git a/src/themes/Adwaita.css b/src/themes/Adwaita.css
index 1e3ec685..2d8fe364 100644
--- a/src/themes/Adwaita.css
+++ b/src/themes/Adwaita.css
@@ -23,21 +23,21 @@ row.expander:checked image.expander-row-arrow:not(:disabled) { color: #3584e4; }
 
 row.expander image.expander-row-arrow:disabled { color: #929595; }
 
-deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
+flap > dimming, deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
 
-deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 0.05); }
+flap > border, deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 
0.05); }
 
-deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
+flap > shadow, deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
 
-deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.left, deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to left, rgba(0, 0, 0, 
0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 0, 0, 0.03), 
rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.right, deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to 
left, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 
0, 0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.up, deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 0.03), rgba(0, 0, 0, 
0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.down, deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 
0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 255, 255, 0.2); }
+flap > outline, deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 
255, 255, 0.2); }
 
 avatar { border-radius: 9999px; -gtk-outline-radius: 9999px; font-weight: bold; }
 
diff --git a/src/themes/HighContrast.css b/src/themes/HighContrast.css
index 2cadd949..4d535ab3 100644
--- a/src/themes/HighContrast.css
+++ b/src/themes/HighContrast.css
@@ -23,21 +23,21 @@ row.expander:checked image.expander-row-arrow:not(:disabled) { color: #1b6acb; }
 
 row.expander image.expander-row-arrow:disabled { color: #929495; }
 
-deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
+flap > dimming, deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
 
-deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: #877b6e; }
+flap > border, deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: #877b6e; }
 
-deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
+flap > shadow, deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
 
-deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.left, deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to left, rgba(0, 0, 0, 
0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 0, 0, 0.03), 
rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.right, deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to 
left, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 
0, 0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.up, deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 0.03), rgba(0, 0, 0, 
0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.down, deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 
0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: transparent; }
+flap > outline, deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: 
transparent; }
 
 avatar { border-radius: 9999px; -gtk-outline-radius: 9999px; font-weight: bold; }
 
diff --git a/src/themes/HighContrastInverse.css b/src/themes/HighContrastInverse.css
index d1951f8a..554e5a9f 100644
--- a/src/themes/HighContrastInverse.css
+++ b/src/themes/HighContrastInverse.css
@@ -23,21 +23,21 @@ row.expander:checked image.expander-row-arrow:not(:disabled) { color: #0f3b71; }
 
 row.expander image.expander-row-arrow:disabled { color: #919191; }
 
-deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.24); }
+flap > dimming, deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.24); }
 
-deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: #686868; }
+flap > border, deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: #686868; }
 
-deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
+flap > shadow, deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
 
-deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 0.06), rgba(0, 0, 
0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.left, deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 
0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to left, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 0, 0, 0.06), rgba(0, 0, 0, 
0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.right, deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to 
left, rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 
0, 0, 0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 0, 0.06), rgba(0, 0, 
0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.up, deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 
0, 0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, rgba(0, 0, 0, 0.1), 
rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 0.06), rgba(0, 0, 0, 
0.02) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.down, deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, 
rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.02) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 
0.06), rgba(0, 0, 0, 0.02) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: transparent; }
+flap > outline, deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: 
transparent; }
 
 avatar { border-radius: 9999px; -gtk-outline-radius: 9999px; font-weight: bold; }
 
diff --git a/src/themes/_fallback-base.scss b/src/themes/_fallback-base.scss
index b0a93e7c..66f977b0 100644
--- a/src/themes/_fallback-base.scss
+++ b/src/themes/_fallback-base.scss
@@ -69,6 +69,7 @@ row.expander {
 
 // Shadows
 
+flap,
 deck,
 leaflet {
   > dimming {
diff --git a/src/themes/fallback.css b/src/themes/fallback.css
index 94eda289..13d103ff 100644
--- a/src/themes/fallback.css
+++ b/src/themes/fallback.css
@@ -23,21 +23,21 @@ row.expander:checked image.expander-row-arrow:not(:disabled) { color: #3584e4; }
 
 row.expander image.expander-row-arrow:disabled { color: #929595; }
 
-deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
+flap > dimming, deck > dimming, leaflet > dimming { background: rgba(0, 0, 0, 0.12); }
 
-deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 0.05); }
+flap > border, deck > border, leaflet > border { min-width: 1px; min-height: 1px; background: rgba(0, 0, 0, 
0.05); }
 
-deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
+flap > shadow, deck > shadow, leaflet > shadow { min-width: 56px; min-height: 56px; }
 
-deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.left, deck > shadow.left, leaflet > shadow.left { background-image: linear-gradient(to right, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to right, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to left, rgba(0, 0, 0, 
0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 0, 0, 0.03), 
rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.right, deck > shadow.right, leaflet > shadow.right { background-image: linear-gradient(to 
left, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to left, rgba(0, 
0, 0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 0, 0.03), rgba(0, 0, 
0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.up, deck > shadow.up, leaflet > shadow.up { background-image: linear-gradient(to bottom, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to bottom, rgba(0, 0, 
0, 0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, rgba(0, 0, 0, 0.05), 
rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 0.03), rgba(0, 0, 0, 
0.01) 7px, rgba(0, 0, 0, 0) 24px); }
+flap > shadow.down, deck > shadow.down, leaflet > shadow.down { background-image: linear-gradient(to top, 
rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.01) 40px, rgba(0, 0, 0, 0) 56px), linear-gradient(to top, rgba(0, 0, 0, 
0.03), rgba(0, 0, 0, 0.01) 7px, rgba(0, 0, 0, 0) 24px); }
 
-deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 255, 255, 0.2); }
+flap > outline, deck > outline, leaflet > outline { min-width: 1px; min-height: 1px; background: rgba(255, 
255, 255, 0.2); }
 
 avatar { border-radius: 9999px; -gtk-outline-radius: 9999px; font-weight: bold; }
 
diff --git a/tests/meson.build b/tests/meson.build
index c80e96e7..83abefda 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -28,6 +28,7 @@ test_names = [
   'test-combo-row',
   'test-deck',
   'test-expander-row',
+  'test-flap',
   'test-header-bar',
   'test-header-group',
   'test-keypad',
diff --git a/tests/test-flap.c b/tests/test-flap.c
new file mode 100644
index 00000000..8492c623
--- /dev/null
+++ b/tests/test-flap.c
@@ -0,0 +1,425 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Author: Alexander Mikhaylenko <alexander mikhaylenko puri sm>
+ */
+
+#include <handy.h>
+
+gint notified;
+
+static void
+notify_cb (GtkWidget *widget, gpointer data)
+{
+  notified++;
+}
+
+static void
+test_hdy_flap_flap (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  GtkWidget *widget = NULL;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::flap", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "flap", &widget, NULL);
+  g_assert_null (widget);
+
+  hdy_flap_set_flap (flap, NULL);
+  g_assert_cmpint (notified, ==, 0);
+
+  widget = gtk_button_new ();
+  hdy_flap_set_flap (flap, widget);
+  g_assert_true (hdy_flap_get_flap (flap) == widget);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "flap", NULL, NULL);
+  g_assert_null (hdy_flap_get_flap (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_separator (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  GtkWidget *widget = NULL;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::separator", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "separator", &widget, NULL);
+  g_assert_null (widget);
+
+  hdy_flap_set_separator (flap, NULL);
+  g_assert_cmpint (notified, ==, 0);
+
+  widget = gtk_button_new ();
+  hdy_flap_set_separator (flap, widget);
+  g_assert_true (hdy_flap_get_separator (flap) == widget);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "separator", NULL, NULL);
+  g_assert_null (hdy_flap_get_separator (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_flap_position (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  GtkPackType position;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::flap-position", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "flap-position", &position, NULL);
+  g_assert_cmpint (position, ==, GTK_PACK_START);
+
+  hdy_flap_set_flap_position (flap, GTK_PACK_START);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_flap_position (flap, GTK_PACK_END);
+  g_assert_cmpint (hdy_flap_get_flap_position (flap), ==, GTK_PACK_END);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "flap-position", GTK_PACK_START, NULL);
+  g_assert_cmpint (hdy_flap_get_flap_position (flap), ==, GTK_PACK_START);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_reveal_flap (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean reveal;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+
+  notified = 0;
+  g_signal_connect (flap, "notify::reveal-flap", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "reveal-flap", &reveal, NULL);
+  g_assert_true (reveal);
+
+  hdy_flap_set_reveal_flap (flap, TRUE);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_reveal_flap (flap, FALSE);
+  g_assert_false (hdy_flap_get_reveal_flap (flap));
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "reveal-flap", TRUE, NULL);
+  g_assert_true (hdy_flap_get_reveal_flap (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_reveal_duration (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  guint duration;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::reveal-duration", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "reveal-duration", &duration, NULL);
+  g_assert_cmpint (duration, ==, 250);
+
+  hdy_flap_set_reveal_duration (flap, 250);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_reveal_duration (flap, 500);
+  g_assert_cmpint (hdy_flap_get_reveal_duration (flap), ==, 500);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "reveal-duration", 100, NULL);
+  g_assert_cmpint (hdy_flap_get_reveal_duration (flap), ==, 100);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_reveal_progress (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gdouble progress;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::reveal-progress", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "reveal-progress", &progress, NULL);
+  g_assert_cmpint (progress, ==, 1.0);
+
+  hdy_flap_set_reveal_flap (flap, FALSE);
+  g_assert_cmpint (hdy_flap_get_reveal_progress (flap), ==, 0.0);
+  g_assert_cmpint (notified, ==, 1);
+
+  hdy_flap_set_reveal_flap (flap, TRUE);
+  g_assert_cmpint (hdy_flap_get_reveal_progress (flap), ==, 1.0);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_fold_policy (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  HdyFlapFoldPolicy policy;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::fold-policy", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "fold-policy", &policy, NULL);
+  g_assert_cmpint (policy, ==, HDY_FLAP_FOLD_POLICY_AUTO);
+
+  hdy_flap_set_fold_policy (flap, HDY_FLAP_FOLD_POLICY_AUTO);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_fold_policy (flap, HDY_FLAP_FOLD_POLICY_NEVER);
+  g_assert_cmpint (hdy_flap_get_fold_policy (flap), ==, HDY_FLAP_FOLD_POLICY_NEVER);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "fold-policy", HDY_FLAP_FOLD_POLICY_ALWAYS, NULL);
+  g_assert_cmpint (hdy_flap_get_fold_policy (flap), ==, HDY_FLAP_FOLD_POLICY_ALWAYS);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_fold_duration (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  guint duration;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::fold-duration", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "fold-duration", &duration, NULL);
+  g_assert_cmpint (duration, ==, 250);
+
+  hdy_flap_set_fold_duration (flap, 250);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_fold_duration (flap, 500);
+  g_assert_cmpint (hdy_flap_get_fold_duration (flap), ==, 500);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "fold-duration", 100, NULL);
+  g_assert_cmpint (hdy_flap_get_fold_duration (flap), ==, 100);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_folded (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean folded;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+  hdy_flap_set_fold_policy (flap, HDY_FLAP_FOLD_POLICY_NEVER);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::folded", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "folded", &folded, NULL);
+  g_assert_false (folded);
+
+  hdy_flap_set_fold_policy (flap, HDY_FLAP_FOLD_POLICY_ALWAYS);
+  g_assert_true (hdy_flap_get_folded (flap));
+  g_assert_cmpint (notified, ==, 1);
+}
+
+static void
+test_hdy_flap_locked (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean locked;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+
+  notified = 0;
+  g_signal_connect (flap, "notify::locked", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "locked", &locked, NULL);
+  g_assert_false (locked);
+
+  hdy_flap_set_locked (flap, FALSE);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_locked (flap, TRUE);
+  g_assert_true (hdy_flap_get_locked (flap));
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "locked", FALSE, NULL);
+  g_assert_false (hdy_flap_get_locked (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_transition_type (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  HdyFlapTransitionType policy;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  notified = 0;
+  g_signal_connect (flap, "notify::transition-type", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "transition-type", &policy, NULL);
+  g_assert_cmpint (policy, ==, HDY_FLAP_TRANSITION_TYPE_OVER);
+
+  hdy_flap_set_transition_type (flap, HDY_FLAP_TRANSITION_TYPE_OVER);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_transition_type (flap, HDY_FLAP_TRANSITION_TYPE_SLIDE);
+  g_assert_cmpint (hdy_flap_get_transition_type (flap), ==, HDY_FLAP_TRANSITION_TYPE_SLIDE);
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "transition-type", HDY_FLAP_TRANSITION_TYPE_UNDER, NULL);
+  g_assert_cmpint (hdy_flap_get_transition_type (flap), ==, HDY_FLAP_TRANSITION_TYPE_UNDER);
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_modal (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean modal;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+
+  notified = 0;
+  g_signal_connect (flap, "notify::modal", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "modal", &modal, NULL);
+  g_assert_true (modal);
+
+  hdy_flap_set_modal (flap, TRUE);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_modal (flap, FALSE);
+  g_assert_false (hdy_flap_get_modal (flap));
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "modal", TRUE, NULL);
+  g_assert_true (hdy_flap_get_modal (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_swipe_to_open (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean swipe_to_open;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+
+  notified = 0;
+  g_signal_connect (flap, "notify::swipe-to-open", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "swipe-to-open", &swipe_to_open, NULL);
+  g_assert_true (swipe_to_open);
+
+  hdy_flap_set_swipe_to_open (flap, TRUE);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_swipe_to_open (flap, FALSE);
+  g_assert_false (hdy_flap_get_swipe_to_open (flap));
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "swipe-to-open", TRUE, NULL);
+  g_assert_true (hdy_flap_get_swipe_to_open (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_hdy_flap_swipe_to_close (void)
+{
+  g_autoptr (HdyFlap) flap = NULL;
+  gboolean swipe_to_close;
+
+  flap = g_object_ref_sink (HDY_FLAP (hdy_flap_new ()));
+  g_assert_nonnull (flap);
+
+  hdy_flap_set_flap (flap, gtk_button_new ());
+
+  notified = 0;
+  g_signal_connect (flap, "notify::swipe-to-close", G_CALLBACK (notify_cb), NULL);
+
+  g_object_get (flap, "swipe-to-close", &swipe_to_close, NULL);
+  g_assert_true (swipe_to_close);
+
+  hdy_flap_set_swipe_to_close (flap, TRUE);
+  g_assert_cmpint (notified, ==, 0);
+
+  hdy_flap_set_swipe_to_close (flap, FALSE);
+  g_assert_false (hdy_flap_get_swipe_to_close (flap));
+  g_assert_cmpint (notified, ==, 1);
+
+  g_object_set (flap, "swipe-to-close", TRUE, NULL);
+  g_assert_true (hdy_flap_get_swipe_to_close (flap));
+  g_assert_cmpint (notified, ==, 2);
+}
+
+gint
+main (gint argc,
+      gchar *argv[])
+{
+  gtk_test_init (&argc, &argv, NULL);
+  hdy_init ();
+
+  g_test_add_func ("/Handy/Flap/flap", test_hdy_flap_flap);
+  g_test_add_func ("/Handy/Flap/separator", test_hdy_flap_separator);
+  g_test_add_func ("/Handy/Flap/flap_position", test_hdy_flap_flap_position);
+  g_test_add_func ("/Handy/Flap/reveal_flap", test_hdy_flap_reveal_flap);
+  g_test_add_func ("/Handy/Flap/reveal_duration", test_hdy_flap_reveal_duration);
+  g_test_add_func ("/Handy/Flap/reveal_progress", test_hdy_flap_reveal_progress);
+  g_test_add_func ("/Handy/Flap/fold_policy", test_hdy_flap_fold_policy);
+  g_test_add_func ("/Handy/Flap/fold_duration", test_hdy_flap_fold_duration);
+  g_test_add_func ("/Handy/Flap/folded", test_hdy_flap_folded);
+  g_test_add_func ("/Handy/Flap/locked", test_hdy_flap_locked);
+  g_test_add_func ("/Handy/Flap/transition_type", test_hdy_flap_transition_type);
+  g_test_add_func ("/Handy/Flap/modal", test_hdy_flap_modal);
+  g_test_add_func ("/Handy/Flap/swipe_to_open", test_hdy_flap_swipe_to_open);
+  g_test_add_func ("/Handy/Flap/swipe_to_close", test_hdy_flap_swipe_to_close);
+
+  return g_test_run ();
+}


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]