629 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			629 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GTK - The GIMP Toolkit
 | |
|  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
 | |
|  *
 | |
|  * 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/>.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
 | |
|  * file for a list of people on the GTK+ Team.  See the ChangeLog
 | |
|  * files for a list of changes.  These files are distributed with
 | |
|  * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| #include "gtkaccellabel.h"
 | |
| #include "gtkcheckmenuitemprivate.h"
 | |
| #include "gtkmarshalers.h"
 | |
| #include "gtkradiomenuitem.h"
 | |
| #include "deprecated/gtkactivatable.h"
 | |
| #include "gtkprivate.h"
 | |
| #include "gtkintl.h"
 | |
| #include "a11y/gtkradiomenuitemaccessible.h"
 | |
| 
 | |
| /**
 | |
|  * SECTION:gtkradiomenuitem
 | |
|  * @Short_description: A choice from multiple check menu items
 | |
|  * @Title: GtkRadioMenuItem
 | |
|  * @See_also: #GtkMenuItem, #GtkCheckMenuItem
 | |
|  *
 | |
|  * A radio menu item is a check menu item that belongs to a group. At each
 | |
|  * instant exactly one of the radio menu items from a group is selected.
 | |
|  *
 | |
|  * The group list does not need to be freed, as each #GtkRadioMenuItem will
 | |
|  * remove itself and its list item when it is destroyed.
 | |
|  *
 | |
|  * The correct way to create a group of radio menu items is approximatively
 | |
|  * this:
 | |
|  *
 | |
|  * ## How to create a group of radio menu items.
 | |
|  *
 | |
|  * |[<!-- language="C" -->
 | |
|  * GSList *group = NULL;
 | |
|  * GtkWidget *item;
 | |
|  * gint i;
 | |
|  *
 | |
|  * for (i = 0; i < 5; i++)
 | |
|  * {
 | |
|  *   item = gtk_radio_menu_item_new_with_label (group, "This is an example");
 | |
|  *   group = gtk_radio_menu_item_get_group (GTK_RADIO_MENU_ITEM (item));
 | |
|  *   if (i == 1)
 | |
|  *     gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (item), TRUE);
 | |
|  * }
 | |
|  * ]|
 | |
|  *
 | |
|  * # CSS nodes
 | |
|  *
 | |
|  * |[<!-- language="plain" -->
 | |
|  * menuitem
 | |
|  * ├── radio.left
 | |
|  * ╰── <child>
 | |
|  * ]|
 | |
|  *
 | |
|  * GtkRadioMenuItem has a main CSS node with name menuitem, and a subnode
 | |
|  * with name radio, which gets the .left or .right style class.
 | |
|  */
 | |
| 
 | |
| struct _GtkRadioMenuItemPrivate
 | |
| {
 | |
|   GSList *group;
 | |
| };
 | |
| 
 | |
| enum {
 | |
|   PROP_0,
 | |
|   PROP_GROUP
 | |
| };
 | |
| 
 | |
| 
 | |
| static void gtk_radio_menu_item_destroy        (GtkWidget             *widget);
 | |
| static void gtk_radio_menu_item_activate       (GtkMenuItem           *menu_item);
 | |
| static void gtk_radio_menu_item_set_property   (GObject               *object,
 | |
| 						guint                  prop_id,
 | |
| 						const GValue          *value,
 | |
| 						GParamSpec            *pspec);
 | |
| static void gtk_radio_menu_item_get_property   (GObject               *object,
 | |
| 						guint                  prop_id,
 | |
| 						GValue                *value,
 | |
| 						GParamSpec            *pspec);
 | |
| 
 | |
| static guint group_changed_signal = 0;
 | |
| 
 | |
| G_DEFINE_TYPE_WITH_PRIVATE (GtkRadioMenuItem, gtk_radio_menu_item, GTK_TYPE_CHECK_MENU_ITEM)
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new:
 | |
|  * @group: (element-type GtkRadioMenuItem) (allow-none): the group to which the
 | |
|  *   radio menu item is to be attached, or %NULL
 | |
|  *
 | |
|  * Creates a new #GtkRadioMenuItem.
 | |
|  *
 | |
|  * Returns: a new #GtkRadioMenuItem
 | |
|  */
 | |
| GtkWidget*
 | |
| gtk_radio_menu_item_new (GSList *group)
 | |
| {
 | |
|   GtkRadioMenuItem *radio_menu_item;
 | |
| 
 | |
|   radio_menu_item = g_object_new (GTK_TYPE_RADIO_MENU_ITEM, NULL);
 | |
| 
 | |
|   gtk_radio_menu_item_set_group (radio_menu_item, group);
 | |
| 
 | |
|   return GTK_WIDGET (radio_menu_item);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_set_property (GObject      *object,
 | |
| 				  guint         prop_id,
 | |
| 				  const GValue *value,
 | |
| 				  GParamSpec   *pspec)
 | |
| {
 | |
|   GtkRadioMenuItem *radio_menu_item;
 | |
| 
 | |
|   radio_menu_item = GTK_RADIO_MENU_ITEM (object);
 | |
| 
 | |
|   switch (prop_id)
 | |
|     {
 | |
|       GSList *slist;
 | |
| 
 | |
|     case PROP_GROUP:
 | |
|       slist = g_value_get_object (value);
 | |
|       if (slist)
 | |
|         slist = gtk_radio_menu_item_get_group ((GtkRadioMenuItem*) g_value_get_object (value));
 | |
|       gtk_radio_menu_item_set_group (radio_menu_item, slist);
 | |
|       break;
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_get_property (GObject    *object,
 | |
| 				  guint       prop_id,
 | |
| 				  GValue     *value,
 | |
| 				  GParamSpec *pspec)
 | |
| {
 | |
|   switch (prop_id)
 | |
|     {
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | |
|       break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_set_group:
 | |
|  * @radio_menu_item: a #GtkRadioMenuItem.
 | |
|  * @group: (element-type GtkRadioMenuItem) (allow-none): the new group, or %NULL.
 | |
|  *
 | |
|  * Sets the group of a radio menu item, or changes it.
 | |
|  */
 | |
| void
 | |
| gtk_radio_menu_item_set_group (GtkRadioMenuItem *radio_menu_item,
 | |
| 			       GSList           *group)
 | |
| {
 | |
|   GtkRadioMenuItemPrivate *priv;
 | |
|   GtkWidget *old_group_singleton = NULL;
 | |
|   GtkWidget *new_group_singleton = NULL;
 | |
| 
 | |
|   g_return_if_fail (GTK_IS_RADIO_MENU_ITEM (radio_menu_item));
 | |
| 
 | |
|   priv = radio_menu_item->priv;
 | |
| 
 | |
|   if (priv->group == group)
 | |
|     return;
 | |
| 
 | |
|   if (priv->group)
 | |
|     {
 | |
|       GSList *slist;
 | |
| 
 | |
|       priv->group = g_slist_remove (priv->group, radio_menu_item);
 | |
| 
 | |
|       if (priv->group && !priv->group->next)
 | |
| 	old_group_singleton = g_object_ref (priv->group->data);
 | |
| 
 | |
|       for (slist = priv->group; slist; slist = slist->next)
 | |
| 	{
 | |
| 	  GtkRadioMenuItem *tmp_item;
 | |
| 	  
 | |
| 	  tmp_item = slist->data;
 | |
| 
 | |
| 	  tmp_item->priv->group = priv->group;
 | |
| 	}
 | |
|     }
 | |
|   
 | |
|   if (group && !group->next)
 | |
|     new_group_singleton = g_object_ref (group->data);
 | |
| 
 | |
|   priv->group = g_slist_prepend (group, radio_menu_item);
 | |
| 
 | |
|   if (group)
 | |
|     {
 | |
|       GSList *slist;
 | |
|       
 | |
|       for (slist = group; slist; slist = slist->next)
 | |
| 	{
 | |
| 	  GtkRadioMenuItem *tmp_item;
 | |
| 	  
 | |
| 	  tmp_item = slist->data;
 | |
| 
 | |
| 	  tmp_item->priv->group = priv->group;
 | |
| 	}
 | |
| 
 | |
|       _gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (radio_menu_item), FALSE);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       _gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (radio_menu_item), TRUE);
 | |
|       /* gtk_widget_set_state (GTK_WIDGET (radio_menu_item), GTK_STATE_ACTIVE);
 | |
|        */
 | |
|     }
 | |
| 
 | |
|   g_object_ref (radio_menu_item);
 | |
| 
 | |
|   g_object_notify (G_OBJECT (radio_menu_item), "group");
 | |
|   g_signal_emit (radio_menu_item, group_changed_signal, 0);
 | |
|   if (old_group_singleton)
 | |
|     {
 | |
|       g_signal_emit (old_group_singleton, group_changed_signal, 0);
 | |
|       g_object_unref (old_group_singleton);
 | |
|     }
 | |
|   if (new_group_singleton)
 | |
|     {
 | |
|       g_signal_emit (new_group_singleton, group_changed_signal, 0);
 | |
|       g_object_unref (new_group_singleton);
 | |
|     }
 | |
| 
 | |
|   g_object_unref (radio_menu_item);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new_with_label:
 | |
|  * @group: (element-type GtkRadioMenuItem) (allow-none):
 | |
|  *         group the radio menu item is inside, or %NULL
 | |
|  * @label: the text for the label
 | |
|  *
 | |
|  * Creates a new #GtkRadioMenuItem whose child is a simple #GtkLabel.
 | |
|  *
 | |
|  * Returns: (transfer none): A new #GtkRadioMenuItem
 | |
|  */
 | |
| GtkWidget*
 | |
| gtk_radio_menu_item_new_with_label (GSList *group,
 | |
| 				    const gchar *label)
 | |
| {
 | |
|   return g_object_new (GTK_TYPE_RADIO_MENU_ITEM,
 | |
|           "group", (group) ? group->data : NULL,
 | |
|           "label", label,
 | |
|           NULL);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new_with_mnemonic:
 | |
|  * @group: (element-type GtkRadioMenuItem) (allow-none):
 | |
|  *         group the radio menu item is inside, or %NULL
 | |
|  * @label: the text of the button, with an underscore in front of the
 | |
|  *         mnemonic character
 | |
|  *
 | |
|  * Creates a new #GtkRadioMenuItem containing a label. The label
 | |
|  * will be created using gtk_label_new_with_mnemonic(), so underscores
 | |
|  * in @label indicate the mnemonic for the menu item.
 | |
|  *
 | |
|  * Returns: a new #GtkRadioMenuItem
 | |
|  */
 | |
| GtkWidget*
 | |
| gtk_radio_menu_item_new_with_mnemonic (GSList *group,
 | |
| 				       const gchar *label)
 | |
| {
 | |
|   return g_object_new (GTK_TYPE_RADIO_MENU_ITEM,
 | |
|           "group", (group) ? group->data : NULL,
 | |
|           "label", label,
 | |
|           "use-underline", TRUE,
 | |
|           NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new_from_widget: (constructor)
 | |
|  * @group: (allow-none): An existing #GtkRadioMenuItem
 | |
|  *
 | |
|  * Creates a new #GtkRadioMenuItem adding it to the same group as @group.
 | |
|  *
 | |
|  * Returns: (transfer none): The new #GtkRadioMenuItem
 | |
|  *
 | |
|  * Since: 2.4
 | |
|  **/
 | |
| GtkWidget *
 | |
| gtk_radio_menu_item_new_from_widget (GtkRadioMenuItem *group)
 | |
| {
 | |
|   GSList *list = NULL;
 | |
|   
 | |
|   g_return_val_if_fail (group == NULL || GTK_IS_RADIO_MENU_ITEM (group), NULL);
 | |
| 
 | |
|   if (group)
 | |
|     list = gtk_radio_menu_item_get_group (group);
 | |
|   
 | |
|   return gtk_radio_menu_item_new (list);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new_with_mnemonic_from_widget: (constructor)
 | |
|  * @group: (allow-none): An existing #GtkRadioMenuItem
 | |
|  * @label: (allow-none): the text of the button, with an underscore in front of the
 | |
|  *         mnemonic character
 | |
|  *
 | |
|  * Creates a new GtkRadioMenuItem containing a label. The label will be
 | |
|  * created using gtk_label_new_with_mnemonic(), so underscores in label
 | |
|  * indicate the mnemonic for the menu item.
 | |
|  *
 | |
|  * The new #GtkRadioMenuItem is added to the same group as @group.
 | |
|  *
 | |
|  * Returns: (transfer none): The new #GtkRadioMenuItem
 | |
|  *
 | |
|  * Since: 2.4
 | |
|  **/
 | |
| GtkWidget *
 | |
| gtk_radio_menu_item_new_with_mnemonic_from_widget (GtkRadioMenuItem *group,
 | |
| 						   const gchar *label)
 | |
| {
 | |
|   GSList *list = NULL;
 | |
| 
 | |
|   g_return_val_if_fail (group == NULL || GTK_IS_RADIO_MENU_ITEM (group), NULL);
 | |
| 
 | |
|   if (group)
 | |
|     list = gtk_radio_menu_item_get_group (group);
 | |
| 
 | |
|   return gtk_radio_menu_item_new_with_mnemonic (list, label);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_new_with_label_from_widget: (constructor)
 | |
|  * @group: (allow-none): an existing #GtkRadioMenuItem
 | |
|  * @label: (allow-none): the text for the label
 | |
|  *
 | |
|  * Creates a new GtkRadioMenuItem whose child is a simple GtkLabel.
 | |
|  * The new #GtkRadioMenuItem is added to the same group as @group.
 | |
|  *
 | |
|  * Returns: (transfer none): The new #GtkRadioMenuItem
 | |
|  *
 | |
|  * Since: 2.4
 | |
|  **/
 | |
| GtkWidget *
 | |
| gtk_radio_menu_item_new_with_label_from_widget (GtkRadioMenuItem *group,
 | |
| 						const gchar *label)
 | |
| {
 | |
|   GSList *list = NULL;
 | |
| 
 | |
|   g_return_val_if_fail (group == NULL || GTK_IS_RADIO_MENU_ITEM (group), NULL);
 | |
| 
 | |
|   if (group)
 | |
|     list = gtk_radio_menu_item_get_group (group);
 | |
| 
 | |
|   return gtk_radio_menu_item_new_with_label (list, label);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_get_group:
 | |
|  * @radio_menu_item: a #GtkRadioMenuItem
 | |
|  *
 | |
|  * Returns the group to which the radio menu item belongs, as a #GList of
 | |
|  * #GtkRadioMenuItem. The list belongs to GTK+ and should not be freed.
 | |
|  *
 | |
|  * Returns: (element-type GtkRadioMenuItem) (transfer none): the group
 | |
|  *     of @radio_menu_item
 | |
|  */
 | |
| GSList*
 | |
| gtk_radio_menu_item_get_group (GtkRadioMenuItem *radio_menu_item)
 | |
| {
 | |
|   g_return_val_if_fail (GTK_IS_RADIO_MENU_ITEM (radio_menu_item), NULL);
 | |
| 
 | |
|   return radio_menu_item->priv->group;
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_class_init (GtkRadioMenuItemClass *klass)
 | |
| {
 | |
|   GObjectClass *gobject_class;
 | |
|   GtkWidgetClass *widget_class;
 | |
|   GtkMenuItemClass *menu_item_class;
 | |
| 
 | |
|   gobject_class = G_OBJECT_CLASS (klass);
 | |
|   widget_class = GTK_WIDGET_CLASS (klass);
 | |
|   menu_item_class = GTK_MENU_ITEM_CLASS (klass);
 | |
| 
 | |
|   gobject_class->set_property = gtk_radio_menu_item_set_property;
 | |
|   gobject_class->get_property = gtk_radio_menu_item_get_property;
 | |
| 
 | |
|   widget_class->destroy = gtk_radio_menu_item_destroy;
 | |
| 
 | |
|   gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_RADIO_MENU_ITEM_ACCESSIBLE);
 | |
| 
 | |
|   menu_item_class->activate = gtk_radio_menu_item_activate;
 | |
| 
 | |
|   /**
 | |
|    * GtkRadioMenuItem:group:
 | |
|    * 
 | |
|    * The radio menu item whose group this widget belongs to.
 | |
|    * 
 | |
|    * Since: 2.8
 | |
|    */
 | |
|   g_object_class_install_property (gobject_class,
 | |
| 				   PROP_GROUP,
 | |
| 				   g_param_spec_object ("group",
 | |
| 							P_("Group"),
 | |
| 							P_("The radio menu item whose group this widget belongs to."),
 | |
| 							GTK_TYPE_RADIO_MENU_ITEM,
 | |
| 							GTK_PARAM_WRITABLE));
 | |
| 
 | |
|   /**
 | |
|    * GtkStyle::group-changed:
 | |
|    * @style: the object which received the signal
 | |
|    *
 | |
|    * Emitted when the group of radio menu items that a radio menu item belongs
 | |
|    * to changes. This is emitted when a radio menu item switches from
 | |
|    * being alone to being part of a group of 2 or more menu items, or
 | |
|    * vice-versa, and when a button is moved from one group of 2 or
 | |
|    * more menu items ton a different one, but not when the composition
 | |
|    * of the group that a menu item belongs to changes.
 | |
|    *
 | |
|    * Since: 2.4
 | |
|    */
 | |
|   group_changed_signal = g_signal_new (I_("group-changed"),
 | |
| 				       G_OBJECT_CLASS_TYPE (gobject_class),
 | |
| 				       G_SIGNAL_RUN_FIRST,
 | |
| 				       G_STRUCT_OFFSET (GtkRadioMenuItemClass, group_changed),
 | |
| 				       NULL, NULL,
 | |
| 				       _gtk_marshal_VOID__VOID,
 | |
| 				       G_TYPE_NONE, 0);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_init (GtkRadioMenuItem *radio_menu_item)
 | |
| {
 | |
|   GtkRadioMenuItemPrivate *priv;
 | |
| 
 | |
|   radio_menu_item->priv = gtk_radio_menu_item_get_instance_private (radio_menu_item);
 | |
|   priv = radio_menu_item->priv;
 | |
| 
 | |
|   priv->group = g_slist_prepend (NULL, radio_menu_item);
 | |
|   gtk_check_menu_item_set_draw_as_radio (GTK_CHECK_MENU_ITEM (radio_menu_item), TRUE);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_destroy (GtkWidget *widget)
 | |
| {
 | |
|   GtkRadioMenuItem *radio_menu_item = GTK_RADIO_MENU_ITEM (widget);
 | |
|   GtkRadioMenuItemPrivate *priv = radio_menu_item->priv;
 | |
|   GtkWidget *old_group_singleton = NULL;
 | |
|   GtkRadioMenuItem *tmp_menu_item;
 | |
|   GSList *tmp_list;
 | |
|   gboolean was_in_group;
 | |
| 
 | |
|   was_in_group = priv->group && priv->group->next;
 | |
| 
 | |
|   priv->group = g_slist_remove (priv->group, radio_menu_item);
 | |
|   if (priv->group && !priv->group->next)
 | |
|     old_group_singleton = priv->group->data;
 | |
| 
 | |
|   tmp_list = priv->group;
 | |
| 
 | |
|   while (tmp_list)
 | |
|     {
 | |
|       tmp_menu_item = tmp_list->data;
 | |
|       tmp_list = tmp_list->next;
 | |
| 
 | |
|       tmp_menu_item->priv->group = priv->group;
 | |
|     }
 | |
| 
 | |
|   /* this radio menu item is no longer in the group */
 | |
|   priv->group = NULL;
 | |
|   
 | |
|   if (old_group_singleton)
 | |
|     g_signal_emit (old_group_singleton, group_changed_signal, 0);
 | |
|   if (was_in_group)
 | |
|     g_signal_emit (radio_menu_item, group_changed_signal, 0);
 | |
| 
 | |
|   GTK_WIDGET_CLASS (gtk_radio_menu_item_parent_class)->destroy (widget);
 | |
| }
 | |
| 
 | |
| static void
 | |
| gtk_radio_menu_item_activate (GtkMenuItem *menu_item)
 | |
| {
 | |
|   GtkRadioMenuItem *radio_menu_item = GTK_RADIO_MENU_ITEM (menu_item);
 | |
|   GtkRadioMenuItemPrivate *priv = radio_menu_item->priv;
 | |
|   GtkCheckMenuItem *check_menu_item = GTK_CHECK_MENU_ITEM (menu_item);
 | |
|   GtkCheckMenuItem *tmp_menu_item;
 | |
|   GtkAction        *action;
 | |
|   GSList *tmp_list;
 | |
|   gboolean active;
 | |
|   gint toggled;
 | |
| 
 | |
|   G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
 | |
| 
 | |
|   action = gtk_activatable_get_related_action (GTK_ACTIVATABLE (menu_item));
 | |
|   if (action && gtk_menu_item_get_submenu (menu_item) == NULL)
 | |
|     gtk_action_activate (action);
 | |
| 
 | |
|   G_GNUC_END_IGNORE_DEPRECATIONS;
 | |
| 
 | |
|   toggled = FALSE;
 | |
| 
 | |
|   active = gtk_check_menu_item_get_active (check_menu_item);
 | |
|   if (active)
 | |
|     {
 | |
|       tmp_menu_item = NULL;
 | |
|       tmp_list = priv->group;
 | |
| 
 | |
|       while (tmp_list)
 | |
| 	{
 | |
| 	  tmp_menu_item = tmp_list->data;
 | |
| 	  tmp_list = tmp_list->next;
 | |
| 
 | |
|           if (gtk_check_menu_item_get_active (tmp_menu_item) &&
 | |
|               tmp_menu_item != check_menu_item)
 | |
| 	    break;
 | |
| 
 | |
| 	  tmp_menu_item = NULL;
 | |
| 	}
 | |
| 
 | |
|       if (tmp_menu_item)
 | |
| 	{
 | |
| 	  toggled = TRUE;
 | |
|           _gtk_check_menu_item_set_active (check_menu_item, !active);
 | |
| 	}
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       toggled = TRUE;
 | |
|       _gtk_check_menu_item_set_active (check_menu_item, !active);
 | |
| 
 | |
|       tmp_list = priv->group;
 | |
|       while (tmp_list)
 | |
| 	{
 | |
| 	  tmp_menu_item = tmp_list->data;
 | |
| 	  tmp_list = tmp_list->next;
 | |
| 
 | |
|           if (gtk_check_menu_item_get_active (tmp_menu_item) &&
 | |
|               tmp_menu_item != check_menu_item)
 | |
| 	    {
 | |
|               gtk_menu_item_activate (GTK_MENU_ITEM (tmp_menu_item));
 | |
| 	      break;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|   if (toggled)
 | |
|     {
 | |
|       gtk_check_menu_item_toggled (check_menu_item);
 | |
|     }
 | |
| 
 | |
|   gtk_widget_queue_draw (GTK_WIDGET (radio_menu_item));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * gtk_radio_menu_item_join_group:
 | |
|  * @radio_menu_item: a #GtkRadioMenuItem
 | |
|  * @group_source: (allow-none): a #GtkRadioMenuItem whose group we are
 | |
|  *   joining, or %NULL to remove the @radio_menu_item from its current
 | |
|  *   group
 | |
|  *
 | |
|  * Joins a #GtkRadioMenuItem object to the group of another #GtkRadioMenuItem
 | |
|  * object.
 | |
|  *
 | |
|  * This function should be used by language bindings to avoid the memory
 | |
|  * manangement of the opaque #GSList of gtk_radio_menu_item_get_group()
 | |
|  * and gtk_radio_menu_item_set_group().
 | |
|  *
 | |
|  * A common way to set up a group of #GtkRadioMenuItem instances is:
 | |
|  *
 | |
|  * |[
 | |
|  *   GtkRadioMenuItem *last_item = NULL;
 | |
|  *
 | |
|  *   while ( ...more items to add... )
 | |
|  *     {
 | |
|  *       GtkRadioMenuItem *radio_item;
 | |
|  *
 | |
|  *       radio_item = gtk_radio_menu_item_new (...);
 | |
|  *
 | |
|  *       gtk_radio_menu_item_join_group (radio_item, last_item);
 | |
|  *       last_item = radio_item;
 | |
|  *     }
 | |
|  * ]|
 | |
|  *
 | |
|  * Since: 3.18
 | |
|  */
 | |
| void
 | |
| gtk_radio_menu_item_join_group (GtkRadioMenuItem *radio_menu_item,
 | |
|                                 GtkRadioMenuItem *group_source)
 | |
| {
 | |
|   g_return_if_fail (GTK_IS_RADIO_MENU_ITEM (radio_menu_item));
 | |
|   g_return_if_fail (group_source == NULL || GTK_IS_RADIO_MENU_ITEM (group_source));
 | |
| 
 | |
|   if (group_source != NULL)
 | |
|     {
 | |
|       GSList *group = gtk_radio_menu_item_get_group (group_source);
 | |
| 
 | |
|       if (group == NULL)
 | |
|         {
 | |
|           /* if the group source does not have a group, we force one */
 | |
|           gtk_radio_menu_item_set_group (group_source, NULL);
 | |
|           group = gtk_radio_menu_item_get_group (group_source);
 | |
|         }
 | |
| 
 | |
|       gtk_radio_menu_item_set_group (radio_menu_item, group);
 | |
|     }
 | |
|   else
 | |
|     gtk_radio_menu_item_set_group (radio_menu_item, NULL);
 | |
| }
 | 
