2019-06-07 23:27:24 +00:00
|
|
|
/* GTK - The GIMP Toolkit
|
|
|
|
* Copyright (C) 2019 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* - Matthias Clasen <mclasen@redhat.com>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "gtkpopoverbar.h"
|
|
|
|
#include "gtkpopovermenu.h"
|
|
|
|
|
|
|
|
#include "gtkbox.h"
|
|
|
|
#include "gtklabel.h"
|
|
|
|
#include "gtkmenubutton.h"
|
|
|
|
#include "gtkintl.h"
|
|
|
|
#include "gtkprivate.h"
|
|
|
|
#include "gtkmarshalers.h"
|
|
|
|
#include "gtkstylecontext.h"
|
|
|
|
#include "gtkgestureclick.h"
|
|
|
|
#include "gtkeventcontrollerkey.h"
|
2019-06-08 03:23:22 +00:00
|
|
|
#include "gtkeventcontrollermotion.h"
|
2019-06-07 23:27:24 +00:00
|
|
|
#include "gtkactionmuxerprivate.h"
|
|
|
|
#include "gtkmenutracker.h"
|
|
|
|
#include "gtkwidgetprivate.h"
|
2019-06-08 03:23:22 +00:00
|
|
|
#include "gtkmain.h"
|
|
|
|
#include "gtknative.h"
|
2019-06-07 23:27:24 +00:00
|
|
|
|
|
|
|
struct _GtkPopoverBar
|
|
|
|
{
|
|
|
|
GtkWidget parent;
|
|
|
|
|
|
|
|
GtkMenuTracker *tracker;
|
|
|
|
GtkWidget *box;
|
2019-06-08 03:23:22 +00:00
|
|
|
GtkWidget *open_popover;
|
2019-06-07 23:27:24 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct _GtkPopoverBarClass GtkPopoverBarClass;
|
|
|
|
struct _GtkPopoverBarClass
|
|
|
|
{
|
|
|
|
GtkWidgetClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (GtkPopoverBar, gtk_popover_bar, GTK_TYPE_WIDGET)
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_init (GtkPopoverBar *bar)
|
|
|
|
{
|
|
|
|
bar->box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
gtk_widget_set_parent (bar->box, GTK_WIDGET (bar));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_dispose (GObject *object)
|
|
|
|
{
|
|
|
|
GtkPopoverBar *bar = GTK_POPOVER_BAR (object);
|
|
|
|
|
|
|
|
g_clear_pointer (&bar->tracker, gtk_menu_tracker_free);
|
|
|
|
g_clear_pointer (&bar->box, gtk_widget_unparent);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_popover_bar_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
G_OBJECT_CLASS (gtk_popover_bar_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_measure (GtkWidget *widget,
|
|
|
|
GtkOrientation orientation,
|
|
|
|
int for_size,
|
|
|
|
int *minimum,
|
|
|
|
int *natural,
|
|
|
|
int *minimum_baseline,
|
|
|
|
int *natural_baseline)
|
|
|
|
{
|
|
|
|
GtkPopoverBar *bar = GTK_POPOVER_BAR (widget);
|
|
|
|
|
|
|
|
gtk_widget_measure (bar->box, orientation, for_size,
|
|
|
|
minimum, natural,
|
|
|
|
minimum_baseline, natural_baseline);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_size_allocate (GtkWidget *widget,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int baseline)
|
|
|
|
{
|
|
|
|
GtkPopoverBar *bar = GTK_POPOVER_BAR (widget);
|
2019-06-08 03:23:22 +00:00
|
|
|
GtkWidget *child;
|
2019-06-07 23:27:24 +00:00
|
|
|
|
|
|
|
gtk_widget_size_allocate (bar->box,
|
|
|
|
&(GtkAllocation) { 0, 0, width, height },
|
|
|
|
baseline);
|
2019-06-08 03:23:22 +00:00
|
|
|
|
|
|
|
for (child = gtk_widget_get_first_child (bar->box);
|
|
|
|
child;
|
|
|
|
child = gtk_widget_get_next_sibling (child))
|
|
|
|
{
|
|
|
|
GtkNative *popover = GTK_NATIVE (g_object_get_data (G_OBJECT (child), "popover"));
|
|
|
|
gtk_native_check_resize (popover);
|
|
|
|
}
|
2019-06-07 23:27:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_bar_class_init (GtkPopoverBarClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->dispose = gtk_popover_bar_dispose;
|
|
|
|
object_class->finalize = gtk_popover_bar_finalize;
|
|
|
|
|
|
|
|
widget_class->measure = gtk_popover_bar_measure;
|
|
|
|
widget_class->size_allocate = gtk_popover_bar_size_allocate;
|
|
|
|
|
|
|
|
gtk_widget_class_set_css_name (widget_class, I_("menubar"));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
tracker_remove (gint position,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkPopoverBar *bar = user_data;
|
|
|
|
GtkWidget *child;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (child = gtk_widget_get_first_child (bar->box), i = 0;
|
|
|
|
child;
|
|
|
|
child = gtk_widget_get_next_sibling (child), i++)
|
|
|
|
{
|
|
|
|
if (i == position)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy (child);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-08 03:23:22 +00:00
|
|
|
static void
|
|
|
|
clicked_cb (GtkGesture *gesture,
|
|
|
|
int n,
|
|
|
|
double x,
|
|
|
|
double y,
|
|
|
|
GtkPopoverBar *bar)
|
|
|
|
{
|
|
|
|
GtkWidget *target;
|
|
|
|
GtkWidget *popover;
|
|
|
|
|
|
|
|
target = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
|
|
|
|
|
|
|
|
popover = GTK_WIDGET (g_object_get_data (G_OBJECT (target), "popover"));
|
|
|
|
if (popover && popover != bar->open_popover)
|
|
|
|
{
|
|
|
|
GtkAllocation alloc;
|
|
|
|
|
|
|
|
if (bar->open_popover)
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (bar->open_popover));
|
|
|
|
|
|
|
|
bar->open_popover = popover;
|
|
|
|
|
|
|
|
gtk_widget_get_allocation (target, &alloc);
|
|
|
|
gtk_popover_set_pointing_to (GTK_POPOVER (popover), &alloc);
|
|
|
|
gtk_popover_popup (GTK_POPOVER (popover));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
enter_cb (GtkEventController *controller,
|
|
|
|
double x,
|
|
|
|
double y,
|
|
|
|
GdkCrossingMode mode,
|
|
|
|
GdkNotifyType type,
|
|
|
|
GtkPopoverBar *bar)
|
|
|
|
{
|
|
|
|
GtkWidget *target;
|
|
|
|
GtkWidget *popover;
|
|
|
|
|
|
|
|
target = gtk_event_controller_get_widget (controller);
|
|
|
|
|
|
|
|
popover = GTK_WIDGET (g_object_get_data (G_OBJECT (target), "popover"));
|
|
|
|
|
|
|
|
if (popover && bar->open_popover && popover != bar->open_popover)
|
|
|
|
{
|
|
|
|
GtkAllocation alloc;
|
|
|
|
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (bar->open_popover));
|
|
|
|
|
|
|
|
bar->open_popover = popover;
|
|
|
|
|
|
|
|
gtk_widget_get_allocation (target, &alloc);
|
|
|
|
gtk_popover_set_pointing_to (GTK_POPOVER (popover), &alloc);
|
|
|
|
gtk_popover_popup (GTK_POPOVER (popover));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
popover_unmap (GtkWidget *popover,
|
|
|
|
GtkPopoverBar *bar)
|
|
|
|
{
|
|
|
|
if (popover == bar->open_popover)
|
|
|
|
bar->open_popover = NULL;
|
|
|
|
}
|
|
|
|
|
2019-06-07 23:27:24 +00:00
|
|
|
static void
|
|
|
|
tracker_insert (GtkMenuTrackerItem *item,
|
|
|
|
gint position,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkPopoverBar *bar = user_data;
|
|
|
|
|
|
|
|
if (gtk_menu_tracker_item_get_has_link (item, G_MENU_LINK_SUBMENU))
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
GMenuModel *model;
|
|
|
|
GtkWidget *sibling;
|
|
|
|
GtkWidget *child;
|
|
|
|
GtkWidget *popover;
|
|
|
|
int i;
|
2019-06-08 03:23:22 +00:00
|
|
|
GtkEventController *controller;
|
2019-06-07 23:27:24 +00:00
|
|
|
|
2019-06-08 03:23:22 +00:00
|
|
|
widget = g_object_new (GTK_TYPE_LABEL, "css-name", "menuitem", NULL);
|
2019-06-07 23:27:24 +00:00
|
|
|
g_object_bind_property (item, "label", widget, "label", G_BINDING_SYNC_CREATE);
|
|
|
|
|
2019-06-08 03:23:22 +00:00
|
|
|
controller = GTK_EVENT_CONTROLLER (gtk_gesture_click_new ());
|
|
|
|
g_signal_connect (controller, "pressed", G_CALLBACK (clicked_cb), bar);
|
|
|
|
gtk_widget_add_controller (GTK_WIDGET (widget), controller);
|
|
|
|
|
|
|
|
controller = gtk_event_controller_motion_new ();
|
|
|
|
gtk_event_controller_set_propagation_limit (controller, GTK_LIMIT_NONE);
|
|
|
|
g_signal_connect (controller, "enter", G_CALLBACK (enter_cb), bar);
|
|
|
|
gtk_widget_add_controller (GTK_WIDGET (widget), controller);
|
|
|
|
|
|
|
|
g_signal_connect (popover, "unmap", G_CALLBACK (popover_unmap), bar);
|
|
|
|
|
2019-06-07 23:27:24 +00:00
|
|
|
model = _gtk_menu_tracker_item_get_link (item, G_MENU_LINK_SUBMENU);
|
|
|
|
popover = gtk_popover_menu_new_from_model (GTK_WIDGET (bar), model);
|
2019-06-08 03:23:22 +00:00
|
|
|
gtk_popover_set_position (GTK_POPOVER (popover), GTK_POS_BOTTOM);
|
2019-06-07 23:27:24 +00:00
|
|
|
gtk_popover_set_has_arrow (GTK_POPOVER (popover), FALSE);
|
2019-06-08 03:23:22 +00:00
|
|
|
gtk_widget_set_halign (popover, GTK_ALIGN_START);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (widget), "popover", popover);
|
2019-06-07 23:27:24 +00:00
|
|
|
|
|
|
|
sibling = NULL;
|
|
|
|
for (child = gtk_widget_get_first_child (bar->box), i = 1;
|
|
|
|
child;
|
|
|
|
child = gtk_widget_get_next_sibling (child), i++)
|
|
|
|
{
|
|
|
|
if (i == position)
|
|
|
|
{
|
|
|
|
sibling = child;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
gtk_box_insert_child_after (GTK_BOX (bar->box), widget, sibling);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
g_warning ("Don't know how to handle this item");
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_bar_new_from_model (GMenuModel *model)
|
|
|
|
{
|
|
|
|
GtkWidget *bar;
|
|
|
|
GtkActionMuxer *muxer;
|
|
|
|
|
|
|
|
bar = g_object_new (GTK_TYPE_POPOVER_BAR, NULL);
|
|
|
|
|
|
|
|
muxer = _gtk_widget_get_action_muxer (bar, TRUE);
|
|
|
|
|
|
|
|
GTK_POPOVER_BAR (bar)->tracker =
|
|
|
|
gtk_menu_tracker_new (GTK_ACTION_OBSERVABLE (muxer),
|
|
|
|
model,
|
|
|
|
FALSE,
|
|
|
|
TRUE,
|
|
|
|
FALSE,
|
|
|
|
NULL,
|
|
|
|
tracker_insert,
|
|
|
|
tracker_remove,
|
|
|
|
bar);
|
|
|
|
|
|
|
|
return bar;
|
|
|
|
}
|