summaryrefslogtreecommitdiff
path: root/modules
diff options
context:
space:
mode:
authorOwen Taylor <otaylor@redhat.com>2003-08-19 21:17:18 +0000
committerOwen Taylor <otaylor@src.gnome.org>2003-08-19 21:17:18 +0000
commit55b0689a1d97522cec96b62bd6c3692b1f9c7654 (patch)
treeffe17ced0996e4f1f56efe9372a8f03edade9679 /modules
parentf9063a112241f8c0579330bb911596d06bf10336 (diff)
downloadgtk+-55b0689a1d97522cec96b62bd6c3692b1f9c7654.tar.gz
Substantially rework the handling of status windows:
Mon Aug 18 17:19:12 2003 Owen Taylor <otaylor@redhat.com> * modules/input/gtkimcontextxim.[ch]: Substantially rework the handling of status windows: - Store the current StatusWindow in the GtkIMContextXIM structure and vice-versa, so we don't have to hunt the window hierarchy on cleanup. - Use the Gtkidget hierarchy instead of/or as well as the GdkWindow hierarchy when finding the toplevel; this helps for things like GtkHandlebox - Watch GtkWidget::hierarchy_changed to catch changes in the toplevel without changes in the GdkWindow (reparenting) - Never create the GtkWindow for the status window unless we have text to display. - Various cleanups, add lots of comments. (#115077, much help from Takuro Ashie and Hidetoshi Tajima in tracking this down and figuring out a fix.) * modules/input/gtkimcontextxim.c (gtk_im_context_xim_focus_in): * modules/input/gtkimcontextxim.c: Track the current screen for each toplevel so that we show the status window on the right screen. (#116340, James Su) * modules/input/gtkimcontextxim.c: If create a new IC when we currently have the focus, call XSetICFocus() on it. * modules/input/gtkimcontextxim.c (get_im): Fix bug with multiple open screens.
Diffstat (limited to 'modules')
-rw-r--r--modules/input/gtkimcontextxim.c571
-rw-r--r--modules/input/gtkimcontextxim.h37
2 files changed, 407 insertions, 201 deletions
diff --git a/modules/input/gtkimcontextxim.c b/modules/input/gtkimcontextxim.c
index aa55961249..2be36e2026 100644
--- a/modules/input/gtkimcontextxim.c
+++ b/modules/input/gtkimcontextxim.c
@@ -27,6 +27,49 @@
#include "gtkimcontextxim.h"
typedef struct _StatusWindow StatusWindow;
+typedef struct _GtkXIMInfo GtkXIMInfo;
+
+struct _GtkIMContextXIM
+{
+ GtkIMContext object;
+
+ GtkXIMInfo *im_info;
+
+ gchar *locale;
+ gchar *mb_charset;
+
+ GdkWindow *client_window;
+ GtkWidget *client_widget;
+
+ /* The status window for this input context; we claim the
+ * status window when we are focused and have created an XIC
+ */
+ StatusWindow *status_window;
+
+ gint preedit_size;
+ gint preedit_length;
+ gunichar *preedit_chars;
+ XIMFeedback *feedbacks;
+
+ gint preedit_cursor;
+
+ XIMCallback preedit_start_callback;
+ XIMCallback preedit_done_callback;
+ XIMCallback preedit_draw_callback;
+ XIMCallback preedit_caret_callback;
+
+ XIMCallback status_start_callback;
+ XIMCallback status_done_callback;
+ XIMCallback status_draw_callback;
+
+ XIC ic;
+
+ guint filter_key_release : 1;
+ guint use_preedit : 1;
+ guint finalizing : 1;
+ guint in_toplevel : 1;
+ guint has_focus : 1;
+};
struct _GtkXIMInfo
{
@@ -52,10 +95,9 @@ struct _StatusWindow
/* Toplevel window to which the status window corresponds */
GtkWidget *toplevel;
-
- /* Signal connection ids; we connect to the toplevel */
- gulong destroy_handler_id;
- gulong configure_handler_id;
+
+ /* Currently focused GtkIMContextXIM for the toplevel, if any */
+ GtkIMContextXIM *context;
};
static void gtk_im_context_xim_class_init (GtkIMContextXIMClass *class);
@@ -83,10 +125,13 @@ static void set_ic_client_window (GtkIMContextXIM *context_xim,
static void setup_styles (GtkXIMInfo *info);
-static void status_window_show (GtkIMContextXIM *context_xim);
-static void status_window_hide (GtkIMContextXIM *context_xim);
-static void status_window_set_text (GtkIMContextXIM *context_xim,
- const gchar *text);
+static void update_client_widget (GtkIMContextXIM *context_xim);
+static void update_status_window (GtkIMContextXIM *context_xim);
+
+static StatusWindow *status_window_get (GtkWidget *toplevel);
+static void status_window_free (StatusWindow *status_window);
+static void status_window_set_text (StatusWindow *status_window,
+ const gchar *text);
static void xim_destroy_callback (XIM xim,
XPointer client_data,
@@ -430,14 +475,19 @@ get_im (GdkWindow *client_window,
tmp_list = open_ims;
while (tmp_list)
{
- info = tmp_list->data;
- if (info->screen == screen &&
- strcmp (info->locale, locale) == 0)
+ GtkXIMInfo *tmp_info = tmp_list->data;
+ if (tmp_info->screen == screen &&
+ strcmp (tmp_info->locale, locale) == 0)
{
- if (info->im)
- return info;
+ if (tmp_info->im)
+ {
+ return tmp_info;
+ }
else
- break;
+ {
+ tmp_info = tmp_info;
+ break;
+ }
}
tmp_list = tmp_list->next;
}
@@ -488,8 +538,9 @@ gtk_im_context_xim_init (GtkIMContextXIM *im_context_xim)
{
im_context_xim->use_preedit = TRUE;
im_context_xim->filter_key_release = FALSE;
- im_context_xim->status_visible = FALSE;
im_context_xim->finalizing = FALSE;
+ im_context_xim->has_focus = FALSE;
+ im_context_xim->in_toplevel = FALSE;
}
static void
@@ -511,8 +562,8 @@ reinitialize_ic (GtkIMContextXIM *context_xim)
if (context_xim->ic)
{
XDestroyIC (context_xim->ic);
- status_window_hide (context_xim);
context_xim->ic = NULL;
+ update_status_window (context_xim);
if (context_xim->preedit_length)
{
@@ -546,6 +597,8 @@ set_ic_client_window (GtkIMContextXIM *context_xim,
context_xim->im_info = get_im (context_xim->client_window, context_xim->locale);
context_xim->im_info->ics = g_slist_prepend (context_xim->im_info->ics, context_xim);
}
+
+ update_client_widget (context_xim);
}
static void
@@ -688,14 +741,17 @@ static void
gtk_im_context_xim_focus_in (GtkIMContext *context)
{
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
- XIC ic = gtk_im_context_xim_get_ic (context_xim);
-
- if (!ic)
- return;
- XSetICFocus (ic);
+ if (!context_xim->has_focus)
+ {
+ XIC ic = gtk_im_context_xim_get_ic (context_xim);
- status_window_show (context_xim);
+ context_xim->has_focus = TRUE;
+ update_status_window (context_xim);
+
+ if (ic)
+ XSetICFocus (ic);
+ }
return;
}
@@ -704,14 +760,17 @@ static void
gtk_im_context_xim_focus_out (GtkIMContext *context)
{
GtkIMContextXIM *context_xim = GTK_IM_CONTEXT_XIM (context);
- XIC ic = gtk_im_context_xim_get_ic (context_xim);
- if (!ic)
- return;
-
- XUnsetICFocus (ic);
-
- status_window_hide (context_xim);
+ if (context_xim->has_focus)
+ {
+ XIC ic = gtk_im_context_xim_get_ic (context_xim);
+
+ context_xim->has_focus = FALSE;
+ update_status_window (context_xim);
+
+ if (ic)
+ XUnsetICFocus (ic);
+ }
return;
}
@@ -1101,18 +1160,13 @@ status_draw_callback (XIC xic,
{
GtkIMContextXIM *context = GTK_IM_CONTEXT_XIM (client_data);
- if (!context->status_visible)
- return;
-
if (call_data->type == XIMTextType)
{
gchar *text;
xim_text_to_utf8 (context, call_data->data.text, &text);
- if (text)
- status_window_set_text (context, text);
- else
- status_window_set_text (context, "");
+ if (context->status_window)
+ status_window_set_text (context->status_window, text ? text : "");
}
else /* bitmap */
{
@@ -1226,200 +1280,389 @@ gtk_im_context_xim_get_ic (GtkIMContextXIM *context_xim)
XGetICValues (xic,
XNFilterEvents, &mask,
NULL);
- context_xim->filter_key_release = (mask & KeyReleaseMask);
+ context_xim->filter_key_release = (mask & KeyReleaseMask) != 0;
}
+
context_xim->ic = xic;
+
+ update_status_window (context_xim);
+
+ if (xic && context_xim->has_focus)
+ XSetICFocus (xic);
}
return context_xim->ic;
}
-/**************************
- * *
- * Status Window handling *
- * *
- **************************/
+/*****************************************************************
+ * Status Window handling
+ *
+ * A status window is a small window attached to the toplevel
+ * that is used to display information to the user about the
+ * current input operation.
+ *
+ * We claim the toplevel's status window for an input context if:
+ *
+ * A) The input context has a toplevel
+ * B) The input context has the focus
+ * C) The input context has an XIC associated with it
+ *
+ * Tracking A) and C) is pretty reliable since we
+ * compute A) and create the XIC for C) ourselves.
+ * For B) we basically have to depend on our callers
+ * calling ::focus-in and ::focus-out at the right time.
+ *
+ * The toplevel is computed by walking up the GdkWindow
+ * hierarchy from context->client_window until we find a
+ * window that is owned by some widget, and then calling
+ * gtk_widget_get_toplevel() on that widget. This should
+ * handle both cases where we might have GdkWindows without widgets,
+ * and cases where GtkWidgets have strange window hierarchies
+ * (like a torn off GtkHandleBox.)
+ *
+ * The status window is visible if and only if there is text
+ * for it; whenever a new GtkIMContextXIM claims the status
+ * window, we blank out any existing text. We actually only
+ * create a GtkWindow for the status window the first time
+ * it is shown; this is an important optimization when we are
+ * using XIM with something like a simple compose-key input
+ * method that never needs a status window.
+ *****************************************************************/
+
+/* Called when we no longer need a status window
+*/
+static void
+disclaim_status_window (GtkIMContextXIM *context_xim)
+{
+ if (context_xim->status_window)
+ {
+ g_assert (context_xim->status_window->context == context_xim);
+
+ status_window_set_text (context_xim->status_window, "");
+
+ context_xim->status_window->context = NULL;
+ context_xim->status_window = NULL;
+ }
+}
-static gboolean
-status_window_expose_event (GtkWidget *widget,
- GdkEventExpose *event)
+/* Called when we need a status window
+ */
+static void
+claim_status_window (GtkIMContextXIM *context_xim)
{
- gdk_draw_rectangle (widget->window,
- widget->style->base_gc [GTK_STATE_NORMAL],
- TRUE,
- 0, 0,
- widget->allocation.width, widget->allocation.height);
- gdk_draw_rectangle (widget->window,
- widget->style->text_gc [GTK_STATE_NORMAL],
- FALSE,
- 0, 0,
- widget->allocation.width - 1, widget->allocation.height - 1);
+ if (!context_xim->status_window && context_xim->client_widget)
+ {
+ GtkWidget *toplevel = gtk_widget_get_toplevel (context_xim->client_widget);
+ if (toplevel && GTK_WIDGET_TOPLEVEL (toplevel))
+ {
+ StatusWindow *status_window = status_window_get (toplevel);
- return FALSE;
+ if (status_window->context)
+ disclaim_status_window (status_window->context);
+
+ status_window->context = context_xim;
+ context_xim->status_window = status_window;
+ }
+ }
}
+/* Basic call made whenever something changed that might cause
+ * us to need, or not to need a status window.
+ */
static void
-status_window_style_set (GtkWidget *toplevel,
- GtkStyle *previous_style,
- GtkWidget *label)
+update_status_window (GtkIMContextXIM *context_xim)
{
- gint i;
+ if (context_xim->ic && context_xim->in_toplevel && context_xim->has_focus)
+ claim_status_window (context_xim);
+ else
+ disclaim_status_window (context_xim);
+}
+
+/* Updates the in_toplevel flag for @context_xim
+ */
+static void
+update_in_toplevel (GtkIMContextXIM *context_xim)
+{
+ if (context_xim->client_widget)
+ {
+ GtkWidget *toplevel = gtk_widget_get_toplevel (context_xim->client_widget);
+
+ context_xim->in_toplevel = (toplevel && GTK_WIDGET_TOPLEVEL (toplevel));
+ }
+ else
+ context_xim->in_toplevel = FALSE;
+
+ /* Some paranoia, in case we don't get a focus out */
+ if (!context_xim->in_toplevel)
+ context_xim->has_focus = FALSE;
- for (i = 0; i < 5; i++)
- gtk_widget_modify_fg (label, i, &toplevel->style->text[i]);
+ update_status_window (context_xim);
}
-/* Frees a status window and removes its link from the status_windows list */
+/* Callback when @widget's toplevel changes. It will always
+ * change from NULL to a window, or a window to NULL;
+ * we use that intermediate NULL state to make sure
+ * that we disclaim the toplevel status window for the old
+ * window.
+ */
static void
-status_window_free (StatusWindow *status_window)
+on_client_widget_hierarchy_changed (GtkWidget *widget,
+ GtkWidget *old_toplevel,
+ GtkIMContextXIM *context_xim)
{
- status_windows = g_slist_remove (status_windows, status_window);
-
- g_signal_handler_disconnect (status_window->toplevel, status_window->destroy_handler_id);
- g_signal_handler_disconnect (status_window->toplevel, status_window->configure_handler_id);
- gtk_widget_destroy (status_window->window);
- g_object_set_data (G_OBJECT (status_window->toplevel), "gtk-im-xim-status-window", NULL);
-
- g_free (status_window);
+ update_in_toplevel (context_xim);
+}
+
+/* Finds the GtkWidget that owns the window, or if none, the
+ * widget owning the nearest parent that has a widget.
+ */
+static GtkWidget *
+widget_for_window (GdkWindow *window)
+{
+ while (window)
+ {
+ gpointer user_data;
+ gdk_window_get_user_data (window, &user_data);
+ if (user_data)
+ return user_data;
+
+ window = gdk_window_get_parent (window);
+ }
+
+ return NULL;
+}
+
+/* Called when context_xim->client_window changes; takes care of
+ * removing and/or setting up our watches for the toplevel
+ */
+static void
+update_client_widget (GtkIMContextXIM *context_xim)
+{
+ GtkWidget *new_client_widget = widget_for_window (context_xim->client_window);
+
+ if (new_client_widget != context_xim->client_widget)
+ {
+ if (context_xim->client_widget)
+ {
+ g_signal_handlers_disconnect_by_func (context_xim->client_widget,
+ G_CALLBACK (on_client_widget_hierarchy_changed),
+ context_xim);
+ }
+ context_xim->client_widget = new_client_widget;
+ if (context_xim->client_widget)
+ {
+ g_signal_connect (context_xim->client_widget, "hierarchy-changed",
+ G_CALLBACK (on_client_widget_hierarchy_changed),
+ context_xim);
+ }
+
+ update_in_toplevel (context_xim);
+ }
+}
+
+/* Called when the toplevel is destroyed; frees the status window
+ */
+static void
+on_status_toplevel_destroy (GtkWidget *toplevel,
+ StatusWindow *status_window)
+{
+ status_window_free (status_window);
+}
+
+/* Called when the screen for the toplevel changes; updates the
+ * screen for the status window to match.
+ */
+static void
+on_status_toplevel_notify_screen (GtkWindow *toplevel,
+ GParamSpec *pspec,
+ StatusWindow *status_window)
+{
+ if (status_window->window)
+ gtk_window_set_screen (GTK_WINDOW (status_window->window),
+ gtk_widget_get_screen (GTK_WIDGET (toplevel)));
}
+/* Called when the toplevel window is moved; updates the position of
+ * the status window to follow it.
+ */
static gboolean
-status_window_configure (GtkWidget *toplevel,
- GdkEventConfigure *event,
- StatusWindow *status_window)
+on_status_toplevel_configure (GtkWidget *toplevel,
+ GdkEventConfigure *event,
+ StatusWindow *status_window)
{
GdkRectangle rect;
GtkRequisition requisition;
gint y;
- gint height = gdk_screen_get_height (gtk_widget_get_screen (toplevel));
-
- gdk_window_get_frame_extents (toplevel->window, &rect);
- gtk_widget_size_request (status_window->window, &requisition);
+ gint height;
- if (rect.y + rect.height + requisition.height < height)
- y = rect.y + rect.height;
- else
- y = height - requisition.height;
+ if (status_window->window)
+ {
+ height = gdk_screen_get_height (gtk_widget_get_screen (toplevel));
- gtk_window_move (GTK_WINDOW (status_window->window), rect.x, y);
+ gdk_window_get_frame_extents (toplevel->window, &rect);
+ gtk_widget_size_request (status_window->window, &requisition);
+
+ if (rect.y + rect.height + requisition.height < height)
+ y = rect.y + rect.height;
+ else
+ y = height - requisition.height;
+
+ gtk_window_move (GTK_WINDOW (status_window->window), rect.x, y);
+ }
return FALSE;
}
-static GtkWidget *
-status_window_get (GtkIMContextXIM *context_xim,
- gboolean create)
+/* Frees a status window and removes its link from the status_windows list
+ */
+static void
+status_window_free (StatusWindow *status_window)
{
- GdkWindow *toplevel_gdk;
- GtkWidget *toplevel;
- GtkWidget *window;
- StatusWindow *status_window;
- GtkWidget *status_label;
- GdkScreen *screen;
- GdkWindow *root_window;
-
- if (!context_xim->client_window)
- return NULL;
+ status_windows = g_slist_remove (status_windows, status_window);
- toplevel_gdk = context_xim->client_window;
- screen = gdk_drawable_get_screen (toplevel_gdk);
- root_window = gdk_screen_get_root_window (screen);
+ if (status_window->context)
+ status_window->context->status_window = NULL;
+
+ g_signal_handlers_disconnect_by_func (status_window->toplevel,
+ G_CALLBACK (on_status_toplevel_destroy),
+ status_window);
+ g_signal_handlers_disconnect_by_func (status_window->toplevel,
+ G_CALLBACK (on_status_toplevel_notify_screen),
+ status_window);
+ g_signal_handlers_disconnect_by_func (status_window->toplevel,
+ G_CALLBACK (on_status_toplevel_configure),
+ status_window);
+
+ if (status_window->window)
+ gtk_widget_destroy (status_window->window);
- while (TRUE)
- {
- GdkWindow *parent = gdk_window_get_parent (toplevel_gdk);
- if (parent == root_window)
- break;
- else
- toplevel_gdk = parent;
- }
+ g_object_set_data (G_OBJECT (status_window->toplevel), "gtk-im-xim-status-window", NULL);
+
+ g_free (status_window);
+}
- gdk_window_get_user_data (toplevel_gdk, (gpointer *)&toplevel);
- if (!toplevel)
- return NULL;
+/* Finds the status window object for a toplevel, creating it if necessary.
+ */
+static StatusWindow *
+status_window_get (GtkWidget *toplevel)
+{
+ StatusWindow *status_window;
status_window = g_object_get_data (G_OBJECT (toplevel), "gtk-im-xim-status-window");
if (status_window)
- return status_window->window;
- else if (!create)
- return NULL;
-
- status_window = g_new (StatusWindow, 1);
- status_window->window = gtk_window_new (GTK_WINDOW_POPUP);
+ return status_window;
+
+ status_window = g_new0 (StatusWindow, 1);
status_window->toplevel = toplevel;
status_windows = g_slist_prepend (status_windows, status_window);
- window = status_window->window;
-
- gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
- gtk_widget_set_app_paintable (window, TRUE);
-
- status_label = gtk_label_new ("");
- gtk_misc_set_padding (GTK_MISC (status_label), 1, 1);
- gtk_widget_show (status_label);
+ g_signal_connect (toplevel, "destroy",
+ G_CALLBACK (on_status_toplevel_destroy),
+ status_window);
+ g_signal_connect (toplevel, "configure_event",
+ G_CALLBACK (on_status_toplevel_configure),
+ status_window);
+ g_signal_connect (toplevel, "notify::screen",
+ G_CALLBACK (on_status_toplevel_notify_screen),
+ status_window);
- gtk_container_add (GTK_CONTAINER (window), status_label);
-
- status_window->destroy_handler_id = g_signal_connect_swapped (toplevel, "destroy",
- G_CALLBACK (status_window_free),
- status_window);
- status_window->configure_handler_id = g_signal_connect (toplevel, "configure_event",
- G_CALLBACK (status_window_configure),
- status_window);
-
- status_window_configure (toplevel, NULL, status_window);
-
- g_signal_connect (window, "style_set",
- G_CALLBACK (status_window_style_set), status_label);
- g_signal_connect (window, "expose_event",
- G_CALLBACK (status_window_expose_event), NULL);
-
g_object_set_data (G_OBJECT (toplevel), "gtk-im-xim-status-window", status_window);
- return window;
+ return status_window;
}
+/* Draw the background (normally white) and border for the status window
+ */
static gboolean
-status_window_has_text (GtkWidget *status_window)
+on_status_window_expose_event (GtkWidget *widget,
+ GdkEventExpose *event)
{
- GtkWidget *label = GTK_BIN (status_window)->child;
- const gchar *text = gtk_label_get_text (GTK_LABEL (label));
+ gdk_draw_rectangle (widget->window,
+ widget->style->base_gc [GTK_STATE_NORMAL],
+ TRUE,
+ 0, 0,
+ widget->allocation.width, widget->allocation.height);
+ gdk_draw_rectangle (widget->window,
+ widget->style->text_gc [GTK_STATE_NORMAL],
+ FALSE,
+ 0, 0,
+ widget->allocation.width - 1, widget->allocation.height - 1);
- return text[0] != '\0';
+ return FALSE;
}
+/* We watch the ::style-set signal for our label widget
+ * and use that to change it's foreground color to match
+ * the 'text' color of the toplevel window. The text/base
+ * pair of colors might be reversed from the fg/bg pair
+ * that are normally used for labels.
+ */
static void
-status_window_show (GtkIMContextXIM *context_xim)
+on_status_window_style_set (GtkWidget *toplevel,
+ GtkStyle *previous_style,
+ GtkWidget *label)
{
- context_xim->status_visible = TRUE;
+ gint i;
+
+ for (i = 0; i < 5; i++)
+ gtk_widget_modify_fg (label, i, &toplevel->style->text[i]);
}
+/* Creates the widgets for the status window; called when we
+ * first need to show text for the status window.
+ */
static void
-status_window_hide (GtkIMContextXIM *context_xim)
+status_window_make_window (StatusWindow *status_window)
{
- GtkWidget *status_window = status_window_get (context_xim, FALSE);
+ GtkWidget *window;
+ GtkWidget *status_label;
+
+ status_window->window = gtk_window_new (GTK_WINDOW_POPUP);
+ window = status_window->window;
- context_xim->status_visible = FALSE;
+ gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
+ gtk_widget_set_app_paintable (window, TRUE);
- if (status_window)
- status_window_set_text (context_xim, "");
+ status_label = gtk_label_new ("");
+ gtk_misc_set_padding (GTK_MISC (status_label), 1, 1);
+ gtk_widget_show (status_label);
+
+ g_signal_connect (window, "style_set",
+ G_CALLBACK (on_status_window_style_set), status_label);
+ gtk_container_add (GTK_CONTAINER (window), status_label);
+
+ g_signal_connect (window, "expose_event",
+ G_CALLBACK (on_status_window_expose_event), NULL);
+
+ gtk_window_set_screen (GTK_WINDOW (status_window->window),
+ gtk_widget_get_screen (status_window->toplevel));
+
+ on_status_toplevel_configure (status_window->toplevel, NULL, status_window);
}
+/* Updates the text in the status window, hiding or
+ * showing the window as necessary.
+ */
static void
-status_window_set_text (GtkIMContextXIM *context_xim,
- const gchar *text)
+status_window_set_text (StatusWindow *status_window,
+ const gchar *text)
{
- GtkWidget *status_window = status_window_get (context_xim, TRUE);
-
- if (status_window)
+ if (text[0])
{
- GtkWidget *label = GTK_BIN (status_window)->child;
- gtk_label_set_text (GTK_LABEL (label), text);
+ GtkWidget *label;
- if (context_xim->status_visible && status_window_has_text (status_window))
- gtk_widget_show (status_window);
- else
- gtk_widget_hide (status_window);
+ if (!status_window->window)
+ status_window_make_window (status_window);
+
+ label = GTK_BIN (status_window->window)->child;
+ gtk_label_set_text (GTK_LABEL (label), text);
+
+ gtk_widget_show (status_window->window);
+ }
+ else
+ {
+ if (status_window->window)
+ gtk_widget_hide (status_window->window);
}
}
diff --git a/modules/input/gtkimcontextxim.h b/modules/input/gtkimcontextxim.h
index 006ecbeacc..bdcdb08cd5 100644
--- a/modules/input/gtkimcontextxim.h
+++ b/modules/input/gtkimcontextxim.h
@@ -41,43 +41,6 @@ extern GType gtk_type_im_context_xim;
typedef struct _GtkIMContextXIM GtkIMContextXIM;
typedef struct _GtkIMContextXIMClass GtkIMContextXIMClass;
-typedef struct _GtkXIMInfo GtkXIMInfo;
-
-struct _GtkIMContextXIM
-{
- GtkIMContext object;
-
- GtkXIMInfo *im_info;
-
- gchar *locale;
- gchar *mb_charset;
-
- GdkWindow *client_window;
-
- gint preedit_size;
- gint preedit_length;
- gunichar *preedit_chars;
- XIMFeedback *feedbacks;
-
- gint preedit_cursor;
-
- XIMCallback preedit_start_callback;
- XIMCallback preedit_done_callback;
- XIMCallback preedit_draw_callback;
- XIMCallback preedit_caret_callback;
-
- XIMCallback status_start_callback;
- XIMCallback status_done_callback;
- XIMCallback status_draw_callback;
-
- XIC ic;
- gboolean filter_key_release;
-
- guint use_preedit : 1;
- guint status_visible : 1;
- guint finalizing : 1;
-};
-
struct _GtkIMContextXIMClass
{
GtkIMContextClass parent_class;