Move dimming callback to ModestWindow
[modest] / src / hildon2 / modest-hildon2-window.c
1 /* Copyright (c) 2008, 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 <hildon/hildon-banner.h>
31 #include <modest-platform.h>
32 #include <hildon/hildon-program.h>
33 #include "modest-marshal.h"
34 #include <modest-maemo-utils.h>
35 #include <modest-defs.h>
36 #include <modest-ui-dimming-rules.h>
37 #include <modest-ui-dimming-manager.h>
38 #include <modest-window-priv.h>
39 #include <modest-hildon2-window.h>
40 #include <modest-ui-actions.h>
41 #include "modest-text-utils.h"
42 #include <hildon/hildon-edit-toolbar.h>
43
44 typedef struct _EditModeRegister {
45         gchar *description;
46         gchar *button_label;
47         GtkWidget *tree_view;
48         GtkSelectionMode mode;
49         ModestHildon2EditModeCallback action;
50 } EditModeRegister;
51
52 /* 'private'/'protected' functions */
53 static void modest_hildon2_window_class_init  (gpointer klass, gpointer class_data);
54 static void modest_hildon2_window_instance_init (GTypeInstance *instance, gpointer g_class);
55 static void modest_hildon2_window_dispose     (GObject *obj);
56
57 static gboolean on_zoom_minus_plus_not_implemented (ModestWindow *window);
58 static void modest_hildon2_window_show_progress (ModestWindow *window,
59                                                  gboolean show);
60 static void setup_menu (ModestHildon2Window *self);
61
62 static void modest_hildon2_window_show_toolbar (ModestWindow *self,
63                                                  gboolean show_toolbar);
64 static void modest_hildon2_window_add_toolbar (ModestWindow *self,
65                                                GtkToolbar *toolbar);
66 static void modest_hildon2_window_add_to_menu (ModestWindow *window,
67                                                const gchar *label,
68                                                const gchar *accelerator,
69                                                ModestWindowMenuCallback callback,
70                                                ModestDimmingCallback dimming_callback);
71 static void modest_hildon2_window_set_title (ModestWindow *self,
72                                              const gchar *title);
73 static gboolean modest_hildon2_window_toggle_menu (HildonWindow *window,
74                                                     guint button,
75                                                     guint32 time);
76 static void modest_hildon2_window_pack_toolbar_not_implemented (ModestHildon2Window *self,
77                                                                 GtkPackType pack_type,
78                                                                 GtkWidget *toolbar);
79 static EditModeRegister *edit_mode_register_new (const gchar *description,
80                                                  const gchar *button_label,
81                                                  GtkTreeView *tree_view,
82                                                  GtkSelectionMode mode,
83                                                  ModestHildon2EditModeCallback action);
84 static void edit_mode_register_destroy (gpointer data);
85 static void edit_toolbar_button_clicked (HildonEditToolbar *toolbar,
86                                          ModestHildon2Window *self);
87 static void edit_toolbar_arrow_clicked (HildonEditToolbar *toolbar,
88                                         ModestHildon2Window *self);
89
90 typedef struct _ModestHildon2WindowPrivate ModestHildon2WindowPrivate;
91 struct _ModestHildon2WindowPrivate {
92
93         GtkWidget *app_menu;
94         ModestDimmingRulesGroup *app_menu_dimming_group;
95         GtkAccelGroup *accel_group;
96
97         /* Edit mode support */
98         gboolean edit_mode;
99         gint edit_command;
100         GtkWidget *edit_toolbar;
101         GtkWidget *current_edit_tree_view;
102         GHashTable *edit_mode_registry;
103 };
104 #define MODEST_HILDON2_WINDOW_GET_PRIVATE(o)  (G_TYPE_INSTANCE_GET_PRIVATE((o), \
105                                                                             MODEST_TYPE_HILDON2_WINDOW, \
106                                                                             ModestHildon2WindowPrivate))
107
108 /* list my signals */
109 enum {
110         EDIT_MODE_CHANGED_SIGNAL,
111         LAST_SIGNAL
112 };
113
114 /* globals */
115 static GtkWindowClass *parent_class = NULL;
116
117 /* uncomment the following if you have defined any signals */
118 static guint signals[LAST_SIGNAL] = {0};
119
120 /************************************************************************/
121
122 GType
123 modest_hildon2_window_get_type (void)
124 {
125         static GType my_type = 0;
126         if (!my_type) {
127                 static const GTypeInfo my_info = {
128                         sizeof(ModestHildon2WindowClass),
129                         NULL,           /* base init */
130                         NULL,           /* base finalize */
131                         (GClassInitFunc) modest_hildon2_window_class_init,
132                         NULL,           /* class finalize */
133                         NULL,           /* class data */
134                         sizeof(ModestHildon2Window),
135                         1,              /* n_preallocs */
136                         (GInstanceInitFunc) modest_hildon2_window_instance_init,
137                         NULL
138                 };
139                 my_type = g_type_register_static (MODEST_TYPE_WINDOW,
140                                                   "ModestHildon2Window",
141                                                   &my_info, 0);
142         }
143         return my_type;
144 }
145
146 static void
147 modest_hildon2_window_class_init (gpointer klass, gpointer class_data)
148 {
149         GObjectClass *gobject_class;
150         gobject_class = (GObjectClass*) klass;
151         ModestWindowClass *modest_window_class = (ModestWindowClass *) klass;
152         HildonWindowClass *hildon_window_class = (HildonWindowClass *) klass;
153         ModestHildon2WindowClass *modest_hildon2_window_class = (ModestHildon2WindowClass *) klass;
154
155         parent_class            = g_type_class_peek_parent (klass);
156         gobject_class->dispose  = modest_hildon2_window_dispose;
157
158         signals[EDIT_MODE_CHANGED_SIGNAL] =
159                 g_signal_new ("edit-mode-changed",
160                               G_TYPE_FROM_CLASS (gobject_class),
161                               G_SIGNAL_RUN_FIRST,
162                               G_STRUCT_OFFSET (ModestHildon2WindowClass, edit_mode_changed),
163                               NULL, NULL,
164                               modest_marshal_VOID__INT_BOOLEAN,
165                               G_TYPE_NONE, 2, G_TYPE_INT, G_TYPE_BOOLEAN);
166
167         g_type_class_add_private (gobject_class, sizeof(ModestHildon2WindowPrivate));
168         
169         hildon_window_class->toggle_menu = modest_hildon2_window_toggle_menu;
170
171         modest_window_class->zoom_minus_func = on_zoom_minus_plus_not_implemented;
172         modest_window_class->zoom_plus_func = on_zoom_minus_plus_not_implemented;
173         modest_window_class->show_toolbar_func = modest_hildon2_window_show_toolbar;
174         modest_window_class->add_toolbar_func = modest_hildon2_window_add_toolbar;
175         modest_window_class->add_to_menu_func = modest_hildon2_window_add_to_menu;
176         modest_window_class->set_title_func = modest_hildon2_window_set_title;
177         modest_window_class->show_progress_func = modest_hildon2_window_show_progress;
178
179         modest_hildon2_window_class->pack_toolbar_func = modest_hildon2_window_pack_toolbar_not_implemented;
180 }
181
182 static void
183 modest_hildon2_window_dispose (GObject *obj)
184 {
185         ModestHildon2WindowPrivate *priv;
186
187         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE(obj);
188
189         if (priv->app_menu_dimming_group) {
190                 g_object_unref (priv->app_menu_dimming_group);
191                 priv->app_menu_dimming_group = NULL;
192         }
193
194         if (priv->edit_mode_registry) {
195                 g_hash_table_unref (priv->edit_mode_registry);
196                 priv->edit_mode_registry = NULL;
197         }
198
199         G_OBJECT_CLASS(parent_class)->dispose (obj);
200 }
201
202 static void
203 modest_hildon2_window_instance_init (GTypeInstance *instance, gpointer g_class)
204 {
205         ModestHildon2Window *self = NULL;       
206         ModestWindowPrivate *parent_priv = NULL;
207         ModestHildon2WindowPrivate *priv = NULL;
208
209         self = (ModestHildon2Window *) instance;
210         parent_priv = MODEST_WINDOW_GET_PRIVATE (self);
211         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
212
213         priv->accel_group = gtk_accel_group_new ();
214
215         priv->edit_mode = FALSE;
216         priv->edit_toolbar = NULL;
217         priv->current_edit_tree_view = NULL;
218         priv->edit_command = MODEST_HILDON2_WINDOW_EDIT_MODE_NONE;
219         priv->edit_mode_registry = g_hash_table_new_full (g_direct_hash, g_direct_equal,
220                                                           NULL, edit_mode_register_destroy);
221
222         parent_priv->ui_dimming_manager = modest_ui_dimming_manager_new();
223         priv->app_menu_dimming_group = modest_dimming_rules_group_new (MODEST_DIMMING_RULES_MENU, FALSE);
224         gtk_window_add_accel_group (GTK_WINDOW (self), priv->accel_group);
225
226         setup_menu (self);
227
228         modest_ui_dimming_manager_insert_rules_group (parent_priv->ui_dimming_manager, 
229                                                       priv->app_menu_dimming_group);
230
231         /* Dont't restore settings here, 
232          * because it requires a gtk_widget_show(), 
233          * and we don't want to do that until later,
234          * so that the UI is not visible for non-menu D-Bus activation.
235          */
236 }
237
238 static gboolean
239 on_zoom_minus_plus_not_implemented (ModestWindow *window)
240 {
241         g_return_val_if_fail (MODEST_IS_HILDON2_WINDOW (window), FALSE);
242
243         hildon_banner_show_information (NULL, NULL, _CS("ckct_ib_cannot_zoom_here"));
244         return FALSE;
245 }
246
247 static void 
248 modest_hildon2_window_pack_toolbar_not_implemented (ModestHildon2Window *self,
249                                                     GtkPackType pack_type,
250                                                     GtkWidget *toolbar)
251 {
252         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
253
254         g_debug ("%s not implemented", __FUNCTION__);
255 }
256
257 void
258 modest_hildon2_window_pack_toolbar (ModestHildon2Window *self,
259                                     GtkPackType pack_type,
260                                     GtkWidget *toolbar)
261 {
262         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
263
264         MODEST_HILDON2_WINDOW_GET_CLASS (self)->pack_toolbar_func (self, pack_type, toolbar);
265 }
266
267 void 
268 modest_hildon2_window_add_button_to_menu (ModestHildon2Window *self,
269                                           GtkButton *button,
270                                           ModestDimmingCallback dimming_callback)
271 {
272         ModestHildon2WindowPrivate *priv;
273
274         g_return_if_fail (MODEST_IS_HILDON2_WINDOW(self));
275         g_return_if_fail (GTK_IS_BUTTON (button));
276         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
277
278         modest_ui_dimming_manager_set_widget_dimming_mode (GTK_WIDGET (button),
279                                                            MODEST_UI_DIMMING_MODE_HIDE);
280
281         if (dimming_callback)
282                 modest_dimming_rules_group_add_widget_rule (priv->app_menu_dimming_group,
283                                                             GTK_WIDGET (button),
284                                                             (GCallback) dimming_callback,
285                                                             MODEST_WINDOW (self));
286         hildon_app_menu_append (HILDON_APP_MENU (priv->app_menu), GTK_BUTTON (button));
287         gtk_widget_show (GTK_WIDGET (button));
288 }
289
290 static void
291 modest_hildon2_window_add_to_menu (ModestWindow *self,
292                                    const gchar *label,
293                                    const gchar *accelerator,
294                                    ModestWindowMenuCallback callback,
295                                    ModestDimmingCallback dimming_callback)
296 {
297         ModestHildon2WindowPrivate *priv = NULL;
298         GtkWidget *button;
299
300         g_return_if_fail (MODEST_IS_HILDON2_WINDOW(self));
301         g_return_if_fail (label && label[0] != '\0');
302         g_return_if_fail (callback != NULL);
303
304         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
305
306         button = gtk_button_new_with_label (label);
307         g_signal_connect_after (G_OBJECT (button), "clicked",
308                                 G_CALLBACK (callback), (gpointer) self);
309
310         if (accelerator != NULL) {
311                 guint accel_key;
312                 GdkModifierType accel_mods;
313
314                 gtk_accelerator_parse (accelerator, &accel_key, &accel_mods);
315                 gtk_widget_add_accelerator (button, "clicked", priv->accel_group,
316                                             accel_key, accel_mods, 0);
317         }
318
319         modest_hildon2_window_add_button_to_menu (MODEST_HILDON2_WINDOW (self), GTK_BUTTON (button), dimming_callback);
320 }
321
322 static void setup_menu (ModestHildon2Window *self)
323 {
324         ModestHildon2WindowPrivate *priv = NULL;
325
326         g_return_if_fail (MODEST_IS_HILDON2_WINDOW(self));
327
328         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
329
330         priv->app_menu = hildon_app_menu_new ();
331
332         /* we expect that the app menu is filled in children using the expected 
333          * add_to_menu methods */
334
335         hildon_stackable_window_set_main_menu (HILDON_STACKABLE_WINDOW (self), 
336                                                HILDON_APP_MENU (priv->app_menu));
337 }
338
339 static gboolean 
340 modest_hildon2_window_toggle_menu (HildonWindow *window,
341                                     guint button,
342                                     guint32 time)
343 {
344         ModestHildon2WindowPrivate *priv = NULL;
345
346         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (window);
347
348         modest_ui_actions_check_menu_dimming_rules (MODEST_WINDOW (window));
349
350         gtk_widget_queue_resize (priv->app_menu);
351
352         return HILDON_WINDOW_CLASS (parent_class)->toggle_menu (window, button, time);
353 }
354
355 static void
356 modest_hildon2_window_show_toolbar (ModestWindow *self,
357                                     gboolean show_toolbar)
358 {
359         /* Empty implementation: Hildon 2.2 implementation
360          * doesn't switch toolbar visibility */
361 }
362
363 static void
364 modest_hildon2_window_add_toolbar (ModestWindow *self,
365                                    GtkToolbar *toolbar)
366 {
367         hildon_window_add_toolbar (HILDON_WINDOW (self),
368                                    toolbar);
369 }
370
371 static void
372 modest_hildon2_window_set_title (ModestWindow *self,
373                                  const gchar *title)
374 {
375         gtk_window_set_title (GTK_WINDOW (self),
376                               title);
377 }
378
379 static void
380 modest_hildon2_window_show_progress (ModestWindow *self,
381                                      gboolean show)
382 {
383         hildon_gtk_window_set_progress_indicator (GTK_WINDOW (self),
384                                                   show);
385 }
386
387 void 
388 modest_hildon2_window_register_edit_mode (ModestHildon2Window *self,
389                                           gint edit_mode_id,
390                                           const gchar *description,
391                                           const gchar *button_label,
392                                           GtkTreeView *tree_view,
393                                           GtkSelectionMode mode,
394                                           ModestHildon2EditModeCallback action)
395 {
396         ModestHildon2WindowPrivate *priv = NULL;
397         EditModeRegister *reg;
398
399         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
400         g_return_if_fail (edit_mode_id >= 0);
401         g_return_if_fail (description);
402         g_return_if_fail (button_label);
403         g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
404
405         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
406
407         reg = (EditModeRegister *) g_hash_table_lookup (priv->edit_mode_registry, GINT_TO_POINTER (edit_mode_id));
408         g_return_if_fail (reg == NULL);
409
410         reg = edit_mode_register_new (description, button_label, tree_view, mode, action);
411         g_hash_table_insert (priv->edit_mode_registry, GINT_TO_POINTER (edit_mode_id), (gpointer) reg);
412 }
413
414 void
415 modest_hildon2_window_set_edit_mode (ModestHildon2Window *self,
416                                      gint edit_mode_id)
417 {
418         ModestHildon2WindowPrivate *priv = NULL;
419         EditModeRegister *reg;
420         GtkTreeSelection *selection;
421
422         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
423         g_return_if_fail (edit_mode_id >= 0);
424
425         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
426         reg = (EditModeRegister *) g_hash_table_lookup (priv->edit_mode_registry, GINT_TO_POINTER (edit_mode_id));
427         g_return_if_fail (reg != NULL);
428
429         if (priv->edit_mode) {
430                 modest_hildon2_window_unset_edit_mode (self);
431         }
432
433         priv->edit_mode = TRUE;
434         priv->edit_command = edit_mode_id;
435
436         priv->current_edit_tree_view = reg->tree_view;
437         g_object_set (G_OBJECT (priv->current_edit_tree_view),
438                       "hildon-ui-mode", HILDON_UI_MODE_EDIT,
439                       NULL);
440
441         selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->current_edit_tree_view));
442         gtk_tree_selection_set_mode (selection, reg->mode);
443         if (reg->mode == GTK_SELECTION_SINGLE || reg->mode == GTK_SELECTION_BROWSE) {
444                 GtkTreeModel *model;
445                 GtkTreeIter iter;
446
447                 model = gtk_tree_view_get_model (GTK_TREE_VIEW (priv->current_edit_tree_view));
448                 if (gtk_tree_model_get_iter_first (model, &iter)) {
449                         gtk_tree_view_scroll_to_point (GTK_TREE_VIEW (priv->current_edit_tree_view), 0, 0);
450                         gtk_tree_selection_select_iter (selection, &iter);
451                 }
452         } else {
453                 gtk_tree_selection_unselect_all (selection);
454         }
455
456         priv->edit_toolbar = hildon_edit_toolbar_new ();
457         hildon_edit_toolbar_set_label (HILDON_EDIT_TOOLBAR (priv->edit_toolbar),
458                                        reg->description);
459         hildon_edit_toolbar_set_button_label (HILDON_EDIT_TOOLBAR (priv->edit_toolbar),
460                                               reg->button_label);
461         modest_hildon2_window_pack_toolbar (self, GTK_PACK_START,
462                                             priv->edit_toolbar);
463
464         g_signal_connect (G_OBJECT (priv->edit_toolbar), "button-clicked",
465                           G_CALLBACK (edit_toolbar_button_clicked), (gpointer) self);
466         g_signal_connect (G_OBJECT (priv->edit_toolbar), "arrow-clicked",
467                           G_CALLBACK (edit_toolbar_arrow_clicked), (gpointer) self);
468
469         gtk_widget_show (priv->edit_toolbar);
470         gtk_widget_queue_resize (priv->current_edit_tree_view);
471         gtk_window_fullscreen (GTK_WINDOW (self));
472
473         g_signal_emit (G_OBJECT (self), signals[EDIT_MODE_CHANGED_SIGNAL], 0,
474                        priv->edit_command, priv->edit_mode);
475 }
476
477 void 
478 modest_hildon2_window_unset_edit_mode (ModestHildon2Window *self)
479 {
480         ModestHildon2WindowPrivate *priv = NULL;
481
482         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
483         priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
484
485         if (priv->edit_toolbar) {
486                 gtk_widget_destroy (priv->edit_toolbar);
487                 priv->edit_toolbar = NULL;
488         }
489
490         if (priv->edit_mode) {
491                 priv->edit_mode = FALSE;
492                 if (priv->current_edit_tree_view) {
493                         g_object_set (G_OBJECT (priv->current_edit_tree_view), 
494                                       "hildon-ui-mode", HILDON_UI_MODE_NORMAL, 
495                                       NULL);
496                         gtk_widget_queue_resize (priv->current_edit_tree_view);
497                         priv->current_edit_tree_view = NULL;
498                 }
499                 gtk_window_unfullscreen (GTK_WINDOW (self));
500                 g_signal_emit (G_OBJECT (self), signals[EDIT_MODE_CHANGED_SIGNAL], 0,
501                                priv->edit_command, priv->edit_mode);
502                 priv->edit_command = MODEST_HILDON2_WINDOW_EDIT_MODE_NONE;
503         }
504 }
505
506 static EditModeRegister *
507 edit_mode_register_new (const gchar *description,
508                         const gchar *button_label,
509                         GtkTreeView *tree_view,
510                         GtkSelectionMode mode,
511                         ModestHildon2EditModeCallback action)
512 {
513         EditModeRegister *reg;
514
515         reg = g_slice_new (EditModeRegister);
516
517         reg->description = g_strdup (description);
518         reg->button_label = g_strdup (button_label);
519         reg->tree_view = g_object_ref (tree_view);
520         reg->mode = mode;
521         reg->action = action;
522
523         return reg;
524 }
525
526 static void 
527 edit_mode_register_destroy (gpointer data)
528 {
529         EditModeRegister *reg = (EditModeRegister *) data;
530
531         g_free (reg->description);
532         g_free (reg->button_label);
533         g_object_unref (reg->tree_view);
534
535         g_slice_free (EditModeRegister, reg);
536 }
537
538 static void
539 edit_toolbar_button_clicked (HildonEditToolbar *toolbar,
540                              ModestHildon2Window *self)
541 {
542         ModestHildon2WindowPrivate *priv = MODEST_HILDON2_WINDOW_GET_PRIVATE (self);
543         EditModeRegister *reg;
544
545         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
546
547         reg = (EditModeRegister *) g_hash_table_lookup (priv->edit_mode_registry, 
548                                                         GINT_TO_POINTER (priv->edit_command));
549
550         if (reg) {
551                 if ((reg->action == NULL) || reg->action (self))
552                         modest_hildon2_window_unset_edit_mode (self);
553         } else {
554                 modest_hildon2_window_unset_edit_mode (self);
555         }
556 }
557
558 static void
559 edit_toolbar_arrow_clicked (HildonEditToolbar *toolbar,
560                             ModestHildon2Window *self)
561 {
562         g_return_if_fail (MODEST_IS_HILDON2_WINDOW (self));
563
564         modest_hildon2_window_unset_edit_mode (self);
565 }
566