2014-10-24 03:38:09 +00:00
|
|
|
/* GTK - The GIMP Toolkit
|
|
|
|
* Copyright © 2014 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* 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 "gtkpopovermenu.h"
|
2019-06-13 00:13:21 +00:00
|
|
|
#include "gtkpopovermenuprivate.h"
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
#include "gtkstack.h"
|
2015-10-30 14:08:18 +00:00
|
|
|
#include "gtkstylecontext.h"
|
2014-10-24 03:38:09 +00:00
|
|
|
#include "gtkintl.h"
|
2019-06-10 03:10:13 +00:00
|
|
|
#include "gtkmenusectionboxprivate.h"
|
2019-06-07 17:05:48 +00:00
|
|
|
#include "gtkmenubutton.h"
|
|
|
|
#include "gtkactionmuxerprivate.h"
|
2019-06-10 03:10:13 +00:00
|
|
|
#include "gtkmenutrackerprivate.h"
|
2017-07-08 10:04:44 +00:00
|
|
|
#include "gtkpopoverprivate.h"
|
2019-06-07 17:05:48 +00:00
|
|
|
#include "gtkwidgetprivate.h"
|
2019-06-09 14:08:18 +00:00
|
|
|
#include "gtkeventcontrollerkey.h"
|
2019-08-27 08:28:29 +00:00
|
|
|
#include "gtkeventcontrollermotion.h"
|
2019-06-09 14:08:18 +00:00
|
|
|
#include "gtkmain.h"
|
2019-06-12 16:35:36 +00:00
|
|
|
#include "gtktypebuiltins.h"
|
|
|
|
#include "gtkbindings.h"
|
2019-08-27 08:28:29 +00:00
|
|
|
#include "gtkmodelbutton.h"
|
|
|
|
#include "gtkpopovermenubar.h"
|
2014-10-24 03:38:09 +00:00
|
|
|
|
|
|
|
|
2014-10-26 23:04:28 +00:00
|
|
|
/**
|
|
|
|
* SECTION:gtkpopovermenu
|
|
|
|
* @Short_description: Popovers to use as menus
|
|
|
|
* @Title: GtkPopoverMenu
|
|
|
|
*
|
|
|
|
* GtkPopoverMenu is a subclass of #GtkPopover that treats its
|
2016-01-19 11:52:58 +00:00
|
|
|
* children like menus and allows switching between them. It is
|
2014-10-26 23:04:28 +00:00
|
|
|
* meant to be used primarily together with #GtkModelButton, but
|
|
|
|
* any widget can be used, such as #GtkSpinButton or #GtkScale.
|
|
|
|
* In this respect, GtkPopoverMenu is more flexible than popovers
|
|
|
|
* that are created from a #GMenuModel with gtk_popover_new_from_model().
|
|
|
|
*
|
2019-03-27 22:37:57 +00:00
|
|
|
* To add a child as a submenu, use gtk_popover_menu_add_submenu().
|
|
|
|
* To let the user open this submenu, add a #GtkModelButton whose
|
|
|
|
* #GtkModelButton:menu-name property is set to the name you've given
|
|
|
|
* to the submenu.
|
|
|
|
*
|
|
|
|
* To add a named submenu in a ui file, set the #GtkWidget:name property
|
|
|
|
* of the widget that you are adding as a child of the popover menu.
|
2014-10-26 23:04:28 +00:00
|
|
|
*
|
|
|
|
* By convention, the first child of a submenu should be a #GtkModelButton
|
|
|
|
* to switch back to the parent menu. Such a button should use the
|
|
|
|
* #GtkModelButton:inverted and #GtkModelButton:centered properties
|
|
|
|
* to achieve a title-like appearance and place the submenu indicator
|
|
|
|
* at the opposite side. To switch back to the main menu, use "main"
|
|
|
|
* as the menu name.
|
|
|
|
*
|
|
|
|
* # Example
|
|
|
|
*
|
|
|
|
* |[
|
|
|
|
* <object class="GtkPopoverMenu">
|
|
|
|
* <child>
|
|
|
|
* <object class="GtkBox">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="margin">10</property>
|
|
|
|
* <child>
|
|
|
|
* <object class="GtkModelButton">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="action-name">win.frob</property>
|
|
|
|
* <property name="text" translatable="yes">Frob</property>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* <child>
|
|
|
|
* <object class="GtkModelButton">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="menu-name">more</property>
|
|
|
|
* <property name="text" translatable="yes">More</property>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* <child>
|
|
|
|
* <object class="GtkBox">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="margin">10</property>
|
2019-03-27 22:37:57 +00:00
|
|
|
* <property name="name">more</property>
|
2014-10-26 23:04:28 +00:00
|
|
|
* <child>
|
|
|
|
* <object class="GtkModelButton">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="action-name">win.foo</property>
|
|
|
|
* <property name="text" translatable="yes">Foo</property>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* <child>
|
|
|
|
* <object class="GtkModelButton">
|
|
|
|
* <property name="visible">True</property>
|
|
|
|
* <property name="action-name">win.bar</property>
|
|
|
|
* <property name="text" translatable="yes">Bar</property>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* </object>
|
|
|
|
* </child>
|
|
|
|
* </object>
|
|
|
|
* ]|
|
2015-10-30 14:08:18 +00:00
|
|
|
*
|
2017-07-08 10:04:44 +00:00
|
|
|
* # CSS Nodes
|
|
|
|
*
|
|
|
|
* #GtkPopoverMenu is just a subclass of #GtkPopover that adds
|
|
|
|
* custom content to it, therefore it has the same CSS nodes.
|
|
|
|
* It is one of the cases that add a .menu style class to the
|
|
|
|
* popover's contents node.
|
2014-10-26 23:04:28 +00:00
|
|
|
*/
|
|
|
|
|
2019-05-19 20:45:42 +00:00
|
|
|
typedef struct _GtkPopoverMenuClass GtkPopoverMenuClass;
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
struct _GtkPopoverMenu
|
|
|
|
{
|
|
|
|
GtkPopover parent_instance;
|
2019-06-13 00:13:21 +00:00
|
|
|
|
|
|
|
GtkWidget *active_item;
|
2019-08-27 08:28:29 +00:00
|
|
|
GtkWidget *open_submenu;
|
|
|
|
GtkWidget *parent_menu;
|
2014-10-24 03:38:09 +00:00
|
|
|
};
|
|
|
|
|
2019-05-19 20:45:42 +00:00
|
|
|
struct _GtkPopoverMenuClass
|
|
|
|
{
|
|
|
|
GtkPopoverClass parent_class;
|
|
|
|
};
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
enum {
|
2014-11-18 12:39:15 +00:00
|
|
|
PROP_VISIBLE_SUBMENU = 1
|
|
|
|
};
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
G_DEFINE_TYPE (GtkPopoverMenu, gtk_popover_menu, GTK_TYPE_POPOVER)
|
|
|
|
|
2019-08-27 08:28:29 +00:00
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_menu_get_parent_menu (GtkPopoverMenu *menu)
|
|
|
|
{
|
|
|
|
return menu->parent_menu;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gtk_popover_menu_set_parent_menu (GtkPopoverMenu *menu,
|
|
|
|
GtkWidget *parent)
|
|
|
|
{
|
|
|
|
menu->parent_menu = parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_menu_get_open_submenu (GtkPopoverMenu *menu)
|
|
|
|
{
|
|
|
|
return menu->open_submenu;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gtk_popover_menu_set_open_submenu (GtkPopoverMenu *menu,
|
|
|
|
GtkWidget *submenu)
|
|
|
|
{
|
|
|
|
menu->open_submenu = submenu;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_menu_get_active_item (GtkPopoverMenu *menu)
|
|
|
|
{
|
|
|
|
return menu->active_item;
|
|
|
|
}
|
|
|
|
|
2019-06-13 00:13:21 +00:00
|
|
|
void
|
|
|
|
gtk_popover_menu_set_active_item (GtkPopoverMenu *menu,
|
|
|
|
GtkWidget *item)
|
|
|
|
{
|
|
|
|
if (menu->active_item != item)
|
|
|
|
{
|
|
|
|
if (menu->active_item)
|
|
|
|
gtk_widget_unset_state_flags (menu->active_item, GTK_STATE_FLAG_SELECTED);
|
|
|
|
|
|
|
|
menu->active_item = item;
|
|
|
|
|
|
|
|
if (menu->active_item)
|
|
|
|
{
|
2019-08-27 08:28:29 +00:00
|
|
|
GtkWidget *popover;
|
|
|
|
|
2019-06-13 00:13:21 +00:00
|
|
|
gtk_widget_set_state_flags (menu->active_item, GTK_STATE_FLAG_SELECTED, FALSE);
|
2019-08-27 08:28:29 +00:00
|
|
|
if (GTK_IS_MODEL_BUTTON (item))
|
|
|
|
g_object_get (item, "popover", &popover, NULL);
|
|
|
|
|
|
|
|
if (!popover || popover != menu->open_submenu)
|
|
|
|
gtk_widget_grab_focus (menu->active_item);
|
|
|
|
|
|
|
|
g_clear_object (&popover);
|
2019-06-13 00:13:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-19 08:26:25 +00:00
|
|
|
static void
|
|
|
|
visible_submenu_changed (GObject *object,
|
|
|
|
GParamSpec *pspec,
|
|
|
|
GtkPopoverMenu *popover)
|
|
|
|
{
|
|
|
|
g_object_notify (G_OBJECT (popover), "visible-submenu");
|
|
|
|
}
|
|
|
|
|
2019-06-09 14:08:18 +00:00
|
|
|
static void
|
2019-10-03 20:13:54 +00:00
|
|
|
focus_out (GtkEventControllerKey *controller,
|
|
|
|
GdkCrossingMode mode,
|
|
|
|
GdkNotifyType detail,
|
|
|
|
GtkPopoverMenu *menu)
|
2019-06-09 14:08:18 +00:00
|
|
|
{
|
2019-10-06 10:50:22 +00:00
|
|
|
GtkWidget *new_focus = gtk_root_get_focus (gtk_widget_get_root (GTK_WIDGET (menu)));
|
|
|
|
|
|
|
|
if (!gtk_event_controller_key_contains_focus (controller) &&
|
|
|
|
new_focus != NULL)
|
2019-08-27 08:28:29 +00:00
|
|
|
{
|
|
|
|
if (menu->parent_menu &&
|
2019-10-03 20:13:54 +00:00
|
|
|
GTK_POPOVER_MENU (menu->parent_menu)->open_submenu == (GtkWidget*) menu)
|
2019-08-27 08:28:29 +00:00
|
|
|
GTK_POPOVER_MENU (menu->parent_menu)->open_submenu = NULL;
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (menu));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
leave_cb (GtkEventController *controller,
|
|
|
|
GdkCrossingMode mode,
|
|
|
|
GdkNotifyType type,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkWidget *target;
|
|
|
|
gboolean is;
|
|
|
|
gboolean contains;
|
|
|
|
|
|
|
|
target = gtk_event_controller_get_widget (controller);
|
|
|
|
|
|
|
|
g_object_get (controller,
|
|
|
|
"is-pointer-focus", &is,
|
|
|
|
"contains-pointer-focus", &contains,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!(is || contains))
|
|
|
|
gtk_popover_menu_set_active_item (GTK_POPOVER_MENU (target), NULL);
|
2019-06-09 14:08:18 +00:00
|
|
|
}
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
static void
|
2014-10-26 22:58:55 +00:00
|
|
|
gtk_popover_menu_init (GtkPopoverMenu *popover)
|
2014-10-24 03:38:09 +00:00
|
|
|
{
|
|
|
|
GtkWidget *stack;
|
2015-10-30 14:08:18 +00:00
|
|
|
GtkStyleContext *style_context;
|
2019-06-09 14:08:18 +00:00
|
|
|
GtkEventController *controller;
|
2014-10-24 03:38:09 +00:00
|
|
|
|
|
|
|
stack = gtk_stack_new ();
|
|
|
|
gtk_stack_set_vhomogeneous (GTK_STACK (stack), FALSE);
|
|
|
|
gtk_stack_set_transition_type (GTK_STACK (stack), GTK_STACK_TRANSITION_TYPE_SLIDE_LEFT_RIGHT);
|
2015-07-19 15:40:28 +00:00
|
|
|
gtk_stack_set_interpolate_size (GTK_STACK (stack), TRUE);
|
2014-10-24 03:38:09 +00:00
|
|
|
gtk_container_add (GTK_CONTAINER (popover), stack);
|
2014-11-19 08:26:25 +00:00
|
|
|
g_signal_connect (stack, "notify::visible-child-name",
|
|
|
|
G_CALLBACK (visible_submenu_changed), popover);
|
2015-10-30 14:08:18 +00:00
|
|
|
|
2019-06-06 18:08:52 +00:00
|
|
|
style_context = gtk_widget_get_style_context (GTK_WIDGET (popover));
|
2015-10-30 14:08:18 +00:00
|
|
|
gtk_style_context_add_class (style_context, GTK_STYLE_CLASS_MENU);
|
2019-06-09 14:08:18 +00:00
|
|
|
|
|
|
|
controller = gtk_event_controller_key_new ();
|
|
|
|
g_signal_connect (controller, "focus-out", G_CALLBACK (focus_out), popover);
|
|
|
|
gtk_widget_add_controller (GTK_WIDGET (popover), controller);
|
2019-08-27 08:28:29 +00:00
|
|
|
|
|
|
|
controller = gtk_event_controller_motion_new ();
|
|
|
|
g_signal_connect (controller, "leave", G_CALLBACK (leave_cb), popover);
|
|
|
|
gtk_widget_add_controller (GTK_WIDGET (popover), controller);
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_menu_map (GtkWidget *widget)
|
|
|
|
{
|
2014-10-26 23:14:01 +00:00
|
|
|
gtk_popover_menu_open_submenu (GTK_POPOVER_MENU (widget), "main");
|
2019-10-06 10:52:11 +00:00
|
|
|
GTK_WIDGET_CLASS (gtk_popover_menu_parent_class)->map (widget);
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_menu_unmap (GtkWidget *widget)
|
|
|
|
{
|
|
|
|
GTK_WIDGET_CLASS (gtk_popover_menu_parent_class)->unmap (widget);
|
2019-10-06 10:13:42 +00:00
|
|
|
gtk_popover_menu_open_submenu (GTK_POPOVER_MENU (widget), "main");
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_menu_add (GtkContainer *container,
|
|
|
|
GtkWidget *child)
|
|
|
|
{
|
2014-10-26 22:58:55 +00:00
|
|
|
GtkWidget *stack;
|
2014-10-24 03:38:09 +00:00
|
|
|
|
2014-10-26 22:58:55 +00:00
|
|
|
stack = gtk_bin_get_child (GTK_BIN (container));
|
|
|
|
|
|
|
|
if (stack == NULL)
|
2014-10-24 03:38:09 +00:00
|
|
|
{
|
2018-07-23 10:01:01 +00:00
|
|
|
GTK_CONTAINER_CLASS (gtk_popover_menu_parent_class)->add (container, child);
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
else
|
2014-10-26 22:58:55 +00:00
|
|
|
{
|
2016-10-15 20:32:51 +00:00
|
|
|
const char *name;
|
2014-10-26 22:58:55 +00:00
|
|
|
|
2019-03-27 22:37:57 +00:00
|
|
|
if (gtk_widget_get_name (child))
|
|
|
|
name = gtk_widget_get_name (child);
|
|
|
|
else if (gtk_stack_get_child_by_name (GTK_STACK (stack), "main"))
|
2014-10-26 22:58:55 +00:00
|
|
|
name = "submenu";
|
|
|
|
else
|
|
|
|
name = "main";
|
|
|
|
|
2019-03-27 22:37:57 +00:00
|
|
|
gtk_popover_menu_add_submenu (GTK_POPOVER_MENU (container), child, name);
|
2014-10-26 22:58:55 +00:00
|
|
|
}
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_menu_remove (GtkContainer *container,
|
|
|
|
GtkWidget *child)
|
|
|
|
{
|
2014-10-26 22:58:55 +00:00
|
|
|
GtkWidget *stack;
|
|
|
|
|
|
|
|
stack = gtk_bin_get_child (GTK_BIN (container));
|
|
|
|
|
|
|
|
if (child == stack)
|
2014-10-24 03:38:09 +00:00
|
|
|
GTK_CONTAINER_CLASS (gtk_popover_menu_parent_class)->remove (container, child);
|
|
|
|
else
|
2014-10-26 22:58:55 +00:00
|
|
|
gtk_container_remove (GTK_CONTAINER (stack), child);
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
2014-11-18 12:39:15 +00:00
|
|
|
static void
|
|
|
|
gtk_popover_menu_get_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkWidget *stack;
|
|
|
|
|
|
|
|
stack = gtk_bin_get_child (GTK_BIN (object));
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_VISIBLE_SUBMENU:
|
|
|
|
g_value_set_string (value, gtk_stack_get_visible_child_name (GTK_STACK (stack)));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gtk_popover_menu_set_property (GObject *object,
|
|
|
|
guint property_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
GtkWidget *stack;
|
|
|
|
|
|
|
|
stack = gtk_bin_get_child (GTK_BIN (object));
|
|
|
|
|
|
|
|
switch (property_id)
|
|
|
|
{
|
|
|
|
case PROP_VISIBLE_SUBMENU:
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (stack), g_value_get_string (value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-09 02:25:12 +00:00
|
|
|
static gboolean
|
|
|
|
gtk_popover_menu_focus (GtkWidget *widget,
|
|
|
|
GtkDirectionType direction)
|
|
|
|
{
|
|
|
|
if (gtk_widget_get_first_child (widget) == NULL)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-27 08:28:29 +00:00
|
|
|
if (GTK_POPOVER_MENU (widget)->open_submenu)
|
|
|
|
{
|
|
|
|
if (gtk_widget_child_focus (GTK_POPOVER_MENU (widget)->open_submenu, direction))
|
|
|
|
return TRUE;
|
|
|
|
if (direction == GTK_DIR_LEFT)
|
|
|
|
{
|
|
|
|
gtk_widget_grab_focus (GTK_POPOVER_MENU (widget)->active_item);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-06-09 02:25:12 +00:00
|
|
|
if (gtk_widget_focus_move (widget, direction))
|
|
|
|
return TRUE;
|
|
|
|
|
2019-08-27 08:28:29 +00:00
|
|
|
if (direction == GTK_DIR_LEFT || direction == GTK_DIR_RIGHT)
|
|
|
|
{
|
|
|
|
/* If we are part of a menubar, we want to let the
|
|
|
|
* menubar use left/right arrows for cycling, else
|
|
|
|
* we eat them.
|
|
|
|
*/
|
|
|
|
if (gtk_widget_get_ancestor (widget, GTK_TYPE_POPOVER_MENU_BAR) ||
|
|
|
|
(gtk_popover_menu_get_parent_menu (GTK_POPOVER_MENU (widget)) &&
|
|
|
|
direction == GTK_DIR_LEFT))
|
|
|
|
return FALSE;
|
|
|
|
else
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if (direction == GTK_DIR_UP || direction == GTK_DIR_DOWN)
|
2019-06-09 02:25:12 +00:00
|
|
|
{
|
|
|
|
GtkWidget *p;
|
|
|
|
|
|
|
|
/* cycle around */
|
|
|
|
for (p = gtk_window_get_focus (GTK_WINDOW (gtk_widget_get_root (widget)));
|
|
|
|
p != widget;
|
|
|
|
p = gtk_widget_get_parent (p))
|
|
|
|
{
|
|
|
|
gtk_widget_set_focus_child (p, NULL);
|
|
|
|
}
|
|
|
|
if (gtk_widget_focus_move (widget, direction))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-06-12 16:35:36 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
add_tab_bindings (GtkBindingSet *binding_set,
|
|
|
|
GdkModifierType modifiers,
|
|
|
|
GtkDirectionType direction)
|
|
|
|
{
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_Tab, modifiers,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_KP_Tab, modifiers,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_arrow_bindings (GtkBindingSet *binding_set,
|
|
|
|
guint keysym,
|
|
|
|
GtkDirectionType direction)
|
|
|
|
{
|
|
|
|
guint keypad_keysym = keysym - GDK_KEY_Left + GDK_KEY_KP_Left;
|
|
|
|
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keysym, 0,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keysym, GDK_CONTROL_MASK,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keypad_keysym, 0,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keypad_keysym, GDK_CONTROL_MASK,
|
|
|
|
"move-focus", 1,
|
|
|
|
GTK_TYPE_DIRECTION_TYPE, direction);
|
|
|
|
}
|
|
|
|
|
2014-10-24 03:38:09 +00:00
|
|
|
static void
|
|
|
|
gtk_popover_menu_class_init (GtkPopoverMenuClass *klass)
|
|
|
|
{
|
|
|
|
GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass);
|
|
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
2014-11-18 12:39:15 +00:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2019-06-12 16:35:36 +00:00
|
|
|
GtkBindingSet *binding_set;
|
2014-11-18 12:39:15 +00:00
|
|
|
|
|
|
|
object_class->set_property = gtk_popover_menu_set_property;
|
|
|
|
object_class->get_property = gtk_popover_menu_get_property;
|
2014-10-24 03:38:09 +00:00
|
|
|
|
|
|
|
widget_class->map = gtk_popover_menu_map;
|
|
|
|
widget_class->unmap = gtk_popover_menu_unmap;
|
2019-06-09 02:25:12 +00:00
|
|
|
widget_class->focus = gtk_popover_menu_focus;
|
2014-10-24 03:38:09 +00:00
|
|
|
|
|
|
|
container_class->add = gtk_popover_menu_add;
|
|
|
|
container_class->remove = gtk_popover_menu_remove;
|
|
|
|
|
2014-11-18 12:39:15 +00:00
|
|
|
g_object_class_install_property (object_class,
|
|
|
|
PROP_VISIBLE_SUBMENU,
|
|
|
|
g_param_spec_string ("visible-submenu",
|
|
|
|
P_("Visible submenu"),
|
|
|
|
P_("The name of the visible submenu"),
|
|
|
|
NULL,
|
|
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
2019-06-12 16:35:36 +00:00
|
|
|
|
|
|
|
binding_set = gtk_binding_set_by_class (klass);
|
|
|
|
|
|
|
|
add_arrow_bindings (binding_set, GDK_KEY_Up, GTK_DIR_UP);
|
|
|
|
add_arrow_bindings (binding_set, GDK_KEY_Down, GTK_DIR_DOWN);
|
|
|
|
add_arrow_bindings (binding_set, GDK_KEY_Left, GTK_DIR_LEFT);
|
|
|
|
add_arrow_bindings (binding_set, GDK_KEY_Right, GTK_DIR_RIGHT);
|
|
|
|
|
|
|
|
add_tab_bindings (binding_set, 0, GTK_DIR_TAB_FORWARD);
|
|
|
|
add_tab_bindings (binding_set, GDK_CONTROL_MASK, GTK_DIR_TAB_FORWARD);
|
|
|
|
add_tab_bindings (binding_set, GDK_SHIFT_MASK, GTK_DIR_TAB_BACKWARD);
|
|
|
|
add_tab_bindings (binding_set, GDK_CONTROL_MASK | GDK_SHIFT_MASK, GTK_DIR_TAB_BACKWARD);
|
2019-06-12 16:55:43 +00:00
|
|
|
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_Return, 0,
|
|
|
|
"activate-default", 0);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_ISO_Enter, 0,
|
|
|
|
"activate-default", 0);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_KP_Enter, 0,
|
|
|
|
"activate-default", 0);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_space, 0,
|
|
|
|
"activate-default", 0);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KEY_KP_Space, 0,
|
|
|
|
"activate-default", 0);
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
|
|
|
|
2014-10-26 23:04:28 +00:00
|
|
|
/**
|
|
|
|
* gtk_popover_menu_new:
|
2019-06-07 17:05:48 +00:00
|
|
|
* @relative_to: (allow-none): #GtkWidget the popover is related to
|
2014-10-26 23:04:28 +00:00
|
|
|
*
|
|
|
|
* Creates a new popover menu.
|
|
|
|
*
|
|
|
|
* Returns: a new #GtkPopoverMenu
|
|
|
|
*/
|
2014-10-24 03:38:09 +00:00
|
|
|
GtkWidget *
|
2019-06-07 17:05:48 +00:00
|
|
|
gtk_popover_menu_new (GtkWidget *relative_to)
|
2014-10-24 03:38:09 +00:00
|
|
|
{
|
2019-06-07 17:05:48 +00:00
|
|
|
GtkWidget *popover;
|
|
|
|
|
|
|
|
g_return_val_if_fail (relative_to == NULL || GTK_IS_WIDGET (relative_to), NULL);
|
|
|
|
|
2019-06-10 23:33:36 +00:00
|
|
|
popover = g_object_new (GTK_TYPE_POPOVER_MENU,
|
|
|
|
"relative-to", relative_to,
|
|
|
|
"autohide", TRUE,
|
|
|
|
NULL);
|
2019-06-07 17:05:48 +00:00
|
|
|
|
|
|
|
return popover;
|
2014-10-24 03:38:09 +00:00
|
|
|
}
|
2014-10-26 23:14:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_popover_menu_open_submenu:
|
|
|
|
* @popover: a #GtkPopoverMenu
|
|
|
|
* @name: the name of the menu to switch to
|
|
|
|
*
|
|
|
|
* Opens a submenu of the @popover. The @name
|
|
|
|
* must be one of the names given to the submenus
|
|
|
|
* of @popover with #GtkPopoverMenu:submenu, or
|
|
|
|
* "main" to switch back to the main menu.
|
|
|
|
*
|
|
|
|
* #GtkModelButton will open submenus automatically
|
|
|
|
* when the #GtkModelButton:menu-name property is set,
|
|
|
|
* so this function is only needed when you are using
|
|
|
|
* other kinds of widgets to initiate menu changes.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_popover_menu_open_submenu (GtkPopoverMenu *popover,
|
|
|
|
const gchar *name)
|
|
|
|
{
|
|
|
|
GtkWidget *stack;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_POPOVER_MENU (popover));
|
|
|
|
|
|
|
|
stack = gtk_bin_get_child (GTK_BIN (popover));
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (stack), name);
|
|
|
|
}
|
2019-03-27 21:52:51 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_popover_menu_add_submenu:
|
|
|
|
* @popover: a #GtkPopoverMenu
|
|
|
|
* @submenu: a widget to add as submenu
|
|
|
|
* @name: the name for the submenu
|
|
|
|
*
|
|
|
|
* Adds a submenu to the popover menu.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
gtk_popover_menu_add_submenu (GtkPopoverMenu *popover,
|
|
|
|
GtkWidget *submenu,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
GtkWidget *stack;
|
|
|
|
|
2019-03-27 22:37:57 +00:00
|
|
|
stack = gtk_bin_get_child (GTK_BIN (popover));
|
2019-03-27 21:52:51 +00:00
|
|
|
|
|
|
|
gtk_stack_add_named (GTK_STACK (stack), submenu, name);
|
2019-06-07 17:05:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_popover_menu_new_from_model:
|
|
|
|
* @relative_to: (allow-none): #GtkWidget the popover is related to
|
|
|
|
* @model: a #GMenuModel
|
|
|
|
*
|
|
|
|
* Creates a #GtkPopoverMenu and populates it according to
|
|
|
|
* @model. The popover is pointed to the @relative_to widget.
|
|
|
|
*
|
|
|
|
* The created buttons are connected to actions found in the
|
|
|
|
* #GtkApplicationWindow to which the popover belongs - typically
|
|
|
|
* by means of being attached to a widget that is contained within
|
|
|
|
* the #GtkApplicationWindows widget hierarchy.
|
|
|
|
*
|
|
|
|
* Actions can also be added using gtk_widget_insert_action_group()
|
|
|
|
* on the menus attach widget or on any of its parent widgets.
|
|
|
|
*
|
2019-08-27 08:28:29 +00:00
|
|
|
* This function creates menus with sliding submenus.
|
|
|
|
* See gtk_popover_menu_new_from_model_full() for a way
|
|
|
|
* to control this.
|
|
|
|
*
|
2019-06-07 17:05:48 +00:00
|
|
|
* Returns: the new #GtkPopoverMenu
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_menu_new_from_model (GtkWidget *relative_to,
|
|
|
|
GMenuModel *model)
|
2019-08-27 08:28:29 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
return gtk_popover_menu_new_from_model_full (relative_to, model, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gtk_popover_menu_new_from_model_full:
|
|
|
|
* @relative_to: (allow-none): #GtkWidget the popover is related to
|
|
|
|
* @model: a #GMenuModel
|
|
|
|
* @flags: flags that affect how the menu is created
|
|
|
|
*
|
|
|
|
* Creates a #GtkPopoverMenu and populates it according to
|
|
|
|
* @model. The popover is pointed to the @relative_to widget.
|
|
|
|
*
|
|
|
|
* The created buttons are connected to actions found in the
|
|
|
|
* #GtkApplicationWindow to which the popover belongs - typically
|
|
|
|
* by means of being attached to a widget that is contained within
|
|
|
|
* the #GtkApplicationWindows widget hierarchy.
|
|
|
|
*
|
|
|
|
* Actions can also be added using gtk_widget_insert_action_group()
|
|
|
|
* on the menus attach widget or on any of its parent widgets.
|
|
|
|
*
|
|
|
|
* The only flag that is supported currently is
|
|
|
|
* #GTK_POPOVER_MENU_NESTED, which makes GTK create traditional,
|
|
|
|
* nested submenus instead of the default sliding submenus.
|
|
|
|
*
|
|
|
|
* Returns: the new #GtkPopoverMenu
|
|
|
|
*/
|
|
|
|
GtkWidget *
|
|
|
|
gtk_popover_menu_new_from_model_full (GtkWidget *relative_to,
|
|
|
|
GMenuModel *model,
|
|
|
|
GtkPopoverMenuFlags flags)
|
2019-06-07 17:05:48 +00:00
|
|
|
{
|
|
|
|
GtkWidget *popover;
|
|
|
|
|
|
|
|
g_return_val_if_fail (relative_to == NULL || GTK_IS_WIDGET (relative_to), NULL);
|
|
|
|
g_return_val_if_fail (G_IS_MENU_MODEL (model), NULL);
|
|
|
|
|
|
|
|
popover = gtk_popover_menu_new (relative_to);
|
2019-08-27 08:28:29 +00:00
|
|
|
gtk_menu_section_box_new_toplevel (GTK_POPOVER_MENU (popover), model, flags);
|
2019-06-07 17:05:48 +00:00
|
|
|
|
|
|
|
return popover;
|
|
|
|
}
|
2019-08-27 08:28:29 +00:00
|
|
|
|