Forward keyboard events to current window from shell
[modest] / src / gtk / modest-shell.c
1 /* Copyright (c) 2009, Nokia Corporation
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  * * Redistributions of source code must retain the above copyright
9  *   notice, this list of conditions and the following disclaimer.
10  * * Redistributions in binary form must reproduce the above copyright
11  *   notice, this list of conditions and the following disclaimer in the
12  *   documentation and/or other materials provided with the distribution.
13  * * Neither the name of the Nokia Corporation nor the names of its
14  *   contributors may be used to endorse or promote products derived from
15  *   this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
18  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
20  * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
21  * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
24  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
25  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
26  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29
30 #include <string.h>
31 #include <modest-shell.h>
32 #include <modest-shell-window.h>
33 #include <modest-icon-names.h>
34 #include <modest-ui-actions.h>
35
36 /* 'private'/'protected' functions */
37 static void modest_shell_class_init (ModestShellClass *klass);
38 static void modest_shell_instance_init (ModestShell *obj);
39 static void modest_shell_finalize   (GObject *obj);
40
41 static void update_title (ModestShell *self);
42
43 static void on_back_button_clicked (GtkToolButton *button, ModestShell *self);
44 static void on_title_button_clicked (GtkToolButton *button, ModestShell *self);
45 static void on_new_msg_button_clicked (GtkToolButton *button, ModestShell *self);
46 static void on_style_set (GtkWidget *widget, GtkStyle *old_style, ModestShell *shell);
47 static gboolean on_key_pressed (GtkWidget *widget, GdkEventKey *event, ModestShell *shell);
48
49
50 typedef struct _ModestShellPrivate ModestShellPrivate;
51 struct _ModestShellPrivate {
52         GtkWidget *main_vbox;
53         GtkWidget *notebook;
54         GtkWidget *top_toolbar;
55         GtkToolItem *new_message_button;
56         GtkToolItem *back_button;
57         GtkToolItem *title_button;
58         GtkWidget *title_label;
59         GtkWidget *subtitle_label;
60
61         GtkWidget *progress_icon;
62         GdkPixbuf **progress_frames;
63         gint next_frame;
64         guint progress_timeout_id;
65 };
66 #define MODEST_SHELL_GET_PRIVATE(o)      (G_TYPE_INSTANCE_GET_PRIVATE((o), \
67                                                                       MODEST_TYPE_SHELL, \
68                                                                       ModestShellPrivate))
69 /* globals */
70 static GObjectClass *parent_class = NULL;
71
72 GType
73 modest_shell_get_type (void)
74 {
75         static GType my_type = 0;
76         if (!my_type) {
77                 static const GTypeInfo my_info = {
78                         sizeof(ModestShellClass),
79                         NULL,           /* base init */
80                         NULL,           /* base finalize */
81                         (GClassInitFunc) modest_shell_class_init,
82                         NULL,           /* class finalize */
83                         NULL,           /* class data */
84                         sizeof(ModestShell),
85                         1,              /* n_preallocs */
86                         (GInstanceInitFunc) modest_shell_instance_init,
87                         NULL
88                 };
89                 my_type = g_type_register_static (GTK_TYPE_WINDOW,
90                                                   "ModestShell",
91                                                   &my_info, 0);
92         }
93         return my_type;
94 }
95
96 static void
97 modest_shell_class_init (ModestShellClass *klass)
98 {
99         GObjectClass *gobject_class;
100
101         gobject_class = (GObjectClass*) klass;
102
103         parent_class            = g_type_class_peek_parent (klass);
104         gobject_class->finalize = modest_shell_finalize;
105
106         g_type_class_add_private (gobject_class, sizeof(ModestShellPrivate));
107
108 }
109
110 static void
111 modest_shell_instance_init (ModestShell *obj)
112 {
113         ModestShellPrivate *priv;
114         GtkWidget *title_vbox;
115         GtkWidget *title_arrow;
116         GtkWidget *new_message_icon;
117         GtkToolItem *separator_toolitem;
118         GtkWidget *top_hbox;
119         GtkWidget *separator;
120
121         priv = MODEST_SHELL_GET_PRIVATE(obj);
122         priv->progress_frames = g_malloc0 (sizeof(GdkPixbuf *)*31);
123         priv->progress_timeout_id = 0;
124         priv->next_frame = 0;
125
126         priv->main_vbox = gtk_vbox_new (FALSE, 0);
127         gtk_widget_show (priv->main_vbox);
128
129         top_hbox = gtk_hbox_new (FALSE, 0);
130         gtk_widget_show (top_hbox);
131         gtk_box_pack_start (GTK_BOX (priv->main_vbox), top_hbox, FALSE, FALSE, 0);
132
133         separator = gtk_hseparator_new ();
134         gtk_widget_show (separator);
135         gtk_box_pack_start (GTK_BOX (priv->main_vbox), separator, FALSE, FALSE, 0);
136
137         priv->top_toolbar = gtk_toolbar_new ();
138         gtk_toolbar_set_style (GTK_TOOLBAR (priv->top_toolbar), GTK_TOOLBAR_BOTH_HORIZ);
139         gtk_toolbar_set_show_arrow (GTK_TOOLBAR (priv->top_toolbar), FALSE);
140         gtk_widget_show (priv->top_toolbar);
141         gtk_box_pack_start (GTK_BOX (top_hbox), priv->top_toolbar, TRUE, TRUE, 0);
142
143         priv->progress_icon = gtk_image_new ();
144         gtk_widget_show (priv->progress_icon);
145         gtk_box_pack_start (GTK_BOX (top_hbox), priv->progress_icon, FALSE, FALSE, 0);
146
147         new_message_icon = gtk_image_new_from_icon_name (MODEST_TOOLBAR_ICON_NEW_MAIL, GTK_ICON_SIZE_LARGE_TOOLBAR);
148         gtk_widget_show (new_message_icon);
149         priv->new_message_button = gtk_tool_button_new (new_message_icon, _("mcen_va_new_email"));
150         g_object_set (priv->new_message_button, "is-important", TRUE, NULL);
151         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->new_message_button, -1);
152         gtk_widget_show (GTK_WIDGET (priv->new_message_button));
153         g_signal_connect (G_OBJECT (priv->new_message_button), "clicked", G_CALLBACK (on_new_msg_button_clicked), obj);
154
155         priv->back_button = gtk_tool_button_new_from_stock (GTK_STOCK_GO_BACK);
156         g_object_set (priv->back_button, "is-important", TRUE, NULL);
157         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->back_button, -1);
158         gtk_widget_show (GTK_WIDGET (priv->back_button));
159         g_signal_connect (G_OBJECT (priv->back_button), "clicked", G_CALLBACK (on_back_button_clicked), obj);
160
161         separator_toolitem = gtk_separator_tool_item_new ();
162         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), separator_toolitem, -1);
163         gtk_widget_show (GTK_WIDGET (separator_toolitem));
164
165         title_vbox = gtk_vbox_new (FALSE, 0);
166         priv->title_label = gtk_label_new (NULL);
167         gtk_label_set_ellipsize (GTK_LABEL (priv->title_label), PANGO_ELLIPSIZE_END);
168         gtk_misc_set_alignment (GTK_MISC (priv->title_label), 0.0, 1.0);
169         priv->subtitle_label = gtk_label_new (NULL);
170         gtk_label_set_ellipsize (GTK_LABEL (priv->subtitle_label), PANGO_ELLIPSIZE_START);
171         gtk_misc_set_alignment (GTK_MISC (priv->subtitle_label), 0.0, 0.0);
172         gtk_widget_show (priv->title_label);
173         gtk_widget_show (priv->subtitle_label);
174         gtk_box_pack_start (GTK_BOX (title_vbox), priv->title_label, TRUE, TRUE, 0);
175         gtk_box_pack_start (GTK_BOX (title_vbox), priv->subtitle_label, FALSE, FALSE, 0);
176         gtk_widget_show (title_vbox);
177
178         priv->title_button = gtk_tool_button_new (NULL, NULL);
179         gtk_widget_show (GTK_WIDGET (priv->title_button));
180         title_arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE);
181         gtk_widget_show (title_arrow);
182         gtk_tool_button_set_icon_widget (GTK_TOOL_BUTTON (priv->title_button), title_arrow);
183         gtk_tool_button_set_label_widget (GTK_TOOL_BUTTON (priv->title_button), title_vbox);
184         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->title_button, -1);
185         gtk_container_child_set (GTK_CONTAINER (priv->top_toolbar), GTK_WIDGET (priv->title_button), "expand", TRUE, NULL);
186         g_object_set (priv->title_button, "is-important", TRUE, NULL);
187         g_signal_connect (G_OBJECT (priv->title_button), "clicked", G_CALLBACK (on_title_button_clicked), obj);
188
189         priv->notebook = gtk_notebook_new ();
190         gtk_notebook_set_show_tabs ((GtkNotebook *)priv->notebook, FALSE);
191         gtk_notebook_set_show_border ((GtkNotebook *)priv->notebook, FALSE);
192         gtk_widget_show (priv->notebook);
193         gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->notebook, TRUE, TRUE, 0);
194         gtk_container_add (GTK_CONTAINER (obj), priv->main_vbox);
195
196         g_signal_connect (G_OBJECT (obj), "style-set", G_CALLBACK (on_style_set), obj);
197
198         guint accel_key;
199         GdkModifierType accel_mods;
200         GtkAccelGroup *accel_group;
201         accel_group = gtk_accel_group_new ();
202         gtk_accelerator_parse ("<Control>n", &accel_key, &accel_mods);
203         gtk_widget_add_accelerator (GTK_WIDGET (priv->new_message_button), "clicked", accel_group,
204                                     accel_key, accel_mods, 0);
205         gtk_accelerator_parse ("Esc", &accel_key, &accel_mods);
206         gtk_widget_add_accelerator (GTK_WIDGET (priv->back_button), "clicked", accel_group,
207                                     accel_key, accel_mods, 0);
208         gtk_accelerator_parse ("F10", &accel_key, &accel_mods);
209         gtk_widget_add_accelerator (GTK_WIDGET (priv->title_button), "clicked", accel_group,
210                                     accel_key, accel_mods, 0);
211         gtk_window_add_accel_group (GTK_WINDOW (obj), accel_group);
212
213         g_signal_connect (G_OBJECT (obj), 
214                           "key-press-event", 
215                           G_CALLBACK (on_key_pressed), obj);
216
217 }
218
219 static void
220 modest_shell_finalize (GObject *obj)
221 {
222         ModestShellPrivate *priv;
223         int n;
224
225         priv = MODEST_SHELL_GET_PRIVATE (obj);
226
227         if (priv->progress_timeout_id) {
228                 g_source_remove (priv->progress_timeout_id);
229         }
230         for (n = 0; n < 31; n++) {
231                 if (priv->progress_frames[n]) {
232                         g_object_unref (priv->progress_frames[n]);
233                 }
234         }
235         g_free (priv->progress_frames);
236
237         G_OBJECT_CLASS(parent_class)->finalize (obj);
238 }
239
240 GtkWidget*
241 modest_shell_new (void)
242 {
243         return (GtkWidget *) g_object_new(MODEST_TYPE_SHELL, NULL);
244 }
245
246 ModestWindow *
247 modest_shell_peek_window (ModestShell *shell)
248 {
249         ModestShellPrivate *priv;
250         gint count;
251
252         priv = MODEST_SHELL_GET_PRIVATE (shell);
253         count = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
254
255         if (count > 0) {
256                 return (ModestWindow *) gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), count - 1);
257         } else {
258                 return NULL;
259         }
260 }
261
262 gboolean
263 modest_shell_delete_window (ModestShell *shell, ModestWindow *window)
264 {
265         ModestShellPrivate *priv;
266         gboolean ret_value;
267
268         priv = MODEST_SHELL_GET_PRIVATE (shell);
269         g_signal_emit_by_name (G_OBJECT (window), "delete-event", NULL, &ret_value);
270         if (ret_value == FALSE) {
271                 gint page_num;
272                 
273                 page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window));
274                 if (page_num != -1) {
275                         gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), page_num);
276                 }
277         }
278
279         update_title (shell);
280
281         return ret_value;
282 }
283
284 void
285 modest_shell_add_window (ModestShell *shell, ModestWindow *window)
286 {
287         ModestShellPrivate *priv;
288
289         priv = MODEST_SHELL_GET_PRIVATE (shell);
290         gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window), NULL);
291         gtk_widget_show (GTK_WIDGET (window));
292         gtk_notebook_set_current_page (GTK_NOTEBOOK (priv->notebook), -1);
293         modest_shell_window_set_shell (MODEST_SHELL_WINDOW (window), shell);
294         update_title (shell);
295 }
296
297 gint
298 modest_shell_count_windows (ModestShell *shell)
299 {
300         ModestShellPrivate *priv;
301
302         priv = MODEST_SHELL_GET_PRIVATE (shell);
303
304         return gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
305 }
306
307 void
308 modest_shell_set_title (ModestShell *shell, ModestWindow *window, const gchar *title)
309 {
310         ModestShellPrivate *priv;
311
312         priv = MODEST_SHELL_GET_PRIVATE (shell);
313
314         gtk_notebook_set_tab_label_text (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window), title);
315
316         update_title (shell);
317 }
318
319 static void
320 show_next_frame (ModestShell *shell)
321 {
322         ModestShellPrivate *priv;
323
324         priv = MODEST_SHELL_GET_PRIVATE (shell);
325
326         gtk_image_set_from_pixbuf (GTK_IMAGE (priv->progress_icon), priv->progress_frames[priv->next_frame]);
327
328         priv->next_frame++;
329         if (priv->next_frame >= 31)
330                 priv->next_frame = 0;
331 }
332
333 static gboolean
334 on_progress_timeout (ModestShell *shell)
335 {
336         show_next_frame (shell);
337         return TRUE;
338 }
339
340 void
341 modest_shell_show_progress (ModestShell *shell, ModestWindow *window, gboolean show)
342 {
343         ModestShellPrivate *priv;
344
345         priv = MODEST_SHELL_GET_PRIVATE (shell);
346
347         if (show) {
348                 if (priv->progress_timeout_id == 0) {
349                         priv->progress_timeout_id = g_timeout_add (100, (GSourceFunc) on_progress_timeout, shell);
350                         show_next_frame (shell);
351                 }
352                 gtk_widget_show (priv->progress_icon);
353         } else {
354                 if (priv->progress_timeout_id) {
355                         g_source_remove (priv->progress_timeout_id);
356                         priv->progress_timeout_id = 0;
357                 }
358                 gtk_widget_hide (priv->progress_icon);
359         }
360 }
361
362 static void
363 update_title (ModestShell *self)
364 {
365         gint n_pages, i;
366         ModestShellPrivate *priv;
367         GtkWidget *child;
368         GString *title_buffer;
369         GString *subtitle_buffer;
370         const gchar *tab_label_text;
371
372         priv = MODEST_SHELL_GET_PRIVATE (self);
373
374         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
375         if (n_pages == 0) {
376                 gtk_label_set_text (GTK_LABEL (priv->title_label), "");
377                 gtk_label_set_text (GTK_LABEL (priv->subtitle_label), "");
378                 return;
379         }
380
381         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), n_pages - 1);
382         title_buffer = g_string_new ("");
383         title_buffer = g_string_append (title_buffer, "<b>");
384         tab_label_text = gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child);
385         if (tab_label_text)
386                 title_buffer = g_string_append (title_buffer, tab_label_text);
387         title_buffer = g_string_append (title_buffer, "</b>");
388         gtk_label_set_markup (GTK_LABEL (priv->title_label), 
389                               title_buffer->str);
390         g_string_free (title_buffer, TRUE);
391
392         subtitle_buffer = g_string_new ("");
393         subtitle_buffer = g_string_append (subtitle_buffer, "<small>");
394         for (i = 0; i < n_pages - 1; i++) {
395         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), i);
396                 if (i != 0) {
397                         subtitle_buffer = g_string_append (subtitle_buffer, " / ");
398                 }
399                 subtitle_buffer = g_string_append (subtitle_buffer,
400                                                    gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child));
401         }
402         subtitle_buffer = g_string_append (subtitle_buffer, "</small>");
403         gtk_label_set_markup (GTK_LABEL (priv->subtitle_label), 
404                               subtitle_buffer->str);
405         g_string_free (subtitle_buffer, TRUE);
406 }
407
408 static void
409 on_back_button_clicked (GtkToolButton *button, ModestShell *self)
410 {
411         ModestShellPrivate *priv;
412         gint n_pages;
413         gboolean delete_event_retval;
414         GtkWidget *child;
415
416         priv = MODEST_SHELL_GET_PRIVATE (self);
417
418         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
419         if (n_pages < 1)
420                 return;
421
422         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
423         g_signal_emit_by_name (G_OBJECT (child), "delete-event", NULL, &delete_event_retval);
424
425         if (!delete_event_retval) {
426                 update_title (self);
427         }
428 }
429
430 static void
431 menu_position_cb (GtkMenu *menu,
432                   gint *x,
433                   gint *y,
434                   gboolean *push_in,
435                   ModestShell *self)
436 {
437         ModestShellPrivate *priv;
438         GtkAllocation *alloc;
439         GdkWindow *parent_window;
440         gint pos_x, pos_y;
441
442         priv = MODEST_SHELL_GET_PRIVATE (self);
443
444         alloc = &(GTK_WIDGET (priv->title_button)->allocation);
445         parent_window = gtk_widget_get_parent_window (GTK_WIDGET (priv->title_button));
446         gdk_window_get_position (parent_window, &pos_x, &pos_y);
447         *x = pos_x + alloc->x;
448         *y = pos_y + alloc->y + alloc->height;
449         *push_in = TRUE;
450         
451 }
452
453 static void
454 on_title_button_clicked (GtkToolButton *button, ModestShell *self)
455 {
456         ModestShellPrivate *priv;
457         gint n_pages;
458         GtkWidget *child;
459         GtkWidget *menu;
460
461         priv = MODEST_SHELL_GET_PRIVATE (self);
462
463         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
464         if (n_pages < 1)
465                 return;
466
467         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
468         menu = modest_shell_window_get_menu (MODEST_SHELL_WINDOW (child));
469
470         if (menu) {
471                 gtk_menu_popup (GTK_MENU (menu), NULL, NULL, 
472                                 (GtkMenuPositionFunc) menu_position_cb, (gpointer) self,
473                                 1, gtk_get_current_event_time ());
474         }
475 }
476
477 static void
478 on_new_msg_button_clicked (GtkToolButton *button, ModestShell *self)
479 {
480         ModestShellPrivate *priv;
481         gint n_pages;
482         GtkWidget *child;
483
484         priv = MODEST_SHELL_GET_PRIVATE (self);
485
486         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
487         if (n_pages < 1)
488                 return;
489
490         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
491
492         modest_ui_actions_on_new_msg (NULL, MODEST_WINDOW (child));
493 }
494
495 static void
496 on_style_set (GtkWidget *widget,
497               GtkStyle *old_style,
498               ModestShell *self)
499 {
500         ModestShellPrivate *priv;
501         gint icon_w, icon_h;
502         GdkPixbuf *progress_pixbuf;
503         int n;
504
505         priv = MODEST_SHELL_GET_PRIVATE (self);
506
507         if (!gtk_icon_size_lookup (GTK_ICON_SIZE_LARGE_TOOLBAR, &icon_w, &icon_h))
508                 return;
509         progress_pixbuf = gtk_icon_theme_load_icon (gtk_icon_theme_get_default (), "process-working", icon_w, 0, NULL);
510
511         for (n = 0; n < 31; n++) {
512                 if (priv->progress_frames[n] != NULL) {
513                         g_object_unref (priv->progress_frames[n]);
514                 }
515                 priv->progress_frames[n] = NULL;
516         }
517
518         if (progress_pixbuf) {
519                 gint max_x, max_y;
520                 gint i, j;
521
522                 icon_w = gdk_pixbuf_get_width (progress_pixbuf) / 8;
523
524                 n = 0;
525                 max_x = 8;
526                 max_y = 4;
527                 for (i = 0; i < 4; i++) {
528                         for (j = 0; j < 8; j++) {
529                                         GdkPixbuf *frame;
530
531                                         if ((i == 0) && (j == 0))
532                                                 continue;
533                                         frame = gdk_pixbuf_new_subpixbuf  (progress_pixbuf,
534                                                                            j*icon_w, i*icon_w,
535                                                                            icon_w, icon_w);
536                                         priv->progress_frames[n] = frame;
537                                         n++;
538                                 }
539                         }
540                 g_object_unref (progress_pixbuf);
541         }
542
543 }
544
545 static gboolean
546 on_key_pressed (GtkWidget *widget,
547                 GdkEventKey *event,
548                 ModestShell *shell)
549 {
550         ModestShellPrivate *priv;
551         gboolean retval;
552         GtkWidget *current_window;
553
554         priv = MODEST_SHELL_GET_PRIVATE (shell);
555
556         current_window = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
557
558         g_signal_emit_by_name (current_window, "key-press-event", event, &retval);
559
560         return retval;
561         
562 }