|
Packit |
98cdb6 |
/* GTK - The GIMP Toolkit
|
|
Packit |
98cdb6 |
* Copyright (C) 2000 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 "config.h"
|
|
Packit |
98cdb6 |
#include "locale.h"
|
|
Packit |
98cdb6 |
#include <string.h>
|
|
Packit |
98cdb6 |
#include <stdlib.h>
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
#include "gtkimcontextxim.h"
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
#include "gtk/gtkintl.h"
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
typedef struct _StatusWindow StatusWindow;
|
|
Packit |
98cdb6 |
typedef struct _GtkXIMInfo GtkXIMInfo;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
struct _GtkIMContextXIM
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContext object;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
GtkXIMInfo *im_info;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gchar *locale;
|
|
Packit |
98cdb6 |
gchar *mb_charset;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
GdkWindow *client_window;
|
|
Packit |
98cdb6 |
GtkWidget *client_widget;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* The status window for this input context; we claim the
|
|
Packit |
98cdb6 |
* status window when we are focused and have created an XIC
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
StatusWindow *status_window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gint preedit_size;
|
|
Packit |
98cdb6 |
gint preedit_length;
|
|
Packit |
98cdb6 |
gunichar *preedit_chars;
|
|
Packit |
98cdb6 |
XIMFeedback *feedbacks;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gint preedit_cursor;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XIMCallback preedit_start_callback;
|
|
Packit |
98cdb6 |
XIMCallback preedit_done_callback;
|
|
Packit |
98cdb6 |
XIMCallback preedit_draw_callback;
|
|
Packit |
98cdb6 |
XIMCallback preedit_caret_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XIMCallback status_start_callback;
|
|
Packit |
98cdb6 |
XIMCallback status_done_callback;
|
|
Packit |
98cdb6 |
XIMCallback status_draw_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XIMCallback string_conversion_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XIC ic;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
guint filter_key_release : 1;
|
|
Packit |
98cdb6 |
guint use_preedit : 1;
|
|
Packit |
98cdb6 |
guint finalizing : 1;
|
|
Packit |
98cdb6 |
guint in_toplevel : 1;
|
|
Packit |
98cdb6 |
guint has_focus : 1;
|
|
Packit |
98cdb6 |
};
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
struct _GtkXIMInfo
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GdkScreen *screen;
|
|
Packit |
98cdb6 |
XIM im;
|
|
Packit |
98cdb6 |
char *locale;
|
|
Packit |
98cdb6 |
XIMStyle preedit_style_setting;
|
|
Packit |
98cdb6 |
XIMStyle status_style_setting;
|
|
Packit |
98cdb6 |
XIMStyle style;
|
|
Packit |
98cdb6 |
GtkSettings *settings;
|
|
Packit |
98cdb6 |
gulong status_set;
|
|
Packit |
98cdb6 |
gulong preedit_set;
|
|
Packit |
98cdb6 |
gulong display_closed_cb;
|
|
Packit |
98cdb6 |
XIMStyles *xim_styles;
|
|
Packit |
98cdb6 |
GSList *ics;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
guint reconnecting :1;
|
|
Packit |
98cdb6 |
guint supports_string_conversion;
|
|
Packit |
98cdb6 |
};
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* A context status window; these are kept in the status_windows list. */
|
|
Packit |
98cdb6 |
struct _StatusWindow
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Toplevel window to which the status window corresponds */
|
|
Packit |
98cdb6 |
GtkWidget *toplevel;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Currently focused GtkIMContextXIM for the toplevel, if any */
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context;
|
|
Packit |
98cdb6 |
};
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_class_init (GtkIMContextXIMClass *class);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_init (GtkIMContextXIM *im_context_xim);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_finalize (GObject *obj);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_set_client_window (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkWindow *client_window);
|
|
Packit |
98cdb6 |
static gboolean gtk_im_context_xim_filter_keypress (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkEventKey *key);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_reset (GtkIMContext *context);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_focus_in (GtkIMContext *context);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_focus_out (GtkIMContext *context);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_set_cursor_location (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkRectangle *area);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_set_use_preedit (GtkIMContext *context,
|
|
Packit |
98cdb6 |
gboolean use_preedit);
|
|
Packit |
98cdb6 |
static void gtk_im_context_xim_get_preedit_string (GtkIMContext *context,
|
|
Packit |
98cdb6 |
gchar **str,
|
|
Packit |
98cdb6 |
PangoAttrList **attrs,
|
|
Packit |
98cdb6 |
gint *cursor_pos);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void reinitialize_ic (GtkIMContextXIM *context_xim);
|
|
Packit |
98cdb6 |
static void set_ic_client_window (GtkIMContextXIM *context_xim,
|
|
Packit |
98cdb6 |
GdkWindow *client_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void setup_styles (GtkXIMInfo *info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void update_client_widget (GtkIMContextXIM *context_xim);
|
|
Packit |
98cdb6 |
static void update_status_window (GtkIMContextXIM *context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static StatusWindow *status_window_get (GtkWidget *toplevel);
|
|
Packit |
98cdb6 |
static void status_window_free (StatusWindow *status_window);
|
|
Packit |
98cdb6 |
static void status_window_set_text (StatusWindow *status_window,
|
|
Packit |
98cdb6 |
const gchar *text);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void xim_destroy_callback (XIM xim,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static XIC gtk_im_context_xim_get_ic (GtkIMContextXIM *context_xim);
|
|
Packit |
98cdb6 |
static void xim_info_display_closed (GdkDisplay *display,
|
|
Packit |
98cdb6 |
gboolean is_error,
|
|
Packit |
98cdb6 |
GtkXIMInfo *info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static GObjectClass *parent_class;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
GType gtk_type_im_context_xim = 0;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static GSList *open_ims = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* List of status windows for different toplevels */
|
|
Packit |
98cdb6 |
static GSList *status_windows = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_register_type (GTypeModule *type_module)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
const GTypeInfo im_context_xim_info =
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
sizeof (GtkIMContextXIMClass),
|
|
Packit |
98cdb6 |
(GBaseInitFunc) NULL,
|
|
Packit |
98cdb6 |
(GBaseFinalizeFunc) NULL,
|
|
Packit |
98cdb6 |
(GClassInitFunc) gtk_im_context_xim_class_init,
|
|
Packit |
98cdb6 |
NULL, /* class_finalize */
|
|
Packit |
98cdb6 |
NULL, /* class_data */
|
|
Packit |
98cdb6 |
sizeof (GtkIMContextXIM),
|
|
Packit |
98cdb6 |
0,
|
|
Packit |
98cdb6 |
(GInstanceInitFunc) gtk_im_context_xim_init,
|
|
Packit |
98cdb6 |
};
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_type_im_context_xim =
|
|
Packit |
98cdb6 |
g_type_module_register_type (type_module,
|
|
Packit |
98cdb6 |
GTK_TYPE_IM_CONTEXT,
|
|
Packit |
98cdb6 |
"GtkIMContextXIM",
|
|
Packit |
98cdb6 |
&im_context_xim_info, 0);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
#define PREEDIT_MASK (XIMPreeditCallbacks | XIMPreeditPosition | \
|
|
Packit |
98cdb6 |
XIMPreeditArea | XIMPreeditNothing | XIMPreeditNone)
|
|
Packit |
98cdb6 |
#define STATUS_MASK (XIMStatusCallbacks | XIMStatusArea | \
|
|
Packit |
98cdb6 |
XIMStatusNothing | XIMStatusNone)
|
|
Packit |
98cdb6 |
#define ALLOWED_MASK (XIMPreeditCallbacks | XIMPreeditNothing | XIMPreeditNone | \
|
|
Packit |
98cdb6 |
XIMStatusCallbacks | XIMStatusNothing | XIMStatusNone)
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static XIMStyle
|
|
Packit |
98cdb6 |
choose_better_style (XIMStyle style1, XIMStyle style2)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIMStyle s1, s2, u;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (style1 == 0) return style2;
|
|
Packit |
98cdb6 |
if (style2 == 0) return style1;
|
|
Packit |
98cdb6 |
if ((style1 & (PREEDIT_MASK | STATUS_MASK))
|
|
Packit |
98cdb6 |
== (style2 & (PREEDIT_MASK | STATUS_MASK)))
|
|
Packit |
98cdb6 |
return style1;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
s1 = style1 & PREEDIT_MASK;
|
|
Packit |
98cdb6 |
s2 = style2 & PREEDIT_MASK;
|
|
Packit |
98cdb6 |
u = s1 | s2;
|
|
Packit |
98cdb6 |
if (s1 != s2) {
|
|
Packit |
98cdb6 |
if (u & XIMPreeditCallbacks)
|
|
Packit |
98cdb6 |
return (s1 == XIMPreeditCallbacks) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMPreeditPosition)
|
|
Packit |
98cdb6 |
return (s1 == XIMPreeditPosition) ? style1 :style2;
|
|
Packit |
98cdb6 |
else if (u & XIMPreeditArea)
|
|
Packit |
98cdb6 |
return (s1 == XIMPreeditArea) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMPreeditNothing)
|
|
Packit |
98cdb6 |
return (s1 == XIMPreeditNothing) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMPreeditNone)
|
|
Packit |
98cdb6 |
return (s1 == XIMPreeditNone) ? style1 : style2;
|
|
Packit |
98cdb6 |
} else {
|
|
Packit |
98cdb6 |
s1 = style1 & STATUS_MASK;
|
|
Packit |
98cdb6 |
s2 = style2 & STATUS_MASK;
|
|
Packit |
98cdb6 |
u = s1 | s2;
|
|
Packit |
98cdb6 |
if (u & XIMStatusCallbacks)
|
|
Packit |
98cdb6 |
return (s1 == XIMStatusCallbacks) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMStatusArea)
|
|
Packit |
98cdb6 |
return (s1 == XIMStatusArea) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMStatusNothing)
|
|
Packit |
98cdb6 |
return (s1 == XIMStatusNothing) ? style1 : style2;
|
|
Packit |
98cdb6 |
else if (u & XIMStatusNone)
|
|
Packit |
98cdb6 |
return (s1 == XIMStatusNone) ? style1 : style2;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
return 0; /* Get rid of stupid warning */
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
reinitialize_all_ics (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GSList *tmp_list;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
for (tmp_list = info->ics; tmp_list; tmp_list = tmp_list->next)
|
|
Packit |
98cdb6 |
reinitialize_ic (tmp_list->data);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_style_change (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMStatusStyle status_style;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_object_get (info->settings,
|
|
Packit |
98cdb6 |
"gtk-im-status-style", &status_style,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
if (status_style == GTK_IM_STATUS_CALLBACK)
|
|
Packit |
98cdb6 |
info->status_style_setting = XIMStatusCallbacks;
|
|
Packit |
98cdb6 |
else if (status_style == GTK_IM_STATUS_NOTHING)
|
|
Packit |
98cdb6 |
info->status_style_setting = XIMStatusNothing;
|
|
Packit |
98cdb6 |
else if (status_style == GTK_IM_STATUS_NONE)
|
|
Packit |
98cdb6 |
info->status_style_setting = XIMStatusNone;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
setup_styles (info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
reinitialize_all_ics (info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
preedit_style_change (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMPreeditStyle preedit_style;
|
|
Packit |
98cdb6 |
g_object_get (info->settings,
|
|
Packit |
98cdb6 |
"gtk-im-preedit-style", &preedit_style,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
if (preedit_style == GTK_IM_PREEDIT_CALLBACK)
|
|
Packit |
98cdb6 |
info->preedit_style_setting = XIMPreeditCallbacks;
|
|
Packit |
98cdb6 |
else if (preedit_style == GTK_IM_PREEDIT_NOTHING)
|
|
Packit |
98cdb6 |
info->preedit_style_setting = XIMPreeditNothing;
|
|
Packit |
98cdb6 |
else if (preedit_style == GTK_IM_PREEDIT_NONE)
|
|
Packit |
98cdb6 |
info->preedit_style_setting = XIMPreeditNone;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
setup_styles (info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
reinitialize_all_ics (info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
setup_styles (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
int i;
|
|
Packit |
98cdb6 |
unsigned long settings_preference;
|
|
Packit |
98cdb6 |
XIMStyles *xim_styles = info->xim_styles;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
settings_preference = info->status_style_setting|info->preedit_style_setting;
|
|
Packit |
98cdb6 |
info->style = 0;
|
|
Packit |
98cdb6 |
if (xim_styles)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
for (i = 0; i < xim_styles->count_styles; i++)
|
|
Packit |
98cdb6 |
if ((xim_styles->supported_styles[i] & ALLOWED_MASK) == xim_styles->supported_styles[i])
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (settings_preference == xim_styles->supported_styles[i])
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
info->style = settings_preference;
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
info->style = choose_better_style (info->style,
|
|
Packit |
98cdb6 |
xim_styles->supported_styles[i]);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
if (info->style == 0)
|
|
Packit |
98cdb6 |
info->style = XIMPreeditNothing | XIMStatusNothing;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
setup_im (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIMValuesList *ic_values = NULL;
|
|
Packit |
98cdb6 |
XIMCallback im_destroy_callback;
|
|
Packit |
98cdb6 |
GdkDisplay *display;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (info->im == NULL)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
im_destroy_callback.client_data = (XPointer)info;
|
|
Packit |
98cdb6 |
im_destroy_callback.callback = (XIMProc)xim_destroy_callback;
|
|
Packit |
98cdb6 |
XSetIMValues (info->im,
|
|
Packit |
98cdb6 |
XNDestroyCallback, &im_destroy_callback,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XGetIMValues (info->im,
|
|
Packit |
98cdb6 |
XNQueryInputStyle, &info->xim_styles,
|
|
Packit |
98cdb6 |
XNQueryICValuesList, &ic_values,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info->settings = gtk_settings_get_for_screen (info->screen);
|
|
Packit |
98cdb6 |
info->status_set = g_signal_connect_swapped (info->settings,
|
|
Packit |
98cdb6 |
"notify::gtk-im-status-style",
|
|
Packit |
98cdb6 |
G_CALLBACK (status_style_change),
|
|
Packit |
98cdb6 |
info);
|
|
Packit |
98cdb6 |
info->preedit_set = g_signal_connect_swapped (info->settings,
|
|
Packit |
98cdb6 |
"notify::gtk-im-preedit-style",
|
|
Packit |
98cdb6 |
G_CALLBACK (preedit_style_change),
|
|
Packit |
98cdb6 |
info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info->supports_string_conversion = FALSE;
|
|
Packit |
98cdb6 |
if (ic_values)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
int i;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
for (i = 0; i < ic_values->count_values; i++)
|
|
Packit |
98cdb6 |
if (strcmp (ic_values->supported_values[i],
|
|
Packit |
98cdb6 |
XNStringConversionCallback) == 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
info->supports_string_conversion = TRUE;
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
#if 0
|
|
Packit |
98cdb6 |
for (i = 0; i < ic_values->count_values; i++)
|
|
Packit |
98cdb6 |
g_print ("%s\n", ic_values->supported_values[i]);
|
|
Packit |
98cdb6 |
for (i = 0; i < xim_styles->count_styles; i++)
|
|
Packit |
98cdb6 |
g_print ("%#x\n", xim_styles->supported_styles[i]);
|
|
Packit |
98cdb6 |
#endif
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XFree (ic_values);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_style_change (info);
|
|
Packit |
98cdb6 |
preedit_style_change (info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
display = gdk_screen_get_display (info->screen);
|
|
Packit |
98cdb6 |
info->display_closed_cb = g_signal_connect (display, "closed",
|
|
Packit |
98cdb6 |
G_CALLBACK (xim_info_display_closed), info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
xim_info_display_closed (GdkDisplay *display,
|
|
Packit |
98cdb6 |
gboolean is_error,
|
|
Packit |
98cdb6 |
GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GSList *ics, *tmp_list;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
open_ims = g_slist_remove (open_ims, info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
ics = info->ics;
|
|
Packit |
98cdb6 |
info->ics = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
for (tmp_list = ics; tmp_list; tmp_list = tmp_list->next)
|
|
Packit |
98cdb6 |
set_ic_client_window (tmp_list->data, NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_slist_free (ics);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (info->status_set)
|
|
Packit |
98cdb6 |
g_signal_handler_disconnect (info->settings, info->status_set);
|
|
Packit |
98cdb6 |
if (info->preedit_set)
|
|
Packit |
98cdb6 |
g_signal_handler_disconnect (info->settings, info->preedit_set);
|
|
Packit |
98cdb6 |
if (info->display_closed_cb)
|
|
Packit |
98cdb6 |
g_signal_handler_disconnect (display, info->display_closed_cb);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (info->xim_styles)
|
|
Packit |
98cdb6 |
XFree (info->xim_styles);
|
|
Packit |
98cdb6 |
g_free (info->locale);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (info->im)
|
|
Packit |
98cdb6 |
XCloseIM (info->im);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
xim_instantiate_callback (Display *display, XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkXIMInfo *info = (GtkXIMInfo*)client_data;
|
|
Packit |
98cdb6 |
XIM im = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
im = XOpenIM (display, NULL, NULL, NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!im)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info->im = im;
|
|
Packit |
98cdb6 |
setup_im (info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XUnregisterIMInstantiateCallback (display, NULL, NULL, NULL,
|
|
Packit |
98cdb6 |
xim_instantiate_callback,
|
|
Packit |
98cdb6 |
(XPointer)info);
|
|
Packit |
98cdb6 |
info->reconnecting = FALSE;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* initialize info->im */
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
xim_info_try_im (GtkXIMInfo *info)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GdkScreen *screen = info->screen;
|
|
Packit |
98cdb6 |
GdkDisplay *display = gdk_screen_get_display (screen);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_assert (info->im == NULL);
|
|
Packit |
98cdb6 |
if (info->reconnecting)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (XSupportsLocale ())
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (!XSetLocaleModifiers (""))
|
|
Packit |
98cdb6 |
g_warning ("Unable to set locale modifiers with XSetLocaleModifiers()");
|
|
Packit |
98cdb6 |
info->im = XOpenIM (GDK_DISPLAY_XDISPLAY (display), NULL, NULL, NULL);
|
|
Packit |
98cdb6 |
if (!info->im)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XRegisterIMInstantiateCallback (GDK_DISPLAY_XDISPLAY(display),
|
|
Packit |
98cdb6 |
NULL, NULL, NULL,
|
|
Packit |
98cdb6 |
xim_instantiate_callback,
|
|
Packit |
98cdb6 |
(XPointer)info);
|
|
Packit |
98cdb6 |
info->reconnecting = TRUE;
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
setup_im (info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
xim_destroy_callback (XIM xim,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkXIMInfo *info = (GtkXIMInfo*)client_data;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info->im = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_signal_handler_disconnect (info->settings, info->status_set);
|
|
Packit |
98cdb6 |
info->status_set = 0;
|
|
Packit |
98cdb6 |
g_signal_handler_disconnect (info->settings, info->preedit_set);
|
|
Packit |
98cdb6 |
info->preedit_set = 0;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
reinitialize_all_ics (info);
|
|
Packit |
98cdb6 |
xim_info_try_im (info);
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static GtkXIMInfo *
|
|
Packit |
98cdb6 |
get_im (GdkWindow *client_window,
|
|
Packit |
98cdb6 |
const char *locale)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GSList *tmp_list;
|
|
Packit |
98cdb6 |
GtkXIMInfo *info;
|
|
Packit |
98cdb6 |
GdkScreen *screen = gdk_window_get_screen (client_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info = NULL;
|
|
Packit |
98cdb6 |
tmp_list = open_ims;
|
|
Packit |
98cdb6 |
while (tmp_list)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkXIMInfo *tmp_info = tmp_list->data;
|
|
Packit |
98cdb6 |
if (tmp_info->screen == screen &&
|
|
Packit |
98cdb6 |
strcmp (tmp_info->locale, locale) == 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (tmp_info->im)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
return tmp_info;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
tmp_info = tmp_info;
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
tmp_list = tmp_list->next;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (info == NULL)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
info = g_new (GtkXIMInfo, 1);
|
|
Packit |
98cdb6 |
open_ims = g_slist_prepend (open_ims, info);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
info->screen = screen;
|
|
Packit |
98cdb6 |
info->locale = g_strdup (locale);
|
|
Packit |
98cdb6 |
info->xim_styles = NULL;
|
|
Packit |
98cdb6 |
info->preedit_style_setting = 0;
|
|
Packit |
98cdb6 |
info->status_style_setting = 0;
|
|
Packit |
98cdb6 |
info->settings = NULL;
|
|
Packit |
98cdb6 |
info->preedit_set = 0;
|
|
Packit |
98cdb6 |
info->status_set = 0;
|
|
Packit |
98cdb6 |
info->display_closed_cb = 0;
|
|
Packit |
98cdb6 |
info->ics = NULL;
|
|
Packit |
98cdb6 |
info->reconnecting = FALSE;
|
|
Packit |
98cdb6 |
info->im = NULL;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
xim_info_try_im (info);
|
|
Packit |
98cdb6 |
return info;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_class_init (GtkIMContextXIMClass *class)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextClass *im_context_class = GTK_IM_CONTEXT_CLASS (class);
|
|
Packit |
98cdb6 |
GObjectClass *gobject_class = G_OBJECT_CLASS (class);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
parent_class = g_type_class_peek_parent (class);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
im_context_class->set_client_window = gtk_im_context_xim_set_client_window;
|
|
Packit |
98cdb6 |
im_context_class->filter_keypress = gtk_im_context_xim_filter_keypress;
|
|
Packit |
98cdb6 |
im_context_class->reset = gtk_im_context_xim_reset;
|
|
Packit |
98cdb6 |
im_context_class->get_preedit_string = gtk_im_context_xim_get_preedit_string;
|
|
Packit |
98cdb6 |
im_context_class->focus_in = gtk_im_context_xim_focus_in;
|
|
Packit |
98cdb6 |
im_context_class->focus_out = gtk_im_context_xim_focus_out;
|
|
Packit |
98cdb6 |
im_context_class->set_cursor_location = gtk_im_context_xim_set_cursor_location;
|
|
Packit |
98cdb6 |
im_context_class->set_use_preedit = gtk_im_context_xim_set_use_preedit;
|
|
Packit |
98cdb6 |
gobject_class->finalize = gtk_im_context_xim_finalize;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_init (GtkIMContextXIM *im_context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
im_context_xim->use_preedit = TRUE;
|
|
Packit |
98cdb6 |
im_context_xim->filter_key_release = FALSE;
|
|
Packit |
98cdb6 |
im_context_xim->finalizing = FALSE;
|
|
Packit |
98cdb6 |
im_context_xim->has_focus = FALSE;
|
|
Packit |
98cdb6 |
im_context_xim->in_toplevel = FALSE;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_finalize (GObject *obj)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (obj);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->finalizing = TRUE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->im_info && !context_xim->im_info->ics->next)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->im_info->reconnecting)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GdkDisplay *display;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
display = gdk_screen_get_display (context_xim->im_info->screen);
|
|
Packit |
98cdb6 |
XUnregisterIMInstantiateCallback (GDK_DISPLAY_XDISPLAY (display),
|
|
Packit |
98cdb6 |
NULL, NULL, NULL,
|
|
Packit |
98cdb6 |
xim_instantiate_callback,
|
|
Packit |
98cdb6 |
(XPointer)context_xim->im_info);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else if (context_xim->im_info->im)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIMCallback im_destroy_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
im_destroy_callback.client_data = NULL;
|
|
Packit |
98cdb6 |
im_destroy_callback.callback = NULL;
|
|
Packit |
98cdb6 |
XSetIMValues (context_xim->im_info->im,
|
|
Packit |
98cdb6 |
XNDestroyCallback, &im_destroy_callback,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
set_ic_client_window (context_xim, NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (context_xim->locale);
|
|
Packit |
98cdb6 |
g_free (context_xim->mb_charset);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
G_OBJECT_CLASS (parent_class)->finalize (obj);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
reinitialize_ic (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->ic)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XDestroyIC (context_xim->ic);
|
|
Packit |
98cdb6 |
context_xim->ic = NULL;
|
|
Packit |
98cdb6 |
update_status_window (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->preedit_length)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->preedit_length = 0;
|
|
Packit |
98cdb6 |
if (!context_xim->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context_xim, "preedit-changed");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
/*
|
|
Packit |
98cdb6 |
reset filter_key_release flag, otherwise keystrokes will be doubled
|
|
Packit |
98cdb6 |
until reconnecting to XIM.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
context_xim->filter_key_release = FALSE;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
set_ic_client_window (GtkIMContextXIM *context_xim,
|
|
Packit |
98cdb6 |
GdkWindow *client_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
reinitialize_ic (context_xim);
|
|
Packit |
98cdb6 |
if (context_xim->client_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->im_info->ics = g_slist_remove (context_xim->im_info->ics, context_xim);
|
|
Packit |
98cdb6 |
context_xim->im_info = NULL;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->client_window = client_window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->client_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->im_info = get_im (context_xim->client_window, context_xim->locale);
|
|
Packit |
98cdb6 |
context_xim->im_info->ics = g_slist_prepend (context_xim->im_info->ics, context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
update_client_widget (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_set_client_window (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkWindow *client_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
set_ic_client_window (context_xim, client_window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
GtkIMContext *
|
|
Packit |
98cdb6 |
gtk_im_context_xim_new (void)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *result;
|
|
Packit |
98cdb6 |
const gchar *charset;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
result = g_object_new (GTK_TYPE_IM_CONTEXT_XIM, NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
result->locale = g_strdup (setlocale (LC_CTYPE, NULL));
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_get_charset (&charset);
|
|
Packit |
98cdb6 |
result->mb_charset = g_strdup (charset);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return GTK_IM_CONTEXT (result);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static char *
|
|
Packit |
98cdb6 |
mb_to_utf8 (GtkIMContextXIM *context_xim,
|
|
Packit |
98cdb6 |
const char *str)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GError *error = NULL;
|
|
Packit |
98cdb6 |
gchar *result;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (strcmp (context_xim->mb_charset, "UTF-8") == 0)
|
|
Packit |
98cdb6 |
result = g_strdup (str);
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
result = g_convert (str, -1,
|
|
Packit |
98cdb6 |
"UTF-8", context_xim->mb_charset,
|
|
Packit |
98cdb6 |
NULL, NULL, &error);
|
|
Packit |
98cdb6 |
if (!result)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_warning ("Error converting text from IM to UTF-8: %s\n", error->message);
|
|
Packit |
98cdb6 |
g_error_free (error);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return result;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static gboolean
|
|
Packit |
98cdb6 |
gtk_im_context_xim_filter_keypress (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkEventKey *event)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
XIC ic = gtk_im_context_xim_get_ic (context_xim);
|
|
Packit |
98cdb6 |
gchar static_buffer[256];
|
|
Packit |
98cdb6 |
gchar *buffer = static_buffer;
|
|
Packit |
98cdb6 |
gint buffer_size = sizeof(static_buffer) - 1;
|
|
Packit |
98cdb6 |
gint num_bytes = 0;
|
|
Packit |
98cdb6 |
KeySym keysym;
|
|
Packit |
98cdb6 |
Status status;
|
|
Packit |
98cdb6 |
gboolean result = FALSE;
|
|
Packit |
98cdb6 |
GdkWindow *root_window = gdk_screen_get_root_window (gdk_window_get_screen (event->window));
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XKeyPressedEvent xevent;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (event->type == GDK_KEY_RELEASE && !context_xim->filter_key_release)
|
|
Packit |
98cdb6 |
return FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
xevent.type = (event->type == GDK_KEY_PRESS) ? KeyPress : KeyRelease;
|
|
Packit |
98cdb6 |
xevent.serial = 0; /* hope it doesn't matter */
|
|
Packit |
98cdb6 |
xevent.send_event = event->send_event;
|
|
Packit |
98cdb6 |
xevent.display = GDK_DRAWABLE_XDISPLAY (event->window);
|
|
Packit |
98cdb6 |
xevent.window = GDK_DRAWABLE_XID (event->window);
|
|
Packit |
98cdb6 |
xevent.root = GDK_DRAWABLE_XID (root_window);
|
|
Packit |
98cdb6 |
xevent.subwindow = xevent.window;
|
|
Packit |
98cdb6 |
xevent.time = event->time;
|
|
Packit |
98cdb6 |
xevent.x = xevent.x_root = 0;
|
|
Packit |
98cdb6 |
xevent.y = xevent.y_root = 0;
|
|
Packit |
98cdb6 |
xevent.state = event->state;
|
|
Packit |
98cdb6 |
xevent.keycode = event->hardware_keycode;
|
|
Packit |
98cdb6 |
xevent.same_screen = True;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (XFilterEvent ((XEvent *)&xevent, GDK_DRAWABLE_XID (context_xim->client_window)))
|
|
Packit |
98cdb6 |
return TRUE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (event->state &
|
|
Packit |
98cdb6 |
(gtk_accelerator_get_default_mod_mask () & ~(GDK_SHIFT_MASK | GDK_CONTROL_MASK)))
|
|
Packit |
98cdb6 |
return FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
again:
|
|
Packit |
98cdb6 |
if (ic)
|
|
Packit |
98cdb6 |
num_bytes = XmbLookupString (ic, &xevent, buffer, buffer_size, &keysym, &status);
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
num_bytes = XLookupString (&xevent, buffer, buffer_size, &keysym, NULL);
|
|
Packit |
98cdb6 |
status = XLookupBoth;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (status == XBufferOverflow)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
buffer_size = num_bytes;
|
|
Packit |
98cdb6 |
if (buffer != static_buffer)
|
|
Packit |
98cdb6 |
g_free (buffer);
|
|
Packit |
98cdb6 |
buffer = g_malloc (num_bytes + 1);
|
|
Packit |
98cdb6 |
goto again;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* I don't know how we should properly handle XLookupKeysym or XLookupBoth
|
|
Packit |
98cdb6 |
* here ... do input methods actually change the keysym? we can't really
|
|
Packit |
98cdb6 |
* feed it back to accelerator processing at this point...
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
if (status == XLookupChars || status == XLookupBoth)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
char *result_utf8;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
buffer[num_bytes] = '\0';
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
result_utf8 = mb_to_utf8 (context_xim, buffer);
|
|
Packit |
98cdb6 |
if (result_utf8)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if ((guchar)result_utf8[0] >= 0x20 &&
|
|
Packit |
98cdb6 |
result_utf8[0] != 0x7f) /* Some IM have a nasty habit of converting
|
|
Packit |
98cdb6 |
* control characters into strings
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "commit", result_utf8);
|
|
Packit |
98cdb6 |
result = TRUE;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (result_utf8);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (buffer != static_buffer)
|
|
Packit |
98cdb6 |
g_free (buffer);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return result;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_focus_in (GtkIMContext *context)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!context_xim->has_focus)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIC ic = gtk_im_context_xim_get_ic (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->has_focus = TRUE;
|
|
Packit |
98cdb6 |
update_status_window (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (ic)
|
|
Packit |
98cdb6 |
XSetICFocus (ic);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_focus_out (GtkIMContext *context)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->has_focus)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIC ic = gtk_im_context_xim_get_ic (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->has_focus = FALSE;
|
|
Packit |
98cdb6 |
update_status_window (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (ic)
|
|
Packit |
98cdb6 |
XUnsetICFocus (ic);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_set_cursor_location (GtkIMContext *context,
|
|
Packit |
98cdb6 |
GdkRectangle *area)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
XIC ic = gtk_im_context_xim_get_ic (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XVaNestedList preedit_attr;
|
|
Packit |
98cdb6 |
XPoint spot;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!ic)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
spot.x = area->x;
|
|
Packit |
98cdb6 |
spot.y = area->y + area->height;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
preedit_attr = XVaCreateNestedList (0,
|
|
Packit |
98cdb6 |
XNSpotLocation, &spot,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
XSetICValues (ic,
|
|
Packit |
98cdb6 |
XNPreeditAttributes, preedit_attr,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
XFree(preedit_attr);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_set_use_preedit (GtkIMContext *context,
|
|
Packit |
98cdb6 |
gboolean use_preedit)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
use_preedit = use_preedit != FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->use_preedit != use_preedit)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->use_preedit = use_preedit;
|
|
Packit |
98cdb6 |
reinitialize_ic (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_reset (GtkIMContext *context)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
XIC ic = gtk_im_context_xim_get_ic (context_xim);
|
|
Packit |
98cdb6 |
gchar *result;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* restore conversion state after resetting ic later */
|
|
Packit |
98cdb6 |
XIMPreeditState preedit_state = XIMPreeditUnKnown;
|
|
Packit |
98cdb6 |
XVaNestedList preedit_attr;
|
|
Packit |
98cdb6 |
gboolean have_preedit_state = FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!ic)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->preedit_length == 0)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
preedit_attr = XVaCreateNestedList(0,
|
|
Packit |
98cdb6 |
XNPreeditState, &preedit_state,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
if (!XGetICValues(ic,
|
|
Packit |
98cdb6 |
XNPreeditAttributes, preedit_attr,
|
|
Packit |
98cdb6 |
NULL))
|
|
Packit |
98cdb6 |
have_preedit_state = TRUE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XFree(preedit_attr);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
result = XmbResetIC (ic);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
preedit_attr = XVaCreateNestedList(0,
|
|
Packit |
98cdb6 |
XNPreeditState, preedit_state,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
if (have_preedit_state)
|
|
Packit |
98cdb6 |
XSetICValues(ic,
|
|
Packit |
98cdb6 |
XNPreeditAttributes, preedit_attr,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XFree(preedit_attr);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (result)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
char *result_utf8 = mb_to_utf8 (context_xim, result);
|
|
Packit |
98cdb6 |
if (result_utf8)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "commit", result_utf8);
|
|
Packit |
98cdb6 |
g_free (result_utf8);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->preedit_length)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->preedit_length = 0;
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "preedit-changed");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XFree (result);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Mask of feedback bits that we render
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
#define FEEDBACK_MASK (XIMReverse | XIMUnderline)
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
add_feedback_attr (PangoAttrList *attrs,
|
|
Packit |
98cdb6 |
const gchar *str,
|
|
Packit |
98cdb6 |
XIMFeedback feedback,
|
|
Packit |
98cdb6 |
gint start_pos,
|
|
Packit |
98cdb6 |
gint end_pos)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
PangoAttribute *attr;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gint start_index = g_utf8_offset_to_pointer (str, start_pos) - str;
|
|
Packit |
98cdb6 |
gint end_index = g_utf8_offset_to_pointer (str, end_pos) - str;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (feedback & XIMUnderline)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
attr = pango_attr_underline_new (PANGO_UNDERLINE_SINGLE);
|
|
Packit |
98cdb6 |
attr->start_index = start_index;
|
|
Packit |
98cdb6 |
attr->end_index = end_index;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
pango_attr_list_change (attrs, attr);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (feedback & XIMReverse)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
attr = pango_attr_foreground_new (0xffff, 0xffff, 0xffff);
|
|
Packit |
98cdb6 |
attr->start_index = start_index;
|
|
Packit |
98cdb6 |
attr->end_index = end_index;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
pango_attr_list_change (attrs, attr);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
attr = pango_attr_background_new (0, 0, 0);
|
|
Packit |
98cdb6 |
attr->start_index = start_index;
|
|
Packit |
98cdb6 |
attr->end_index = end_index;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
pango_attr_list_change (attrs, attr);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (feedback & ~FEEDBACK_MASK)
|
|
Packit |
98cdb6 |
g_warning ("Unrendered feedback style: %#lx", feedback & ~FEEDBACK_MASK);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_get_preedit_string (GtkIMContext *context,
|
|
Packit |
98cdb6 |
gchar **str,
|
|
Packit |
98cdb6 |
PangoAttrList **attrs,
|
|
Packit |
98cdb6 |
gint *cursor_pos)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
gchar *utf8 = g_ucs4_to_utf8 (context_xim->preedit_chars, context_xim->preedit_length, NULL, NULL, NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (attrs)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
int i;
|
|
Packit |
98cdb6 |
XIMFeedback last_feedback = 0;
|
|
Packit |
98cdb6 |
gint start = -1;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
*attrs = pango_attr_list_new ();
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
for (i = 0; i < context_xim->preedit_length; i++)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
XIMFeedback new_feedback = context_xim->feedbacks[i] & FEEDBACK_MASK;
|
|
Packit |
98cdb6 |
if (new_feedback != last_feedback)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (start >= 0)
|
|
Packit |
98cdb6 |
add_feedback_attr (*attrs, utf8, last_feedback, start, i);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
last_feedback = new_feedback;
|
|
Packit |
98cdb6 |
start = i;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (start >= 0)
|
|
Packit |
98cdb6 |
add_feedback_attr (*attrs, utf8, last_feedback, start, i);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (str)
|
|
Packit |
98cdb6 |
*str = utf8;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
g_free (utf8);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (cursor_pos)
|
|
Packit |
98cdb6 |
*cursor_pos = context_xim->preedit_cursor;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static int
|
|
Packit |
98cdb6 |
preedit_start_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContext *context = GTK_IM_CONTEXT (client_data);
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!context_xim->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "preedit-start");
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return -1; /* No length limit */
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
preedit_done_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContext *context = GTK_IM_CONTEXT (client_data);
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->preedit_length)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->preedit_length = 0;
|
|
Packit |
98cdb6 |
if (!context_xim->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context_xim, "preedit-changed");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!context_xim->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "preedit-end");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static gint
|
|
Packit |
98cdb6 |
xim_text_to_utf8 (GtkIMContextXIM *context, XIMText *xim_text, gchar **text)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
gint text_length = 0;
|
|
Packit |
98cdb6 |
GError *error = NULL;
|
|
Packit |
98cdb6 |
gchar *result = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (xim_text && xim_text->string.multi_byte)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (xim_text->encoding_is_wchar)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_warning ("Wide character return from Xlib not currently supported");
|
|
Packit |
98cdb6 |
*text = NULL;
|
|
Packit |
98cdb6 |
return 0;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (strcmp (context->mb_charset, "UTF-8") == 0)
|
|
Packit |
98cdb6 |
result = g_strdup (xim_text->string.multi_byte);
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
result = g_convert (xim_text->string.multi_byte,
|
|
Packit |
98cdb6 |
-1,
|
|
Packit |
98cdb6 |
"UTF-8",
|
|
Packit |
98cdb6 |
context->mb_charset,
|
|
Packit |
98cdb6 |
NULL, NULL, &error);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (result)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
text_length = g_utf8_strlen (result, -1);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (text_length != xim_text->length)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_warning ("Size mismatch when converting text from input method: supplied length = %d\n, result length = %d", xim_text->length, text_length);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_warning ("Error converting text from IM to UCS-4: %s", error->message);
|
|
Packit |
98cdb6 |
g_error_free (error);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
*text = NULL;
|
|
Packit |
98cdb6 |
return 0;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
*text = result;
|
|
Packit |
98cdb6 |
return text_length;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
*text = NULL;
|
|
Packit |
98cdb6 |
return 0;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
preedit_draw_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XIMPreeditDrawCallbackStruct *call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context = GTK_IM_CONTEXT_XIM (client_data);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XIMText *new_xim_text = call_data->text;
|
|
Packit |
98cdb6 |
gint new_text_length;
|
|
Packit |
98cdb6 |
gunichar *new_text = NULL;
|
|
Packit |
98cdb6 |
gint i;
|
|
Packit |
98cdb6 |
gint diff;
|
|
Packit |
98cdb6 |
gint new_length;
|
|
Packit |
98cdb6 |
gchar *tmp;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gint chg_first = CLAMP (call_data->chg_first, 0, context->preedit_length);
|
|
Packit |
98cdb6 |
gint chg_length = CLAMP (call_data->chg_length, 0, context->preedit_length - chg_first);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context->preedit_cursor = call_data->caret;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (chg_first != call_data->chg_first || chg_length != call_data->chg_length)
|
|
Packit |
98cdb6 |
g_warning ("Invalid change to preedit string, first=%d length=%d (orig length == %d)",
|
|
Packit |
98cdb6 |
call_data->chg_first, call_data->chg_length, context->preedit_length);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
new_text_length = xim_text_to_utf8 (context, new_xim_text, &tmp);
|
|
Packit |
98cdb6 |
if (tmp)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
new_text = g_utf8_to_ucs4_fast (tmp, -1, NULL);
|
|
Packit |
98cdb6 |
g_free (tmp);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
diff = new_text_length - chg_length;
|
|
Packit |
98cdb6 |
new_length = context->preedit_length + diff;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (new_length > context->preedit_size)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context->preedit_size = new_length;
|
|
Packit |
98cdb6 |
context->preedit_chars = g_renew (gunichar, context->preedit_chars, new_length);
|
|
Packit |
98cdb6 |
context->feedbacks = g_renew (XIMFeedback, context->feedbacks, new_length);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (diff < 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
for (i = chg_first + chg_length ; i < context->preedit_length; i++)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context->preedit_chars[i + diff] = context->preedit_chars[i];
|
|
Packit |
98cdb6 |
context->feedbacks[i + diff] = context->feedbacks[i];
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
for (i = context->preedit_length - 1; i >= chg_first + chg_length ; i--)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context->preedit_chars[i + diff] = context->preedit_chars[i];
|
|
Packit |
98cdb6 |
context->feedbacks[i + diff] = context->feedbacks[i];
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
for (i = 0; i < new_text_length; i++)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context->preedit_chars[chg_first + i] = new_text[i];
|
|
Packit |
98cdb6 |
context->feedbacks[chg_first + i] = new_xim_text->feedback[i];
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context->preedit_length += diff;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (new_text);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!context->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "preedit-changed");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
preedit_caret_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XIMPreeditCaretCallbackStruct *call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context = GTK_IM_CONTEXT_XIM (client_data);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (call_data->direction == XIMAbsolutePosition)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context->preedit_cursor = call_data->position;
|
|
Packit |
98cdb6 |
if (!context->finalizing)
|
|
Packit |
98cdb6 |
g_signal_emit_by_name (context, "preedit-changed");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_warning ("Caret movement command: %d %d %d not supported",
|
|
Packit |
98cdb6 |
call_data->position, call_data->direction, call_data->style);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_start_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_done_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_draw_callback (XIC xic,
|
|
Packit |
98cdb6 |
XPointer client_data,
|
|
Packit |
98cdb6 |
XIMStatusDrawCallbackStruct *call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context = GTK_IM_CONTEXT_XIM (client_data);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (call_data->type == XIMTextType)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
gchar *text;
|
|
Packit |
98cdb6 |
xim_text_to_utf8 (context, call_data->data.text, &text);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context->status_window)
|
|
Packit |
98cdb6 |
status_window_set_text (context->status_window, text ? text : "");
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else /* bitmap */
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_print ("Status drawn with bitmap - id = %#lx\n", call_data->data.bitmap);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
string_conversion_callback (XIC xic, XPointer client_data, XPointer call_data)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim;
|
|
Packit |
98cdb6 |
XIMStringConversionCallbackStruct *conv_data;
|
|
Packit |
98cdb6 |
gchar *surrounding;
|
|
Packit |
98cdb6 |
gint cursor_index;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim = (GtkIMContextXIM *)client_data;
|
|
Packit |
98cdb6 |
conv_data = (XIMStringConversionCallbackStruct *)call_data;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (gtk_im_context_get_surrounding ((GtkIMContext *)context_xim,
|
|
Packit |
98cdb6 |
&surrounding, &cursor_index))
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
gchar *text = NULL;
|
|
Packit |
98cdb6 |
gsize text_len = 0;
|
|
Packit |
98cdb6 |
gint subst_offset = 0, subst_nchars = 0;
|
|
Packit |
98cdb6 |
gint i;
|
|
Packit |
98cdb6 |
gchar *p = surrounding + cursor_index, *q;
|
|
Packit |
98cdb6 |
gshort position = (gshort)conv_data->position;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (position > 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
for (i = position; i > 0 && *p; --i)
|
|
Packit |
98cdb6 |
p = g_utf8_next_char (p);
|
|
Packit |
98cdb6 |
if (i > 0)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
/* According to X11R6.4 Xlib - C Library Reference Manual
|
|
Packit |
98cdb6 |
* section 13.5.7.3 String Conversion Callback,
|
|
Packit |
98cdb6 |
* XIMStringConversionPosition is starting position _relative_
|
|
Packit |
98cdb6 |
* to current client's cursor position. So it should be able
|
|
Packit |
98cdb6 |
* to be negative, or referring to a position before the cursor
|
|
Packit |
98cdb6 |
* would be impossible. But current X protocol defines this as
|
|
Packit |
98cdb6 |
* unsigned short. So, compiler may warn about the value range
|
|
Packit |
98cdb6 |
* here. We hope the X protocol is fixed soon.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
else if (position < 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
for (i = position; i < 0 && p > surrounding; ++i)
|
|
Packit |
98cdb6 |
p = g_utf8_prev_char (p);
|
|
Packit |
98cdb6 |
if (i < 0)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
switch (conv_data->direction)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
case XIMForwardChar:
|
|
Packit |
98cdb6 |
for (i = conv_data->factor, q = p; i > 0 && *q; --i)
|
|
Packit |
98cdb6 |
q = g_utf8_next_char (q);
|
|
Packit |
98cdb6 |
if (i > 0)
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
text = g_locale_from_utf8 (p, q - p, NULL, &text_len, NULL);
|
|
Packit |
98cdb6 |
subst_offset = position;
|
|
Packit |
98cdb6 |
subst_nchars = conv_data->factor;
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
case XIMBackwardChar:
|
|
Packit |
98cdb6 |
for (i = conv_data->factor, q = p; i > 0 && q > surrounding; --i)
|
|
Packit |
98cdb6 |
q = g_utf8_prev_char (q);
|
|
Packit |
98cdb6 |
if (i > 0)
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
text = g_locale_from_utf8 (q, p - q, NULL, &text_len, NULL);
|
|
Packit |
98cdb6 |
subst_offset = position - conv_data->factor;
|
|
Packit |
98cdb6 |
subst_nchars = conv_data->factor;
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
case XIMForwardWord:
|
|
Packit |
98cdb6 |
case XIMBackwardWord:
|
|
Packit |
98cdb6 |
case XIMCaretUp:
|
|
Packit |
98cdb6 |
case XIMCaretDown:
|
|
Packit |
98cdb6 |
case XIMNextLine:
|
|
Packit |
98cdb6 |
case XIMPreviousLine:
|
|
Packit |
98cdb6 |
case XIMLineStart:
|
|
Packit |
98cdb6 |
case XIMLineEnd:
|
|
Packit |
98cdb6 |
case XIMAbsolutePosition:
|
|
Packit |
98cdb6 |
case XIMDontChange:
|
|
Packit |
98cdb6 |
default:
|
|
Packit |
98cdb6 |
break;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
/* block out any failure happenning to "text", including conversion */
|
|
Packit |
98cdb6 |
if (text)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
conv_data->text = (XIMStringConversionText *)
|
|
Packit |
98cdb6 |
malloc (sizeof (XIMStringConversionText));
|
|
Packit |
98cdb6 |
if (conv_data->text)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
conv_data->text->length = text_len;
|
|
Packit |
98cdb6 |
conv_data->text->feedback = NULL;
|
|
Packit |
98cdb6 |
conv_data->text->encoding_is_wchar = False;
|
|
Packit |
98cdb6 |
conv_data->text->string.mbs = (char *)malloc (text_len);
|
|
Packit |
98cdb6 |
if (conv_data->text->string.mbs)
|
|
Packit |
98cdb6 |
memcpy (conv_data->text->string.mbs, text, text_len);
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
free (conv_data->text);
|
|
Packit |
98cdb6 |
conv_data->text = NULL;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (text);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
if (conv_data->operation == XIMStringConversionSubstitution
|
|
Packit |
98cdb6 |
&& subst_nchars > 0)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
gtk_im_context_delete_surrounding ((GtkIMContext *)context_xim,
|
|
Packit |
98cdb6 |
subst_offset, subst_nchars);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (surrounding);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static XVaNestedList
|
|
Packit |
98cdb6 |
set_preedit_callback (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->preedit_start_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->preedit_start_callback.callback = (XIMProc)preedit_start_callback;
|
|
Packit |
98cdb6 |
context_xim->preedit_done_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->preedit_done_callback.callback = (XIMProc)preedit_done_callback;
|
|
Packit |
98cdb6 |
context_xim->preedit_draw_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->preedit_draw_callback.callback = (XIMProc)preedit_draw_callback;
|
|
Packit |
98cdb6 |
context_xim->preedit_caret_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->preedit_caret_callback.callback = (XIMProc)preedit_caret_callback;
|
|
Packit |
98cdb6 |
return XVaCreateNestedList (0,
|
|
Packit |
98cdb6 |
XNPreeditStartCallback, &context_xim->preedit_start_callback,
|
|
Packit |
98cdb6 |
XNPreeditDoneCallback, &context_xim->preedit_done_callback,
|
|
Packit |
98cdb6 |
XNPreeditDrawCallback, &context_xim->preedit_draw_callback,
|
|
Packit |
98cdb6 |
XNPreeditCaretCallback, &context_xim->preedit_caret_callback,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static XVaNestedList
|
|
Packit |
98cdb6 |
set_status_callback (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
context_xim->status_start_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->status_start_callback.callback = (XIMProc)status_start_callback;
|
|
Packit |
98cdb6 |
context_xim->status_done_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->status_done_callback.callback = (XIMProc)status_done_callback;
|
|
Packit |
98cdb6 |
context_xim->status_draw_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->status_draw_callback.callback = (XIMProc)status_draw_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return XVaCreateNestedList (0,
|
|
Packit |
98cdb6 |
XNStatusStartCallback, &context_xim->status_start_callback,
|
|
Packit |
98cdb6 |
XNStatusDoneCallback, &context_xim->status_done_callback,
|
|
Packit |
98cdb6 |
XNStatusDrawCallback, &context_xim->status_draw_callback,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
set_string_conversion_callback (GtkIMContextXIM *context_xim, XIC xic)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (!context_xim->im_info->supports_string_conversion)
|
|
Packit |
98cdb6 |
return;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->string_conversion_callback.client_data = (XPointer)context_xim;
|
|
Packit |
98cdb6 |
context_xim->string_conversion_callback.callback = (XIMProc)string_conversion_callback;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
XSetICValues (xic,
|
|
Packit |
98cdb6 |
XNStringConversionCallback,
|
|
Packit |
98cdb6 |
(XPointer)&context_xim->string_conversion_callback,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
static XIC
|
|
Packit |
98cdb6 |
gtk_im_context_xim_get_ic (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->im_info == NULL || context_xim->im_info->im == NULL)
|
|
Packit |
98cdb6 |
return NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!context_xim->ic)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
const char *name1 = NULL;
|
|
Packit |
98cdb6 |
XVaNestedList list1 = NULL;
|
|
Packit |
98cdb6 |
const char *name2 = NULL;
|
|
Packit |
98cdb6 |
XVaNestedList list2 = NULL;
|
|
Packit |
98cdb6 |
XIMStyle im_style = 0;
|
|
Packit |
98cdb6 |
XIC xic = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (context_xim->use_preedit &&
|
|
Packit |
98cdb6 |
(context_xim->im_info->style & PREEDIT_MASK) == XIMPreeditCallbacks)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
im_style |= XIMPreeditCallbacks;
|
|
Packit |
98cdb6 |
name1 = XNPreeditAttributes;
|
|
Packit |
98cdb6 |
list1 = set_preedit_callback (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else if ((context_xim->im_info->style & PREEDIT_MASK) == XIMPreeditNone)
|
|
Packit |
98cdb6 |
im_style |= XIMPreeditNone;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
im_style |= XIMPreeditNothing;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if ((context_xim->im_info->style & STATUS_MASK) == XIMStatusCallbacks)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
im_style |= XIMStatusCallbacks;
|
|
Packit |
98cdb6 |
if (name1 == NULL)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
name1 = XNStatusAttributes;
|
|
Packit |
98cdb6 |
list1 = set_status_callback (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
name2 = XNStatusAttributes;
|
|
Packit |
98cdb6 |
list2 = set_status_callback (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else if ((context_xim->im_info->style & STATUS_MASK) == XIMStatusNone)
|
|
Packit |
98cdb6 |
im_style |= XIMStatusNone;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
im_style |= XIMStatusNothing;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
xic = XCreateIC (context_xim->im_info->im,
|
|
Packit |
98cdb6 |
XNInputStyle, im_style,
|
|
Packit |
98cdb6 |
XNClientWindow, GDK_DRAWABLE_XID (context_xim->client_window),
|
|
Packit |
98cdb6 |
name1, list1,
|
|
Packit |
98cdb6 |
name2, list2,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
if (list1)
|
|
Packit |
98cdb6 |
XFree (list1);
|
|
Packit |
98cdb6 |
if (list2)
|
|
Packit |
98cdb6 |
XFree (list2);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (xic)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
/* Don't filter key released events with XFilterEvents unless
|
|
Packit |
98cdb6 |
* input methods ask for. This is a workaround for Solaris input
|
|
Packit |
98cdb6 |
* method bug in C and European locales. It doubles each key
|
|
Packit |
98cdb6 |
* stroke if both key pressed and released events are filtered.
|
|
Packit |
98cdb6 |
* (bugzilla #81759)
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
gulong mask = 0xaaaaaaaa;
|
|
Packit |
98cdb6 |
XGetICValues (xic,
|
|
Packit |
98cdb6 |
XNFilterEvents, &mask,
|
|
Packit |
98cdb6 |
NULL);
|
|
Packit |
98cdb6 |
context_xim->filter_key_release = (mask & KeyReleaseMask) != 0;
|
|
Packit |
98cdb6 |
set_string_conversion_callback (context_xim, xic);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->ic = xic;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
update_status_window (context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (xic && context_xim->has_focus)
|
|
Packit |
98cdb6 |
XSetICFocus (xic);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
return context_xim->ic;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/*****************************************************************
|
|
Packit |
98cdb6 |
* Status Window handling
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* A status window is a small window attached to the toplevel
|
|
Packit |
98cdb6 |
* that is used to display information to the user about the
|
|
Packit |
98cdb6 |
* current input operation.
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* We claim the toplevel's status window for an input context if:
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* A) The input context has a toplevel
|
|
Packit |
98cdb6 |
* B) The input context has the focus
|
|
Packit |
98cdb6 |
* C) The input context has an XIC associated with it
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* Tracking A) and C) is pretty reliable since we
|
|
Packit |
98cdb6 |
* compute A) and create the XIC for C) ourselves.
|
|
Packit |
98cdb6 |
* For B) we basically have to depend on our callers
|
|
Packit |
98cdb6 |
* calling ::focus-in and ::focus-out at the right time.
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* The toplevel is computed by walking up the GdkWindow
|
|
Packit |
98cdb6 |
* hierarchy from context->client_window until we find a
|
|
Packit |
98cdb6 |
* window that is owned by some widget, and then calling
|
|
Packit |
98cdb6 |
* gtk_widget_get_toplevel() on that widget. This should
|
|
Packit |
98cdb6 |
* handle both cases where we might have GdkWindows without widgets,
|
|
Packit |
98cdb6 |
* and cases where GtkWidgets have strange window hierarchies
|
|
Packit |
98cdb6 |
* (like a torn off GtkHandleBox.)
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* The status window is visible if and only if there is text
|
|
Packit |
98cdb6 |
* for it; whenever a new GtkIMContextXIM claims the status
|
|
Packit |
98cdb6 |
* window, we blank out any existing text. We actually only
|
|
Packit |
98cdb6 |
* create a GtkWindow for the status window the first time
|
|
Packit |
98cdb6 |
* it is shown; this is an important optimization when we are
|
|
Packit |
98cdb6 |
* using XIM with something like a simple compose-key input
|
|
Packit |
98cdb6 |
* method that never needs a status window.
|
|
Packit |
98cdb6 |
*****************************************************************/
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when we no longer need a status window
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
disclaim_status_window (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_assert (context_xim->status_window->context == context_xim);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_window_set_text (context_xim->status_window, "");
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->status_window->context = NULL;
|
|
Packit |
98cdb6 |
context_xim->status_window = NULL;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when we need a status window
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
claim_status_window (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (!context_xim->status_window && context_xim->client_widget)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *toplevel = gtk_widget_get_toplevel (context_xim->client_widget);
|
|
Packit |
98cdb6 |
if (toplevel && gtk_widget_is_toplevel (toplevel))
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
StatusWindow *status_window = status_window_get (toplevel);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (status_window->context)
|
|
Packit |
98cdb6 |
disclaim_status_window (status_window->context);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_window->context = context_xim;
|
|
Packit |
98cdb6 |
context_xim->status_window = status_window;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Basic call made whenever something changed that might cause
|
|
Packit |
98cdb6 |
* us to need, or not to need a status window.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
update_status_window (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->ic && context_xim->in_toplevel && context_xim->has_focus)
|
|
Packit |
98cdb6 |
claim_status_window (context_xim);
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
disclaim_status_window (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Updates the in_toplevel flag for @context_xim
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
update_in_toplevel (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->client_widget)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *toplevel = gtk_widget_get_toplevel (context_xim->client_widget);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
context_xim->in_toplevel = (toplevel && gtk_widget_is_toplevel (toplevel));
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
context_xim->in_toplevel = FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Some paranoia, in case we don't get a focus out */
|
|
Packit |
98cdb6 |
if (!context_xim->in_toplevel)
|
|
Packit |
98cdb6 |
context_xim->has_focus = FALSE;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
update_status_window (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Callback when @widget's toplevel changes. It will always
|
|
Packit |
98cdb6 |
* change from NULL to a window, or a window to NULL;
|
|
Packit |
98cdb6 |
* we use that intermediate NULL state to make sure
|
|
Packit |
98cdb6 |
* that we disclaim the toplevel status window for the old
|
|
Packit |
98cdb6 |
* window.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
on_client_widget_hierarchy_changed (GtkWidget *widget,
|
|
Packit |
98cdb6 |
GtkWidget *old_toplevel,
|
|
Packit |
98cdb6 |
GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
update_in_toplevel (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Finds the GtkWidget that owns the window, or if none, the
|
|
Packit |
98cdb6 |
* widget owning the nearest parent that has a widget.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static GtkWidget *
|
|
Packit |
98cdb6 |
widget_for_window (GdkWindow *window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
while (window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
gpointer user_data;
|
|
Packit |
98cdb6 |
gdk_window_get_user_data (window, &user_data);
|
|
Packit |
98cdb6 |
if (user_data)
|
|
Packit |
98cdb6 |
return user_data;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
window = gdk_window_get_parent (window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return NULL;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when context_xim->client_window changes; takes care of
|
|
Packit |
98cdb6 |
* removing and/or setting up our watches for the toplevel
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
update_client_widget (GtkIMContextXIM *context_xim)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *new_client_widget = widget_for_window (context_xim->client_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (new_client_widget != context_xim->client_widget)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (context_xim->client_widget)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_signal_handlers_disconnect_by_func (context_xim->client_widget,
|
|
Packit |
98cdb6 |
G_CALLBACK (on_client_widget_hierarchy_changed),
|
|
Packit |
98cdb6 |
context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
context_xim->client_widget = new_client_widget;
|
|
Packit |
98cdb6 |
if (context_xim->client_widget)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
g_signal_connect (context_xim->client_widget, "hierarchy-changed",
|
|
Packit |
98cdb6 |
G_CALLBACK (on_client_widget_hierarchy_changed),
|
|
Packit |
98cdb6 |
context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
update_in_toplevel (context_xim);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when the toplevel is destroyed; frees the status window
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
on_status_toplevel_destroy (GtkWidget *toplevel,
|
|
Packit |
98cdb6 |
StatusWindow *status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
status_window_free (status_window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when the screen for the toplevel changes; updates the
|
|
Packit |
98cdb6 |
* screen for the status window to match.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
on_status_toplevel_notify_screen (GtkWindow *toplevel,
|
|
Packit |
98cdb6 |
GParamSpec *pspec,
|
|
Packit |
98cdb6 |
StatusWindow *status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (status_window->window)
|
|
Packit |
98cdb6 |
gtk_window_set_screen (GTK_WINDOW (status_window->window),
|
|
Packit |
98cdb6 |
gtk_widget_get_screen (GTK_WIDGET (toplevel)));
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Called when the toplevel window is moved; updates the position of
|
|
Packit |
98cdb6 |
* the status window to follow it.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static gboolean
|
|
Packit |
98cdb6 |
on_status_toplevel_configure (GtkWidget *toplevel,
|
|
Packit |
98cdb6 |
GdkEventConfigure *event,
|
|
Packit |
98cdb6 |
StatusWindow *status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GdkRectangle rect;
|
|
Packit |
98cdb6 |
GtkRequisition requisition;
|
|
Packit |
98cdb6 |
gint y;
|
|
Packit |
98cdb6 |
gint height;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (status_window->window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
height = gdk_screen_get_height (gtk_widget_get_screen (toplevel));
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gdk_window_get_frame_extents (toplevel->window, &rect);
|
|
Packit |
98cdb6 |
gtk_widget_size_request (status_window->window, &requisition);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (rect.y + rect.height + requisition.height < height)
|
|
Packit |
98cdb6 |
y = rect.y + rect.height;
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
y = height - requisition.height;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_window_move (GTK_WINDOW (status_window->window), rect.x, y);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return FALSE;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Frees a status window and removes its link from the status_windows list
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_window_free (StatusWindow *status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
status_windows = g_slist_remove (status_windows, status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (status_window->context)
|
|
Packit |
98cdb6 |
status_window->context->status_window = NULL;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_signal_handlers_disconnect_by_func (status_window->toplevel,
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_destroy),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
g_signal_handlers_disconnect_by_func (status_window->toplevel,
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_notify_screen),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
g_signal_handlers_disconnect_by_func (status_window->toplevel,
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_configure),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (status_window->window)
|
|
Packit |
98cdb6 |
gtk_widget_destroy (status_window->window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_object_set_data (G_OBJECT (status_window->toplevel), "gtk-im-xim-status-window", NULL);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_free (status_window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Finds the status window object for a toplevel, creating it if necessary.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static StatusWindow *
|
|
Packit |
98cdb6 |
status_window_get (GtkWidget *toplevel)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
StatusWindow *status_window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_window = g_object_get_data (G_OBJECT (toplevel), "gtk-im-xim-status-window");
|
|
Packit |
98cdb6 |
if (status_window)
|
|
Packit |
98cdb6 |
return status_window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_window = g_new0 (StatusWindow, 1);
|
|
Packit |
98cdb6 |
status_window->toplevel = toplevel;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_windows = g_slist_prepend (status_windows, status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_signal_connect (toplevel, "destroy",
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_destroy),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
g_signal_connect (toplevel, "configure-event",
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_configure),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
g_signal_connect (toplevel, "notify::screen",
|
|
Packit |
98cdb6 |
G_CALLBACK (on_status_toplevel_notify_screen),
|
|
Packit |
98cdb6 |
status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
g_object_set_data (G_OBJECT (toplevel), "gtk-im-xim-status-window", status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
return status_window;
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Creates the widgets for the status window; called when we
|
|
Packit |
98cdb6 |
* first need to show text for the status window.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_window_make_window (StatusWindow *status_window)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *window;
|
|
Packit |
98cdb6 |
GtkWidget *status_label;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_window->window = gtk_window_new (GTK_WINDOW_POPUP);
|
|
Packit |
98cdb6 |
window = status_window->window;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
status_label = gtk_label_new ("");
|
|
Packit |
98cdb6 |
gtk_misc_set_padding (GTK_MISC (status_label), 1, 1);
|
|
Packit |
98cdb6 |
gtk_widget_show (status_label);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_container_add (GTK_CONTAINER (window), status_label);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_window_set_screen (GTK_WINDOW (status_window->window),
|
|
Packit |
98cdb6 |
gtk_widget_get_screen (status_window->toplevel));
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
on_status_toplevel_configure (status_window->toplevel, NULL, status_window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/* Updates the text in the status window, hiding or
|
|
Packit |
98cdb6 |
* showing the window as necessary.
|
|
Packit |
98cdb6 |
*/
|
|
Packit |
98cdb6 |
static void
|
|
Packit |
98cdb6 |
status_window_set_text (StatusWindow *status_window,
|
|
Packit |
98cdb6 |
const gchar *text)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (text[0])
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkWidget *label;
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
if (!status_window->window)
|
|
Packit |
98cdb6 |
status_window_make_window (status_window);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
label = GTK_BIN (status_window->window)->child;
|
|
Packit |
98cdb6 |
gtk_label_set_text (GTK_LABEL (label), text);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
gtk_widget_show (status_window->window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
else
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
if (status_window->window)
|
|
Packit |
98cdb6 |
gtk_widget_hide (status_window->window);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
/**
|
|
Packit |
98cdb6 |
* gtk_im_context_xim_shutdown:
|
|
Packit |
98cdb6 |
*
|
|
Packit |
98cdb6 |
* Destroys all the status windows that are kept by the XIM contexts. This
|
|
Packit |
98cdb6 |
* function should only be called by the XIM module exit routine.
|
|
Packit |
98cdb6 |
**/
|
|
Packit |
98cdb6 |
void
|
|
Packit |
98cdb6 |
gtk_im_context_xim_shutdown (void)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
while (status_windows)
|
|
Packit |
98cdb6 |
status_window_free (status_windows->data);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
while (open_ims)
|
|
Packit |
98cdb6 |
{
|
|
Packit |
98cdb6 |
GtkXIMInfo *info = open_ims->data;
|
|
Packit |
98cdb6 |
GdkDisplay *display = gdk_screen_get_display (info->screen);
|
|
Packit |
98cdb6 |
|
|
Packit |
98cdb6 |
xim_info_display_closed (display, FALSE, info);
|
|
Packit |
98cdb6 |
open_ims = g_slist_remove_link (open_ims, open_ims);
|
|
Packit |
98cdb6 |
}
|
|
Packit |
98cdb6 |
}
|