Adding a patch by Santtu Lakkala that add an 'input' signal to the code dialog that...
[hildon] / src / hildon-code-dialog.c
1 /*
2  * This file is a part of hildon
3  *
4  * Copyright (C) 2006 Nokia Corporation, all rights reserved.
5  *
6  * Contact: Michael Dominic Kostrzewa <michael.kostrzewa@nokia.com>
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public License
10  * as published by the Free Software Foundation; version 2.1 of
11  * the License.
12  *
13  * This library is distributed in the hope that it will be useful, but
14  * WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with this library; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
21  * 02110-1301 USA
22  *
23  */
24
25 /**
26  * SECTION:hildon-code-dialog
27  * @short_description: A keypad-like widget used to enter pincodes.
28  *
29  * #HildonCodeDialog displays a keypad that can be used to enter 
30  * numerical pin codes or lock codes. It emits a 'input' signal each time 
31  * an input action is performed on the dialog.
32  *
33  */
34
35 /* FIXME We need property access in this widget */
36
37 #ifdef                                          HAVE_CONFIG_H
38 #include                                        <config.h>
39 #endif
40
41 #include                                        "hildon-code-dialog.h"
42 #include                                        "hildon-defines.h"
43 #include                                        "hildon-banner.h"
44
45 #include                                        <gdk/gdkkeysyms.h>
46 #include                                        <gtk/gtkbutton.h>
47 #include                                        <gtk/gtkentry.h>
48 #include                                        <gtk/gtkicontheme.h>
49 #include                                        <gtk/gtktable.h>
50 #include                                        <gtk/gtkvbox.h>
51 #include                                        <gtk/gtkbbox.h>
52 #include                                        <gtk/gtklabel.h>
53 #include                                        <gtk/gtkalignment.h>
54 #include                                        <libintl.h>
55 #include                                        "hildon-code-dialog-private.h"
56
57 #define                                         HEIGHT (38-HILDON_MARGIN_DEFAULT)
58
59 #define                                         WIDTH  (60-HILDON_MARGIN_DEFAULT)
60
61 #define                                         BACKSPACE_ICON "qgn_calculator_backspace"
62
63 #define                                         _(String) \
64                                                 dgettext("hildon-libs", String)
65
66 #define                                         c_(String) \
67                                                 dgettext("hildon-common-strings", String)
68
69 #define                                         DEVICELOCK_OK _("secu_enter_lock_code_dialog_ok")
70
71 #define                                         DEVICELOCK_CANCEL _("secu_enter_lock_code_dialog_cancel")
72
73 #define                                         DEVICELOCK_TITLE _("secu_application_title")
74
75 #define                                         DEVICELOCK_MAX_CHAR_REACHED c_("ckdg_ib_maximum_characters_reached")
76         
77 #define                                         MAX_PINCODE_LEN (10)
78
79 static void 
80 hildon_code_dialog_class_init                   (HildonCodeDialogClass *cd_class);
81
82 static void 
83 hildon_code_dialog_init                         (HildonCodeDialog *self);
84
85 static void 
86 hildon_code_dialog_button_clicked               (GtkButton *buttonm,
87                                                  gpointer user_data);
88
89 static void 
90 hildon_code_dialog_insert_text                  (GtkEditable *editable,
91                                                  gchar *new_text,
92                                                  gint new_text_length,
93                                                  gint *position,
94                                                  gpointer user_data);
95
96 static gboolean 
97 hildon_code_dialog_key_press_event              (GtkWidget *widget,
98                                                  GdkEventKey *event,
99                                                  gpointer user_data);
100
101 static void 
102 hildon_code_dialog_real_input                   (HildonCodeDialog *dialog);
103
104 static void
105 hildon_code_dialog_input                        (HildonCodeDialog *dialog);
106
107 static GtkDialogClass*                          parent_class = NULL;
108
109 static guint                                    input_signal;
110
111 /**
112  * hildon_code_dialog_get_type:
113  *
114  * Initializes and returns the type of a hildon code dialog.
115  *
116  * @Returns: GType of #HildonCodeDialog
117  */
118 GType G_GNUC_CONST
119 hildon_code_dialog_get_type                     (void)
120 {
121     static GType type = 0;
122
123     if (!type)
124     {
125         static const GTypeInfo info = 
126         {
127             sizeof (HildonCodeDialogClass),
128             NULL, /* base_init */
129             NULL, /* base_finalize */
130             (GClassInitFunc) hildon_code_dialog_class_init,
131             NULL,       /* class_finalize */
132             NULL,       /* class_data */
133             sizeof (HildonCodeDialog),
134             0,  /* n_preallocs */
135             (GInstanceInitFunc) hildon_code_dialog_init
136         };
137         type = g_type_register_static (GTK_TYPE_DIALOG,
138                 "HildonCodeDialog", &info, 0);
139     }
140     return type;
141 }
142
143 static void
144 hildon_code_dialog_class_init                   (HildonCodeDialogClass *cd_class)
145 {
146     parent_class = GTK_DIALOG_CLASS (g_type_class_peek_parent (cd_class));
147     g_type_class_add_private (cd_class, sizeof (HildonCodeDialogPrivate));
148
149     cd_class->input = hildon_code_dialog_real_input;
150
151     /* FIXME Document this signal! */
152     input_signal = g_signal_new("input",
153                                 HILDON_TYPE_CODE_DIALOG,
154                                 G_SIGNAL_RUN_LAST,
155                                 G_STRUCT_OFFSET (HildonCodeDialogClass, input),
156                                 NULL, NULL,
157                                 g_cclosure_marshal_VOID__VOID,
158                                 G_TYPE_NONE,
159                                 0);
160 }
161
162 static void 
163 hildon_code_dialog_init                         (HildonCodeDialog *dialog)
164 {
165     HildonCodeDialogPrivate *priv;
166     gint i, x, y;
167     GtkWidget *dialog_vbox1 = NULL;
168     GtkWidget *table = NULL;
169     GtkWidget *alignment = NULL;
170     GtkWidget *vbox1 = NULL;
171     GtkWidget *image1 = NULL;
172     GtkWidget *dialog_action_area1 = NULL;
173     GdkGeometry hints;
174     GtkWidget *okButton;
175     GtkWidget *cancelButton;
176
177     priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
178     g_assert (priv);
179     
180     const gchar* numstrs[10] = {
181         "0","1","2","3","4","5","6","7","8","9"
182     };
183
184     GdkPixbuf* pixbuf = NULL;
185     GtkIconTheme* icon_theme = NULL;
186     GtkIconInfo *icon_info = NULL;
187     gint base_size = 0;
188
189     /* Set default title */
190     gtk_window_set_title (GTK_WINDOW (dialog), DEVICELOCK_TITLE);
191
192     gtk_window_set_type_hint (GTK_WINDOW (dialog), GDK_WINDOW_TYPE_HINT_DIALOG);
193
194     hints.min_width  = -1;
195     hints.min_height = -1;
196     hints.max_width  = -1;
197     hints.max_height = -1;
198
199     gtk_window_set_geometry_hints (GTK_WINDOW (dialog), GTK_WIDGET (dialog), &hints,
200             GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE);
201
202     table = gtk_table_new (4, 3, FALSE);
203     gtk_table_set_row_spacings (GTK_TABLE (table), HILDON_MARGIN_DEFAULT );
204     gtk_table_set_col_spacings (GTK_TABLE (table), HILDON_MARGIN_DEFAULT );
205
206     dialog_vbox1 = GTK_DIALOG (dialog)->vbox;
207     vbox1 = gtk_vbox_new (FALSE, 0);
208     gtk_box_set_spacing (GTK_BOX (vbox1), HILDON_MARGIN_DOUBLE);
209
210     priv->help_text = gtk_label_new ("");
211     alignment = gtk_alignment_new (0.5,0,1,1);
212     gtk_container_add (GTK_CONTAINER (alignment), priv->help_text);
213
214     priv->entry = gtk_entry_new ();
215     
216     GTK_WIDGET_UNSET_FLAGS (GTK_WIDGET (priv->entry), GTK_CAN_FOCUS);
217     gtk_entry_set_invisible_char (GTK_ENTRY (priv->entry), g_utf8_get_char ("*"));
218     gtk_entry_set_alignment (GTK_ENTRY (priv->entry), 1.0);
219
220     gtk_editable_set_editable (GTK_EDITABLE (priv->entry),FALSE);
221     gtk_entry_set_visibility (GTK_ENTRY (priv->entry), FALSE);
222
223     gtk_box_pack_start (GTK_BOX (vbox1), alignment, TRUE,FALSE,0);
224     gtk_box_pack_start (GTK_BOX (vbox1), priv->entry, TRUE,FALSE,0);
225     gtk_box_pack_start (GTK_BOX (vbox1), table, FALSE,TRUE,0);
226
227     gtk_box_pack_start (GTK_BOX (dialog_vbox1), vbox1, FALSE,TRUE,0);
228
229     for(i = 1;i <= 3; i++) {
230         priv->buttons[0][i-1] = gtk_button_new_with_mnemonic (numstrs[i]);
231         gtk_widget_set_size_request (priv->buttons[0][i-1], WIDTH, HEIGHT);
232         gtk_table_attach_defaults (GTK_TABLE (table), priv->buttons[0][i-1],
233                 i-1, i, 0, 1);
234     }
235
236     for(i = 4;i <= 6;i++) {
237         priv->buttons[1][i-4] = gtk_button_new_with_mnemonic (numstrs[i]);
238         gtk_widget_set_size_request (priv->buttons[1][i-4], WIDTH, HEIGHT);
239         gtk_table_attach_defaults (GTK_TABLE (table), priv->buttons[1][i-4],
240                 i-4, i-3, 1, 2);
241     }
242
243     for(i=7;i<=9;i++) {
244         priv->buttons[2][i-7] = gtk_button_new_with_mnemonic (numstrs[i]);
245         gtk_widget_set_size_request (priv->buttons[2][i-7], WIDTH, HEIGHT);
246         gtk_table_attach_defaults (GTK_TABLE (table), priv->buttons[2][i-7],
247                 i-7, i-6, 2, 3);
248     }
249
250     priv->buttons[3][0] = priv->buttons[3][1] = 
251         gtk_button_new_with_mnemonic (numstrs[0]);
252     gtk_widget_set_size_request (priv->buttons[3][0], WIDTH, HEIGHT);
253     gtk_table_attach (GTK_TABLE (table), priv->buttons[3][0],
254             0,2,3,4, (GtkAttachOptions) (GTK_FILL),
255             (GtkAttachOptions) (0), 0, 0);
256
257     priv->buttons[3][2] = gtk_button_new ();
258     gtk_widget_set_size_request (priv->buttons[3][2], WIDTH, HEIGHT);
259     gtk_table_attach_defaults (GTK_TABLE (table), priv->buttons[3][2],
260             2, 3, 3, 4);
261
262     icon_theme = gtk_icon_theme_get_default ();
263
264     icon_info = gtk_icon_theme_lookup_icon (icon_theme, BACKSPACE_ICON, 1,
265             GTK_ICON_LOOKUP_NO_SVG);
266     base_size = gtk_icon_info_get_base_size (icon_info);
267     gtk_icon_info_free (icon_info);
268     icon_info = NULL;
269     pixbuf = gtk_icon_theme_load_icon (icon_theme,
270             BACKSPACE_ICON, base_size,
271             GTK_ICON_LOOKUP_NO_SVG,
272             NULL);
273
274     image1 = gtk_image_new_from_pixbuf (pixbuf);
275     g_object_unref (G_OBJECT(pixbuf));
276     gtk_container_add (GTK_CONTAINER (priv->buttons[3][2]), image1);
277     dialog_action_area1 = GTK_DIALOG (dialog)->action_area;
278     gtk_button_box_set_layout (GTK_BUTTON_BOX (dialog_action_area1),
279             GTK_BUTTONBOX_END);
280
281     okButton = gtk_dialog_add_button (GTK_DIALOG (dialog) ,DEVICELOCK_OK,
282             GTK_RESPONSE_OK);
283     cancelButton =
284         gtk_dialog_add_button (GTK_DIALOG (dialog), DEVICELOCK_CANCEL,
285                 GTK_RESPONSE_CANCEL);
286     
287     gtk_widget_set_sensitive (okButton, FALSE);
288
289     priv->buttons[4][0] = priv->buttons[4][1] = okButton;
290     priv->buttons[4][2] = cancelButton;
291
292     /*
293        Connect signals.
294     */
295     g_signal_connect (G_OBJECT (priv->entry), "insert_text",
296             G_CALLBACK (hildon_code_dialog_insert_text), dialog);
297     
298     gtk_entry_set_max_length (GTK_ENTRY (priv->entry), MAX_PINCODE_LEN);
299
300     for (x = 0; x < 3; x++)
301     {
302         for (y = 0; y < 3; y++)
303         {
304             g_signal_connect (G_OBJECT (priv->buttons[x][y]), "clicked",
305                 G_CALLBACK (hildon_code_dialog_button_clicked), dialog);
306             g_signal_connect (G_OBJECT (priv->buttons[x][y]), "key-press-event",
307                 G_CALLBACK (hildon_code_dialog_key_press_event), dialog);
308         }
309     }
310             
311     g_signal_connect (G_OBJECT (priv->buttons[3][0]), "clicked",
312                 G_CALLBACK (hildon_code_dialog_button_clicked), dialog);
313     g_signal_connect (G_OBJECT (priv->buttons[3][0]), "key-press-event",
314                 G_CALLBACK (hildon_code_dialog_key_press_event), dialog);
315     
316     g_signal_connect (G_OBJECT (priv->buttons[3][2]), "clicked",
317                 G_CALLBACK (hildon_code_dialog_button_clicked), dialog);
318     g_signal_connect (G_OBJECT (priv->buttons[3][2]), "key-press-event",
319                 G_CALLBACK (hildon_code_dialog_key_press_event), dialog);
320         
321     g_signal_connect (G_OBJECT (okButton), "key-press-event",
322                 G_CALLBACK(hildon_code_dialog_key_press_event), dialog);
323     
324     g_signal_connect (G_OBJECT (cancelButton), "key-press-event",
325                 G_CALLBACK (hildon_code_dialog_key_press_event), dialog);
326
327     gtk_widget_show_all (GTK_WIDGET (GTK_DIALOG (dialog)->vbox));
328 }
329
330 static void 
331 hildon_code_dialog_button_clicked               (GtkButton *button, 
332                                                  gpointer user_data)
333 {
334     HildonCodeDialog *dialog = HILDON_CODE_DIALOG (user_data);
335     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
336     g_assert (priv);
337
338     const char *number = gtk_button_get_label (button);
339
340     if (number && *number )
341     {
342         gtk_entry_append_text (GTK_ENTRY (priv->entry), number);
343     }
344     else
345     {
346         /* Backspace */
347         gchar *text = g_strdup (gtk_entry_get_text (GTK_ENTRY (priv->entry)));
348         gchar *pos;
349         
350         pos = text;
351
352         while (*pos != '\0')
353         {
354             pos ++;
355         }
356
357         pos = g_utf8_find_prev_char (text, pos);
358
359         if (pos)
360         {
361             *pos=0;
362         }
363
364         gtk_entry_set_text (GTK_ENTRY (priv->entry), text);
365
366         if (*text == 0)
367         {
368             gtk_widget_set_sensitive (priv->buttons[4][0], FALSE);
369         }
370
371         g_free (text);
372     }
373 }
374
375 static void 
376 hildon_code_dialog_insert_text                  (GtkEditable *editable,
377                                                  gchar *new_text,
378                                                  gint new_text_length,
379                                                  gint *position,
380                                                  gpointer user_data)
381 {
382     HildonCodeDialog *dialog = HILDON_CODE_DIALOG (user_data);
383     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
384     gchar *text = g_strdup(gtk_entry_get_text (GTK_ENTRY (priv->entry)));
385     glong length = g_utf8_strlen (text, -1);
386     g_free (text);
387     g_assert (priv);
388
389     if (length == MAX_PINCODE_LEN)
390     {
391         hildon_banner_show_information (GTK_WIDGET (dialog),
392                                         NULL,
393                                         DEVICELOCK_MAX_CHAR_REACHED);
394     }
395
396     else if (! length)
397     { 
398         /* make the Ok button sensitive */
399         gtk_widget_set_sensitive (priv->buttons[4][0], TRUE);
400     }
401
402     hildon_code_dialog_input (dialog);
403 }
404
405 static gboolean 
406 hildon_code_dialog_key_press_event              (GtkWidget *widget,
407                                                  GdkEventKey *event,
408                                                  gpointer user_data)
409 {
410     HildonCodeDialog *dialog = HILDON_CODE_DIALOG (user_data);
411     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
412     GtkWidget *new_widget = widget;
413     gint x, y;
414     
415     g_assert (priv);
416
417     for (x = 0; x < 5; x++)
418     {
419         for (y = 0; y < 3; y++)
420         {
421             if (priv->buttons[x][y] == widget)
422                 goto found;
423         }
424     }
425     return FALSE;
426
427 found:
428
429     while (new_widget == widget)
430     {
431         switch (event->keyval)
432         {
433             case GDK_Up:
434                 x = (x+4)%5;
435                 break;
436
437             case GDK_Down:
438                 x = (x+1)%5;
439                 break;
440
441             case GDK_Left:
442                 y = (y+2)%3;
443                 break;
444
445             case GDK_Right:
446                 y = (y+1)%3;
447                 break;
448
449             default:
450                 return FALSE;
451         }
452                 
453         new_widget = priv->buttons[x][y];
454     }
455
456     gtk_widget_grab_focus (new_widget);
457
458     return TRUE;
459 }
460
461 /**
462  * hildon_code_dialog_new:
463  *
464  * Use this function to create a new HildonCodeDialog.
465  *
466  * Return value: A @HildonCodeDialog.
467  **/
468 GtkWidget*
469 hildon_code_dialog_new                          (void)
470 {
471     HildonCodeDialog *dialog = g_object_new (HILDON_TYPE_CODE_DIALOG, NULL);
472
473     return GTK_WIDGET (dialog);
474 }
475
476 /**
477  * hildon_code_dialog_get_code:
478  * @dialog: The #HildonCodeDialog from which to get the entered code
479  *
480  * Use this function to access the code entered by the user.
481  *
482  * Return value: The entered code.
483  **/
484 const gchar*
485 hildon_code_dialog_get_code                     (HildonCodeDialog *dialog)
486 {
487     g_return_val_if_fail (HILDON_IS_CODE_DIALOG (dialog), NULL);
488     
489     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
490     g_assert (priv);
491
492     return gtk_entry_get_text (GTK_ENTRY (priv->entry));
493 }
494
495 /**
496  * hildon_code_dialog_clear_clode:
497  * @dialog: The #HildonCodeDialog whose entry should be cleared:
498  *
499  * Use this function to clear the user entered code.
500  **/
501 void 
502 hildon_code_dialog_clear_code                   (HildonCodeDialog *dialog)
503 {
504     g_return_if_fail (HILDON_IS_CODE_DIALOG (dialog));
505
506     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
507     g_assert (priv);
508
509     gtk_entry_set_text (GTK_ENTRY (priv->entry), "");
510     gtk_widget_set_sensitive (priv->buttons[4][0], FALSE);
511 }
512
513 /**
514  * hildon_code_dialog_set_help_text:
515  * @dialog: The #HildonCodeDialog whose entry should be cleared:
516  * @text: The text to use in the help label.
517  *
518  * Use this function to set the text that will be displayd in the
519  * help label
520  **/
521 void 
522 hildon_code_dialog_set_help_text                (HildonCodeDialog *dialog, 
523                                                  const gchar *text)
524 {
525     g_return_if_fail (HILDON_IS_CODE_DIALOG (dialog));
526
527     HildonCodeDialogPrivate *priv = HILDON_CODE_DIALOG_GET_PRIVATE (dialog);
528     g_assert (priv);
529
530     gtk_label_set_text (GTK_LABEL (priv->help_text), text);
531 }
532
533 static void 
534 hildon_code_dialog_real_input                   (HildonCodeDialog *dialog)
535 {
536 }
537
538 static void 
539 hildon_code_dialog_input                        (HildonCodeDialog *dialog)
540 {
541     /* Emit the signal */
542     g_signal_emit (dialog, input_signal, 0);
543 }