|
|
|
@@ -50,34 +50,39 @@
|
|
|
|
|
# include <pango/pangowin32.h>
|
|
|
|
|
#endif /* STRICT */
|
|
|
|
|
|
|
|
|
|
/* #define BUFSIZE 4096 */
|
|
|
|
|
/* Determines what happens when focus is lost while preedit is in process. */
|
|
|
|
|
typedef enum {
|
|
|
|
|
/* Preedit is committed. */
|
|
|
|
|
GTK_WIN32_IME_FOCUS_BEHAVIOR_COMMIT,
|
|
|
|
|
/* Preedit is discarded. */
|
|
|
|
|
GTK_WIN32_IME_FOCUS_BEHAVIOR_DISCARD,
|
|
|
|
|
/* Preedit follows the cursor (that means it will appear in the widget
|
|
|
|
|
* that receives the focus) */
|
|
|
|
|
GTK_WIN32_IME_FOCUS_BEHAVIOR_FOLLOW,
|
|
|
|
|
} GtkWin32IMEFocusBehavior;
|
|
|
|
|
|
|
|
|
|
#define IS_DEAD_KEY(k) \
|
|
|
|
|
((k) >= GDK_KEY_dead_grave && (k) <= (GDK_KEY_dead_dasia+1))
|
|
|
|
|
|
|
|
|
|
#define FREE_PREEDIT_BUFFER(ctx) \
|
|
|
|
|
{ \
|
|
|
|
|
g_free((ctx)->priv->comp_str); \
|
|
|
|
|
g_free((ctx)->priv->read_str); \
|
|
|
|
|
(ctx)->priv->comp_str = NULL; \
|
|
|
|
|
(ctx)->priv->read_str = NULL; \
|
|
|
|
|
(ctx)->priv->comp_str_len = 0; \
|
|
|
|
|
(ctx)->priv->read_str_len = 0; \
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct _GtkIMContextIMEPrivate
|
|
|
|
|
{
|
|
|
|
|
/* save IME context when the client window is focused out */
|
|
|
|
|
DWORD conversion_mode;
|
|
|
|
|
DWORD sentence_mode;
|
|
|
|
|
|
|
|
|
|
LPVOID comp_str;
|
|
|
|
|
DWORD comp_str_len;
|
|
|
|
|
LPVOID read_str;
|
|
|
|
|
DWORD read_str_len;
|
|
|
|
|
|
|
|
|
|
/* When pretend_empty_preedit is set to TRUE,
|
|
|
|
|
* gtk_im_context_ime_get_preedit_string() will return an empty string
|
|
|
|
|
* instead of the actual content of ImmGetCompositionStringW().
|
|
|
|
|
*
|
|
|
|
|
* This is necessary because GtkEntry expects the preedit buffer to be
|
|
|
|
|
* cleared before commit() is called, otherwise it leads to an assertion
|
|
|
|
|
* failure in Pango. However, since we emit the commit() signal while
|
|
|
|
|
* handling the WM_IME_COMPOSITION message, the IME buffer will be non-empty,
|
|
|
|
|
* so we temporarily set this flag while emmiting the appropriate signals.
|
|
|
|
|
*
|
|
|
|
|
* See also:
|
|
|
|
|
* https://bugzilla.gnome.org/show_bug.cgi?id=787142
|
|
|
|
|
* https://gitlab.gnome.org/GNOME/gtk/commit/c255ba68fc2c918dd84da48a472e7973d3c00b03
|
|
|
|
|
*/
|
|
|
|
|
gboolean pretend_empty_preedit;
|
|
|
|
|
guint32 dead_key_keyval;
|
|
|
|
|
GtkWin32IMEFocusBehavior focus_behavior;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -166,12 +171,7 @@ gtk_im_context_ime_init (GtkIMContextIME *context_ime)
|
|
|
|
|
context_ime->commit_string = NULL;
|
|
|
|
|
|
|
|
|
|
context_ime->priv = g_malloc0 (sizeof (GtkIMContextIMEPrivate));
|
|
|
|
|
context_ime->priv->conversion_mode = 0;
|
|
|
|
|
context_ime->priv->sentence_mode = 0;
|
|
|
|
|
context_ime->priv->comp_str = NULL;
|
|
|
|
|
context_ime->priv->comp_str_len = 0;
|
|
|
|
|
context_ime->priv->read_str = NULL;
|
|
|
|
|
context_ime->priv->read_str_len = 0;
|
|
|
|
|
context_ime->priv->focus_behavior = GTK_WIN32_IME_FOCUS_BEHAVIOR_COMMIT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -184,8 +184,6 @@ gtk_im_context_ime_dispose (GObject *obj)
|
|
|
|
|
if (context_ime->client_surface)
|
|
|
|
|
gtk_im_context_ime_set_client_widget (context, NULL);
|
|
|
|
|
|
|
|
|
|
FREE_PREEDIT_BUFFER (context_ime);
|
|
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (gtk_im_context_ime_parent_class)->dispose (obj);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -193,7 +191,6 @@ gtk_im_context_ime_dispose (GObject *obj)
|
|
|
|
|
static void
|
|
|
|
|
gtk_im_context_ime_finalize (GObject *obj)
|
|
|
|
|
{
|
|
|
|
|
/* GtkIMContext *context = GTK_IM_CONTEXT (obj); */
|
|
|
|
|
GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (obj);
|
|
|
|
|
|
|
|
|
|
g_free (context_ime->priv);
|
|
|
|
@@ -251,30 +248,27 @@ gtk_im_context_ime_set_client_widget (GtkIMContext *context,
|
|
|
|
|
GtkWidget *widget)
|
|
|
|
|
{
|
|
|
|
|
GtkIMContextIME *context_ime;
|
|
|
|
|
GdkSurface *client_surface;
|
|
|
|
|
GdkSurface *client_surface = NULL;
|
|
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_IM_CONTEXT_IME (context));
|
|
|
|
|
context_ime = GTK_IM_CONTEXT_IME (context);
|
|
|
|
|
|
|
|
|
|
client_surface = NULL;
|
|
|
|
|
if (widget)
|
|
|
|
|
client_surface = gtk_native_get_surface (gtk_widget_get_native (widget));
|
|
|
|
|
|
|
|
|
|
if (client_surface)
|
|
|
|
|
if (client_surface != NULL)
|
|
|
|
|
{
|
|
|
|
|
HIMC himc;
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
|
|
|
|
|
hwnd = gdk_win32_surface_get_impl_hwnd (client_surface);
|
|
|
|
|
himc = ImmGetContext (hwnd);
|
|
|
|
|
HWND hwnd = gdk_win32_surface_get_impl_hwnd (client_surface);
|
|
|
|
|
HIMC himc = ImmGetContext (hwnd);
|
|
|
|
|
if (himc)
|
|
|
|
|
{
|
|
|
|
|
context_ime->opened = ImmGetOpenStatus (himc);
|
|
|
|
|
ImmGetConversionStatus (himc,
|
|
|
|
|
&context_ime->priv->conversion_mode,
|
|
|
|
|
&context_ime->priv->sentence_mode);
|
|
|
|
|
ImmReleaseContext (hwnd, himc);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
context_ime->opened = ImmGetOpenStatus (himc);
|
|
|
|
|
ImmReleaseContext (hwnd, himc);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
context_ime->opened = FALSE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (context_ime->focus)
|
|
|
|
|
{
|
|
|
|
@@ -427,11 +421,10 @@ gtk_im_context_ime_reset (GtkIMContext *context)
|
|
|
|
|
if (!himc)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
ImmNotifyIME (himc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
|
|
|
|
|
|
|
|
|
|
if (context_ime->preediting)
|
|
|
|
|
{
|
|
|
|
|
if (ImmGetOpenStatus (himc))
|
|
|
|
|
ImmNotifyIME (himc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
|
|
|
|
|
|
|
|
|
|
context_ime->preediting = FALSE;
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
}
|
|
|
|
@@ -441,12 +434,17 @@ gtk_im_context_ime_reset (GtkIMContext *context)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
get_utf8_preedit_string (GtkIMContextIME *context_ime, int *pos_ret)
|
|
|
|
|
get_utf8_preedit_string (GtkIMContextIME *context_ime,
|
|
|
|
|
int kind,
|
|
|
|
|
int *pos_ret)
|
|
|
|
|
{
|
|
|
|
|
gunichar2 *utf16str = NULL;
|
|
|
|
|
glong size;
|
|
|
|
|
char *utf8str = NULL;
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
HIMC himc;
|
|
|
|
|
int pos = 0;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
|
|
if (pos_ret)
|
|
|
|
|
*pos_ret = 0;
|
|
|
|
@@ -458,58 +456,31 @@ get_utf8_preedit_string (GtkIMContextIME *context_ime, int *pos_ret)
|
|
|
|
|
if (!himc)
|
|
|
|
|
return g_strdup ("");
|
|
|
|
|
|
|
|
|
|
if (context_ime->preediting)
|
|
|
|
|
size = ImmGetCompositionStringW (himc, kind, NULL, 0);
|
|
|
|
|
|
|
|
|
|
if (size > 0)
|
|
|
|
|
{
|
|
|
|
|
glong len;
|
|
|
|
|
utf16str = g_malloc (size);
|
|
|
|
|
|
|
|
|
|
len = ImmGetCompositionStringW (himc, GCS_COMPSTR, NULL, 0);
|
|
|
|
|
if (len > 0)
|
|
|
|
|
ImmGetCompositionStringW (himc, kind, utf16str, size);
|
|
|
|
|
utf8str = g_utf16_to_utf8 (utf16str, size / sizeof (gunichar2),
|
|
|
|
|
NULL, NULL, &error);
|
|
|
|
|
if (error)
|
|
|
|
|
{
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gpointer buf = g_alloca (len);
|
|
|
|
|
g_warning ("%s", error->message);
|
|
|
|
|
g_error_free (error);
|
|
|
|
|
|
|
|
|
|
ImmGetCompositionStringW (himc, GCS_COMPSTR, buf, len);
|
|
|
|
|
len /= 2;
|
|
|
|
|
utf8str = g_utf16_to_utf8 (buf, len, NULL, NULL, &error);
|
|
|
|
|
if (error)
|
|
|
|
|
{
|
|
|
|
|
g_warning ("%s", error->message);
|
|
|
|
|
g_error_free (error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pos_ret)
|
|
|
|
|
{
|
|
|
|
|
pos = ImmGetCompositionStringW (himc, GCS_CURSORPOS, NULL, 0);
|
|
|
|
|
if (pos < 0 || len < pos)
|
|
|
|
|
{
|
|
|
|
|
g_warning ("ImmGetCompositionString: "
|
|
|
|
|
"Invalid cursor position!");
|
|
|
|
|
pos = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (context_ime->commit_string)
|
|
|
|
|
if (pos_ret)
|
|
|
|
|
{
|
|
|
|
|
if (utf8str)
|
|
|
|
|
pos = ImmGetCompositionStringW (himc, GCS_CURSORPOS, NULL, 0);
|
|
|
|
|
if (pos < 0 || size < pos)
|
|
|
|
|
{
|
|
|
|
|
char *utf8str_new = g_strdup (utf8str);
|
|
|
|
|
|
|
|
|
|
/* Note: We *don't* want to update context_ime->commit_string here!
|
|
|
|
|
* Otherwise it will be updated repeatedly, not what we want!
|
|
|
|
|
*/
|
|
|
|
|
g_free (utf8str);
|
|
|
|
|
utf8str = g_strconcat (context_ime->commit_string,
|
|
|
|
|
utf8str_new,
|
|
|
|
|
NULL);
|
|
|
|
|
g_free (utf8str_new);
|
|
|
|
|
pos += g_utf8_strlen (context_ime->commit_string, -1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
utf8str = g_strdup (context_ime->commit_string);
|
|
|
|
|
pos = g_utf8_strlen (context_ime->commit_string, -1);
|
|
|
|
|
g_warning ("ImmGetCompositionString: "
|
|
|
|
|
"Invalid cursor position!");
|
|
|
|
|
pos = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@@ -523,6 +494,7 @@ get_utf8_preedit_string (GtkIMContextIME *context_ime, int *pos_ret)
|
|
|
|
|
*pos_ret = pos;
|
|
|
|
|
|
|
|
|
|
ImmReleaseContext (hwnd, himc);
|
|
|
|
|
g_free (utf16str);
|
|
|
|
|
|
|
|
|
|
return utf8str;
|
|
|
|
|
}
|
|
|
|
@@ -534,6 +506,7 @@ get_pango_attr_list (GtkIMContextIME *context_ime, const char *utf8str)
|
|
|
|
|
PangoAttrList *attrs = pango_attr_list_new ();
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
HIMC himc;
|
|
|
|
|
guint8 *buf = NULL;
|
|
|
|
|
|
|
|
|
|
if (!context_ime->client_surface)
|
|
|
|
|
return attrs;
|
|
|
|
@@ -545,7 +518,6 @@ get_pango_attr_list (GtkIMContextIME *context_ime, const char *utf8str)
|
|
|
|
|
if (context_ime->preediting)
|
|
|
|
|
{
|
|
|
|
|
const char *schr = utf8str, *echr;
|
|
|
|
|
guint8 *buf;
|
|
|
|
|
guint16 f_red, f_green, f_blue, b_red, b_green, b_blue;
|
|
|
|
|
glong len, spos = 0, epos, sidx = 0, eidx;
|
|
|
|
|
PangoAttribute *attr;
|
|
|
|
@@ -554,7 +526,7 @@ get_pango_attr_list (GtkIMContextIME *context_ime, const char *utf8str)
|
|
|
|
|
* get attributes list of IME.
|
|
|
|
|
*/
|
|
|
|
|
len = ImmGetCompositionStringW (himc, GCS_COMPATTR, NULL, 0);
|
|
|
|
|
buf = g_alloca (len);
|
|
|
|
|
buf = g_malloc (len);
|
|
|
|
|
ImmGetCompositionStringW (himc, GCS_COMPATTR, buf, len);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
@@ -623,6 +595,7 @@ get_pango_attr_list (GtkIMContextIME *context_ime, const char *utf8str)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImmReleaseContext (hwnd, himc);
|
|
|
|
|
g_free (buf);
|
|
|
|
|
|
|
|
|
|
return attrs;
|
|
|
|
|
}
|
|
|
|
@@ -640,20 +613,18 @@ gtk_im_context_ime_get_preedit_string (GtkIMContext *context,
|
|
|
|
|
|
|
|
|
|
context_ime = GTK_IM_CONTEXT_IME (context);
|
|
|
|
|
|
|
|
|
|
utf8str = get_utf8_preedit_string (context_ime, &pos);
|
|
|
|
|
if (!context_ime->focus || context_ime->priv->pretend_empty_preedit)
|
|
|
|
|
utf8str = g_strdup ("");
|
|
|
|
|
else
|
|
|
|
|
utf8str = get_utf8_preedit_string (context_ime, GCS_COMPSTR, &pos);
|
|
|
|
|
|
|
|
|
|
if (attrs)
|
|
|
|
|
*attrs = get_pango_attr_list (context_ime, utf8str);
|
|
|
|
|
|
|
|
|
|
if (str)
|
|
|
|
|
{
|
|
|
|
|
*str = utf8str;
|
|
|
|
|
}
|
|
|
|
|
*str = utf8str;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_free (utf8str);
|
|
|
|
|
utf8str = NULL;
|
|
|
|
|
}
|
|
|
|
|
utf8str = NULL;
|
|
|
|
|
|
|
|
|
|
if (cursor_pos)
|
|
|
|
|
*cursor_pos = pos;
|
|
|
|
@@ -674,41 +645,45 @@ gtk_im_context_ime_focus_in (GtkIMContext *context)
|
|
|
|
|
/* switch current context */
|
|
|
|
|
context_ime->focus = TRUE;
|
|
|
|
|
|
|
|
|
|
hwnd = gdk_win32_surface_get_impl_hwnd (context_ime->client_surface);
|
|
|
|
|
himc = ImmGetContext (hwnd);
|
|
|
|
|
if (!himc)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
toplevel = context_ime->client_surface;
|
|
|
|
|
if (GDK_IS_SURFACE (toplevel))
|
|
|
|
|
{
|
|
|
|
|
gdk_win32_display_add_filter (GDK_WIN32_DISPLAY (gdk_surface_get_display (toplevel)),
|
|
|
|
|
gtk_im_context_ime_message_filter, context_ime);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
if (!GDK_IS_SURFACE (toplevel))
|
|
|
|
|
{
|
|
|
|
|
g_warning ("gtk_im_context_ime_focus_in(): "
|
|
|
|
|
"cannot find toplevel window.");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* restore preedit context */
|
|
|
|
|
ImmSetConversionStatus (himc,
|
|
|
|
|
context_ime->priv->conversion_mode,
|
|
|
|
|
context_ime->priv->sentence_mode);
|
|
|
|
|
hwnd = gdk_win32_surface_get_impl_hwnd (toplevel);
|
|
|
|
|
himc = ImmGetContext (hwnd);
|
|
|
|
|
if (!himc)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (context_ime->opened)
|
|
|
|
|
gdk_win32_display_add_filter (GDK_WIN32_DISPLAY (gdk_surface_get_display (toplevel)),
|
|
|
|
|
gtk_im_context_ime_message_filter, context_ime);
|
|
|
|
|
|
|
|
|
|
/* restore preedit context */
|
|
|
|
|
context_ime->opened = ImmGetOpenStatus (himc);
|
|
|
|
|
|
|
|
|
|
switch (context_ime->priv->focus_behavior)
|
|
|
|
|
{
|
|
|
|
|
if (!ImmGetOpenStatus (himc))
|
|
|
|
|
ImmSetOpenStatus (himc, TRUE);
|
|
|
|
|
if (context_ime->preediting)
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_COMMIT:
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_DISCARD:
|
|
|
|
|
gtk_im_context_ime_reset (context);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_FOLLOW:
|
|
|
|
|
{
|
|
|
|
|
ImmSetCompositionStringW (himc,
|
|
|
|
|
SCS_SETSTR,
|
|
|
|
|
context_ime->priv->comp_str,
|
|
|
|
|
context_ime->priv->comp_str_len, NULL, 0);
|
|
|
|
|
FREE_PREEDIT_BUFFER (context_ime);
|
|
|
|
|
gchar *utf8str = get_utf8_preedit_string (context_ime, GCS_COMPSTR, NULL);
|
|
|
|
|
if (utf8str != NULL && strlen(utf8str) > 0)
|
|
|
|
|
{
|
|
|
|
|
context_ime->preediting = TRUE;
|
|
|
|
|
gtk_im_context_ime_set_cursor_location (context, NULL);
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-start");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
}
|
|
|
|
|
g_free (utf8str);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* clean */
|
|
|
|
@@ -720,78 +695,66 @@ static void
|
|
|
|
|
gtk_im_context_ime_focus_out (GtkIMContext *context)
|
|
|
|
|
{
|
|
|
|
|
GtkIMContextIME *context_ime = GTK_IM_CONTEXT_IME (context);
|
|
|
|
|
GdkSurface *toplevel;
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
HIMC himc;
|
|
|
|
|
gboolean was_preediting;
|
|
|
|
|
|
|
|
|
|
if (!GDK_IS_SURFACE (context_ime->client_surface))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* switch current context */
|
|
|
|
|
was_preediting = context_ime->preediting;
|
|
|
|
|
context_ime->opened = FALSE;
|
|
|
|
|
context_ime->preediting = FALSE;
|
|
|
|
|
context_ime->focus = FALSE;
|
|
|
|
|
|
|
|
|
|
hwnd = gdk_win32_surface_get_impl_hwnd (context_ime->client_surface);
|
|
|
|
|
himc = ImmGetContext (hwnd);
|
|
|
|
|
if (!himc)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* save preedit context */
|
|
|
|
|
ImmGetConversionStatus (himc,
|
|
|
|
|
&context_ime->priv->conversion_mode,
|
|
|
|
|
&context_ime->priv->sentence_mode);
|
|
|
|
|
|
|
|
|
|
if (ImmGetOpenStatus (himc))
|
|
|
|
|
switch (context_ime->priv->focus_behavior)
|
|
|
|
|
{
|
|
|
|
|
gboolean preediting = context_ime->preediting;
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_COMMIT:
|
|
|
|
|
if (was_preediting)
|
|
|
|
|
{
|
|
|
|
|
gchar *utf8str = get_utf8_preedit_string (context_ime, GCS_COMPSTR, NULL);
|
|
|
|
|
|
|
|
|
|
if (preediting)
|
|
|
|
|
{
|
|
|
|
|
FREE_PREEDIT_BUFFER (context_ime);
|
|
|
|
|
context_ime->priv->pretend_empty_preedit = TRUE;
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-end");
|
|
|
|
|
g_signal_emit_by_name (context, "commit", utf8str);
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-start");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
context_ime->priv->pretend_empty_preedit = FALSE;
|
|
|
|
|
g_free (utf8str);
|
|
|
|
|
}
|
|
|
|
|
/* fallthrough */
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_DISCARD:
|
|
|
|
|
gtk_im_context_ime_reset (context);
|
|
|
|
|
|
|
|
|
|
context_ime->priv->comp_str_len
|
|
|
|
|
= ImmGetCompositionStringW (himc, GCS_COMPSTR, NULL, 0);
|
|
|
|
|
context_ime->priv->comp_str
|
|
|
|
|
= g_malloc (context_ime->priv->comp_str_len);
|
|
|
|
|
ImmGetCompositionStringW (himc, GCS_COMPSTR,
|
|
|
|
|
context_ime->priv->comp_str,
|
|
|
|
|
context_ime->priv->comp_str_len);
|
|
|
|
|
/* Callbacks triggered by im_context_ime_reset() could set the focus back to our
|
|
|
|
|
context. In that case, we want to exit here. */
|
|
|
|
|
|
|
|
|
|
context_ime->priv->read_str_len
|
|
|
|
|
= ImmGetCompositionStringW (himc, GCS_COMPREADSTR, NULL, 0);
|
|
|
|
|
context_ime->priv->read_str
|
|
|
|
|
= g_malloc (context_ime->priv->read_str_len);
|
|
|
|
|
ImmGetCompositionStringW (himc, GCS_COMPREADSTR,
|
|
|
|
|
context_ime->priv->read_str,
|
|
|
|
|
context_ime->priv->read_str_len);
|
|
|
|
|
}
|
|
|
|
|
if (context_ime->focus)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
ImmSetOpenStatus (himc, FALSE);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
context_ime->opened = TRUE;
|
|
|
|
|
context_ime->preediting = preediting;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
context_ime->opened = FALSE;
|
|
|
|
|
context_ime->preediting = FALSE;
|
|
|
|
|
case GTK_WIN32_IME_FOCUS_BEHAVIOR_FOLLOW:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* remove event filter */
|
|
|
|
|
toplevel = context_ime->client_surface;
|
|
|
|
|
if (GDK_IS_SURFACE (toplevel))
|
|
|
|
|
if (GDK_IS_SURFACE (context_ime->client_surface))
|
|
|
|
|
{
|
|
|
|
|
gdk_win32_display_remove_filter (GDK_WIN32_DISPLAY (gdk_surface_get_display (toplevel)),
|
|
|
|
|
gdk_win32_display_remove_filter (GDK_WIN32_DISPLAY (gdk_surface_get_display (context_ime->client_surface)),
|
|
|
|
|
gtk_im_context_ime_message_filter,
|
|
|
|
|
context_ime);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
|
|
|
|
|
if (was_preediting)
|
|
|
|
|
{
|
|
|
|
|
g_warning ("gtk_im_context_ime_focus_out(): "
|
|
|
|
|
"cannot find toplevel window.");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-end");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* clean */
|
|
|
|
|
ImmReleaseContext (hwnd, himc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@@ -1012,16 +975,16 @@ gtk_im_context_ime_message_filter (GdkWin32Display *display,
|
|
|
|
|
get_window_position (context_ime->client_surface, &wx, &wy);
|
|
|
|
|
/* FIXME! */
|
|
|
|
|
{
|
|
|
|
|
HWND hwnd_top;
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
POINT pt;
|
|
|
|
|
RECT rc;
|
|
|
|
|
|
|
|
|
|
hwnd_top =
|
|
|
|
|
hwnd =
|
|
|
|
|
gdk_win32_surface_get_impl_hwnd (context_ime->client_surface);
|
|
|
|
|
GetWindowRect (hwnd_top, &rc);
|
|
|
|
|
GetWindowRect (hwnd, &rc);
|
|
|
|
|
pt.x = wx;
|
|
|
|
|
pt.y = wy;
|
|
|
|
|
ClientToScreen (hwnd_top, &pt);
|
|
|
|
|
ClientToScreen (hwnd, &pt);
|
|
|
|
|
wx = pt.x - rc.left;
|
|
|
|
|
wy = pt.y - rc.top;
|
|
|
|
|
}
|
|
|
|
@@ -1037,38 +1000,32 @@ gtk_im_context_ime_message_filter (GdkWin32Display *display,
|
|
|
|
|
|
|
|
|
|
if (msg->lParam & GCS_RESULTSTR)
|
|
|
|
|
{
|
|
|
|
|
gsize len;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gchar *utf8str = get_utf8_preedit_string (context_ime, GCS_RESULTSTR, NULL);
|
|
|
|
|
|
|
|
|
|
len = ImmGetCompositionStringW (himc, GCS_RESULTSTR, NULL, 0);
|
|
|
|
|
|
|
|
|
|
if (len > 0)
|
|
|
|
|
if (utf8str)
|
|
|
|
|
{
|
|
|
|
|
gpointer buf = g_alloca (len);
|
|
|
|
|
ImmGetCompositionStringW (himc, GCS_RESULTSTR, buf, len);
|
|
|
|
|
len /= 2;
|
|
|
|
|
context_ime->commit_string = g_utf16_to_utf8 (buf, len, NULL, NULL, &error);
|
|
|
|
|
if (error)
|
|
|
|
|
{
|
|
|
|
|
g_warning ("%s", error->message);
|
|
|
|
|
g_error_free (error);
|
|
|
|
|
}
|
|
|
|
|
context_ime->priv->pretend_empty_preedit = TRUE;
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-end");
|
|
|
|
|
|
|
|
|
|
if (context_ime->commit_string)
|
|
|
|
|
{
|
|
|
|
|
g_signal_emit_by_name (context, "commit", context_ime->commit_string);
|
|
|
|
|
g_free (context_ime->commit_string);
|
|
|
|
|
context_ime->commit_string = NULL;
|
|
|
|
|
retval = GDK_WIN32_MESSAGE_FILTER_REMOVE;
|
|
|
|
|
}
|
|
|
|
|
g_signal_emit_by_name (context, "commit", utf8str);
|
|
|
|
|
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-start");
|
|
|
|
|
g_signal_emit_by_name (context, "preedit-changed");
|
|
|
|
|
context_ime->priv->pretend_empty_preedit = FALSE;
|
|
|
|
|
|
|
|
|
|
retval = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (utf8str);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (context_ime->use_preedit)
|
|
|
|
|
retval = GDK_WIN32_MESSAGE_FILTER_REMOVE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case WM_IME_STARTCOMPOSITION:
|
|
|
|
|
context_ime->preediting = TRUE;
|
|
|
|
|
gtk_im_context_ime_set_cursor_location (context, NULL);
|
|
|
|
|