1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * GtkQueryTips: Query onscreen widgets for their tooltips
5 * Copyright (C) 1998 Tim Janik
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the
19 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20 * Boston, MA 02111-1307, USA.
24 * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
25 * file for a list of people on the GTK+ Team. See the ChangeLog
26 * files for a list of changes. These files are distributed with
27 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
30 #undef GTK_DISABLE_DEPRECATED
33 #include "gtktipsquery.h"
34 #include "gtksignal.h"
35 #include "gtktooltips.h"
37 #include "gtkmarshalers.h"
44 /* --- arguments --- */
59 SIGNAL_WIDGET_ENTERED,
60 SIGNAL_WIDGET_SELECTED,
64 /* --- prototypes --- */
65 static void gtk_tips_query_class_init (GtkTipsQueryClass *class);
66 static void gtk_tips_query_init (GtkTipsQuery *tips_query);
67 static void gtk_tips_query_destroy (GtkObject *object);
68 static gint gtk_tips_query_event (GtkWidget *widget,
70 static void gtk_tips_query_set_arg (GtkObject *object,
73 static void gtk_tips_query_get_arg (GtkObject *object,
76 static void gtk_tips_query_real_start_query (GtkTipsQuery *tips_query);
77 static void gtk_tips_query_real_stop_query (GtkTipsQuery *tips_query);
78 static void gtk_tips_query_widget_entered (GtkTipsQuery *tips_query,
80 const gchar *tip_text,
81 const gchar *tip_private);
84 /* --- variables --- */
85 static GtkLabelClass *parent_class = NULL;
86 static guint tips_query_signals[SIGNAL_LAST] = { 0 };
89 /* --- functions --- */
91 gtk_tips_query_get_type (void)
93 static GtkType tips_query_type = 0;
97 static const GtkTypeInfo tips_query_info =
100 sizeof (GtkTipsQuery),
101 sizeof (GtkTipsQueryClass),
102 (GtkClassInitFunc) gtk_tips_query_class_init,
103 (GtkObjectInitFunc) gtk_tips_query_init,
104 /* reserved_1 */ NULL,
105 /* reserved_2 */ NULL,
106 (GtkClassInitFunc) NULL,
110 tips_query_type = gtk_type_unique (gtk_label_get_type (), &tips_query_info);
113 return tips_query_type;
117 gtk_tips_query_class_init (GtkTipsQueryClass *class)
119 GtkObjectClass *object_class;
120 GtkWidgetClass *widget_class;
122 object_class = (GtkObjectClass*) class;
123 widget_class = (GtkWidgetClass*) class;
125 parent_class = gtk_type_class (gtk_label_get_type ());
128 object_class->set_arg = gtk_tips_query_set_arg;
129 object_class->get_arg = gtk_tips_query_get_arg;
130 object_class->destroy = gtk_tips_query_destroy;
132 widget_class->event = gtk_tips_query_event;
134 class->start_query = gtk_tips_query_real_start_query;
135 class->stop_query = gtk_tips_query_real_stop_query;
136 class->widget_entered = gtk_tips_query_widget_entered;
137 class->widget_selected = NULL;
139 gtk_object_add_arg_type ("GtkTipsQuery::emit-always", GTK_TYPE_BOOL, GTK_ARG_READWRITE | G_PARAM_STATIC_NAME, ARG_EMIT_ALWAYS);
140 gtk_object_add_arg_type ("GtkTipsQuery::caller", GTK_TYPE_WIDGET, GTK_ARG_READWRITE | G_PARAM_STATIC_NAME, ARG_CALLER);
141 gtk_object_add_arg_type ("GtkTipsQuery::label-inactive", GTK_TYPE_STRING, GTK_ARG_READWRITE | G_PARAM_STATIC_NAME, ARG_LABEL_INACTIVE);
142 gtk_object_add_arg_type ("GtkTipsQuery::label-no-tip", GTK_TYPE_STRING, GTK_ARG_READWRITE | G_PARAM_STATIC_NAME, ARG_LABEL_NO_TIP);
144 tips_query_signals[SIGNAL_START_QUERY] =
145 gtk_signal_new (I_("start-query"),
147 GTK_CLASS_TYPE (object_class),
148 GTK_SIGNAL_OFFSET (GtkTipsQueryClass, start_query),
149 _gtk_marshal_VOID__VOID,
151 tips_query_signals[SIGNAL_STOP_QUERY] =
152 gtk_signal_new (I_("stop-query"),
154 GTK_CLASS_TYPE (object_class),
155 GTK_SIGNAL_OFFSET (GtkTipsQueryClass, stop_query),
156 _gtk_marshal_VOID__VOID,
158 tips_query_signals[SIGNAL_WIDGET_ENTERED] =
159 gtk_signal_new (I_("widget-entered"),
161 GTK_CLASS_TYPE (object_class),
162 GTK_SIGNAL_OFFSET (GtkTipsQueryClass, widget_entered),
163 _gtk_marshal_VOID__OBJECT_STRING_STRING,
168 tips_query_signals[SIGNAL_WIDGET_SELECTED] =
169 g_signal_new (I_("widget-selected"),
170 G_TYPE_FROM_CLASS(object_class),
172 G_STRUCT_OFFSET(GtkTipsQueryClass, widget_selected),
173 _gtk_boolean_handled_accumulator, NULL,
174 _gtk_marshal_BOOLEAN__OBJECT_STRING_STRING_BOXED,
179 GDK_TYPE_EVENT | G_SIGNAL_TYPE_STATIC_SCOPE);
183 gtk_tips_query_init (GtkTipsQuery *tips_query)
185 tips_query->emit_always = FALSE;
186 tips_query->in_query = FALSE;
187 tips_query->label_inactive = g_strdup ("");
188 tips_query->label_no_tip = g_strdup (_("--- No Tip ---"));
189 tips_query->caller = NULL;
190 tips_query->last_crossed = NULL;
191 tips_query->query_cursor = NULL;
193 gtk_label_set_text (GTK_LABEL (tips_query), tips_query->label_inactive);
197 gtk_tips_query_set_arg (GtkObject *object,
201 GtkTipsQuery *tips_query;
203 tips_query = GTK_TIPS_QUERY (object);
207 case ARG_EMIT_ALWAYS:
208 tips_query->emit_always = (GTK_VALUE_BOOL (*arg) != FALSE);
211 gtk_tips_query_set_caller (tips_query, GTK_WIDGET (GTK_VALUE_OBJECT (*arg)));
213 case ARG_LABEL_INACTIVE:
214 gtk_tips_query_set_labels (tips_query, GTK_VALUE_STRING (*arg), tips_query->label_no_tip);
216 case ARG_LABEL_NO_TIP:
217 gtk_tips_query_set_labels (tips_query, tips_query->label_inactive, GTK_VALUE_STRING (*arg));
225 gtk_tips_query_get_arg (GtkObject *object,
229 GtkTipsQuery *tips_query;
231 tips_query = GTK_TIPS_QUERY (object);
235 case ARG_EMIT_ALWAYS:
236 GTK_VALUE_BOOL (*arg) = tips_query->emit_always;
239 GTK_VALUE_OBJECT (*arg) = (GtkObject*) tips_query->caller;
241 case ARG_LABEL_INACTIVE:
242 GTK_VALUE_STRING (*arg) = g_strdup (tips_query->label_inactive);
244 case ARG_LABEL_NO_TIP:
245 GTK_VALUE_STRING (*arg) = g_strdup (tips_query->label_no_tip);
248 arg->type = GTK_TYPE_INVALID;
254 gtk_tips_query_destroy (GtkObject *object)
256 GtkTipsQuery *tips_query = GTK_TIPS_QUERY (object);
258 if (tips_query->in_query)
259 gtk_tips_query_stop_query (tips_query);
261 gtk_tips_query_set_caller (tips_query, NULL);
263 g_free (tips_query->label_inactive);
264 tips_query->label_inactive = NULL;
265 g_free (tips_query->label_no_tip);
266 tips_query->label_no_tip = NULL;
268 GTK_OBJECT_CLASS (parent_class)->destroy (object);
272 gtk_tips_query_new (void)
274 GtkTipsQuery *tips_query;
276 tips_query = gtk_type_new (gtk_tips_query_get_type ());
278 return GTK_WIDGET (tips_query);
282 gtk_tips_query_set_labels (GtkTipsQuery *tips_query,
283 const gchar *label_inactive,
284 const gchar *label_no_tip)
288 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
289 g_return_if_fail (label_inactive != NULL);
290 g_return_if_fail (label_no_tip != NULL);
292 old = tips_query->label_inactive;
293 tips_query->label_inactive = g_strdup (label_inactive);
295 old = tips_query->label_no_tip;
296 tips_query->label_no_tip = g_strdup (label_no_tip);
301 gtk_tips_query_set_caller (GtkTipsQuery *tips_query,
304 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
305 g_return_if_fail (tips_query->in_query == FALSE);
307 g_return_if_fail (GTK_IS_WIDGET (caller));
310 g_object_ref (caller);
312 if (tips_query->caller)
313 g_object_unref (tips_query->caller);
315 tips_query->caller = caller;
319 gtk_tips_query_start_query (GtkTipsQuery *tips_query)
321 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
322 g_return_if_fail (tips_query->in_query == FALSE);
323 g_return_if_fail (gtk_widget_get_realized (GTK_WIDGET (tips_query)));
325 tips_query->in_query = TRUE;
326 gtk_signal_emit (GTK_OBJECT (tips_query), tips_query_signals[SIGNAL_START_QUERY]);
330 gtk_tips_query_stop_query (GtkTipsQuery *tips_query)
332 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
333 g_return_if_fail (tips_query->in_query == TRUE);
335 gtk_signal_emit (GTK_OBJECT (tips_query), tips_query_signals[SIGNAL_STOP_QUERY]);
336 tips_query->in_query = FALSE;
340 gtk_tips_query_real_start_query (GtkTipsQuery *tips_query)
344 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
346 tips_query->query_cursor = gdk_cursor_new_for_display (gtk_widget_get_display (GTK_WIDGET (tips_query)),
348 failure = gdk_pointer_grab (GTK_WIDGET (tips_query)->window,
350 GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK |
351 GDK_ENTER_NOTIFY_MASK | GDK_LEAVE_NOTIFY_MASK,
353 tips_query->query_cursor,
357 gdk_cursor_unref (tips_query->query_cursor);
358 tips_query->query_cursor = NULL;
360 gtk_grab_add (GTK_WIDGET (tips_query));
364 gtk_tips_query_real_stop_query (GtkTipsQuery *tips_query)
366 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
368 gtk_grab_remove (GTK_WIDGET (tips_query));
369 if (tips_query->query_cursor)
371 gdk_display_pointer_ungrab (gtk_widget_get_display (GTK_WIDGET (tips_query)),
373 gdk_cursor_unref (tips_query->query_cursor);
374 tips_query->query_cursor = NULL;
376 if (tips_query->last_crossed)
378 g_object_unref (tips_query->last_crossed);
379 tips_query->last_crossed = NULL;
382 gtk_label_set_text (GTK_LABEL (tips_query), tips_query->label_inactive);
386 gtk_tips_query_widget_entered (GtkTipsQuery *tips_query,
388 const gchar *tip_text,
389 const gchar *tip_private)
391 g_return_if_fail (GTK_IS_TIPS_QUERY (tips_query));
394 tip_text = tips_query->label_no_tip;
396 if (!g_str_equal (GTK_LABEL (tips_query)->label, (gchar*) tip_text))
397 gtk_label_set_text (GTK_LABEL (tips_query), tip_text);
401 gtk_tips_query_emit_widget_entered (GtkTipsQuery *tips_query,
404 GtkTooltipsData *tdata;
406 if (widget == (GtkWidget*) tips_query)
410 tdata = gtk_tooltips_data_get (widget);
414 if (!widget && tips_query->last_crossed)
416 gtk_signal_emit (GTK_OBJECT (tips_query),
417 tips_query_signals[SIGNAL_WIDGET_ENTERED],
421 g_object_unref (tips_query->last_crossed);
422 tips_query->last_crossed = NULL;
424 else if (widget && widget != tips_query->last_crossed)
426 g_object_ref (widget);
427 if (tdata || tips_query->emit_always)
428 gtk_signal_emit (GTK_OBJECT (tips_query),
429 tips_query_signals[SIGNAL_WIDGET_ENTERED],
431 tdata ? tdata->tip_text : NULL,
432 tdata ? tdata->tip_private : NULL);
433 if (tips_query->last_crossed)
434 g_object_unref (tips_query->last_crossed);
435 tips_query->last_crossed = widget;
440 gtk_tips_query_event (GtkWidget *widget,
443 GtkTipsQuery *tips_query;
444 GtkWidget *event_widget;
445 gboolean event_handled;
447 g_return_val_if_fail (GTK_IS_TIPS_QUERY (widget), FALSE);
449 tips_query = GTK_TIPS_QUERY (widget);
450 if (!tips_query->in_query)
452 if (GTK_WIDGET_CLASS (parent_class)->event)
453 return GTK_WIDGET_CLASS (parent_class)->event (widget, event);
458 event_widget = gtk_get_event_widget (event);
460 event_handled = FALSE;
463 GdkWindow *pointer_window;
465 case GDK_LEAVE_NOTIFY:
467 pointer_window = gdk_window_get_pointer (event_widget->window, NULL, NULL, NULL);
469 pointer_window = NULL;
473 gpointer event_widget_ptr;
474 gdk_window_get_user_data (pointer_window, &event_widget_ptr);
475 event_widget = event_widget_ptr;
477 gtk_tips_query_emit_widget_entered (tips_query, event_widget);
478 event_handled = TRUE;
481 case GDK_ENTER_NOTIFY:
482 gtk_tips_query_emit_widget_entered (tips_query, event_widget);
483 event_handled = TRUE;
486 case GDK_BUTTON_PRESS:
487 case GDK_BUTTON_RELEASE:
490 if (event_widget == (GtkWidget*) tips_query ||
491 event_widget == tips_query->caller)
492 gtk_tips_query_stop_query (tips_query);
496 GtkTooltipsData *tdata;
499 tdata = gtk_tooltips_data_get (event_widget);
500 if (tdata || tips_query->emit_always)
501 gtk_signal_emit (GTK_OBJECT (tips_query),
502 tips_query_signals[SIGNAL_WIDGET_SELECTED],
504 tdata ? tdata->tip_text : NULL,
505 tdata ? tdata->tip_private : NULL,
510 gtk_tips_query_stop_query (tips_query);
513 event_handled = TRUE;
520 return event_handled;
523 #define __GTK_TIPS_QUERY_C__
524 #include "gtkaliasdef.c"