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>
56 #ifdef MODEST_USE_IPHB
57 #include <iphbd/libiphb.h>
60 /* Label child of a captioned */
61 #define CAPTIONED_LABEL_CHILD "captioned-label"
63 #ifdef MODEST_PLATFORM_MAEMO
64 #define INTERNAL_MMC_USB_MODE "/system/osso/af/internal-mmc-used-over-usb"
67 static osso_context_t *__osso_context = NULL; /* urgh global */
70 modest_maemo_utils_get_osso_context (void)
73 __osso_context = osso_initialize(PACKAGE,PACKAGE_VERSION,
76 return __osso_context;
80 get_properties_cb (DBusMessage *message)
83 DBusMessageIter dict_iter;
84 DBusMessageIter dict_entry_iter;
86 gchar *bt_name = NULL;
87 int key_type, array_type, msg_type;
89 dbus_error_init(&err);
90 if (dbus_set_error_from_message (&err, message)) {
91 g_warning ("%s: %s", __FUNCTION__, err.message);
95 dbus_message_iter_init (message, &iter);
96 msg_type = dbus_message_iter_get_arg_type (&iter);
97 dbus_message_iter_recurse (&iter, &dict_iter);
99 while ((array_type = dbus_message_iter_get_arg_type (&dict_iter)) == DBUS_TYPE_DICT_ENTRY) {
101 dbus_message_iter_recurse (&dict_iter, &dict_entry_iter);
103 while ((key_type = dbus_message_iter_get_arg_type (&dict_entry_iter)) == DBUS_TYPE_STRING) {
104 DBusMessageIter dict_entry_content_iter;
108 int dict_entry_content_type;
110 dbus_message_iter_get_basic (&dict_entry_iter, &key);
111 dbus_message_iter_next (&dict_entry_iter);
112 dict_entry_type = dbus_message_iter_get_arg_type (&dict_entry_iter);
113 dbus_message_iter_recurse (&dict_entry_iter, &dict_entry_content_iter);
114 dict_entry_content_type = dbus_message_iter_get_arg_type (&dict_entry_content_iter);
116 if (dict_entry_content_type == DBUS_TYPE_STRING) {
117 dbus_message_iter_get_basic ( &dict_entry_content_iter, &value );
119 if (strcmp (key, "Name") == 0 ) {
120 g_debug ("-------------Name %s", value);
125 dbus_message_iter_next (&dict_entry_iter);
128 if (key_type != DBUS_TYPE_INVALID)
131 dbus_message_iter_next (&dict_iter);
134 /* Save device name */
136 g_debug ("Setting the device name %s", bt_name);
137 modest_conf_set_string (modest_runtime_get_conf(),
138 MODEST_CONF_DEVICE_NAME, bt_name,
144 get_default_adapter_cb (DBusConnection *conn,
145 DBusMessage *message)
147 DBusMessageIter iter;
150 DBusMessage *msg, *adapterMsg;
152 dbus_message_iter_init (message, &iter);
153 dbus_message_iter_get_basic (&iter, &path);
158 adapterMsg = dbus_message_new_method_call ("org.bluez", path,
162 dbus_error_init (&error);
163 msg = dbus_connection_send_with_reply_and_block (conn, adapterMsg, -1, &error);
165 g_debug ("Getting the properties");
166 get_properties_cb (msg);
167 dbus_message_unref (msg);
169 dbus_message_unref (adapterMsg);
173 modest_maemo_utils_get_device_name (void)
175 static DBusConnection *conn = NULL;
176 DBusMessage *request;
180 dbus_error_init (&error);
182 conn = dbus_bus_get (DBUS_BUS_SYSTEM, &error);
184 g_printerr ("modest: cannot get on the dbus: %s: %s\n",
185 error.name, error.message);
186 dbus_error_free (&error);
191 /* Get the default adapter */
192 request = dbus_message_new_method_call ("org.bluez", "/" ,
196 msg = dbus_connection_send_with_reply_and_block (conn, request, -1, &error);
198 g_debug ("Getting the default adapter");
199 get_default_adapter_cb (conn, msg);
200 dbus_message_unref (msg);
203 dbus_message_unref (request);
204 if (dbus_error_is_set (&error))
205 dbus_error_free (&error);
209 modest_maemo_utils_setup_images_filechooser (GtkFileChooser *chooser)
211 GtkFileFilter *file_filter;
212 GList *image_mimetypes_list;
216 g_return_if_fail (GTK_IS_FILE_CHOOSER (chooser));
218 conf_folder = modest_conf_get_string (modest_runtime_get_conf (),
219 MODEST_CONF_LATEST_INSERT_IMAGE_PATH, NULL);
220 if (conf_folder && conf_folder[0] != '\0') {
221 gtk_file_chooser_set_current_folder_uri (chooser, conf_folder);
223 gchar *images_folder;
224 /* Set the default folder to images folder */
225 images_folder = (gchar *) g_strdup(g_get_user_special_dir (G_USER_DIRECTORY_PICTURES));
226 if (!images_folder) {
228 images_folder = g_build_filename (g_getenv (MODEST_MAEMO_UTILS_MYDOCS_ENV),
229 MODEST_MAEMO_UTILS_DEFAULT_IMAGE_FOLDER, NULL);
231 gtk_file_chooser_set_current_folder (chooser, images_folder);
232 g_free (images_folder);
234 g_free (conf_folder);
236 /* Set the images mime filter */
237 file_filter = gtk_file_filter_new ();
238 image_mimetypes_list = hildon_mime_get_mime_types_for_category (HILDON_MIME_CATEGORY_IMAGES);
239 for (node = image_mimetypes_list; node != NULL; node = g_list_next (node)) {
240 gtk_file_filter_add_mime_type (file_filter, node->data);
242 gtk_file_chooser_set_filter (chooser, file_filter);
243 hildon_mime_types_list_free (image_mimetypes_list);
247 static gboolean match_all (TnyList *list, GObject *item, gpointer match_data)
253 modest_maemo_utils_select_attachments (GtkWindow *window, TnyList *att_list, gboolean include_msgs)
256 TnyIterator *iterator;
258 GtkCellRenderer *renderer;
261 gboolean result = TRUE;
262 gint attachments_added = 0;
264 model = GTK_TREE_MODEL (gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_OBJECT));
265 for (iterator = tny_list_create_iterator (att_list);
266 !tny_iterator_is_done (iterator);
267 tny_iterator_next (iterator)) {
270 gchar *filename = NULL;
272 part = (TnyMimePart *) tny_iterator_get_current (iterator);
274 /* Ignore purged attachments and messages if ignore is
276 if (!(tny_mime_part_is_purged (part) ||
277 (TNY_IS_MSG (part) && !include_msgs))) {
279 if (TNY_IS_MSG (part)) {
280 TnyHeader *header = tny_msg_get_header (TNY_MSG (part));
281 filename = tny_header_dup_subject (header);
282 g_object_unref (header);
284 filename = g_strdup (tny_mime_part_get_filename (part));
286 if ((filename == NULL) || (filename[0] == '\0')) {
288 filename = g_strdup (_("mail_va_no_subject"));
290 gtk_list_store_append (GTK_LIST_STORE (model), &iter);
291 gtk_list_store_set (GTK_LIST_STORE (model), &iter, 0, filename, 1, part, -1);
292 attachments_added ++;
295 g_object_unref (part);
297 g_object_unref (iterator);
299 selector = GTK_WIDGET (hildon_touch_selector_new ());
300 renderer = gtk_cell_renderer_text_new ();
301 g_object_set((GObject *) renderer, "ellipsize", PANGO_ELLIPSIZE_END, NULL);
302 hildon_touch_selector_append_column ((HildonTouchSelector *) selector, model, renderer,
304 hildon_touch_selector_set_column_selection_mode ((HildonTouchSelector *) selector,
305 HILDON_TOUCH_SELECTOR_SELECTION_MODE_MULTIPLE);
307 dialog = hildon_picker_dialog_new (window);
308 gtk_window_set_title (GTK_WINDOW (dialog), (attachments_added > 1)?
309 _("mcen_ti_select_attachments_title"):_("mcen_ti_select_attachment_title"));
310 hildon_picker_dialog_set_selector (HILDON_PICKER_DIALOG (dialog), (HildonTouchSelector *) selector);
311 hildon_touch_selector_unselect_all ((HildonTouchSelector *) selector, 0);
312 hildon_picker_dialog_set_done_label (HILDON_PICKER_DIALOG (dialog), _HL("wdgt_bd_done"));
314 response = gtk_dialog_run (GTK_DIALOG (dialog));
316 if (response == GTK_RESPONSE_OK) {
317 GList *selected_rows, *node;
319 tny_list_remove_matches (att_list, match_all, NULL);
320 selected_rows = hildon_touch_selector_get_selected_rows ((HildonTouchSelector *) selector, 0);
321 for (node = selected_rows; node != NULL; node = g_list_next (node)) {
326 path = (GtkTreePath *) node->data;
327 gtk_tree_model_get_iter (model, &iter, path);
328 gtk_tree_model_get (model, &iter, 1, &selected, -1);
329 tny_list_append (att_list, selected);
331 if (tny_list_get_length (att_list) == 0)
334 g_list_foreach (selected_rows, (GFunc) gtk_tree_path_free, NULL);
335 g_list_free (selected_rows);
340 gtk_widget_destroy (dialog);
342 g_object_unref (model);
347 #ifdef MODEST_PLATFORM_MAEMO
349 modest_maemo_utils_in_usb_mode ()
351 return modest_conf_get_bool (modest_runtime_get_conf (), INTERNAL_MMC_USB_MODE, NULL);
355 #ifdef MODEST_USE_IPHB
357 typedef struct _ModestHeartbeatSource {
362 } ModestHeartbeatSource;
364 static gboolean modest_heartbeat_prepare (GSource* source, gint *timeout)
371 modest_heartbeat_check(GSource* source)
373 return ((ModestHeartbeatSource *) source)->poll.revents != 0;
376 static gboolean modest_heartbeat_dispatch (GSource *source, GSourceFunc callback, gpointer userdata)
378 if (callback(userdata))
380 ModestHeartbeatSource *hb_source = (ModestHeartbeatSource *) source;
382 g_source_remove_poll (source, &(hb_source->poll));
384 int min = MAX(hb_source->interval - 30, 5);
385 iphb_wait(hb_source->iphb, min, min + 60, 0);
387 hb_source->poll.fd = iphb_get_fd(hb_source->iphb);
388 hb_source->poll.events = G_IO_IN;
389 hb_source->poll.revents = 0;
391 g_source_add_poll(source, &(hb_source->poll));
400 modest_heartbeat_finalize (GSource* source)
402 ModestHeartbeatSource* hb_source = (ModestHeartbeatSource *) source;
403 hb_source->iphb = iphb_close(hb_source->iphb);
406 GSourceFuncs modest_heartbeat_funcs =
408 modest_heartbeat_prepare,
409 modest_heartbeat_check,
410 modest_heartbeat_dispatch,
411 modest_heartbeat_finalize,
417 modest_heartbeat_source_new (void)
420 ModestHeartbeatSource *hb_source;
427 iphb = iphb_open (&hb_interval);
431 source = g_source_new (&modest_heartbeat_funcs, sizeof (ModestHeartbeatSource));
432 hb_source = (ModestHeartbeatSource *) source;
433 g_source_set_priority (source, G_PRIORITY_DEFAULT_IDLE);
434 hb_source->iphb = iphb;
435 hb_source->interval = hb_interval;
437 min = MAX(hb_interval - 30, 5);
438 iphb_wait(hb_source->iphb, min, min + 60, 0);
440 hb_source->poll.fd = iphb_get_fd(hb_source->iphb);
441 hb_source->poll.events = G_IO_IN;
442 hb_source->poll.revents = 0;
444 g_source_add_poll(source, &(hb_source->poll));
446 source = g_idle_source_new ();
453 modest_heartbeat_add (GSourceFunc function,
459 source = modest_heartbeat_source_new ();
460 g_source_set_callback (source, function, userdata, NULL);
461 id = g_source_attach (source, NULL);
462 g_source_unref (source);
469 modest_heartbeat_add (GSourceFunc function,
472 return g_idle_add (function, userdata);