Fix scrolling animation glitches. Increase footer press area.
[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
26 #include <hildon/hildon.h>
27 #include <rtcom-eventlogger/eventlogger.h>
28 #include <sqlite3.h>
29 #include <string.h>
30 #include <libosso-abook/osso-abook-init.h>
31 #include <libosso-abook/osso-abook-aggregator.h>
32 #include <libosso-abook/osso-abook-contact.h>
33 #include <libosso-abook/osso-abook-waitable.h>
34 #include <libosso-abook/osso-abook-contact.h>
35 #include <libosso-abook/osso-abook-touch-contact-starter.h>
36 #include <libosso-abook/osso-abook-temporary-contact-dialog.h>
37 #include <libosso-abook/osso-abook-account-manager.h>
38
39 #include <telepathy-glib/interfaces.h>
40 #include <telepathy-glib/dbus.h>
41 #include <rtcom-telepathy-glib/extensions.h>
42
43 #define EL_HOME_APPLET_GET_PRIVATE(obj) ( \
44         G_TYPE_INSTANCE_GET_PRIVATE (obj, \
45                 EL_TYPE_HOME_APPLET, ELHomeAppletPrivate))
46
47 #define BOX_WIDTH 352
48 #define BOX_HEIGHT 284
49
50 #define CONTENT_OFFSET_X HILDON_MARGIN_HALF
51 #define CONTENT_OFFSET_Y_TOP 4*HILDON_MARGIN_HALF
52 #define CONTENT_OFFSET_Y_BOTTOM HILDON_MARGIN_HALF
53 #define C_WIDTH (BOX_WIDTH - 2*CONTENT_OFFSET_X)
54 #define C_HEIGHT (BOX_HEIGHT - (CONTENT_OFFSET_Y_TOP + CONTENT_OFFSET_Y_BOTTOM))
55
56 #define HEADER_HEIGHT 48
57 #define FOOTER_HEIGHT 24
58 #define FOOTER_HEIGHT_PRESS FOOTER_HEIGHT*2 /* approx, used only for checking clicks */
59 #define FOOTER_WIDTH C_WIDTH/4
60 #define FOOTER_WIDTH_PRESS (FOOTER_WIDTH + FOOTER_WIDTH/2) /* approx, used only for checking clicks, bigger than controls */
61
62 #define MESSAGE_HEIGHT (C_HEIGHT - HEADER_HEIGHT - FOOTER_HEIGHT)
63 #define MESSAGE_WIDTH (C_WIDTH - 2*HILDON_MARGIN_DEFAULT)
64
65 #define BOX_RADIOUS 10
66
67 #define SCROLL_PERIOD 100 /* ms */
68 #define SCROLL_STEP 1 /* pixel */
69 #define TEXT_Y_OFFSET (HEADER_HEIGHT + HILDON_MARGIN_HALF)
70
71 #define NOTIFICATION_UI_DBUS_NAME     "org.freedesktop.Telepathy.Client.NotificationUI"
72 #define NOTIFICATION_UI_DBUS_PATH     "/org/freedesktop/Telepathy/Client/NotificationUI"
73 #define NOTIFICATION_UI_DBUS_IFACE    "com.nokia.RtcomNotificationUi"
74
75 typedef enum {
76         SELECTED_NONE,
77         SELECTED_HEADER,
78         SELECTED_BODY,
79         SELECTED_FOOTER
80 } WidgetActiveSelection;
81
82 struct _ELHomeAppletPrivate
83 {
84         RTComEl *eventlogger;
85
86         GtkWidget *sender;
87         /* GtkWidget *icon; */
88         GtkWidget *unread;
89         GtkWidget *received;
90         GtkWidget *empty;
91         GtkWidget *cut_message;
92         GtkWidget *avatar;
93
94         gchar *message;
95         gint event_id;
96
97         WidgetActiveSelection active;
98
99         guint unread_count;
100
101         struct {
102                 float red;
103                 float green;
104                 float blue;
105         } active_color;
106         guint8 border_color[4];
107         PangoFontDescription *font_desc;
108
109         guint idle_id;
110
111         cairo_surface_t *message_surface;
112
113         gboolean scroll_on_click;
114         gint scroll_offset;
115         gint hidden_message_height;
116         guint scroll_anim_id;
117
118         OssoABookRoster *aggregator;
119         OssoABookWaitableClosure *aggregator_ready_closure;
120         gchar *contact_id;
121         gchar *remote_id;
122         gchar *local_id;
123         gchar *group_uid;
124         OssoABookContact *contact;
125 };
126
127 HD_DEFINE_PLUGIN_MODULE (ELHomeApplet, el_home_applet, HD_TYPE_HOME_PLUGIN_ITEM);
128
129 const gchar* g_module_check_init (GModule *module);
130 const gchar*
131 g_module_check_init (GModule *module)
132 {
133         g_module_make_resident (module);
134         return NULL;
135 }
136
137 static void
138 el_home_applet_class_finalize (ELHomeAppletClass *klass)
139 {
140 }
141
142 static void
143 el_home_applet_realize (GtkWidget *widget)
144 {
145         GdkScreen *screen;
146
147         screen = gtk_widget_get_screen (widget);
148         gtk_widget_set_colormap (widget,
149                                  gdk_screen_get_rgba_colormap (screen));
150
151         gtk_widget_set_app_paintable (widget,
152                                       TRUE);
153
154         GTK_WIDGET_CLASS (el_home_applet_parent_class)->realize (widget);
155 }
156
157 /*
158  * Thanks http://cairographics.org/cookbook/roundedrectangles/
159  */
160 static void
161 rounded_rectangle (cairo_t *cr,
162                    double   x,
163                    double   y,
164                    double   w,
165                    double   h,
166                    double   r)
167 {
168         cairo_move_to (cr, x + r, y);
169         cairo_line_to (cr, x + w - r, y);
170         cairo_curve_to (cr, x + w, y,
171                         x + w, y,
172                         x + w, y + r);
173         cairo_line_to (cr, x + w, y + h - r);
174         cairo_curve_to (cr, x + w, y + h,
175                         x + w, y + h,
176                         x + w - r, y + h);
177         cairo_line_to (cr, x + r, y + h);
178         cairo_curve_to (cr, x, y + h,
179                         x, y + h,
180                         x, y + h - r);
181         cairo_line_to (cr, x, y + r);
182         cairo_curve_to (cr, x, y,
183                         x, y,
184                         x + r, y);
185 }
186
187 static cairo_surface_t*
188 draw_text (cairo_t              *cr,
189            PangoFontDescription *desc,
190            const gchar          *text,
191            gint                  width,
192            gint                 *height)
193 {
194         PangoLayout *layout;
195         PangoRectangle extent;
196
197         cairo_surface_t *gdk_surface, *result_surface;
198         cairo_t *msg_cr;
199
200         /* Create a PangoLayout, set the font and text */
201         layout = pango_cairo_create_layout (cr);
202         pango_layout_set_text (layout,
203                                text,
204                                -1);
205         pango_layout_set_font_description (layout, desc);
206
207         pango_layout_set_wrap (layout, PANGO_WRAP_WORD_CHAR);
208         pango_layout_set_width (layout, PANGO_SCALE*width);
209
210         pango_layout_get_pixel_extents (layout, NULL, &extent);
211         *height = extent.height;
212
213         gdk_surface = cairo_get_target (cr);
214         result_surface = cairo_surface_create_similar
215                 (gdk_surface,
216                  CAIRO_CONTENT_COLOR_ALPHA,
217                  width,
218                  extent.height);
219         msg_cr = cairo_create (result_surface);
220
221         pango_cairo_update_layout (msg_cr, layout);
222         /* draw shadow */
223         cairo_move_to (msg_cr, 1, 1);
224         cairo_set_source_rgba (msg_cr, 0.2, 0.2, 0.2, 0.8);
225         pango_cairo_show_layout (msg_cr, layout);
226
227         /* draw fg */
228         cairo_move_to (msg_cr, 0, 0);
229         cairo_set_source_rgba (msg_cr, 1.0, 1.0, 1.0, 1.0);
230         pango_cairo_show_layout (msg_cr, layout);
231
232         cairo_destroy (msg_cr);
233         g_object_unref (layout);
234
235         return result_surface;
236 }
237
238 static gboolean
239 stop_scroll_anim (ELHomeAppletPrivate *priv)
240 {
241         gboolean result = priv->scroll_anim_id > 0;
242
243         if (result) {
244                 g_source_remove (priv->scroll_anim_id);
245                 priv->scroll_anim_id = 0;
246                 priv->scroll_on_click = FALSE;
247                 gtk_widget_hide (priv->cut_message);
248         }
249
250         return result;
251 }
252
253 static void
254 style_set_cb (GtkWidget *widget,
255               GtkStyle  *previous_style,
256               ELHomeApplet *self)
257 {
258         ELHomeAppletPrivate *priv = self->priv;
259         GdkColor color;
260         GtkStyle *font_style;
261
262         font_style = gtk_rc_get_style_by_paths (gtk_widget_get_settings (widget),
263                                                 "SystemFont",
264                                                 NULL,
265                                                 G_TYPE_NONE);
266         if (font_style && font_style->font_desc) {
267                 if (priv->font_desc)
268                         pango_font_description_free (priv->font_desc);
269                 priv->font_desc = pango_font_description_copy (font_style->font_desc);
270         }
271
272         if (gtk_style_lookup_color (widget->style,
273                                     "ActiveTextColor",
274                                     &color)) {
275                 priv->active_color.red = color.red/(float)G_MAXUINT16;
276                 priv->active_color.green = color.green/(float)G_MAXUINT16;
277                 priv->active_color.blue = color.blue/(float)G_MAXUINT16;
278
279                 priv->border_color[0] = color.red;
280                 priv->border_color[1] = color.green;
281                 priv->border_color[2] = color.blue;
282                 priv->border_color[3] = 255;
283         }
284 }
285
286 static void
287 reset_scroll (ELHomeApplet *self)
288 {
289         ELHomeAppletPrivate *priv = self->priv;
290
291         if (stop_scroll_anim (self->priv)) {
292                 priv->scroll_on_click = TRUE;/* priv->scroll_offset; */
293                 priv->scroll_offset = 0;
294                 if (priv->scroll_on_click)
295                         gtk_widget_show (priv->cut_message);
296         }
297 }
298
299 static void
300 notify_on_current_desktop (GObject      *object,
301                            GParamSpec   *unused G_GNUC_UNUSED,
302                            ELHomeApplet *self)
303 {
304         gboolean on;
305
306         g_object_get (object, "is-on-current-desktop", &on, NULL);
307         if (!on) {
308                 reset_scroll (self);
309                 gtk_widget_queue_draw (GTK_WIDGET (self));
310         }
311 }
312
313 static gboolean
314 expose_event (GtkWidget *self, GdkEventExpose *event)
315 {
316         ELHomeAppletPrivate *priv = EL_HOME_APPLET(self)->priv;
317         cairo_t *cr;
318         cairo_pattern_t *grad;
319
320         cr = gdk_cairo_create (self->window);
321         gdk_cairo_region (cr, event->region);
322         cairo_clip (cr);
323
324         cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
325
326         /* draw bound box */
327         cairo_set_source_rgba (cr, 0.4f, 0.4f, 0.4f, 0.1f);
328         cairo_set_line_width (cr, 3.0f);
329
330         rounded_rectangle (cr,
331                            CONTENT_OFFSET_X,
332                            CONTENT_OFFSET_Y_TOP,
333                            BOX_WIDTH - 2*CONTENT_OFFSET_X,
334                            BOX_HEIGHT - (CONTENT_OFFSET_Y_TOP + CONTENT_OFFSET_Y_BOTTOM),
335                            BOX_RADIOUS);
336
337         cairo_close_path (cr);
338         cairo_stroke (cr);
339
340         /* draw header */
341         cairo_set_line_width (cr, 1.0f);
342
343         cairo_translate (cr, CONTENT_OFFSET_X, CONTENT_OFFSET_Y_TOP);
344         cairo_move_to (cr, 0, HEADER_HEIGHT);
345         cairo_line_to (cr, 0, BOX_RADIOUS);
346         cairo_curve_to (cr, 0, 0, 0, 0, BOX_RADIOUS, 0);
347         cairo_line_to (cr, C_WIDTH - BOX_RADIOUS, 0);
348         cairo_curve_to (cr, C_WIDTH, 0, C_WIDTH, 0, C_WIDTH, BOX_RADIOUS);
349         cairo_line_to (cr, C_WIDTH, HEADER_HEIGHT);
350         cairo_line_to (cr, 0, HEADER_HEIGHT);
351
352         cairo_close_path (cr);
353
354         switch (priv->active) {
355         case SELECTED_HEADER:
356                 cairo_set_source_rgba (cr,
357                                        priv->active_color.red,
358                                        priv->active_color.green,
359                                        priv->active_color.blue,
360                                        0.8f);
361                 break;
362         default:
363                 cairo_set_source_rgba (cr, 0.2f, 0.2f, 0.2f, 0.8f);
364         }
365
366         cairo_fill (cr);
367
368         cairo_move_to (cr, 0, HEADER_HEIGHT);
369         cairo_line_to (cr, C_WIDTH, HEADER_HEIGHT);
370         cairo_set_source_rgba (cr,
371                                priv->active_color.red,
372                                priv->active_color.green,
373                                priv->active_color.blue,
374                                1.0f);
375         cairo_stroke (cr);
376
377         /* draw body */
378         if (!priv->message) {
379                 cairo_move_to (cr, 0, HEADER_HEIGHT);
380                 cairo_line_to (cr, 0, C_HEIGHT - BOX_RADIOUS);
381                 cairo_curve_to (cr, 0, C_HEIGHT, 0, C_HEIGHT, BOX_RADIOUS, C_HEIGHT);
382                 cairo_line_to (cr, C_WIDTH - BOX_RADIOUS, C_HEIGHT);
383                 cairo_curve_to (cr, C_WIDTH, C_HEIGHT, C_WIDTH, C_HEIGHT, C_WIDTH, C_HEIGHT - BOX_RADIOUS);
384                 cairo_line_to (cr, C_WIDTH, HEADER_HEIGHT);
385                 cairo_line_to (cr, 0, HEADER_HEIGHT);
386                 cairo_close_path (cr);
387         }
388         else
389                 cairo_rectangle (cr, 0, HEADER_HEIGHT,
390                                  C_WIDTH, MESSAGE_HEIGHT);
391
392         /* draw body filling depending on (in)active state */
393         grad = cairo_pattern_create_linear (0, HEADER_HEIGHT,
394                                             0, C_HEIGHT - FOOTER_HEIGHT);
395
396         switch (priv->active) {
397         case SELECTED_BODY:
398                 cairo_pattern_add_color_stop_rgba (grad,
399                                                    0.5f,
400                                                    priv->active_color.red,
401                                                    priv->active_color.green,
402                                                    priv->active_color.blue,
403                                                    0.8f);
404                 cairo_pattern_add_color_stop_rgba (grad,
405                                                    1.0f,
406                                                    priv->active_color.red/2,
407                                                    priv->active_color.green/2,
408                                                    priv->active_color.blue/2,
409                                                    0.8f);
410                 break;
411         default:
412                 cairo_pattern_add_color_stop_rgba (grad, 0.5f,
413                                                    0.4f, 0.4f, 0.4f, 0.8f);
414                 cairo_pattern_add_color_stop_rgba (grad, 1.0f,
415                                                    0.2f, 0.2f, 0.2f, 0.8f);
416         }
417
418         cairo_set_source (cr, grad);
419         cairo_fill (cr);
420
421         cairo_pattern_destroy (grad);
422
423         if (priv->message) {
424                 /* draw footer unread part bg */
425                 cairo_move_to (cr, 0, C_HEIGHT - FOOTER_HEIGHT);
426                 cairo_line_to (cr, 0, C_HEIGHT - BOX_RADIOUS);
427                 cairo_curve_to (cr, 0, C_HEIGHT, 0, C_HEIGHT, BOX_RADIOUS, C_HEIGHT);
428                 cairo_line_to (cr, FOOTER_WIDTH, C_HEIGHT);
429                 cairo_line_to (cr, FOOTER_WIDTH, C_HEIGHT - FOOTER_HEIGHT);
430                 cairo_line_to (cr, 0, C_HEIGHT - FOOTER_HEIGHT);
431                 cairo_close_path (cr);
432
433                 /* draw body filling depending on (in)active state */
434                 switch (priv->active) {
435                 case SELECTED_FOOTER:
436                         cairo_set_source_rgba (cr,
437                                                priv->active_color.red,
438                                                priv->active_color.green,
439                                                priv->active_color.blue,
440                                                0.8f);
441                         break;
442                 default:
443                         cairo_set_source_rgba (cr, 0.1f, 0.1f, 0.1f, 0.9f);
444                 }
445                 cairo_fill (cr);
446
447                 /* draw footer received part bg */
448                 cairo_move_to (cr, FOOTER_WIDTH, C_HEIGHT - FOOTER_HEIGHT);
449                 cairo_rel_line_to (cr, 0, FOOTER_HEIGHT);
450                 cairo_rel_line_to (cr, C_WIDTH - FOOTER_WIDTH - BOX_RADIOUS, 0);
451                 cairo_rel_curve_to (cr, BOX_RADIOUS, 0, BOX_RADIOUS, 0, BOX_RADIOUS, -BOX_RADIOUS);
452                 cairo_rel_line_to (cr, 0, -(FOOTER_HEIGHT - BOX_RADIOUS));
453                 cairo_rel_line_to (cr, -(C_WIDTH - FOOTER_WIDTH), 0);
454                 cairo_close_path (cr);
455
456                 cairo_set_source_rgba (cr, 0.2f, 0.2f, 0.2f, 0.8f);
457                 cairo_fill (cr);
458
459                 /* draw message */
460                 if (!priv->message_surface) {
461                         gint height;
462
463                         priv->message_surface = draw_text (cr,
464                                                            priv->font_desc,
465                                                            priv->message,
466                                                            MESSAGE_WIDTH,
467                                                            &height);
468
469                         priv->hidden_message_height = height - MESSAGE_HEIGHT;
470                         priv->scroll_on_click = priv->hidden_message_height > 0;
471                         if (priv->scroll_on_click)
472                                 gtk_widget_show (priv->cut_message);
473                 }
474
475                 cairo_rectangle (cr,
476                                  2*CONTENT_OFFSET_X,
477                                  TEXT_Y_OFFSET,
478                                  MESSAGE_WIDTH,
479                                  MESSAGE_HEIGHT);
480                 cairo_clip (cr);
481
482                 cairo_set_source_surface (cr,
483                                           priv->message_surface,
484                                           2*CONTENT_OFFSET_X,
485                                           TEXT_Y_OFFSET - priv->scroll_offset);
486                 cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
487                 cairo_paint (cr);
488         }
489
490         cairo_destroy (cr);
491
492         return GTK_WIDGET_CLASS (el_home_applet_parent_class)->expose_event (self, event);
493 }
494
495
496 static void
497 clean_state (ELHomeApplet *self)
498 {
499         ELHomeAppletPrivate *priv = self->priv;
500
501         if (priv->message) {
502                 g_free (priv->message);
503                 priv->message = NULL;
504         }
505
506         if (priv->contact_id) {
507                 g_free (priv->contact_id);
508                 priv->contact_id = NULL;
509         }
510         if (priv->local_id) {
511                 g_free (priv->local_id);
512                 priv->local_id = NULL;
513         }
514         if (priv->remote_id) {
515                 g_free (priv->remote_id);
516                 priv->remote_id = NULL;
517         }
518         if (priv->group_uid) {
519                 g_free (priv->group_uid);
520                 priv->group_uid = NULL;
521         }
522
523         if (priv->contact) {
524                 g_object_unref (priv->contact);
525                 priv->contact = NULL;
526         }
527         if (priv->aggregator) {
528                 if (priv->aggregator_ready_closure){
529                         osso_abook_waitable_cancel (OSSO_ABOOK_WAITABLE (priv->aggregator),
530                                                     priv->aggregator_ready_closure);
531                         priv->aggregator_ready_closure = NULL;
532                 }
533                 osso_abook_roster_stop (priv->aggregator);
534                 g_object_unref (priv->aggregator);
535                 priv->aggregator = NULL;
536         }
537 }
538
539 static void
540 dispose (GObject *self)
541 {
542         ELHomeAppletPrivate *priv = EL_HOME_APPLET(self)->priv;
543
544         stop_scroll_anim (priv);
545         if (priv->idle_id) {
546                 g_source_remove (priv->idle_id);
547                 priv->idle_id = 0;
548         }
549         if (priv->eventlogger) {
550                 g_object_unref (priv->eventlogger);
551                 priv->eventlogger = NULL;
552         }
553         if (priv->font_desc) {
554                 pango_font_description_free (priv->font_desc);
555                 priv->font_desc = NULL;
556         }
557
558         clean_state (EL_HOME_APPLET (self));
559
560         G_OBJECT_CLASS (el_home_applet_parent_class)->dispose (self);
561 }
562
563 static void
564 finalize (GObject *self)
565 {
566         G_OBJECT_CLASS (el_home_applet_parent_class)->finalize (self);
567 }
568
569 static void
570 aggregator_ready_cb (OssoABookWaitable *waitable,
571                      const GError      *error,
572                      gpointer           userdata)
573 {
574         ELHomeApplet *self = EL_HOME_APPLET(userdata);
575         ELHomeAppletPrivate *priv = self->priv;
576         GList *contacts = NULL;
577
578         priv->aggregator_ready_closure = NULL;
579
580         if (error) {
581                 g_warning ("Failed to create aggregator: %s", error->message);
582                 return;
583         }
584
585         if (priv->contact_id) {
586                 contacts = osso_abook_aggregator_lookup
587                         (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
588                          priv->contact_id);
589         }
590         else if (priv->local_id && priv->remote_id) {
591                 if (g_strcmp0 (priv->local_id, "ring/tel/ring" == 0)) {
592                         contacts = osso_abook_aggregator_find_contacts_for_phone_number
593                                 (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
594                                  priv->remote_id,
595                                  TRUE);
596                 }
597                 else {
598                         McAccount *account;
599                         account = osso_abook_account_manager_lookup_by_name
600                                 (NULL,
601                                  priv->local_id);
602                         if (account) {
603                                 contacts = osso_abook_aggregator_find_contacts_for_im_contact
604                                         (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
605                                          priv->remote_id,
606                                          account);
607                         }
608                 }
609         }
610
611         if (contacts && contacts->data) {
612                 GdkPixbuf *avatar_image;
613
614                 priv->contact = g_object_ref (OSSO_ABOOK_CONTACT (contacts->data));
615                 gtk_label_set_text (GTK_LABEL (priv->sender),
616                                     osso_abook_contact_get_display_name (priv->contact));
617
618                 avatar_image = osso_abook_avatar_get_image_rounded
619                         (OSSO_ABOOK_AVATAR (priv->contact),
620                          HILDON_ICON_PIXEL_SIZE_THUMB,
621                          HILDON_ICON_PIXEL_SIZE_THUMB,
622                          TRUE,
623                          -1,
624                          priv->border_color);
625
626                 if (avatar_image) {
627                         gtk_image_set_from_pixbuf (GTK_IMAGE (priv->avatar),
628                                                    avatar_image);
629                         gtk_widget_show (priv->avatar);
630                         g_object_unref (avatar_image);
631                 }
632                 gtk_widget_queue_draw (GTK_WIDGET (self));
633         }
634 }
635
636 static void
637 resolve_contact (ELHomeApplet *self)
638 {
639         ELHomeAppletPrivate *priv = self->priv;
640         EBookQuery *query = NULL;
641         GError *error = NULL;
642
643         if (priv->local_id && priv->remote_id) {
644                 const gchar *vcard = osso_abook_account_manager_get_vcard_field
645                         (NULL, priv->local_id);
646                 if (vcard)
647                         query = e_book_query_vcard_field_test (vcard,
648                                                                E_BOOK_QUERY_IS,
649                                                                priv->remote_id);
650                 else
651                         query = e_book_query_any_field_contains (priv->remote_id);
652         }
653
654         if (query) {
655                 priv->aggregator = osso_abook_aggregator_new_with_query (NULL,
656                                                                          query,
657                                                                          NULL,
658                                                                          1,
659                                                                          &error);
660                 e_book_query_unref (query);
661         }
662         if (error) {
663                 g_warning ("Failed to create aggregator: %s", error->message);
664                 g_error_free (error);
665                 return;
666         }
667
668         if (priv->aggregator) {
669                 priv->aggregator_ready_closure = osso_abook_waitable_call_when_ready
670                         (OSSO_ABOOK_WAITABLE (priv->aggregator),
671                          aggregator_ready_cb,
672                          self, NULL);
673
674                 osso_abook_roster_start (priv->aggregator);
675         }
676 }
677
678 static gchar*
679 format_time (time_t t)
680 {
681         static const guint RESULT_SIZE = 32;
682
683         time_t now;
684         struct tm now_tm, t_tm;
685         const gchar *format = "%x %X";
686         gchar *result = g_malloc0 (RESULT_SIZE);
687
688         now = time (NULL);
689         localtime_r (&now, &now_tm);
690         localtime_r (&t, &t_tm);
691
692         if ((now_tm.tm_year == t_tm.tm_year) &&
693             (now_tm.tm_mon  == t_tm.tm_mon) &&
694             (now_tm.tm_mday == t_tm.tm_mday))
695                 format = "%X";
696
697         strftime (result, RESULT_SIZE, format, &t_tm);
698
699         return result;
700 }
701
702 static void
703 show_event (ELHomeApplet *self, RTComElIter *it)
704 {
705         ELHomeAppletPrivate *priv = self->priv;
706         gchar *remote = NULL;
707         gchar *received = NULL;
708         /* const gchar *icon_name = NULL; */
709
710         if (it && rtcom_el_iter_first (it)) {
711                 rtcom_el_iter_dup_string (it, "free-text", &priv->message);
712                 if (priv->message) {
713                         /* const gchar *service; */
714                         time_t received_t;
715
716                         rtcom_el_iter_get_int (it, "id", &priv->event_id);
717                         if (rtcom_el_iter_get_int (it, "start-time", (gint*)&received_t))
718                                 received = format_time (received_t);
719
720                         if (rtcom_el_iter_dup_string (it, "remote-uid", &priv->remote_id)) {
721                                 if (priv->remote_id && priv->remote_id[0])  {
722                                         if (!rtcom_el_iter_dup_string (it, "remote-name", &remote))
723                                                 remote = g_strdup (priv->remote_id);
724
725                                         rtcom_el_iter_dup_string (it, "remote-ebook-uid", &priv->contact_id);
726                                         rtcom_el_iter_dup_string (it, "local-uid", &priv->local_id);
727                                 }
728                                 else if (priv->remote_id) {
729                                         g_free (priv->remote_id);
730                                         priv->remote_id = NULL;
731                                 }
732                         }
733                         rtcom_el_iter_dup_string (it, "group-uid", &priv->group_uid);
734 #if 0
735                         service = rtcom_el_iter_get_service (it);
736                         if (!g_strcmp0 (service, "RTCOM_EL_SERVICE_SMS"))
737                                 icon_name = "chat_unread_sms";
738                         else if (!g_strcmp0 (service, "RTCOM_EL_SERVICE_CHAT"))
739                                 icon_name = "chat_unread_chat";
740 #endif
741                 }
742         }
743         else {
744                 priv->event_id = -1;
745         }
746
747         gtk_widget_hide (priv->avatar);
748
749         if (priv->message) {
750                 gtk_widget_hide (priv->empty);
751         }
752         else {
753                 gtk_widget_show (priv->empty);
754         }
755
756         gtk_label_set_text (GTK_LABEL (priv->received), received);
757
758 #if 0
759         gtk_widget_hide (priv->avatar);
760         if (icon_name) {
761                 const gchar *current_icon_name;
762                 gtk_image_get_icon_name (GTK_IMAGE (priv->icon),
763                                          &current_icon_name,
764                                          NULL);
765                 if (g_strcmp0 (current_icon_name, icon_name))
766                         gtk_image_set_from_icon_name (GTK_IMAGE (priv->icon),
767                                                       icon_name,
768                                                       HILDON_ICON_SIZE_FINGER);
769                 gtk_widget_show (priv->icon);
770         }
771         else
772                 gtk_widget_hide (priv->icon);
773 #endif
774
775         if (remote)
776                 gtk_label_set_text (GTK_LABEL (priv->sender), remote);
777         else
778                 gtk_label_set_text (GTK_LABEL (priv->sender), priv->remote_id);
779         g_free (remote);
780
781         stop_scroll_anim (priv);
782         priv->scroll_offset = 0;
783         if (priv->message_surface) {
784                 cairo_surface_destroy (priv->message_surface);
785                 priv->message_surface = NULL;
786         }
787
788         gtk_widget_hide (priv->cut_message);
789         gtk_widget_queue_draw (GTK_WIDGET (self));
790 }
791
792 static RTComElIter*
793 make_query (RTComEl *el, gint event_id)
794 {
795         RTComElQuery *query = NULL;
796         RTComElIter *it = NULL;
797
798         static const gchar *services[] = {"RTCOM_EL_SERVICE_SMS",
799                                           "RTCOM_EL_SERVICE_CHAT",
800                                           NULL};
801         static const gchar *event_types[] = {"RTCOM_EL_EVENTTYPE_SMS_INBOUND",
802                                              "RTCOM_EL_EVENTTYPE_CHAT_INBOUND",
803                                              NULL};
804
805         query = rtcom_el_query_new (el);
806         rtcom_el_query_set_limit (query, 1);
807         if (event_id >= 0) {
808                 rtcom_el_query_prepare (query,
809                                         "is-read", FALSE, RTCOM_EL_OP_EQUAL,
810                                         "id", event_id, RTCOM_EL_OP_EQUAL,
811                                         "service", services, RTCOM_EL_OP_IN_STRV,
812                                         "event-type", event_types, RTCOM_EL_OP_IN_STRV,
813                                         NULL);
814         }
815         else {
816                 rtcom_el_query_prepare (query,
817                                         "is-read", FALSE, RTCOM_EL_OP_EQUAL,
818                                         "service", services, RTCOM_EL_OP_IN_STRV,
819                                         "event-type", event_types, RTCOM_EL_OP_IN_STRV,
820                                         NULL);
821         }
822         it = rtcom_el_get_events (el, query);
823         g_object_unref (query);
824
825         return it;
826 }
827
828 static void
829 update_unread_label (ELHomeApplet *self)
830 {
831         ELHomeAppletPrivate *priv = self->priv;
832
833         if (priv->unread_count > 0) {
834                 gchar *text;
835                 text = g_strdup_printf
836                         ("%d<span foreground=\"red\" rise=\"5000\">*</span>",
837                          priv->unread_count);
838
839                 gtk_label_set_markup (GTK_LABEL (priv->unread), text);
840                 g_free (text);
841         }
842         else
843                 gtk_label_set_text (GTK_LABEL (priv->unread), NULL);
844 }
845
846 static gint
847 query_unread_events (RTComEl *el)
848 {
849         sqlite3 *db;
850         sqlite3_stmt *stmt;
851         int ret;
852         gint count = 0;
853
854         g_object_get (el, "db", &db, NULL);
855
856         if (sqlite3_prepare_v2 (db,
857                                 "SELECT SUM(total_events)-SUM(read_events) FROM GroupCache;",
858                                 -1,
859                                 &stmt,
860                                 NULL) != SQLITE_OK) {
861                 g_error ("%s: can't compile SQL", G_STRFUNC);
862                 return -1;
863         }
864
865         while (SQLITE_BUSY == (ret = sqlite3_step (stmt)));
866
867         if (ret == SQLITE_ROW) {
868                 count = sqlite3_column_int (stmt, 0);
869         }
870         else {
871                 g_error ("%s: error while executing SQL", G_STRFUNC);
872         }
873
874         sqlite3_finalize (stmt);
875
876         return count;
877 }
878
879 static void
880 read_event (ELHomeApplet *self)
881 {
882         ELHomeAppletPrivate *priv = self->priv;
883         RTComElIter *it = NULL;
884
885         clean_state (self);
886
887         it = make_query (priv->eventlogger, -1);
888         show_event (self, it);
889         resolve_contact (self);
890         if (it) g_object_unref (it);
891 }
892
893 static void
894 remove_notification (ELHomeApplet *self)
895 {
896         ELHomeAppletPrivate *priv = self->priv;
897
898         DBusGConnection* conn;
899         GError *error;
900         DBusGProxy *proxy;
901         GPtrArray *conv_structs;
902         GType conv_structs_type;
903         GValueArray *account_info;
904         GValue value = {0, };
905         DBusGProxyCall *call;
906
907         if (!(priv->remote_id && priv->local_id))
908                 return;
909
910         conn = hd_home_plugin_item_get_dbus_g_connection (HD_HOME_PLUGIN_ITEM (self),
911                                                           DBUS_BUS_SESSION,
912                                                           &error);
913         if (!conn) {
914                 g_error ("Failed get dbus g connection %s", error->message);
915                 g_error_free (error);
916                 return;
917         }
918
919         proxy = dbus_g_proxy_new_for_name (conn,
920                                            NOTIFICATION_UI_DBUS_NAME,
921                                            NOTIFICATION_UI_DBUS_PATH,
922                                            NOTIFICATION_UI_DBUS_IFACE);
923
924         conv_structs = g_ptr_array_sized_new (1);
925         account_info = g_value_array_new (2);
926
927         g_value_init (&value, G_TYPE_STRING);
928         g_value_set_string (&value, priv->local_id);
929         g_value_array_append (account_info, &value);
930         g_value_unset (&value);
931
932         g_value_init (&value, G_TYPE_STRING);
933         g_value_set_string (&value, priv->remote_id);
934         g_value_array_append (account_info, &value);
935         g_value_unset (&value);
936
937         g_ptr_array_add (conv_structs, account_info);
938
939         conv_structs_type = dbus_g_type_get_collection
940                 ("GPtrArray",
941                  dbus_g_type_get_struct ("GValueArray",
942                                          G_TYPE_STRING,
943                                          G_TYPE_STRING,
944                                          G_TYPE_INVALID));
945
946         call = dbus_g_proxy_begin_call (proxy,
947                                         "ClearConversationNotifications",
948                                         NULL, NULL, NULL,
949                                         conv_structs_type,
950                                         conv_structs,
951                                         G_TYPE_INVALID);
952
953         g_value_array_free (account_info);
954         g_ptr_array_free (conv_structs, TRUE);
955
956         g_object_unref (proxy);
957 }
958
959 static void
960 mark_as_read (ELHomeApplet *self)
961 {
962         ELHomeAppletPrivate *priv = self->priv;
963
964         if (priv->event_id >= 0) {
965                 rtcom_el_set_read_event (priv->eventlogger,
966                                          priv->event_id,
967                                          TRUE,
968                                          NULL);
969                 remove_notification (self);
970         }
971 }
972
973 static void
974 open_conversation (ELHomeApplet *self)
975 {
976         ELHomeAppletPrivate *priv = self->priv;
977         McAccount *account;
978         const gchar *persistent_id = NULL;
979
980         if (!((priv->remote_id || priv->group_uid) && priv->local_id))
981                 return;
982
983         account = osso_abook_account_manager_lookup_by_name (NULL,
984                                                              priv->local_id);
985         if (!account)
986                 return;
987
988         if (priv->group_uid &&
989             g_str_has_prefix (priv->group_uid, "group:")) {
990                 persistent_id = strchr (priv->group_uid, '-');
991                 if (persistent_id)
992                         persistent_id++;
993         }
994
995         if (persistent_id && persistent_id[0] != '\0') {
996                 GHashTable *properties = tp_asv_new
997                         (TP_IFACE_CHANNEL ".ChannelType", G_TYPE_STRING,
998                          TP_IFACE_CHANNEL_TYPE_TEXT,
999                          TP_IFACE_CHANNEL ".TargetHandleType", G_TYPE_UINT,
1000                          TP_HANDLE_TYPE_NONE,
1001                          RTCOM_TP_IFACE_CHANNEL_INTERFACE_PERSISTENT ".PersistentID",
1002                          G_TYPE_STRING, persistent_id,
1003                          NULL);
1004
1005                 mc_account_channelrequest_ht (account,
1006                                               properties,
1007                                               time (NULL),
1008                                               NULL,
1009                                               MC_ACCOUNT_CR_FLAG_USE_EXISTING,
1010                                               NULL, NULL, NULL, NULL);
1011
1012                 g_hash_table_unref (properties);
1013         }
1014         else if (priv->remote_id) {
1015                 McAccountChannelrequestData request;
1016
1017                 MC_ACCOUNT_CRD_INIT (&request);
1018                 MC_ACCOUNT_CRD_SET (&request, channel_type, TP_IFACE_QUARK_CHANNEL_TYPE_TEXT);
1019                 MC_ACCOUNT_CRD_SET (&request, target_handle_type, TP_HANDLE_TYPE_CONTACT);
1020                 MC_ACCOUNT_CRD_SET (&request, target_id, priv->remote_id);
1021
1022                 mc_account_channelrequest (account,
1023                                            &request,
1024                                            time (NULL),
1025                                            NULL,
1026                                            MC_ACCOUNT_CR_FLAG_USE_EXISTING,
1027                                            NULL, NULL, NULL, NULL);
1028         }
1029 }
1030
1031 static gboolean
1032 read_new_event (ELHomeApplet *self)
1033 {
1034         ELHomeAppletPrivate *priv = self->priv;
1035
1036         read_event (self);
1037         priv->unread_count = query_unread_events (priv->eventlogger);
1038         update_unread_label (self);
1039
1040         priv->idle_id = 0;
1041
1042         return FALSE;
1043 }
1044
1045 static void
1046 add_new_idle (ELHomeApplet *self)
1047 {
1048         ELHomeAppletPrivate *priv = self->priv;
1049
1050         if (priv->idle_id)
1051                 g_source_remove (priv->idle_id);
1052         priv->idle_id = g_idle_add ((GSourceFunc)read_new_event,
1053                                     self);
1054 }
1055
1056 static void
1057 new_event_cb (RTComEl      *backend,
1058               gint          event_id,
1059               const gchar  *local_uid,
1060               const gchar  *remote_uid,
1061               const gchar  *remote_ebook_uid,
1062               const gchar  *group_uid,
1063               const gchar  *service,
1064               ELHomeApplet *self)
1065 {
1066         /* TODO: avoid updating if not related */
1067         add_new_idle (self);
1068 }
1069
1070 static gboolean
1071 scroll_anim_cb (ELHomeApplet *self)
1072 {
1073         ELHomeAppletPrivate *priv = self->priv;
1074         gboolean to_continue;
1075
1076         priv->scroll_offset += SCROLL_STEP;
1077         gtk_widget_queue_draw_area (GTK_WIDGET (self),
1078                                     3*CONTENT_OFFSET_X,
1079                                     HEADER_HEIGHT + CONTENT_OFFSET_Y_TOP,
1080                                     MESSAGE_WIDTH,
1081                                     MESSAGE_HEIGHT);
1082
1083         to_continue = priv->scroll_offset <= priv->hidden_message_height;
1084         if (!to_continue) {
1085                 priv->scroll_anim_id = 0;
1086                 gtk_widget_hide (priv->cut_message);
1087         }
1088
1089         return to_continue;
1090 }
1091
1092 static gboolean
1093 button_press_event_cb (GtkWidget      *widget,
1094                        GdkEventButton *event,
1095                        ELHomeApplet   *self)
1096 {
1097         ELHomeAppletPrivate *priv = self->priv;
1098
1099         if (priv->event_id > 0) {
1100                 if (event->y < CONTENT_OFFSET_Y_TOP + HEADER_HEIGHT) {
1101                         if (priv->aggregator &&
1102                             osso_abook_waitable_is_ready
1103                             (OSSO_ABOOK_WAITABLE (priv->aggregator), NULL))
1104                                 priv->active = SELECTED_HEADER;
1105                 }
1106                 else if (event->y > (BOX_HEIGHT - CONTENT_OFFSET_Y_BOTTOM - FOOTER_HEIGHT_PRESS) &&
1107                          event->x < FOOTER_WIDTH_PRESS)
1108                         priv->active = SELECTED_FOOTER;
1109                 else
1110                         priv->active = SELECTED_BODY;
1111
1112                 gtk_widget_queue_draw (widget);
1113         }
1114
1115         return TRUE;
1116 }
1117
1118 static GtkWidget*
1119 create_contact_starter_dialog (OssoABookAggregator *aggregator, const gchar *contact_id)
1120 {
1121         GtkWidget *dialog = NULL;
1122         GList *contacts = osso_abook_aggregator_lookup (aggregator, contact_id);
1123         if (contacts && contacts->data) {
1124                 GtkWidget *starter =
1125                         osso_abook_touch_contact_starter_new_with_contact
1126                         (NULL,
1127                          OSSO_ABOOK_CONTACT (contacts->data));
1128                 dialog = osso_abook_touch_contact_starter_dialog_new
1129                         (NULL,
1130                          OSSO_ABOOK_TOUCH_CONTACT_STARTER (starter));
1131                 gtk_widget_show_all (starter);
1132         }
1133
1134         g_list_free (contacts);
1135
1136         return dialog;
1137 }
1138
1139 static GtkWidget*
1140 create_temporary_contact_dialog (const gchar *remote_id,
1141                                  const gchar *account_id)
1142 {
1143         GtkWidget *dialog = NULL;
1144         const gchar *vcard = NULL;
1145         McAccount *account = NULL;
1146
1147         if (account_id) {
1148             vcard = osso_abook_account_manager_get_vcard_field (NULL, account_id);
1149             account = osso_abook_account_manager_lookup_by_name (NULL, account_id);
1150         }
1151
1152         if (vcard && account) {
1153                 EVCardAttribute *attribute = e_vcard_attribute_new (NULL, vcard);
1154
1155                 e_vcard_attribute_add_value (attribute, remote_id);
1156                 dialog = osso_abook_temporary_contact_dialog_new
1157                         (NULL,
1158                          NULL, /*EBook            *book,*/
1159                          attribute,
1160                          account);
1161                 g_signal_connect (dialog,
1162                                   "response",
1163                                   G_CALLBACK (gtk_widget_destroy),
1164                                   NULL);
1165                 e_vcard_attribute_free (attribute);
1166         }
1167
1168         return dialog;
1169 }
1170
1171 static gboolean
1172 button_release_event_cb (GtkWidget      *widget,
1173                          GdkEventButton *event,
1174                          ELHomeApplet   *self)
1175 {
1176         ELHomeAppletPrivate *priv = self->priv;
1177
1178         switch (priv->active) {
1179         case SELECTED_BODY:
1180                 reset_scroll (self);
1181                 open_conversation (self);
1182                 break;
1183         case SELECTED_HEADER: {
1184                 GtkWidget *dialog = NULL;
1185
1186                 reset_scroll (self);
1187
1188                 if (priv->aggregator && priv->contact_id)
1189                         dialog = create_contact_starter_dialog
1190                                 (OSSO_ABOOK_AGGREGATOR (priv->aggregator),
1191                                  priv->contact_id);
1192                 if (!dialog &&
1193                     priv->remote_id &&
1194                     priv->local_id)
1195                         dialog = create_temporary_contact_dialog (priv->remote_id,
1196                                                                   priv->local_id);
1197
1198                 if (dialog)
1199                         gtk_widget_show (dialog);
1200         }
1201                 break;
1202         case SELECTED_FOOTER:
1203                 if (priv->scroll_on_click) {
1204                         priv->scroll_on_click = FALSE;
1205                         priv->scroll_anim_id = g_timeout_add (SCROLL_PERIOD,
1206                                                               (GSourceFunc)scroll_anim_cb,
1207                                                               self);
1208                 }
1209                 else
1210 #ifndef DEBUG_LAYOUT
1211                         mark_as_read (self);
1212 #endif
1213                 break;
1214         default:;
1215         }
1216
1217         priv->active = SELECTED_NONE;
1218         gtk_widget_queue_draw (widget);
1219
1220         return TRUE;
1221 }
1222
1223 static gboolean
1224 leave_notify_event_cb (GtkWidget        *widget,
1225                        GdkEventCrossing *event,
1226                        ELHomeApplet     *self)
1227 {
1228         ELHomeAppletPrivate *priv = self->priv;
1229
1230         switch (priv->active) {
1231         case SELECTED_FOOTER:
1232                 stop_scroll_anim (priv);
1233                 /* fall down */
1234         case SELECTED_HEADER:
1235         case SELECTED_BODY:
1236                 gtk_widget_queue_draw (widget);
1237                 break;
1238         default:;
1239         }
1240
1241         priv->active = SELECTED_NONE;
1242         return FALSE;
1243 }
1244
1245 static void
1246 el_home_applet_init (ELHomeApplet *self)
1247 {
1248         ELHomeAppletPrivate *priv;
1249         GtkWidget *event_box;
1250         GtkWidget *hbox, *vbox, *align, *footer;
1251
1252         self->priv = EL_HOME_APPLET_GET_PRIVATE (self);
1253         priv = self->priv;
1254
1255         gtk_widget_set_app_paintable (GTK_WIDGET (self), TRUE);
1256
1257         priv->unread = gtk_label_new ("12");
1258         gtk_misc_set_alignment (GTK_MISC (priv->unread),
1259                                 0.0f,
1260                                 0.5f);
1261         hildon_helper_set_logical_font (priv->unread, "SmallSystemFont");
1262         /* gtk_widget_set_size_request (priv->unread, */
1263         /*                              -1, */
1264         /*                              HEADER_HEIGHT); */
1265 #if 0
1266         priv->icon = gtk_image_new_from_icon_name ("chat_unread_sms",
1267                                                    HILDON_ICON_SIZE_FINGER);
1268         gtk_misc_set_alignment (GTK_MISC (priv->icon),
1269                                 0.5f,
1270                                 0.5f);
1271 #endif
1272         priv->avatar = gtk_image_new ();
1273         gtk_misc_set_alignment (GTK_MISC (priv->avatar),
1274                                 0.5f,
1275                                 0.5f);
1276
1277         priv->sender = gtk_label_new ("asdf asdf asdf asdf asdf");
1278         gtk_misc_set_alignment (GTK_MISC (priv->sender),
1279                                 0.5f,
1280                                 0.55f);
1281         gtk_label_set_ellipsize (GTK_LABEL (priv->sender),
1282                                  PANGO_ELLIPSIZE_END);
1283         gtk_widget_set_name (priv->sender, "hildon-shadow-label");
1284         hildon_helper_set_logical_font (priv->sender, "SystemFont");
1285         gtk_widget_set_size_request (priv->sender,
1286                                      -1,
1287                                      HILDON_ICON_PIXEL_SIZE_THUMB);
1288
1289         priv->message = g_strdup ("One two three four five six seven eight nine ten");
1290
1291         /* TODO: l10n */
1292         priv->empty = gtk_label_new ("No new messages");
1293         gtk_widget_set_name (priv->empty, "hildon-shadow-label");
1294         GTK_WIDGET_SET_FLAGS (priv->empty, GTK_NO_SHOW_ALL);
1295
1296         priv->received = gtk_label_new ("aewf aewf aewf awef");
1297         gtk_misc_set_alignment (GTK_MISC (priv->received),
1298                                 1.0f,
1299                                 0.5f);
1300         hildon_helper_set_logical_font (priv->received, "SmallSystemFont");
1301         gtk_widget_set_name (priv->received, "hildon-shadow-label");
1302
1303
1304         priv->cut_message = gtk_label_new ("...");
1305         gtk_misc_set_alignment (GTK_MISC (priv->cut_message),
1306                                 0.5f,
1307                                 0.0f);
1308         hildon_helper_set_logical_font (priv->cut_message, "SmallSystemFont");
1309         gtk_widget_set_name (priv->cut_message, "hildon-shadow-label");
1310         GTK_WIDGET_SET_FLAGS (priv->cut_message, GTK_NO_SHOW_ALL);
1311
1312         hbox = gtk_hbox_new (FALSE, 0);
1313         /* gtk_box_pack_start (GTK_BOX (hbox), priv->icon, FALSE, FALSE, 0); */
1314         gtk_box_pack_start (GTK_BOX (hbox), priv->sender, TRUE, TRUE, 0);
1315         gtk_box_pack_start (GTK_BOX (hbox), priv->avatar, FALSE, FALSE, 0);
1316
1317         footer = gtk_hbox_new (FALSE, HILDON_MARGIN_DEFAULT);
1318         gtk_box_pack_start (GTK_BOX (footer), priv->unread, FALSE, FALSE, 0);
1319         gtk_box_pack_start (GTK_BOX (footer), priv->cut_message, TRUE, TRUE, 0);
1320         gtk_box_pack_end (GTK_BOX (footer), priv->received, FALSE, FALSE, 0);
1321
1322         vbox = gtk_vbox_new (FALSE, 0);
1323         gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
1324         gtk_box_pack_start (GTK_BOX (vbox), priv->empty, TRUE, TRUE, 0);
1325         gtk_box_pack_end (GTK_BOX (vbox), footer, FALSE, FALSE, 0);
1326
1327         align = gtk_alignment_new (0.5f, 0.0f, 1.0f, 1.0f);
1328         gtk_alignment_set_padding (GTK_ALIGNMENT (align),
1329                                    0, 0, HILDON_MARGIN_DEFAULT, HILDON_MARGIN_DEFAULT);
1330
1331         gtk_container_set_border_width (GTK_CONTAINER (vbox), HILDON_MARGIN_HALF);
1332
1333         event_box = gtk_event_box_new ();
1334         gtk_event_box_set_visible_window (GTK_EVENT_BOX (event_box), FALSE);
1335         gtk_widget_set_size_request (event_box, BOX_WIDTH, BOX_HEIGHT);
1336
1337         gtk_container_add (GTK_CONTAINER (align), vbox);
1338         gtk_container_add (GTK_CONTAINER (event_box), align);
1339         gtk_container_add (GTK_CONTAINER (self), event_box);
1340
1341         g_signal_connect (event_box,
1342                           "button-press-event",
1343                           G_CALLBACK (button_press_event_cb),
1344                           self);
1345         g_signal_connect (event_box,
1346                           "button-release-event",
1347                           G_CALLBACK (button_release_event_cb),
1348                           self);
1349         g_signal_connect (event_box,
1350                           "leave-notify-event",
1351                           G_CALLBACK (leave_notify_event_cb),
1352                           self);
1353
1354         g_signal_connect (event_box,
1355                           "style-set",
1356                           G_CALLBACK (style_set_cb),
1357                           self);
1358         g_signal_connect (self,
1359                           "notify::is-on-current-desktop",
1360                           G_CALLBACK (notify_on_current_desktop),
1361                           self);
1362
1363         gtk_widget_show_all (GTK_WIDGET (event_box));
1364
1365 #ifndef DEBUG_LAYOUT
1366         priv->eventlogger = rtcom_el_new ();
1367         g_signal_connect (priv->eventlogger,
1368                           "new-event",
1369                           G_CALLBACK (new_event_cb),
1370                           self);
1371         g_signal_connect (priv->eventlogger,
1372                           "event-updated",
1373                           G_CALLBACK (new_event_cb),
1374                           self);
1375
1376         read_new_event (self);
1377
1378         osso_abook_init_with_name (PACKAGE, NULL);
1379 #endif
1380 }
1381
1382 static void
1383 el_home_applet_class_init (ELHomeAppletClass *klass)
1384 {
1385         GObjectClass *object_class = G_OBJECT_CLASS (klass);
1386         GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
1387
1388         object_class->dispose = dispose;
1389         object_class->finalize = finalize;
1390         widget_class->expose_event = expose_event;
1391         widget_class->realize = el_home_applet_realize;
1392
1393         g_type_class_add_private (klass, sizeof (ELHomeAppletPrivate));
1394 }
1395