[libadwaita/wip/exalm/tabs: 1/6] Add AdwTabView
- From: Adrien Plazas <aplazas src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [libadwaita/wip/exalm/tabs: 1/6] Add AdwTabView
- Date: Wed, 12 May 2021 11:03:06 +0000 (UTC)
commit 2cdd7e2eec2d64a6fa91567a1199c4506428ea01
Author: Alexander Mikhaylenko <alexm gnome org>
Date: Sun Sep 13 02:23:26 2020 +0500
Add AdwTabView
doc/adwaita-docs.xml | 1 +
doc/meson.build | 1 +
src/adw-tab-view-private.h | 30 +
src/adw-tab-view.c | 3399 ++++++++++++++++++++
src/adw-tab-view.h | 213 ++
src/adwaita.gresources.xml | 1 +
src/adwaita.h | 1 +
.../status/adw-tab-icon-missing-symbolic.svg | 32 +
src/meson.build | 2 +
tests/meson.build | 1 +
tests/test-tab-view.c | 1151 +++++++
11 files changed, 4832 insertions(+)
---
diff --git a/doc/adwaita-docs.xml b/doc/adwaita-docs.xml
index d3be7205..790da5e1 100644
--- a/doc/adwaita-docs.xml
+++ b/doc/adwaita-docs.xml
@@ -62,6 +62,7 @@
<xi:include href="xml/adw-status-page.xml"/>
<xi:include href="xml/adw-swipeable.xml"/>
<xi:include href="xml/adw-swipe-tracker.xml"/>
+ <xi:include href="xml/adw-tab-view.xml"/>
<xi:include href="xml/adw-value-object.xml"/>
<xi:include href="xml/adw-view-switcher.xml"/>
<xi:include href="xml/adw-view-switcher-bar.xml"/>
diff --git a/doc/meson.build b/doc/meson.build
index cd2ea0ee..7c578115 100644
--- a/doc/meson.build
+++ b/doc/meson.build
@@ -20,6 +20,7 @@ private_headers = [
'adw-preferences-page-private.h',
'adw-shadow-helper-private.h',
'adw-swipe-tracker-private.h',
+ 'adw-tab-view-private.h',
'adw-view-switcher-button-private.h',
'adw-window-mixin-private.h',
]
diff --git a/src/adw-tab-view-private.h b/src/adw-tab-view-private.h
new file mode 100644
index 00000000..f75f4698
--- /dev/null
+++ b/src/adw-tab-view-private.h
@@ -0,0 +1,30 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Author: Alexander Mikhaylenko <alexander mikhaylenko puri sm>
+ */
+
+#pragma once
+
+#if !defined(_ADWAITA_INSIDE) && !defined(ADWAITA_COMPILATION)
+#error "Only <adwaita.h> can be included directly."
+#endif
+
+#include "adw-tab-view.h"
+
+G_BEGIN_DECLS
+
+gboolean adw_tab_view_select_first_page (AdwTabView *self);
+gboolean adw_tab_view_select_last_page (AdwTabView *self);
+
+void adw_tab_view_detach_page (AdwTabView *self,
+ AdwTabPage *page);
+void adw_tab_view_attach_page (AdwTabView *self,
+ AdwTabPage *page,
+ int position);
+
+AdwTabView *adw_tab_view_create_window (AdwTabView *self);
+
+G_END_DECLS
diff --git a/src/adw-tab-view.c b/src/adw-tab-view.c
new file mode 100644
index 00000000..7756858c
--- /dev/null
+++ b/src/adw-tab-view.c
@@ -0,0 +1,3399 @@
+/*
+ * Copyright (C) 2020-2021 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Author: Alexander Mikhaylenko <alexander mikhaylenko puri sm>
+ */
+
+#include "config.h"
+
+#include "adw-tab-view-private.h"
+
+#include "adw-gizmo-private.h"
+
+/* FIXME replace with groups */
+static GSList *tab_view_list;
+
+/**
+ * SECTION:adw-tab-view
+ * @short_description: A dynamic tabbed container
+ * @title: AdwTabView
+ * @See_also: #AdwTabBar
+ *
+ * #AdwTabView is a container which shows one child at a time. While it provides
+ * keyboard shortcuts for switching between pages, it does not provide a visible
+ * tab bar and relies on external widgets for that, such as #AdwTabBar.
+ *
+ * #AdwTabView maintains a #AdwTabPage object for each page,which holds
+ * additional per-page properties. You can obtain the #AdwTabPage for a page
+ * with adw_tab_view_get_page(), and as return value for adw_tab_view_append()
+ * and other functions for adding children.
+ *
+ * #AdwTabView only aims to be useful for dynamic tabs in multi-window
+ * document-based applications, such as web browsers, file managers, text
+ * editors or terminals. It does not aim to replace #GtkNotebook for use cases
+ * such as tabbed dialogs.
+ *
+ * As such, it does not support disabling page reordering or detaching, or
+ * adding children via #GtkBuilder.
+ *
+ * # CSS nodes
+ *
+ * #AdwTabView has a main CSS node with the name tabview.
+ *
+ * It contains the subnode overlay, which contains subnodes stack and widget.
+ * The stack subnode contains the added pages.
+ *
+ * |[<!-- language="plain" -->
+ * tabview
+ * ├── stack
+ * │ ╰── [ Children ]
+ * ╰── widget
+ * ]|
+ *
+ * Since: 1.0
+ */
+
+struct _AdwTabPage
+{
+ GObject parent_instance;
+
+ GtkWidget *child;
+ AdwTabPage *parent;
+ gboolean selected;
+ gboolean pinned;
+ char *title;
+ char *tooltip;
+ GIcon *icon;
+ gboolean loading;
+ GIcon *indicator_icon;
+ gboolean indicator_activatable;
+ gboolean needs_attention;
+
+ gboolean closing;
+};
+
+G_DEFINE_TYPE (AdwTabPage, adw_tab_page, G_TYPE_OBJECT)
+
+enum {
+ PAGE_PROP_0,
+ PAGE_PROP_CHILD,
+ PAGE_PROP_PARENT,
+ PAGE_PROP_SELECTED,
+ PAGE_PROP_PINNED,
+ PAGE_PROP_TITLE,
+ PAGE_PROP_TOOLTIP,
+ PAGE_PROP_ICON,
+ PAGE_PROP_LOADING,
+ PAGE_PROP_INDICATOR_ICON,
+ PAGE_PROP_INDICATOR_ACTIVATABLE,
+ PAGE_PROP_NEEDS_ATTENTION,
+ LAST_PAGE_PROP
+};
+
+static GParamSpec *page_props[LAST_PAGE_PROP];
+
+struct _AdwTabView
+{
+ GtkWidget parent_instance;
+
+ GtkStack *stack;
+ GListStore *children;
+
+ int n_pages;
+ int n_pinned_pages;
+ AdwTabPage *selected_page;
+ GIcon *default_icon;
+ GMenuModel *menu_model;
+
+ int transfer_count;
+
+ GtkWidget *shortcut_widget;
+ GtkEventController *shortcut_controller;
+
+ GtkSelectionModel *pages;
+};
+
+G_DEFINE_TYPE (AdwTabView, adw_tab_view, GTK_TYPE_WIDGET)
+
+enum {
+ PROP_0,
+ PROP_N_PAGES,
+ PROP_N_PINNED_PAGES,
+ PROP_IS_TRANSFERRING_PAGE,
+ PROP_SELECTED_PAGE,
+ PROP_DEFAULT_ICON,
+ PROP_MENU_MODEL,
+ PROP_SHORTCUT_WIDGET,
+ PROP_PAGES,
+ LAST_PROP
+};
+
+static GParamSpec *props[LAST_PROP];
+
+enum {
+ SIGNAL_PAGE_ATTACHED,
+ SIGNAL_PAGE_DETACHED,
+ SIGNAL_PAGE_REORDERED,
+ SIGNAL_CLOSE_PAGE,
+ SIGNAL_SETUP_MENU,
+ SIGNAL_CREATE_WINDOW,
+ SIGNAL_INDICATOR_ACTIVATED,
+ SIGNAL_LAST_SIGNAL,
+};
+
+static guint signals[SIGNAL_LAST_SIGNAL];
+
+static void
+set_page_selected (AdwTabPage *self,
+ gboolean selected)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ selected = !!selected;
+
+ if (self->selected == selected)
+ return;
+
+ self->selected = selected;
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_SELECTED]);
+}
+
+static void
+set_page_pinned (AdwTabPage *self,
+ gboolean pinned)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ pinned = !!pinned;
+
+ if (self->pinned == pinned)
+ return;
+
+ self->pinned = pinned;
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_PINNED]);
+}
+
+static void set_page_parent (AdwTabPage *self,
+ AdwTabPage *parent);
+
+static void
+page_parent_notify_cb (AdwTabPage *self)
+{
+ AdwTabPage *grandparent = adw_tab_page_get_parent (self->parent);
+
+ self->parent = NULL;
+
+ if (grandparent)
+ set_page_parent (self, grandparent);
+ else
+ g_object_notify_by_pspec (G_OBJECT (self), props[PAGE_PROP_PARENT]);
+}
+
+static void
+set_page_parent (AdwTabPage *self,
+ AdwTabPage *parent)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (parent) || parent == NULL);
+
+ if (self->parent == parent)
+ return;
+
+ if (self->parent)
+ g_object_weak_unref (G_OBJECT (self->parent),
+ (GWeakNotify) page_parent_notify_cb,
+ self);
+
+ self->parent = parent;
+
+ if (self->parent)
+ g_object_weak_ref (G_OBJECT (self->parent),
+ (GWeakNotify) page_parent_notify_cb,
+ self);
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PAGE_PROP_PARENT]);
+}
+
+static void
+adw_tab_page_dispose (GObject *object)
+{
+ AdwTabPage *self = ADW_TAB_PAGE (object);
+
+ set_page_parent (self, NULL);
+
+ G_OBJECT_CLASS (adw_tab_page_parent_class)->dispose (object);
+}
+
+static void
+adw_tab_page_finalize (GObject *object)
+{
+ AdwTabPage *self = (AdwTabPage *)object;
+
+ g_clear_object (&self->child);
+ g_clear_pointer (&self->title, g_free);
+ g_clear_pointer (&self->tooltip, g_free);
+ g_clear_object (&self->icon);
+ g_clear_object (&self->indicator_icon);
+
+ G_OBJECT_CLASS (adw_tab_page_parent_class)->finalize (object);
+}
+
+static void
+adw_tab_page_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ AdwTabPage *self = ADW_TAB_PAGE (object);
+
+ switch (prop_id) {
+ case PAGE_PROP_CHILD:
+ g_value_set_object (value, adw_tab_page_get_child (self));
+ break;
+
+ case PAGE_PROP_PARENT:
+ g_value_set_object (value, adw_tab_page_get_parent (self));
+ break;
+
+ case PAGE_PROP_SELECTED:
+ g_value_set_boolean (value, adw_tab_page_get_selected (self));
+ break;
+
+ case PAGE_PROP_PINNED:
+ g_value_set_boolean (value, adw_tab_page_get_pinned (self));
+ break;
+
+ case PAGE_PROP_TITLE:
+ g_value_set_string (value, adw_tab_page_get_title (self));
+ break;
+
+ case PAGE_PROP_TOOLTIP:
+ g_value_set_string (value, adw_tab_page_get_tooltip (self));
+ break;
+
+ case PAGE_PROP_ICON:
+ g_value_set_object (value, adw_tab_page_get_icon (self));
+ break;
+
+ case PAGE_PROP_LOADING:
+ g_value_set_boolean (value, adw_tab_page_get_loading (self));
+ break;
+
+ case PAGE_PROP_INDICATOR_ICON:
+ g_value_set_object (value, adw_tab_page_get_indicator_icon (self));
+ break;
+
+ case PAGE_PROP_INDICATOR_ACTIVATABLE:
+ g_value_set_boolean (value, adw_tab_page_get_indicator_activatable (self));
+ break;
+
+ case PAGE_PROP_NEEDS_ATTENTION:
+ g_value_set_boolean (value, adw_tab_page_get_needs_attention (self));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+adw_tab_page_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ AdwTabPage *self = ADW_TAB_PAGE (object);
+
+ switch (prop_id) {
+ case PAGE_PROP_CHILD:
+ g_set_object (&self->child, g_value_get_object (value));
+ break;
+
+ case PAGE_PROP_PARENT:
+ set_page_parent (self, g_value_get_object (value));
+ break;
+
+ case PAGE_PROP_TITLE:
+ adw_tab_page_set_title (self, g_value_get_string (value));
+ break;
+
+ case PAGE_PROP_TOOLTIP:
+ adw_tab_page_set_tooltip (self, g_value_get_string (value));
+ break;
+
+ case PAGE_PROP_ICON:
+ adw_tab_page_set_icon (self, g_value_get_object (value));
+ break;
+
+ case PAGE_PROP_LOADING:
+ adw_tab_page_set_loading (self, g_value_get_boolean (value));
+ break;
+
+ case PAGE_PROP_INDICATOR_ICON:
+ adw_tab_page_set_indicator_icon (self, g_value_get_object (value));
+ break;
+
+ case PAGE_PROP_INDICATOR_ACTIVATABLE:
+ adw_tab_page_set_indicator_activatable (self, g_value_get_boolean (value));
+ break;
+
+ case PAGE_PROP_NEEDS_ATTENTION:
+ adw_tab_page_set_needs_attention (self, g_value_get_boolean (value));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+adw_tab_page_class_init (AdwTabPageClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->dispose = adw_tab_page_dispose;
+ object_class->finalize = adw_tab_page_finalize;
+ object_class->get_property = adw_tab_page_get_property;
+ object_class->set_property = adw_tab_page_set_property;
+
+ /**
+ * AdwTabPage:child:
+ *
+ * The child of the page.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_CHILD] =
+ g_param_spec_object ("child",
+ "Child",
+ "The child of the page",
+ GTK_TYPE_WIDGET,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
+
+ /**
+ * AdwTabPage:parent:
+ *
+ * The parent page of the page.
+ *
+ * See adw_tab_view_add_page() and adw_tab_view_close_page().
+
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_PARENT] =
+ g_param_spec_object ("parent",
+ "Parent",
+ "The parent page of the page",
+ ADW_TYPE_TAB_PAGE,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:selected:
+ *
+ * Whether the page is selected.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_SELECTED] =
+ g_param_spec_boolean ("selected",
+ "Selected",
+ "Whether the page is selected",
+ FALSE,
+ G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:pinned:
+ *
+ * Whether the page is pinned. See adw_tab_view_set_page_pinned().
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_PINNED] =
+ g_param_spec_boolean ("pinned",
+ "Pinned",
+ "Whether the page is pinned",
+ FALSE,
+ G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:title:
+ *
+ * The title of the page.
+ *
+ * #AdwTabBar will display it in the center of the tab unless it's pinned,
+ * and will use it as a tooltip unless #AdwTabPage:tooltip is set.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_TITLE] =
+ g_param_spec_string ("title",
+ "Title",
+ "The title of the page",
+ NULL,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:tooltip:
+ *
+ * The tooltip of the page, marked up with the Pango text markup language.
+ *
+ * If not set, #AdwTabBar will use #AdwTabPage:title as a tooltip instead.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_TOOLTIP] =
+ g_param_spec_string ("tooltip",
+ "Tooltip",
+ "The tooltip of the page",
+ NULL,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:icon:
+ *
+ * The icon of the page, displayed next to the title.
+ *
+ * #AdwTabBar will not show the icon if #AdwTabPage:loading is set to %TRUE,
+ * or if the page is pinned and #AdwTabPage:indicator-icon is set.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_ICON] =
+ g_param_spec_object ("icon",
+ "Icon",
+ "The icon of the page",
+ G_TYPE_ICON,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:loading:
+ *
+ * Whether the page is loading.
+ *
+ * If set to %TRUE, #AdwTabBar will display a spinner in place of icon.
+ *
+ * If the page is pinned and #AdwTabPage:indicator-icon is set, the loading
+ * status will not be visible.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_LOADING] =
+ g_param_spec_boolean ("loading",
+ "Loading",
+ "Whether the page is loading",
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:indicator-icon:
+ *
+ * An indicator icon for the page.
+ *
+ * A common use case is an audio or camera indicator in a web browser.
+ *
+ * #AdwTabPage will show it at the beginning of the tab, alongside icon
+ * representing #AdwTabPage:icon or loading spinner.
+ *
+ * If the page is pinned, the indicator will be shown instead of icon or
+ * spinner.
+ *
+ * If #AdwTabPage:indicator-activatable is set to %TRUE, the indicator icon
+ * can act as a button.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_INDICATOR_ICON] =
+ g_param_spec_object ("indicator-icon",
+ "Indicator icon",
+ "An indicator icon for the page",
+ G_TYPE_ICON,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:indicator-activatable:
+ *
+ * Whether the indicator icon is activatable.
+ *
+ * If set to %TRUE, #AdwTabView::indicator-activated will be emitted when
+ * the indicator icon is clicked.
+ *
+ * If #AdwTabPage:indicator-icon is not set, does nothing.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_INDICATOR_ACTIVATABLE] =
+ g_param_spec_boolean ("indicator-activatable",
+ "Indicator activatable",
+ "Whether the indicator icon is activatable",
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabPage:needs-attention:
+ *
+ * Whether the page needs attention.
+ *
+ * #AdwTabBar will display a glow under the tab representing the page if set
+ * to %TRUE. If the tab is not visible, the corresponding edge of the tab bar
+ * will be highlighted.
+ *
+ * Since: 1.0
+ */
+ page_props[PAGE_PROP_NEEDS_ATTENTION] =
+ g_param_spec_boolean ("needs-attention",
+ "Needs attention",
+ "Whether the page needs attention",
+ FALSE,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ g_object_class_install_properties (object_class, LAST_PAGE_PROP, page_props);
+}
+
+static void
+adw_tab_page_init (AdwTabPage *self)
+{
+}
+
+#define ADW_TYPE_TAB_PAGES (adw_tab_pages_get_type ())
+
+G_DECLARE_FINAL_TYPE (AdwTabPages, adw_tab_pages, ADW, TAB_PAGES, GObject)
+
+struct _AdwTabPages
+{
+ GObject parent_instance;
+
+ AdwTabView *view;
+};
+
+static GType
+adw_tab_pages_get_item_type (GListModel *model)
+{
+ return ADW_TYPE_TAB_PAGE;
+}
+
+static guint
+adw_tab_pages_get_n_items (GListModel *model)
+{
+ AdwTabPages *self = ADW_TAB_PAGES (model);
+
+ return self->view->n_pages;
+}
+
+static gpointer
+adw_tab_pages_get_item (GListModel *model,
+ guint position)
+{
+ AdwTabPages *self = ADW_TAB_PAGES (model);
+ AdwTabPage *page = adw_tab_view_get_nth_page (self->view, position);
+
+ if (!page)
+ return NULL;
+
+ return g_object_ref (page);
+}
+
+static void
+adw_tab_pages_list_model_init (GListModelInterface *iface)
+{
+ iface->get_item_type = adw_tab_pages_get_item_type;
+ iface->get_n_items = adw_tab_pages_get_n_items;
+ iface->get_item = adw_tab_pages_get_item;
+}
+
+static gboolean
+adw_tab_pages_is_selected (GtkSelectionModel *model,
+ guint position)
+{
+ AdwTabPages *self = ADW_TAB_PAGES (model);
+ AdwTabPage *page;
+
+ page = adw_tab_view_get_nth_page (self->view, position);
+
+ return page->selected;
+}
+
+static gboolean
+adw_tab_pages_select_item (GtkSelectionModel *model,
+ guint position,
+ gboolean exclusive)
+{
+ AdwTabPages *self = ADW_TAB_PAGES (model);
+ AdwTabPage *page;
+
+ page = adw_tab_view_get_nth_page (self->view, position);
+
+ adw_tab_view_set_selected_page (self->view, page);
+
+ return TRUE;
+}
+
+static void
+adw_tab_pages_selection_model_init (GtkSelectionModelInterface *iface)
+{
+ iface->is_selected = adw_tab_pages_is_selected;
+ iface->select_item = adw_tab_pages_select_item;
+}
+
+G_DEFINE_TYPE_WITH_CODE (AdwTabPages, adw_tab_pages, G_TYPE_OBJECT,
+ G_IMPLEMENT_INTERFACE (G_TYPE_LIST_MODEL, adw_tab_pages_list_model_init)
+ G_IMPLEMENT_INTERFACE (GTK_TYPE_SELECTION_MODEL,
adw_tab_pages_selection_model_init))
+
+static void
+adw_tab_pages_init (AdwTabPages *self)
+{
+}
+
+static void
+adw_tab_pages_class_init (AdwTabPagesClass *klass)
+{
+}
+
+static GtkSelectionModel *
+adw_tab_pages_new (AdwTabView *view)
+{
+ AdwTabPages *pages;
+
+ pages = g_object_new (ADW_TYPE_TAB_PAGES, NULL);
+ pages->view = view;
+
+ return GTK_SELECTION_MODEL (pages);
+}
+
+static gboolean
+object_handled_accumulator (GSignalInvocationHint *ihint,
+ GValue *return_accu,
+ const GValue *handler_return,
+ gpointer data)
+{
+ GObject *object = g_value_get_object (handler_return);
+
+ g_value_set_object (return_accu, object);
+
+ return !object;
+}
+
+static void
+begin_transfer_for_group (AdwTabView *self)
+{
+ GSList *l;
+
+ for (l = tab_view_list; l; l = l->next) {
+ AdwTabView *view = l->data;
+
+ view->transfer_count++;
+
+ if (view->transfer_count == 1)
+ g_object_notify_by_pspec (G_OBJECT (view), props[PROP_IS_TRANSFERRING_PAGE]);
+ }
+}
+
+static void
+end_transfer_for_group (AdwTabView *self)
+{
+ GSList *l;
+
+ for (l = tab_view_list; l; l = l->next) {
+ AdwTabView *view = l->data;
+
+ view->transfer_count--;
+
+ if (view->transfer_count == 0)
+ g_object_notify_by_pspec (G_OBJECT (view), props[PROP_IS_TRANSFERRING_PAGE]);
+ }
+}
+
+static void
+set_n_pages (AdwTabView *self,
+ int n_pages)
+{
+ if (n_pages == self->n_pages)
+ return;
+
+ self->n_pages = n_pages;
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_N_PAGES]);
+}
+
+static void
+set_n_pinned_pages (AdwTabView *self,
+ int n_pinned_pages)
+{
+ if (n_pinned_pages == self->n_pinned_pages)
+ return;
+
+ self->n_pinned_pages = n_pinned_pages;
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_N_PINNED_PAGES]);
+}
+
+static inline gboolean
+page_belongs_to_this_view (AdwTabView *self,
+ AdwTabPage *page)
+{
+ if (!page)
+ return FALSE;
+
+ return gtk_widget_get_parent (page->child) == GTK_WIDGET (self->stack);
+}
+
+static inline gboolean
+is_descendant_of (AdwTabPage *page,
+ AdwTabPage *parent)
+{
+ while (page && page != parent)
+ page = adw_tab_page_get_parent (page);
+
+ return page == parent;
+}
+
+static void
+attach_page (AdwTabView *self,
+ AdwTabPage *page,
+ int position)
+{
+ GtkWidget *child = adw_tab_page_get_child (page);
+ AdwTabPage *parent;
+
+ g_list_store_insert (self->children, position, page);
+
+ gtk_stack_add_child (self->stack, child);
+
+ g_object_freeze_notify (G_OBJECT (self));
+
+ set_n_pages (self, self->n_pages + 1);
+
+ if (adw_tab_page_get_pinned (page))
+ set_n_pinned_pages (self, self->n_pinned_pages + 1);
+
+ g_object_thaw_notify (G_OBJECT (self));
+
+ parent = adw_tab_page_get_parent (page);
+
+ if (parent && !page_belongs_to_this_view (self, parent))
+ set_page_parent (page, NULL);
+
+ g_signal_emit (self, signals[SIGNAL_PAGE_ATTACHED], 0, page, position);
+
+ if (self->pages)
+ g_list_model_items_changed (G_LIST_MODEL (self->pages), position, 0, 1);
+}
+
+static void
+set_selected_page (AdwTabView *self,
+ AdwTabPage *selected_page,
+ gboolean notify_pages)
+{
+ guint old_position = GTK_INVALID_LIST_POSITION;
+ guint new_position = GTK_INVALID_LIST_POSITION;
+
+ if (self->selected_page == selected_page)
+ return;
+
+ if (self->selected_page) {
+ if (notify_pages && self->pages)
+ old_position = adw_tab_view_get_page_position (self, self->selected_page);
+
+ set_page_selected (self->selected_page, FALSE);
+ }
+
+ self->selected_page = selected_page;
+
+ if (self->selected_page) {
+ if (notify_pages && self->pages)
+ new_position = adw_tab_view_get_page_position (self, self->selected_page);
+
+ gtk_stack_set_visible_child (self->stack,
+ adw_tab_page_get_child (selected_page));
+ set_page_selected (self->selected_page, TRUE);
+ }
+
+ if (notify_pages && self->pages) {
+ if (old_position == GTK_INVALID_LIST_POSITION && new_position == GTK_INVALID_LIST_POSITION)
+ ; /* nothing to do */
+ else if (old_position == GTK_INVALID_LIST_POSITION)
+ gtk_selection_model_selection_changed (self->pages, new_position, 1);
+ else if (new_position == GTK_INVALID_LIST_POSITION)
+ gtk_selection_model_selection_changed (self->pages, old_position, 1);
+ else
+ gtk_selection_model_selection_changed (self->pages,
+ MIN (old_position, new_position),
+ MAX (old_position, new_position) -
+ MIN (old_position, new_position) + 1);
+ }
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SELECTED_PAGE]);
+}
+
+static void
+select_previous_page (AdwTabView *self,
+ AdwTabPage *page)
+{
+ int pos = adw_tab_view_get_page_position (self, page);
+ AdwTabPage *parent;
+
+ if (page != self->selected_page)
+ return;
+
+ parent = adw_tab_page_get_parent (page);
+
+ if (parent && pos > 0) {
+ AdwTabPage *prev_page = adw_tab_view_get_nth_page (self, pos - 1);
+
+ /* This usually means we opened a few pages from the same page in a row, or
+ * the previous page is the parent. Switch there. */
+ if (is_descendant_of (prev_page, parent)) {
+ adw_tab_view_set_selected_page (self, prev_page);
+
+ return;
+ }
+
+ /* Pinned pages are special in that opening a page from a pinned parent
+ * will place it not directly after the parent, but after the last pinned
+ * page. This means that if we're closing the first non-pinned page, we need
+ * to jump to the parent directly instead of the previous page which might
+ * be different. */
+ if (adw_tab_page_get_pinned (prev_page) &&
+ adw_tab_page_get_pinned (parent)) {
+ adw_tab_view_set_selected_page (self, parent);
+
+ return;
+ }
+ }
+
+ if (adw_tab_view_select_next_page (self))
+ return;
+
+ adw_tab_view_select_previous_page (self);
+}
+
+static void
+detach_page (AdwTabView *self,
+ AdwTabPage *page,
+ gboolean notify_pages)
+{
+ int pos = adw_tab_view_get_page_position (self, page);
+ GtkWidget *child;
+
+ select_previous_page (self, page);
+
+ child = adw_tab_page_get_child (page);
+
+ g_object_ref (page);
+ g_object_ref (child);
+
+ if (self->n_pages == 1)
+ set_selected_page (self, NULL, notify_pages);
+
+ g_list_store_remove (self->children, pos);
+
+ g_object_freeze_notify (G_OBJECT (self));
+
+ set_n_pages (self, self->n_pages - 1);
+
+ if (adw_tab_page_get_pinned (page))
+ set_n_pinned_pages (self, self->n_pinned_pages - 1);
+
+ g_object_thaw_notify (G_OBJECT (self));
+
+ gtk_stack_remove (self->stack, child);
+
+ g_signal_emit (self, signals[SIGNAL_PAGE_DETACHED], 0, page, pos);
+
+ if (notify_pages && self->pages)
+ g_list_model_items_changed (G_LIST_MODEL (self->pages), pos, 1, 0);
+
+ g_object_unref (child);
+ g_object_unref (page);
+}
+
+static AdwTabPage *
+insert_page (AdwTabView *self,
+ GtkWidget *child,
+ AdwTabPage *parent,
+ int position,
+ gboolean pinned)
+{
+ g_autoptr (AdwTabPage) page =
+ g_object_new (ADW_TYPE_TAB_PAGE,
+ "child", child,
+ "parent", parent,
+ NULL);
+
+ set_page_pinned (page, pinned);
+
+ attach_page (self, page, position);
+
+ if (!self->selected_page)
+ adw_tab_view_set_selected_page (self, page);
+
+ return page;
+}
+
+static gboolean
+close_page_cb (AdwTabView *self,
+ AdwTabPage *page)
+{
+ adw_tab_view_close_page_finish (self, page,
+ !adw_tab_page_get_pinned (page));
+
+ return GDK_EVENT_STOP;
+}
+
+static gboolean
+select_page (AdwTabView *self,
+ GtkDirectionType direction,
+ gboolean last)
+{
+ gboolean is_rtl, success = last;
+
+ if (!self->selected_page)
+ return FALSE;
+
+ is_rtl = gtk_widget_get_direction (GTK_WIDGET (self)) == GTK_TEXT_DIR_RTL;
+
+ if (direction == GTK_DIR_LEFT)
+ direction = is_rtl ? GTK_DIR_TAB_FORWARD : GTK_DIR_TAB_BACKWARD;
+ else if (direction == GTK_DIR_RIGHT)
+ direction = is_rtl ? GTK_DIR_TAB_BACKWARD : GTK_DIR_TAB_FORWARD;
+
+ if (direction == GTK_DIR_TAB_BACKWARD) {
+ if (last)
+ success = adw_tab_view_select_first_page (self);
+ else
+ success = adw_tab_view_select_previous_page (self);
+ } else if (direction == GTK_DIR_TAB_FORWARD) {
+ if (last)
+ success = adw_tab_view_select_last_page (self);
+ else
+ success = adw_tab_view_select_next_page (self);
+ }
+
+ gtk_widget_grab_focus (adw_tab_page_get_child (self->selected_page));
+
+ return success;
+}
+
+static gboolean
+reorder_page (AdwTabView *self,
+ GtkDirectionType direction,
+ gboolean last)
+{
+ gboolean is_rtl, success = last;
+
+ if (!self->selected_page)
+ return FALSE;
+
+ is_rtl = gtk_widget_get_direction (GTK_WIDGET (self)) == GTK_TEXT_DIR_RTL;
+
+ if (direction == GTK_DIR_LEFT)
+ direction = is_rtl ? GTK_DIR_TAB_FORWARD : GTK_DIR_TAB_BACKWARD;
+ else if (direction == GTK_DIR_RIGHT)
+ direction = is_rtl ? GTK_DIR_TAB_BACKWARD : GTK_DIR_TAB_FORWARD;
+
+ if (direction == GTK_DIR_TAB_BACKWARD) {
+ if (last)
+ success = adw_tab_view_reorder_first (self, self->selected_page);
+ else
+ success = adw_tab_view_reorder_backward (self, self->selected_page);
+ } else if (direction == GTK_DIR_TAB_FORWARD) {
+ if (last)
+ success = adw_tab_view_reorder_last (self, self->selected_page);
+ else
+ success = adw_tab_view_reorder_forward (self, self->selected_page);
+ }
+
+ return success;
+}
+
+static inline gboolean
+handle_select_reorder_shortcuts (AdwTabView *self,
+ guint keyval,
+ GdkModifierType state,
+ guint keysym,
+ GtkDirectionType direction,
+ gboolean last)
+{
+ /* All keypad keysyms are aligned at the same order as non-keypad ones */
+ guint keypad_keysym = keysym - GDK_KEY_Left + GDK_KEY_KP_Left;
+ gboolean success = FALSE;
+
+ if (keyval != keysym && keyval != keypad_keysym)
+ return GDK_EVENT_PROPAGATE;
+
+ if (state == GDK_CONTROL_MASK)
+ success = select_page (self, direction, last);
+ else if (state == (GDK_CONTROL_MASK | GDK_SHIFT_MASK))
+ success = reorder_page (self, direction, last);
+ else
+ return GDK_EVENT_PROPAGATE;
+
+ if (!success)
+ gtk_widget_error_bell (GTK_WIDGET (self));
+
+ return GDK_EVENT_STOP;
+}
+
+static gboolean
+select_page_cb (GtkWidget *widget,
+ GVariant *args,
+ AdwTabView *self)
+{
+ GtkDirectionType direction;
+ gboolean last, loop, success = FALSE;
+
+ if (!adw_tab_view_get_selected_page (self))
+ return GDK_EVENT_PROPAGATE;
+
+ g_variant_get (args, "(hbb)", &direction, &last, &loop);
+
+ if (direction == GTK_DIR_TAB_BACKWARD) {
+ if (last)
+ success = adw_tab_view_select_first_page (self);
+ else
+ success = adw_tab_view_select_previous_page (self);
+
+ if (!success && loop) {
+ AdwTabPage *page = adw_tab_view_get_nth_page (self, self->n_pages - 1);
+
+ adw_tab_view_set_selected_page (self, page);
+
+ success = TRUE;
+ }
+ } else if (direction == GTK_DIR_TAB_FORWARD) {
+ if (last)
+ success = adw_tab_view_select_last_page (self);
+ else
+ success = adw_tab_view_select_next_page (self);
+
+ if (!success && loop) {
+ AdwTabPage *page = adw_tab_view_get_nth_page (self, 0);
+
+ adw_tab_view_set_selected_page (self, page);
+
+ success = TRUE;
+ }
+ }
+
+ if (!success)
+ gtk_widget_error_bell (GTK_WIDGET (self));
+
+ return success;
+}
+
+static inline void
+add_switch_shortcut (AdwTabView *self,
+ GtkEventController *controller,
+ guint keysym,
+ guint keypad_keysym,
+ GdkModifierType modifiers,
+ GtkDirectionType direction,
+ gboolean last,
+ gboolean loop)
+{
+ GtkShortcutTrigger *trigger;
+ GtkShortcutAction *action;
+ GtkShortcut *shortcut;
+
+ trigger = gtk_alternative_trigger_new (gtk_keyval_trigger_new (keysym, modifiers),
+ gtk_keyval_trigger_new (keypad_keysym, modifiers));
+ action = gtk_callback_action_new ((GtkShortcutFunc) select_page_cb, self, NULL);
+ shortcut = gtk_shortcut_new (trigger, action);
+
+ gtk_shortcut_set_arguments (shortcut, g_variant_new ("(hbb)", direction, last, loop));
+ gtk_shortcut_controller_add_shortcut (GTK_SHORTCUT_CONTROLLER (controller),
+ shortcut);
+}
+
+static gboolean
+reorder_page_cb (GtkWidget *widget,
+ GVariant *args,
+ AdwTabView *self)
+{
+ GtkDirectionType direction;
+ gboolean last, success = FALSE;
+ AdwTabPage *page = adw_tab_view_get_selected_page (self);
+
+ if (!page)
+ return GDK_EVENT_PROPAGATE;
+
+ g_variant_get (args, "(hb)", &direction, &last);
+
+ if (direction == GTK_DIR_TAB_BACKWARD) {
+ if (last)
+ success = adw_tab_view_reorder_first (self, page);
+ else
+ success = adw_tab_view_reorder_backward (self, page);
+ } else if (direction == GTK_DIR_TAB_FORWARD) {
+ if (last)
+ success = adw_tab_view_reorder_last (self, page);
+ else
+ success = adw_tab_view_reorder_forward (self, page);
+ }
+
+ if (!success)
+ gtk_widget_error_bell (GTK_WIDGET (self));
+
+ return success;
+}
+
+static inline void
+add_reorder_shortcut (AdwTabView *self,
+ GtkEventController *controller,
+ guint keysym,
+ guint keypad_keysym,
+ GtkDirectionType direction,
+ gboolean last)
+{
+ GtkShortcutTrigger *trigger;
+ GtkShortcutAction *action;
+ GtkShortcut *shortcut;
+
+ trigger = gtk_alternative_trigger_new (gtk_keyval_trigger_new (keysym, GDK_CONTROL_MASK | GDK_SHIFT_MASK),
+ gtk_keyval_trigger_new (keypad_keysym, GDK_CONTROL_MASK |
GDK_SHIFT_MASK));
+ action = gtk_callback_action_new ((GtkShortcutFunc) reorder_page_cb, self, NULL);
+ shortcut = gtk_shortcut_new (trigger, action);
+
+ gtk_shortcut_set_arguments (shortcut, g_variant_new ("(hb)", direction, last));
+ gtk_shortcut_controller_add_shortcut (GTK_SHORTCUT_CONTROLLER (controller),
+ shortcut);
+}
+
+static gboolean
+select_nth_page_cb (GtkWidget *widget,
+ GVariant *args,
+ AdwTabView *self)
+{
+ gint8 n_page = g_variant_get_byte (args);
+ AdwTabPage *page;
+
+ if (n_page >= self->n_pages)
+ return GDK_EVENT_PROPAGATE;
+
+ page = adw_tab_view_get_nth_page (self, n_page);
+ adw_tab_view_set_selected_page (self, page);
+
+ return GDK_EVENT_STOP;
+}
+
+static inline void
+add_switch_nth_page_shortcut (AdwTabView *self,
+ GtkEventController *controller,
+ guint keysym,
+ guint keypad_keysym,
+ int n_page)
+{
+ GtkShortcutTrigger *trigger;
+ GtkShortcutAction *action;
+ GtkShortcut *shortcut;
+
+ trigger = gtk_alternative_trigger_new (gtk_keyval_trigger_new (keysym, GDK_ALT_MASK),
+ gtk_keyval_trigger_new (keypad_keysym, GDK_ALT_MASK));
+ action = gtk_callback_action_new ((GtkShortcutFunc) select_nth_page_cb, self, NULL);
+ shortcut = gtk_shortcut_new (trigger, action);
+
+ gtk_shortcut_set_arguments (shortcut, g_variant_new_byte (n_page));
+ gtk_shortcut_controller_add_shortcut (GTK_SHORTCUT_CONTROLLER (controller),
+ shortcut);
+}
+
+static void
+init_shortcuts (AdwTabView *self,
+ GtkEventController *controller)
+{
+ int i;
+
+ add_switch_shortcut (self, controller,
+ GDK_KEY_Tab, GDK_KEY_KP_Tab, GDK_CONTROL_MASK,
+ GTK_DIR_TAB_FORWARD, FALSE, TRUE);
+ add_switch_shortcut (self, controller,
+ GDK_KEY_Tab, GDK_KEY_KP_Tab, GDK_CONTROL_MASK | GDK_SHIFT_MASK,
+ GTK_DIR_TAB_BACKWARD, FALSE, TRUE);
+ add_switch_shortcut (self, controller,
+ GDK_KEY_Page_Up, GDK_KEY_KP_Page_Up, GDK_CONTROL_MASK,
+ GTK_DIR_TAB_BACKWARD, FALSE, FALSE);
+ add_switch_shortcut (self, controller,
+ GDK_KEY_Page_Down, GDK_KEY_KP_Page_Down, GDK_CONTROL_MASK,
+ GTK_DIR_TAB_FORWARD, FALSE, FALSE);
+ add_switch_shortcut (self, controller,
+ GDK_KEY_Home, GDK_KEY_KP_Home, GDK_CONTROL_MASK,
+ GTK_DIR_TAB_BACKWARD, TRUE, FALSE);
+ add_switch_shortcut (self, controller,
+ GDK_KEY_End, GDK_KEY_KP_End, GDK_CONTROL_MASK,
+ GTK_DIR_TAB_FORWARD, TRUE, FALSE);
+
+ add_reorder_shortcut (self, controller,
+ GDK_KEY_Page_Up, GDK_KEY_KP_Page_Up,
+ GTK_DIR_TAB_BACKWARD, FALSE);
+ add_reorder_shortcut (self, controller,
+ GDK_KEY_Page_Down, GDK_KEY_KP_Page_Down,
+ GTK_DIR_TAB_FORWARD, FALSE);
+ add_reorder_shortcut (self, controller,
+ GDK_KEY_Home, GDK_KEY_KP_Home,
+ GTK_DIR_TAB_BACKWARD, TRUE);
+ add_reorder_shortcut (self, controller,
+ GDK_KEY_End, GDK_KEY_KP_End,
+ GTK_DIR_TAB_FORWARD, TRUE);
+
+ for (i = 0; i < 10; i++)
+ add_switch_nth_page_shortcut (self,
+ controller,
+ GDK_KEY_0 + i,
+ GDK_KEY_KP_0 + i,
+ (i + 9) % 10); /* Alt+0 means page 10, not 0 */
+}
+
+static void
+shortcut_widget_notify_cb (AdwTabView *self)
+{
+ gtk_widget_remove_controller (self->shortcut_widget, self->shortcut_controller);
+ self->shortcut_controller = NULL;
+ self->shortcut_widget = NULL;
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SHORTCUT_WIDGET]);
+}
+
+static void
+adw_tab_view_dispose (GObject *object)
+{
+ AdwTabView *self = ADW_TAB_VIEW (object);
+
+ adw_tab_view_set_shortcut_widget (self, NULL);
+
+ if (self->pages)
+ g_list_model_items_changed (G_LIST_MODEL (self->pages), 0, self->n_pages, 0);
+
+ while (self->n_pages) {
+ AdwTabPage *page = adw_tab_view_get_nth_page (self, 0);
+
+ detach_page (self, page, FALSE);
+ }
+
+ g_clear_object (&self->children);
+
+ g_clear_pointer ((GtkWidget **) &self->stack, gtk_widget_unparent);
+
+ G_OBJECT_CLASS (adw_tab_view_parent_class)->dispose (object);
+}
+
+static void
+adw_tab_view_finalize (GObject *object)
+{
+ AdwTabView *self = (AdwTabView *) object;
+
+ if (self->pages)
+ g_object_remove_weak_pointer (G_OBJECT (self->pages),
+ (gpointer *) &self->pages);
+
+ g_clear_object (&self->default_icon);
+ g_clear_object (&self->menu_model);
+
+ tab_view_list = g_slist_remove (tab_view_list, self);
+
+ G_OBJECT_CLASS (adw_tab_view_parent_class)->finalize (object);
+}
+
+static void
+adw_tab_view_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ AdwTabView *self = ADW_TAB_VIEW (object);
+
+ switch (prop_id) {
+ case PROP_N_PAGES:
+ g_value_set_int (value, adw_tab_view_get_n_pages (self));
+ break;
+
+ case PROP_N_PINNED_PAGES:
+ g_value_set_int (value, adw_tab_view_get_n_pinned_pages (self));
+ break;
+
+ case PROP_IS_TRANSFERRING_PAGE:
+ g_value_set_boolean (value, adw_tab_view_get_is_transferring_page (self));
+ break;
+
+ case PROP_SELECTED_PAGE:
+ g_value_set_object (value, adw_tab_view_get_selected_page (self));
+ break;
+
+ case PROP_DEFAULT_ICON:
+ g_value_set_object (value, adw_tab_view_get_default_icon (self));
+ break;
+
+ case PROP_MENU_MODEL:
+ g_value_set_object (value, adw_tab_view_get_menu_model (self));
+ break;
+
+ case PROP_SHORTCUT_WIDGET:
+ g_value_set_object (value, adw_tab_view_get_shortcut_widget (self));
+ break;
+
+ case PROP_PAGES:
+ g_value_set_object (value, adw_tab_view_get_pages (self));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+adw_tab_view_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ AdwTabView *self = ADW_TAB_VIEW (object);
+
+ switch (prop_id) {
+ case PROP_SELECTED_PAGE:
+ adw_tab_view_set_selected_page (self, g_value_get_object (value));
+ break;
+
+ case PROP_DEFAULT_ICON:
+ adw_tab_view_set_default_icon (self, g_value_get_object (value));
+ break;
+
+ case PROP_MENU_MODEL:
+ adw_tab_view_set_menu_model (self, g_value_get_object (value));
+ break;
+
+ case PROP_SHORTCUT_WIDGET:
+ adw_tab_view_set_shortcut_widget (self, g_value_get_object (value));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+adw_tab_view_class_init (AdwTabViewClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ object_class->dispose = adw_tab_view_dispose;
+ object_class->finalize = adw_tab_view_finalize;
+ object_class->get_property = adw_tab_view_get_property;
+ object_class->set_property = adw_tab_view_set_property;
+
+ /**
+ * AdwTabView:n-pages:
+ *
+ * The number of pages in the tab view.
+ *
+ * Since: 1.0
+ */
+ props[PROP_N_PAGES] =
+ g_param_spec_int ("n-pages",
+ "Number of pages",
+ "The number of pages in the tab view",
+ 0, G_MAXINT, 0,
+ G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:n-pinned-pages:
+ *
+ * The number of pinned pages in the tab view.
+ *
+ * See adw_tab_view_set_page_pinned().
+ *
+ * Since: 1.0
+ */
+ props[PROP_N_PINNED_PAGES] =
+ g_param_spec_int ("n-pinned-pages",
+ "Number of pinned pages",
+ "The number of pinned pages in the tab view",
+ 0, G_MAXINT, 0,
+ G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:is-transferring-page:
+ *
+ * Whether a page is being transferred.
+ *
+ * This property will be set to %TRUE when a drag-n-drop tab transfer starts
+ * on any #AdwTabView, and to %FALSE after it ends.
+ *
+ * During the transfer, children cannot receive pointer input and a tab can
+ * be safely dropped on the tab view.
+ *
+ * Since: 1.0
+ */
+ props[PROP_IS_TRANSFERRING_PAGE] =
+ g_param_spec_boolean ("is-transferring-page",
+ "Is transferring page",
+ "Whether a page is being transferred",
+ FALSE,
+ G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:selected-page:
+ *
+ * The currently selected page.
+ *
+ * Since: 1.0
+ */
+ props[PROP_SELECTED_PAGE] =
+ g_param_spec_object ("selected-page",
+ "Selected page",
+ "The currently selected page",
+ ADW_TYPE_TAB_PAGE,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:default-icon:
+ *
+ * Default page icon.
+ *
+ * If a page doesn't provide its own icon via #AdwTabPage:icon, default icon
+ * may be used instead for contexts where having an icon is necessary.
+ *
+ * #AdwTabBar will use default icon for pinned tabs in case the page is not
+ * loading, doesn't have an icon and an indicator. Default icon is never used
+ * for tabs that aren't pinned.
+ *
+ * Since: 1.0
+ */
+ props[PROP_DEFAULT_ICON] =
+ g_param_spec_object ("default-icon",
+ "Default icon",
+ "Default page icon",
+ G_TYPE_ICON,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:menu-model:
+ *
+ * Tab context menu model.
+ *
+ * When a context menu is shown for a tab, it will be constructed from the
+ * provided menu model. Use #AdwTabView::setup-menu signal to set up the menu
+ * actions for the particular tab.
+ *
+ * Since: 1.0
+ */
+ props[PROP_MENU_MODEL] =
+ g_param_spec_object ("menu-model",
+ "Menu model",
+ "Tab context menu model",
+ G_TYPE_MENU_MODEL,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:shortcut-widget:
+ *
+ * Tab shortcut widget, has the following shortcuts:
+ * * Ctrl+Page Up - switch to the previous page
+ * * Ctrl+Page Down - switch to the next page
+ * * Ctrl+Home - switch to the first page
+ * * Ctrl+End - switch to the last page
+ * * Ctrl+Shift+Page Up - move the current page backward
+ * * Ctrl+Shift+Page Down - move the current page forward
+ * * Ctrl+Shift+Home - move the current page at the start
+ * * Ctrl+Shift+End - move the current page at the end
+ * * Ctrl+Tab - switch to the next page, with looping
+ * * Ctrl+Shift+Tab - switch to the previous page, with looping
+ * * Alt+1-9 - switch to pages 1-9
+ * * Alt+0 - switch to page 10
+ *
+ * These shortcuts are always available on @self, this property is useful if
+ * they should be available globally.
+ *
+ * Since: 1.0
+ */
+ props[PROP_SHORTCUT_WIDGET] =
+ g_param_spec_object ("shortcut-widget",
+ "Shortcut widget",
+ "Tab shortcut widget",
+ GTK_TYPE_WIDGET,
+ G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
+
+ /**
+ * AdwTabView:pages:
+ *
+ * A selection model with the tab view's pages.
+ *
+ * It can be used to keep an up-to-date view.
+ *
+ * Since: 1.0
+ */
+ props[PROP_PAGES] =
+ g_param_spec_object ("pages",
+ "Pages",
+ "A selection model with the tab view's pages",
+ GTK_TYPE_SELECTION_MODEL,
+ G_PARAM_READABLE);
+
+ g_object_class_install_properties (object_class, LAST_PROP, props);
+
+ /**
+ * AdwTabView::page-attached:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @position: the position of the page, starting from 0
+ *
+ * This signal is emitted when a page has been created or transferred to
+ * @self.
+ *
+ * A typical reason to connect to this signal would be to connect to page
+ * signals for things such as updating window title.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_PAGE_ATTACHED] =
+ g_signal_new ("page-attached",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 2,
+ ADW_TYPE_TAB_PAGE, G_TYPE_INT);
+
+ /**
+ * AdwTabView::page-detached:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @position: the position of the removed page, starting from 0
+ *
+ * This signal is emitted when a page has been removed or transferred to
+ * another view.
+ *
+ * A typical reason to connect to this signal would be to disconnect signal
+ * handlers connected in the #AdwTabView::page-attached handler.
+ *
+ * It is important not to try and destroy the page child in the handler of
+ * this function as the child might merely be moved to another window; use
+ * child dispose handler for that or do it in sync with your
+ * adw_tab_view_close_page_finish() calls.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_PAGE_DETACHED] =
+ g_signal_new ("page-detached",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 2,
+ ADW_TYPE_TAB_PAGE, G_TYPE_INT);
+
+ /**
+ * AdwTabView::page-reordered:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @position: the position @page was moved to, starting at 0
+ *
+ * This signal is emitted after @page has been reordered to @position.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_PAGE_REORDERED] =
+ g_signal_new ("page-reordered",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 2,
+ ADW_TYPE_TAB_PAGE, G_TYPE_INT);
+
+ /**
+ * AdwTabView::close-page:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * This signal is emitted after adw_tab_view_close_page() has been called for
+ * @page.
+ *
+ * The handler is expected to call adw_tab_view_close_page_finish() to confirm
+ * or reject the closing.
+ *
+ * The default handler will immediately confirm closing for non-pinned pages,
+ * or reject it for pinned pages, equivalent to the following example:
+ *
+ * |[<!-- language="C" -->
+ * static gboolean
+ * close_page_cb (AdwTabView *view,
+ * AdwTabPage *page,
+ * gpointer user_data)
+ * {
+ * adw_tab_view_close_page_finish (view, page, !adw_tab_page_get_pinned (page));
+ *
+ * return GDK_EVENT_STOP;
+ * }
+ * ]|
+ *
+ * The adw_tab_view_close_page_finish() doesn't have to happen during the
+ * handler, so can be used to do asynchronous checks before confirming the
+ * closing.
+ *
+ * A typical reason to connect to this signal is to show a confirmation dialog
+ * for closing a tab.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_CLOSE_PAGE] =
+ g_signal_new ("close-page",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ g_signal_accumulator_true_handled,
+ NULL, NULL,
+ G_TYPE_BOOLEAN,
+ 1,
+ ADW_TYPE_TAB_PAGE);
+
+ /**
+ * AdwTabView::setup-menu:
+ * @self: a #AdwTabView
+ * @page: a page of @self, or %NULL
+ *
+ * This signal is emitted before a context menu is opened for @page, and after
+ * it's closed, in the latter case the @page will be set to %NULL.
+ *
+ * It can be used to set up menu actions before showing the menu, for example
+ * disable actions not applicable to @page.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_SETUP_MENU] =
+ g_signal_new ("setup-menu",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 1,
+ ADW_TYPE_TAB_PAGE);
+
+ /**
+ * AdwTabView::create-window:
+ * @self: a #AdwTabView
+ *
+ * This signal is emitted when a tab is dropped onto desktop and should be
+ * transferred into a new window.
+ *
+ * The signal handler is expected to create a new window, position it as
+ * needed and return its #AdwTabView that the page will be transferred into.
+ *
+ * Returns: (transfer none) (nullable): the #AdwTabView from the new window
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_CREATE_WINDOW] =
+ g_signal_new ("create-window",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ object_handled_accumulator,
+ NULL, NULL,
+ ADW_TYPE_TAB_VIEW,
+ 0);
+
+ /**
+ * AdwTabView::indicator-activated:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * This signal is emitted after the indicator icon on @page has been activated.
+ *
+ * See #AdwTabPage:indicator-icon and #AdwTabPage:indicator-activatable.
+ *
+ * Since: 1.0
+ */
+ signals[SIGNAL_INDICATOR_ACTIVATED] =
+ g_signal_new ("indicator-activated",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE,
+ 1,
+ ADW_TYPE_TAB_PAGE);
+
+ g_signal_override_class_handler ("close-page",
+ G_TYPE_FROM_CLASS (klass),
+ G_CALLBACK (close_page_cb));
+
+ gtk_widget_class_set_css_name (widget_class, "tabview");
+ gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_BIN_LAYOUT);
+}
+
+static void
+adw_tab_view_init (AdwTabView *self)
+{
+ GtkEventController *controller;
+
+ self->children = g_list_store_new (ADW_TYPE_TAB_PAGE);
+ self->default_icon = G_ICON (g_themed_icon_new ("adw-tab-icon-missing-symbolic"));
+
+ self->stack = GTK_STACK (gtk_stack_new ());
+ gtk_widget_show (GTK_WIDGET (self->stack));
+ gtk_widget_set_parent (GTK_WIDGET (self->stack), GTK_WIDGET (self));
+
+ g_object_bind_property (self, "is-transferring-page",
+ self->stack, "can-focus",
+ G_BINDING_INVERT_BOOLEAN);
+
+ tab_view_list = g_slist_prepend (tab_view_list, self);
+
+ controller = gtk_shortcut_controller_new ();
+
+ init_shortcuts (self, controller);
+
+ gtk_widget_add_controller (GTK_WIDGET (self), controller);
+}
+
+/**
+ * adw_tab_page_get_child:
+ * @self: a #AdwTabPage
+ *
+ * Gets the child of @self.
+ *
+ * Returns: (transfer none): the child of @self
+ *
+ * Since: 1.0
+ */
+GtkWidget *
+adw_tab_page_get_child (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->child;
+}
+
+/**
+ * adw_tab_page_get_parent:
+ * @self: a #AdwTabPage
+ *
+ * Gets the parent page of @self, or %NULL if the @self does not have a parent.
+ *
+ * See adw_tab_view_add_page() and adw_tab_view_close_page().
+ *
+ * Returns: (transfer none) (nullable): the parent page of @self, or %NULL
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_page_get_parent (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->parent;
+}
+
+/**
+ * adw_tab_page_get_selected:
+ * @self: a #AdwTabPage
+ *
+ * Gets whether @self is selected. See adw_tab_view_set_selected_page().
+ *
+ * Returns: whether @self is selected
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_page_get_selected (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), FALSE);
+
+ return self->selected;
+}
+
+/**
+ * adw_tab_page_get_pinned:
+ * @self: a #AdwTabPage
+ *
+ * Gets whether @self is pinned. See adw_tab_view_set_page_pinned().
+ *
+ * Returns: whether @self is pinned
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_page_get_pinned (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), FALSE);
+
+ return self->pinned;
+}
+
+/**
+ * adw_tab_page_get_title:
+ * @self: a #AdwTabPage
+ *
+ * Gets the title of @self, see adw_tab_page_set_title().
+ *
+ * Returns: (nullable): the title of @self
+ *
+ * Since: 1.0
+ */
+const char *
+adw_tab_page_get_title (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->title;
+}
+
+/**
+ * adw_tab_page_set_title:
+ * @self: a #AdwTabPage
+ * @title: (nullable): the title of @self
+ *
+ * Sets the title of @self.
+ *
+ * #AdwTabBar will display it in the center of the tab representing @self
+ * unless it's pinned, and will use it as a tooltip unless #AdwTabPage:tooltip
+ * is set.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_title (AdwTabPage *self,
+ const char *title)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ if (!g_strcmp0 (title, self->title))
+ return;
+
+ g_clear_pointer (&self->title, g_free);
+ self->title = g_strdup (title);
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_TITLE]);
+}
+
+/**
+ * adw_tab_page_get_tooltip:
+ * @self: a #AdwTabPage
+ *
+ * Gets the tooltip of @self, see adw_tab_page_set_tooltip().
+ *
+ * Returns: (nullable): the tooltip of @self
+ *
+ * Since: 1.0
+ */
+const char *
+adw_tab_page_get_tooltip (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->tooltip;
+}
+
+/**
+ * adw_tab_page_set_tooltip:
+ * @self: a #AdwTabPage
+ * @tooltip: (nullable): the tooltip of @self
+ *
+ * Sets the tooltip of @self, marked up with the Pango text markup language.
+ *
+ * If not set, #AdwTabBar will use #AdwTabPage:title as a tooltip instead.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_tooltip (AdwTabPage *self,
+ const char *tooltip)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ if (!g_strcmp0 (tooltip, self->tooltip))
+ return;
+
+ g_clear_pointer (&self->tooltip, g_free);
+ self->tooltip = g_strdup (tooltip);
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_TOOLTIP]);
+}
+
+/**
+ * adw_tab_page_get_icon:
+ * @self: a #AdwTabPage
+ *
+ * Gets the icon of @self, see adw_tab_page_set_icon().
+ *
+ * Returns: (transfer none) (nullable): the icon of @self
+ *
+ * Since: 1.0
+ */
+GIcon *
+adw_tab_page_get_icon (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->icon;
+}
+
+/**
+ * adw_tab_page_set_icon:
+ * @self: a #AdwTabPage
+ * @icon: (nullable): the icon of @self
+ *
+ * Sets the icon of @self, displayed next to the title.
+ *
+ * #AdwTabBar will not show the icon if #AdwTabPage:loading is set to %TRUE,
+ * or if @self is pinned and #AdwTabPage:indicator-icon is set.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_icon (AdwTabPage *self,
+ GIcon *icon)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+ g_return_if_fail (G_IS_ICON (icon) || icon == NULL);
+
+ if (self->icon == icon)
+ return;
+
+ g_set_object (&self->icon, icon);
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_ICON]);
+}
+
+/**
+ * adw_tab_page_get_loading:
+ * @self: a #AdwTabPage
+ *
+ * Gets whether @self is loading, see adw_tab_page_set_loading().
+ *
+ * Returns: whether @self is loading
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_page_get_loading (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), FALSE);
+
+ return self->loading;
+}
+
+/**
+ * adw_tab_page_set_loading:
+ * @self: a #AdwTabPage
+ * @loading: whether @self is loading
+ *
+ * Sets wether @self is loading.
+ *
+ * If set to %TRUE, #AdwTabBar will display a spinner in place of icon.
+ *
+ * If @self is pinned and #AdwTabPage:indicator-icon is set, the loading status
+ * will not be visible.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_loading (AdwTabPage *self,
+ gboolean loading)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ loading = !!loading;
+
+ if (self->loading == loading)
+ return;
+
+ self->loading = loading;
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_LOADING]);
+}
+
+/**
+ * adw_tab_page_get_indicator_icon:
+ * @self: a #AdwTabPage
+ *
+ * Gets the indicator icon of @self, see adw_tab_page_set_indicator_icon().
+ *
+ * Returns: (transfer none) (nullable): the indicator icon of @self
+ *
+ * Since: 1.0
+ */
+GIcon *
+adw_tab_page_get_indicator_icon (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), NULL);
+
+ return self->indicator_icon;
+}
+
+/**
+ * adw_tab_page_set_indicator_icon:
+ * @self: a #AdwTabPage
+ * @indicator_icon: (nullable): the indicator icon of @self
+ *
+ * Sets the indicator icon of @self.
+ *
+ * A common use case is an audio or camera indicator in a web browser.
+ *
+ * #AdwTabPage will show it at the beginning of the tab, alongside icon
+ * representing #AdwTabPage:icon or loading spinner.
+ *
+ * If the page is pinned, the indicator will be shown instead of icon or spinner.
+ *
+ * If #AdwTabPage:indicator-activatable is set to %TRUE, indicator icon
+ * can act as a button.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_indicator_icon (AdwTabPage *self,
+ GIcon *indicator_icon)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+ g_return_if_fail (G_IS_ICON (indicator_icon) || indicator_icon == NULL);
+
+ if (self->indicator_icon == indicator_icon)
+ return;
+
+ g_set_object (&self->indicator_icon, indicator_icon);
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_INDICATOR_ICON]);
+}
+
+/**
+ * adw_tab_page_get_indicator_activatable:
+ * @self: a #AdwTabPage
+ *
+ *
+ * Gets whether the indicator of @self is activatable, see
+ * adw_tab_page_set_indicator_activatable().
+ *
+ * Returns: whether the indicator is activatable
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_page_get_indicator_activatable (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), FALSE);
+
+ return self->indicator_activatable;
+}
+
+/**
+ * adw_tab_page_set_indicator_activatable:
+ * @self: a #AdwTabPage
+ * @activatable: whether the indicator is activatable
+ *
+ * sets whether the indicator of @self is activatable.
+ *
+ * If set to %TRUE, #AdwTabView::indicator-activated will be emitted when
+ * the indicator is clicked.
+ *
+ * If #AdwTabPage:indicator-icon is not set, does nothing.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_indicator_activatable (AdwTabPage *self,
+ gboolean activatable)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ activatable = !!activatable;
+
+ if (self->indicator_activatable == activatable)
+ return;
+
+ self->indicator_activatable = activatable;
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_INDICATOR_ACTIVATABLE]);
+}
+
+/**
+ * adw_tab_page_get_needs_attention:
+ * @self: a #AdwTabPage
+ *
+ * Gets whether @self needs attention, see adw_tab_page_set_needs_attention().
+ *
+ * Returns: whether @self needs attention
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_page_get_needs_attention (AdwTabPage *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (self), FALSE);
+
+ return self->needs_attention;
+}
+
+/**
+ * adw_tab_page_set_needs_attention:
+ * @self: a #AdwTabPage
+ * @needs_attention: whether @self needs attention
+ *
+ * Sets whether @self needs attention.
+ *
+ * #AdwTabBar will display a glow under the tab representing @self if set to
+ * %TRUE. If the tab is not visible, the corresponding edge of the tab bar will
+ * be highlighted.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_page_set_needs_attention (AdwTabPage *self,
+ gboolean needs_attention)
+{
+ g_return_if_fail (ADW_IS_TAB_PAGE (self));
+
+ needs_attention = !!needs_attention;
+
+ if (self->needs_attention == needs_attention)
+ return;
+
+ self->needs_attention = needs_attention;
+
+ g_object_notify_by_pspec (G_OBJECT (self), page_props[PAGE_PROP_NEEDS_ATTENTION]);
+}
+
+/**
+ * adw_tab_view_new:
+ *
+ * Creates a new #AdwTabView widget.
+ *
+ * Returns: a new #AdwTabView
+ *
+ * Since: 1.0
+ */
+AdwTabView *
+adw_tab_view_new (void)
+{
+ return g_object_new (ADW_TYPE_TAB_VIEW, NULL);
+}
+
+/**
+ * adw_tab_view_get_n_pages:
+ * @self: a #AdwTabView
+ *
+ * Gets the number of pages in @self.
+ *
+ * Returns: the number of pages in @self
+ *
+ * Since: 1.0
+ */
+int
+adw_tab_view_get_n_pages (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), 0);
+
+ return self->n_pages;
+}
+
+/**
+ * adw_tab_view_get_n_pinned_pages:
+ * @self: a #AdwTabView
+ *
+ * Gets the number of pinned pages in @self.
+ *
+ * See adw_tab_view_set_page_pinned().
+ *
+ * Returns: the number of pinned pages in @self
+ *
+ * Since: 1.0
+ */
+int
+adw_tab_view_get_n_pinned_pages (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), 0);
+
+ return self->n_pinned_pages;
+}
+
+/**
+ * adw_tab_view_get_is_transferring_page:
+ * @self: a #AdwTabView
+ *
+ * Whether a page is being transferred.
+ *
+ * Gets the value of #AdwTabView:is-transferring-page property.
+ *
+ * Returns: whether a page is being transferred
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_get_is_transferring_page (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+
+ return self->transfer_count > 0;
+}
+
+/**
+ * adw_tab_view_get_selected_page:
+ * @self: a #AdwTabView
+ *
+ * Gets the currently selected page in @self.
+ *
+ * Returns: (transfer none) (nullable): the selected page in @self
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_get_selected_page (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+
+ return self->selected_page;
+}
+
+/**
+ * adw_tab_view_set_selected_page:
+ * @self: a #AdwTabView
+ * @selected_page: a page in @self
+ *
+ * Sets the currently selected page in @self.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_set_selected_page (AdwTabView *self,
+ AdwTabPage *selected_page)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+
+ if (self->n_pages > 0) {
+ g_return_if_fail (ADW_IS_TAB_PAGE (selected_page));
+ g_return_if_fail (page_belongs_to_this_view (self, selected_page));
+ } else {
+ g_return_if_fail (selected_page == NULL);
+ }
+
+ set_selected_page (self, selected_page, TRUE);
+}
+
+/**
+ * adw_tab_view_select_previous_page:
+ * @self: a #AdwTabView
+ *
+ * Selects the page before the currently selected page.
+ *
+ * If the first page was already selected, this function does nothing.
+ *
+ * Returns: %TRUE if the selected page was changed, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_select_previous_page (AdwTabView *self)
+{
+ AdwTabPage *page;
+ int pos;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+
+ if (!self->selected_page)
+ return FALSE;
+
+ pos = adw_tab_view_get_page_position (self, self->selected_page);
+
+ if (pos <= 0)
+ return FALSE;
+
+ page = adw_tab_view_get_nth_page (self, pos - 1);
+
+ adw_tab_view_set_selected_page (self, page);
+
+ return TRUE;
+}
+
+/**
+ * adw_tab_view_select_next_page:
+ * @self: a #AdwTabView
+ *
+ * Selects the page after the currently selected page.
+ *
+ * If the last page was already selected, this function does nothing.
+ *
+ * Returns: %TRUE if the selected page was changed, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_select_next_page (AdwTabView *self)
+{
+ AdwTabPage *page;
+ int pos;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+
+ if (!self->selected_page)
+ return FALSE;
+
+ pos = adw_tab_view_get_page_position (self, self->selected_page);
+
+ if (pos >= self->n_pages - 1)
+ return FALSE;
+
+ page = adw_tab_view_get_nth_page (self, pos + 1);
+
+ adw_tab_view_set_selected_page (self, page);
+
+ return TRUE;
+}
+
+gboolean
+adw_tab_view_select_first_page (AdwTabView *self)
+{
+ AdwTabPage *page;
+ int pos;
+ gboolean pinned;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+
+ if (!self->selected_page)
+ return FALSE;
+
+ pinned = adw_tab_page_get_pinned (self->selected_page);
+ pos = pinned ? 0 : self->n_pinned_pages;
+
+ page = adw_tab_view_get_nth_page (self, pos);
+
+ /* If we're on the first non-pinned tab, go to the first pinned tab */
+ if (page == self->selected_page && !pinned)
+ page = adw_tab_view_get_nth_page (self, 0);
+
+ if (page == self->selected_page)
+ return FALSE;
+
+ adw_tab_view_set_selected_page (self, page);
+
+ return TRUE;
+}
+
+gboolean
+adw_tab_view_select_last_page (AdwTabView *self)
+{
+ AdwTabPage *page;
+ int pos;
+ gboolean pinned;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+
+ if (!self->selected_page)
+ return FALSE;
+
+ pinned = adw_tab_page_get_pinned (self->selected_page);
+ pos = (pinned ? self->n_pinned_pages : self->n_pages) - 1;
+
+ page = adw_tab_view_get_nth_page (self, pos);
+
+ /* If we're on the last pinned tab, go to the last non-pinned tab */
+ if (page == self->selected_page && pinned)
+ page = adw_tab_view_get_nth_page (self, self->n_pages - 1);
+
+ if (page == self->selected_page)
+ return FALSE;
+
+ adw_tab_view_set_selected_page (self, page);
+
+ return TRUE;
+}
+
+/**
+ * adw_tab_view_get_default_icon:
+ * @self: a #AdwTabView
+ *
+ * Gets default icon of @self, see adw_tab_view_set_default_icon().
+ *
+ * Returns: (transfer none): the default icon of @self.
+ *
+ * Since: 1.0
+ */
+GIcon *
+adw_tab_view_get_default_icon (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+
+ return self->default_icon;
+}
+
+/**
+ * adw_tab_view_set_default_icon:
+ * @self: a #AdwTabView
+ * @default_icon: the default icon
+ *
+ * Sets default page icon for @self.
+ *
+ * If a page doesn't provide its own icon via #AdwTabPage:icon, default icon
+ * may be used instead for contexts where having an icon is necessary.
+ *
+ * #AdwTabBar will use default icon for pinned tabs in case the page is not
+ * loading, doesn't have an icon and an indicator. Default icon is never used
+ * for tabs that aren't pinned.
+ *
+ * By default, 'adw-tab-icon-missing-symbolic' icon is used.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_set_default_icon (AdwTabView *self,
+ GIcon *default_icon)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (G_IS_ICON (default_icon));
+
+ if (self->default_icon == default_icon)
+ return;
+
+ g_set_object (&self->default_icon, default_icon);
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_DEFAULT_ICON]);
+}
+
+/**
+ * adw_tab_view_get_menu_model:
+ * @self: a #AdwTabView
+ *
+ * Gets the tab context menu model for @self, see adw_tab_view_set_menu_model().
+ *
+ * Returns: (transfer none) (nullable): the tab context menu model for @self
+ *
+ * Since: 1.0
+ */
+GMenuModel *
+adw_tab_view_get_menu_model (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+
+ return self->menu_model;
+}
+
+/**
+ * adw_tab_view_set_menu_model:
+ * @self: a #AdwTabView
+ * @menu_model: (nullable): a menu model
+ *
+ * Sets the tab context menu model for @self.
+ *
+ * When a context menu is shown for a tab, it will be constructed from the
+ * provided menu model. Use #AdwTabView::setup-menu signal to set up the menu
+ * actions for the particular tab.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_set_menu_model (AdwTabView *self,
+ GMenuModel *menu_model)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (G_IS_MENU_MODEL (menu_model));
+
+ if (self->menu_model == menu_model)
+ return;
+
+ g_set_object (&self->menu_model, menu_model);
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_MENU_MODEL]);
+}
+
+/**
+ * adw_tab_view_get_shortcut_widget:
+ * @self: a #AdwTabView
+ *
+ * Gets the shortcut widget for @self, see adw_tab_view_set_shortcut_widget().
+ *
+ * Returns: (transfer none) (nullable): the shortcut widget for @self
+ *
+ * Since: 1.0
+ */
+GtkWidget *
+adw_tab_view_get_shortcut_widget (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+
+ return self->shortcut_widget;
+}
+
+/**
+ * adw_tab_view_set_shortcut_widget:
+ * @self: a #AdwTabView
+ * @widget: (nullable): a shortcut widget
+ *
+ * Sets the shortcut widget for @self.
+ *
+ * Registers the following shortcuts on @widget:
+ * * Ctrl+Page Up - switch to the previous page
+ * * Ctrl+Page Down - switch to the next page
+ * * Ctrl+Home - switch to the first page
+ * * Ctrl+End - switch to the last page
+ * * Ctrl+Shift+Page Up - move the current page backward
+ * * Ctrl+Shift+Page Down - move the current page forward
+ * * Ctrl+Shift+Home - move the current page at the start
+ * * Ctrl+Shift+End - move the current page at the end
+ * * Ctrl+Tab - switch to the next page, with looping
+ * * Ctrl+Shift+Tab - switch to the previous page, with looping
+ * * Alt+1-9 - switch to pages 1-9
+ * * Alt+0 - switch to page 10
+ *
+ * These shortcuts are always available on @self, this function is useful if
+ * they should be available globally.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_set_shortcut_widget (AdwTabView *self,
+ GtkWidget *widget)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (GTK_IS_WIDGET (widget) || widget == NULL);
+
+ if (widget == self->shortcut_widget)
+ return;
+
+ if (self->shortcut_widget) {
+ gtk_widget_remove_controller (self->shortcut_widget, self->shortcut_controller);
+ self->shortcut_controller = NULL;
+
+ g_object_weak_unref (G_OBJECT (self->shortcut_widget),
+ (GWeakNotify) shortcut_widget_notify_cb,
+ self);
+ }
+
+ self->shortcut_widget = widget;
+
+ if (self->shortcut_widget) {
+ g_object_weak_ref (G_OBJECT (self->shortcut_widget),
+ (GWeakNotify) shortcut_widget_notify_cb,
+ self);
+
+ self->shortcut_controller = gtk_shortcut_controller_new ();
+
+ init_shortcuts (self, self->shortcut_controller);
+
+ gtk_widget_add_controller (self->shortcut_widget, self->shortcut_controller);
+ }
+
+ g_object_notify_by_pspec (G_OBJECT (self), props[PROP_SHORTCUT_WIDGET]);
+}
+
+/**
+ * adw_tab_view_set_page_pinned:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @pinned: whether @page should be pinned
+ *
+ * Pins or unpins @page.
+ *
+ * Pinned pages are guaranteed to be placed before all non-pinned pages; at any
+ * given moment the first #AdwTabView:n-pinned-pages pages in @self are
+ * guaranteed to be pinned.
+ *
+ * When a page is pinned or unpinned, it's automatically reordered: pinning a
+ * page moves it after other pinned pages; unpinning a page moves it before
+ * other non-pinned pages.
+ *
+ * Pinned pages can still be reordered between each other.
+ *
+ * #AdwTabBar will display pinned pages in a compact form, never showing the
+ * title or close button, and only showing a single icon, selected in the
+ * following order:
+ *
+ * 1. #AdwTabPage:indicator-icon
+ * 2. A spinner if #AdwTabPage:loading is %TRUE
+ * 3. #AdwTabPage:icon
+ * 4. #AdwTabView:default-icon
+ *
+ * Pinned pages cannot be closed by default, see #AdwTabView::close-page for how
+ * to override that behavior.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_set_page_pinned (AdwTabView *self,
+ AdwTabPage *page,
+ gboolean pinned)
+{
+ int old_pos, new_pos;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ pinned = !!pinned;
+
+ if (adw_tab_page_get_pinned (page) == pinned)
+ return;
+
+ old_pos = adw_tab_view_get_page_position (self, page);
+
+ g_object_ref (page);
+
+ g_list_store_remove (self->children, old_pos);
+
+ new_pos = self->n_pinned_pages;
+
+ if (!pinned)
+ new_pos--;
+
+ g_list_store_insert (self->children, new_pos, page);
+
+ g_object_unref (page);
+
+ set_n_pinned_pages (self, new_pos + (pinned ? 1 : 0));
+ set_page_pinned (page, pinned);
+
+ if (self->pages) {
+ int min = MIN (old_pos, new_pos);
+ int n_changed = MAX (old_pos, new_pos) - min + 1;
+
+ g_list_model_items_changed (G_LIST_MODEL (self->pages), min, n_changed, n_changed);
+ }
+}
+
+/**
+ * adw_tab_view_get_page:
+ * @self: a #AdwTabView
+ * @child: a child in @self
+ *
+ * Gets the #AdwTabPage object representing @child.
+ *
+ * Returns: (transfer none): the #AdwTabPage representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_get_page (AdwTabView *self,
+ GtkWidget *child)
+{
+ int i;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+ g_return_val_if_fail (gtk_widget_get_parent (child) == GTK_WIDGET (self->stack), NULL);
+
+ for (i = 0; i < self->n_pages; i++) {
+ AdwTabPage *page = adw_tab_view_get_nth_page (self, i);
+
+ if (adw_tab_page_get_child (page) == child)
+ return page;
+ }
+
+ g_assert_not_reached ();
+}
+
+/**
+ * adw_tab_view_get_nth_page:
+ * @self: a #AdwTabView
+ * @position: the index of the page in @self, starting from 0
+ *
+ * Gets the #AdwTabPage representing the child at @position.
+ *
+ * Returns: (transfer none): the page object at @position
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_get_nth_page (AdwTabView *self,
+ int position)
+{
+ g_autoptr (AdwTabPage) page = NULL;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (position >= 0, NULL);
+ g_return_val_if_fail (position < self->n_pages, NULL);
+
+ page = g_list_model_get_item (G_LIST_MODEL (self->children), (guint) position);
+
+ return page;
+}
+
+/**
+ * adw_tab_view_get_page_position:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Finds the position of @page in @self, starting from 0.
+ *
+ * Returns: the position of @page in @self
+ *
+ * Since: 1.0
+ */
+int
+adw_tab_view_get_page_position (AdwTabView *self,
+ AdwTabPage *page)
+{
+ int i;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), -1);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), -1);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), -1);
+
+ for (i = 0; i < self->n_pages; i++) {
+ AdwTabPage *p = adw_tab_view_get_nth_page (self, i);
+
+ if (page == p)
+ return i;
+ }
+
+ g_assert_not_reached ();
+}
+
+/**
+ * adw_tab_view_add_page:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ * @parent: (nullable): a parent page for @child, or %NULL
+ *
+ * Adds @child to @self with @parent as the parent.
+ *
+ * This function can be used to automatically position new pages, and to select
+ * the correct page when this page is closed while being selected (see
+ * adw_tab_view_close_page()).
+ *
+ * If @parent is %NULL, this function is equivalent to adw_tab_view_append().
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_add_page (AdwTabView *self,
+ GtkWidget *child,
+ AdwTabPage *parent)
+{
+ int position;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (parent) || parent == NULL, NULL);
+
+ if (parent) {
+ AdwTabPage *page;
+
+ g_return_val_if_fail (page_belongs_to_this_view (self, parent), NULL);
+
+ if (adw_tab_page_get_pinned (parent))
+ position = self->n_pinned_pages - 1;
+ else
+ position = adw_tab_view_get_page_position (self, parent);
+
+ do {
+ position++;
+
+ if (position >= self->n_pages)
+ break;
+
+ page = adw_tab_view_get_nth_page (self, position);
+ } while (is_descendant_of (page, parent));
+ } else {
+ position = self->n_pages;
+ }
+
+ return insert_page (self, child, parent, position, FALSE);
+}
+
+/**
+ * adw_tab_view_insert:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ * @position: the position to add @child at, starting from 0
+ *
+ * Inserts a non-pinned page at @position.
+ *
+ * It's an error to try to insert a page before a pinned page, in that case
+ * adw_tab_view_insert_pinned() should be used instead.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_insert (AdwTabView *self,
+ GtkWidget *child,
+ int position)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+ g_return_val_if_fail (position >= self->n_pinned_pages, NULL);
+ g_return_val_if_fail (position <= self->n_pages, NULL);
+
+ return insert_page (self, child, NULL, position, FALSE);
+}
+
+/**
+ * adw_tab_view_prepend:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ *
+ * Inserts @child as the first non-pinned page.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_prepend (AdwTabView *self,
+ GtkWidget *child)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+
+ return insert_page (self, child, NULL, self->n_pinned_pages, FALSE);
+}
+
+/**
+ * adw_tab_view_append:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ *
+ * Inserts @child as the last non-pinned page.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_append (AdwTabView *self,
+ GtkWidget *child)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+
+ return insert_page (self, child, NULL, self->n_pages, FALSE);
+}
+
+/**
+ * adw_tab_view_insert_pinned:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ * @position: the position to add @child at, starting from 0
+ *
+ * Inserts a pinned page at @position.
+ *
+ * It's an error to try to insert a pinned page after a non-pinned page, in
+ * that case adw_tab_view_insert() should be used instead.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_insert_pinned (AdwTabView *self,
+ GtkWidget *child,
+ int position)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+ g_return_val_if_fail (position >= 0, NULL);
+ g_return_val_if_fail (position <= self->n_pinned_pages, NULL);
+
+ return insert_page (self, child, NULL, position, TRUE);
+}
+
+/**
+ * adw_tab_view_prepend_pinned:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ *
+ * Inserts @child as the first pinned page.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_prepend_pinned (AdwTabView *self,
+ GtkWidget *child)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+
+ return insert_page (self, child, NULL, 0, TRUE);
+}
+
+/**
+ * adw_tab_view_append_pinned:
+ * @self: a #AdwTabView
+ * @child: a widget to add
+ *
+ * Inserts @child as the last pinned page.
+ *
+ * Returns: (transfer none): the page object representing @child
+ *
+ * Since: 1.0
+ */
+AdwTabPage *
+adw_tab_view_append_pinned (AdwTabView *self,
+ GtkWidget *child)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+ g_return_val_if_fail (GTK_IS_WIDGET (child), NULL);
+
+ return insert_page (self, child, NULL, self->n_pinned_pages, TRUE);
+}
+
+/**
+ * adw_tab_view_close_page:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Requests to close @page.
+ *
+ * Calling this function will result in #AdwTabView::close-page signal being
+ * emitted for @page. Closing the page can then be confirmed or denied via
+ * adw_tab_view_close_page_finish().
+ *
+ * If the page is waiting for a adw_tab_view_close_page_finish() call, this
+ * function will do nothing.
+ *
+ * The default handler for #AdwTabView::close-page will immediately confirm
+ * closing the page if it's non-pinned, or reject it if it's pinned. This
+ * behavior can be changed by registering your own handler for that signal.
+ *
+ * If @page was selected, another page will be selected instead:
+ *
+ * If the #AdwTabPage:parent value is %NULL, the next page will be selected when
+ * possible, or if the page was already last, the previous page will be selected
+ * instead.
+ *
+ * If it's not %NULL, the previous page will be selected if it's a
+ * descendant (possibly indirect) of the parent. If both the previous page and
+ * the parent are pinned, the parent will be selected instead.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_close_page (AdwTabView *self,
+ AdwTabPage *page)
+{
+ gboolean ret;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ if (page->closing)
+ return;
+
+ page->closing = TRUE;
+ g_signal_emit (self, signals[SIGNAL_CLOSE_PAGE], 0, page, &ret);
+}
+
+/**
+ * adw_tab_view_close_page_finish:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @confirm: whether to confirm or deny closing @page
+ *
+ * Completes a adw_tab_view_close_page() call for @page.
+ *
+ * If @confirm is %TRUE, @page will be closed. If it's %FALSE, ite will be
+ * reverted to its previous state and adw_tab_view_close_page() can be called
+ * for it again.
+ *
+ * This function should not be called unless a custom handler for
+ * #AdwTabView::close-page is used.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_close_page_finish (AdwTabView *self,
+ AdwTabPage *page,
+ gboolean confirm)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+ g_return_if_fail (page->closing);
+
+ page->closing = FALSE;
+
+ if (confirm)
+ detach_page (self, page, TRUE);
+}
+
+/**
+ * adw_tab_view_close_other_pages:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Requests to close all pages other than @page.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_close_other_pages (AdwTabView *self,
+ AdwTabPage *page)
+{
+ int i;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ for (i = self->n_pages - 1; i >= 0; i--) {
+ AdwTabPage *p = adw_tab_view_get_nth_page (self, i);
+
+ if (p == page)
+ continue;
+
+ adw_tab_view_close_page (self, p);
+ }
+}
+
+/**
+ * adw_tab_view_close_pages_before:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Requests to close all pages before @page.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_close_pages_before (AdwTabView *self,
+ AdwTabPage *page)
+{
+ int pos, i;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ pos = adw_tab_view_get_page_position (self, page);
+
+ for (i = pos - 1; i >= 0; i--) {
+ AdwTabPage *p = adw_tab_view_get_nth_page (self, i);
+
+ adw_tab_view_close_page (self, p);
+ }
+}
+
+/**
+ * adw_tab_view_close_pages_after:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Requests to close all pages after @page.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_close_pages_after (AdwTabView *self,
+ AdwTabPage *page)
+{
+ int pos, i;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ pos = adw_tab_view_get_page_position (self, page);
+
+ for (i = self->n_pages - 1; i > pos; i--) {
+ AdwTabPage *p = adw_tab_view_get_nth_page (self, i);
+
+ adw_tab_view_close_page (self, p);
+ }
+}
+
+/**
+ * adw_tab_view_reorder_page:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @position: the position to insert the page at, starting at 0
+ *
+ * Reorders @page to @position.
+ *
+ * It's a programmer error to try to reorder a pinned page after a non-pinned
+ * one, or a non-pinned page before a pinned one.
+ *
+ * Returns: %TRUE if @page was moved, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_reorder_page (AdwTabView *self,
+ AdwTabPage *page,
+ int position)
+{
+ int original_pos;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), FALSE);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), FALSE);
+
+ if (adw_tab_page_get_pinned (page)) {
+ g_return_val_if_fail (position >= 0, FALSE);
+ g_return_val_if_fail (position < self->n_pinned_pages, FALSE);
+ } else {
+ g_return_val_if_fail (position >= self->n_pinned_pages, FALSE);
+ g_return_val_if_fail (position < self->n_pages, FALSE);
+ }
+
+ original_pos = adw_tab_view_get_page_position (self, page);
+
+ if (original_pos == position)
+ return FALSE;
+
+ g_object_ref (page);
+
+ g_list_store_remove (self->children, original_pos);
+ g_list_store_insert (self->children, position, page);
+
+ g_object_unref (page);
+
+ g_signal_emit (self, signals[SIGNAL_PAGE_REORDERED], 0, page, position);
+
+ if (self->pages) {
+ int min = MIN (original_pos, position);
+ int n_changed = MAX (original_pos, position) - min + 1;
+
+ g_list_model_items_changed (G_LIST_MODEL (self->pages), min, n_changed, n_changed);
+ }
+
+ return TRUE;
+}
+
+/**
+ * adw_tab_view_reorder_backward:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Reorders @page to before its previous page if possible.
+ *
+ * Returns: %TRUE if @page was moved, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_reorder_backward (AdwTabView *self,
+ AdwTabPage *page)
+{
+ gboolean pinned;
+ int pos, first;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), FALSE);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), FALSE);
+
+ pos = adw_tab_view_get_page_position (self, page);
+
+ pinned = adw_tab_page_get_pinned (page);
+ first = pinned ? 0 : self->n_pinned_pages;
+
+ if (pos <= first)
+ return FALSE;
+
+ return adw_tab_view_reorder_page (self, page, pos - 1);
+}
+
+/**
+ * adw_tab_view_reorder_forward:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Reorders @page to after its next page if possible.
+ *
+ * Returns: %TRUE if @page was moved, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_reorder_forward (AdwTabView *self,
+ AdwTabPage *page)
+{
+ gboolean pinned;
+ int pos, last;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), FALSE);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), FALSE);
+
+ pos = adw_tab_view_get_page_position (self, page);
+
+ pinned = adw_tab_page_get_pinned (page);
+ last = (pinned ? self->n_pinned_pages : self->n_pages) - 1;
+
+ if (pos >= last)
+ return FALSE;
+
+ return adw_tab_view_reorder_page (self, page, pos + 1);
+}
+
+/**
+ * adw_tab_view_reorder_first:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Reorders @page to the first possible position.
+ *
+ * Returns: %TRUE if @page was moved, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_reorder_first (AdwTabView *self,
+ AdwTabPage *page)
+{
+ gboolean pinned;
+ int pos;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), FALSE);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), FALSE);
+
+ pinned = adw_tab_page_get_pinned (page);
+ pos = pinned ? 0 : self->n_pinned_pages;
+
+ return adw_tab_view_reorder_page (self, page, pos);
+}
+
+/**
+ * adw_tab_view_reorder_last:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ *
+ * Reorders @page to the last possible position.
+ *
+ * Returns: %TRUE if @page was moved, %FALSE otherwise
+ *
+ * Since: 1.0
+ */
+gboolean
+adw_tab_view_reorder_last (AdwTabView *self,
+ AdwTabPage *page)
+{
+ gboolean pinned;
+ int pos;
+
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), FALSE);
+ g_return_val_if_fail (ADW_IS_TAB_PAGE (page), FALSE);
+ g_return_val_if_fail (page_belongs_to_this_view (self, page), FALSE);
+
+ pinned = adw_tab_page_get_pinned (page);
+ pos = (pinned ? self->n_pinned_pages : self->n_pages) - 1;
+
+ return adw_tab_view_reorder_page (self, page, pos);
+}
+
+void
+adw_tab_view_detach_page (AdwTabView *self,
+ AdwTabPage *page)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+
+ g_object_ref (page);
+
+ begin_transfer_for_group (self);
+
+ detach_page (self, page, TRUE);
+}
+
+void
+adw_tab_view_attach_page (AdwTabView *self,
+ AdwTabPage *page,
+ int position)
+{
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (!page_belongs_to_this_view (self, page));
+ g_return_if_fail (position >= 0);
+ g_return_if_fail (position <= self->n_pages);
+
+ attach_page (self, page, position);
+
+ adw_tab_view_set_selected_page (self, page);
+
+ end_transfer_for_group (self);
+
+ g_object_unref (page);
+}
+
+/**
+ * adw_tab_view_transfer_page:
+ * @self: a #AdwTabView
+ * @page: a page of @self
+ * @other_view: the tab view to transfer the page to
+ * @position: the position to insert the page at, starting at 0
+ *
+ * Transfers @page from @self to @other_view. The @page object will be reused.
+ *
+ * It's a programmer error to try to insert a pinned page after a non-pinned
+ * one, or a non-pinned page before a pinned one.
+ *
+ * Since: 1.0
+ */
+void
+adw_tab_view_transfer_page (AdwTabView *self,
+ AdwTabPage *page,
+ AdwTabView *other_view,
+ int position)
+{
+ gboolean pinned;
+
+ g_return_if_fail (ADW_IS_TAB_VIEW (self));
+ g_return_if_fail (ADW_IS_TAB_PAGE (page));
+ g_return_if_fail (ADW_IS_TAB_VIEW (other_view));
+ g_return_if_fail (page_belongs_to_this_view (self, page));
+ g_return_if_fail (position >= 0);
+ g_return_if_fail (position <= other_view->n_pages);
+
+ pinned = adw_tab_page_get_pinned (page);
+
+ g_return_if_fail (!pinned || position <= other_view->n_pinned_pages);
+ g_return_if_fail (pinned || position >= other_view->n_pinned_pages);
+
+ adw_tab_view_detach_page (self, page);
+ adw_tab_view_attach_page (other_view, page, position);
+}
+
+/**
+ * adw_tab_view_get_pages:
+ * @self: a #AdwTabView
+ *
+ * Returns a #GListModel that contains the pages of the tab view, and can be
+ * used to keep an up-to-date view. The model also implements #GtkSelectionModel
+ * and can be used to track the selected page.
+ *
+ * Returns: (transfer full): a #GtkSelectionModel for the pages of @self
+ *
+ * Since: 1.0
+ */
+GtkSelectionModel *
+adw_tab_view_get_pages (AdwTabView *self)
+{
+ g_return_val_if_fail (ADW_IS_TAB_VIEW (self), NULL);
+
+ if (self->pages)
+ return g_object_ref (self->pages);
+
+ self->pages = adw_tab_pages_new (self);
+ g_object_add_weak_pointer (G_OBJECT (self->pages),
+ (gpointer *) &self->pages);
+
+ return self->pages;
+}
+
+AdwTabView *
+adw_tab_view_create_window (AdwTabView *self)
+{
+ AdwTabView *new_view = NULL;
+
+ g_signal_emit (self, signals[SIGNAL_CREATE_WINDOW], 0, &new_view);
+
+ if (!new_view) {
+ g_critical ("AdwTabView::create-window handler must not return NULL");
+
+ return NULL;
+ }
+
+ new_view->transfer_count = self->transfer_count;
+
+ return new_view;
+}
diff --git a/src/adw-tab-view.h b/src/adw-tab-view.h
new file mode 100644
index 00000000..2268dce9
--- /dev/null
+++ b/src/adw-tab-view.h
@@ -0,0 +1,213 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Author: Alexander Mikhaylenko <alexander mikhaylenko puri sm>
+ */
+
+#pragma once
+
+#if !defined(_ADWAITA_INSIDE) && !defined(ADWAITA_COMPILATION)
+#error "Only <adwaita.h> can be included directly."
+#endif
+
+#include "adw-version.h"
+
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define ADW_TYPE_TAB_PAGE (adw_tab_page_get_type())
+
+ADW_AVAILABLE_IN_ALL
+G_DECLARE_FINAL_TYPE (AdwTabPage, adw_tab_page, ADW, TAB_PAGE, GObject)
+
+ADW_AVAILABLE_IN_ALL
+GtkWidget *adw_tab_page_get_child (AdwTabPage *self);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_page_get_parent (AdwTabPage *self);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_page_get_selected (AdwTabPage *self);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_page_get_pinned (AdwTabPage *self);
+
+ADW_AVAILABLE_IN_ALL
+const char *adw_tab_page_get_title (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_title (AdwTabPage *self,
+ const char *title);
+
+ADW_AVAILABLE_IN_ALL
+const char *adw_tab_page_get_tooltip (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_tooltip (AdwTabPage *self,
+ const char *tooltip);
+
+ADW_AVAILABLE_IN_ALL
+GIcon *adw_tab_page_get_icon (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_icon (AdwTabPage *self,
+ GIcon *icon);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_page_get_loading (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_loading (AdwTabPage *self,
+ gboolean loading);
+
+ADW_AVAILABLE_IN_ALL
+GIcon *adw_tab_page_get_indicator_icon (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_indicator_icon (AdwTabPage *self,
+ GIcon *indicator_icon);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_page_get_indicator_activatable (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_indicator_activatable (AdwTabPage *self,
+ gboolean activatable);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_page_get_needs_attention (AdwTabPage *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_page_set_needs_attention (AdwTabPage *self,
+ gboolean needs_attention);
+
+#define ADW_TYPE_TAB_VIEW (adw_tab_view_get_type())
+
+ADW_AVAILABLE_IN_ALL
+G_DECLARE_FINAL_TYPE (AdwTabView, adw_tab_view, ADW, TAB_VIEW, GtkWidget)
+
+ADW_AVAILABLE_IN_ALL
+AdwTabView *adw_tab_view_new (void);
+
+ADW_AVAILABLE_IN_ALL
+int adw_tab_view_get_n_pages (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+int adw_tab_view_get_n_pinned_pages (AdwTabView *self);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_get_is_transferring_page (AdwTabView *self);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_get_selected_page (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_set_selected_page (AdwTabView *self,
+ AdwTabPage *selected_page);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_select_previous_page (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_select_next_page (AdwTabView *self);
+
+ADW_AVAILABLE_IN_ALL
+GIcon *adw_tab_view_get_default_icon (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_set_default_icon (AdwTabView *self,
+ GIcon *default_icon);
+
+ADW_AVAILABLE_IN_ALL
+GMenuModel *adw_tab_view_get_menu_model (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_set_menu_model (AdwTabView *self,
+ GMenuModel *menu_model);
+
+ADW_AVAILABLE_IN_ALL
+GtkWidget *adw_tab_view_get_shortcut_widget (AdwTabView *self);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_set_shortcut_widget (AdwTabView *self,
+ GtkWidget *widget);
+
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_set_page_pinned (AdwTabView *self,
+ AdwTabPage *page,
+ gboolean pinned);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_get_page (AdwTabView *self,
+ GtkWidget *child);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_get_nth_page (AdwTabView *self,
+ int position);
+
+ADW_AVAILABLE_IN_ALL
+int adw_tab_view_get_page_position (AdwTabView *self,
+ AdwTabPage *page);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_add_page (AdwTabView *self,
+ GtkWidget *child,
+ AdwTabPage *parent);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_insert (AdwTabView *self,
+ GtkWidget *child,
+ int position);
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_prepend (AdwTabView *self,
+ GtkWidget *child);
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_append (AdwTabView *self,
+ GtkWidget *child);
+
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_insert_pinned (AdwTabView *self,
+ GtkWidget *child,
+ int position);
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_prepend_pinned (AdwTabView *self,
+ GtkWidget *child);
+ADW_AVAILABLE_IN_ALL
+AdwTabPage *adw_tab_view_append_pinned (AdwTabView *self,
+ GtkWidget *child);
+
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_close_page (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_close_page_finish (AdwTabView *self,
+ AdwTabPage *page,
+ gboolean confirm);
+
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_close_other_pages (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_close_pages_before (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_close_pages_after (AdwTabView *self,
+ AdwTabPage *page);
+
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_reorder_page (AdwTabView *self,
+ AdwTabPage *page,
+ int position);
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_reorder_backward (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_reorder_forward (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_reorder_first (AdwTabView *self,
+ AdwTabPage *page);
+ADW_AVAILABLE_IN_ALL
+gboolean adw_tab_view_reorder_last (AdwTabView *self,
+ AdwTabPage *page);
+
+ADW_AVAILABLE_IN_ALL
+void adw_tab_view_transfer_page (AdwTabView *self,
+ AdwTabPage *page,
+ AdwTabView *other_view,
+ int position);
+
+ADW_AVAILABLE_IN_ALL
+GtkSelectionModel *adw_tab_view_get_pages (AdwTabView *self);
+
+G_END_DECLS
diff --git a/src/adwaita.gresources.xml b/src/adwaita.gresources.xml
index 1bee5f86..c1de08b3 100644
--- a/src/adwaita.gresources.xml
+++ b/src/adwaita.gresources.xml
@@ -5,6 +5,7 @@
<file>glsl/mask.glsl</file>
<file preprocess="xml-stripblanks">icons/scalable/actions/adw-expander-arrow-symbolic.svg</file>
<file preprocess="xml-stripblanks">icons/scalable/status/avatar-default-symbolic.svg</file>
+ <file preprocess="xml-stripblanks">icons/scalable/status/adw-tab-icon-missing-symbolic.svg</file>
</gresource>
<gresource prefix="/org/gnome/Adwaita/ui">
<file preprocess="xml-stripblanks">adw-action-row.ui</file>
diff --git a/src/adwaita.h b/src/adwaita.h
index 86fcd235..b5ccc780 100644
--- a/src/adwaita.h
+++ b/src/adwaita.h
@@ -50,6 +50,7 @@ G_BEGIN_DECLS
#include "adw-status-page.h"
#include "adw-swipe-tracker.h"
#include "adw-swipeable.h"
+#include "adw-tab-view.h"
#include "adw-value-object.h"
#include "adw-view-switcher.h"
#include "adw-view-switcher-bar.h"
diff --git a/src/icons/scalable/status/adw-tab-icon-missing-symbolic.svg
b/src/icons/scalable/status/adw-tab-icon-missing-symbolic.svg
new file mode 100644
index 00000000..48807915
--- /dev/null
+++ b/src/icons/scalable/status/adw-tab-icon-missing-symbolic.svg
@@ -0,0 +1,32 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+ xmlns:dc="http://purl.org/dc/elements/1.1/"
+ xmlns:cc="http://creativecommons.org/ns#"
+ xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+ xmlns:svg="http://www.w3.org/2000/svg"
+ xmlns="http://www.w3.org/2000/svg"
+ height="16"
+ id="svg7384"
+ version="1.1"
+ width="16">
+ <metadata
+ id="metadata90">
+ <rdf:RDF>
+ <cc:Work
+ rdf:about="">
+ <dc:format>image/svg+xml</dc:format>
+ <dc:type
+ rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+ <dc:title>Gnome Symbolic Icon Theme</dc:title>
+ </cc:Work>
+ </rdf:RDF>
+ </metadata>
+ <title
+ id="title9167">Gnome Symbolic Icon Theme</title>
+ <defs
+ id="defs7386" />
+ <path
+
style="color:#bebebe;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:medium;line-height:normal;font-family:sans-serif;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration:none;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000000;letter-spacing:normal;word-spacing:normal;text-transform:none;writing-mode:lr-tb;direction:ltr;text-orientation:mixed;dominant-baseline:auto;baseline-shift:baseline;text-anchor:start;white-space:normal;shape-padding:0;clip-rule:nonzero;display:inline;overflow:visible;visibility:visible;opacity:1;isolation:auto;mix-blend-mode:normal;color-interpolation:sRGB;color-interpolation-filters:linearRGB;solid-color:#000000;solid-opacity:1;vector-effect:none;fill:#241f31;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:2;st
roke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;color-rendering:auto;image-rendering:auto;shape-rendering:auto;text-rendering:auto;enable-background:accumulate"
+ d="M 2.1992188 0 C 0.94446875 0 -2.9605947e-16 1.0997689 0 2.3183594 L 0 4 L 2 4 L 2 2.3183594 C 2
2.0752625 2.1256288 2 2.1992188 2 L 4 2 L 4 0 L 2.1992188 0 z M 6 0 L 6 2 L 10 2 L 10 0 L 6 0 z M 12 0 L 12 2
L 13.800781 2 C 13.874371 2 14 2.0752716 14 2.3183594 L 14 4 L 16 4 L 16 2.3183594 C 16 1.0997598 15.055531
-2.9605947e-16 13.800781 0 L 12 0 z M 0 6 L 0 10 L 2 10 L 2 6 L 0 6 z M 14 6 L 14 10 L 16 10 L 16 6 L 14 6 z
M 0 12 L 0 13.681641 C 0 14.900231 0.94447875 16 2.1992188 16 L 4 16 L 4 14 L 2.1992188 14 C 2.1256188 14 2
13.924738 2 13.681641 L 2 12 L 0 12 z M 14 12 L 14 13.681641 C 14 13.924729 13.874381 14 13.800781 14 L 12 14
L 12 16 L 13.800781 16 C 15.055521 16 16 14.90024 16 13.681641 L 16 12 L 14 12 z M 6 14 L 6 16 L 10 16 L 10
14 L 6 14 z "
+ id="path983" />
+</svg>
diff --git a/src/meson.build b/src/meson.build
index 8390098c..53ac3437 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -93,6 +93,7 @@ src_headers = [
'adw-status-page.h',
'adw-swipe-tracker.h',
'adw-swipeable.h',
+ 'adw-tab-view.h',
'adw-value-object.h',
'adw-view-switcher.h',
'adw-view-switcher-bar.h',
@@ -147,6 +148,7 @@ src_sources = [
'adw-status-page.c',
'adw-swipe-tracker.c',
'adw-swipeable.c',
+ 'adw-tab-view.c',
'adw-value-object.c',
'adw-view-switcher.c',
'adw-view-switcher-bar.c',
diff --git a/tests/meson.build b/tests/meson.build
index 0f0a29fc..d75f86ac 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -38,6 +38,7 @@ test_names = [
'test-preferences-window',
'test-squeezer',
'test-status-page',
+ 'test-tab-view',
'test-value-object',
'test-view-switcher',
'test-view-switcher-bar',
diff --git a/tests/test-tab-view.c b/tests/test-tab-view.c
new file mode 100644
index 00000000..4368c076
--- /dev/null
+++ b/tests/test-tab-view.c
@@ -0,0 +1,1151 @@
+/*
+ * Copyright (C) 2020 Purism SPC
+ *
+ * SPDX-License-Identifier: LGPL-2.1+
+ *
+ * Author: Alexander Mikhaylenko <alexander mikhaylenko puri sm>
+ */
+
+#include <adwaita.h>
+
+int notified;
+
+static void
+notify_cb (GtkWidget *widget, gpointer data)
+{
+ notified++;
+}
+
+static void
+add_pages (AdwTabView *view,
+ AdwTabPage **pages,
+ int n,
+ int n_pinned)
+{
+ int i;
+
+ for (i = 0; i < n_pinned; i++)
+ pages[i] = adw_tab_view_append_pinned (view, gtk_button_new ());
+
+ for (i = n_pinned; i < n; i++)
+ pages[i] = adw_tab_view_append (view, gtk_button_new ());
+}
+
+static void
+assert_page_positions (AdwTabView *view,
+ AdwTabPage **pages,
+ int n,
+ int n_pinned,
+ ...)
+{
+ va_list args;
+ int i;
+
+ va_start (args, n_pinned);
+
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, n);
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, n_pinned);
+
+ for (i = 0; i < n; i++) {
+ int index = va_arg (args, int);
+
+ if (index >= 0)
+ g_assert_cmpint (adw_tab_view_get_page_position (view, pages[index]), ==, i);
+ }
+
+ va_end (args);
+}
+
+static gboolean
+close_noop (void)
+{
+ return GDK_EVENT_STOP;
+}
+
+static void
+test_adw_tab_view_n_pages (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ int n_pages;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::n-pages", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "n-pages", &n_pages, NULL);
+ g_assert_cmpint (n_pages, ==, 0);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_object_get (view, "n-pages", &n_pages, NULL);
+ g_assert_cmpint (n_pages, ==, 1);
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, 1);
+ g_assert_cmpint (notified, ==, 1);
+
+ adw_tab_view_append (view, gtk_button_new ());
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, 2);
+ g_assert_cmpint (notified, ==, 2);
+
+ adw_tab_view_append_pinned (view, gtk_button_new ());
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, 3);
+ g_assert_cmpint (notified, ==, 3);
+
+ adw_tab_view_reorder_forward (view, page);
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, 3);
+ g_assert_cmpint (notified, ==, 3);
+
+ adw_tab_view_close_page (view, page);
+ g_assert_cmpint (adw_tab_view_get_n_pages (view), ==, 2);
+ g_assert_cmpint (notified, ==, 4);
+}
+
+static void
+test_adw_tab_view_n_pinned_pages (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ int n_pages;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::n-pinned-pages", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "n-pinned-pages", &n_pages, NULL);
+ g_assert_cmpint (n_pages, ==, 0);
+
+ adw_tab_view_append_pinned (view, gtk_button_new ());
+ g_object_get (view, "n-pinned-pages", &n_pages, NULL);
+ g_assert_cmpint (n_pages, ==, 1);
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, 1);
+ g_assert_cmpint (notified, ==, 1);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, 1);
+ g_assert_cmpint (notified, ==, 1);
+
+ adw_tab_view_set_page_pinned (view, page, TRUE);
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, 2);
+ g_assert_cmpint (notified, ==, 2);
+
+ adw_tab_view_reorder_backward (view, page);
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, 2);
+ g_assert_cmpint (notified, ==, 2);
+
+ adw_tab_view_set_page_pinned (view, page, FALSE);
+ g_assert_cmpint (adw_tab_view_get_n_pinned_pages (view), ==, 1);
+ g_assert_cmpint (notified, ==, 3);
+}
+
+static void
+test_adw_tab_view_default_icon (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ GIcon *icon = NULL;
+ g_autoptr (GIcon) icon1 = g_themed_icon_new ("go-previous-symbolic");
+ g_autoptr (GIcon) icon2 = g_themed_icon_new ("go-next-symbolic");
+ g_autofree char *icon_str = NULL;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::default-icon", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "default-icon", &icon, NULL);
+ icon_str = g_icon_to_string (icon);
+ g_assert_cmpstr (icon_str, ==, "adw-tab-icon-missing-symbolic");
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_view_set_default_icon (view, icon1);
+ g_object_get (view, "default-icon", &icon, NULL);
+ g_assert_true (icon == icon1);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (view, "default-icon", icon2, NULL);
+ g_assert_true (adw_tab_view_get_default_icon (view) == icon2);
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_view_menu_model (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ GMenuModel *model = NULL;
+ g_autoptr (GMenuModel) model1 = G_MENU_MODEL (g_menu_new ());
+ g_autoptr (GMenuModel) model2 = G_MENU_MODEL (g_menu_new ());
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::menu-model", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "menu-model", &model, NULL);
+ g_assert_null (model);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_view_set_menu_model (view, model1);
+ g_object_get (view, "menu-model", &model, NULL);
+ g_assert_true (model == model1);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (view, "menu-model", model2, NULL);
+ g_assert_true (adw_tab_view_get_menu_model (view) == model2);
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_view_shortcut_widget (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ GtkWidget *widget = NULL;
+ g_autoptr (GtkWidget) widget1 = g_object_ref_sink (gtk_button_new ());
+ g_autoptr (GtkWidget) widget2 = g_object_ref_sink (gtk_button_new ());
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::shortcut-widget", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "shortcut-widget", &widget, NULL);
+ g_assert_null (widget);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_view_set_shortcut_widget (view, widget1);
+ g_object_get (view, "shortcut-widget", &widget, NULL);
+ g_assert_true (widget == widget1);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (view, "shortcut-widget", widget2, NULL);
+ g_assert_true (adw_tab_view_get_shortcut_widget (view) == widget2);
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_view_pages (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ GtkWidget *child1, *child2, *child3;
+ AdwTabPage *page1, *page2, *page3;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ child1 = gtk_button_new ();
+ child2 = gtk_button_new ();
+ child3 = gtk_button_new ();
+
+ page1 = adw_tab_view_append_pinned (view, child1);
+ page2 = adw_tab_view_append (view, child2);
+ page3 = adw_tab_view_append (view, child3);
+
+ g_assert_true (adw_tab_view_get_nth_page (view, 0) == page1);
+ g_assert_true (adw_tab_view_get_nth_page (view, 1) == page2);
+ g_assert_true (adw_tab_view_get_nth_page (view, 2) == page3);
+
+ g_assert_true (adw_tab_view_get_page (view, child1) == page1);
+ g_assert_true (adw_tab_view_get_page (view, child2) == page2);
+ g_assert_true (adw_tab_view_get_page (view, child3) == page3);
+
+ g_assert_cmpint (adw_tab_view_get_page_position (view, page1), ==, 0);
+ g_assert_cmpint (adw_tab_view_get_page_position (view, page2), ==, 1);
+ g_assert_cmpint (adw_tab_view_get_page_position (view, page3), ==, 2);
+
+ g_assert_true (adw_tab_page_get_child (page1) == child1);
+ g_assert_true (adw_tab_page_get_child (page2) == child2);
+ g_assert_true (adw_tab_page_get_child (page3) == child3);
+}
+
+static void
+test_adw_tab_view_select (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page1, *page2, *selected_page;
+ gboolean ret;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ notified = 0;
+ g_signal_connect (view, "notify::selected-page", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (view, "selected-page", &selected_page, NULL);
+ g_assert_null (selected_page);
+
+ page1 = adw_tab_view_append (view, gtk_button_new ());
+ g_object_get (view, "selected-page", &selected_page, NULL);
+ g_assert_true (selected_page == page1);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page1);
+ g_assert_true (adw_tab_page_get_selected (page1));
+ g_assert_cmpint (notified, ==, 1);
+
+ page2 = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_true (adw_tab_view_get_selected_page (view) == page1);
+ g_assert_true (adw_tab_page_get_selected (page1));
+ g_assert_false (adw_tab_page_get_selected (page2));
+ g_assert_cmpint (notified, ==, 1);
+
+ adw_tab_view_set_selected_page (view, page2);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page2);
+ g_assert_cmpint (notified, ==, 2);
+
+ g_object_set (view, "selected-page", page1, NULL);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page1);
+ g_assert_cmpint (notified, ==, 3);
+
+ ret = adw_tab_view_select_previous_page (view);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page1);
+ g_assert_false (ret);
+ g_assert_cmpint (notified, ==, 3);
+
+ ret = adw_tab_view_select_next_page (view);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page2);
+ g_assert_true (ret);
+ g_assert_cmpint (notified, ==, 4);
+
+ ret = adw_tab_view_select_next_page (view);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page2);
+ g_assert_false (ret);
+ g_assert_cmpint (notified, ==, 4);
+
+ ret = adw_tab_view_select_previous_page (view);
+ g_assert_true (adw_tab_view_get_selected_page (view) == page1);
+ g_assert_true (ret);
+ g_assert_cmpint (notified, ==, 5);
+}
+
+static void
+test_adw_tab_view_add_basic (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[6];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ pages[0] = adw_tab_view_append (view, gtk_button_new ());
+ assert_page_positions (view, pages, 1, 0,
+ 0);
+
+ pages[1] = adw_tab_view_prepend (view, gtk_button_new ());
+ assert_page_positions (view, pages, 2, 0,
+ 1, 0);
+
+ pages[2] = adw_tab_view_insert (view, gtk_button_new (), 1);
+ assert_page_positions (view, pages, 3, 0,
+ 1, 2, 0);
+
+ pages[3] = adw_tab_view_prepend_pinned (view, gtk_button_new ());
+ assert_page_positions (view, pages, 4, 1,
+ 3, 1, 2, 0);
+
+ pages[4] = adw_tab_view_append_pinned (view, gtk_button_new ());
+ assert_page_positions (view, pages, 5, 2,
+ 3, 4, 1, 2, 0);
+
+ pages[5] = adw_tab_view_insert_pinned (view, gtk_button_new (), 1);
+ assert_page_positions (view, pages, 6, 3,
+ 3, 5, 4, 1, 2, 0);
+}
+
+static void
+test_adw_tab_view_add_auto (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[17];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 3, 3);
+ assert_page_positions (view, pages, 3, 3,
+ 0, 1, 2);
+
+ /* No parent */
+
+ pages[3] = adw_tab_view_add_page (view, gtk_button_new (), NULL);
+ g_assert_null (adw_tab_page_get_parent (pages[3]));
+ assert_page_positions (view, pages, 4, 3,
+ 0, 1, 2, 3);
+
+ pages[4] = adw_tab_view_add_page (view, gtk_button_new (), NULL);
+ g_assert_null (adw_tab_page_get_parent (pages[4]));
+ assert_page_positions (view, pages, 5, 3,
+ 0, 1, 2, 3, 4);
+
+ pages[5] = adw_tab_view_add_page (view, gtk_button_new (), NULL);
+ g_assert_null (adw_tab_page_get_parent (pages[5]));
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ /* Parent is a regular page */
+
+ pages[6] = adw_tab_view_add_page (view, gtk_button_new (), pages[4]);
+ g_assert_true (adw_tab_page_get_parent (pages[6]) == pages[4]);
+ assert_page_positions (view, pages, 7, 3,
+ 0, 1, 2, 3, 4, 6, 5);
+
+ pages[7] = adw_tab_view_add_page (view, gtk_button_new (), pages[4]);
+ g_assert_true (adw_tab_page_get_parent (pages[7]) == pages[4]);
+ assert_page_positions (view, pages, 8, 3,
+ 0, 1, 2, 3, 4, 6, 7, 5);
+
+ pages[8] = adw_tab_view_add_page (view, gtk_button_new (), pages[6]);
+ g_assert_true (adw_tab_page_get_parent (pages[8]) == pages[6]);
+ assert_page_positions (view, pages, 9, 3,
+ 0, 1, 2, 3, 4, 6, 8, 7, 5);
+
+ pages[9] = adw_tab_view_add_page (view, gtk_button_new (), pages[6]);
+ g_assert_true (adw_tab_page_get_parent (pages[9]) == pages[6]);
+ assert_page_positions (view, pages, 10, 3,
+ 0, 1, 2, 3, 4, 6, 8, 9, 7, 5);
+
+ pages[10] = adw_tab_view_add_page (view, gtk_button_new (), pages[4]);
+ g_assert_true (adw_tab_page_get_parent (pages[10]) == pages[4]);
+ assert_page_positions (view, pages, 11, 3,
+ 0, 1, 2, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ /* Parent is a pinned page */
+
+ pages[11] = adw_tab_view_add_page (view, gtk_button_new (), pages[1]);
+ g_assert_true (adw_tab_page_get_parent (pages[11]) == pages[1]);
+ assert_page_positions (view, pages, 12, 3,
+ 0, 1, 2, 11, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ pages[12] = adw_tab_view_add_page (view, gtk_button_new (), pages[11]);
+ g_assert_true (adw_tab_page_get_parent (pages[12]) == pages[11]);
+ assert_page_positions (view, pages, 13, 3,
+ 0, 1, 2, 11, 12, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ pages[13] = adw_tab_view_add_page (view, gtk_button_new (), pages[1]);
+ g_assert_true (adw_tab_page_get_parent (pages[13]) == pages[1]);
+ assert_page_positions (view, pages, 14, 3,
+ 0, 1, 2, 11, 12, 13, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ pages[14] = adw_tab_view_add_page (view, gtk_button_new (), pages[0]);
+ g_assert_true (adw_tab_page_get_parent (pages[14]) == pages[0]);
+ assert_page_positions (view, pages, 15, 3,
+ 0, 1, 2, 14, 11, 12, 13, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ pages[15] = adw_tab_view_add_page (view, gtk_button_new (), pages[1]);
+ g_assert_true (adw_tab_page_get_parent (pages[15]) == pages[1]);
+ assert_page_positions (view, pages, 16, 3,
+ 0, 1, 2, 15, 14, 11, 12, 13, 3, 4, 6, 8, 9, 7, 10, 5);
+
+ /* Parent is the last page */
+
+ pages[16] = adw_tab_view_add_page (view, gtk_button_new (), pages[5]);
+ g_assert_true (adw_tab_page_get_parent (pages[16]) == pages[5]);
+ assert_page_positions (view, pages, 17, 3,
+ 0, 1, 2, 15, 14, 11, 12, 13, 3, 4, 6, 8, 9, 7, 10, 5, 16);
+}
+
+static void
+test_adw_tab_view_reorder (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[6];
+ gboolean ret;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 6, 3);
+
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_page (view, pages[1], 1);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_page (view, pages[1], 0);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 0, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_page (view, pages[1], 1);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_page (view, pages[5], 5);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_page (view, pages[5], 4);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 5, 4);
+
+ ret = adw_tab_view_reorder_page (view, pages[5], 5);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+}
+
+static void
+test_adw_tab_view_reorder_first_last (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[6];
+ gboolean ret;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 6, 3);
+
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_first (view, pages[0]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_last (view, pages[0]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_last (view, pages[0]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_first (view, pages[0]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_first (view, pages[3]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_last (view, pages[3]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 4, 5, 3);
+
+ ret = adw_tab_view_reorder_last (view, pages[3]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 4, 5, 3);
+
+ ret = adw_tab_view_reorder_first (view, pages[3]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+}
+
+static void
+test_adw_tab_view_reorder_forward_backward (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[6];
+ gboolean ret;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 6, 3);
+
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_backward (view, pages[0]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_forward (view, pages[0]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 0, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_forward (view, pages[2]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 0, 2, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_backward (view, pages[2]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_backward (view, pages[3]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 3, 4, 5);
+
+ ret = adw_tab_view_reorder_forward (view, pages[3]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 4, 3, 5);
+
+ ret = adw_tab_view_reorder_forward (view, pages[5]);
+ g_assert_false (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 4, 3, 5);
+
+ ret = adw_tab_view_reorder_backward (view, pages[5]);
+ g_assert_true (ret);
+ assert_page_positions (view, pages, 6, 3,
+ 1, 2, 0, 4, 5, 3);
+}
+
+static void
+test_adw_tab_view_pin (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[4];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ /* Test specifically pinning with only 1 page */
+ pages[0] = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_false (adw_tab_page_get_pinned (pages[0]));
+ assert_page_positions (view, pages, 1, 0,
+ 0);
+
+ adw_tab_view_set_page_pinned (view, pages[0], TRUE);
+ g_assert_true (adw_tab_page_get_pinned (pages[0]));
+ assert_page_positions (view, pages, 1, 1,
+ 0);
+
+ adw_tab_view_set_page_pinned (view, pages[0], FALSE);
+ g_assert_false (adw_tab_page_get_pinned (pages[0]));
+ assert_page_positions (view, pages, 1, 0,
+ 0);
+
+ pages[1] = adw_tab_view_append (view, gtk_button_new ());
+ pages[2] = adw_tab_view_append (view, gtk_button_new ());
+ pages[3] = adw_tab_view_append (view, gtk_button_new ());
+ assert_page_positions (view, pages, 4, 0,
+ 0, 1, 2, 3);
+
+ adw_tab_view_set_page_pinned (view, pages[2], TRUE);
+ assert_page_positions (view, pages, 4, 1,
+ 2, 0, 1, 3);
+
+ adw_tab_view_set_page_pinned (view, pages[1], TRUE);
+ assert_page_positions (view, pages, 4, 2,
+ 2, 1, 0, 3);
+
+ adw_tab_view_set_page_pinned (view, pages[0], TRUE);
+ assert_page_positions (view, pages, 4, 3,
+ 2, 1, 0, 3);
+
+ adw_tab_view_set_page_pinned (view, pages[1], FALSE);
+ assert_page_positions (view, pages, 4, 2,
+ 2, 0, 1, 3);
+}
+
+static void
+test_adw_tab_view_close (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[3];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 3, 0);
+
+ adw_tab_view_set_selected_page (view, pages[1]);
+
+ assert_page_positions (view, pages, 3, 0,
+ 0, 1, 2);
+
+ adw_tab_view_close_page (view, pages[1]);
+ assert_page_positions (view, pages, 2, 0,
+ 0, 2);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[2]);
+
+ adw_tab_view_close_page (view, pages[2]);
+ assert_page_positions (view, pages, 1, 0,
+ 0);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[0]);
+
+ adw_tab_view_close_page (view, pages[0]);
+ assert_page_positions (view, pages, 0, 0);
+ g_assert_null (adw_tab_view_get_selected_page (view));
+}
+
+static void
+test_adw_tab_view_close_other (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[6];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 6, 3);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 3, 4, 5);
+
+ adw_tab_view_close_other_pages (view, pages[4]);
+ assert_page_positions (view, pages, 4, 3,
+ 0, 1, 2, 4);
+
+ adw_tab_view_close_other_pages (view, pages[2]);
+ assert_page_positions (view, pages, 3, 3,
+ 0, 1, 2);
+}
+
+static void
+test_adw_tab_view_close_before_after (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[10];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 10, 3);
+ assert_page_positions (view, pages, 10, 3,
+ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9);
+
+ adw_tab_view_close_pages_before (view, pages[3]);
+ assert_page_positions (view, pages, 10, 3,
+ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9);
+
+ adw_tab_view_close_pages_before (view, pages[5]);
+ assert_page_positions (view, pages, 8, 3,
+ 0, 1, 2, 5, 6, 7, 8, 9);
+
+ adw_tab_view_close_pages_after (view, pages[7]);
+ assert_page_positions (view, pages, 6, 3,
+ 0, 1, 2, 5, 6, 7);
+
+ adw_tab_view_close_pages_after (view, pages[0]);
+ assert_page_positions (view, pages, 3, 3,
+ 0, 1, 2);
+}
+
+static gboolean
+close_page_position_cb (AdwTabView *view,
+ AdwTabPage *page)
+{
+ int position = adw_tab_view_get_page_position (view, page);
+
+ adw_tab_view_close_page_finish (view, page, (position % 2) > 0);
+
+ return GDK_EVENT_STOP;
+}
+
+static void
+test_adw_tab_view_close_signal (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[10];
+ gulong handler;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ /* Allow closing pages with odd positions, including pinned */
+ handler = g_signal_connect (view, "close-page",
+ G_CALLBACK (close_page_position_cb), NULL);
+
+ add_pages (view, pages, 10, 3);
+ assert_page_positions (view, pages, 10, 3,
+ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9);
+
+ adw_tab_view_close_other_pages (view, pages[5]);
+ assert_page_positions (view, pages, 6, 2,
+ 0, 2, 4, 5, 6, 8);
+
+ g_signal_handler_disconnect (view, handler);
+
+ /* Defer closing */
+ handler = g_signal_connect (view, "close-page", G_CALLBACK (close_noop), NULL);
+
+ adw_tab_view_close_page (view, pages[0]);
+ assert_page_positions (view, pages, 6, 2,
+ 0, 2, 4, 5, 6, 8);
+
+ adw_tab_view_close_page_finish (view, pages[0], FALSE);
+ assert_page_positions (view, pages, 6, 2,
+ 0, 2, 4, 5, 6, 8);
+
+ adw_tab_view_close_page (view, pages[0]);
+ assert_page_positions (view, pages, 6, 2,
+ 0, 2, 4, 5, 6, 8);
+
+ adw_tab_view_close_page_finish (view, pages[0], TRUE);
+ assert_page_positions (view, pages, 5, 1,
+ 2, 4, 5, 6, 8);
+
+ g_signal_handler_disconnect (view, handler);
+}
+
+static void
+test_adw_tab_view_close_select (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *pages[14];
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ add_pages (view, pages, 9, 3);
+ pages[9] = adw_tab_view_add_page (view, gtk_button_new (), pages[4]);
+ pages[10] = adw_tab_view_add_page (view, gtk_button_new (), pages[4]);
+ pages[11] = adw_tab_view_add_page (view, gtk_button_new (), pages[9]);
+ pages[12] = adw_tab_view_add_page (view, gtk_button_new (), pages[1]);
+ pages[13] = adw_tab_view_add_page (view, gtk_button_new (), pages[1]);
+
+ assert_page_positions (view, pages, 14, 3,
+ 0, 1, 2, 12, 13, 3, 4, 9, 11, 10, 5, 6, 7, 8);
+
+ /* Nothing happens when closing unselected pages */
+
+ adw_tab_view_set_selected_page (view, pages[0]);
+
+ adw_tab_view_close_page (view, pages[8]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[0]);
+
+ /* No parent */
+
+ assert_page_positions (view, pages, 13, 3,
+ 0, 1, 2, 12, 13, 3, 4, 9, 11, 10, 5, 6, 7);
+
+ adw_tab_view_set_selected_page (view, pages[6]);
+
+ adw_tab_view_close_page (view, pages[6]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[7]);
+
+ adw_tab_view_close_page (view, pages[7]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[5]);
+
+ /* Regular parent */
+
+ assert_page_positions (view, pages, 11, 3,
+ 0, 1, 2, 12, 13, 3, 4, 9, 11, 10, 5);
+
+ adw_tab_view_set_selected_page (view, pages[10]);
+
+ adw_tab_view_close_page (view, pages[10]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[11]);
+
+ adw_tab_view_close_page (view, pages[11]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[9]);
+
+ adw_tab_view_close_page (view, pages[9]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[4]);
+
+ adw_tab_view_close_page (view, pages[4]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[5]);
+
+ /* Pinned parent */
+
+ assert_page_positions (view, pages, 7, 3,
+ 0, 1, 2, 12, 13, 3, 5);
+
+ adw_tab_view_set_selected_page (view, pages[13]);
+
+ adw_tab_view_close_page (view, pages[13]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[12]);
+
+ adw_tab_view_close_page (view, pages[12]);
+ g_assert_true (adw_tab_view_get_selected_page (view) == pages[1]);
+}
+
+static void
+test_adw_tab_view_transfer (void)
+{
+ g_autoptr (AdwTabView) view1 = NULL;
+ g_autoptr (AdwTabView) view2 = NULL;
+ AdwTabPage *pages1[4], *pages2[4];
+
+ view1 = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view1);
+
+ view2 = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view2);
+
+ add_pages (view1, pages1, 4, 2);
+ assert_page_positions (view1, pages1, 4, 2,
+ 0, 1, 2, 3);
+
+ add_pages (view2, pages2, 4, 2);
+ assert_page_positions (view2, pages2, 4, 2,
+ 0, 1, 2, 3);
+
+ adw_tab_view_transfer_page (view1, pages1[1], view2, 1);
+ assert_page_positions (view1, pages1, 3, 1,
+ 0, 2, 3);
+ assert_page_positions (view2, pages2, 5, 3,
+ 0, -1, 1, 2, 3);
+ g_assert_true (adw_tab_view_get_nth_page (view2, 1) == pages1[1]);
+
+ adw_tab_view_transfer_page (view2, pages2[3], view1, 2);
+ assert_page_positions (view1, pages1, 4, 1,
+ 0, 2, -1, 3);
+ assert_page_positions (view2, pages2, 4, 3,
+ 0, -1, 1, 2);
+ g_assert_true (adw_tab_view_get_nth_page (view1, 2) == pages2[3]);
+}
+
+static void
+test_adw_tab_page_title (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ const char *title;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::title", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "title", &title, NULL);
+ g_assert_null (title);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_title (page, "Some title");
+ g_object_get (page, "title", &title, NULL);
+ g_assert_cmpstr (title, ==, "Some title");
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "title", "Some other title", NULL);
+ g_assert_cmpstr (adw_tab_page_get_title (page), ==, "Some other title");
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_tooltip (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ const char *tooltip;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::tooltip", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "tooltip", &tooltip, NULL);
+ g_assert_null (tooltip);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_tooltip (page, "Some tooltip");
+ g_object_get (page, "tooltip", &tooltip, NULL);
+ g_assert_cmpstr (tooltip, ==, "Some tooltip");
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "tooltip", "Some other tooltip", NULL);
+ g_assert_cmpstr (adw_tab_page_get_tooltip (page), ==, "Some other tooltip");
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_icon (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ GIcon *icon = NULL;
+ g_autoptr (GIcon) icon1 = g_themed_icon_new ("go-previous-symbolic");
+ g_autoptr (GIcon) icon2 = g_themed_icon_new ("go-next-symbolic");
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::icon", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "icon", &icon, NULL);
+ g_assert_null (icon);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_icon (page, icon1);
+ g_object_get (page, "icon", &icon, NULL);
+ g_assert_true (icon == icon1);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "icon", icon2, NULL);
+ g_assert_true (adw_tab_page_get_icon (page) == icon2);
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_loading (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ gboolean loading;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::loading", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "loading", &loading, NULL);
+ g_assert_false (loading);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_loading (page, TRUE);
+ g_object_get (page, "loading", &loading, NULL);
+ g_assert_true (loading);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "loading", FALSE, NULL);
+ g_assert_false (adw_tab_page_get_loading (page));
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_indicator_icon (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ GIcon *icon = NULL;
+ g_autoptr (GIcon) icon1 = g_themed_icon_new ("go-previous-symbolic");
+ g_autoptr (GIcon) icon2 = g_themed_icon_new ("go-next-symbolic");
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::indicator-icon", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "indicator-icon", &icon, NULL);
+ g_assert_null (icon);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_indicator_icon (page, icon1);
+ g_object_get (page, "indicator-icon", &icon, NULL);
+ g_assert_true (icon == icon1);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "indicator-icon", icon2, NULL);
+ g_assert_true (adw_tab_page_get_indicator_icon (page) == icon2);
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_indicator_activatable (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ gboolean activatable;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::indicator-activatable", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "indicator-activatable", &activatable, NULL);
+ g_assert_false (activatable);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_indicator_activatable (page, TRUE);
+ g_object_get (page, "indicator-activatable", &activatable, NULL);
+ g_assert_true (activatable);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "indicator-activatable", FALSE, NULL);
+ g_assert_false (adw_tab_page_get_indicator_activatable (page));
+ g_assert_cmpint (notified, ==, 2);
+}
+
+static void
+test_adw_tab_page_needs_attention (void)
+{
+ g_autoptr (AdwTabView) view = NULL;
+ AdwTabPage *page;
+ gboolean needs_attention;
+
+ view = g_object_ref_sink (ADW_TAB_VIEW (adw_tab_view_new ()));
+ g_assert_nonnull (view);
+
+ page = adw_tab_view_append (view, gtk_button_new ());
+ g_assert_nonnull (page);
+
+ notified = 0;
+ g_signal_connect (page, "notify::needs-attention", G_CALLBACK (notify_cb), NULL);
+
+ g_object_get (page, "needs-attention", &needs_attention, NULL);
+ g_assert_false (needs_attention);
+ g_assert_cmpint (notified, ==, 0);
+
+ adw_tab_page_set_needs_attention (page, TRUE);
+ g_object_get (page, "needs-attention", &needs_attention, NULL);
+ g_assert_true (needs_attention);
+ g_assert_cmpint (notified, ==, 1);
+
+ g_object_set (page, "needs-attention", FALSE, NULL);
+ g_assert_false (adw_tab_page_get_needs_attention (page));
+ g_assert_cmpint (notified, ==, 2);
+}
+
+int
+main (int argc,
+ char *argv[])
+{
+ gtk_test_init (&argc, &argv, NULL);
+ adw_init ();
+
+ g_test_add_func ("/Adwaita/TabView/n_pages", test_adw_tab_view_n_pages);
+ g_test_add_func ("/Adwaita/TabView/n_pinned_pages", test_adw_tab_view_n_pinned_pages);
+ g_test_add_func ("/Adwaita/TabView/default_icon", test_adw_tab_view_default_icon);
+ g_test_add_func ("/Adwaita/TabView/menu_model", test_adw_tab_view_menu_model);
+ g_test_add_func ("/Adwaita/TabView/shortcut_widget", test_adw_tab_view_shortcut_widget);
+ g_test_add_func ("/Adwaita/TabView/pages", test_adw_tab_view_pages);
+ g_test_add_func ("/Adwaita/TabView/select", test_adw_tab_view_select);
+ g_test_add_func ("/Adwaita/TabView/add_basic", test_adw_tab_view_add_basic);
+ g_test_add_func ("/Adwaita/TabView/add_auto", test_adw_tab_view_add_auto);
+ g_test_add_func ("/Adwaita/TabView/reorder", test_adw_tab_view_reorder);
+ g_test_add_func ("/Adwaita/TabView/reorder_first_last", test_adw_tab_view_reorder_first_last);
+ g_test_add_func ("/Adwaita/TabView/reorder_forward_backward", test_adw_tab_view_reorder_forward_backward);
+ g_test_add_func ("/Adwaita/TabView/pin", test_adw_tab_view_pin);
+ g_test_add_func ("/Adwaita/TabView/close", test_adw_tab_view_close);
+ g_test_add_func ("/Adwaita/TabView/close_other", test_adw_tab_view_close_other);
+ g_test_add_func ("/Adwaita/TabView/close_before_after", test_adw_tab_view_close_before_after);
+ g_test_add_func ("/Adwaita/TabView/close_signal", test_adw_tab_view_close_signal);
+ g_test_add_func ("/Adwaita/TabView/close_select", test_adw_tab_view_close_select);
+ g_test_add_func ("/Adwaita/TabView/transfer", test_adw_tab_view_transfer);
+ g_test_add_func ("/Adwaita/TabPage/title", test_adw_tab_page_title);
+ g_test_add_func ("/Adwaita/TabPage/tooltip", test_adw_tab_page_tooltip);
+ g_test_add_func ("/Adwaita/TabPage/icon", test_adw_tab_page_icon);
+ g_test_add_func ("/Adwaita/TabPage/loading", test_adw_tab_page_loading);
+ g_test_add_func ("/Adwaita/TabPage/indicator_icon", test_adw_tab_page_indicator_icon);
+ g_test_add_func ("/Adwaita/TabPage/indicator_activatable", test_adw_tab_page_indicator_activatable);
+ g_test_add_func ("/Adwaita/TabPage/needs_attention", test_adw_tab_page_needs_attention);
+
+ return g_test_run ();
+}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]