Blame gtk/gtkmnemonichash.c

Packit 98cdb6
/* gtkmnemonichash.c: Sets of mnemonics with cycling
Packit 98cdb6
 *
Packit 98cdb6
 * GTK - The GIMP Toolkit
Packit 98cdb6
 * Copyright (C) 2002, Red Hat Inc.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "gtkmnemonichash.h"
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
struct _GtkMnemnonicHash
Packit 98cdb6
{
Packit 98cdb6
  GHashTable *hash;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
GtkMnemonicHash *
Packit 98cdb6
_gtk_mnemonic_hash_new (void)
Packit 98cdb6
{
Packit 98cdb6
  GtkMnemonicHash *mnemonic_hash = g_new (GtkMnemonicHash, 1);
Packit 98cdb6
Packit 98cdb6
  mnemonic_hash->hash = g_hash_table_new (g_direct_hash, NULL);
Packit 98cdb6
Packit 98cdb6
  return mnemonic_hash;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
mnemonic_hash_free_foreach (gpointer	key,
Packit 98cdb6
			    gpointer	value,
Packit 98cdb6
			    gpointer	user)
Packit 98cdb6
{
Packit 98cdb6
  guint keyval = GPOINTER_TO_UINT (key);
Packit 98cdb6
  GSList *targets = value;
Packit 98cdb6
Packit 98cdb6
  gchar *name = gtk_accelerator_name (keyval, 0);
Packit 98cdb6
      
Packit 98cdb6
  g_warning ("mnemonic \"%s\" wasn't removed for widget (%p)",
Packit 98cdb6
	     name, targets->data);
Packit 98cdb6
  g_free (name);
Packit 98cdb6
  
Packit 98cdb6
  g_slist_free (targets);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_mnemonic_hash_free (GtkMnemonicHash *mnemonic_hash)
Packit 98cdb6
{
Packit 98cdb6
  g_hash_table_foreach (mnemonic_hash->hash,
Packit 98cdb6
			mnemonic_hash_free_foreach,
Packit 98cdb6
			NULL);
Packit 98cdb6
Packit 98cdb6
  g_hash_table_destroy (mnemonic_hash->hash);
Packit 98cdb6
  g_free (mnemonic_hash);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_mnemonic_hash_add (GtkMnemonicHash *mnemonic_hash,
Packit 98cdb6
			guint            keyval,
Packit 98cdb6
			GtkWidget       *target)
Packit 98cdb6
{
Packit 98cdb6
  gpointer key = GUINT_TO_POINTER (keyval);
Packit 98cdb6
  GSList *targets, *new_targets;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (GTK_IS_WIDGET (target));
Packit 98cdb6
  
Packit 98cdb6
  targets = g_hash_table_lookup (mnemonic_hash->hash, key);
Packit 98cdb6
  g_return_if_fail (g_slist_find (targets, target) == NULL);
Packit 98cdb6
Packit 98cdb6
  new_targets = g_slist_append (targets, target);
Packit 98cdb6
  if (new_targets != targets)
Packit 98cdb6
    g_hash_table_insert (mnemonic_hash->hash, key, new_targets);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_mnemonic_hash_remove (GtkMnemonicHash *mnemonic_hash,
Packit 98cdb6
			   guint           keyval,
Packit 98cdb6
			   GtkWidget      *target)
Packit 98cdb6
{
Packit 98cdb6
  gpointer key = GUINT_TO_POINTER (keyval);
Packit 98cdb6
  GSList *targets, *new_targets;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (GTK_IS_WIDGET (target));
Packit 98cdb6
  
Packit 98cdb6
  targets = g_hash_table_lookup (mnemonic_hash->hash, key);
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (targets && g_slist_find (targets, target) != NULL);
Packit 98cdb6
Packit 98cdb6
  new_targets = g_slist_remove (targets, target);
Packit 98cdb6
  if (new_targets != targets)
Packit 98cdb6
    {
Packit 98cdb6
      if (new_targets == NULL)
Packit 98cdb6
	g_hash_table_remove (mnemonic_hash->hash, key);
Packit 98cdb6
      else
Packit 98cdb6
	g_hash_table_insert (mnemonic_hash->hash, key, new_targets);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gtk_mnemonic_hash_activate (GtkMnemonicHash *mnemonic_hash,
Packit 98cdb6
			     guint            keyval)
Packit 98cdb6
{
Packit 98cdb6
  GSList *list, *targets;
Packit 98cdb6
  GtkWidget *widget, *chosen_widget;
Packit 98cdb6
  gboolean overloaded;
Packit 98cdb6
Packit 98cdb6
  targets = g_hash_table_lookup (mnemonic_hash->hash,
Packit 98cdb6
				 GUINT_TO_POINTER (keyval));
Packit 98cdb6
  if (!targets)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
  
Packit 98cdb6
  overloaded = FALSE;
Packit 98cdb6
  chosen_widget = NULL;
Packit 98cdb6
  for (list = targets; list; list = list->next)
Packit 98cdb6
    {
Packit 98cdb6
      widget = GTK_WIDGET (list->data);
Packit 98cdb6
      
Packit 98cdb6
      if (gtk_widget_is_sensitive (widget) &&
Packit 98cdb6
	  gtk_widget_get_mapped (widget) &&
Packit 98cdb6
          widget->window &&
Packit 98cdb6
	  gdk_window_is_viewable (widget->window))
Packit 98cdb6
	{
Packit 98cdb6
	  if (chosen_widget)
Packit 98cdb6
	    {
Packit 98cdb6
	      overloaded = TRUE;
Packit 98cdb6
	      break;
Packit 98cdb6
	    }
Packit 98cdb6
	  else
Packit 98cdb6
	    chosen_widget = widget;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (chosen_widget)
Packit 98cdb6
    {
Packit 98cdb6
      /* For round robin we put the activated entry on
Packit 98cdb6
       * the end of the list after activation
Packit 98cdb6
       */
Packit 98cdb6
      targets = g_slist_remove (targets, chosen_widget);
Packit 98cdb6
      targets = g_slist_append (targets, chosen_widget);
Packit 98cdb6
      g_hash_table_insert (mnemonic_hash->hash,
Packit 98cdb6
			   GUINT_TO_POINTER (keyval),
Packit 98cdb6
			   targets);
Packit 98cdb6
Packit 98cdb6
      return gtk_widget_mnemonic_activate (chosen_widget, overloaded);
Packit 98cdb6
    }
Packit 98cdb6
  return FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GSList *
Packit 98cdb6
_gtk_mnemonic_hash_lookup (GtkMnemonicHash *mnemonic_hash,
Packit 98cdb6
			   guint            keyval)
Packit 98cdb6
{
Packit 98cdb6
  return g_hash_table_lookup (mnemonic_hash->hash, GUINT_TO_POINTER (keyval));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
mnemonic_hash_foreach_func (gpointer key,
Packit 98cdb6
			    gpointer value,
Packit 98cdb6
			    gpointer data)
Packit 98cdb6
{
Packit 98cdb6
  struct {
Packit 98cdb6
    GtkMnemonicHashForeach func;
Packit 98cdb6
    gpointer func_data;
Packit 98cdb6
  } *info = data;
Packit 98cdb6
Packit 98cdb6
  guint keyval = GPOINTER_TO_UINT (key);
Packit 98cdb6
  GSList *targets = value;
Packit 98cdb6
  
Packit 98cdb6
  (*info->func) (keyval, targets, info->func_data);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_mnemonic_hash_foreach (GtkMnemonicHash       *mnemonic_hash,
Packit 98cdb6
			    GtkMnemonicHashForeach func,
Packit 98cdb6
			    gpointer               func_data)
Packit 98cdb6
{
Packit 98cdb6
  struct {
Packit 98cdb6
    GtkMnemonicHashForeach func;
Packit 98cdb6
    gpointer func_data;
Packit 98cdb6
  } info;
Packit 98cdb6
  
Packit 98cdb6
  info.func = func;
Packit 98cdb6
  info.func_data = func_data;
Packit 98cdb6
Packit 98cdb6
  g_hash_table_foreach (mnemonic_hash->hash,
Packit 98cdb6
			mnemonic_hash_foreach_func,
Packit 98cdb6
			&info;;
Packit 98cdb6
}