Blame gtk/gtkaccelmap.c

Packit 98cdb6
/* GTK - The GIMP Toolkit
Packit 98cdb6
 * Copyright (C) 1998, 2001 Tim Janik
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 "config.h"
Packit 98cdb6
Packit 98cdb6
#include "gtkaccelmap.h"
Packit 98cdb6
Packit 98cdb6
#include "gtkmarshalers.h"
Packit 98cdb6
#include "gtkwindow.h"  /* in lack of GtkAcceleratable */
Packit 98cdb6
#include "gtkintl.h" 
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
#include <glib/gstdio.h>
Packit 98cdb6
Packit 98cdb6
#include <string.h>
Packit 98cdb6
#include <errno.h>
Packit 98cdb6
#include <fcntl.h>
Packit 98cdb6
#ifdef HAVE_UNISTD_H
Packit 98cdb6
#include <unistd.h>
Packit 98cdb6
#endif
Packit 98cdb6
#ifdef G_OS_WIN32
Packit 98cdb6
#include <io.h>
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
/* --- structures --- */
Packit 98cdb6
struct _GtkAccelMap
Packit 98cdb6
{
Packit 98cdb6
  GObject parent_instance;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
struct _GtkAccelMapClass
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass parent_class;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
typedef struct {
Packit 98cdb6
  const gchar *accel_path;
Packit 98cdb6
  guint        accel_key;
Packit 98cdb6
  guint        accel_mods;
Packit 98cdb6
  guint	       std_accel_key;
Packit 98cdb6
  guint	       std_accel_mods;
Packit 98cdb6
  guint        changed    :  1;
Packit 98cdb6
  guint        lock_count : 15;
Packit 98cdb6
  GSList      *groups;
Packit 98cdb6
} AccelEntry;
Packit 98cdb6
Packit 98cdb6
/* --- signals --- */
Packit 98cdb6
enum {
Packit 98cdb6
  CHANGED,
Packit 98cdb6
  LAST_SIGNAL
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
/* --- variables --- */
Packit 98cdb6
Packit 98cdb6
static GHashTable  *accel_entry_ht = NULL;	/* accel_path -> AccelEntry */
Packit 98cdb6
static GSList      *accel_filters = NULL;
Packit 98cdb6
static gulong	    accel_map_signals[LAST_SIGNAL] = { 0, };
Packit 98cdb6
static GtkAccelMap *accel_map;
Packit 98cdb6
Packit 98cdb6
/* --- prototypes --- */
Packit 98cdb6
static void do_accel_map_changed (AccelEntry *entry);
Packit 98cdb6
Packit 98cdb6
/* --- functions --- */
Packit 98cdb6
static guint
Packit 98cdb6
accel_entry_hash (gconstpointer key)
Packit 98cdb6
{
Packit 98cdb6
  const AccelEntry *entry = key;
Packit 98cdb6
Packit 98cdb6
  return g_str_hash (entry->accel_path);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
accel_entry_equal (gconstpointer key1,
Packit 98cdb6
		   gconstpointer key2)
Packit 98cdb6
{
Packit 98cdb6
  const AccelEntry *entry1 = key1;
Packit 98cdb6
  const AccelEntry *entry2 = key2;
Packit 98cdb6
Packit 98cdb6
  return g_str_equal (entry1->accel_path, entry2->accel_path);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static inline AccelEntry*
Packit 98cdb6
accel_path_lookup (const gchar *accel_path)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry ekey;
Packit 98cdb6
Packit 98cdb6
  ekey.accel_path = accel_path;
Packit 98cdb6
Packit 98cdb6
  /* safety NULL check for return_if_fail()s */
Packit 98cdb6
  return accel_path ? g_hash_table_lookup (accel_entry_ht, &ekey) : NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_accel_map_init (void)
Packit 98cdb6
{
Packit 98cdb6
  g_assert (accel_entry_ht == NULL);
Packit 98cdb6
Packit 98cdb6
  accel_entry_ht = g_hash_table_new (accel_entry_hash, accel_entry_equal);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gtk_accel_path_is_valid (const gchar *accel_path)
Packit 98cdb6
{
Packit 98cdb6
  gchar *p;
Packit 98cdb6
Packit 98cdb6
  if (!accel_path || accel_path[0] != '<' ||
Packit 98cdb6
      accel_path[1] == '<' || accel_path[1] == '>' || !accel_path[1])
Packit 98cdb6
    return FALSE;
Packit 98cdb6
  p = strchr (accel_path, '>');
Packit 98cdb6
  if (!p || (p[1] != 0 && p[1] != '/'))
Packit 98cdb6
    return FALSE;
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_add_entry:
Packit 98cdb6
 * @accel_path: valid accelerator path
Packit 98cdb6
 * @accel_key:  the accelerator key
Packit 98cdb6
 * @accel_mods: the accelerator modifiers
Packit 98cdb6
 *
Packit 98cdb6
 * Registers a new accelerator with the global accelerator map.
Packit 98cdb6
 * This function should only be called once per @accel_path
Packit 98cdb6
 * with the canonical @accel_key and @accel_mods for this path.
Packit 98cdb6
 * To change the accelerator during runtime programatically, use
Packit 98cdb6
 * gtk_accel_map_change_entry().
Packit 98cdb6
 * The accelerator path must consist of "<WINDOWTYPE>/Category1/Category2/.../Action",
Packit 98cdb6
 * where <WINDOWTYPE> should be a unique application-specific identifier, that
Packit 98cdb6
 * corresponds to the kind of window the accelerator is being used in, e.g. "Gimp-Image",
Packit 98cdb6
 * "Abiword-Document" or "Gnumeric-Settings".
Packit 98cdb6
 * The Category1/.../Action portion is most appropriately chosen by the action the
Packit 98cdb6
 * accelerator triggers, i.e. for accelerators on menu items, choose the item's menu path,
Packit 98cdb6
 * e.g. "File/Save As", "Image/View/Zoom" or "Edit/Select All".
Packit 98cdb6
 * So a full valid accelerator path may look like:
Packit 98cdb6
 * "<Gimp-Toolbox>/File/Dialogs/Tool Options...".
Packit 98cdb6
 * 
Packit 98cdb6
 * Note that @accel_path string will be stored in a #GQuark. Therefore, if you
Packit 98cdb6
 * pass a static string, you can save some memory by interning it first with 
Packit 98cdb6
 * g_intern_static_string().
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_add_entry (const gchar    *accel_path,
Packit 98cdb6
			 guint           accel_key,
Packit 98cdb6
			 GdkModifierType accel_mods)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (_gtk_accel_path_is_valid (accel_path));
Packit 98cdb6
Packit 98cdb6
  if (!accel_key)
Packit 98cdb6
    accel_mods = 0;
Packit 98cdb6
  else
Packit 98cdb6
    accel_mods &= gtk_accelerator_get_default_mod_mask ();
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
  if (entry)
Packit 98cdb6
    {
Packit 98cdb6
      if (!entry->std_accel_key && !entry->std_accel_mods &&
Packit 98cdb6
	  (accel_key || accel_mods))
Packit 98cdb6
	{
Packit 98cdb6
	  entry->std_accel_key = accel_key;
Packit 98cdb6
	  entry->std_accel_mods = accel_mods;
Packit 98cdb6
	  if (!entry->changed)
Packit 98cdb6
	    gtk_accel_map_change_entry (entry->accel_path, accel_key, accel_mods, TRUE);
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      entry = g_slice_new0 (AccelEntry);
Packit 98cdb6
      entry->accel_path = g_intern_string (accel_path);
Packit 98cdb6
      entry->std_accel_key = accel_key;
Packit 98cdb6
      entry->std_accel_mods = accel_mods;
Packit 98cdb6
      entry->accel_key = accel_key;
Packit 98cdb6
      entry->accel_mods = accel_mods;
Packit 98cdb6
      entry->changed = FALSE;
Packit 98cdb6
      entry->lock_count = 0;
Packit 98cdb6
      g_hash_table_insert (accel_entry_ht, entry, entry);
Packit 98cdb6
Packit 98cdb6
      do_accel_map_changed (entry);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_lookup_entry:
Packit 98cdb6
 * @accel_path:  a valid accelerator path
Packit 98cdb6
 * @key:         the accelerator key to be filled in (optional)
Packit 98cdb6
 * @returns:     %TRUE if @accel_path is known, %FALSE otherwise
Packit 98cdb6
 *
Packit 98cdb6
 * Looks up the accelerator entry for @accel_path and fills in @key.
Packit 98cdb6
 */
Packit 98cdb6
gboolean
Packit 98cdb6
gtk_accel_map_lookup_entry (const gchar *accel_path,
Packit 98cdb6
			    GtkAccelKey *key)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (_gtk_accel_path_is_valid (accel_path), FALSE);
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
  if (entry && key)
Packit 98cdb6
    {
Packit 98cdb6
      key->accel_key = entry->accel_key;
Packit 98cdb6
      key->accel_mods = entry->accel_mods;
Packit 98cdb6
      key->accel_flags = 0;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return entry ? TRUE : FALSE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
hash2slist_foreach (gpointer  key,
Packit 98cdb6
		    gpointer  value,
Packit 98cdb6
		    gpointer  user_data)
Packit 98cdb6
{
Packit 98cdb6
  GSList **slist_p = user_data;
Packit 98cdb6
Packit 98cdb6
  *slist_p = g_slist_prepend (*slist_p, value);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GSList*
Packit 98cdb6
g_hash_table_slist_values (GHashTable *hash_table)
Packit 98cdb6
{
Packit 98cdb6
  GSList *slist = NULL;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (hash_table != NULL, NULL);
Packit 98cdb6
Packit 98cdb6
  g_hash_table_foreach (hash_table, hash2slist_foreach, &slist);
Packit 98cdb6
Packit 98cdb6
  return slist;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* if simulate==TRUE, return whether accel_path can be changed to
Packit 98cdb6
 * accel_key && accel_mods. otherwise, return whether accel_path
Packit 98cdb6
 * was actually changed.
Packit 98cdb6
 */
Packit 98cdb6
static gboolean
Packit 98cdb6
internal_change_entry (const gchar    *accel_path,
Packit 98cdb6
		       guint           accel_key,
Packit 98cdb6
		       GdkModifierType accel_mods,
Packit 98cdb6
		       gboolean        replace,
Packit 98cdb6
		       gboolean	       simulate)
Packit 98cdb6
{
Packit 98cdb6
  GSList *node, *slist, *win_list, *group_list, *replace_list = NULL;
Packit 98cdb6
  GHashTable *group_hm, *window_hm;
Packit 98cdb6
  gboolean change_accel, removable, can_change = TRUE, seen_accel = FALSE;
Packit 98cdb6
  GQuark entry_quark;
Packit 98cdb6
  AccelEntry *entry = accel_path_lookup (accel_path);
Packit 98cdb6
Packit 98cdb6
  /* not much todo if there's no entry yet */
Packit 98cdb6
  if (!entry)
Packit 98cdb6
    {
Packit 98cdb6
      if (!simulate)
Packit 98cdb6
	{
Packit 98cdb6
	  gtk_accel_map_add_entry (accel_path, 0, 0);
Packit 98cdb6
	  entry = accel_path_lookup (accel_path);
Packit 98cdb6
	  entry->accel_key = accel_key;
Packit 98cdb6
	  entry->accel_mods = accel_mods;
Packit 98cdb6
	  entry->changed = TRUE;
Packit 98cdb6
Packit 98cdb6
	  do_accel_map_changed (entry);
Packit 98cdb6
	}
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* if there's nothing to change, not much todo either */
Packit 98cdb6
  if (entry->accel_key == accel_key && entry->accel_mods == accel_mods)
Packit 98cdb6
    {
Packit 98cdb6
      if (!simulate)
Packit 98cdb6
	entry->changed = TRUE;
Packit 98cdb6
      return simulate ? TRUE : FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* The no-change case has already been handled, so 
Packit 98cdb6
   * simulate doesn't make a difference here.
Packit 98cdb6
   */
Packit 98cdb6
  if (entry->lock_count > 0)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  /* nobody's interested, easy going */
Packit 98cdb6
  if (!entry->groups)
Packit 98cdb6
    {
Packit 98cdb6
      if (!simulate)
Packit 98cdb6
	{
Packit 98cdb6
	  entry->accel_key = accel_key;
Packit 98cdb6
	  entry->accel_mods = accel_mods;
Packit 98cdb6
	  entry->changed = TRUE;
Packit 98cdb6
Packit 98cdb6
	  do_accel_map_changed (entry);
Packit 98cdb6
	}
Packit 98cdb6
      return TRUE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* 1) fetch all accel groups affected by this entry */
Packit 98cdb6
  entry_quark = g_quark_try_string (entry->accel_path);
Packit 98cdb6
  group_hm = g_hash_table_new (NULL, NULL);
Packit 98cdb6
  window_hm = g_hash_table_new (NULL, NULL);
Packit 98cdb6
  for (slist = entry->groups; slist; slist = slist->next)
Packit 98cdb6
    g_hash_table_insert (group_hm, slist->data, slist->data);
Packit 98cdb6
Packit 98cdb6
  /* 2) collect acceleratables affected */
Packit 98cdb6
  group_list = g_hash_table_slist_values (group_hm);
Packit 98cdb6
  for (slist = group_list; slist; slist = slist->next)
Packit 98cdb6
    {
Packit 98cdb6
      GtkAccelGroup *group = slist->data;
Packit 98cdb6
Packit 98cdb6
      for (node = group->acceleratables; node; node = node->next)
Packit 98cdb6
	g_hash_table_insert (window_hm, node->data, node->data);
Packit 98cdb6
    }
Packit 98cdb6
  g_slist_free (group_list);
Packit 98cdb6
Packit 98cdb6
  /* 3) include all accel groups used by acceleratables */
Packit 98cdb6
  win_list = g_hash_table_slist_values (window_hm);
Packit 98cdb6
  g_hash_table_destroy (window_hm);
Packit 98cdb6
  for (slist = win_list; slist; slist = slist->next)
Packit 98cdb6
    for (node = gtk_accel_groups_from_object (slist->data); node; node = node->next)
Packit 98cdb6
      g_hash_table_insert (group_hm, node->data, node->data);
Packit 98cdb6
  group_list = g_hash_table_slist_values (group_hm);
Packit 98cdb6
  g_hash_table_destroy (group_hm);
Packit 98cdb6
  
Packit 98cdb6
  /* 4) walk the acceleratables and figure whether they occupy accel_key&accel_mods */
Packit 98cdb6
  if (accel_key)
Packit 98cdb6
    for (slist = win_list; slist; slist = slist->next)
Packit 98cdb6
      if (GTK_IS_WINDOW (slist->data))	/* bad kludge in lack of a GtkAcceleratable */
Packit 98cdb6
	if (_gtk_window_query_nonaccels (slist->data, accel_key, accel_mods))
Packit 98cdb6
	  {
Packit 98cdb6
	    seen_accel = TRUE;
Packit 98cdb6
	    break;
Packit 98cdb6
	  }
Packit 98cdb6
  removable = !seen_accel;
Packit 98cdb6
  
Packit 98cdb6
  /* 5) walk all accel groups and search for locks */
Packit 98cdb6
  if (removable)
Packit 98cdb6
    for (slist = group_list; slist; slist = slist->next)
Packit 98cdb6
      {
Packit 98cdb6
	GtkAccelGroup *group = slist->data;
Packit 98cdb6
	GtkAccelGroupEntry *ag_entry;
Packit 98cdb6
	guint i, n;
Packit 98cdb6
	
Packit 98cdb6
	n = 0;
Packit 98cdb6
	ag_entry = entry->accel_key ? gtk_accel_group_query (group, entry->accel_key, entry->accel_mods, &n) : NULL;
Packit 98cdb6
	for (i = 0; i < n; i++)
Packit 98cdb6
	  if (ag_entry[i].accel_path_quark == entry_quark)
Packit 98cdb6
	    {
Packit 98cdb6
	      can_change = !(ag_entry[i].key.accel_flags & GTK_ACCEL_LOCKED);
Packit 98cdb6
	      if (!can_change)
Packit 98cdb6
		goto break_loop_step5;
Packit 98cdb6
	    }
Packit 98cdb6
	
Packit 98cdb6
	n = 0;
Packit 98cdb6
	ag_entry = accel_key ? gtk_accel_group_query (group, accel_key, accel_mods, &n) : NULL;
Packit 98cdb6
	for (i = 0; i < n; i++)
Packit 98cdb6
	  {
Packit 98cdb6
	    seen_accel = TRUE;
Packit 98cdb6
	    removable = !group->lock_count && !(ag_entry[i].key.accel_flags & GTK_ACCEL_LOCKED);
Packit 98cdb6
	    if (!removable)
Packit 98cdb6
	      goto break_loop_step5;
Packit 98cdb6
	    if (ag_entry[i].accel_path_quark)
Packit 98cdb6
	      replace_list = g_slist_prepend (replace_list, GUINT_TO_POINTER (ag_entry[i].accel_path_quark));
Packit 98cdb6
	  }
Packit 98cdb6
      }
Packit 98cdb6
 break_loop_step5:
Packit 98cdb6
  
Packit 98cdb6
  /* 6) check whether we can remove existing accelerators */
Packit 98cdb6
  if (removable && can_change)
Packit 98cdb6
    for (slist = replace_list; slist; slist = slist->next)
Packit 98cdb6
      if (!internal_change_entry (g_quark_to_string (GPOINTER_TO_UINT (slist->data)), 0, 0, FALSE, TRUE))
Packit 98cdb6
	{
Packit 98cdb6
	  removable = FALSE;
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
  
Packit 98cdb6
  /* 7) check conditions and proceed if possible */
Packit 98cdb6
  change_accel = can_change && (!seen_accel || (removable && replace));
Packit 98cdb6
  
Packit 98cdb6
  if (change_accel && !simulate)
Packit 98cdb6
    {
Packit 98cdb6
      /* ref accel groups */
Packit 98cdb6
      for (slist = group_list; slist; slist = slist->next)
Packit 98cdb6
	g_object_ref (slist->data);
Packit 98cdb6
Packit 98cdb6
      /* 8) remove existing accelerators */
Packit 98cdb6
      for (slist = replace_list; slist; slist = slist->next)
Packit 98cdb6
	internal_change_entry (g_quark_to_string (GPOINTER_TO_UINT (slist->data)), 0, 0, FALSE, FALSE);
Packit 98cdb6
Packit 98cdb6
      /* 9) install new accelerator */
Packit 98cdb6
      entry->accel_key = accel_key;
Packit 98cdb6
      entry->accel_mods = accel_mods;
Packit 98cdb6
      entry->changed = TRUE;
Packit 98cdb6
Packit 98cdb6
      for (slist = group_list; slist; slist = slist->next)
Packit 98cdb6
	_gtk_accel_group_reconnect (slist->data, g_quark_from_string (entry->accel_path));
Packit 98cdb6
Packit 98cdb6
      /* unref accel groups */
Packit 98cdb6
      for (slist = group_list; slist; slist = slist->next)
Packit 98cdb6
	g_object_unref (slist->data);
Packit 98cdb6
Packit 98cdb6
      do_accel_map_changed (entry);
Packit 98cdb6
    }
Packit 98cdb6
  g_slist_free (replace_list);
Packit 98cdb6
  g_slist_free (group_list);
Packit 98cdb6
  g_slist_free (win_list);
Packit 98cdb6
Packit 98cdb6
  return change_accel;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_change_entry:
Packit 98cdb6
 * @accel_path:  a valid accelerator path
Packit 98cdb6
 * @accel_key:   the new accelerator key
Packit 98cdb6
 * @accel_mods:  the new accelerator modifiers
Packit 98cdb6
 * @replace:     %TRUE if other accelerators may be deleted upon conflicts
Packit 98cdb6
 * @returns:     %TRUE if the accelerator could be changed, %FALSE otherwise
Packit 98cdb6
 *
Packit 98cdb6
 * Changes the @accel_key and @accel_mods currently associated with @accel_path.
Packit 98cdb6
 * Due to conflicts with other accelerators, a change may not always be possible,
Packit 98cdb6
 * @replace indicates whether other accelerators may be deleted to resolve such
Packit 98cdb6
 * conflicts. A change will only occur if all conflicts could be resolved (which
Packit 98cdb6
 * might not be the case if conflicting accelerators are locked). Successful
Packit 98cdb6
 * changes are indicated by a %TRUE return value.
Packit 98cdb6
 * 
Packit 98cdb6
 * Note that @accel_path string will be stored in a #GQuark. Therefore, if you
Packit 98cdb6
 * pass a static string, you can save some memory by interning it first with 
Packit 98cdb6
 * g_intern_static_string().
Packit 98cdb6
 */
Packit 98cdb6
gboolean
Packit 98cdb6
gtk_accel_map_change_entry (const gchar    *accel_path,
Packit 98cdb6
			    guint           accel_key,
Packit 98cdb6
			    GdkModifierType accel_mods,
Packit 98cdb6
			    gboolean        replace)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (_gtk_accel_path_is_valid (accel_path), FALSE);
Packit 98cdb6
Packit 98cdb6
  return internal_change_entry (accel_path, accel_key, accel_key ? accel_mods : 0, replace, FALSE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static guint
Packit 98cdb6
accel_map_parse_accel_path (GScanner *scanner)
Packit 98cdb6
{
Packit 98cdb6
  guint accel_key = 0;
Packit 98cdb6
  GdkModifierType accel_mods = 0;
Packit 98cdb6
  gchar *path, *accel;
Packit 98cdb6
  
Packit 98cdb6
  /* parse accel path */
Packit 98cdb6
  g_scanner_get_next_token (scanner);
Packit 98cdb6
  if (scanner->token != G_TOKEN_STRING)
Packit 98cdb6
    return G_TOKEN_STRING;
Packit 98cdb6
Packit 98cdb6
  /* test if the next token is an accelerator */
Packit 98cdb6
  g_scanner_peek_next_token (scanner);
Packit 98cdb6
  if (scanner->next_token != G_TOKEN_STRING)
Packit 98cdb6
    {
Packit 98cdb6
      /* if not so, eat that token and error out */
Packit 98cdb6
      g_scanner_get_next_token (scanner);
Packit 98cdb6
      return G_TOKEN_STRING;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* get the full accelerator specification */
Packit 98cdb6
  path = g_strdup (scanner->value.v_string);
Packit 98cdb6
  g_scanner_get_next_token (scanner);
Packit 98cdb6
  accel = g_strdup (scanner->value.v_string);
Packit 98cdb6
Packit 98cdb6
  /* ensure the entry is present */
Packit 98cdb6
  gtk_accel_map_add_entry (path, 0, 0);
Packit 98cdb6
Packit 98cdb6
  /* and propagate it */
Packit 98cdb6
  gtk_accelerator_parse (accel, &accel_key, &accel_mods);
Packit 98cdb6
  gtk_accel_map_change_entry (path, accel_key, accel_mods, TRUE);
Packit 98cdb6
Packit 98cdb6
  g_free (accel);
Packit 98cdb6
  g_free (path);
Packit 98cdb6
Packit 98cdb6
  /* check correct statement end */
Packit 98cdb6
  g_scanner_get_next_token (scanner);
Packit 98cdb6
  if (scanner->token != ')')
Packit 98cdb6
    return ')';
Packit 98cdb6
  else
Packit 98cdb6
    return G_TOKEN_NONE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
accel_map_parse_statement (GScanner *scanner)
Packit 98cdb6
{
Packit 98cdb6
  guint expected_token;
Packit 98cdb6
Packit 98cdb6
  g_scanner_get_next_token (scanner);
Packit 98cdb6
Packit 98cdb6
  if (scanner->token == G_TOKEN_SYMBOL)
Packit 98cdb6
    {
Packit 98cdb6
      guint (*parser_func) (GScanner*);
Packit 98cdb6
Packit 98cdb6
      parser_func = (guint (*) (GScanner *))scanner->value.v_symbol;
Packit 98cdb6
Packit 98cdb6
      expected_token = parser_func (scanner);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    expected_token = G_TOKEN_SYMBOL;
Packit 98cdb6
Packit 98cdb6
  /* skip rest of statement on errrors
Packit 98cdb6
   */
Packit 98cdb6
  if (expected_token != G_TOKEN_NONE)
Packit 98cdb6
    {
Packit 98cdb6
      register guint level;
Packit 98cdb6
Packit 98cdb6
      level = 1;
Packit 98cdb6
      if (scanner->token == ')')
Packit 98cdb6
	level--;
Packit 98cdb6
      if (scanner->token == '(')
Packit 98cdb6
	level++;
Packit 98cdb6
Packit 98cdb6
      while (!g_scanner_eof (scanner) && level > 0)
Packit 98cdb6
	{
Packit 98cdb6
	  g_scanner_get_next_token (scanner);
Packit 98cdb6
Packit 98cdb6
	  if (scanner->token == '(')
Packit 98cdb6
	    level++;
Packit 98cdb6
	  else if (scanner->token == ')')
Packit 98cdb6
	    level--;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_load_scanner:
Packit 98cdb6
 * @scanner: a #GScanner which has already been provided with an input file
Packit 98cdb6
 *
Packit 98cdb6
 * #GScanner variant of gtk_accel_map_load().
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_load_scanner (GScanner *scanner)
Packit 98cdb6
{
Packit 98cdb6
  gboolean skip_comment_single;
Packit 98cdb6
  gboolean symbol_2_token;
Packit 98cdb6
  gchar *cpair_comment_single;
Packit 98cdb6
  gpointer saved_symbol;
Packit 98cdb6
  
Packit 98cdb6
  g_return_if_fail (scanner != NULL);
Packit 98cdb6
Packit 98cdb6
  /* configure scanner */
Packit 98cdb6
  skip_comment_single = scanner->config->skip_comment_single;
Packit 98cdb6
  scanner->config->skip_comment_single = TRUE;
Packit 98cdb6
  cpair_comment_single = scanner->config->cpair_comment_single;
Packit 98cdb6
  scanner->config->cpair_comment_single = ";\n";
Packit 98cdb6
  symbol_2_token = scanner->config->symbol_2_token;
Packit 98cdb6
  scanner->config->symbol_2_token = FALSE;
Packit 98cdb6
  saved_symbol = g_scanner_lookup_symbol (scanner, "gtk_accel_path");
Packit 98cdb6
  g_scanner_scope_add_symbol (scanner, 0, "gtk_accel_path", 
Packit 98cdb6
			      accel_map_parse_accel_path);
Packit 98cdb6
Packit 98cdb6
  /* outer parsing loop
Packit 98cdb6
   */
Packit 98cdb6
  g_scanner_peek_next_token (scanner);
Packit 98cdb6
  while (scanner->next_token == '(')
Packit 98cdb6
    {
Packit 98cdb6
      g_scanner_get_next_token (scanner);
Packit 98cdb6
Packit 98cdb6
      accel_map_parse_statement (scanner);
Packit 98cdb6
Packit 98cdb6
      g_scanner_peek_next_token (scanner);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  /* restore config */
Packit 98cdb6
  scanner->config->skip_comment_single = skip_comment_single;
Packit 98cdb6
  scanner->config->cpair_comment_single = cpair_comment_single;
Packit 98cdb6
  scanner->config->symbol_2_token = symbol_2_token;
Packit 98cdb6
  g_scanner_scope_remove_symbol (scanner, 0, "gtk_accel_path");
Packit 98cdb6
  if (saved_symbol)
Packit 98cdb6
    g_scanner_scope_add_symbol (scanner, 0, "gtk_accel_path", saved_symbol);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_load_fd:
Packit 98cdb6
 * @fd: a valid readable file descriptor
Packit 98cdb6
 *
Packit 98cdb6
 * Filedescriptor variant of gtk_accel_map_load().
Packit 98cdb6
 *
Packit 98cdb6
 * Note that the file descriptor will not be closed by this function.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_load_fd (gint fd)
Packit 98cdb6
{
Packit 98cdb6
  GScanner *scanner;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (fd >= 0);
Packit 98cdb6
Packit 98cdb6
  /* create and setup scanner */
Packit 98cdb6
  scanner = g_scanner_new (NULL);
Packit 98cdb6
  g_scanner_input_file (scanner, fd);
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_load_scanner (scanner);
Packit 98cdb6
Packit 98cdb6
  g_scanner_destroy (scanner);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_load:
Packit 98cdb6
 * @file_name: a file containing accelerator specifications,
Packit 98cdb6
 *   in the GLib file name encoding
Packit 98cdb6
 *
Packit 98cdb6
 * Parses a file previously saved with gtk_accel_map_save() for
Packit 98cdb6
 * accelerator specifications, and propagates them accordingly.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_load (const gchar *file_name)
Packit 98cdb6
{
Packit 98cdb6
  gint fd;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (file_name != NULL);
Packit 98cdb6
Packit 98cdb6
  if (!g_file_test (file_name, G_FILE_TEST_IS_REGULAR))
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  fd = g_open (file_name, O_RDONLY, 0);
Packit 98cdb6
  if (fd < 0)
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_load_fd (fd);
Packit 98cdb6
Packit 98cdb6
  close (fd);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
write_all (gint   fd,
Packit 98cdb6
	   gchar *buf,
Packit 98cdb6
	   gsize  to_write)
Packit 98cdb6
{
Packit 98cdb6
  while (to_write > 0)
Packit 98cdb6
    {
Packit 98cdb6
      gssize count = write (fd, buf, to_write);
Packit 98cdb6
      if (count < 0)
Packit 98cdb6
	{
Packit 98cdb6
	  if (errno != EINTR)
Packit 98cdb6
	    return FALSE;
Packit 98cdb6
	}
Packit 98cdb6
      else
Packit 98cdb6
	{
Packit 98cdb6
	  to_write -= count;
Packit 98cdb6
	  buf += count;
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
accel_map_print (gpointer        data,
Packit 98cdb6
		 const gchar    *accel_path,
Packit 98cdb6
		 guint           accel_key,
Packit 98cdb6
		 GdkModifierType accel_mods,
Packit 98cdb6
		 gboolean        changed)
Packit 98cdb6
{
Packit 98cdb6
  GString *gstring = g_string_new (changed ? NULL : "; ");
Packit 98cdb6
  gint fd = GPOINTER_TO_INT (data);
Packit 98cdb6
  gchar *tmp, *name;
Packit 98cdb6
Packit 98cdb6
  g_string_append (gstring, "(gtk_accel_path \"");
Packit 98cdb6
Packit 98cdb6
  tmp = g_strescape (accel_path, NULL);
Packit 98cdb6
  g_string_append (gstring, tmp);
Packit 98cdb6
  g_free (tmp);
Packit 98cdb6
Packit 98cdb6
  g_string_append (gstring, "\" \"");
Packit 98cdb6
Packit 98cdb6
  name = gtk_accelerator_name (accel_key, accel_mods);
Packit 98cdb6
  tmp = g_strescape (name, NULL);
Packit 98cdb6
  g_free (name);
Packit 98cdb6
  g_string_append (gstring, tmp);
Packit 98cdb6
  g_free (tmp);
Packit 98cdb6
Packit 98cdb6
  g_string_append (gstring, "\")\n");
Packit 98cdb6
Packit 98cdb6
  write_all (fd, gstring->str, gstring->len);
Packit 98cdb6
Packit 98cdb6
  g_string_free (gstring, TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_save_fd:
Packit 98cdb6
 * @fd: a valid writable file descriptor
Packit 98cdb6
 *
Packit 98cdb6
 * Filedescriptor variant of gtk_accel_map_save().
Packit 98cdb6
 *
Packit 98cdb6
 * Note that the file descriptor will not be closed by this function.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_save_fd (gint fd)
Packit 98cdb6
{
Packit 98cdb6
  GString *gstring;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (fd >= 0);
Packit 98cdb6
Packit 98cdb6
  gstring = g_string_new ("; ");
Packit 98cdb6
  if (g_get_prgname ())
Packit 98cdb6
    g_string_append (gstring, g_get_prgname ());
Packit 98cdb6
  g_string_append (gstring, " GtkAccelMap rc-file         -*- scheme -*-\n");
Packit 98cdb6
  g_string_append (gstring, "; this file is an automated accelerator map dump\n");
Packit 98cdb6
  g_string_append (gstring, ";\n");
Packit 98cdb6
Packit 98cdb6
  write_all (fd, gstring->str, gstring->len);
Packit 98cdb6
  
Packit 98cdb6
  g_string_free (gstring, TRUE);
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_foreach (GINT_TO_POINTER (fd), accel_map_print);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_save:
Packit 98cdb6
 * @file_name: the name of the file to contain accelerator specifications,
Packit 98cdb6
 *   in the GLib file name encoding
Packit 98cdb6
 *
Packit 98cdb6
 * Saves current accelerator specifications (accelerator path, key
Packit 98cdb6
 * and modifiers) to @file_name.
Packit 98cdb6
 * The file is written in a format suitable to be read back in by
Packit 98cdb6
 * gtk_accel_map_load().
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_save (const gchar *file_name)
Packit 98cdb6
{
Packit 98cdb6
  gint fd;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (file_name != NULL);
Packit 98cdb6
Packit 98cdb6
  fd = g_open (file_name, O_CREAT | O_TRUNC | O_WRONLY, 0644);
Packit 98cdb6
  if (fd < 0)
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_save_fd (fd);
Packit 98cdb6
Packit 98cdb6
  close (fd);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_foreach:
Packit 98cdb6
 * @data:         data to be passed into @foreach_func
Packit 98cdb6
 * @foreach_func: function to be executed for each accel map entry which
Packit 98cdb6
 *                is not filtered out
Packit 98cdb6
 *
Packit 98cdb6
 * Loops over the entries in the accelerator map whose accel path 
Packit 98cdb6
 * doesn't match any of the filters added with gtk_accel_map_add_filter(), 
Packit 98cdb6
 * and execute @foreach_func on each. The signature of @foreach_func is 
Packit 98cdb6
 * that of #GtkAccelMapForeach, the @changed parameter indicates whether
Packit 98cdb6
 * this accelerator was changed during runtime (thus, would need
Packit 98cdb6
 * saving during an accelerator map dump).
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_foreach (gpointer           data,
Packit 98cdb6
		       GtkAccelMapForeach foreach_func)
Packit 98cdb6
{
Packit 98cdb6
  GSList *entries, *slist, *node;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (foreach_func != NULL);
Packit 98cdb6
Packit 98cdb6
  entries = g_hash_table_slist_values (accel_entry_ht);
Packit 98cdb6
  for (slist = entries; slist; slist = slist->next)
Packit 98cdb6
    {
Packit 98cdb6
      AccelEntry *entry = slist->data;
Packit 98cdb6
      gboolean changed = entry->accel_key != entry->std_accel_key || entry->accel_mods != entry->std_accel_mods;
Packit 98cdb6
Packit 98cdb6
      for (node = accel_filters; node; node = node->next)
Packit 98cdb6
	if (g_pattern_match_string (node->data, entry->accel_path))
Packit 98cdb6
	  goto skip_accel;
Packit 98cdb6
      foreach_func (data, entry->accel_path, entry->accel_key, entry->accel_mods, changed);
Packit 98cdb6
    skip_accel:
Packit 98cdb6
      /* noop */;
Packit 98cdb6
    }
Packit 98cdb6
  g_slist_free (entries);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_foreach_unfiltered:
Packit 98cdb6
 * @data:         data to be passed into @foreach_func
Packit 98cdb6
 * @foreach_func: function to be executed for each accel map entry
Packit 98cdb6
 *
Packit 98cdb6
 * Loops over all entries in the accelerator map, and execute
Packit 98cdb6
 * @foreach_func on each. The signature of @foreach_func is that of
Packit 98cdb6
 * #GtkAccelMapForeach, the @changed parameter indicates whether
Packit 98cdb6
 * this accelerator was changed during runtime (thus, would need
Packit 98cdb6
 * saving during an accelerator map dump).
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_foreach_unfiltered (gpointer           data,
Packit 98cdb6
				  GtkAccelMapForeach foreach_func)
Packit 98cdb6
{
Packit 98cdb6
  GSList *entries, *slist;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (foreach_func != NULL);
Packit 98cdb6
Packit 98cdb6
  entries = g_hash_table_slist_values (accel_entry_ht);
Packit 98cdb6
  for (slist = entries; slist; slist = slist->next)
Packit 98cdb6
    {
Packit 98cdb6
      AccelEntry *entry = slist->data;
Packit 98cdb6
      gboolean changed = entry->accel_key != entry->std_accel_key || entry->accel_mods != entry->std_accel_mods;
Packit 98cdb6
Packit 98cdb6
      foreach_func (data, entry->accel_path, entry->accel_key, entry->accel_mods, changed);
Packit 98cdb6
    }
Packit 98cdb6
  g_slist_free (entries);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_add_filter:
Packit 98cdb6
 * @filter_pattern: a pattern (see #GPatternSpec)
Packit 98cdb6
 *
Packit 98cdb6
 * Adds a filter to the global list of accel path filters.
Packit 98cdb6
 *
Packit 98cdb6
 * Accel map entries whose accel path matches one of the filters
Packit 98cdb6
 * are skipped by gtk_accel_map_foreach().
Packit 98cdb6
 *
Packit 98cdb6
 * This function is intended for GTK+ modules that create their own
Packit 98cdb6
 * menus, but don't want them to be saved into the applications accelerator
Packit 98cdb6
 * map dump.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_add_filter (const gchar *filter_pattern)
Packit 98cdb6
{
Packit 98cdb6
  GPatternSpec *pspec;
Packit 98cdb6
  GSList *slist;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (filter_pattern != NULL);
Packit 98cdb6
Packit 98cdb6
  pspec = g_pattern_spec_new (filter_pattern);
Packit 98cdb6
  for (slist = accel_filters; slist; slist = slist->next)
Packit 98cdb6
    if (g_pattern_spec_equal (pspec, slist->data))
Packit 98cdb6
      {
Packit 98cdb6
	g_pattern_spec_free (pspec);
Packit 98cdb6
	return;
Packit 98cdb6
      }
Packit 98cdb6
  accel_filters = g_slist_prepend (accel_filters, pspec);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_accel_map_add_group (const gchar   *accel_path,
Packit 98cdb6
			  GtkAccelGroup *accel_group)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (_gtk_accel_path_is_valid (accel_path));
Packit 98cdb6
  g_return_if_fail (GTK_IS_ACCEL_GROUP (accel_group));
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
  if (!entry)
Packit 98cdb6
    {
Packit 98cdb6
      gtk_accel_map_add_entry (accel_path, 0, 0);
Packit 98cdb6
      entry = accel_path_lookup (accel_path);
Packit 98cdb6
    }
Packit 98cdb6
  entry->groups = g_slist_prepend (entry->groups, accel_group);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_accel_map_remove_group (const gchar   *accel_path,
Packit 98cdb6
			     GtkAccelGroup *accel_group)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
  g_return_if_fail (entry != NULL);
Packit 98cdb6
  g_return_if_fail (g_slist_find (entry->groups, accel_group));
Packit 98cdb6
Packit 98cdb6
  entry->groups = g_slist_remove (entry->groups, accel_group);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_lock_path:
Packit 98cdb6
 * @accel_path: a valid accelerator path
Packit 98cdb6
 * 
Packit 98cdb6
 * Locks the given accelerator path. If the accelerator map doesn't yet contain
Packit 98cdb6
 * an entry for @accel_path, a new one is created.
Packit 98cdb6
 *
Packit 98cdb6
 * Locking an accelerator path prevents its accelerator from being changed 
Packit 98cdb6
 * during runtime. A locked accelerator path can be unlocked by 
Packit 98cdb6
 * gtk_accel_map_unlock_path(). Refer to gtk_accel_map_change_entry() 
Packit 98cdb6
 * for information about runtime accelerator changes.
Packit 98cdb6
 *
Packit 98cdb6
 * If called more than once, @accel_path remains locked until
Packit 98cdb6
 * gtk_accel_map_unlock_path() has been called an equivalent number
Packit 98cdb6
 * of times.
Packit 98cdb6
 *
Packit 98cdb6
 * Note that locking of individual accelerator paths is independent from 
Packit 98cdb6
 * locking the #GtkAccelGroup containing them. For runtime accelerator
Packit 98cdb6
 * changes to be possible both the accelerator path and its #GtkAccelGroup
Packit 98cdb6
 * have to be unlocked. 
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.4
Packit 98cdb6
 **/
Packit 98cdb6
void 
Packit 98cdb6
gtk_accel_map_lock_path (const gchar *accel_path)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (_gtk_accel_path_is_valid (accel_path));
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
  
Packit 98cdb6
  if (!entry)
Packit 98cdb6
    {
Packit 98cdb6
      gtk_accel_map_add_entry (accel_path, 0, 0);
Packit 98cdb6
      entry = accel_path_lookup (accel_path);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  entry->lock_count += 1;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_unlock_path:
Packit 98cdb6
 * @accel_path: a valid accelerator path
Packit 98cdb6
 * 
Packit 98cdb6
 * Undoes the last call to gtk_accel_map_lock_path() on this @accel_path.
Packit 98cdb6
 * Refer to gtk_accel_map_lock_path() for information about accelerator path locking.
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.4
Packit 98cdb6
 **/
Packit 98cdb6
void 
Packit 98cdb6
gtk_accel_map_unlock_path (const gchar *accel_path)
Packit 98cdb6
{
Packit 98cdb6
  AccelEntry *entry;
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (_gtk_accel_path_is_valid (accel_path));
Packit 98cdb6
Packit 98cdb6
  entry = accel_path_lookup (accel_path);
Packit 98cdb6
Packit 98cdb6
  g_return_if_fail (entry != NULL && entry->lock_count > 0);
Packit 98cdb6
Packit 98cdb6
  entry->lock_count -= 1;  
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GtkAccelMap, gtk_accel_map, G_TYPE_OBJECT)
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_accel_map_class_init (GtkAccelMapClass *accel_map_class)
Packit 98cdb6
{
Packit 98cdb6
  /**
Packit 98cdb6
   * GtkAccelMap::changed:
Packit 98cdb6
   * @object: the global accel map object
Packit 98cdb6
   * @accel_path: the path of the accelerator that changed
Packit 98cdb6
   * @accel_key: the key value for the new accelerator
Packit 98cdb6
   * @accel_mods: the modifier mask for the new accelerator
Packit 98cdb6
   *
Packit 98cdb6
   * Notifies of a change in the global accelerator map.
Packit 98cdb6
   * The path is also used as the detail for the signal,
Packit 98cdb6
   * so it is possible to connect to
Packit 98cdb6
   * changed::<replaceable>accel_path</replaceable>.
Packit 98cdb6
   *
Packit 98cdb6
   * Since: 2.4
Packit 98cdb6
   */
Packit 98cdb6
  accel_map_signals[CHANGED] = g_signal_new (I_("changed"),
Packit 98cdb6
					     G_TYPE_FROM_CLASS (accel_map_class),
Packit 98cdb6
					     G_SIGNAL_DETAILED|G_SIGNAL_RUN_LAST,
Packit 98cdb6
					     0,
Packit 98cdb6
					     NULL, NULL,
Packit 98cdb6
					     _gtk_marshal_VOID__STRING_UINT_FLAGS,
Packit 98cdb6
					     G_TYPE_NONE, 3,
Packit 98cdb6
					     G_TYPE_STRING, G_TYPE_UINT, GDK_TYPE_MODIFIER_TYPE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_accel_map_init (GtkAccelMap *accel_map)
Packit 98cdb6
{
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * gtk_accel_map_get:
Packit 98cdb6
 * 
Packit 98cdb6
 * Gets the singleton global #GtkAccelMap object. This object
Packit 98cdb6
 * is useful only for notification of changes to the accelerator
Packit 98cdb6
 * map via the ::changed signal; it isn't a parameter to the
Packit 98cdb6
 * other accelerator map functions.
Packit 98cdb6
 * 
Packit 98cdb6
 * Return value: the global #GtkAccelMap object
Packit 98cdb6
 *
Packit 98cdb6
 * Since: 2.4
Packit 98cdb6
 **/
Packit 98cdb6
GtkAccelMap *
Packit 98cdb6
gtk_accel_map_get (void)
Packit 98cdb6
{
Packit 98cdb6
  if (!accel_map)
Packit 98cdb6
    accel_map = g_object_new (GTK_TYPE_ACCEL_MAP, NULL);
Packit 98cdb6
Packit 98cdb6
  return accel_map;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
do_accel_map_changed (AccelEntry *entry)
Packit 98cdb6
{
Packit 98cdb6
  if (accel_map)
Packit 98cdb6
    g_signal_emit (accel_map,
Packit 98cdb6
		   accel_map_signals[CHANGED],
Packit 98cdb6
		   g_quark_from_string (entry->accel_path),
Packit 98cdb6
		   entry->accel_path,
Packit 98cdb6
		   entry->accel_key,
Packit 98cdb6
		   entry->accel_mods);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#if defined (G_OS_WIN32) && !defined (_WIN64)
Packit 98cdb6
Packit 98cdb6
#undef gtk_accel_map_load
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_load (const gchar *file_name)
Packit 98cdb6
{
Packit 98cdb6
  gchar *utf8_file_name = g_locale_to_utf8 (file_name, -1, NULL, NULL, NULL);
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_load_utf8 (utf8_file_name);
Packit 98cdb6
Packit 98cdb6
  g_free (utf8_file_name);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#undef gtk_accel_map_save
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
gtk_accel_map_save (const gchar *file_name)
Packit 98cdb6
{
Packit 98cdb6
  gchar *utf8_file_name = g_locale_to_utf8 (file_name, -1, NULL, NULL, NULL);
Packit 98cdb6
Packit 98cdb6
  gtk_accel_map_save_utf8 (utf8_file_name);
Packit 98cdb6
Packit 98cdb6
  g_free (utf8_file_name);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
#define __GTK_ACCEL_MAP_C__
Packit 98cdb6
#include "gtkaliasdef.c"