1 /* Copyright (c) 2006, Nokia Corporation
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
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.
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.
30 #ifndef DBUS_API_SUBJECT_TO_CHANGE
31 #define DBUS_API_SUBJECT_TO_CHANGE
32 #endif /*DBUS_API_SUBJECT_TO_CHANGE*/
34 #include <dbus/dbus.h>
35 #include <dbus/dbus-glib-lowlevel.h>
37 #include <glib/gstdio.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"
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>
58 * For getting and tracking the Bluetooth name
60 #define BTNAME_SERVICE "org.bluez"
61 #define BTNAME_REQUEST_IF "org.bluez.Adapter"
62 #define BTNAME_SIGNAL_IF "org.bluez.Adapter"
63 #define BTNAME_SIGNAL_PATH "/org/bluez/hci0"
65 #define BTNAME_REQ_GET_PROPERTIES "GetProperties"
66 #define BTNAME_REQ_DEFAULT_ADAPTER "DefaultAdapter"
67 #define BTNAME_SIG_CHANGED "NameChanged"
69 #define BTNAME_MATCH_RULE "type='signal',interface='" BTNAME_SIGNAL_IF \
70 "',member='" BTNAME_SIG_CHANGED "'"
72 /* Label child of a captioned */
73 #define CAPTIONED_LABEL_CHILD "captioned-label"
76 static osso_context_t *__osso_context = NULL; /* urgh global */
79 modest_maemo_utils_get_osso_context (void)
82 g_warning ("%s: __osso_context == NULL", __FUNCTION__);
84 return __osso_context;
88 modest_maemo_utils_set_osso_context (osso_context_t *osso_context)
90 g_return_if_fail (osso_context);
91 __osso_context = osso_context;
95 /* update_device_name_from_msg (DBusMessage *message) */
97 /* DBusError error; */
98 /* DBusMessageIter iter; */
100 /* dbus_error_init (&error); */
102 /* if (dbus_set_error_from_message (&error, message)) { */
103 /* g_printerr ("modest: failed to get bluetooth name: %s\n", error.message); */
104 /* dbus_error_free (&error); */
106 /* const gchar *device_name; */
107 /* if (!dbus_message_iter_init (message, &iter)) { */
108 /* g_printerr ("modest: message did not have argument\n"); */
111 /* dbus_message_iter_get_basic (&iter, &device_name); */
112 /* modest_conf_set_string (modest_runtime_get_conf(), */
113 /* MODEST_CONF_DEVICE_NAME, device_name, */
120 /* on_device_name_received (DBusPendingCall *call, void *user_data) */
122 /* DBusMessage *message; */
124 /* g_return_if_fail (dbus_pending_call_get_completed (call)); */
126 /* message = dbus_pending_call_steal_reply (call); */
127 /* if (!message) { */
128 /* g_printerr ("modest: no reply on device name query\n"); */
132 /* update_device_name_from_msg (message); */
133 /* dbus_message_unref (message); */
137 /* static DBusHandlerResult */
138 /* handle_dbus_signal (DBusConnection *conn, DBusMessage *msg, gpointer data) */
140 /* if (dbus_message_is_signal(msg, BTNAME_SIGNAL_IF, BTNAME_SIG_CHANGED)) */
141 /* update_device_name_from_msg (msg); */
143 /* return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; */
147 get_properties_cb (DBusPendingCall *call,
150 DBusMessageIter iter;
151 DBusMessageIter dict_iter;
152 DBusMessageIter dict_entry_iter;
154 gchar *bt_name = NULL;
155 DBusMessage *message;
156 int key_type, array_type, msg_type;
158 if ( !dbus_pending_call_get_completed ( call ) )
159 g_warning ("%s: Call not completed!", __FUNCTION__);
161 message = dbus_pending_call_steal_reply ( call );
164 g_warning ("%s: Message is NULL", __FUNCTION__);
167 dbus_error_init(&err);
168 if (dbus_set_error_from_message (&err, message)) {
169 g_warning ("%s: %s", __FUNCTION__, err.message);
174 dbus_message_iter_init (message, &iter);
175 msg_type = dbus_message_iter_get_arg_type (&iter);
176 dbus_message_iter_recurse (&iter, &dict_iter);
178 while ((array_type = dbus_message_iter_get_arg_type (&dict_iter)) == DBUS_TYPE_DICT_ENTRY) {
180 dbus_message_iter_recurse (&dict_iter, &dict_entry_iter);
182 while ((key_type = dbus_message_iter_get_arg_type (&dict_entry_iter)) == DBUS_TYPE_STRING) {
183 DBusMessageIter dict_entry_content_iter;
187 int dict_entry_content_type;
189 dbus_message_iter_get_basic (&dict_entry_iter, &key);
190 dbus_message_iter_next (&dict_entry_iter);
191 dict_entry_type = dbus_message_iter_get_arg_type (&dict_entry_iter);
192 dbus_message_iter_recurse (&dict_entry_iter, &dict_entry_content_iter);
193 dict_entry_content_type = dbus_message_iter_get_arg_type (&dict_entry_content_iter);
195 if (dict_entry_content_type == DBUS_TYPE_STRING) {
196 dbus_message_iter_get_basic ( &dict_entry_content_iter, &value );
198 if ( strcmp ( key, "Name" ) == 0 ) {
199 g_debug ("-------------Name %s", value);
203 dbus_message_iter_next (&dict_entry_iter);
206 if (key_type != DBUS_TYPE_INVALID)
209 dbus_message_iter_next (&dict_iter);
212 /* Save device name */
214 modest_conf_set_string (modest_runtime_get_conf(),
215 MODEST_CONF_DEVICE_NAME, bt_name,
221 get_default_adapter_cb (DBusPendingCall *call,
224 DBusMessage *message;
225 DBusMessageIter iter;
228 message = dbus_pending_call_steal_reply(call);
230 // Todo extract msg info from here
231 dbus_message_iter_init ( message, &iter );
233 dbus_message_iter_get_basic (&iter, &path);
235 DBusConnection *conn;
236 DBusMessage *adapterMsg = dbus_message_new_method_call(BTNAME_SERVICE, path,
238 BTNAME_REQ_GET_PROPERTIES);
239 DBusPendingCall *call = NULL;
241 conn = dbus_bus_get (DBUS_BUS_SYSTEM, NULL);
243 if (conn && dbus_connection_send_with_reply(conn, adapterMsg, &call, -1) ) {
244 dbus_pending_call_set_notify(call, get_properties_cb, NULL, NULL);
245 dbus_pending_call_block(call);
246 dbus_pending_call_unref(call);
249 g_warning ("Failed to get the default bluetooth adapter");
254 modest_maemo_utils_get_device_name (void)
256 static DBusConnection *conn = NULL;
257 DBusMessage *request;
259 DBusPendingCall *call = NULL;
261 dbus_error_init (&error);
263 conn = dbus_bus_get (DBUS_BUS_SYSTEM, &error);
265 g_printerr ("modest: cannot get on the dbus: %s: %s\n",
266 error.name, error.message);
267 dbus_error_free (&error);
272 /* Get the default adapter */
273 request = dbus_message_new_method_call(BTNAME_SERVICE, "/" ,
275 BTNAME_REQ_DEFAULT_ADAPTER);
277 if (dbus_connection_send_with_reply(conn, request, &call, -1)) {
278 dbus_pending_call_set_notify(call, get_default_adapter_cb, NULL, NULL);
279 dbus_pending_call_block(call);
280 dbus_pending_call_unref(call);
283 /* request = dbus_message_new_method_call (BTNAME_SERVICE, BTNAME_REQUEST_PATH, */
284 /* BTNAME_REQUEST_IF, BTNAME_REQ_GET); */
285 /* if (!request) { */
286 /* /\* should we free the connection? *\/ */
287 /* g_printerr ("modest: dbus_message_new_method_call failed\n"); */
290 /* dbus_message_set_auto_start (request, TRUE); */
291 /* if (dbus_connection_send_with_reply (conn, request, &call, -1)) { */
292 /* dbus_pending_call_set_notify (call, on_device_name_received, */
294 /* dbus_pending_call_unref (call); */
296 /* dbus_message_unref (request); */
298 /* dbus_connection_setup_with_g_main (conn, NULL); */
299 /* dbus_bus_add_match (conn, BTNAME_MATCH_RULE, &error); */
300 /* if (dbus_error_is_set(&error)) { */
301 /* g_printerr ("modest: dbus_bus_add_match failed: %s\n", error.message); */
302 /* dbus_error_free (&error); */
305 /* if (!dbus_connection_add_filter(conn, handle_dbus_signal, NULL, NULL)) */
306 /* g_printerr ("modest: dbus_connection_add_filter failed\n"); */
310 modest_maemo_utils_setup_images_filechooser (GtkFileChooser *chooser)
312 GtkFileFilter *file_filter;
313 GList *image_mimetypes_list;
317 g_return_if_fail (GTK_IS_FILE_CHOOSER (chooser));
319 conf_folder = modest_conf_get_string (modest_runtime_get_conf (), MODEST_CONF_LATEST_INSERT_IMAGE_PATH, NULL);
320 if (conf_folder && conf_folder[0] != '\0') {
321 gtk_file_chooser_set_current_folder_uri (chooser, conf_folder);
323 gchar *images_folder;
324 /* Set the default folder to images folder */
325 images_folder = g_build_filename (g_getenv (MODEST_MAEMO_UTILS_MYDOCS_ENV),
326 MODEST_MAEMO_UTILS_DEFAULT_IMAGE_FOLDER, NULL);
327 gtk_file_chooser_set_current_folder (chooser, images_folder);
328 g_free (images_folder);
330 g_free (conf_folder);
332 /* Set the images mime filter */
333 file_filter = gtk_file_filter_new ();
334 #ifdef MODEST_HAVE_HILDON0_WIDGETS
335 image_mimetypes_list = osso_mime_get_mime_types_for_category (OSSO_MIME_CATEGORY_IMAGES);
337 image_mimetypes_list = hildon_mime_get_mime_types_for_category (HILDON_MIME_CATEGORY_IMAGES);
339 for (node = image_mimetypes_list; node != NULL; node = g_list_next (node)) {
340 gtk_file_filter_add_mime_type (file_filter, node->data);
342 gtk_file_chooser_set_filter (chooser, file_filter);
343 #ifdef MODEST_HAVE_HILDON0_WIDGETS
344 osso_mime_types_list_free (image_mimetypes_list);
346 hildon_mime_types_list_free (image_mimetypes_list);
352 modest_maemo_set_thumbable_scrollbar (GtkScrolledWindow *win,
355 g_return_if_fail (GTK_IS_SCROLLED_WINDOW(win));
356 #ifdef MODEST_HAVE_HILDON1_WIDGETS
357 hildon_helper_set_thumb_scrollbar (win, thumbable);
358 #endif /* MODEST_HAVE_HILDON1_WIDGETS */
362 modest_maemo_utils_get_manager_menubar_as_menu (GtkUIManager *manager,
363 const gchar *item_name)
367 GList *children, *iter;
369 menubar = gtk_ui_manager_get_widget (manager, item_name);
370 new_menu = gtk_menu_new ();
372 children = gtk_container_get_children (GTK_CONTAINER (menubar));
373 for (iter = children; iter != NULL; iter = g_list_next (iter)) {
376 menu = GTK_WIDGET (iter->data);
377 gtk_widget_reparent (menu, new_menu);
380 g_list_free (children);
386 * modest_maemo_utils_create_captioned:
387 * @title_size_group: a #GtkSizeGroup
388 * @value_size_group: a #GtkSizeGroup
390 * @control: a #GtkWidget
392 * this creates a widget (a #GtkHBox) with a control, and a label
393 * (@string) captioning it. It also uses the proper size groups for title
396 * Returns: a widget containing the control and a proper label.
399 modest_maemo_utils_create_captioned (GtkSizeGroup *title_size_group,
400 GtkSizeGroup *value_size_group,
405 return modest_maemo_utils_create_captioned_with_size_type (title_size_group,
414 * modest_maemo_utils_create_captioned_with_size_type:
415 * @title_size_group: a #GtkSizeGroup
416 * @value_size_group: a #GtkSizeGroup
418 * @control: a #GtkWidget
419 * @size_type: a #HildonSizeType
421 * this creates a widget (a #GtkHBox) with a control, and a label
422 * (@string) captioning it. It also uses the proper size groups for title
425 * Returns: a widget containing the control and a proper label.
428 modest_maemo_utils_create_captioned_with_size_type (GtkSizeGroup *title_size_group,
429 GtkSizeGroup *value_size_group,
433 HildonSizeType size_type)
440 label = gtk_label_new (NULL);
441 gtk_label_set_markup (GTK_LABEL (label), title);
443 label = gtk_label_new (title);
445 align = gtk_alignment_new (0.0, 0.0, 1.0, 1.0);
446 gtk_alignment_set_padding (GTK_ALIGNMENT (align), 0, 0, MODEST_MARGIN_DOUBLE, MODEST_MARGIN_TRIPLE);
448 gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
449 hildon_gtk_widget_set_theme_size (label, HILDON_SIZE_FINGER_HEIGHT);
450 gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5);
451 gtk_widget_show (label);
452 gtk_widget_show (align);
453 box = gtk_hbox_new (FALSE, 0);
454 gtk_container_add (GTK_CONTAINER (align), label);
455 gtk_box_pack_start (GTK_BOX (box), align, FALSE, FALSE, 0);
456 gtk_box_pack_start (GTK_BOX (box), control, TRUE, TRUE, 0);
457 if (title_size_group)
458 gtk_size_group_add_widget (title_size_group, label);
459 if (value_size_group)
460 gtk_size_group_add_widget (value_size_group, control);
462 hildon_gtk_widget_set_theme_size (control, size_type);
464 g_object_set_data (G_OBJECT (box), CAPTIONED_LABEL_CHILD, label);
470 * modest_maemo_utils_captioned_set_label:
471 * @captioned: a #GtkWidget built as captioned
472 * @new_label: a string
473 * @use_markup: a #gboolean
475 * set a new label for the captioned
478 modest_maemo_utils_captioned_set_label (GtkWidget *captioned,
479 const gchar *new_label,
484 g_return_if_fail (GTK_IS_WIDGET (captioned));
486 label = g_object_get_data (G_OBJECT (captioned), CAPTIONED_LABEL_CHILD);
487 g_return_if_fail (GTK_IS_LABEL (label));
490 gtk_label_set_markup (GTK_LABEL (label), new_label);
492 gtk_label_set_text (GTK_LABEL (label), new_label);
497 * modest_maemo_utils_set_hbutton_layout:
498 * @title_sizegroup: a #GtkSizeGroup, or %NULL
499 * @value_sizegroup: a #GtkSizeGroup, or %NULL
501 * @button: a #HildonButton
503 * Configures the alignment and layout of @button. If @title_sizegroup is provided,
504 * the title will be aligned to the left using it. If @value_sizegroup is provided,
505 * the value will be aligned to the left using it. It also sets the title
508 * The alignment is left for the title and for the value.
511 modest_maemo_utils_set_hbutton_layout (GtkSizeGroup *title_sizegroup,
512 GtkSizeGroup *value_sizegroup,
516 hildon_button_set_title (HILDON_BUTTON (button), title);
518 hildon_button_add_title_size_group (HILDON_BUTTON (button), title_sizegroup);
520 hildon_button_add_value_size_group (HILDON_BUTTON (button), value_sizegroup);
521 hildon_button_set_alignment (HILDON_BUTTON (button), 0.0, 0.5, 1.0, 0.0);
522 hildon_button_set_title_alignment (HILDON_BUTTON (button), 0.0, 0.5);
523 hildon_button_set_value_alignment (HILDON_BUTTON (button), 0.0, 0.5);
527 modest_maemo_utils_set_vbutton_layout (GtkSizeGroup *sizegroup,
531 hildon_button_set_title (HILDON_BUTTON (button), title);
533 hildon_button_add_title_size_group (HILDON_BUTTON (button), sizegroup);
534 hildon_button_add_value_size_group (HILDON_BUTTON (button), sizegroup);
536 hildon_button_set_alignment (HILDON_BUTTON (button), 0.0, 0.5, 1.0, 0.0);
537 hildon_button_set_title_alignment (HILDON_BUTTON (button), 0.0, 0.5);
538 hildon_button_set_value_alignment (HILDON_BUTTON (button), 0.0, 0.5);
542 modest_maemo_utils_create_group_box (const gchar *label_text, GtkWidget *contents)
547 label = gtk_label_new (label_text);
548 gtk_widget_show (label);
550 box = gtk_vbox_new (FALSE, MODEST_MARGIN_HALF);
551 gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0);
552 gtk_box_pack_start (GTK_BOX (box), contents, TRUE, TRUE, 0);
553 gtk_widget_show (box);
558 static gboolean match_all (TnyList *list, GObject *item, gpointer match_data)
564 modest_maemo_utils_select_attachments (GtkWindow *window, TnyList *att_list, gboolean include_msgs)
567 TnyIterator *iterator;
569 GtkCellRenderer *renderer;
572 gboolean result = TRUE;
573 gint attachments_added = 0;
575 model = GTK_TREE_MODEL (gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_OBJECT));
576 for (iterator = tny_list_create_iterator (att_list);
577 !tny_iterator_is_done (iterator);
578 tny_iterator_next (iterator)) {
581 gchar *filename = NULL;
583 part = (TnyMimePart *) tny_iterator_get_current (iterator);
585 /* Ignore purged attachments and messages if ignore is
587 if (!(tny_mime_part_is_purged (part) ||
588 (TNY_IS_MSG (part) && !include_msgs))) {
590 if (TNY_IS_MSG (part)) {
591 TnyHeader *header = tny_msg_get_header (TNY_MSG (part));
592 filename = tny_header_dup_subject (header);
593 if ((filename == NULL) || (filename[0] == '\0')) {
595 filename = g_strdup (_("mail_va_no_subject"));
597 g_object_unref (header);
599 filename = g_strdup (tny_mime_part_get_filename (part));
601 gtk_list_store_append (GTK_LIST_STORE (model), &iter);
602 gtk_list_store_set (GTK_LIST_STORE (model), &iter, 0, filename, 1, part, -1);
603 attachments_added ++;
605 g_object_unref (part);
609 selector = GTK_WIDGET (hildon_touch_selector_new ());
610 renderer = gtk_cell_renderer_text_new ();
611 hildon_touch_selector_append_column (HILDON_TOUCH_SELECTOR (selector), model, renderer,
613 hildon_touch_selector_set_column_selection_mode (HILDON_TOUCH_SELECTOR (selector),
614 HILDON_TOUCH_SELECTOR_SELECTION_MODE_MULTIPLE);
616 dialog = hildon_picker_dialog_new (window);
617 gtk_window_set_title (GTK_WINDOW (dialog), (attachments_added > 1)?
618 _("mcen_ti_select_attachments_title"):_("mcen_ti_select_attachment_title"));
619 hildon_picker_dialog_set_selector (HILDON_PICKER_DIALOG (dialog), HILDON_TOUCH_SELECTOR (selector));
620 hildon_picker_dialog_set_done_label (HILDON_PICKER_DIALOG (dialog), _HL("wdgt_bd_done"));
622 response = gtk_dialog_run (GTK_DIALOG (dialog));
624 if (response == GTK_RESPONSE_OK) {
625 GList *selected_rows, *node;
627 tny_list_remove_matches (att_list, match_all, NULL);
628 selected_rows = hildon_touch_selector_get_selected_rows (HILDON_TOUCH_SELECTOR (selector), 0);
629 for (node = selected_rows; node != NULL; node = g_list_next (node)) {
634 path = (GtkTreePath *) node->data;
635 gtk_tree_model_get_iter (model, &iter, path);
636 gtk_tree_model_get (model, &iter, 1, &selected, -1);
637 tny_list_append (att_list, selected);
639 if (tny_list_get_length (att_list) == 0)
645 gtk_widget_destroy (dialog);
647 g_object_unref (model);