Ensure hildon-home will not start in safe mode
[conv-inbox] / src / el-home-applet.c
1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- */
2 /*
3  *  Copyright (C) 2009 Artem Garmash. All rights reserved.
4  *
5  *  This program is free software; you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation; either version 2 of the License, or
8  *  (at your option) any later version.
9  *
10  *  This program is distributed in the hope that it will be useful,
11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  *  GNU General Public License for more details.
14  *
15  *  You should have received a copy of the GNU General Public License
16  *  along with this program; if not, write to the Free Software
17  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18  *
19  * Contact: Artem Garmash <artemgarmash@gmail.com>
20  *
21  */
22
23 #include "config.h"
24 #include "el-home-applet.h"
25 #include <libintl.h>
26 #include <hildon/hildon.h>
27 #include <rtcom-eventlogger/eventlogger.h>
28 #include <sqlite3.h>
29 #include <string.h>
30 #include <gconf/gconf-client.h>
31 #include <libosso-abook/osso-abook-init.h>
32 #include <libosso-abook/osso-abook-aggregator.h>
33 #include <libosso-abook/osso-abook-contact.h>
34 #include <libosso-abook/osso-abook-waitable.h>
35 #include <libosso-abook/osso-abook-presence.h>
36 #include <libosso-abook/osso-abook-touch-contact-starter.h>
37 #include <libosso-abook/osso-abook-temporary-contact-dialog.h>
38 #include <libosso-abook/osso-abook-account-manager.h>
39
40 #include <telepathy-glib/interfaces.h>
41 #include <telepathy-glib/dbus.h>
42 #include <rtcom-telepathy-glib/extensions.h>
43
44 #define EL_HOME_APPLET_GET_PRIVATE(obj) ( \
45         G_TYPE_INSTANCE_GET_PRIVATE (obj, \
46                 EL_TYPE_HOME_APPLET, ELHomeAppletPrivate))
47
48 #define BOX_WIDTH 352
49 #define BOX_HEIGHT 284
50
51 #define CONTENT_OFFSET_X HILDON_MARGIN_HALF
52 #define CONTENT_OFFSET_Y_TOP 4*HILDON_MARGIN_HALF
53 #define CONTENT_OFFSET_Y_BOTTOM HILDON_MARGIN_HALF
54 #define C_WIDTH (BOX_WIDTH - 2*CONTENT_OFFSET_X)
55 #define C_HEIGHT (BOX_HEIGHT - (CONTENT_OFFSET_Y_TOP + CONTENT_OFFSET_Y_BOTTOM))
56
57 #define HEADER_HEIGHT 48
58 #define FOOTER_HEIGHT 24
59 #define FOOTER_HEIGHT_PRESS FOOTER_HEIGHT*2 /* approx, used only for checking clicks */
60 #define FOOTER_WIDTH C_WIDTH/4
61 #define FOOTER_WIDTH_PRESS (FOOTER_WIDTH + FOOTER_WIDTH/2) /* approx, used only for checking clicks, bigger than controls */
62
63 #define MESSAGE_HEIGHT (C_HEIGHT - HEADER_HEIGHT - FOOTER_HEIGHT)
64 #define MESSAGE_WIDTH (C_WIDTH - 2*HILDON_MARGIN_DEFAULT)
65
66 #define AVATAR_SIZE HILDON_ICON_PIXEL_SIZE_THUMB
67
68 #define AVATAR_X (C_WIDTH - AVATAR_SIZE - HILDON_MARGIN_DEFAULT)
69 #define AVATAR_Y 3*HILDON_MARGIN_HALF
70
71 #define BOX_RADIOUS 20
72
73 #define SCROLL_PERIOD 100 /* ms */
74 #define SCROLL_STEP 1 /* pixel */
75 #define TEXT_Y_OFFSET (HEADER_HEIGHT + HILDON_MARGIN_HALF)
76
77 #define NOTIFICATION_UI_DBUS_NAME     "org.freedesktop.Telepathy.Client.NotificationUI"
78 #define NOTIFICATION_UI_DBUS_PATH     "/org/freedesktop/Telepathy/Client/NotificationUI"
79 #define NOTIFICATION_UI_DBUS_IFACE    "com.nokia.RtcomNotificationUi"
80
81 #define CONVERSATIONS_UI_DBUS_NAME     "com.nokia.MessagingUI"
82 #define CONVERSATIONS_UI_DBUS_PATH     "/com/nokia/MessagingUI"
83 #define CONVERSATIONS_UI_DBUS_IFACE    "com.nokia.MessagingUI"
84
85 static const gchar *conv_services[] = {"RTCOM_EL_SERVICE_SMS",
86                                        "RTCOM_EL_SERVICE_CHAT",
87                                        NULL};
88 static const gchar *conv_event_types[] = {"RTCOM_EL_EVENTTYPE_SMS_INBOUND",
89                                           "RTCOM_EL_EVENTTYPE_CHAT_INBOUND",
90                                           NULL};
91
92 typedef enum {
93         SELECTED_NONE,
94         SELECTED_HEADER,
95         SELECTED_BODY,
96         SELECTED_FOOTER
97 } WidgetActiveSelection;
98
99 struct _ELHomeAppletPrivate
100 {
101         RTComEl *eventlogger;
102
103         GtkWidget *sender;
104         GtkWidget *icon;
105         GtkWidget *unread;
106         GtkWidget *received;
107         GtkWidget *cut_message;
108
109         /* empty view*/
110         GtkWidget *empty;
111         GtkWidget *sms_total;
112         GtkWidget *chat_total;
113
114         gchar *message;
115         gint event_id;
116
117         WidgetActiveSelection active;
118
119         guint unread_count;
120
121         struct {
122                 float red;
123                 float green;
124                 float blue;
125         } active_color;
126         PangoFontDescription *font_desc;
127
128         GdkPixbuf *avatar_pixbuf;
129         GdkPixbuf *presence_pixbuf;
130
131         guint idle_id;
132
133         cairo_surface_t *message_surface;
134
135         gboolean scroll_on_click;
136         gint scroll_offset;
137         gint hidden_message_height;
138         guint scroll_anim_id;
139
140         OssoABookRoster *aggregator;
141         OssoABookWaitableClosure *aggregator_ready_closure;
142         gchar *contact_id;
143         gchar *remote_id;
144         gchar *local_id;
145         gchar *group_uid;
146         OssoABookContact *contact;
147
148         gboolean time_fmt_24h;
149
150         guint init_timer;
151 };
152
153 HD_DEFINE_PLUGIN_MODULE (ELHomeApplet, el_home_applet, HD_TYPE_HOME_PLUGIN_ITEM);
154
155 const gchar* g_module_check_init (GModule *module);
156 const gchar*
157 g_module_check_init (GModule *module)
158 {
159         g_module_make_resident (module);
160         return NULL;
161 }
162
163 static void
164 el_home_applet_class_finalize (ELHomeAppletClass *klass)
165 {
166 }
167
168 static void
169 el_home_applet_realize (GtkWidget *widget)
170 {
171         GdkScreen *screen;
172
173         screen = gtk_widget_get_screen (widget);
174         gtk_widget_set_colormap (widget,
175                                  gdk_screen_get_rgba_colormap (screen));
176
177         gtk_widget_set_app_paintable (widget,
178                                       TRUE);
179
180         GTK_WIDGET_CLASS (el_home_applet_parent_class)->realize (widget);
181 }
182
183 enum {
184         ROUND_CORNER_TL = 1,
185         ROUND_CORNER_TR = 1<<1,
186         ROUND_CORNER_BL = 1<<2,
187         ROUND_CORNER_BR = 1<<3,
188         ROUND_CORNER_ALL = ROUND_CORNER_TL | ROUND_CORNER_TR |
189                            ROUND_CORNER_BL | ROUND_CORNER_BR
190 };
191
192 /**
193  * Draw rectangle with optional round corners.
194  *
195  * @x
196  * @y
197  * @w width
198  * @h height
199  * @r round corner radious
200  * @round_corners define which corners draw round, ROUND_CORNER_TL,
201  *                ROUND_CORNER_TR, ROUND_CORNER_BL, ROUND_CORNER_BR
202  */
203 static void
204 rounded_rectangle (cairo_t *cr,
205                    double x,
206                    double y,
207                    double w,
208                    double h,
209                    double r,
210                    guint round_corners)
211 {
212         if (round_corners & ROUND_CORNER_TL)
213                 cairo_move_to (cr, x + r, y);
214         else
215                 cairo_move_to (cr, x, y);
216
217         if (round_corners & ROUND_CORNER_TR) {
218                 cairo_line_to (cr, x + w - r, y);
219                 cairo_rel_curve_to (cr,
220                                     r, 0,
221                                     r, 0,
222                                     r, r);
223         }
224         else
225                 cairo_line_to (cr, x + w, y);
226
227         if (round_corners & ROUND_CORNER_BR) {
228                 cairo_line_to (cr, x + w, y + h - r);
229                 cairo_rel_curve_to (cr,
230                                     0, r,
231                                     0, r,
232                                     -r, r);
233         }
234         else
235                 cairo_line_to (cr, x + w, y + h);
236
237         if (round_corners & ROUND_CORNER_BL) {
238                 cairo_line_to (cr, x + r, y + h);
239                 cairo_rel_curve_to (cr,
240                                     -r, 0,
241                                     -r, 0,
242                                     -r, -r);
243         }
244         else
245                 cairo_line_to (cr, x, y + h);
246
247         if (round_corners & ROUND_CORNER_TL) {
248                 cairo_line_to (cr, x, y + r);
249                 cairo_rel_curve_to (cr,
250                                     0, -r,
251                                     0, -r,
252                                     r, -r);
253         }
254         else
255                 cairo_line_to (cr, x, y);
256 }
257
258 static cairo_surface_t*
259 draw_text (cairo_t              *cr,
260            PangoFontDescription *desc,
261            const gchar          *text,
262            gint                  width,
263            gint                 *height)
264 {
265         PangoLayout *layout;
266         PangoRectangle extent;
267
268         cairo_surface_t *gdk_surface, *result_surface;
269         cairo_t *msg_cr;
270
271         /* Create a PangoLayout, set the font and text */
272         layout = pango_cairo_create_layout (cr);
273         pango_layout_set_text (layout,
274                                text,
275                                -1);
276         pango_layout_set_font_description (layout, desc);
277
278         pango_layout_set_wrap (layout, PANGO_WRAP_WORD_CHAR);
279         pango_layout_set_width (layout, PANGO_SCALE*width);
280
281         pango_layout_get_pixel_extents (layout, NULL, &extent);
282         *height = extent.height;
283
284         gdk_surface = cairo_get_target (cr);
285         result_surface = cairo_surface_create_similar
286                 (gdk_surface,
287                  CAIRO_CONTENT_COLOR_ALPHA,
288                  width,
289                  extent.height);
290         msg_cr = cairo_create (result_surface);
291
292         pango_cairo_update_layout (msg_cr, layout);
293         /* draw shadow */
294         cairo_move_to (msg_cr, 1, 1);
295         cairo_set_source_rgba (msg_cr, 0.2, 0.2, 0.2, 0.8);
296         pango_cairo_show_layout (msg_cr, layout);
297
298         /* draw fg */
299         cairo_move_to (msg_cr, 0, 0);
300         cairo_set_source_rgba (msg_cr, 1.0, 1.0, 1.0, 1.0);
301         pango_cairo_show_layout (msg_cr, layout);
302
303         cairo_destroy (msg_cr);
304         g_object_unref (layout);
305
306         return result_surface;
307 }
308
309 static gboolean
310 stop_scroll_anim (ELHomeAppletPrivate *priv)
311 {
312         gboolean result = priv->scroll_anim_id > 0;
313
314         if (result) {
315                 g_source_remove (priv->scroll_anim_id);
316                 priv->scroll_anim_id = 0;
317                 priv->scroll_on_click = FALSE;
318                 gtk_widget_hide (priv->cut_message);
319         }
320
321         return result;
322 }
323
324 static void
325 style_set_cb (GtkWidget *widget,
326               GtkStyle  *previous_style,
327               ELHomeApplet *self)
328 {
329         ELHomeAppletPrivate *priv = self->priv;
330         GdkColor color;
331         GtkStyle *font_style;
332
333         font_style = gtk_rc_get_style_by_paths (gtk_widget_get_settings (widget),
334                                                 "SystemFont",
335                                                 NULL,
336                                                 G_TYPE_NONE);
337         if (font_style && font_style->font_desc) {
338                 if (priv->font_desc)
339                         pango_font_description_free (priv->font_desc);
340                 priv->font_desc = pango_font_description_copy (font_style->font_desc);
341         }
342
343         if (gtk_style_lookup_color (widget->style,
344                                     "ActiveTextColor",
345                                     &color)) {
346                 priv->active_color.red = color.red/(float)G_MAXUINT16;
347                 priv->active_color.green = color.green/(float)G_MAXUINT16;
348                 priv->active_color.blue = color.blue/(float)G_MAXUINT16;
349         }
350 }
351
352 static void
353 reset_scroll (ELHomeApplet *self)
354 {
355         ELHomeAppletPrivate *priv = self->priv;
356
357         if (stop_scroll_anim (self->priv)) {
358                 priv->scroll_on_click = TRUE;/* priv->scroll_offset; */
359                 priv->scroll_offset = 0;
360                 if (priv->scroll_on_click)
361                         gtk_widget_show (priv->cut_message);
362         }
363 }
364
365 static void
366 notify_on_current_desktop (GObject      *object,
367                            GParamSpec   *unused G_GNUC_UNUSED,
368                            ELHomeApplet *self)
369 {
370         gboolean on;
371
372         g_object_get (object, "is-on-current-desktop", &on, NULL);
373         if (!on) {
374                 reset_scroll (self);
375                 gtk_widget_queue_draw (GTK_WIDGET (self));
376         }
377 }
378
379 static gboolean
380 expose_event (GtkWidget *self, GdkEventExpose *event)
381 {
382         ELHomeAppletPrivate *priv = EL_HOME_APPLET(self)->priv;
383         cairo_t *cr;
384         cairo_pattern_t *grad;
385
386         cr = gdk_cairo_create (self->window);
387         gdk_cairo_region (cr, event->region);
388         cairo_clip (cr);
389
390         cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
391
392         /* draw bound box */
393         cairo_set_source_rgba (cr, 0.4f, 0.4f, 0.4f, 0.1f);
394         cairo_set_line_width (cr, 3.0f);
395
396         rounded_rectangle (cr,
397                            CONTENT_OFFSET_X,
398                            CONTENT_OFFSET_Y_TOP,
399                            BOX_WIDTH - 2*CONTENT_OFFSET_X,
400                            BOX_HEIGHT - (CONTENT_OFFSET_Y_TOP + CONTENT_OFFSET_Y_BOTTOM),
401                            BOX_RADIOUS,
402                            ROUND_CORNER_ALL);
403
404         cairo_close_path (cr);
405         cairo_stroke (cr);
406
407         /* draw header */
408         cairo_set_line_width (cr, 1.0f);
409
410         cairo_translate (cr, CONTENT_OFFSET_X, CONTENT_OFFSET_Y_TOP);
411         rounded_rectangle (cr,
412                            0, 0,
413                            C_WIDTH, HEADER_HEIGHT,
414                            BOX_RADIOUS,
415                            ROUND_CORNER_TL | ROUND_CORNER_TR);
416         cairo_close_path (cr);
417
418         switch (priv->active) {
419         case SELECTED_HEADER:
420                 cairo_set_source_rgba (cr,
421                                        priv->active_color.red,
422                                        priv->active_color.green,
423                                        priv->active_color.blue,
424                                        0.8f);
425                 break;
426         default:
427                 cairo_set_source_rgba (cr, 0.2f, 0.2f, 0.2f, 0.8f);
428         }
429
430         cairo_fill (cr);
431
432         cairo_move_to (cr, 0, HEADER_HEIGHT);
433         cairo_line_to (cr, C_WIDTH, HEADER_HEIGHT);
434         cairo_set_source_rgba (cr,
435                                priv->active_color.red,
436                                priv->active_color.green,
437                                priv->active_color.blue,
438                                1.0f);
439         cairo_stroke (cr);
440
441         /* draw body */
442         if (!priv->message) {
443                 rounded_rectangle (cr,
444                                    0, HEADER_HEIGHT,
445                                    C_WIDTH, C_HEIGHT,
446                                    BOX_RADIOUS,
447                                    ROUND_CORNER_BL | ROUND_CORNER_BR);
448                 cairo_close_path (cr);
449         }
450         else
451                 cairo_rectangle (cr, 0, HEADER_HEIGHT,
452                                  C_WIDTH, MESSAGE_HEIGHT);
453
454         /* draw body filling depending on (in)active state */
455         grad = cairo_pattern_create_linear (0, HEADER_HEIGHT,
456                                             0, C_HEIGHT - FOOTER_HEIGHT);
457
458         switch (priv->active) {
459         case SELECTED_BODY:
460                 cairo_pattern_add_color_stop_rgba (grad,
461                                                    0.5f,
462                                                    priv->active_color.red,
463                                                    priv->active_color.green,
464                                                    priv->active_color.blue,
465                                                    0.8f);
466                 cairo_pattern_add_color_stop_rgba (grad,
467                                                    1.0f,
468                                                    priv->active_color.red/2,
469                                                    priv->active_color.green/2,
470                                                    priv->active_color.blue/2,
471                                                    0.8f);
472                 break;
473         default:
474                 cairo_pattern_add_color_stop_rgba (grad, 0.5f,
475                                                    0.4f, 0.4f, 0.4f, 0.8f);
476                 cairo_pattern_add_color_stop_rgba (grad, 1.0f,
477                                                    0.2f, 0.2f, 0.2f, 0.8f);
478         }
479
480         cairo_set_source (cr, grad);
481         cairo_fill (cr);
482
483         cairo_pattern_destroy (grad);
484
485         /* draw avatar */
486         if (priv->avatar_pixbuf) {
487                 rounded_rectangle (cr,
488                                    AVATAR_X, -AVATAR_Y,
489                                    AVATAR_SIZE, AVATAR_SIZE,
490                                    BOX_RADIOUS,
491                                    ROUND_CORNER_ALL);
492                 cairo_close_path (cr);
493
494                 gdk_cairo_set_source_pixbuf (cr,
495                                              priv->avatar_pixbuf,
496                                              AVATAR_X,
497                                              -AVATAR_Y);
498                 cairo_fill_preserve (cr);
499
500                 cairo_set_source_rgba (cr,
501                                        priv->active_color.red,
502                                        priv->active_color.green,
503                                        priv->active_color.blue,
504                                        1.0f);
505                 cairo_stroke (cr);
506         }
507         if (priv->presence_pixbuf) {
508                 guint x = C_WIDTH - HILDON_ICON_PIXEL_SIZE_XSMALL - HILDON_MARGIN_DEFAULT;
509                 guint y = (HEADER_HEIGHT - HILDON_ICON_PIXEL_SIZE_XSMALL)/2;
510
511                 if (priv->avatar_pixbuf)
512                         x -= AVATAR_SIZE + HILDON_MARGIN_DEFAULT;
513
514                 cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
515                 gdk_cairo_set_source_pixbuf (cr,
516                                              priv->presence_pixbuf,
517                                              x,
518                                              y);
519                 cairo_paint (cr);
520         }
521
522         if (priv->message) {
523                 /* draw footer unread part bg */
524                 rounded_rectangle (cr,
525                                    0, C_HEIGHT - FOOTER_HEIGHT,
526                                    FOOTER_WIDTH, FOOTER_HEIGHT,
527                                    BOX_RADIOUS,
528                                    ROUND_CORNER_BL);
529                 cairo_close_path (cr);
530
531                 /* draw body filling depending on (in)active state */
532                 switch (priv->active) {
533                 case SELECTED_FOOTER:
534                         cairo_set_source_rgba (cr,
535                                                priv->active_color.red,
536                                                priv->active_color.green,
537                                                priv->active_color.blue,
538                                                0.8f);
539                         break;
540                 default:
541                         cairo_set_source_rgba (cr, 0.1f, 0.1f, 0.1f, 0.9f);
542                 }
543                 cairo_fill (cr);
544
545                 /* draw footer received part bg */
546                 rounded_rectangle (cr,
547                                    FOOTER_WIDTH, C_HEIGHT - FOOTER_HEIGHT,
548                                    C_WIDTH - FOOTER_WIDTH, FOOTER_HEIGHT,
549                                    BOX_RADIOUS,
550                                    ROUND_CORNER_BR);
551                 cairo_close_path (cr);
552
553                 cairo_set_source_rgba (cr, 0.2f, 0.2f, 0.2f, 0.8f);
554                 cairo_fill (cr);
555
556                 /* draw message */
557                 if (!priv->message_surface) {
558                         gint height;
559
560                         priv->message_surface = draw_text (cr,
561                                                            priv->font_desc,
562                                                            priv->message,
563                                                            MESSAGE_WIDTH,
564                                                            &height);
565
566                         priv->hidden_message_height = height - MESSAGE_HEIGHT;
567                         priv->scroll_on_click = priv->hidden_message_height > 0;
568                         if (priv->scroll_on_click)
569                                 gtk_widget_show (priv->cut_message);
570                 }
571
572                 cairo_rectangle (cr,
573                                  2*CONTENT_OFFSET_X,
574                                  TEXT_Y_OFFSET,
575                                  MESSAGE_WIDTH,
576                                  MESSAGE_HEIGHT);
577                 cairo_clip (cr);
578
579                 cairo_set_source_surface (cr,
580                                           priv->message_surface,
581                                           2*CONTENT_OFFSET_X,
582                                           TEXT_Y_OFFSET - priv->scroll_offset);
583                 cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
584                 cairo_paint (cr);
585         }
586
587         cairo_destroy (cr);
588
589         return GTK_WIDGET_CLASS (el_home_applet_parent_class)->expose_event (self, event);
590 }
591
592 static void
593 resize_sender (ELHomeAppletPrivate *priv)
594 {
595         guint width = C_WIDTH;
596
597         if (priv->avatar_pixbuf) {
598                 width -= AVATAR_SIZE + HILDON_MARGIN_DEFAULT;
599         }
600
601         if (priv->presence_pixbuf) {
602                 width -= HILDON_ICON_PIXEL_SIZE_XSMALL + HILDON_MARGIN_DEFAULT;
603         }
604
605         gtk_widget_set_size_request (priv->sender,
606                                      width,
607                                      HILDON_ICON_PIXEL_SIZE_THUMB);
608 }
609
610 static void
611 update_presence_pixbuf (ELHomeApplet *self,
612                         OssoABookPresence *presence)
613 {
614         ELHomeAppletPrivate *priv = self->priv;
615         const gchar *icon_name = osso_abook_presence_get_icon_name (presence);
616         gboolean resize = !!priv->presence_pixbuf ^ !!icon_name; /* logical via bit XOR */
617
618         if (priv->presence_pixbuf) {
619                 g_object_unref (priv->presence_pixbuf);
620                 priv->presence_pixbuf = NULL;
621         }
622
623         if (icon_name)
624                 priv->presence_pixbuf = gtk_icon_theme_load_icon
625                         (gtk_icon_theme_get_default (),
626                          icon_name,
627                          HILDON_ICON_PIXEL_SIZE_XSMALL,
628                          0, NULL);
629         if (resize)
630                 resize_sender (priv);
631
632         gtk_widget_queue_draw (GTK_WIDGET (self));
633 }
634
635 static void
636 presence_updated (OssoABookPresence *presence,
637                   GParamSpec *spec,
638                   gpointer *user_data)
639 {
640         ELHomeApplet *self = EL_HOME_APPLET(user_data);
641
642         if (!OSSO_ABOOK_IS_CONTACT(self->priv->contact))
643                 return;
644
645         update_presence_pixbuf (self, presence);
646 }
647
648 static void
649 show_contact (ELHomeApplet *self)
650 {
651         ELHomeAppletPrivate *priv = self->priv;
652
653         g_return_if_fail (priv->contact);
654
655         gtk_label_set_text (GTK_LABEL (priv->sender),
656                             osso_abook_contact_get_display_name (priv->contact));
657         resize_sender (priv);
658         gtk_widget_queue_draw (GTK_WIDGET (self));
659 }
660
661 static void
662 resolve_contact (ELHomeApplet *self)
663 {
664         ELHomeAppletPrivate *priv = self->priv;
665         GList *contacts = NULL;
666
667         if (priv->contact_id) {
668                 contacts = osso_abook_aggregator_lookup
669                         (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
670                          priv->contact_id);
671         }
672         else if (priv->local_id && priv->remote_id) {
673                 if (g_strcmp0 (priv->local_id, "ring/tel/ring" == 0)) {
674                         contacts = osso_abook_aggregator_find_contacts_for_phone_number
675                                 (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
676                                  priv->remote_id,
677                                  TRUE);
678                 }
679                 else {
680                         McAccount *account;
681                         account = osso_abook_account_manager_lookup_by_name
682                                 (NULL,
683                                  priv->local_id);
684                         if (account) {
685                                 contacts = osso_abook_aggregator_find_contacts_for_im_contact
686                                         (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
687                                          priv->remote_id,
688                                          account);
689                         }
690                 }
691         }
692
693         if (contacts && contacts->data) {
694                 priv->contact = g_object_ref (OSSO_ABOOK_CONTACT (contacts->data));
695                 g_signal_connect (priv->contact,
696                                   "notify::presence-status",
697                                   G_CALLBACK (presence_updated),
698                                   self);
699                 priv->avatar_pixbuf = osso_abook_avatar_get_image_scaled
700                         (OSSO_ABOOK_AVATAR (priv->contact),
701                          HILDON_ICON_PIXEL_SIZE_THUMB,
702                          HILDON_ICON_PIXEL_SIZE_THUMB,
703                          TRUE);
704                 update_presence_pixbuf (self,
705                                         OSSO_ABOOK_PRESENCE (priv->contact));
706                 show_contact (self);
707         }
708 }
709
710 static void
711 contacts_added (OssoABookRoster  *roster,
712                 OssoABookContact **contacts,
713                 gpointer          userdata)
714 {
715         ELHomeApplet *self = EL_HOME_APPLET (userdata);
716         ELHomeAppletPrivate *priv = self->priv;
717
718         if (!priv->contact)
719                 resolve_contact (self);
720 }
721
722 static void
723 reset_contact (ELHomeApplet *self, gboolean resize)
724 {
725         ELHomeAppletPrivate *priv = self->priv;
726
727         if (priv->avatar_pixbuf) {
728                 g_object_unref (priv->avatar_pixbuf);
729                 priv->avatar_pixbuf = NULL;
730         }
731
732         if (priv->presence_pixbuf) {
733                 g_object_unref (priv->presence_pixbuf);
734                 priv->presence_pixbuf = NULL;
735         }
736
737         if (priv->contact) {
738                 g_signal_handlers_disconnect_by_func (priv->contact,
739                                                       presence_updated,
740                                                       self);
741                 g_object_unref (priv->contact);
742                 priv->contact = NULL;
743         }
744
745         if (resize)
746                 resize_sender (priv);
747 }
748
749 static void
750 contacts_removed (OssoABookRoster *roster,
751                   const gchar     **ids,
752                   gpointer         userdata)
753 {
754         ELHomeApplet *self = EL_HOME_APPLET (userdata);
755         ELHomeAppletPrivate *priv = self->priv;
756
757         if (priv->contact) {
758                 const gchar **contact_id;
759                 const gchar *uid = osso_abook_contact_get_uid (priv->contact);
760
761                 for (contact_id = ids; *contact_id; contact_id++) {
762                         if (strcmp (*contact_id, priv->contact_id) == 0) {
763                                 reset_contact (self, TRUE);
764
765                                 gtk_widget_queue_draw (GTK_WIDGET (self));
766                                 return;
767                         }
768                         if (strcmp (*contact_id, uid) == 0) {
769                                 reset_contact (self, TRUE);
770                                 resolve_contact (self);
771                                 gtk_widget_queue_draw (GTK_WIDGET (self));
772                                 return;
773                         }
774                 }
775         }
776 }
777
778 static void
779 clean_contact (ELHomeApplet *self, gboolean resize)
780 {
781         ELHomeAppletPrivate *priv = self->priv;
782
783         reset_contact (self, resize);
784
785         if (priv->aggregator) {
786                 if (priv->aggregator_ready_closure){
787                         osso_abook_waitable_cancel (OSSO_ABOOK_WAITABLE (priv->aggregator),
788                                                     priv->aggregator_ready_closure);
789                         priv->aggregator_ready_closure = NULL;
790                 }
791                 g_signal_handlers_disconnect_by_func (priv->aggregator,
792                                                       contacts_added,
793                                                       self);
794                 g_signal_handlers_disconnect_by_func (priv->aggregator,
795                                                       contacts_removed,
796                                                       self);
797                 osso_abook_roster_stop (priv->aggregator);
798                 g_object_unref (priv->aggregator);
799                 priv->aggregator = NULL;
800         }
801 }
802
803 static void
804 clean_state (ELHomeApplet *self)
805 {
806         ELHomeAppletPrivate *priv = self->priv;
807
808         if (priv->message) {
809                 g_free (priv->message);
810                 priv->message = NULL;
811         }
812
813         if (priv->contact_id) {
814                 g_free (priv->contact_id);
815                 priv->contact_id = NULL;
816         }
817         if (priv->local_id) {
818                 g_free (priv->local_id);
819                 priv->local_id = NULL;
820         }
821         if (priv->remote_id) {
822                 g_free (priv->remote_id);
823                 priv->remote_id = NULL;
824         }
825         if (priv->group_uid) {
826                 g_free (priv->group_uid);
827                 priv->group_uid = NULL;
828         }
829 }
830
831 static void
832 dispose (GObject *self)
833 {
834         ELHomeAppletPrivate *priv = EL_HOME_APPLET(self)->priv;
835
836         if (priv->init_timer) {
837                 g_source_remove (priv->init_timer);
838                 priv->init_timer = 0;
839         }
840
841         stop_scroll_anim (priv);
842         if (priv->idle_id) {
843                 g_source_remove (priv->idle_id);
844                 priv->idle_id = 0;
845         }
846         if (priv->eventlogger) {
847                 g_object_unref (priv->eventlogger);
848                 priv->eventlogger = NULL;
849         }
850         if (priv->font_desc) {
851                 pango_font_description_free (priv->font_desc);
852                 priv->font_desc = NULL;
853         }
854
855         clean_state (EL_HOME_APPLET (self));
856         clean_contact (EL_HOME_APPLET (self), FALSE);
857
858         G_OBJECT_CLASS (el_home_applet_parent_class)->dispose (self);
859 }
860
861 static void
862 finalize (GObject *self)
863 {
864         G_OBJECT_CLASS (el_home_applet_parent_class)->finalize (self);
865 }
866
867 static void
868 aggregator_ready_cb (OssoABookWaitable *waitable,
869                      const GError      *error,
870                      gpointer           userdata)
871 {
872         ELHomeApplet *self = EL_HOME_APPLET (userdata);
873         ELHomeAppletPrivate *priv = self->priv;
874
875         priv->aggregator_ready_closure = NULL;
876
877         if (error) {
878                 g_warning ("Failed to create aggregator: %s", error->message);
879                 return;
880         }
881
882         g_signal_connect (priv->aggregator,
883                           "contacts-added",
884                           G_CALLBACK (contacts_added),
885                           self);
886         g_signal_connect (priv->aggregator,
887                           "contacts-removed",
888                           G_CALLBACK (contacts_removed),
889                           self);
890
891         resolve_contact (self);
892 }
893
894 static void
895 start_aggregator (ELHomeApplet *self)
896 {
897         ELHomeAppletPrivate *priv = self->priv;
898         EBookQuery *query = NULL;
899         GError *error = NULL;
900
901         if (priv->local_id && priv->remote_id) {
902                 const gchar *vcard = osso_abook_account_manager_get_vcard_field
903                         (NULL, priv->local_id);
904                 if (vcard)
905                         query = e_book_query_vcard_field_test (vcard,
906                                                                E_BOOK_QUERY_IS,
907                                                                priv->remote_id);
908                 else
909                         query = e_book_query_any_field_contains (priv->remote_id);
910         }
911
912         if (query) {
913                 priv->aggregator = osso_abook_aggregator_new_with_query (NULL,
914                                                                          query,
915                                                                          NULL,
916                                                                          1,
917                                                                          &error);
918                 e_book_query_unref (query);
919         }
920         if (error) {
921                 g_warning ("Failed to create aggregator: %s", error->message);
922                 g_error_free (error);
923                 return;
924         }
925
926         if (priv->aggregator) {
927                 priv->aggregator_ready_closure = osso_abook_waitable_call_when_ready
928                         (OSSO_ABOOK_WAITABLE (priv->aggregator),
929                          aggregator_ready_cb,
930                          self, NULL);
931
932                 osso_abook_roster_start (priv->aggregator);
933         }
934 }
935
936 static gchar*
937 format_time (time_t t, gboolean time_fmt_24h)
938 {
939         static const guint RESULT_SIZE = 64;
940
941         time_t now;
942         struct tm now_tm, t_tm;
943         const gchar *time_format;
944         gchar *result = g_malloc0 (RESULT_SIZE);
945
946         now = time (NULL);
947         localtime_r (&now, &now_tm);
948         localtime_r (&t, &t_tm);
949
950         if (time_fmt_24h)
951                 time_format = "wdgt_va_24h_time";
952         else
953                 time_format = now_tm.tm_hour > 11 ?
954                         "wdgt_va_12h_time_pm":
955                         "wdgt_va_12h_time_am";
956
957         if ((now_tm.tm_year == t_tm.tm_year) &&
958             (now_tm.tm_mon  == t_tm.tm_mon) &&
959             (now_tm.tm_mday == t_tm.tm_mday))
960                 strftime (result,
961                           RESULT_SIZE,
962                           dgettext ("hildon-libs", time_format),
963                           &t_tm);
964         else {
965                 gchar *full_format = g_strdup_printf ("%s %s",
966                                                       dgettext ("hildon-libs", "wdgt_va_date"),
967                                                       dgettext ("hildon-libs", time_format));
968                 strftime (result, RESULT_SIZE, full_format, &t_tm);
969                 g_free (full_format);
970         }
971
972         return result;
973 }
974
975 static void
976 show_event (ELHomeApplet *self, RTComElIter *it)
977 {
978         ELHomeAppletPrivate *priv = self->priv;
979         const gchar *remote = NULL;
980         gchar *received = NULL;
981         GValueArray *event = NULL;
982
983         if (it && rtcom_el_iter_first (it)) {
984
985                 event = rtcom_el_iter_get_valuearray (it,
986                                                       "id",
987                                                       "start-time",
988                                                       "local-uid",
989                                                       "remote-uid",
990                                                       "remote-name",
991                                                       "remote-ebook-uid",
992                                                       "free-text",
993                                                       "group-uid",
994                                                       NULL);
995                 if (event) {
996                         time_t received_t;
997 #define _VARR_DUP_STR(array, i) g_value_dup_string (g_value_array_get_nth ((array), (i)))
998
999                         priv->event_id = g_value_get_int (g_value_array_get_nth (event, 0));
1000                         received_t = g_value_get_int (g_value_array_get_nth (event, 1));
1001                         received = format_time (received_t, priv->time_fmt_24h);
1002                         priv->local_id = _VARR_DUP_STR (event, 2);
1003                         priv->remote_id = _VARR_DUP_STR (event, 3);
1004                         if (priv->remote_id && priv->remote_id[0]) {
1005                                 remote = g_value_get_string (g_value_array_get_nth (event, 4));
1006                                 if (!remote)
1007                                         remote = priv->remote_id;
1008                                 priv->contact_id = _VARR_DUP_STR (event, 5);
1009                         }
1010                         else if (priv->remote_id) {
1011                                 g_free (priv->remote_id);
1012                                 priv->remote_id = NULL;
1013                         }
1014
1015                         priv->message = _VARR_DUP_STR (event, 6);
1016                         priv->group_uid = _VARR_DUP_STR (event, 7);
1017
1018 #undef _VARR_DUP_STR
1019                 }
1020         }
1021         else {
1022                 priv->event_id = -1;
1023         }
1024
1025         if (priv->message) {
1026                 gtk_widget_hide (priv->empty);
1027         }
1028         else {
1029                 gtk_widget_show (priv->empty);
1030         }
1031
1032         gtk_label_set_text (GTK_LABEL (priv->received), received);
1033
1034         if (remote)
1035                 gtk_label_set_text (GTK_LABEL (priv->sender), remote);
1036         else
1037                 gtk_label_set_text (GTK_LABEL (priv->sender), priv->remote_id);
1038
1039         stop_scroll_anim (priv);
1040         priv->scroll_offset = 0;
1041         if (priv->message_surface) {
1042                 cairo_surface_destroy (priv->message_surface);
1043                 priv->message_surface = NULL;
1044         }
1045
1046         if (event)
1047                 g_value_array_free (event);
1048
1049         gtk_widget_hide (priv->cut_message);
1050         gtk_widget_queue_draw (GTK_WIDGET (self));
1051 }
1052
1053 static RTComElIter*
1054 make_query (RTComEl *el, gint event_id)
1055 {
1056         RTComElQuery *query = NULL;
1057         RTComElIter *it = NULL;
1058
1059         query = rtcom_el_query_new (el);
1060         rtcom_el_query_set_limit (query, 1);
1061         if (event_id >= 0) {
1062                 rtcom_el_query_prepare (query,
1063                                         "id", event_id, RTCOM_EL_OP_EQUAL,
1064                                         NULL);
1065         }
1066         else {
1067                 rtcom_el_query_prepare (query,
1068                                         "is-read", FALSE, RTCOM_EL_OP_EQUAL,
1069                                         "service", conv_services, RTCOM_EL_OP_IN_STRV,
1070                                         "event-type", conv_event_types, RTCOM_EL_OP_IN_STRV,
1071                                         NULL);
1072         }
1073         it = rtcom_el_get_events (el, query);
1074         g_object_unref (query);
1075
1076         return it;
1077 }
1078
1079 static void
1080 update_unread_label (ELHomeApplet *self)
1081 {
1082         ELHomeAppletPrivate *priv = self->priv;
1083
1084         if (priv->unread_count > 0) {
1085                 gchar *text;
1086                 text = g_strdup_printf
1087                         ("%d<span foreground=\"red\" rise=\"5000\">*</span>",
1088                          priv->unread_count);
1089
1090                 gtk_label_set_markup (GTK_LABEL (priv->unread), text);
1091                 g_free (text);
1092         }
1093         else
1094                 gtk_label_set_text (GTK_LABEL (priv->unread), NULL);
1095 }
1096
1097 static gint
1098 query_unread_events (RTComEl *el)
1099 {
1100         sqlite3 *db;
1101         sqlite3_stmt *stmt;
1102         int ret;
1103         gint count = 0;
1104
1105         g_object_get (el, "db", &db, NULL);
1106
1107         if (sqlite3_prepare_v2 (db,
1108                                 "SELECT SUM(total_events)-SUM(read_events) FROM GroupCache;",
1109                                 -1,
1110                                 &stmt,
1111                                 NULL) != SQLITE_OK) {
1112                 g_error ("%s: can't compile SQL", G_STRFUNC);
1113                 return -1;
1114         }
1115
1116         while (SQLITE_BUSY == (ret = sqlite3_step (stmt)));
1117
1118         if (ret == SQLITE_ROW) {
1119                 count = sqlite3_column_int (stmt, 0);
1120         }
1121         else {
1122                 g_error ("%s: error while executing SQL", G_STRFUNC);
1123         }
1124
1125         sqlite3_finalize (stmt);
1126
1127         return count;
1128 }
1129
1130 static gboolean
1131 query_read_events (RTComEl *el, const gchar *service, gint *events, gint *conversations)
1132 {
1133         sqlite3 *db;
1134         sqlite3_stmt *stmt;
1135         int ret;
1136         gboolean result = TRUE;
1137
1138         g_object_get (el, "db", &db, NULL);
1139
1140         if (sqlite3_prepare_v2 (db,
1141                                 "SELECT SUM(total_events), COUNT(group_uid) FROM GroupCache, Services "
1142                                 "WHERE GroupCache.service_id=Services.id AND Services.name=?;",
1143                                 -1,
1144                                 &stmt,
1145                                 NULL) != SQLITE_OK) {
1146                 g_error ("%s: can't compile SQL", G_STRFUNC);
1147                 return FALSE;
1148         }
1149         if (sqlite3_bind_text (stmt, 1, service, -1, SQLITE_STATIC) != SQLITE_OK)  {
1150                 g_error ("Failed to bind %s to SQL stmt", service);
1151                 result = FALSE;
1152                 goto DONE;
1153         }
1154
1155         while (SQLITE_BUSY == (ret = sqlite3_step (stmt)));
1156
1157         if (ret == SQLITE_ROW) {
1158                 *events = sqlite3_column_int (stmt, 0);
1159                 *conversations = sqlite3_column_int (stmt, 1);
1160         }
1161         else {
1162                 g_error ("%s: error while executing SQL", G_STRFUNC);
1163                 result = FALSE;
1164                 goto DONE;
1165         }
1166
1167  DONE:
1168         sqlite3_finalize (stmt);
1169
1170         return result;
1171 }
1172
1173 static void
1174 am_ready (OssoABookAccountManager *manager,
1175           const GError            *error,
1176           gpointer                 user_data)
1177 {
1178         ELHomeApplet *self = EL_HOME_APPLET (user_data);
1179         ELHomeAppletPrivate *priv = self->priv;
1180
1181         if (!error &&
1182             priv->local_id &&
1183             !GTK_WIDGET_VISIBLE (priv->icon)) {
1184                 McAccount *account;
1185
1186                 account = osso_abook_account_manager_lookup_by_name (NULL,
1187                                                                      priv->local_id);
1188                 if (account) {
1189                         McProfile *profile = mc_profile_lookup (mc_account_compat_get_profile (account));
1190                         const gchar *icon_name = mc_profile_get_icon_name (profile);
1191                         if (icon_name) {
1192                                 gtk_image_set_from_icon_name (GTK_IMAGE (priv->icon),
1193                                                               icon_name,
1194                                                               HILDON_ICON_SIZE_XSMALL);
1195                                 gtk_widget_show (priv->icon);
1196                         }
1197                 }
1198         }
1199 }
1200
1201 static void
1202 read_event (ELHomeApplet *self)
1203 {
1204         ELHomeAppletPrivate *priv = self->priv;
1205         RTComElIter *it = NULL;
1206         const gchar *icon_name = NULL;
1207         gchar *remote_id;
1208         gchar *local_id;
1209
1210         remote_id = g_strdup (priv->remote_id);
1211         local_id = g_strdup (priv->local_id);
1212
1213         clean_state (self);
1214
1215         it = make_query (priv->eventlogger, -1);
1216         show_event (self, it);
1217
1218         if (it) g_object_unref (it);
1219
1220         if (priv->event_id >= 0) {
1221                 gboolean new_account = g_strcmp0 (priv->local_id, local_id);
1222
1223                 if (g_strcmp0 (priv->remote_id, remote_id) ||
1224                     new_account ||
1225                     !priv->contact) {
1226                         clean_contact (self, TRUE);
1227                         start_aggregator (self);
1228                 }
1229                 else if (priv->contact) {
1230                         show_contact (self);
1231                 }
1232
1233                 if (new_account) {
1234                         if (g_strcmp0 (priv->local_id, "ring/tel/ring") == 0) {
1235                                 icon_name = "general_sms";
1236                         }
1237                         else {
1238                                 McAccount *account;
1239                                 OssoABookAccountManager *am = osso_abook_account_manager_get_default ();
1240                                 if (!osso_abook_waitable_is_ready (OSSO_ABOOK_WAITABLE (am), NULL)) {
1241                                         osso_abook_account_manager_call_when_ready  (am,
1242                                                                                      am_ready,
1243                                                                                      self,
1244                                                                                      NULL);
1245                                 }
1246                                 else {
1247                                         account = osso_abook_account_manager_lookup_by_name (NULL,
1248                                                                                              priv->local_id);
1249                                         if (account) {
1250                                                 McProfile *profile = mc_profile_lookup (mc_account_compat_get_profile (account));
1251                                                 icon_name = mc_profile_get_icon_name (profile);
1252                                         }
1253                                 }
1254                         }
1255
1256                         if (icon_name) {
1257                                 gtk_image_set_from_icon_name (GTK_IMAGE (priv->icon),
1258                                                               icon_name,
1259                                                               HILDON_ICON_SIZE_XSMALL);
1260                                 gtk_widget_show (priv->icon);
1261                         }
1262                         else
1263                                 gtk_widget_hide (priv->icon);
1264                 }
1265         }
1266         else {
1267                 gchar *text;
1268                 gint n_sms_events = 0, n_sms_convs = 0;
1269                 gint n_chat_events = 0, n_chat_convs = 0;
1270                 const gchar *fmt = "%d <span size=\"small\">(%d)</span>";
1271
1272                 query_read_events (priv->eventlogger,
1273                                    "RTCOM_EL_SERVICE_SMS",
1274                                    &n_sms_events, &n_sms_convs);
1275                 query_read_events (priv->eventlogger,
1276                                    "RTCOM_EL_SERVICE_CHAT",
1277                                    &n_chat_events, &n_chat_convs);
1278
1279                 text = g_strdup_printf (fmt, n_sms_convs, n_sms_events);
1280                 gtk_label_set_markup (GTK_LABEL (priv->sms_total), text);
1281                 g_free (text);
1282
1283                 text = g_strdup_printf (fmt, n_chat_convs, n_chat_events);
1284                 gtk_label_set_markup (GTK_LABEL (priv->chat_total), text);
1285                 g_free (text);
1286
1287                 gtk_label_set_text (GTK_LABEL (priv->sender),
1288                                     dgettext ("rtcom-messaging-ui",
1289                                               "messaging_ap_conversations"));
1290
1291                 clean_contact (self, TRUE);
1292                 gtk_widget_hide (priv->icon);
1293         }
1294
1295         g_free (local_id);
1296         g_free (remote_id);
1297 }
1298
1299 static void
1300 remove_notification (ELHomeApplet *self)
1301 {
1302         ELHomeAppletPrivate *priv = self->priv;
1303
1304         DBusGConnection *conn;
1305         GError *error;
1306         DBusGProxy *proxy;
1307         GPtrArray *conv_structs;
1308         GType conv_structs_type;
1309         GValueArray *account_info;
1310         GValue value = {0, };
1311         DBusGProxyCall *call;
1312
1313         if (!(priv->remote_id && priv->local_id))
1314                 return;
1315
1316         conn = hd_home_plugin_item_get_dbus_g_connection (HD_HOME_PLUGIN_ITEM (self),
1317                                                           DBUS_BUS_SESSION,
1318                                                           &error);
1319         if (!conn) {
1320                 g_error ("Failed get dbus g connection %s", error->message);
1321                 g_error_free (error);
1322                 return;
1323         }
1324
1325         proxy = dbus_g_proxy_new_for_name (conn,
1326                                            NOTIFICATION_UI_DBUS_NAME,
1327                                            NOTIFICATION_UI_DBUS_PATH,
1328                                            NOTIFICATION_UI_DBUS_IFACE);
1329
1330         conv_structs = g_ptr_array_sized_new (1);
1331         account_info = g_value_array_new (2);
1332
1333         g_value_init (&value, G_TYPE_STRING);
1334         g_value_set_string (&value, priv->local_id);
1335         g_value_array_append (account_info, &value);
1336         g_value_unset (&value);
1337
1338         g_value_init (&value, G_TYPE_STRING);
1339         g_value_set_string (&value, priv->remote_id);
1340         g_value_array_append (account_info, &value);
1341         g_value_unset (&value);
1342
1343         g_ptr_array_add (conv_structs, account_info);
1344
1345         conv_structs_type = dbus_g_type_get_collection
1346                 ("GPtrArray",
1347                  dbus_g_type_get_struct ("GValueArray",
1348                                          G_TYPE_STRING,
1349                                          G_TYPE_STRING,
1350                                          G_TYPE_INVALID));
1351
1352         call = dbus_g_proxy_begin_call (proxy,
1353                                         "ClearConversationNotifications",
1354                                         NULL, NULL, NULL,
1355                                         conv_structs_type,
1356                                         conv_structs,
1357                                         G_TYPE_INVALID);
1358
1359         g_value_array_free (account_info);
1360         g_ptr_array_free (conv_structs, TRUE);
1361
1362         g_object_unref (proxy);
1363 }
1364
1365 static void
1366 mark_as_read (ELHomeApplet *self)
1367 {
1368         ELHomeAppletPrivate *priv = self->priv;
1369
1370         if (priv->event_id >= 0) {
1371                 rtcom_el_set_read_event (priv->eventlogger,
1372                                          priv->event_id,
1373                                          TRUE,
1374                                          NULL);
1375                 remove_notification (self);
1376         }
1377 }
1378
1379 static void
1380 launch_conversations (ELHomeApplet *self)
1381 {
1382         DBusConnection *conn;
1383         DBusMessage *message;
1384         DBusError error;
1385
1386         dbus_error_init (&error);
1387         conn = hd_home_plugin_item_get_dbus_connection (HD_HOME_PLUGIN_ITEM (self),
1388                                                         DBUS_BUS_SESSION,
1389                                                         &error);
1390         if (!conn) {
1391                 if (dbus_error_is_set (&error)) {
1392                         g_error ("Failed to get dbus connection %s", error.message);
1393                         dbus_error_free (&error);
1394                 }
1395                 return;
1396         }
1397
1398         message = dbus_message_new_method_call (CONVERSATIONS_UI_DBUS_NAME,
1399                                                 CONVERSATIONS_UI_DBUS_PATH,
1400                                                 CONVERSATIONS_UI_DBUS_IFACE,
1401                                                 "top_application");
1402         dbus_message_set_no_reply (message, TRUE);
1403
1404         if (dbus_connection_send (conn, message, NULL))
1405                 dbus_connection_flush (conn);
1406         dbus_message_unref (message);
1407
1408         dbus_connection_close (conn);
1409 }
1410
1411 static void
1412 open_conversation (ELHomeApplet *self)
1413 {
1414         ELHomeAppletPrivate *priv = self->priv;
1415         McAccount *account;
1416         const gchar *persistent_id = NULL;
1417
1418         if (!((priv->remote_id || priv->group_uid) && priv->local_id))
1419                 return;
1420
1421         account = osso_abook_account_manager_lookup_by_name (NULL,
1422                                                              priv->local_id);
1423         if (!account)
1424                 return;
1425
1426         if (priv->group_uid &&
1427             g_str_has_prefix (priv->group_uid, "group:")) {
1428                 persistent_id = strchr (priv->group_uid, '-');
1429                 if (persistent_id)
1430                         persistent_id++;
1431         }
1432
1433         if (persistent_id && persistent_id[0] != '\0') {
1434                 GHashTable *properties = tp_asv_new
1435                         (TP_IFACE_CHANNEL ".ChannelType", G_TYPE_STRING,
1436                          TP_IFACE_CHANNEL_TYPE_TEXT,
1437                          TP_IFACE_CHANNEL ".TargetHandleType", G_TYPE_UINT,
1438                          TP_HANDLE_TYPE_NONE,
1439                          RTCOM_TP_IFACE_CHANNEL_INTERFACE_PERSISTENT ".PersistentID",
1440                          G_TYPE_STRING, persistent_id,
1441                          NULL);
1442
1443                 mc_account_channelrequest_ht (account,
1444                                               properties,
1445                                               time (NULL),
1446                                               NULL,
1447                                               MC_ACCOUNT_CR_FLAG_USE_EXISTING,
1448                                               NULL, NULL, NULL, NULL);
1449
1450                 g_hash_table_unref (properties);
1451         }
1452         else if (priv->remote_id) {
1453                 McAccountChannelrequestData request;
1454
1455                 MC_ACCOUNT_CRD_INIT (&request);
1456                 MC_ACCOUNT_CRD_SET (&request, channel_type, TP_IFACE_QUARK_CHANNEL_TYPE_TEXT);
1457                 MC_ACCOUNT_CRD_SET (&request, target_handle_type, TP_HANDLE_TYPE_CONTACT);
1458                 MC_ACCOUNT_CRD_SET (&request, target_id, priv->remote_id);
1459
1460                 mc_account_channelrequest (account,
1461                                            &request,
1462                                            time (NULL),
1463                                            NULL,
1464                                            MC_ACCOUNT_CR_FLAG_USE_EXISTING,
1465                                            NULL, NULL, NULL, NULL);
1466         }
1467 }
1468
1469 static gboolean
1470 read_new_event (ELHomeApplet *self)
1471 {
1472         ELHomeAppletPrivate *priv = self->priv;
1473
1474         read_event (self);
1475
1476         if (priv->event_id >= 0)
1477                 priv->unread_count = query_unread_events (priv->eventlogger);
1478         else
1479                 priv->unread_count = 0;
1480
1481         update_unread_label (self);
1482
1483         priv->idle_id = 0;
1484
1485         return FALSE;
1486 }
1487
1488 static void
1489 add_new_idle (ELHomeApplet *self)
1490 {
1491         ELHomeAppletPrivate *priv = self->priv;
1492
1493         if (priv->idle_id)
1494                 g_source_remove (priv->idle_id);
1495         priv->idle_id = g_idle_add ((GSourceFunc)read_new_event,
1496                                     self);
1497 }
1498
1499 static void
1500 new_event_cb (RTComEl      *backend,
1501               gint          event_id,
1502               const gchar  *local_uid,
1503               const gchar  *remote_uid,
1504               const gchar  *remote_ebook_uid,
1505               const gchar  *group_uid,
1506               const gchar  *service,
1507               ELHomeApplet *self)
1508 {
1509         if (service && service[0] != '\0') {
1510                 const gchar** conv_service = conv_services;
1511                 do {
1512                         if (!g_strcmp0 (*conv_service, service)) {
1513                                 add_new_idle (self);
1514                                 return;
1515                         }
1516                 }
1517                 while(*++conv_service);
1518         }
1519         else
1520                 add_new_idle (self);
1521 }
1522
1523 static void
1524 all_deleted_cb (RTComEl      *backend,
1525                 const gchar  *service,
1526                 ELHomeApplet *self)
1527 {
1528         new_event_cb (backend,
1529                       0,
1530                       NULL,
1531                       NULL,
1532                       NULL,
1533                       NULL,
1534                       service,
1535                       self);
1536 }
1537
1538 static void
1539 refresh_hint_cb (RTComEl     *backend,
1540                  ELHomeApplet *self)
1541 {
1542         add_new_idle (self);
1543 }
1544
1545 static gboolean
1546 scroll_anim_cb (ELHomeApplet *self)
1547 {
1548         ELHomeAppletPrivate *priv = self->priv;
1549         gboolean to_continue;
1550
1551         priv->scroll_offset += SCROLL_STEP;
1552         gtk_widget_queue_draw_area (GTK_WIDGET (self),
1553                                     3*CONTENT_OFFSET_X,
1554                                     HEADER_HEIGHT + CONTENT_OFFSET_Y_TOP,
1555                                     MESSAGE_WIDTH,
1556                                     MESSAGE_HEIGHT);
1557
1558         to_continue = priv->scroll_offset <= priv->hidden_message_height;
1559         if (!to_continue) {
1560                 priv->scroll_anim_id = 0;
1561                 gtk_widget_hide (priv->cut_message);
1562         }
1563
1564         return to_continue;
1565 }
1566
1567 static gboolean
1568 button_press_event_cb (GtkWidget      *widget,
1569                        GdkEventButton *event,
1570                        ELHomeApplet   *self)
1571 {
1572         ELHomeAppletPrivate *priv = self->priv;
1573
1574         if (priv->event_id >= 0) {
1575                 if (event->y < CONTENT_OFFSET_Y_TOP + HEADER_HEIGHT) {
1576                         if (priv->aggregator &&
1577                             osso_abook_waitable_is_ready
1578                             (OSSO_ABOOK_WAITABLE (priv->aggregator), NULL))
1579                                 priv->active = SELECTED_HEADER;
1580                 }
1581                 else if (event->y > (BOX_HEIGHT - CONTENT_OFFSET_Y_BOTTOM - FOOTER_HEIGHT_PRESS) &&
1582                          event->x < FOOTER_WIDTH_PRESS)
1583                         priv->active = SELECTED_FOOTER;
1584                 else
1585                         priv->active = SELECTED_BODY;
1586         }
1587         else {
1588                 priv->active = SELECTED_BODY;
1589         }
1590
1591         gtk_widget_queue_draw (widget);
1592
1593         return TRUE;
1594 }
1595
1596 static GtkWidget*
1597 create_contact_starter_dialog (OssoABookAggregator *aggregator, const gchar *contact_id)
1598 {
1599         GtkWidget *dialog = NULL;
1600         GList *contacts = osso_abook_aggregator_lookup (aggregator, contact_id);
1601         if (contacts && contacts->data) {
1602                 GtkWidget *starter =
1603                         osso_abook_touch_contact_starter_new_with_contact
1604                         (NULL,
1605                          OSSO_ABOOK_CONTACT (contacts->data));
1606                 dialog = osso_abook_touch_contact_starter_dialog_new
1607                         (NULL,
1608                          OSSO_ABOOK_TOUCH_CONTACT_STARTER (starter));
1609                 gtk_widget_show_all (starter);
1610         }
1611
1612         g_list_free (contacts);
1613
1614         return dialog;
1615 }
1616
1617 static GtkWidget*
1618 create_temporary_contact_dialog (const gchar *remote_id,
1619                                  const gchar *account_id)
1620 {
1621         GtkWidget *dialog = NULL;
1622         const gchar *vcard = NULL;
1623         McAccount *account = NULL;
1624
1625         if (account_id) {
1626             vcard = osso_abook_account_manager_get_vcard_field (NULL, account_id);
1627             account = osso_abook_account_manager_lookup_by_name (NULL, account_id);
1628         }
1629
1630         if (vcard && account) {
1631                 EVCardAttribute *attribute = e_vcard_attribute_new (NULL, vcard);
1632
1633                 e_vcard_attribute_add_value (attribute, remote_id);
1634                 dialog = osso_abook_temporary_contact_dialog_new
1635                         (NULL,
1636                          NULL, /*EBook            *book,*/
1637                          attribute,
1638                          account);
1639                 g_signal_connect (dialog,
1640                                   "response",
1641                                   G_CALLBACK (gtk_widget_destroy),
1642                                   NULL);
1643                 e_vcard_attribute_free (attribute);
1644         }
1645
1646         return dialog;
1647 }
1648
1649 static gboolean
1650 button_release_event_cb (GtkWidget      *widget,
1651                          GdkEventButton *event,
1652                          ELHomeApplet   *self)
1653 {
1654         ELHomeAppletPrivate *priv = self->priv;
1655
1656         switch (priv->active) {
1657         case SELECTED_BODY:
1658                 if (priv->event_id >= 0) {
1659                         reset_scroll (self);
1660                         open_conversation (self);
1661                 }
1662                 else
1663                         launch_conversations (self);
1664                 break;
1665         case SELECTED_HEADER: {
1666                 GtkWidget *dialog = NULL;
1667
1668                 reset_scroll (self);
1669
1670                 if (priv->aggregator && priv->contact_id)
1671                         dialog = create_contact_starter_dialog
1672                                 (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
1673                                  priv->contact_id);
1674                 if (!dialog &&
1675                     priv->remote_id &&
1676                     priv->local_id)
1677                         dialog = create_temporary_contact_dialog (priv->remote_id,
1678                                                                   priv->local_id);
1679
1680                 if (dialog)
1681                         gtk_widget_show (dialog);
1682         }
1683                 break;
1684         case SELECTED_FOOTER:
1685                 if (priv->scroll_on_click) {
1686                         priv->scroll_on_click = FALSE;
1687                         priv->scroll_anim_id = g_timeout_add (SCROLL_PERIOD,
1688                                                               (GSourceFunc)scroll_anim_cb,
1689                                                               self);
1690                 }
1691                 else
1692                         mark_as_read (self);
1693                 break;
1694         default:;
1695         }
1696
1697         priv->active = SELECTED_NONE;
1698         gtk_widget_queue_draw (widget);
1699
1700         return TRUE;
1701 }
1702
1703 static gboolean
1704 leave_notify_event_cb (GtkWidget        *widget,
1705                        GdkEventCrossing *event,
1706                        ELHomeApplet     *self)
1707 {
1708         ELHomeAppletPrivate *priv = self->priv;
1709
1710         switch (priv->active) {
1711         case SELECTED_FOOTER:
1712                 stop_scroll_anim (priv);
1713                 /* fall down */
1714         case SELECTED_HEADER:
1715         case SELECTED_BODY:
1716                 gtk_widget_queue_draw (widget);
1717                 break;
1718         default:;
1719         }
1720
1721         priv->active = SELECTED_NONE;
1722         return FALSE;
1723 }
1724
1725 static gboolean
1726 init_eventlogger (ELHomeApplet *self)
1727 {
1728         ELHomeAppletPrivate *priv = self->priv;
1729
1730         priv->eventlogger = rtcom_el_new ();
1731
1732         /* check that db is initialized */
1733         gpointer *db = NULL;
1734         g_object_get (priv->eventlogger, "db", &db, NULL);
1735         if (!db) {
1736                 static int trial = 0;
1737
1738                 g_object_unref (priv->eventlogger);
1739                 priv->eventlogger = NULL;
1740
1741                 if (trial == 0) {
1742                         trial++;
1743                         priv->init_timer = g_timeout_add_seconds (5,
1744                                                                   (GSourceFunc)init_eventlogger,
1745                                                                   self);
1746                         return TRUE; /* return value doesn't matter */
1747                 }
1748                 else if (trial < 5) {
1749                         trial++;
1750                         return TRUE;
1751                 }
1752                 else {
1753                         g_error ("Failed to init eventlogger");
1754                         return FALSE;
1755                 }
1756         }
1757
1758         g_signal_connect (priv->eventlogger,
1759                           "new-event",
1760                           G_CALLBACK (new_event_cb),
1761                           self);
1762         g_signal_connect (priv->eventlogger,
1763                           "event-updated",
1764                           G_CALLBACK (new_event_cb),
1765                           self);
1766         g_signal_connect (priv->eventlogger,
1767                           "event-deleted",
1768                           G_CALLBACK (new_event_cb),
1769                           self);
1770         g_signal_connect (priv->eventlogger,
1771                           "all-deleted",
1772                           G_CALLBACK (all_deleted_cb),
1773                           self);
1774         g_signal_connect (priv->eventlogger,
1775                           "refresh-hint",
1776                           G_CALLBACK (refresh_hint_cb),
1777                           self);
1778
1779         add_new_idle (self);
1780
1781         priv->init_timer = 0;
1782
1783         return FALSE;
1784 }
1785
1786 static void
1787 el_home_applet_init (ELHomeApplet *self)
1788 {
1789         ELHomeAppletPrivate *priv;
1790         GtkWidget *event_box;
1791         GtkWidget *hbox, *vbox, *align, *footer;
1792         GtkWidget *w;
1793         GConfClient *gconf;
1794
1795         self->priv = EL_HOME_APPLET_GET_PRIVATE (self);
1796         priv = self->priv;
1797
1798         gtk_widget_set_app_paintable (GTK_WIDGET (self), TRUE);
1799
1800         priv->unread = gtk_label_new (NULL);
1801         gtk_misc_set_alignment (GTK_MISC (priv->unread),
1802                                 0.0f,
1803                                 0.5f);
1804         hildon_helper_set_logical_font (priv->unread, "SmallSystemFont");
1805
1806         priv->icon = gtk_image_new ();
1807         gtk_misc_set_alignment (GTK_MISC (priv->icon),
1808                                 0.5f,
1809                                 0.5f);
1810
1811         priv->sender = gtk_label_new (NULL);
1812         gtk_misc_set_alignment (GTK_MISC (priv->sender),
1813                                 0.5f,
1814                                 0.55f);
1815         gtk_label_set_ellipsize (GTK_LABEL (priv->sender),
1816                                  PANGO_ELLIPSIZE_END);
1817         gtk_widget_set_name (priv->sender, "hildon-shadow-label");
1818         hildon_helper_set_logical_font (priv->sender, "SystemFont");
1819         gtk_widget_set_size_request (priv->sender,
1820                                      C_WIDTH,
1821                                      HILDON_ICON_PIXEL_SIZE_THUMB);
1822
1823         /* construt empty table */
1824         priv->empty = gtk_fixed_new ();
1825
1826         w = gtk_image_new_from_icon_name ("general_sms", HILDON_ICON_SIZE_FINGER);
1827         gtk_fixed_put (GTK_FIXED (priv->empty), w,
1828                        4*HILDON_MARGIN_DOUBLE,
1829                        2*HILDON_MARGIN_DOUBLE);
1830
1831         w = gtk_image_new_from_icon_name ("general_chat", HILDON_ICON_SIZE_FINGER);
1832         gtk_fixed_put (GTK_FIXED (priv->empty), w,
1833                        4*HILDON_MARGIN_DOUBLE,
1834                        3*HILDON_MARGIN_DOUBLE + HILDON_ICON_PIXEL_SIZE_FINGER);
1835
1836         priv->sms_total = gtk_label_new (NULL);
1837         gtk_widget_set_name (priv->sms_total, "hildon-shadow-label");
1838         gtk_fixed_put (GTK_FIXED (priv->empty), priv->sms_total,
1839                        5*HILDON_MARGIN_DOUBLE  + HILDON_ICON_PIXEL_SIZE_FINGER,
1840                        2*HILDON_MARGIN_DOUBLE + HILDON_MARGIN_HALF);
1841
1842         priv->chat_total = gtk_label_new (NULL);
1843         gtk_widget_set_name (priv->chat_total, "hildon-shadow-label");
1844         gtk_fixed_put (GTK_FIXED (priv->empty), priv->chat_total,
1845                        5*HILDON_MARGIN_DOUBLE  + HILDON_ICON_PIXEL_SIZE_FINGER,
1846                        3*HILDON_MARGIN_DOUBLE + HILDON_MARGIN_HALF + HILDON_ICON_PIXEL_SIZE_FINGER);
1847
1848         gtk_widget_show_all (GTK_WIDGET (priv->empty));
1849         gtk_widget_hide (GTK_WIDGET (priv->empty));
1850         GTK_WIDGET_SET_FLAGS (priv->empty, GTK_NO_SHOW_ALL);
1851
1852         priv->received = gtk_label_new (NULL);
1853         gtk_misc_set_alignment (GTK_MISC (priv->received),
1854                                 1.0f,
1855                                 0.5f);
1856         hildon_helper_set_logical_font (priv->received, "SmallSystemFont");
1857         gtk_widget_set_name (priv->received, "hildon-shadow-label");
1858
1859
1860         priv->cut_message = gtk_label_new ("...");
1861         gtk_misc_set_alignment (GTK_MISC (priv->cut_message),
1862                                 0.5f,
1863                                 0.0f);
1864         hildon_helper_set_logical_font (priv->cut_message, "SmallSystemFont");
1865         gtk_widget_set_name (priv->cut_message, "hildon-shadow-label");
1866         GTK_WIDGET_SET_FLAGS (priv->cut_message, GTK_NO_SHOW_ALL);
1867
1868         hbox = gtk_hbox_new (FALSE, 0);
1869         gtk_box_pack_start (GTK_BOX (hbox), priv->sender, FALSE, FALSE, 0);
1870
1871         footer = gtk_hbox_new (FALSE, HILDON_MARGIN_DEFAULT);
1872         gtk_box_pack_start (GTK_BOX (footer), priv->unread, FALSE, FALSE, 0);
1873         gtk_box_pack_start (GTK_BOX (footer), priv->cut_message, TRUE, TRUE, 0);
1874         gtk_box_pack_end (GTK_BOX (footer), priv->icon, FALSE, FALSE, 0);
1875         gtk_box_pack_end (GTK_BOX (footer), priv->received, FALSE, FALSE, 0);
1876
1877         vbox = gtk_vbox_new (FALSE, 0);
1878         gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
1879         gtk_box_pack_start (GTK_BOX (vbox), priv->empty, TRUE, TRUE, 0);
1880         gtk_box_pack_end (GTK_BOX (vbox), footer, FALSE, FALSE, 0);
1881
1882         align = gtk_alignment_new (0.5f, 0.0f, 1.0f, 1.0f);
1883         gtk_alignment_set_padding (GTK_ALIGNMENT (align),
1884                                    0, 0, HILDON_MARGIN_DEFAULT, HILDON_MARGIN_DEFAULT);
1885
1886         gtk_container_set_border_width (GTK_CONTAINER (vbox), HILDON_MARGIN_HALF);
1887
1888         event_box = gtk_event_box_new ();
1889         gtk_event_box_set_visible_window (GTK_EVENT_BOX (event_box), FALSE);
1890         gtk_widget_set_size_request (event_box, BOX_WIDTH, BOX_HEIGHT);
1891
1892         gtk_container_add (GTK_CONTAINER (align), vbox);
1893         gtk_container_add (GTK_CONTAINER (event_box), align);
1894         gtk_container_add (GTK_CONTAINER (self), event_box);
1895
1896         g_signal_connect (event_box,
1897                           "button-press-event",
1898                           G_CALLBACK (button_press_event_cb),
1899                           self);
1900         g_signal_connect (event_box,
1901                           "button-release-event",
1902                           G_CALLBACK (button_release_event_cb),
1903                           self);
1904         g_signal_connect (event_box,
1905                           "leave-notify-event",
1906                           G_CALLBACK (leave_notify_event_cb),
1907                           self);
1908
1909         g_signal_connect (event_box,
1910                           "style-set",
1911                           G_CALLBACK (style_set_cb),
1912                           self);
1913         g_signal_connect (self,
1914                           "notify::is-on-current-desktop",
1915                           G_CALLBACK (notify_on_current_desktop),
1916                           self);
1917
1918         gtk_widget_show_all (GTK_WIDGET (event_box));
1919
1920
1921         osso_abook_init_with_name (PACKAGE, NULL);
1922
1923         gconf = gconf_client_get_default ();
1924         priv->time_fmt_24h = gconf_client_get_bool (gconf,
1925                                                     "/apps/clock/time-format",
1926                                                     NULL);
1927         g_object_unref (gconf);
1928
1929         init_eventlogger (self);
1930 }
1931
1932 static void
1933 el_home_applet_class_init (ELHomeAppletClass *klass)
1934 {
1935         GObjectClass *object_class = G_OBJECT_CLASS (klass);
1936         GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
1937
1938         object_class->dispose = dispose;
1939         object_class->finalize = finalize;
1940         widget_class->expose_event = expose_event;
1941         widget_class->realize = el_home_applet_realize;
1942
1943         g_type_class_add_private (klass, sizeof (ELHomeAppletPrivate));
1944 }