This reverts commit 7cecc8e524.
It is impossible to use the selection object of the menu while it is
collapsed (collapsed menus deselect everything for a start), so even
though the original patch was correct, the followup issues are too big
to solve this quickly to a release.
https://bugzilla.gnome.org/show_bug.cgi?id=707786
		
	
		
			
				
	
	
		
			436 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			436 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* GTK+ - accessibility implementations
 | 
						|
 * Copyright 2004 Sun Microsystems 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 <glib/gi18n-lib.h>
 | 
						|
#include <gtk/gtk.h>
 | 
						|
#include "gtkcomboboxaccessible.h"
 | 
						|
 | 
						|
struct _GtkComboBoxAccessiblePrivate
 | 
						|
{
 | 
						|
  gchar         *name;
 | 
						|
  gint           old_selection;
 | 
						|
  gboolean       popup_set;
 | 
						|
};
 | 
						|
 | 
						|
static void atk_action_interface_init    (AtkActionIface    *iface);
 | 
						|
static void atk_selection_interface_init (AtkSelectionIface *iface);
 | 
						|
 | 
						|
G_DEFINE_TYPE_WITH_CODE (GtkComboBoxAccessible, gtk_combo_box_accessible, GTK_TYPE_CONTAINER_ACCESSIBLE,
 | 
						|
                         G_ADD_PRIVATE (GtkComboBoxAccessible)
 | 
						|
                         G_IMPLEMENT_INTERFACE (ATK_TYPE_ACTION, atk_action_interface_init)
 | 
						|
                         G_IMPLEMENT_INTERFACE (ATK_TYPE_SELECTION, atk_selection_interface_init))
 | 
						|
 | 
						|
static void
 | 
						|
changed_cb (GtkWidget *widget)
 | 
						|
{
 | 
						|
  GtkComboBox *combo_box;
 | 
						|
  AtkObject *obj;
 | 
						|
  GtkComboBoxAccessible *accessible;
 | 
						|
  gint index;
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX (widget);
 | 
						|
 | 
						|
  index = gtk_combo_box_get_active (combo_box);
 | 
						|
  obj = gtk_widget_get_accessible (widget);
 | 
						|
  accessible = GTK_COMBO_BOX_ACCESSIBLE (obj);
 | 
						|
  if (accessible->priv->old_selection != index)
 | 
						|
    {
 | 
						|
      accessible->priv->old_selection = index;
 | 
						|
      g_object_notify (G_OBJECT (obj), "accessible-name");
 | 
						|
      g_signal_emit_by_name (obj, "selection-changed");
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_combo_box_accessible_initialize (AtkObject *obj,
 | 
						|
                                     gpointer   data)
 | 
						|
{
 | 
						|
  GtkComboBox *combo_box;
 | 
						|
  GtkComboBoxAccessible *accessible;
 | 
						|
  AtkObject *popup;
 | 
						|
 | 
						|
  ATK_OBJECT_CLASS (gtk_combo_box_accessible_parent_class)->initialize (obj, data);
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX (data);
 | 
						|
  accessible = GTK_COMBO_BOX_ACCESSIBLE (obj);
 | 
						|
 | 
						|
  g_signal_connect (combo_box, "changed", G_CALLBACK (changed_cb), NULL);
 | 
						|
  accessible->priv->old_selection = gtk_combo_box_get_active (combo_box);
 | 
						|
 | 
						|
  popup = gtk_combo_box_get_popup_accessible (combo_box);
 | 
						|
  if (popup)
 | 
						|
    {
 | 
						|
      atk_object_set_parent (popup, obj);
 | 
						|
      accessible->priv->popup_set = TRUE;
 | 
						|
    }
 | 
						|
  if (gtk_combo_box_get_has_entry (combo_box))
 | 
						|
    atk_object_set_parent (gtk_widget_get_accessible (gtk_bin_get_child (GTK_BIN (combo_box))), obj);
 | 
						|
 | 
						|
  obj->role = ATK_ROLE_COMBO_BOX;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_combo_box_accessible_finalize (GObject *object)
 | 
						|
{
 | 
						|
  GtkComboBoxAccessible *combo_box = GTK_COMBO_BOX_ACCESSIBLE (object);
 | 
						|
 | 
						|
  g_free (combo_box->priv->name);
 | 
						|
 | 
						|
  G_OBJECT_CLASS (gtk_combo_box_accessible_parent_class)->finalize (object);
 | 
						|
}
 | 
						|
 | 
						|
static const gchar *
 | 
						|
gtk_combo_box_accessible_get_name (AtkObject *obj)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
  GtkComboBox *combo_box;
 | 
						|
  GtkComboBoxAccessible *accessible;
 | 
						|
  GtkTreeIter iter;
 | 
						|
  const gchar *name;
 | 
						|
  GtkTreeModel *model;
 | 
						|
  gint n_columns;
 | 
						|
  gint i;
 | 
						|
 | 
						|
  name = ATK_OBJECT_CLASS (gtk_combo_box_accessible_parent_class)->get_name (obj);
 | 
						|
  if (name)
 | 
						|
    return name;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (obj));
 | 
						|
  if (widget == NULL)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX (widget);
 | 
						|
  accessible = GTK_COMBO_BOX_ACCESSIBLE (obj);
 | 
						|
  if (gtk_combo_box_get_active_iter (combo_box, &iter))
 | 
						|
    {
 | 
						|
      model = gtk_combo_box_get_model (combo_box);
 | 
						|
      n_columns = gtk_tree_model_get_n_columns (model);
 | 
						|
      for (i = 0; i < n_columns; i++)
 | 
						|
        {
 | 
						|
          GValue value = G_VALUE_INIT;
 | 
						|
 | 
						|
          gtk_tree_model_get_value (model, &iter, i, &value);
 | 
						|
          if (G_VALUE_HOLDS_STRING (&value))
 | 
						|
            {
 | 
						|
              g_free (accessible->priv->name);
 | 
						|
              accessible->priv->name =  g_strdup (g_value_get_string (&value));
 | 
						|
              g_value_unset (&value);
 | 
						|
              break;
 | 
						|
            }
 | 
						|
          else
 | 
						|
            g_value_unset (&value);
 | 
						|
        }
 | 
						|
    }
 | 
						|
  return accessible->priv->name;
 | 
						|
}
 | 
						|
 | 
						|
static gint
 | 
						|
gtk_combo_box_accessible_get_n_children (AtkObject* obj)
 | 
						|
{
 | 
						|
  gint n_children = 0;
 | 
						|
  GtkWidget *widget;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (obj));
 | 
						|
  if (widget == NULL)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  n_children++;
 | 
						|
  if (gtk_combo_box_get_has_entry (GTK_COMBO_BOX (widget)))
 | 
						|
    n_children++;
 | 
						|
 | 
						|
  return n_children;
 | 
						|
}
 | 
						|
 | 
						|
static AtkObject *
 | 
						|
gtk_combo_box_accessible_ref_child (AtkObject *obj,
 | 
						|
                                    gint       i)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
  AtkObject *child;
 | 
						|
  GtkComboBoxAccessible *box;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (obj));
 | 
						|
  if (widget == NULL)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  if (i == 0)
 | 
						|
    {
 | 
						|
      child = gtk_combo_box_get_popup_accessible (GTK_COMBO_BOX (widget));
 | 
						|
      box = GTK_COMBO_BOX_ACCESSIBLE (obj);
 | 
						|
      if (!box->priv->popup_set)
 | 
						|
        {
 | 
						|
          atk_object_set_parent (child, obj);
 | 
						|
          box->priv->popup_set = TRUE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
  else if (i == 1 && gtk_combo_box_get_has_entry (GTK_COMBO_BOX (widget)))
 | 
						|
    {
 | 
						|
      child = gtk_widget_get_accessible (gtk_bin_get_child (GTK_BIN (widget)));
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      return NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  return g_object_ref (child);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_combo_box_accessible_class_init (GtkComboBoxAccessibleClass *klass)
 | 
						|
{
 | 
						|
  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
 | 
						|
  AtkObjectClass *class = ATK_OBJECT_CLASS (klass);
 | 
						|
 | 
						|
  gobject_class->finalize = gtk_combo_box_accessible_finalize;
 | 
						|
 | 
						|
  class->get_name = gtk_combo_box_accessible_get_name;
 | 
						|
  class->get_n_children = gtk_combo_box_accessible_get_n_children;
 | 
						|
  class->ref_child = gtk_combo_box_accessible_ref_child;
 | 
						|
  class->initialize = gtk_combo_box_accessible_initialize;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_combo_box_accessible_init (GtkComboBoxAccessible *combo_box)
 | 
						|
{
 | 
						|
  combo_box->priv = gtk_combo_box_accessible_get_instance_private (combo_box);
 | 
						|
  combo_box->priv->old_selection = -1;
 | 
						|
  combo_box->priv->name = NULL;
 | 
						|
  combo_box->priv->popup_set = FALSE;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_combo_box_accessible_do_action (AtkAction *action,
 | 
						|
                                    gint       i)
 | 
						|
{
 | 
						|
  GtkComboBox *combo_box;
 | 
						|
  GtkWidget *widget;
 | 
						|
  gboolean popup_shown;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (action));
 | 
						|
  if (widget == NULL)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  if (!gtk_widget_get_sensitive (widget) || !gtk_widget_get_visible (widget))
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  if (i != 0)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX (widget);
 | 
						|
  g_object_get (combo_box, "popup-shown", &popup_shown, NULL);
 | 
						|
  if (popup_shown)
 | 
						|
    gtk_combo_box_popdown (combo_box);
 | 
						|
  else
 | 
						|
    gtk_combo_box_popup (combo_box);
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static gint
 | 
						|
gtk_combo_box_accessible_get_n_actions (AtkAction *action)
 | 
						|
{
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
 | 
						|
static const gchar *
 | 
						|
gtk_combo_box_accessible_get_keybinding (AtkAction *action,
 | 
						|
                                         gint       i)
 | 
						|
{
 | 
						|
  GtkComboBoxAccessible *combo_box;
 | 
						|
  GtkWidget *widget;
 | 
						|
  GtkWidget *label;
 | 
						|
  AtkRelationSet *set;
 | 
						|
  AtkRelation *relation;
 | 
						|
  GPtrArray *target;
 | 
						|
  gpointer target_object;
 | 
						|
  guint key_val;
 | 
						|
  gchar *return_value = NULL;
 | 
						|
 | 
						|
  if (i != 0)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX_ACCESSIBLE (action);
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (combo_box));
 | 
						|
  if (widget == NULL)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  set = atk_object_ref_relation_set (ATK_OBJECT (action));
 | 
						|
  if (set == NULL)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  label = NULL;
 | 
						|
  relation = atk_relation_set_get_relation_by_type (set, ATK_RELATION_LABELLED_BY);
 | 
						|
  if (relation)
 | 
						|
    {
 | 
						|
      target = atk_relation_get_target (relation);
 | 
						|
      target_object = g_ptr_array_index (target, 0);
 | 
						|
      label = gtk_accessible_get_widget (GTK_ACCESSIBLE (target_object));
 | 
						|
    }
 | 
						|
  g_object_unref (set);
 | 
						|
  if (GTK_IS_LABEL (label))
 | 
						|
    {
 | 
						|
      key_val = gtk_label_get_mnemonic_keyval (GTK_LABEL (label));
 | 
						|
      if (key_val != GDK_KEY_VoidSymbol)
 | 
						|
        return_value = gtk_accelerator_name (key_val, GDK_MOD1_MASK);
 | 
						|
    }
 | 
						|
 | 
						|
  return return_value;
 | 
						|
}
 | 
						|
 | 
						|
static const gchar *
 | 
						|
gtk_combo_box_accessible_action_get_name (AtkAction *action,
 | 
						|
                                          gint       i)
 | 
						|
{
 | 
						|
  if (i == 0)
 | 
						|
    return "press";
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static const gchar *
 | 
						|
gtk_combo_box_accessible_action_get_localized_name (AtkAction *action,
 | 
						|
                                                    gint       i)
 | 
						|
{
 | 
						|
  if (i == 0)
 | 
						|
    return C_("Action name", "Press");
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static const gchar *
 | 
						|
gtk_combo_box_accessible_action_get_description (AtkAction *action,
 | 
						|
                                                 gint       i)
 | 
						|
{
 | 
						|
  if (i == 0)
 | 
						|
    return C_("Action description", "Presses the combobox");
 | 
						|
  return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
atk_action_interface_init (AtkActionIface *iface)
 | 
						|
{
 | 
						|
  iface->do_action = gtk_combo_box_accessible_do_action;
 | 
						|
  iface->get_n_actions = gtk_combo_box_accessible_get_n_actions;
 | 
						|
  iface->get_keybinding = gtk_combo_box_accessible_get_keybinding;
 | 
						|
  iface->get_name = gtk_combo_box_accessible_action_get_name;
 | 
						|
  iface->get_localized_name = gtk_combo_box_accessible_action_get_localized_name;
 | 
						|
  iface->get_description = gtk_combo_box_accessible_action_get_description;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_combo_box_accessible_add_selection (AtkSelection *selection,
 | 
						|
                                        gint          i)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (selection));
 | 
						|
  if (widget == NULL)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  gtk_combo_box_set_active (GTK_COMBO_BOX (widget), i);
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_combo_box_accessible_clear_selection (AtkSelection *selection)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (selection));
 | 
						|
  if (widget == NULL)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  gtk_combo_box_set_active (GTK_COMBO_BOX (widget), -1);
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static AtkObject *
 | 
						|
gtk_combo_box_accessible_ref_selection (AtkSelection *selection,
 | 
						|
                                        gint          i)
 | 
						|
{
 | 
						|
  GtkComboBox *combo_box;
 | 
						|
  GtkWidget *widget;
 | 
						|
  AtkObject *obj;
 | 
						|
  gint index;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (selection));
 | 
						|
  if (widget == NULL)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  if (i != 0)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  combo_box = GTK_COMBO_BOX (widget);
 | 
						|
 | 
						|
  obj = gtk_combo_box_get_popup_accessible (combo_box);
 | 
						|
  index = gtk_combo_box_get_active (combo_box);
 | 
						|
 | 
						|
  return atk_object_ref_accessible_child (obj, index);
 | 
						|
}
 | 
						|
 | 
						|
static gint
 | 
						|
gtk_combo_box_accessible_get_selection_count (AtkSelection *selection)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (selection));
 | 
						|
  if (widget == NULL)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  return (gtk_combo_box_get_active (GTK_COMBO_BOX (widget)) == -1) ? 0 : 1;
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_combo_box_accessible_is_child_selected (AtkSelection *selection,
 | 
						|
                                            gint          i)
 | 
						|
{
 | 
						|
  GtkWidget *widget;
 | 
						|
  gint j;
 | 
						|
 | 
						|
  widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (selection));
 | 
						|
 | 
						|
  if (widget == NULL)
 | 
						|
    return FALSE;
 | 
						|
 | 
						|
  j = gtk_combo_box_get_active (GTK_COMBO_BOX (widget));
 | 
						|
 | 
						|
  return (j == i);
 | 
						|
}
 | 
						|
 | 
						|
static gboolean
 | 
						|
gtk_combo_box_accessible_remove_selection (AtkSelection *selection,
 | 
						|
                                           gint          i)
 | 
						|
{
 | 
						|
  if (atk_selection_is_child_selected (selection, i))
 | 
						|
    atk_selection_clear_selection (selection);
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
atk_selection_interface_init (AtkSelectionIface *iface)
 | 
						|
{
 | 
						|
  iface->add_selection = gtk_combo_box_accessible_add_selection;
 | 
						|
  iface->clear_selection = gtk_combo_box_accessible_clear_selection;
 | 
						|
  iface->ref_selection = gtk_combo_box_accessible_ref_selection;
 | 
						|
  iface->get_selection_count = gtk_combo_box_accessible_get_selection_count;
 | 
						|
  iface->is_child_selected = gtk_combo_box_accessible_is_child_selected;
 | 
						|
  iface->remove_selection = gtk_combo_box_accessible_remove_selection;
 | 
						|
}
 |