Put shell menu below the button that launches it.
[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
47
48 typedef struct _ModestShellPrivate ModestShellPrivate;
49 struct _ModestShellPrivate {
50         GtkWidget *main_vbox;
51         GtkWidget *notebook;
52         GtkWidget *top_toolbar;
53         GtkToolItem *new_message_button;
54         GtkToolItem *back_button;
55         GtkToolItem *title_button;
56         GtkWidget *title_label;
57         GtkWidget *subtitle_label;
58 };
59 #define MODEST_SHELL_GET_PRIVATE(o)      (G_TYPE_INSTANCE_GET_PRIVATE((o), \
60                                                                       MODEST_TYPE_SHELL, \
61                                                                       ModestShellPrivate))
62 /* globals */
63 static GObjectClass *parent_class = NULL;
64
65 GType
66 modest_shell_get_type (void)
67 {
68         static GType my_type = 0;
69         if (!my_type) {
70                 static const GTypeInfo my_info = {
71                         sizeof(ModestShellClass),
72                         NULL,           /* base init */
73                         NULL,           /* base finalize */
74                         (GClassInitFunc) modest_shell_class_init,
75                         NULL,           /* class finalize */
76                         NULL,           /* class data */
77                         sizeof(ModestShell),
78                         1,              /* n_preallocs */
79                         (GInstanceInitFunc) modest_shell_instance_init,
80                         NULL
81                 };
82                 my_type = g_type_register_static (GTK_TYPE_WINDOW,
83                                                   "ModestShell",
84                                                   &my_info, 0);
85         }
86         return my_type;
87 }
88
89 static void
90 modest_shell_class_init (ModestShellClass *klass)
91 {
92         GObjectClass *gobject_class;
93
94         gobject_class = (GObjectClass*) klass;
95
96         parent_class            = g_type_class_peek_parent (klass);
97         gobject_class->finalize = modest_shell_finalize;
98
99         g_type_class_add_private (gobject_class, sizeof(ModestShellPrivate));
100
101 }
102
103 static void
104 modest_shell_instance_init (ModestShell *obj)
105 {
106         ModestShellPrivate *priv;
107         GtkWidget *title_vbox;
108         GtkWidget *new_message_icon;
109         GtkToolItem *separator_toolitem;
110
111         priv = MODEST_SHELL_GET_PRIVATE(obj);
112
113         priv->main_vbox = gtk_vbox_new (FALSE, 0);
114         gtk_widget_show (priv->main_vbox);
115
116         priv->top_toolbar = gtk_toolbar_new ();
117         gtk_toolbar_set_style (GTK_TOOLBAR (priv->top_toolbar), GTK_TOOLBAR_BOTH_HORIZ);
118         gtk_widget_show (priv->top_toolbar);
119         gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->top_toolbar, FALSE, FALSE, 0);
120
121         new_message_icon = gtk_image_new_from_icon_name (MODEST_TOOLBAR_ICON_NEW_MAIL, GTK_ICON_SIZE_LARGE_TOOLBAR);
122         gtk_widget_show (new_message_icon);
123         priv->new_message_button = gtk_tool_button_new (new_message_icon, _("mcen_va_new_email"));
124         g_object_set (priv->new_message_button, "is-important", TRUE, NULL);
125         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->new_message_button, -1);
126         gtk_widget_show (GTK_WIDGET (priv->new_message_button));
127         g_signal_connect (G_OBJECT (priv->new_message_button), "clicked", G_CALLBACK (on_new_msg_button_clicked), obj);
128
129         priv->back_button = gtk_tool_button_new_from_stock (GTK_STOCK_GO_BACK);
130         g_object_set (priv->back_button, "is-important", TRUE, NULL);
131         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->back_button, -1);
132         gtk_widget_show (GTK_WIDGET (priv->back_button));
133         g_signal_connect (G_OBJECT (priv->back_button), "clicked", G_CALLBACK (on_back_button_clicked), obj);
134
135         separator_toolitem = gtk_separator_tool_item_new ();
136         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), separator_toolitem, -1);
137         gtk_widget_show (GTK_WIDGET (separator_toolitem));
138
139         title_vbox = gtk_vbox_new (FALSE, 0);
140         priv->title_label = gtk_label_new (NULL);
141         gtk_misc_set_alignment (GTK_MISC (priv->title_label), 0.0, 1.0);
142         priv->subtitle_label = gtk_label_new (NULL);
143         gtk_misc_set_alignment (GTK_MISC (priv->subtitle_label), 0.0, 0.0);
144         gtk_widget_show (priv->title_label);
145         gtk_widget_show (priv->subtitle_label);
146         gtk_box_pack_start (GTK_BOX (title_vbox), priv->title_label, TRUE, TRUE, 0);
147         gtk_box_pack_start (GTK_BOX (title_vbox), priv->subtitle_label, FALSE, FALSE, 0);
148         gtk_widget_show (title_vbox);
149
150         priv->title_button = gtk_tool_button_new (NULL, NULL);
151         gtk_widget_show (GTK_WIDGET (priv->title_button));
152         gtk_tool_button_set_label_widget (GTK_TOOL_BUTTON (priv->title_button), title_vbox);
153         gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->title_button, -1);
154         gtk_container_child_set (GTK_CONTAINER (priv->top_toolbar), GTK_WIDGET (priv->title_button), "expand", TRUE, NULL);
155         g_object_set (priv->title_button, "is-important", TRUE, NULL);
156         g_signal_connect (G_OBJECT (priv->title_button), "clicked", G_CALLBACK (on_title_button_clicked), obj);
157
158         priv->notebook = gtk_notebook_new ();
159         gtk_notebook_set_show_tabs ((GtkNotebook *)priv->notebook, FALSE);
160         gtk_notebook_set_show_border ((GtkNotebook *)priv->notebook, FALSE);
161         gtk_widget_show (priv->notebook);
162         gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->notebook, TRUE, TRUE, 0);
163         gtk_container_add (GTK_CONTAINER (obj), priv->main_vbox);
164
165 }
166
167 static void
168 modest_shell_finalize (GObject *obj)
169 {
170         G_OBJECT_CLASS(parent_class)->finalize (obj);
171 }
172
173 GtkWidget*
174 modest_shell_new (void)
175 {
176         return (GtkWidget *) g_object_new(MODEST_TYPE_SHELL, NULL);
177 }
178
179 ModestWindow *
180 modest_shell_peek_window (ModestShell *shell)
181 {
182         ModestShellPrivate *priv;
183         gint count;
184
185         priv = MODEST_SHELL_GET_PRIVATE (shell);
186         count = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
187
188         if (count > 0) {
189                 return (ModestWindow *) gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), count - 1);
190         } else {
191                 return NULL;
192         }
193 }
194
195 gboolean
196 modest_shell_delete_window (ModestShell *shell, ModestWindow *window)
197 {
198         ModestShellPrivate *priv;
199         gboolean ret_value;
200
201         priv = MODEST_SHELL_GET_PRIVATE (shell);
202         g_signal_emit_by_name (G_OBJECT (window), "delete-event", NULL, &ret_value);
203         if (ret_value == FALSE) {
204                 gint page_num;
205                 
206                 page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window));
207                 if (page_num != -1) {
208                         gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), page_num);
209                 }
210         }
211
212         update_title (shell);
213
214         return ret_value;
215 }
216
217 void
218 modest_shell_add_window (ModestShell *shell, ModestWindow *window)
219 {
220         ModestShellPrivate *priv;
221
222         priv = MODEST_SHELL_GET_PRIVATE (shell);
223         gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window), NULL);
224         gtk_widget_show (GTK_WIDGET (window));
225         gtk_notebook_set_current_page (GTK_NOTEBOOK (priv->notebook), -1);
226         modest_shell_window_set_shell (MODEST_SHELL_WINDOW (window), shell);
227         update_title (shell);
228 }
229
230 gint
231 modest_shell_count_windows (ModestShell *shell)
232 {
233         ModestShellPrivate *priv;
234
235         priv = MODEST_SHELL_GET_PRIVATE (shell);
236
237         return gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
238 }
239
240 void
241 modest_shell_set_title (ModestShell *shell, ModestWindow *window, const gchar *title)
242 {
243         ModestShellPrivate *priv;
244
245         priv = MODEST_SHELL_GET_PRIVATE (shell);
246
247         gtk_notebook_set_tab_label_text (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window), title);
248
249         update_title (shell);
250 }
251
252 void
253 modest_shell_show_progress (ModestShell *shell, ModestWindow *window, gboolean show)
254 {
255 }
256
257 static void
258 update_title (ModestShell *self)
259 {
260         gint n_pages, i;
261         ModestShellPrivate *priv;
262         GtkWidget *child;
263         GString *title_buffer;
264         GString *subtitle_buffer;
265
266         priv = MODEST_SHELL_GET_PRIVATE (self);
267
268         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
269         if (n_pages == 0) {
270                 gtk_label_set_text (GTK_LABEL (priv->title_label), "");
271                 gtk_label_set_text (GTK_LABEL (priv->subtitle_label), "");
272                 return;
273         }
274
275         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), n_pages - 1);
276         title_buffer = g_string_new ("");
277         title_buffer = g_string_append (title_buffer, "<b>");
278         title_buffer = g_string_append (title_buffer, gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child));
279         title_buffer = g_string_append (title_buffer, "</b>");
280         gtk_label_set_markup (GTK_LABEL (priv->title_label), 
281                               title_buffer->str);
282         g_string_free (title_buffer, TRUE);
283
284         subtitle_buffer = g_string_new ("");
285         subtitle_buffer = g_string_append (subtitle_buffer, "<small>");
286         for (i = 0; i < n_pages - 1; i++) {
287         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), i);
288                 if (i != 0) {
289                         subtitle_buffer = g_string_append (subtitle_buffer, " / ");
290                 }
291                 subtitle_buffer = g_string_append (subtitle_buffer,
292                                                    gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child));
293         }
294         subtitle_buffer = g_string_append (subtitle_buffer, "</small>");
295         gtk_label_set_markup (GTK_LABEL (priv->subtitle_label), 
296                               subtitle_buffer->str);
297         g_string_free (subtitle_buffer, TRUE);
298 }
299
300 static void
301 on_back_button_clicked (GtkToolButton *button, ModestShell *self)
302 {
303         ModestShellPrivate *priv;
304         gint n_pages;
305         gboolean delete_event_retval;
306         GtkWidget *child;
307
308         priv = MODEST_SHELL_GET_PRIVATE (self);
309
310         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
311         if (n_pages < 1)
312                 return;
313
314         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
315         g_signal_emit_by_name (G_OBJECT (child), "delete-event", NULL, &delete_event_retval);
316
317         if (!delete_event_retval) {
318                 update_title (self);
319         }
320 }
321
322 static void
323 menu_position_cb (GtkMenu *menu,
324                   gint *x,
325                   gint *y,
326                   gboolean *push_in,
327                   ModestShell *self)
328 {
329         ModestShellPrivate *priv;
330         GtkAllocation *alloc;
331         GdkWindow *parent_window;
332         gint pos_x, pos_y;
333
334         priv = MODEST_SHELL_GET_PRIVATE (self);
335
336         alloc = &(GTK_WIDGET (priv->title_button)->allocation);
337         parent_window = gtk_widget_get_parent_window (GTK_WIDGET (priv->title_button));
338         gdk_window_get_position (parent_window, &pos_x, &pos_y);
339         *x = pos_x + alloc->x;
340         *y = pos_y + alloc->y + alloc->height;
341         *push_in = TRUE;
342         
343 }
344
345 static void
346 on_title_button_clicked (GtkToolButton *button, ModestShell *self)
347 {
348         ModestShellPrivate *priv;
349         gint n_pages;
350         GtkWidget *child;
351         GtkWidget *menu;
352
353         priv = MODEST_SHELL_GET_PRIVATE (self);
354
355         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
356         if (n_pages < 1)
357                 return;
358
359         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
360         menu = modest_shell_window_get_menu (MODEST_SHELL_WINDOW (child));
361
362         if (menu) {
363                 gtk_menu_popup (GTK_MENU (menu), NULL, NULL, 
364                                 (GtkMenuPositionFunc) menu_position_cb, (gpointer) self,
365                                 1, gtk_get_current_event_time ());
366         }
367 }
368
369 static void
370 on_new_msg_button_clicked (GtkToolButton *button, ModestShell *self)
371 {
372         ModestShellPrivate *priv;
373         gint n_pages;
374         GtkWidget *child;
375
376         priv = MODEST_SHELL_GET_PRIVATE (self);
377
378         n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
379         if (n_pages < 1)
380                 return;
381
382         child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
383
384         modest_ui_actions_on_new_msg (NULL, MODEST_WINDOW (child));
385 }