mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-16 07:04:29 +00:00
306 lines
9.9 KiB
C
306 lines
9.9 KiB
C
/*
|
|
* Copyright © 2011 Canonical Limited
|
|
*
|
|
* 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 licence, 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, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Author: Ryan Lortie <desrt@desrt.ca>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkmodelmenuitem.h"
|
|
|
|
#include "gtkaccelmapprivate.h"
|
|
#include "gtkmodelmenu.h"
|
|
|
|
struct _GtkModelMenuItem
|
|
{
|
|
GtkCheckMenuItem parent_instance;
|
|
|
|
GActionGroup *actions;
|
|
const gchar *action_name;
|
|
gboolean has_indicator;
|
|
gboolean can_activate;
|
|
GVariant *target;
|
|
};
|
|
|
|
typedef GtkCheckMenuItemClass GtkModelMenuItemClass;
|
|
|
|
static void gtk_model_menu_item_observer_iface_init (GActionObserverInterface *iface);
|
|
G_DEFINE_TYPE_WITH_CODE (GtkModelMenuItem, gtk_model_menu_item, GTK_TYPE_CHECK_MENU_ITEM,
|
|
G_IMPLEMENT_INTERFACE (G_TYPE_ACTION_OBSERVER, gtk_model_menu_item_observer_iface_init))
|
|
|
|
static void
|
|
gtk_model_menu_item_activate (GtkMenuItem *menu_item)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (menu_item);
|
|
|
|
if (item->can_activate)
|
|
g_action_group_activate_action (item->actions, item->action_name, item->target);
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_toggle_size_request (GtkMenuItem *menu_item,
|
|
gint *requisition)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (menu_item);
|
|
|
|
if (item->has_indicator)
|
|
GTK_MENU_ITEM_CLASS (gtk_model_menu_item_parent_class)
|
|
->toggle_size_request (menu_item, requisition);
|
|
|
|
else
|
|
*requisition = 0;
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_draw_indicator (GtkCheckMenuItem *check_item,
|
|
cairo_t *cr)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (check_item);
|
|
|
|
if (item->has_indicator)
|
|
GTK_CHECK_MENU_ITEM_CLASS (gtk_model_menu_item_parent_class)
|
|
->draw_indicator (check_item, cr);
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_set_active (GtkModelMenuItem *item,
|
|
gboolean active)
|
|
{
|
|
GtkCheckMenuItem *checkitem = GTK_CHECK_MENU_ITEM (item);
|
|
|
|
if (gtk_check_menu_item_get_active (checkitem) != active)
|
|
{
|
|
_gtk_check_menu_item_set_active (checkitem, active);
|
|
g_object_notify (G_OBJECT (checkitem), "active");
|
|
gtk_check_menu_item_toggled (checkitem);
|
|
gtk_widget_queue_draw (GTK_WIDGET (item));
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_action_added (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
const GVariantType *parameter_type,
|
|
gboolean enabled,
|
|
GVariant *state)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (observer);
|
|
|
|
/* we can only activate the item if we have the correct type of parameter */
|
|
item->can_activate = (item->target == NULL && parameter_type == NULL) ||
|
|
(item->target != NULL && parameter_type != NULL &&
|
|
g_variant_is_of_type (item->target, parameter_type));
|
|
|
|
if (item->can_activate)
|
|
{
|
|
if (item->target != NULL && state != NULL)
|
|
{
|
|
/* actions with states and targets are radios */
|
|
gboolean selected;
|
|
|
|
selected = g_variant_equal (state, item->target);
|
|
gtk_check_menu_item_set_draw_as_radio (GTK_CHECK_MENU_ITEM (item), TRUE);
|
|
gtk_model_menu_item_set_active (item, selected);
|
|
item->has_indicator = TRUE;
|
|
}
|
|
|
|
else if (state != NULL && g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
|
|
{
|
|
/* boolean state actions without target are checks */
|
|
gtk_check_menu_item_set_draw_as_radio (GTK_CHECK_MENU_ITEM (item), FALSE);
|
|
gtk_model_menu_item_set_active (item, g_variant_get_boolean (state));
|
|
item->has_indicator = TRUE;
|
|
}
|
|
|
|
else
|
|
{
|
|
/* stateless items are just plain actions */
|
|
gtk_model_menu_item_set_active (item, FALSE);
|
|
item->has_indicator = FALSE;
|
|
}
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (item), enabled);
|
|
gtk_widget_queue_resize (GTK_WIDGET (item));
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_action_enabled_changed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
gboolean enabled)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (observer);
|
|
|
|
if (!item->can_activate)
|
|
return;
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (item), item->can_activate && enabled);
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_action_state_changed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name,
|
|
GVariant *state)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (observer);
|
|
|
|
if (!item->can_activate)
|
|
return;
|
|
|
|
if (item->target)
|
|
gtk_model_menu_item_set_active (item, g_variant_equal (state, item->target));
|
|
|
|
else if (g_variant_is_of_type (state, G_VARIANT_TYPE_BOOLEAN))
|
|
gtk_model_menu_item_set_active (item, g_variant_get_boolean (state));
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_action_removed (GActionObserver *observer,
|
|
GActionObservable *observable,
|
|
const gchar *action_name)
|
|
{
|
|
GtkModelMenuItem *item = GTK_MODEL_MENU_ITEM (observer);
|
|
|
|
if (!item->can_activate)
|
|
return;
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (item), FALSE);
|
|
gtk_model_menu_item_set_active (item, FALSE);
|
|
item->has_indicator = FALSE;
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (item));
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_setup (GtkModelMenuItem *item,
|
|
GMenuModel *model,
|
|
gint item_index,
|
|
GActionObservable *actions,
|
|
GtkAccelGroup *accels)
|
|
{
|
|
GMenuAttributeIter *iter;
|
|
GMenuModel *submenu;
|
|
const gchar *key;
|
|
GVariant *value;
|
|
|
|
if ((submenu = g_menu_model_get_item_link (model, item_index, "submenu")))
|
|
{
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), gtk_model_menu_create_menu (submenu, actions, accels));
|
|
g_object_unref (submenu);
|
|
}
|
|
|
|
iter = g_menu_model_iterate_item_attributes (model, item_index);
|
|
while (g_menu_attribute_iter_get_next (iter, &key, &value))
|
|
{
|
|
if (g_str_equal (key, "label") && g_variant_is_of_type (value, G_VARIANT_TYPE_STRING))
|
|
gtk_menu_item_set_label (GTK_MENU_ITEM (item), g_variant_get_string (value, NULL));
|
|
|
|
else if (g_str_equal (key, "action") && g_variant_is_of_type (value, G_VARIANT_TYPE_STRING))
|
|
item->action_name = g_variant_get_string (value, NULL);
|
|
|
|
else if (g_str_equal (key, "target"))
|
|
item->target = g_variant_ref (value);
|
|
|
|
g_variant_unref (value);
|
|
}
|
|
g_object_unref (iter);
|
|
|
|
gtk_menu_item_set_use_underline (GTK_MENU_ITEM (item), TRUE);
|
|
|
|
if (item->action_name)
|
|
{
|
|
const GVariantType *type;
|
|
gboolean enabled;
|
|
GVariant *state;
|
|
gchar *path;
|
|
|
|
/* observer already causes us to hold a hard ref on the group */
|
|
item->actions = G_ACTION_GROUP (actions);
|
|
|
|
g_action_observable_register_observer (actions, item->action_name, G_ACTION_OBSERVER (item));
|
|
|
|
if (g_action_group_query_action (G_ACTION_GROUP (actions), item->action_name, &enabled, &type, NULL, NULL, &state))
|
|
{
|
|
gtk_model_menu_item_action_added (G_ACTION_OBSERVER (item), actions, item->action_name, type, enabled, state);
|
|
if (state != NULL)
|
|
g_variant_unref (state);
|
|
}
|
|
|
|
else
|
|
gtk_widget_set_sensitive (GTK_WIDGET (item), FALSE);
|
|
|
|
path = _gtk_accel_path_for_action (item->action_name, item->target);
|
|
gtk_menu_item_set_accel_path (GTK_MENU_ITEM (item), path);
|
|
g_free (path);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_finalize (GObject *object)
|
|
{
|
|
G_OBJECT_CLASS (gtk_model_menu_item_parent_class)
|
|
->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_init (GtkModelMenuItem *item)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_observer_iface_init (GActionObserverInterface *iface)
|
|
{
|
|
iface->action_added = gtk_model_menu_item_action_added;
|
|
iface->action_enabled_changed = gtk_model_menu_item_action_enabled_changed;
|
|
iface->action_state_changed = gtk_model_menu_item_action_state_changed;
|
|
iface->action_removed = gtk_model_menu_item_action_removed;
|
|
}
|
|
|
|
static void
|
|
gtk_model_menu_item_class_init (GtkModelMenuItemClass *class)
|
|
{
|
|
GtkCheckMenuItemClass *check_class = GTK_CHECK_MENU_ITEM_CLASS (class);
|
|
GtkMenuItemClass *item_class = GTK_MENU_ITEM_CLASS (class);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
check_class->draw_indicator = gtk_model_menu_item_draw_indicator;
|
|
|
|
item_class->activate = gtk_model_menu_item_activate;
|
|
item_class->toggle_size_request = gtk_model_menu_item_toggle_size_request;
|
|
|
|
object_class->finalize = gtk_model_menu_item_finalize;
|
|
}
|
|
|
|
GtkMenuItem *
|
|
gtk_model_menu_item_new (GMenuModel *model,
|
|
gint item_index,
|
|
GActionObservable *actions,
|
|
GtkAccelGroup *accels)
|
|
{
|
|
GtkModelMenuItem *item;
|
|
|
|
item = g_object_new (GTK_TYPE_MODEL_MENU_ITEM, NULL);
|
|
|
|
gtk_model_menu_item_setup (item, model, item_index, actions, accels);
|
|
|
|
return GTK_MENU_ITEM (item);
|
|
}
|