2ca89a502cef465d0a9643e785062e8c80ca5e06
[modest] / src / hildon2 / modest-maemo-utils.c
1 /* Copyright (c) 2006, 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 #ifndef DBUS_API_SUBJECT_TO_CHANGE
31 #define DBUS_API_SUBJECT_TO_CHANGE
32 #endif /*DBUS_API_SUBJECT_TO_CHANGE*/
33
34 #include <dbus/dbus.h>
35 #include <dbus/dbus-glib-lowlevel.h>
36 #include <glib.h>
37 #include <glib/gstdio.h>
38 #include <errno.h>
39 #include <string.h> /* for strlen */
40 #include <modest-runtime.h>
41 #include <libgnomevfs/gnome-vfs.h>
42 #include <tny-fs-stream.h>
43 #include <tny-camel-account.h>
44 #include <tny-status.h>
45 #include <tny-camel-transport-account.h>
46 #include <tny-camel-imap-store-account.h>
47 #include <tny-camel-pop-store-account.h>
48 #include "modest-hildon-includes.h"
49
50 #include <modest-defs.h>
51 #include "modest-maemo-utils.h"
52 #include "modest-text-utils.h"
53 #include "modest-platform.h"
54 #include "modest-ui-constants.h"
55 #include <hildon/hildon-picker-dialog.h>
56
57 /* Label child of a captioned */
58 #define CAPTIONED_LABEL_CHILD "captioned-label"
59
60
61 static osso_context_t *__osso_context = NULL; /* urgh global */
62
63 osso_context_t *
64 modest_maemo_utils_get_osso_context (void)
65 {
66         if (!__osso_context) 
67                 __osso_context = osso_initialize(PACKAGE,PACKAGE_VERSION,
68                                                  FALSE, NULL);
69
70         return __osso_context;
71 }
72
73 static void
74 get_properties_cb (DBusMessage *message)
75 {
76         DBusMessageIter iter;
77         DBusMessageIter dict_iter;
78         DBusMessageIter dict_entry_iter;
79         DBusError err;
80         gchar *bt_name = NULL;
81         int key_type, array_type, msg_type;
82
83         dbus_error_init(&err);
84         if (dbus_set_error_from_message (&err, message)) {
85                 g_warning ("%s: %s", __FUNCTION__, err.message);
86         }
87
88         /* Get msg type */
89         dbus_message_iter_init (message, &iter);
90         msg_type = dbus_message_iter_get_arg_type (&iter);
91         dbus_message_iter_recurse (&iter, &dict_iter);
92
93         while ((array_type = dbus_message_iter_get_arg_type (&dict_iter)) == DBUS_TYPE_DICT_ENTRY) {
94
95                 dbus_message_iter_recurse (&dict_iter, &dict_entry_iter);
96
97                 while ((key_type = dbus_message_iter_get_arg_type (&dict_entry_iter)) == DBUS_TYPE_STRING) {
98                         DBusMessageIter dict_entry_content_iter;
99                         char *key;
100                         char *value;
101                         int dict_entry_type;
102                         int dict_entry_content_type;
103
104                         dbus_message_iter_get_basic (&dict_entry_iter, &key);
105                         dbus_message_iter_next (&dict_entry_iter);
106                         dict_entry_type = dbus_message_iter_get_arg_type (&dict_entry_iter);
107                         dbus_message_iter_recurse (&dict_entry_iter, &dict_entry_content_iter);
108                         dict_entry_content_type = dbus_message_iter_get_arg_type (&dict_entry_content_iter);
109
110                         if (dict_entry_content_type == DBUS_TYPE_STRING) {
111                                 dbus_message_iter_get_basic ( &dict_entry_content_iter, &value );
112
113                                 if (strcmp (key, "Name") == 0 ) {
114                                         g_debug ("-------------Name %s", value);
115                                         bt_name = value;
116                                         break;
117                                 }
118                         }
119                         dbus_message_iter_next (&dict_entry_iter);
120                 }
121
122                 if (key_type != DBUS_TYPE_INVALID)
123                         break;
124
125                 dbus_message_iter_next (&dict_iter);
126         }
127
128         /* Save device name */
129         if (bt_name) {
130                 g_debug ("Setting the device name %s", bt_name);
131                 modest_conf_set_string (modest_runtime_get_conf(),
132                                         MODEST_CONF_DEVICE_NAME, bt_name,
133                                         NULL);
134         }
135 }
136
137 static void
138 get_default_adapter_cb (DBusConnection *conn,
139                         DBusMessage *message)
140 {
141         DBusMessageIter iter;
142         gchar* path = NULL;
143         DBusError error;
144         DBusMessage *msg, *adapterMsg;
145
146         dbus_message_iter_init (message, &iter);
147         dbus_message_iter_get_basic (&iter, &path);
148
149         if (!path)
150                 return;
151
152         adapterMsg = dbus_message_new_method_call ("org.bluez", path,
153                                                    "org.bluez.Adapter",
154                                                    "GetProperties");
155
156         dbus_error_init (&error);
157         msg = dbus_connection_send_with_reply_and_block (conn, adapterMsg, -1, &error);
158         if (msg) {
159                 g_debug ("Getting the properties");
160                 get_properties_cb (msg);
161                 dbus_message_unref (msg);
162         }
163         dbus_message_unref (adapterMsg);
164 }
165
166 void
167 modest_maemo_utils_get_device_name (void)
168 {
169         static DBusConnection *conn = NULL;
170         DBusMessage *request;
171         DBusError error;
172         DBusMessage *msg;
173
174         dbus_error_init (&error);
175         if (!conn) {
176                 conn = dbus_bus_get (DBUS_BUS_SYSTEM, &error);
177                 if (!conn) {
178                         g_printerr ("modest: cannot get on the dbus: %s: %s\n",
179                                     error.name, error.message);
180                         dbus_error_free (&error);
181                         return;
182                 }
183         }
184
185         /* Get the default adapter */
186         request = dbus_message_new_method_call ("org.bluez", "/" ,
187                                                 "org.bluez.Manager",
188                                                 "DefaultAdapter");
189
190         msg = dbus_connection_send_with_reply_and_block (conn, request, -1, &error);
191         if (msg) {
192                 g_debug ("Getting the default adapter");
193                 get_default_adapter_cb (conn, msg);
194                 dbus_message_unref (msg);
195         }
196         dbus_message_unref (request);
197 }
198
199 void
200 modest_maemo_utils_setup_images_filechooser (GtkFileChooser *chooser)
201 {
202         GtkFileFilter *file_filter;
203         GList *image_mimetypes_list;
204         GList *node;
205         gchar *conf_folder;
206
207         g_return_if_fail (GTK_IS_FILE_CHOOSER (chooser));
208
209         conf_folder = modest_conf_get_string (modest_runtime_get_conf (), MODEST_CONF_LATEST_INSERT_IMAGE_PATH, NULL);
210         if (conf_folder && conf_folder[0] != '\0') {
211                 gtk_file_chooser_set_current_folder_uri (chooser, conf_folder);
212         } else {
213                 gchar *images_folder;
214                 /* Set the default folder to images folder */
215                 images_folder = g_build_filename (g_getenv (MODEST_MAEMO_UTILS_MYDOCS_ENV),
216                                                   MODEST_MAEMO_UTILS_DEFAULT_IMAGE_FOLDER, NULL);
217                 gtk_file_chooser_set_current_folder (chooser, images_folder);
218                 g_free (images_folder);
219         }
220         g_free (conf_folder);
221
222         /* Set the images mime filter */
223         file_filter = gtk_file_filter_new ();
224 #ifdef MODEST_HAVE_HILDON0_WIDGETS
225         image_mimetypes_list = osso_mime_get_mime_types_for_category (OSSO_MIME_CATEGORY_IMAGES);
226 #else
227         image_mimetypes_list = hildon_mime_get_mime_types_for_category (HILDON_MIME_CATEGORY_IMAGES);
228 #endif
229         for (node = image_mimetypes_list; node != NULL; node = g_list_next (node)) {
230                 gtk_file_filter_add_mime_type (file_filter, node->data);
231         }
232         gtk_file_chooser_set_filter (chooser, file_filter);
233 #ifdef MODEST_HAVE_HILDON0_WIDGETS
234         osso_mime_types_list_free (image_mimetypes_list);
235 #else
236         hildon_mime_types_list_free (image_mimetypes_list);
237 #endif
238
239 }
240
241 void
242 modest_maemo_set_thumbable_scrollbar (GtkScrolledWindow *win, 
243                                       gboolean thumbable)
244 {
245         g_return_if_fail (GTK_IS_SCROLLED_WINDOW(win));
246 #ifdef MODEST_HAVE_HILDON1_WIDGETS              
247         hildon_helper_set_thumb_scrollbar (win, thumbable);
248 #endif /* MODEST_HAVE_HILDON1_WIDGETS */
249 }
250
251 GtkWidget *
252 modest_maemo_utils_get_manager_menubar_as_menu (GtkUIManager *manager,
253                                                 const gchar *item_name)
254 {
255         GtkWidget *new_menu;
256         GtkWidget *menubar;
257         GList *children, *iter;
258
259         menubar = gtk_ui_manager_get_widget (manager, item_name);
260         new_menu = gtk_menu_new ();
261
262         children = gtk_container_get_children (GTK_CONTAINER (menubar));
263         for (iter = children; iter != NULL; iter = g_list_next (iter)) {
264                 GtkWidget *menu;
265
266                 menu = GTK_WIDGET (iter->data);
267                 gtk_widget_reparent (menu, new_menu);
268         }
269         
270         g_list_free (children);
271
272         return new_menu;
273 }
274
275 /**
276  * modest_maemo_utils_create_captioned:
277  * @title_size_group: a #GtkSizeGroup
278  * @value_size_group: a #GtkSizeGroup
279  * @title: a string
280  * @control: a #GtkWidget
281  *
282  * this creates a widget (a #GtkHBox) with a control, and a label
283  * (@string) captioning it. It also uses the proper size groups for title
284  * and control.
285  *
286  * Returns: a widget containing the control and a proper label.
287  */
288 GtkWidget *
289 modest_maemo_utils_create_captioned    (GtkSizeGroup *title_size_group,
290                                         GtkSizeGroup *value_size_group,
291                                         const gchar *title,
292                                         gboolean use_markup,
293                                         GtkWidget *control)
294 {
295         return modest_maemo_utils_create_captioned_with_size_type (title_size_group,
296                                                                    value_size_group,
297                                                                    title,
298                                                                    use_markup,
299                                                                    control,
300                                                                    0);
301 }
302
303 /**
304  * modest_maemo_utils_create_captioned_with_size_type:
305  * @title_size_group: a #GtkSizeGroup
306  * @value_size_group: a #GtkSizeGroup
307  * @title: a string
308  * @control: a #GtkWidget
309  * @size_type: a #HildonSizeType
310  *
311  * this creates a widget (a #GtkHBox) with a control, and a label
312  * (@string) captioning it. It also uses the proper size groups for title
313  * and control.
314  *
315  * Returns: a widget containing the control and a proper label.
316  */
317 GtkWidget *
318 modest_maemo_utils_create_captioned_with_size_type    (GtkSizeGroup *title_size_group,
319                                                        GtkSizeGroup *value_size_group,
320                                                        const gchar *title,
321                                                        gboolean use_markup,
322                                                        GtkWidget *control,
323                                                        HildonSizeType size_type)
324 {
325         GtkWidget *label;
326         GtkWidget *align;
327         GtkWidget *box;
328   
329         if (use_markup) {
330                 label = gtk_label_new (NULL);
331                 gtk_label_set_markup (GTK_LABEL (label), title);
332         } else {
333                 label = gtk_label_new (title);
334         }
335         align = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
336         gtk_alignment_set_padding (GTK_ALIGNMENT (align), 0, 0, MODEST_MARGIN_DOUBLE, MODEST_MARGIN_DOUBLE);
337
338         gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
339         hildon_gtk_widget_set_theme_size (label, HILDON_SIZE_FINGER_HEIGHT);
340         gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
341         gtk_widget_show (label);
342         gtk_widget_show (align);
343         box = gtk_hbox_new (FALSE, 0);
344         gtk_container_add (GTK_CONTAINER (align), label);
345         gtk_box_pack_start (GTK_BOX (box), align, FALSE, FALSE, 0);
346         gtk_box_pack_start (GTK_BOX (box), control, TRUE, TRUE, 0);
347         if (title_size_group)
348                 gtk_size_group_add_widget (title_size_group, label);
349         if (value_size_group)
350                 gtk_size_group_add_widget (value_size_group, control);
351
352         hildon_gtk_widget_set_theme_size (control, size_type);
353
354         g_object_set_data (G_OBJECT (box), CAPTIONED_LABEL_CHILD, label);
355
356         return box;
357 }
358
359 /**
360  * modest_maemo_utils_captioned_set_label:
361  * @captioned: a #GtkWidget built as captioned
362  * @new_label: a string
363  * @use_markup: a #gboolean
364  *
365  * set a new label for the captioned
366  */
367 void
368 modest_maemo_utils_captioned_set_label (GtkWidget *captioned,
369                                         const gchar *new_label,
370                                         gboolean use_markup)
371 {
372         GtkWidget *label;
373
374         g_return_if_fail (GTK_IS_WIDGET (captioned));
375
376         label = g_object_get_data (G_OBJECT (captioned), CAPTIONED_LABEL_CHILD);
377         g_return_if_fail (GTK_IS_LABEL (label));
378
379         if (use_markup) {
380                 gtk_label_set_markup (GTK_LABEL (label), new_label);
381         } else {
382                 gtk_label_set_text (GTK_LABEL (label), new_label);
383         }
384 }
385
386 /**
387  * modest_maemo_utils_captioned_get_label_widget:
388  * @captioned: a #GtkWidget built as captioned
389  *
390  * obtains the label widget for the captioned
391  *
392  * Returns: a #GtkLabel
393  */
394 GtkWidget *
395 modest_maemo_utils_captioned_get_label_widget (GtkWidget *captioned)
396 {
397         GtkWidget *label;
398
399         g_return_val_if_fail (GTK_IS_WIDGET (captioned), NULL);
400
401         label = g_object_get_data (G_OBJECT (captioned), CAPTIONED_LABEL_CHILD);
402         g_return_val_if_fail (GTK_IS_LABEL (label), NULL);
403
404         return label;
405
406 }
407
408 /**
409  * modest_maemo_utils_set_hbutton_layout:
410  * @title_sizegroup: a #GtkSizeGroup, or %NULL
411  * @value_sizegroup: a #GtkSizeGroup, or %NULL
412  * @title: a string
413  * @button: a #HildonButton
414  *
415  * Configures the alignment and layout of @button. If @title_sizegroup is provided,
416  * the title will be aligned to the left using it. If @value_sizegroup is provided,
417  * the value will be aligned to the left using it. It also sets the title
418  * of the button.
419  *
420  * The alignment is left for the title and for the value.
421  */
422 void
423 modest_maemo_utils_set_hbutton_layout (GtkSizeGroup *title_sizegroup, 
424                                        GtkSizeGroup *value_sizegroup,
425                                        const gchar *title, 
426                                        GtkWidget *button)
427 {
428         hildon_button_set_title (HILDON_BUTTON (button), title);
429         if (title_sizegroup)
430                 hildon_button_add_title_size_group (HILDON_BUTTON (button), title_sizegroup);
431         if (value_sizegroup)
432                 hildon_button_add_value_size_group (HILDON_BUTTON (button), value_sizegroup);
433         hildon_button_set_alignment (HILDON_BUTTON (button), 0.0, 0.5, 1.0, 0.0);
434         hildon_button_set_title_alignment (HILDON_BUTTON (button), 0.0, 0.5);
435         hildon_button_set_value_alignment (HILDON_BUTTON (button), 0.0, 0.5);
436 }
437
438 void
439 modest_maemo_utils_set_vbutton_layout (GtkSizeGroup *sizegroup, 
440                                        const gchar *title, 
441                                        GtkWidget *button)
442 {
443         hildon_button_set_title (HILDON_BUTTON (button), title);
444         if (sizegroup) {
445                 hildon_button_add_title_size_group (HILDON_BUTTON (button), sizegroup);
446                 hildon_button_add_value_size_group (HILDON_BUTTON (button), sizegroup);
447         }
448         hildon_button_set_alignment (HILDON_BUTTON (button), 0.0, 0.5, 1.0, 0.0);
449         hildon_button_set_title_alignment (HILDON_BUTTON (button), 0.0, 0.5);
450         hildon_button_set_value_alignment (HILDON_BUTTON (button), 0.0, 0.5);
451 }
452
453 GtkWidget *
454 modest_maemo_utils_create_group_box (const gchar *label_text, GtkWidget *contents)
455 {
456         GtkWidget *label;
457         GtkWidget *box;
458
459         label = gtk_label_new (label_text);
460         gtk_widget_show (label);
461
462         box = gtk_vbox_new (FALSE, MODEST_MARGIN_HALF);
463         gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0);
464         gtk_box_pack_start (GTK_BOX (box), contents, TRUE, TRUE, 0);
465         gtk_widget_show (box);
466
467         return box;
468 }
469
470 static gboolean match_all (TnyList *list, GObject *item, gpointer match_data)
471 {
472         return TRUE;
473 }
474
475 gboolean
476 modest_maemo_utils_select_attachments (GtkWindow *window, TnyList *att_list, gboolean include_msgs)
477 {
478         GtkTreeModel *model;
479         TnyIterator *iterator;
480         GtkWidget *selector;
481         GtkCellRenderer *renderer;
482         GtkWidget *dialog;
483         gint response;
484         gboolean result = TRUE;
485         gint attachments_added = 0;
486
487         model = GTK_TREE_MODEL (gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_OBJECT));
488         for (iterator = tny_list_create_iterator (att_list);
489              !tny_iterator_is_done (iterator);
490              tny_iterator_next (iterator)) {
491                 GtkTreeIter iter;
492                 TnyMimePart *part;
493                 gchar *filename = NULL;
494
495                 part = (TnyMimePart *) tny_iterator_get_current (iterator);
496
497                 /* Ignore purged attachments and messages if ignore is
498                    set to TRUE */
499                 if (!(tny_mime_part_is_purged (part) ||
500                       (TNY_IS_MSG (part) && !include_msgs))) {
501
502                         if (TNY_IS_MSG (part)) {
503                                 TnyHeader *header = tny_msg_get_header (TNY_MSG (part));
504                                 filename = tny_header_dup_subject (header);
505                                 g_object_unref (header);
506                         } else {
507                                 filename = g_strdup (tny_mime_part_get_filename (part));
508                         }
509                         if ((filename == NULL) || (filename[0] == '\0')) {
510                                 g_free (filename);
511                                 filename = g_strdup (_("mail_va_no_subject"));
512                         }
513                         gtk_list_store_append (GTK_LIST_STORE (model), &iter);
514                         gtk_list_store_set (GTK_LIST_STORE (model), &iter, 0, filename, 1, part, -1);
515                         attachments_added ++;
516                         g_free (filename);
517                 }
518                 g_object_unref (part);
519         }
520         g_object_unref (iterator);
521
522         selector = GTK_WIDGET (hildon_touch_selector_new ());
523         renderer = gtk_cell_renderer_text_new ();
524         hildon_touch_selector_append_column (HILDON_TOUCH_SELECTOR (selector), model, renderer,
525                                              "text", 0, NULL);
526         hildon_touch_selector_set_column_selection_mode (HILDON_TOUCH_SELECTOR (selector), 
527                                                          HILDON_TOUCH_SELECTOR_SELECTION_MODE_MULTIPLE);
528
529         dialog = hildon_picker_dialog_new (window);
530         gtk_window_set_title (GTK_WINDOW (dialog), (attachments_added > 1)?
531                               _("mcen_ti_select_attachments_title"):_("mcen_ti_select_attachment_title"));
532         hildon_picker_dialog_set_selector (HILDON_PICKER_DIALOG (dialog), HILDON_TOUCH_SELECTOR (selector));
533         hildon_picker_dialog_set_done_label (HILDON_PICKER_DIALOG (dialog), _HL("wdgt_bd_done"));
534
535         response = gtk_dialog_run (GTK_DIALOG (dialog));
536
537         if (response == GTK_RESPONSE_OK) {
538                 GList *selected_rows, *node;
539
540                 tny_list_remove_matches (att_list, match_all, NULL);
541                 selected_rows = hildon_touch_selector_get_selected_rows (HILDON_TOUCH_SELECTOR (selector), 0);
542                 for (node = selected_rows; node != NULL; node = g_list_next (node)) {
543                         GtkTreePath *path;
544                         GObject *selected;
545                         GtkTreeIter iter;
546
547                         path = (GtkTreePath *) node->data;
548                         gtk_tree_model_get_iter (model, &iter, path);
549                         gtk_tree_model_get (model, &iter, 1, &selected, -1);
550                         tny_list_append (att_list, selected);
551                 }
552                 if (tny_list_get_length (att_list) == 0)
553                         result = FALSE;
554
555                 g_list_foreach (selected_rows, (GFunc) gtk_tree_path_free, NULL);
556                 g_list_free (selected_rows);
557         } else {
558                 result = FALSE;
559         }
560
561         gtk_widget_destroy (dialog);
562
563         g_object_unref (model);
564
565         return result;
566 }