Blame gdk/x11/gdkim-x11.c

Packit Service fb6fa5
/* GDK - The GIMP Drawing Kit
Packit Service fb6fa5
 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * This library is free software; you can redistribute it and/or
Packit Service fb6fa5
 * modify it under the terms of the GNU Lesser General Public
Packit Service fb6fa5
 * License as published by the Free Software Foundation; either
Packit Service fb6fa5
 * version 2 of the License, or (at your option) any later version.
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * This library is distributed in the hope that it will be useful,
Packit Service fb6fa5
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service fb6fa5
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
Packit Service fb6fa5
 * Lesser General Public License for more details.
Packit Service fb6fa5
 *
Packit Service fb6fa5
 * You should have received a copy of the GNU Lesser General Public
Packit Service fb6fa5
 * License along with this library; if not, write to the
Packit Service fb6fa5
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit Service fb6fa5
 * Boston, MA 02111-1307, USA.
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
/*
Packit Service fb6fa5
 * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
Packit Service fb6fa5
 * file for a list of people on the GTK+ Team.  See the ChangeLog
Packit Service fb6fa5
 * files for a list of changes.  These files are distributed with
Packit Service fb6fa5
 * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
Packit Service fb6fa5
 */
Packit Service fb6fa5
Packit Service fb6fa5
#include "config.h"
Packit Service fb6fa5
Packit Service fb6fa5
#include <locale.h>
Packit Service fb6fa5
#include <stdlib.h>
Packit Service fb6fa5
#include <string.h>
Packit Service fb6fa5
Packit Service fb6fa5
#include "gdkx.h"
Packit Service fb6fa5
#include "gdk.h"		/* For gdk_flush() */
Packit Service fb6fa5
#include "gdkpixmap.h"
Packit Service fb6fa5
#include "gdkinternals.h"
Packit Service fb6fa5
#include "gdkdisplay-x11.h"
Packit Service fb6fa5
#include "gdkalias.h"
Packit Service fb6fa5
Packit Service fb6fa5
Packit Service fb6fa5
/* If this variable is FALSE, it indicates that we should
Packit Service fb6fa5
 * avoid trying to use multibyte conversion functions and
Packit Service fb6fa5
 * assume everything is 1-byte per character
Packit Service fb6fa5
 */
Packit Service fb6fa5
static gboolean gdk_use_mb;
Packit Service fb6fa5
Packit Service fb6fa5
void
Packit Service fb6fa5
_gdk_x11_initialize_locale (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  wchar_t result;
Packit Service fb6fa5
  gchar *current_locale;
Packit Service fb6fa5
  static char *last_locale = NULL;
Packit Service fb6fa5
Packit Service fb6fa5
  gdk_use_mb = FALSE;
Packit Service fb6fa5
Packit Service fb6fa5
  current_locale = setlocale (LC_ALL, NULL);
Packit Service fb6fa5
Packit Service fb6fa5
  if (last_locale && strcmp (last_locale, current_locale) == 0)
Packit Service fb6fa5
    return;
Packit Service fb6fa5
Packit Service fb6fa5
  g_free (last_locale);
Packit Service fb6fa5
  last_locale = g_strdup (current_locale);
Packit Service fb6fa5
Packit Service fb6fa5
  if (XSupportsLocale ())
Packit Service fb6fa5
    XSetLocaleModifiers ("");
Packit Service fb6fa5
Packit Service fb6fa5
  if ((strcmp (current_locale, "C")) && (strcmp (current_locale, "POSIX")))
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gdk_use_mb = TRUE;
Packit Service fb6fa5
Packit Service fb6fa5
#ifndef X_LOCALE
Packit Service fb6fa5
      /* Detect ancient GNU libc, where mb == UTF8. Not useful unless it's
Packit Service fb6fa5
       * really a UTF8 locale. The below still probably will
Packit Service fb6fa5
       * screw up on Greek, Cyrillic, etc, encoded as UTF8.
Packit Service fb6fa5
       */
Packit Service fb6fa5
      
Packit Service fb6fa5
      if ((MB_CUR_MAX == 2) &&
Packit Service fb6fa5
	  (mbstowcs (&result, "\xdd\xa5", 1) > 0) &&
Packit Service fb6fa5
	  result == 0x765)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  if ((strlen (current_locale) < 4) ||
Packit Service fb6fa5
	      g_ascii_strcasecmp (current_locale + strlen(current_locale) - 4,
Packit Service fb6fa5
				  "utf8"))
Packit Service fb6fa5
	    gdk_use_mb = FALSE;
Packit Service fb6fa5
	}
Packit Service fb6fa5
#endif /* X_LOCALE */
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  GDK_NOTE (XIM,
Packit Service fb6fa5
	    g_message ("%s multi-byte string functions.", 
Packit Service fb6fa5
		       gdk_use_mb ? "Using" : "Not using"));
Packit Service fb6fa5
  
Packit Service fb6fa5
  return;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
gchar*
Packit Service fb6fa5
gdk_set_locale (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (!setlocale (LC_ALL,""))
Packit Service fb6fa5
    g_warning ("locale not supported by C library");
Packit Service fb6fa5
Packit Service fb6fa5
  _gdk_x11_initialize_locale ();
Packit Service fb6fa5
  
Packit Service fb6fa5
  return setlocale (LC_ALL, NULL);
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
static GdkDisplay *
Packit Service fb6fa5
find_a_display (void)
Packit Service fb6fa5
{
Packit Service fb6fa5
  GdkDisplay *display = gdk_display_get_default ();
Packit Service fb6fa5
Packit Service fb6fa5
  if (!display)
Packit Service fb6fa5
    display = _gdk_displays->data;
Packit Service fb6fa5
Packit Service fb6fa5
  return display;
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
/**
Packit Service fb6fa5
 * gdk_wcstombs:
Packit Service fb6fa5
 * @src: a wide character string.
Packit Service fb6fa5
 * 
Packit Service fb6fa5
 * Converts a wide character string to a multi-byte string.
Packit Service fb6fa5
 * (The function name comes from an acronym of 'Wide Character String TO
Packit Service fb6fa5
 * Multi-Byte String').
Packit Service fb6fa5
 * 
Packit Service fb6fa5
 * Return value: the multi-byte string corresponding to @src, or %NULL if the
Packit Service fb6fa5
 * conversion failed. The returned string should be freed with g_free() when no
Packit Service fb6fa5
 * longer needed.
Packit Service fb6fa5
 **/
Packit Service fb6fa5
gchar *
Packit Service fb6fa5
gdk_wcstombs (const GdkWChar *src)
Packit Service fb6fa5
{
Packit Service fb6fa5
  gchar *mbstr;
Packit Service fb6fa5
Packit Service fb6fa5
  if (gdk_use_mb)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      GdkDisplay *display = find_a_display ();
Packit Service fb6fa5
      Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
Packit Service fb6fa5
      XTextProperty tpr;
Packit Service fb6fa5
Packit Service fb6fa5
      if (sizeof(wchar_t) != sizeof(GdkWChar))
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  gint i;
Packit Service fb6fa5
	  wchar_t *src_alt;
Packit Service fb6fa5
	  for (i=0; src[i]; i++);
Packit Service fb6fa5
	  src_alt = g_new (wchar_t, i+1);
Packit Service fb6fa5
	  for (; i>=0; i--)
Packit Service fb6fa5
	    src_alt[i] = src[i];
Packit Service fb6fa5
	  if (XwcTextListToTextProperty (xdisplay, &src_alt, 1, XTextStyle, &tpr)
Packit Service fb6fa5
	      != Success)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      g_free (src_alt);
Packit Service fb6fa5
	      return NULL;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  g_free (src_alt);
Packit Service fb6fa5
	}
Packit Service fb6fa5
      else
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  wchar_t *tmp;
Packit Service fb6fa5
	  
Packit Service fb6fa5
	  if (XwcTextListToTextProperty (xdisplay, &tmp, 1,
Packit Service fb6fa5
					 XTextStyle, &tpr) != Success)
Packit Service fb6fa5
	    {
Packit Service fb6fa5
	      return NULL;
Packit Service fb6fa5
	    }
Packit Service fb6fa5
	  
Packit Service fb6fa5
	  src = (GdkWChar *)tmp;
Packit Service fb6fa5
	}
Packit Service fb6fa5
      /*
Packit Service fb6fa5
       * We must copy the string into an area allocated by glib, because
Packit Service fb6fa5
       * the string 'tpr.value' must be freed by XFree().
Packit Service fb6fa5
       */
Packit Service fb6fa5
      mbstr = g_strdup((gchar *)tpr.value);
Packit Service fb6fa5
      XFree (tpr.value);
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gint length = 0;
Packit Service fb6fa5
      gint i;
Packit Service fb6fa5
Packit Service fb6fa5
      while (src[length] != 0)
Packit Service fb6fa5
	length++;
Packit Service fb6fa5
      
Packit Service fb6fa5
      mbstr = g_new (gchar, length + 1);
Packit Service fb6fa5
Packit Service fb6fa5
      for (i=0; i
Packit Service fb6fa5
	mbstr[i] = src[i];
Packit Service fb6fa5
    }
Packit Service fb6fa5
Packit Service fb6fa5
  return mbstr;
Packit Service fb6fa5
}
Packit Service fb6fa5
/**
Packit Service fb6fa5
 * gdk_mbstowcs:
Packit Service fb6fa5
 * @dest: the space to place the converted wide character string into.
Packit Service fb6fa5
 * @src: the multi-byte string to convert, which must be nul-terminated.
Packit Service fb6fa5
 * @dest_max: the maximum number of wide characters to place in @dest.
Packit Service fb6fa5
 * 
Packit Service fb6fa5
 * Converts a multi-byte string to a wide character string.
Packit Service fb6fa5
 * (The function name comes from an acronym of 'Multi-Byte String TO Wide
Packit Service fb6fa5
 * Character String').
Packit Service fb6fa5
 * 
Packit Service fb6fa5
 * Return value: the number of wide characters written into @dest, or -1 if 
Packit Service fb6fa5
 *   the conversion failed.
Packit Service fb6fa5
 **/
Packit Service fb6fa5
  
Packit Service fb6fa5
gint
Packit Service fb6fa5
gdk_mbstowcs (GdkWChar *dest, const gchar *src, gint dest_max)
Packit Service fb6fa5
{
Packit Service fb6fa5
  if (gdk_use_mb)
Packit Service fb6fa5
    {
Packit Service fb6fa5
      GdkDisplay *display = find_a_display ();
Packit Service fb6fa5
      Display *xdisplay = GDK_DISPLAY_XDISPLAY (display);
Packit Service fb6fa5
      XTextProperty tpr;
Packit Service fb6fa5
      wchar_t **wstrs, *wstr_src;
Packit Service fb6fa5
      gint num_wstrs;
Packit Service fb6fa5
      gint len_cpy;
Packit Service fb6fa5
      if (XmbTextListToTextProperty (xdisplay, (char **)&src, 1, XTextStyle,
Packit Service fb6fa5
				     &tpr)
Packit Service fb6fa5
	  != Success)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  /* NoMem or LocaleNotSupp */
Packit Service fb6fa5
	  return -1;
Packit Service fb6fa5
	}
Packit Service fb6fa5
      if (XwcTextPropertyToTextList (xdisplay, &tpr, &wstrs, &num_wstrs)
Packit Service fb6fa5
	  != Success)
Packit Service fb6fa5
	{
Packit Service fb6fa5
	  /* InvalidChar */
Packit Service fb6fa5
	  XFree(tpr.value);
Packit Service fb6fa5
	  return -1;
Packit Service fb6fa5
	}
Packit Service fb6fa5
      XFree(tpr.value);
Packit Service fb6fa5
      if (num_wstrs == 0)
Packit Service fb6fa5
	return 0;
Packit Service fb6fa5
      wstr_src = wstrs[0];
Packit Service fb6fa5
      for (len_cpy=0; len_cpy
Packit Service fb6fa5
	dest[len_cpy] = wstr_src[len_cpy];
Packit Service fb6fa5
      XwcFreeStringList (wstrs);
Packit Service fb6fa5
      return len_cpy;
Packit Service fb6fa5
    }
Packit Service fb6fa5
  else
Packit Service fb6fa5
    {
Packit Service fb6fa5
      gint i;
Packit Service fb6fa5
Packit Service fb6fa5
      for (i=0; i
Packit Service fb6fa5
	dest[i] = src[i];
Packit Service fb6fa5
Packit Service fb6fa5
      return i;
Packit Service fb6fa5
    }
Packit Service fb6fa5
}
Packit Service fb6fa5
Packit Service fb6fa5
#define __GDK_IM_X11_C__
Packit Service fb6fa5
#include "gdkaliasdef.c"