]> Pileus Git - ~andy/gtk/blob - gtk/gtkinputdialog.c
Fixes #136082 and #135265, patch by Morten Welinder.
[~andy/gtk] / gtk / gtkinputdialog.c
1 /* GTK - The GIMP Toolkit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the
16  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  */
19
20 /*
21  * gtkinputdialog.c
22  *
23  * Copyright 1997 Owen Taylor <owt1@cornell.edu>
24  *
25  */
26
27 /*
28  * Modified by the GTK+ Team and others 1997-2000.  See the AUTHORS
29  * file for a list of people on the GTK+ Team.  See the ChangeLog
30  * files for a list of changes.  These files are distributed with
31  * GTK+ at ftp://ftp.gtk.org/pub/gtk/. 
32  */
33
34
35 #include <config.h>
36 #include <glib/gprintf.h>
37 #include <stdlib.h>
38 #include "gdk/gdkkeysyms.h"
39 #include "gtkbutton.h"
40 #include "gtkentry.h"
41 #include "gtkhbox.h"
42 #include "gtkhseparator.h"
43 #include "gtkinputdialog.h"
44 #include "gtklabel.h"
45 #include "gtklistitem.h"
46 #include "gtkmain.h"
47 #include "gtkmarshalers.h"
48 #include "gtkmenu.h"
49 #include "gtkmenuitem.h"
50 #include "gtknotebook.h"
51 #include "gtkscrolledwindow.h"
52 #include "gtkstock.h"
53 #include "gtktable.h"
54 #include "gtkvbox.h"
55
56 #undef GTK_DISABLE_DEPRECATED
57 #include "gtkoptionmenu.h"
58 #define GTK_DISABLE_DEPRECATED
59
60 #include "gtkintl.h"
61
62 typedef struct _GtkInputDialogPrivate GtkInputDialogPrivate;
63 typedef struct _GtkInputKeyInfo       GtkInputKeyInfo;
64
65 struct _GtkInputDialogPrivate
66 {
67   GtkWidget *device_menu;
68   GtkWidget *device_optionmenu;
69   GtkWidget *no_devices_label;
70   GtkWidget *main_vbox;
71 };
72
73 struct _GtkInputKeyInfo
74 {
75   gint       index;
76   GtkWidget *entry;
77   GtkInputDialog *inputd;
78 };
79
80 enum
81 {
82   ENABLE_DEVICE,
83   DISABLE_DEVICE,
84   LAST_SIGNAL
85 };
86
87
88 #define AXIS_LIST_WIDTH 160
89 #define AXIS_LIST_HEIGHT 175
90
91 #define KEYS_LIST_WIDTH 200
92 #define KEYS_LIST_HEIGHT 175
93
94 /* Forward declarations */
95
96 static void gtk_input_dialog_class_init       (GtkInputDialogClass *klass);
97 static void gtk_input_dialog_init             (GtkInputDialog      *inputd);
98 static void gtk_input_dialog_screen_changed   (GtkWidget           *widget,
99                                                GdkScreen           *previous_screen);
100 static void gtk_input_dialog_set_device       (GtkWidget           *widget,
101                                                gpointer             data);
102 static void gtk_input_dialog_set_mapping_mode (GtkWidget           *w,
103                                                gpointer             data);
104 static void gtk_input_dialog_set_axis         (GtkWidget           *widget,
105                                                gpointer             data);
106 static void gtk_input_dialog_fill_axes        (GtkInputDialog      *inputd,
107                                                GdkDevice           *info);
108 static void gtk_input_dialog_set_key          (GtkInputKeyInfo     *key,
109                                                guint                keyval,
110                                                GdkModifierType      modifiers);
111 static gboolean gtk_input_dialog_key_press    (GtkWidget           *widget,
112                                                GdkEventKey         *event,
113                                                GtkInputKeyInfo     *key);
114 static void gtk_input_dialog_clear_key        (GtkWidget           *widget,
115                                                GtkInputKeyInfo     *key);
116 static void gtk_input_dialog_destroy_key      (GtkWidget           *widget,
117                                                GtkInputKeyInfo     *key);
118 static void gtk_input_dialog_fill_keys        (GtkInputDialog      *inputd,
119                                                GdkDevice           *info);
120
121 static GtkObjectClass *parent_class = NULL;
122 static guint input_dialog_signals[LAST_SIGNAL] = { 0 };
123
124 GType
125 gtk_input_dialog_get_type (void)
126 {
127   static GType input_dialog_type = 0;
128
129   if (!input_dialog_type)
130     {
131       static const GTypeInfo input_dialog_info =
132       {
133         sizeof (GtkInputDialogClass),
134         NULL,           /* base_init */
135         NULL,           /* base_finalize */
136         (GClassInitFunc) gtk_input_dialog_class_init,
137         NULL,           /* class_finalize */
138         NULL,           /* class_data */
139         sizeof (GtkInputDialog),
140         0,              /* n_preallocs */
141         (GInstanceInitFunc) gtk_input_dialog_init,
142       };
143
144       input_dialog_type =
145         g_type_register_static (GTK_TYPE_DIALOG, "GtkInputDialog",
146                                 &input_dialog_info, 0);
147     }
148
149   return input_dialog_type;
150 }
151
152 GtkInputDialogPrivate *
153 gtk_input_dialog_get_private (GtkInputDialog *input_dialog)
154 {
155   GtkInputDialogPrivate *private;
156   static GQuark private_quark = 0;
157
158   if (!private_quark)
159     private_quark = g_quark_from_static_string ("gtk-input-dialog-private");
160
161   private = g_object_get_qdata (G_OBJECT (input_dialog), private_quark);
162
163   if (!private)
164     {
165       private = g_new0 (GtkInputDialogPrivate, 1);
166       g_object_set_qdata_full (G_OBJECT (input_dialog), private_quark,
167                                private, g_free);
168     }
169
170   return private;
171 }
172
173 static GtkInputDialog *
174 input_dialog_from_widget (GtkWidget *widget)
175 {
176   GtkWidget *toplevel;
177   
178   if (GTK_IS_MENU_ITEM (widget))
179     {
180       GtkMenu *menu = GTK_MENU (widget->parent);
181       widget = gtk_menu_get_attach_widget (menu);
182     }
183
184   toplevel = gtk_widget_get_toplevel (widget);
185   return GTK_INPUT_DIALOG (toplevel);
186 }
187
188 static void
189 gtk_input_dialog_class_init (GtkInputDialogClass *klass)
190 {
191   GtkWidgetClass *widget_class = (GtkWidgetClass *)klass;
192   
193   parent_class = g_type_class_peek_parent (klass);
194
195   widget_class->screen_changed = gtk_input_dialog_screen_changed;
196   
197   klass->enable_device = NULL;
198   klass->disable_device = NULL;
199
200   input_dialog_signals[ENABLE_DEVICE] =
201     g_signal_new ("enable_device",
202                   G_OBJECT_CLASS_TYPE (klass),
203                   G_SIGNAL_RUN_LAST,
204                   G_STRUCT_OFFSET (GtkInputDialogClass, enable_device),
205                   NULL, NULL,
206                   _gtk_marshal_VOID__OBJECT,
207                   G_TYPE_NONE, 1,
208                   GDK_TYPE_DEVICE);
209
210   input_dialog_signals[DISABLE_DEVICE] =
211     g_signal_new ("disable_device",
212                   G_OBJECT_CLASS_TYPE (klass),
213                   G_SIGNAL_RUN_LAST,
214                   G_STRUCT_OFFSET (GtkInputDialogClass, disable_device),
215                   NULL, NULL,
216                   _gtk_marshal_VOID__OBJECT,
217                   G_TYPE_NONE, 1,
218                   GDK_TYPE_DEVICE);
219 }
220
221 static void
222 gtk_input_dialog_init (GtkInputDialog *inputd)
223 {
224   GtkInputDialogPrivate *private = gtk_input_dialog_get_private (inputd);
225   GtkWidget *util_box;
226   GtkWidget *label;
227   GtkWidget *mapping_menu;
228   GtkWidget *menuitem;
229   GtkWidget *separator;
230   GtkWidget *notebook;
231
232   gtk_widget_push_composite_child ();
233
234   gtk_window_set_title (GTK_WINDOW (inputd), _("Input"));
235
236   /* main vbox */
237
238   private->main_vbox = gtk_vbox_new (FALSE, 4);
239   gtk_container_set_border_width (GTK_CONTAINER (private->main_vbox), 5);
240   gtk_box_pack_start (GTK_BOX (GTK_DIALOG (inputd)->vbox), private->main_vbox,
241                       TRUE, TRUE, 0);
242
243   private->no_devices_label = gtk_label_new (_("No extended input devices"));
244   gtk_box_pack_start (GTK_BOX (GTK_DIALOG (inputd)->vbox),
245                       private->no_devices_label,
246                       TRUE, TRUE, 0);
247
248   /* menu for selecting device */
249
250   private->device_menu = gtk_menu_new ();
251
252   util_box = gtk_hbox_new (FALSE, 2);
253   gtk_box_pack_start (GTK_BOX (private->main_vbox), util_box, FALSE, FALSE, 0);
254
255   label = gtk_label_new_with_mnemonic (_("_Device:"));
256   gtk_box_pack_start (GTK_BOX (util_box), label, FALSE, FALSE, 2);
257
258   private->device_optionmenu = gtk_option_menu_new ();
259   gtk_label_set_mnemonic_widget (GTK_LABEL (label), private->device_optionmenu);
260   gtk_box_pack_start (GTK_BOX (util_box), private->device_optionmenu, TRUE, TRUE, 2);
261   gtk_widget_show (private->device_optionmenu);
262   gtk_option_menu_set_menu (GTK_OPTION_MENU (private->device_optionmenu), private->device_menu);
263
264   gtk_widget_show (label);
265
266   /* Device options */
267
268   /* mapping mode option menu */
269
270   mapping_menu = gtk_menu_new ();
271
272   menuitem = gtk_menu_item_new_with_label(_("Disabled"));
273   gtk_menu_shell_append (GTK_MENU_SHELL (mapping_menu), menuitem);
274   gtk_widget_show (menuitem);
275   g_signal_connect (menuitem, "activate",
276                     G_CALLBACK (gtk_input_dialog_set_mapping_mode),
277                     GINT_TO_POINTER (GDK_MODE_DISABLED));
278
279   menuitem = gtk_menu_item_new_with_label(_("Screen"));
280   gtk_menu_shell_append (GTK_MENU_SHELL (mapping_menu), menuitem);
281   gtk_widget_show (menuitem);
282   g_signal_connect (menuitem, "activate",
283                     G_CALLBACK (gtk_input_dialog_set_mapping_mode),
284                     GINT_TO_POINTER (GDK_MODE_SCREEN));
285
286   menuitem = gtk_menu_item_new_with_label(_("Window"));
287   gtk_menu_shell_append (GTK_MENU_SHELL (mapping_menu), menuitem);
288   gtk_widget_show (menuitem);
289   g_signal_connect (menuitem, "activate",
290                     G_CALLBACK (gtk_input_dialog_set_mapping_mode),
291                     GINT_TO_POINTER (GDK_MODE_WINDOW));
292
293   label = gtk_label_new_with_mnemonic (_("_Mode: "));
294   gtk_box_pack_start (GTK_BOX (util_box), label, FALSE, FALSE, 2);
295   
296   inputd->mode_optionmenu = gtk_option_menu_new ();
297   gtk_label_set_mnemonic_widget (GTK_LABEL (label), inputd->mode_optionmenu);
298   gtk_box_pack_start (GTK_BOX (util_box), inputd->mode_optionmenu, FALSE, FALSE, 2);
299   gtk_widget_show (inputd->mode_optionmenu);
300   gtk_option_menu_set_menu (GTK_OPTION_MENU (inputd->mode_optionmenu), mapping_menu);
301
302   gtk_widget_show(label);
303
304   gtk_widget_show (util_box);
305
306   util_box = gtk_hbox_new (FALSE, 2);
307   gtk_box_pack_start (GTK_BOX(private->main_vbox), util_box, FALSE, FALSE, 0);
308
309   gtk_widget_show (label);
310   gtk_widget_show (util_box);
311
312   separator = gtk_hseparator_new();
313   gtk_box_pack_start (GTK_BOX (private->main_vbox), separator, FALSE, TRUE, 0);
314   gtk_widget_show (separator);
315
316   /* Notebook */
317
318   notebook = gtk_notebook_new ();
319   gtk_box_pack_start (GTK_BOX (private->main_vbox), notebook, TRUE, TRUE, 0);
320   gtk_widget_show (notebook);
321       
322   /*  The axis listbox  */
323
324   label = gtk_label_new_with_mnemonic (_("_Axes"));
325
326   inputd->axis_listbox = gtk_scrolled_window_new (NULL, NULL);
327   gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW(inputd->axis_listbox),
328                                   GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
329       
330   gtk_widget_set_size_request (inputd->axis_listbox,
331                                AXIS_LIST_WIDTH, AXIS_LIST_HEIGHT);
332   gtk_notebook_append_page (GTK_NOTEBOOK(notebook), 
333                             inputd->axis_listbox, label);
334
335   gtk_widget_show (inputd->axis_listbox);
336
337   inputd->axis_list = 0;
338
339   /* Keys listbox */
340
341   label = gtk_label_new_with_mnemonic (_("_Keys"));
342
343   inputd->keys_listbox = gtk_scrolled_window_new (NULL, NULL);
344   gtk_widget_set_size_request (inputd->keys_listbox,
345                                KEYS_LIST_WIDTH, KEYS_LIST_HEIGHT);
346   gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (inputd->keys_listbox),
347                                   GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
348   gtk_notebook_append_page (GTK_NOTEBOOK (notebook), 
349                             inputd->keys_listbox, label);
350
351   gtk_widget_show (inputd->keys_listbox);
352
353   inputd->keys_list = 0;
354
355   inputd->save_button = gtk_button_new_from_stock (GTK_STOCK_SAVE);
356   GTK_WIDGET_SET_FLAGS (inputd->save_button, GTK_CAN_DEFAULT);
357   gtk_box_pack_start (GTK_BOX (GTK_DIALOG(inputd)->action_area),
358                       inputd->save_button, TRUE, TRUE, 0);
359   gtk_widget_show (inputd->save_button);
360
361   inputd->close_button = gtk_button_new_from_stock (GTK_STOCK_CLOSE);
362   GTK_WIDGET_SET_FLAGS (inputd->close_button, GTK_CAN_DEFAULT);
363   gtk_box_pack_start (GTK_BOX (GTK_DIALOG(inputd)->action_area),
364                       inputd->close_button, TRUE, TRUE, 0);
365
366   gtk_widget_show (inputd->close_button);
367   gtk_widget_grab_default (inputd->close_button);
368
369   gtk_widget_pop_composite_child ();
370
371   gtk_input_dialog_screen_changed (GTK_WIDGET (inputd), NULL);
372 }
373
374 static void
375 gtk_input_dialog_screen_changed (GtkWidget *widget,
376                                  GdkScreen *previous_screen)
377 {
378   GtkInputDialog *inputd = GTK_INPUT_DIALOG (widget);
379   GtkInputDialogPrivate *private = gtk_input_dialog_get_private (inputd);
380   
381   GList *device_info = NULL;
382   GdkDevice *core_pointer = NULL;
383   GList *tmp_list;
384
385   if (gtk_widget_has_screen (widget))
386     {
387       GdkDisplay *display;
388       
389       display = gtk_widget_get_display (widget);
390       device_info = gdk_display_list_devices (display);
391       core_pointer = gdk_display_get_core_pointer (display);
392     }
393
394   inputd->current_device = NULL;
395   gtk_container_foreach (GTK_CONTAINER (private->device_menu),
396                          (GtkCallback)gtk_widget_destroy, NULL);
397   
398   if (g_list_length(device_info) <= 1) /* only core device */
399     {
400       gtk_widget_hide (private->main_vbox);
401       gtk_widget_show (private->no_devices_label);
402       gtk_widget_set_sensitive(inputd->save_button, FALSE);
403     }
404   else
405     {
406       gtk_widget_show (private->main_vbox);
407       gtk_widget_hide (private->no_devices_label);
408       gtk_widget_set_sensitive(inputd->save_button, TRUE);
409
410       for (tmp_list = device_info; tmp_list; tmp_list = tmp_list->next)
411         {
412           GdkDevice *info = tmp_list->data;
413           if (info != core_pointer)
414             {
415               GtkWidget *menuitem;
416               
417               menuitem = gtk_menu_item_new_with_label (info->name);
418               
419               gtk_menu_shell_append (GTK_MENU_SHELL (private->device_menu),
420                                      menuitem);
421               gtk_widget_show (menuitem);
422               g_signal_connect (menuitem, "activate",
423                                 G_CALLBACK (gtk_input_dialog_set_device),
424                                 info);
425             }
426         }
427       
428       gtk_input_dialog_set_device (widget, device_info->data);
429       gtk_option_menu_set_history (GTK_OPTION_MENU (private->device_optionmenu), 0);
430     }
431 }
432      
433 GtkWidget*
434 gtk_input_dialog_new (void)
435 {
436   GtkInputDialog *inputd;
437
438   inputd = g_object_new (GTK_TYPE_INPUT_DIALOG, NULL);
439
440   return GTK_WIDGET (inputd);
441 }
442
443 static void
444 gtk_input_dialog_set_device (GtkWidget *w,
445                              gpointer   data)
446 {
447   GdkDevice *device = data;
448   GtkInputDialog *inputd = input_dialog_from_widget (w);
449
450   inputd->current_device = device;
451
452   gtk_input_dialog_fill_axes (inputd, device);
453   gtk_input_dialog_fill_keys (inputd, device);
454
455   gtk_option_menu_set_history (GTK_OPTION_MENU (inputd->mode_optionmenu),
456                                device->mode);
457 }
458
459 static void
460 gtk_input_dialog_set_mapping_mode (GtkWidget *w,
461                                    gpointer   data)
462 {
463   GtkInputDialog *inputd = input_dialog_from_widget (w);
464   GdkDevice *info = inputd->current_device;
465   GdkInputMode old_mode = info->mode;
466   GdkInputMode mode = GPOINTER_TO_INT (data);
467
468   if (!info)
469     return;
470
471   if (mode != old_mode)
472     {
473       if (gdk_device_set_mode (info, mode))
474         {
475           if (mode == GDK_MODE_DISABLED)
476             g_signal_emit (inputd,
477                            input_dialog_signals[DISABLE_DEVICE],
478                            0,
479                            info);
480           else
481             g_signal_emit (inputd,
482                            input_dialog_signals[ENABLE_DEVICE],
483                            0,
484                            info);
485         }
486       else
487         gtk_option_menu_set_history (GTK_OPTION_MENU (inputd->mode_optionmenu),
488                                      old_mode);
489
490       /* FIXME: error dialog ? */
491     }
492 }
493
494 static void
495 gtk_input_dialog_set_axis (GtkWidget *w,
496                            gpointer   data)
497 {
498   GdkAxisUse use = GPOINTER_TO_INT(data) & 0xFFFF;
499   GdkAxisUse old_use;
500   GdkAxisUse *new_axes;
501   GtkInputDialog *inputd = input_dialog_from_widget (w);
502   GdkDevice *info = inputd->current_device;
503
504   gint axis = (GPOINTER_TO_INT(data) >> 16) - 1;
505   gint old_axis;
506   int i;
507
508   if (!info)
509     return;
510
511   new_axes = g_new (GdkAxisUse, info->num_axes);
512   old_axis = -1;
513   for (i=0;i<info->num_axes;i++)
514     {
515       new_axes[i] = info->axes[i].use;
516       if (info->axes[i].use == use)
517         old_axis = i;
518     }
519
520   if (axis != -1)
521     old_use = info->axes[axis].use;
522   else
523     old_use = GDK_AXIS_IGNORE;
524
525   if (axis == old_axis)
526     return;
527
528   /* we must always have an x and a y axis */
529   if ((axis == -1 && (use == GDK_AXIS_X || use == GDK_AXIS_Y)) ||
530       (old_axis == -1 && (old_use == GDK_AXIS_X || old_use == GDK_AXIS_Y)))
531     {
532       gtk_option_menu_set_history (
533                 GTK_OPTION_MENU (inputd->axis_items[use]),
534                 old_axis + 1);
535     }
536   else
537     {
538       if (axis != -1)
539         gdk_device_set_axis_use (info, axis, use);
540
541       if (old_axis != -1)
542         gdk_device_set_axis_use (info, old_axis, old_use);
543
544       if (old_use != GDK_AXIS_IGNORE)
545         {
546           gtk_option_menu_set_history (
547                 GTK_OPTION_MENU (inputd->axis_items[old_use]),
548                 old_axis + 1);
549         }
550     }
551
552   g_free (new_axes);
553 }
554
555 static void
556 gtk_input_dialog_fill_axes(GtkInputDialog *inputd, GdkDevice *info)
557 {
558   static const char *axis_use_strings[GDK_AXIS_LAST] =
559   {
560     "",
561     N_("X"),
562     N_("Y"),
563     N_("Pressure"),
564     N_("X Tilt"),
565     N_("Y Tilt"),
566     N_("Wheel")
567   };
568
569   int i,j;
570   GtkWidget *menu;
571   GtkWidget *option_menu;
572   GtkWidget *label;
573
574   /* remove all the old items */
575   if (inputd->axis_list)
576     {
577       gtk_widget_hide (inputd->axis_list);      /* suppress resizes (or get warnings) */
578       gtk_widget_destroy (inputd->axis_list);
579     }
580   inputd->axis_list = gtk_table_new (GDK_AXIS_LAST, 2, 0);
581   gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (inputd->axis_listbox), 
582                                          inputd->axis_list);
583   gtk_widget_show (inputd->axis_list);
584
585   gtk_widget_realize (inputd->axis_list);
586   gdk_window_set_background (inputd->axis_list->window,
587                              &inputd->axis_list->style->base[GTK_STATE_NORMAL]);
588
589   for (i=GDK_AXIS_X;i<GDK_AXIS_LAST;i++)
590     {
591       /* create the label */
592
593       label = gtk_label_new (_(axis_use_strings[i]));
594       gtk_table_attach (GTK_TABLE (inputd->axis_list), label, 0, 1, i, i+1, 
595                         0, 0, 2, 2);
596
597       /* and the use option menu */
598       menu = gtk_menu_new();
599
600       for (j = -1; j < info->num_axes; j++)
601         {
602           char buffer[16];
603           GtkWidget *menu_item;
604
605           if (j == -1)
606             menu_item = gtk_menu_item_new_with_label (_("none"));
607           else
608             {
609               g_snprintf (buffer, sizeof (buffer), "%d", j+1);
610               menu_item = gtk_menu_item_new_with_label (buffer);
611             }
612           g_signal_connect (menu_item, "activate",
613                             G_CALLBACK (gtk_input_dialog_set_axis),
614                             GINT_TO_POINTER (0x10000 * (j + 1) + i));
615           gtk_widget_show (menu_item);
616           gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
617         }
618
619       inputd->axis_items[i] = option_menu = gtk_option_menu_new ();
620       gtk_table_attach (GTK_TABLE (inputd->axis_list), option_menu, 
621                         1, 2, i, i+1, 0, 0, 2, 2);
622
623       gtk_widget_show (option_menu);
624       gtk_option_menu_set_menu (GTK_OPTION_MENU (option_menu), menu);
625       for (j = 0; j < info->num_axes; j++)
626         if (info->axes[j].use == (GdkAxisUse) i)
627           {
628             gtk_option_menu_set_history (GTK_OPTION_MENU (option_menu), j+1);
629             break;
630           }
631
632       gtk_widget_show (label);
633     }
634 }
635
636 static void 
637 gtk_input_dialog_clear_key (GtkWidget *widget, GtkInputKeyInfo *key)
638 {
639   if (!key->inputd->current_device)
640     return;
641   
642   gtk_entry_set_text (GTK_ENTRY(key->entry), _("(disabled)"));
643   gdk_device_set_key (key->inputd->current_device, key->index, 0, 0);
644 }
645
646 static void 
647 gtk_input_dialog_set_key (GtkInputKeyInfo *key,
648                           guint keyval, GdkModifierType modifiers)
649 {
650   GString *str;
651   gchar chars[2];
652
653   if (keyval)
654     {
655       str = g_string_new (NULL);
656       
657       if (modifiers & GDK_SHIFT_MASK)
658         g_string_append (str, "Shift+");
659       if (modifiers & GDK_CONTROL_MASK)
660         g_string_append (str, "Ctrl+");
661       if (modifiers & GDK_MOD1_MASK)
662         g_string_append (str, "Alt+");
663       
664       if ((keyval >= 0x20) && (keyval <= 0xFF))
665         {
666           chars[0] = keyval;
667           chars[1] = 0;
668           g_string_append (str, chars);
669         }
670       else
671         g_string_append (str, _("(unknown)"));
672       gtk_entry_set_text (GTK_ENTRY(key->entry), str->str);
673
674       g_string_free (str, TRUE);
675     }
676   else
677     {
678       gtk_entry_set_text (GTK_ENTRY(key->entry), _("(disabled)"));
679     }
680 }
681
682 static gboolean
683 gtk_input_dialog_key_press (GtkWidget *widget, 
684                             GdkEventKey *event,
685                             GtkInputKeyInfo *key)
686 {
687   if (!key->inputd->current_device)
688     return FALSE;
689   
690   gtk_input_dialog_set_key (key, event->keyval, event->state & 0xFF);
691   gdk_device_set_key (key->inputd->current_device, key->index, 
692                       event->keyval, event->state & 0xFF);
693
694   g_signal_stop_emission_by_name (widget, "key_press_event");
695   
696   return TRUE;
697 }
698
699 static void 
700 gtk_input_dialog_destroy_key (GtkWidget *widget, GtkInputKeyInfo *key)
701 {
702   g_free (key);
703 }
704
705 static void
706 gtk_input_dialog_fill_keys(GtkInputDialog *inputd, GdkDevice *info)
707 {
708   int i;
709   GtkWidget *label;
710   GtkWidget *button;
711
712   char buffer[32];
713   
714   /* remove all the old items */
715   if (inputd->keys_list)
716     {
717       gtk_widget_hide (inputd->keys_list);      /* suppress resizes (or get warnings) */
718       gtk_widget_destroy (inputd->keys_list);
719     }
720
721   inputd->keys_list = gtk_table_new (info->num_keys, 3, FALSE);
722   gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (inputd->keys_listbox), 
723                                          inputd->keys_list);
724   gtk_widget_show (inputd->keys_list);
725
726   gtk_widget_realize (inputd->keys_list);
727   gdk_window_set_background (inputd->keys_list->window,
728                              &inputd->keys_list->style->base[GTK_STATE_NORMAL]);
729
730   for (i=0;i<info->num_keys;i++)
731     {
732       GtkInputKeyInfo *key = g_new (GtkInputKeyInfo, 1);
733       key->index = i;
734       key->inputd = inputd;
735
736       /* create the label */
737
738       g_snprintf (buffer, sizeof (buffer), "%d", i+1);
739       label = gtk_label_new (buffer);
740       gtk_table_attach (GTK_TABLE (inputd->keys_list), label, 0, 1, i, i+1, 
741                         0, 0, 2, 2);
742       gtk_widget_show (label);
743
744       /* the entry */
745
746       key->entry = gtk_entry_new ();
747       gtk_table_attach (GTK_TABLE (inputd->keys_list), key->entry, 1, 2, i, i+1, 
748                         GTK_EXPAND | GTK_FILL , 0, 2, 2);
749       gtk_widget_show (key->entry);
750
751       g_signal_connect (key->entry, "key_press_event",
752                         G_CALLBACK (gtk_input_dialog_key_press), key);
753       g_signal_connect (key->entry, "destroy",
754                         G_CALLBACK (gtk_input_dialog_destroy_key), key);
755       
756       /* and clear button */
757
758       button = gtk_button_new_with_label (_("clear"));
759       gtk_table_attach (GTK_TABLE (inputd->keys_list), button, 2, 3, i, i+1, 
760                         0, 0, 2, 2);
761       gtk_widget_show (button);
762
763       g_signal_connect (button, "clicked",
764                         G_CALLBACK (gtk_input_dialog_clear_key), key);
765
766       gtk_input_dialog_set_key (key, info->keys[i].keyval,
767                                 info->keys[i].modifiers);
768     }
769 }