3 * Copyright (C) 2003 Takuro Ashie
4 * Copyright (C) 2003-2004 Kazuki IWAMOTO
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library. If not, see <http://www.gnu.org/licenses/>.
22 * Please see the following site for the detail of Windows IME API.
23 * http://msdn.microsoft.com/library/default.asp?url=/library/en-us/appendix/hh/appendix/imeimes2_35ph.asp
26 #ifdef GTK_DISABLE_DEPRECATED
27 #undef GTK_DISABLE_DEPRECATED
30 #include "gtkimcontextime.h"
32 #include "imm-extra.h"
34 #include "gdk/win32/gdkwin32.h"
35 #include "gdk/gdkkeysyms.h"
37 #include <pango/pango.h>
42 # include <pango/pangowin32.h>
47 # include <pango/pangowin32.h>
50 /* #define BUFSIZE 4096 */
52 #define FREE_PREEDIT_BUFFER(ctx) \
54 g_free((ctx)->priv->comp_str); \
55 g_free((ctx)->priv->read_str); \
56 (ctx)->priv->comp_str = NULL; \
57 (ctx)->priv->read_str = NULL; \
58 (ctx)->priv->comp_str_len = 0; \
59 (ctx)->priv->read_str_len = 0; \
63 struct _GtkIMContextIMEPrivate
65 /* save IME context when the client window is focused out */
66 DWORD conversion_mode;
76 /* GObject class methods */
77 static void gtk_im_context_ime_class_init (GtkIMContextIMEClass *class);
78 static void gtk_im_context_ime_init (GtkIMContextIME *context_ime);
79 static void gtk_im_context_ime_dispose (GObject *obj);
80 static void gtk_im_context_ime_finalize (GObject *obj);
82 static void gtk_im_context_ime_set_property (GObject *object,
86 static void gtk_im_context_ime_get_property (GObject *object,
91 /* GtkIMContext's virtual functions */
92 static void gtk_im_context_ime_set_client_window (GtkIMContext *context,
93 GdkWindow *client_window);
94 static gboolean gtk_im_context_ime_filter_keypress (GtkIMContext *context,
96 static void gtk_im_context_ime_reset (GtkIMContext *context);
97 static void gtk_im_context_ime_get_preedit_string (GtkIMContext *context,
99 PangoAttrList **attrs,
101 static void gtk_im_context_ime_focus_in (GtkIMContext *context);
102 static void gtk_im_context_ime_focus_out (GtkIMContext *context);
103 static void gtk_im_context_ime_set_cursor_location (GtkIMContext *context,
105 static void gtk_im_context_ime_set_use_preedit (GtkIMContext *context,
106 gboolean use_preedit);
108 /* GtkIMContextIME's private functions */
109 static void gtk_im_context_ime_set_preedit_font (GtkIMContext *context);
111 static GdkFilterReturn
112 gtk_im_context_ime_message_filter (GdkXEvent *xevent,
115 static void get_window_position (GdkWindow *win,
118 static void cb_client_widget_hierarchy_changed (GtkWidget *widget,
120 GtkIMContextIME *context_ime);
122 GType gtk_type_im_context_ime = 0;
123 static GObjectClass *parent_class;
127 gtk_im_context_ime_register_type (GTypeModule *type_module)
129 const GTypeInfo im_context_ime_info = {
130 sizeof (GtkIMContextIMEClass),
131 (GBaseInitFunc) NULL,
132 (GBaseFinalizeFunc) NULL,
133 (GClassInitFunc) gtk_im_context_ime_class_init,
134 NULL, /* class_finalize */
135 NULL, /* class_data */
136 sizeof (GtkIMContextIME),
138 (GInstanceInitFunc) gtk_im_context_ime_init,
141 gtk_type_im_context_ime =
142 g_type_module_register_type (type_module,
144 "GtkIMContextIME", &im_context_ime_info, 0);
148 gtk_im_context_ime_class_init (GtkIMContextIMEClass *class)
150 GtkIMContextClass *im_context_class = GTK_IM_CONTEXT_CLASS (class);
151 GObjectClass *gobject_class = G_OBJECT_CLASS (class);
153 parent_class = g_type_class_peek_parent (class);
155 gobject_class->finalize = gtk_im_context_ime_finalize;
156 gobject_class->dispose = gtk_im_context_ime_dispose;
157 gobject_class->set_property = gtk_im_context_ime_set_property;
158 gobject_class->get_property = gtk_im_context_ime_get_property;
160 im_context_class->set_client_window = gtk_im_context_ime_set_client_window;
161 im_context_class->filter_keypress = gtk_im_context_ime_filter_keypress;
162 im_context_class->reset = gtk_im_context_ime_reset;
163 im_context_class->get_preedit_string = gtk_im_context_ime_get_preedit_string;
164 im_context_class->focus_in = gtk_im_context_ime_focus_in;
165 im_context_class->focus_out = gtk_im_context_ime_focus_out;
166 im_context_class->set_cursor_location = gtk_im_context_ime_set_cursor_location;
167 im_context_class->set_use_preedit = gtk_im_context_ime_set_use_preedit;
172 gtk_im_context_ime_init (GtkIMContextIME *context_ime)
174 context_ime->client_window = NULL;
175 context_ime->toplevel = NULL;
176 context_ime->use_preedit = TRUE;
177 context_ime->preediting = FALSE;
178 context_ime->opened = FALSE;
179 context_ime->focus = FALSE;
180 context_ime->cursor_location.x = 0;
181 context_ime->cursor_location.y = 0;
182 context_ime->cursor_location.width = 0;
183 context_ime->cursor_location.height = 0;
185 context_ime->priv = g_malloc0 (sizeof (GtkIMContextIMEPrivate));
186 context_ime->priv->conversion_mode = 0;
187 context_ime->priv->sentence_mode = 0;
188 context_ime->priv->comp_str = NULL;
189 context_ime->priv->comp_str_len = 0;
190 context_ime->priv->read_str = NULL;
191 context_ime->priv->read_str_len = 0;
196 gtk_im_context_ime_dispose (GObject *obj)
198 GtkIMContext *context = GTK_IM_CONTEXT (obj);
199 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (obj);
201 if (context_ime->client_window)
202 gtk_im_context_ime_set_client_window (context, NULL);
204 FREE_PREEDIT_BUFFER (context_ime);
206 if (G_OBJECT_CLASS (parent_class)->dispose)
207 G_OBJECT_CLASS (parent_class)->dispose (obj);
212 gtk_im_context_ime_finalize (GObject *obj)
214 /* GtkIMContext *context = GTK_IM_CONTEXT (obj); */
215 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (obj);
217 g_free (context_ime->priv);
218 context_ime->priv = NULL;
220 if (G_OBJECT_CLASS (parent_class)->finalize)
221 G_OBJECT_CLASS (parent_class)->finalize (obj);
226 gtk_im_context_ime_set_property (GObject *object,
231 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (object);
233 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context_ime));
244 gtk_im_context_ime_get_property (GObject *object,
249 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (object);
251 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context_ime));
262 gtk_im_context_ime_new (void)
264 return g_object_new (GTK_TYPE_IM_CONTEXT_IME, NULL);
269 gtk_im_context_ime_set_client_window (GtkIMContext *context,
270 GdkWindow *client_window)
272 GtkIMContextIME *context_ime;
274 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context));
275 context_ime = GTK_IM_CONTEXT_IME (context);
282 hwnd = gdk_win32_window_get_impl_hwnd (client_window);
283 himc = ImmGetContext (hwnd);
286 context_ime->opened = ImmGetOpenStatus (himc);
287 ImmGetConversionStatus (himc,
288 &context_ime->priv->conversion_mode,
289 &context_ime->priv->sentence_mode);
290 ImmReleaseContext (hwnd, himc);
293 else if (context_ime->focus)
295 gtk_im_context_ime_focus_out (context);
298 context_ime->client_window = client_window;
303 gtk_im_context_ime_filter_keypress (GtkIMContext *context,
306 GtkIMContextIME *context_ime;
307 gboolean retval = FALSE;
310 g_return_val_if_fail (GTK_IS_IM_CONTEXT_IME (context), FALSE);
311 g_return_val_if_fail (event, FALSE);
313 if (event->type == GDK_KEY_RELEASE)
316 if (event->state & GDK_CONTROL_MASK)
319 context_ime = GTK_IM_CONTEXT_IME (context);
321 if (!context_ime->focus)
324 if (!GDK_IS_WINDOW (context_ime->client_window))
327 c = gdk_keyval_to_unicode (event->keyval);
331 int len = g_unichar_to_utf8 (c, utf8);
334 g_signal_emit_by_name (context_ime, "commit", utf8);
343 gtk_im_context_ime_reset (GtkIMContext *context)
345 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (context);
349 if (!context_ime->client_window)
352 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
353 himc = ImmGetContext (hwnd);
357 if (context_ime->preediting)
359 if (ImmGetOpenStatus (himc))
360 ImmNotifyIME (himc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
362 context_ime->preediting = FALSE;
363 g_signal_emit_by_name (context, "preedit-changed");
366 ImmReleaseContext (hwnd, himc);
371 get_utf8_preedit_string (GtkIMContextIME *context_ime, gint *pos_ret)
373 gchar *utf8str = NULL;
381 if (!context_ime->client_window)
382 return g_strdup ("");
383 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
384 himc = ImmGetContext (hwnd);
386 return g_strdup ("");
388 if (context_ime->preediting)
392 len = ImmGetCompositionStringW (himc, GCS_COMPSTR, NULL, 0);
395 GError *error = NULL;
396 gpointer buf = g_alloca (len);
398 ImmGetCompositionStringW (himc, GCS_COMPSTR, buf, len);
400 utf8str = g_utf16_to_utf8 (buf, len, NULL, NULL, &error);
403 g_warning ("%s", error->message);
404 g_error_free (error);
409 pos = ImmGetCompositionStringW (himc, GCS_CURSORPOS, NULL, 0);
410 if (pos < 0 || len < pos)
412 g_warning ("ImmGetCompositionString: "
413 "Invalid cursor position!");
422 utf8str = g_strdup ("");
429 ImmReleaseContext (hwnd, himc);
435 static PangoAttrList *
436 get_pango_attr_list (GtkIMContextIME *context_ime, const gchar *utf8str)
438 PangoAttrList *attrs = pango_attr_list_new ();
442 if (!context_ime->client_window)
444 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
445 himc = ImmGetContext (hwnd);
449 if (context_ime->preediting)
451 const gchar *schr = utf8str, *echr;
453 guint16 f_red, f_green, f_blue, b_red, b_green, b_blue;
454 glong len, spos = 0, epos, sidx = 0, eidx;
455 PangoAttribute *attr;
458 * get attributes list of IME.
460 len = ImmGetCompositionStringW (himc, GCS_COMPATTR, NULL, 0);
461 buf = g_alloca (len);
462 ImmGetCompositionStringW (himc, GCS_COMPATTR, buf, len);
465 * schr and echr are pointer in utf8str.
467 for (echr = g_utf8_next_char (utf8str); *schr != '\0';
468 echr = g_utf8_next_char (echr))
471 * spos and epos are buf(attributes list of IME) position by
473 * Using the wide-char API, this value is same with UTF-8 offset.
475 epos = g_utf8_pointer_to_offset (utf8str, echr);
478 * sidx and eidx are positions in utf8str by bytes.
480 eidx = echr - utf8str;
483 * convert attributes list to PangoAttriute.
485 if (*echr == '\0' || buf[spos] != buf[epos])
489 case ATTR_TARGET_CONVERTED:
490 attr = pango_attr_underline_new (PANGO_UNDERLINE_DOUBLE);
491 attr->start_index = sidx;
492 attr->end_index = eidx;
493 pango_attr_list_change (attrs, attr);
494 f_red = f_green = f_blue = 0;
495 b_red = b_green = b_blue = 0xffff;
497 case ATTR_TARGET_NOTCONVERTED:
498 f_red = f_green = f_blue = 0xffff;
499 b_red = b_green = b_blue = 0;
501 case ATTR_INPUT_ERROR:
502 f_red = f_green = f_blue = 0;
503 b_red = b_green = b_blue = 0x7fff;
505 default: /* ATTR_INPUT,ATTR_CONVERTED,ATTR_FIXEDCONVERTED */
506 attr = pango_attr_underline_new (PANGO_UNDERLINE_SINGLE);
507 attr->start_index = sidx;
508 attr->end_index = eidx;
509 pango_attr_list_change (attrs, attr);
510 f_red = f_green = f_blue = 0;
511 b_red = b_green = b_blue = 0xffff;
513 attr = pango_attr_foreground_new (f_red, f_green, f_blue);
514 attr->start_index = sidx;
515 attr->end_index = eidx;
516 pango_attr_list_change (attrs, attr);
517 attr = pango_attr_background_new (b_red, b_green, b_blue);
518 attr->start_index = sidx;
519 attr->end_index = eidx;
520 pango_attr_list_change (attrs, attr);
529 ImmReleaseContext (hwnd, himc);
536 gtk_im_context_ime_get_preedit_string (GtkIMContext *context,
538 PangoAttrList **attrs,
541 gchar *utf8str = NULL;
543 GtkIMContextIME *context_ime;
545 context_ime = GTK_IM_CONTEXT_IME (context);
547 utf8str = get_utf8_preedit_string (context_ime, &pos);
550 *attrs = get_pango_attr_list (context_ime, utf8str);
568 gtk_im_context_ime_focus_in (GtkIMContext *context)
570 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (context);
572 GtkWidget *widget = NULL;
576 if (!GDK_IS_WINDOW (context_ime->client_window))
579 /* swtich current context */
580 context_ime->focus = TRUE;
582 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
583 himc = ImmGetContext (hwnd);
587 toplevel = gdk_window_get_toplevel (context_ime->client_window);
588 if (GDK_IS_WINDOW (toplevel))
590 gdk_window_add_filter (toplevel,
591 gtk_im_context_ime_message_filter, context_ime);
592 top_hwnd = gdk_win32_window_get_impl_hwnd (toplevel);
594 context_ime->toplevel = toplevel;
598 g_warning ("gtk_im_context_ime_focus_in(): "
599 "cannot find toplevel window.");
603 /* trace reparenting (probably no need) */
604 gdk_window_get_user_data (context_ime->client_window, (gpointer) & widget);
605 if (GTK_IS_WIDGET (widget))
607 g_signal_connect (widget, "hierarchy-changed",
608 G_CALLBACK (cb_client_widget_hierarchy_changed),
616 /* restore preedit context */
617 ImmSetConversionStatus (himc,
618 context_ime->priv->conversion_mode,
619 context_ime->priv->sentence_mode);
621 if (context_ime->opened)
623 if (!ImmGetOpenStatus (himc))
624 ImmSetOpenStatus (himc, TRUE);
625 if (context_ime->preediting)
627 ImmSetCompositionStringW (himc,
629 context_ime->priv->comp_str,
630 context_ime->priv->comp_str_len, NULL, 0);
631 FREE_PREEDIT_BUFFER (context_ime);
636 ImmReleaseContext (hwnd, himc);
641 gtk_im_context_ime_focus_out (GtkIMContext *context)
643 GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (context);
645 GtkWidget *widget = NULL;
649 if (!GDK_IS_WINDOW (context_ime->client_window))
652 /* swtich current context */
653 context_ime->focus = FALSE;
655 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
656 himc = ImmGetContext (hwnd);
660 /* save preedit context */
661 ImmGetConversionStatus (himc,
662 &context_ime->priv->conversion_mode,
663 &context_ime->priv->sentence_mode);
665 if (ImmGetOpenStatus (himc))
667 gboolean preediting = context_ime->preediting;
671 FREE_PREEDIT_BUFFER (context_ime);
673 context_ime->priv->comp_str_len
674 = ImmGetCompositionStringW (himc, GCS_COMPSTR, NULL, 0);
675 context_ime->priv->comp_str
676 = g_malloc (context_ime->priv->comp_str_len);
677 ImmGetCompositionStringW (himc, GCS_COMPSTR,
678 context_ime->priv->comp_str,
679 context_ime->priv->comp_str_len);
681 context_ime->priv->read_str_len
682 = ImmGetCompositionStringW (himc, GCS_COMPREADSTR, NULL, 0);
683 context_ime->priv->read_str
684 = g_malloc (context_ime->priv->read_str_len);
685 ImmGetCompositionStringW (himc, GCS_COMPREADSTR,
686 context_ime->priv->read_str,
687 context_ime->priv->read_str_len);
690 ImmSetOpenStatus (himc, FALSE);
692 context_ime->opened = TRUE;
693 context_ime->preediting = preediting;
697 context_ime->opened = FALSE;
698 context_ime->preediting = FALSE;
701 /* remove signal handler */
702 gdk_window_get_user_data (context_ime->client_window, (gpointer) & widget);
703 if (GTK_IS_WIDGET (widget))
705 g_signal_handlers_disconnect_by_func
707 G_CALLBACK (cb_client_widget_hierarchy_changed), context_ime);
710 /* remove event fileter */
711 toplevel = gdk_window_get_toplevel (context_ime->client_window);
712 if (GDK_IS_WINDOW (toplevel))
714 gdk_window_remove_filter (toplevel,
715 gtk_im_context_ime_message_filter,
717 top_hwnd = gdk_win32_window_get_impl_hwnd (toplevel);
719 context_ime->toplevel = NULL;
723 g_warning ("gtk_im_context_ime_focus_out(): "
724 "cannot find toplevel window.");
728 ImmReleaseContext (hwnd, himc);
733 gtk_im_context_ime_set_cursor_location (GtkIMContext *context,
737 GtkIMContextIME *context_ime;
742 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context));
744 context_ime = GTK_IM_CONTEXT_IME (context);
746 context_ime->cursor_location = *area;
748 if (!context_ime->client_window)
751 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
752 himc = ImmGetContext (hwnd);
756 get_window_position (context_ime->client_window, &wx, &wy);
757 cf.dwStyle = CFS_POINT;
758 cf.ptCurrentPos.x = wx + context_ime->cursor_location.x;
759 cf.ptCurrentPos.y = wy + context_ime->cursor_location.y;
760 ImmSetCompositionWindow (himc, &cf);
762 ImmReleaseContext (hwnd, himc);
767 gtk_im_context_ime_set_use_preedit (GtkIMContext *context,
768 gboolean use_preedit)
770 GtkIMContextIME *context_ime;
772 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context));
773 context_ime = GTK_IM_CONTEXT_IME (context);
775 context_ime->use_preedit = use_preedit;
776 if (context_ime->preediting)
781 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
782 himc = ImmGetContext (hwnd);
786 /* FIXME: What to do? */
788 ImmReleaseContext (hwnd, himc);
794 gtk_im_context_ime_set_preedit_font (GtkIMContext *context)
796 GtkIMContextIME *context_ime;
797 GtkWidget *widget = NULL;
800 HKL ime = GetKeyboardLayout (0);
803 PangoContext *pango_context;
807 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context));
809 context_ime = GTK_IM_CONTEXT_IME (context);
810 if (!context_ime->client_window)
813 gdk_window_get_user_data (context_ime->client_window, (gpointer) &widget);
814 if (!GTK_IS_WIDGET (widget))
817 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
818 himc = ImmGetContext (hwnd);
823 pango_context = gtk_widget_get_pango_context (widget);
827 /* Try to make sure we use a font that actually can show the
828 * language in question.
831 switch (PRIMARYLANGID (LOWORD (ime)))
838 switch (SUBLANGID (LOWORD (ime)))
840 case SUBLANG_CHINESE_TRADITIONAL:
841 lang = "zh_TW"; break;
842 case SUBLANG_CHINESE_SIMPLIFIED:
843 lang = "zh_CN"; break;
844 case SUBLANG_CHINESE_HONGKONG:
845 lang = "zh_HK"; break;
846 case SUBLANG_CHINESE_SINGAPORE:
847 lang = "zh_SG"; break;
848 case SUBLANG_CHINESE_MACAU:
849 lang = "zh_MO"; break;
860 /* We know what language it is. Look for a character, any
861 * character, that language needs.
863 PangoLanguage *pango_lang = pango_language_from_string (lang);
864 PangoFontset *fontset =
865 pango_context_load_fontset (pango_context,
866 gtk_widget_get_style (widget)->font_desc,
869 g_utf8_to_ucs4 (pango_language_get_sample_string (pango_lang),
870 -1, NULL, NULL, NULL);
871 wc = 0x4E00; /* In all CJK languages? */
876 for (i = 0; sample[i]; i++)
877 if (g_unichar_iswide (sample[i]))
884 font = pango_fontset_get_font (fontset, wc);
885 g_object_unref (fontset);
888 font = pango_context_load_font (pango_context, gtk_widget_get_style (widget)->font_desc);
893 logfont = pango_win32_font_logfont (font);
895 ImmSetCompositionFont (himc, logfont);
897 g_object_unref (font);
901 ImmReleaseContext (hwnd, himc);
905 static GdkFilterReturn
906 gtk_im_context_ime_message_filter (GdkXEvent *xevent,
910 GtkIMContext *context;
911 GtkIMContextIME *context_ime;
914 MSG *msg = (MSG *) xevent;
915 GdkFilterReturn retval = GDK_FILTER_CONTINUE;
917 g_return_val_if_fail (GTK_IS_IM_CONTEXT_IME (data), retval);
919 context = GTK_IM_CONTEXT (data);
920 context_ime = GTK_IM_CONTEXT_IME (data);
921 if (!context_ime->focus)
924 hwnd = gdk_win32_window_get_impl_hwnd (context_ime->client_window);
925 himc = ImmGetContext (hwnd);
929 switch (msg->message)
931 case WM_IME_COMPOSITION:
936 get_window_position (context_ime->client_window, &wx, &wy);
944 gdk_win32_window_get_impl_hwnd (gdk_window_get_toplevel
945 (context_ime->client_window));
946 GetWindowRect (hwnd_top, &rc);
949 ClientToScreen (hwnd_top, &pt);
954 cf.dwStyle = CFS_CANDIDATEPOS;
955 cf.ptCurrentPos.x = wx + context_ime->cursor_location.x;
956 cf.ptCurrentPos.y = wy + context_ime->cursor_location.y
957 + context_ime->cursor_location.height;
958 ImmSetCandidateWindow (himc, &cf);
960 if ((msg->lParam & GCS_COMPSTR))
961 g_signal_emit_by_name (context, "preedit-changed");
963 if (msg->lParam & GCS_RESULTSTR)
966 gchar *utf8str = NULL;
967 GError *error = NULL;
969 len = ImmGetCompositionStringW (himc, GCS_RESULTSTR, NULL, 0);
973 gpointer buf = g_alloca (len);
974 ImmGetCompositionStringW (himc, GCS_RESULTSTR, buf, len);
976 utf8str = g_utf16_to_utf8 (buf, len, NULL, NULL, &error);
979 g_warning ("%s", error->message);
980 g_error_free (error);
986 g_signal_emit_by_name (context, "commit", utf8str);
992 if (context_ime->use_preedit)
997 case WM_IME_STARTCOMPOSITION:
998 context_ime->preediting = TRUE;
999 gtk_im_context_ime_set_cursor_location (context, NULL);
1000 g_signal_emit_by_name (context, "preedit-start");
1001 if (context_ime->use_preedit)
1005 case WM_IME_ENDCOMPOSITION:
1006 context_ime->preediting = FALSE;
1007 g_signal_emit_by_name (context, "preedit-changed");
1008 g_signal_emit_by_name (context, "preedit-end");
1009 if (context_ime->use_preedit)
1014 switch (msg->wParam)
1016 case IMN_SETOPENSTATUS:
1017 context_ime->opened = ImmGetOpenStatus (himc);
1018 gtk_im_context_ime_set_preedit_font (context);
1029 ImmReleaseContext (hwnd, himc);
1035 * x and y must be initialized to 0.
1038 get_window_position (GdkWindow *win, gint *x, gint *y)
1040 GdkWindow *parent, *toplevel;
1043 g_return_if_fail (GDK_IS_WINDOW (win));
1044 g_return_if_fail (x && y);
1046 gdk_window_get_position (win, &wx, &wy);
1049 parent = gdk_window_get_parent (win);
1050 toplevel = gdk_window_get_toplevel (win);
1052 if (parent && parent != toplevel)
1053 get_window_position (parent, x, y);
1058 * probably, this handler isn't needed.
1061 cb_client_widget_hierarchy_changed (GtkWidget *widget,
1063 GtkIMContextIME *context_ime)
1065 GdkWindow *new_toplevel;
1067 g_return_if_fail (GTK_IS_WIDGET (widget));
1068 g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context_ime));
1070 if (!context_ime->client_window)
1072 if (!context_ime->focus)
1075 new_toplevel = gdk_window_get_toplevel (context_ime->client_window);
1076 if (context_ime->toplevel == new_toplevel)
1079 /* remove filter from old toplevel */
1080 if (GDK_IS_WINDOW (context_ime->toplevel))
1082 gdk_window_remove_filter (context_ime->toplevel,
1083 gtk_im_context_ime_message_filter,
1090 /* add filter to new toplevel */
1091 if (GDK_IS_WINDOW (new_toplevel))
1093 gdk_window_add_filter (new_toplevel,
1094 gtk_im_context_ime_message_filter, context_ime);
1100 context_ime->toplevel = new_toplevel;