]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-account-widget.c
add a simple version of the Facebook widget
[empathy.git] / libempathy-gtk / empathy-account-widget.c
1 /*
2  * Copyright (C) 2006-2007 Imendio AB
3  * Copyright (C) 2007-2009 Collabora Ltd.
4  *
5  * This program is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU General Public License as
7  * published by the Free Software Foundation; either version 2 of the
8  * License, or (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 GNU
13  * General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public
16  * License along with this program; if not, write to the
17  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
18  * Boston, MA  02110-1301  USA
19  *
20  * Authors: Xavier Claessens <xclaesse@gmail.com>
21  *          Martyn Russell <martyn@imendio.com>
22  *          Cosimo Cecchi <cosimo.cecchi@collabora.co.uk>
23  *          Jonathan Tellier <jonathan.tellier@gmail.com>
24  */
25
26 #include <config.h>
27
28 #include <string.h>
29
30 #include <gtk/gtk.h>
31 #include <glib/gi18n-lib.h>
32
33 #ifdef HAVE_MOBLIN
34 #include <nbtk/nbtk-gtk.h>
35 #endif
36
37 #include <libempathy/empathy-utils.h>
38
39 #include <telepathy-glib/account.h>
40 #include <telepathy-glib/account-manager.h>
41 #include <telepathy-glib/connection-manager.h>
42 #include <telepathy-glib/util.h>
43 #include <dbus/dbus-protocol.h>
44
45 #include "empathy-account-widget.h"
46 #include "empathy-account-widget-private.h"
47 #include "empathy-account-widget-sip.h"
48 #include "empathy-account-widget-irc.h"
49 #include "empathy-ui-utils.h"
50
51 #define DEBUG_FLAG EMPATHY_DEBUG_ACCOUNT
52 #include <libempathy/empathy-debug.h>
53
54 G_DEFINE_TYPE (EmpathyAccountWidget, empathy_account_widget, G_TYPE_OBJECT)
55
56 typedef struct {
57   EmpathyAccountSettings *settings;
58
59   GtkWidget *table_common_settings;
60   GtkWidget *apply_button;
61   GtkWidget *cancel_button;
62   GtkWidget *entry_password;
63   GtkWidget *button_forget;
64   GtkWidget *spinbutton_port;
65   GtkWidget *enabled_checkbox;
66   GtkWidget *radiobutton_reuse;
67
68   gboolean simple;
69
70   gboolean contains_pending_changes;
71
72   /* An EmpathyAccountWidget can be used to either create an account or
73    * modify it. When we are creating an account, this member is set to TRUE */
74   gboolean creating_account;
75
76   /* if TRUE, the GTK+ destroy signal has been fired and so the widgets
77    * embedded in this account widget can't be used any more
78    * workaround because some async callbacks can be called after the
79    * widget has been destroyed */
80   gboolean destroyed;
81
82   TpAccountManager *account_manager;
83
84   GtkWidget *param_account_widget;
85   GtkWidget *param_password_widget;
86
87   gboolean dispose_run;
88 } EmpathyAccountWidgetPriv;
89
90 enum {
91   PROP_PROTOCOL = 1,
92   PROP_SETTINGS,
93   PROP_SIMPLE,
94   PROP_CREATING_ACCOUNT
95 };
96
97 enum {
98   HANDLE_APPLY,
99   ACCOUNT_CREATED,
100   CANCELLED,
101   LAST_SIGNAL
102 };
103
104 static guint signals[LAST_SIGNAL] = { 0 };
105
106 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyAccountWidget)
107 #define CHANGED_TIMEOUT 300
108
109 static void
110 account_widget_set_control_buttons_sensitivity (EmpathyAccountWidget *self,
111     gboolean sensitive)
112 {
113   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
114
115   if (!priv->simple)
116     {
117       gtk_widget_set_sensitive (priv->apply_button, sensitive);
118       gtk_widget_set_sensitive (
119           priv->cancel_button, sensitive || priv->creating_account);
120     }
121 }
122
123 static void
124 account_widget_handle_control_buttons_sensitivity (EmpathyAccountWidget *self)
125 {
126   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
127   gboolean is_valid;
128
129   is_valid = empathy_account_settings_is_valid (priv->settings);
130
131   if (!priv->simple)
132       account_widget_set_control_buttons_sensitivity (self, is_valid);
133
134   g_signal_emit (self, signals[HANDLE_APPLY], 0, is_valid);
135 }
136
137 static void
138 account_widget_entry_changed_common (EmpathyAccountWidget *self,
139     GtkEntry *entry, gboolean focus)
140 {
141   const gchar *str;
142   const gchar *param_name;
143   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
144
145   str = gtk_entry_get_text (entry);
146   param_name = g_object_get_data (G_OBJECT (entry), "param_name");
147
148   if (EMP_STR_EMPTY (str))
149     {
150       const gchar *value = NULL;
151
152       empathy_account_settings_unset (priv->settings, param_name);
153
154       if (focus)
155         {
156           value = empathy_account_settings_get_string (priv->settings,
157               param_name);
158           DEBUG ("Unset %s and restore to %s", param_name, value);
159           gtk_entry_set_text (entry, value ? value : "");
160         }
161     }
162   else
163     {
164       DEBUG ("Setting %s to %s", param_name,
165           tp_strdiff (param_name, "password") ? str : "***");
166       empathy_account_settings_set_string (priv->settings, param_name, str);
167     }
168 }
169
170 static void
171 account_widget_entry_changed_cb (GtkEditable *entry,
172     EmpathyAccountWidget *self)
173 {
174   account_widget_entry_changed_common (self, GTK_ENTRY (entry), FALSE);
175   empathy_account_widget_changed (self);
176 }
177
178 static void
179 account_widget_int_changed_cb (GtkWidget *widget,
180     EmpathyAccountWidget *self)
181 {
182   const gchar *param_name;
183   gint value;
184   const gchar *signature;
185   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
186
187   value = gtk_spin_button_get_value_as_int (GTK_SPIN_BUTTON (widget));
188   param_name = g_object_get_data (G_OBJECT (widget), "param_name");
189
190   signature = empathy_account_settings_get_dbus_signature (priv->settings,
191     param_name);
192   g_return_if_fail (signature != NULL);
193
194   DEBUG ("Setting %s to %d", param_name, value);
195
196   switch ((int)*signature)
197     {
198     case DBUS_TYPE_INT16:
199     case DBUS_TYPE_INT32:
200       empathy_account_settings_set_int32 (priv->settings, param_name, value);
201       break;
202     case DBUS_TYPE_INT64:
203       empathy_account_settings_set_int64 (priv->settings, param_name, value);
204       break;
205     case DBUS_TYPE_UINT16:
206     case DBUS_TYPE_UINT32:
207       empathy_account_settings_set_uint32 (priv->settings, param_name, value);
208       break;
209     case DBUS_TYPE_UINT64:
210       empathy_account_settings_set_uint64 (priv->settings, param_name, value);
211       break;
212     default:
213       g_return_if_reached ();
214     }
215
216   empathy_account_widget_changed (self);
217 }
218
219 static void
220 account_widget_checkbutton_toggled_cb (GtkWidget *widget,
221     EmpathyAccountWidget *self)
222 {
223   gboolean     value;
224   gboolean     default_value;
225   const gchar *param_name;
226   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
227
228   value = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (widget));
229   param_name = g_object_get_data (G_OBJECT (widget), "param_name");
230
231   /* FIXME: This is ugly! checkbox don't have a "not-set" value so we
232    * always unset the param and set the value if different from the
233    * default value. */
234   empathy_account_settings_unset (priv->settings, param_name);
235   default_value = empathy_account_settings_get_boolean (priv->settings,
236       param_name);
237
238   if (default_value == value)
239     {
240       DEBUG ("Unset %s and restore to %d", param_name, default_value);
241     }
242   else
243     {
244       DEBUG ("Setting %s to %d", param_name, value);
245       empathy_account_settings_set_boolean (priv->settings, param_name, value);
246     }
247
248   empathy_account_widget_changed (self);
249 }
250
251 static void
252 account_widget_forget_clicked_cb (GtkWidget *button,
253     EmpathyAccountWidget *self)
254 {
255   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
256   const gchar *param_name;
257
258   param_name = g_object_get_data (G_OBJECT (priv->entry_password),
259       "param_name");
260
261   DEBUG ("Unset %s", param_name);
262   empathy_account_settings_unset (priv->settings, param_name);
263   gtk_entry_set_text (GTK_ENTRY (priv->entry_password), "");
264
265   empathy_account_widget_changed (self);
266 }
267
268 static void
269 account_widget_password_changed_cb (GtkWidget *entry,
270     EmpathyAccountWidget *self)
271 {
272   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
273   const gchar *str;
274
275   str = gtk_entry_get_text (GTK_ENTRY (entry));
276   gtk_widget_set_sensitive (priv->button_forget, !EMP_STR_EMPTY (str));
277
278   priv->contains_pending_changes = TRUE;
279 }
280
281 static void
282 account_widget_jabber_ssl_toggled_cb (GtkWidget *checkbutton_ssl,
283     EmpathyAccountWidget *self)
284 {
285   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
286   gboolean   value;
287   gint32       port = 0;
288
289   value = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbutton_ssl));
290   port = empathy_account_settings_get_uint32 (priv->settings, "port");
291
292   if (value)
293     {
294       if (port == 5222 || port == 0)
295         port = 5223;
296     }
297   else
298     {
299       if (port == 5223 || port == 0)
300         port = 5222;
301     }
302
303   gtk_spin_button_set_value (GTK_SPIN_BUTTON (priv->spinbutton_port), port);
304
305   priv->contains_pending_changes = TRUE;
306 }
307
308 static void
309 account_widget_combobox_changed_cb (GtkWidget *widget,
310     EmpathyAccountWidget *self)
311 {
312   GtkTreeIter iter;
313   GtkTreeModel *model;
314   const gchar *value;
315   const GValue *v;
316   const gchar *default_value = NULL;
317   const gchar *param_name;
318   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
319
320   if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (widget), &iter))
321     return;
322
323   model = gtk_combo_box_get_model (GTK_COMBO_BOX (widget));
324   /* the param value is stored in the first column */
325   gtk_tree_model_get (model, &iter, 0, &value, -1);
326
327   param_name = g_object_get_data (G_OBJECT (widget), "param_name");
328
329   v = empathy_account_settings_get_default (priv->settings, param_name);
330   if (v != NULL)
331     default_value = g_value_get_string (v);
332
333   if (!tp_strdiff (value, default_value))
334     {
335       DEBUG ("Unset %s and restore to %s", param_name, default_value);
336       empathy_account_settings_unset (priv->settings, param_name);
337     }
338   else
339     {
340       DEBUG ("Setting %s to %s", param_name, value);
341       empathy_account_settings_set_string (priv->settings, param_name, value);
342     }
343
344   empathy_account_widget_changed (self);
345 }
346
347 void
348 empathy_account_widget_setup_widget (EmpathyAccountWidget *self,
349     GtkWidget *widget,
350     const gchar *param_name)
351 {
352   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
353
354   g_object_set_data_full (G_OBJECT (widget), "param_name",
355       g_strdup (param_name), g_free);
356
357   if (GTK_IS_SPIN_BUTTON (widget))
358     {
359       gint value = 0;
360       const gchar *signature;
361
362       signature = empathy_account_settings_get_dbus_signature (priv->settings,
363           param_name);
364       g_return_if_fail (signature != NULL);
365
366       switch ((int)*signature)
367         {
368           case DBUS_TYPE_INT16:
369           case DBUS_TYPE_INT32:
370             value = empathy_account_settings_get_int32 (priv->settings,
371               param_name);
372             break;
373           case DBUS_TYPE_INT64:
374             value = empathy_account_settings_get_int64 (priv->settings,
375               param_name);
376             break;
377           case DBUS_TYPE_UINT16:
378           case DBUS_TYPE_UINT32:
379             value = empathy_account_settings_get_uint32 (priv->settings,
380               param_name);
381             break;
382           case DBUS_TYPE_UINT64:
383             value = empathy_account_settings_get_uint64 (priv->settings,
384                 param_name);
385             break;
386           default:
387             g_return_if_reached ();
388         }
389
390       gtk_spin_button_set_value (GTK_SPIN_BUTTON (widget), value);
391
392       g_signal_connect (widget, "value-changed",
393           G_CALLBACK (account_widget_int_changed_cb),
394           self);
395     }
396   else if (GTK_IS_ENTRY (widget))
397     {
398       const gchar *str = NULL;
399
400       str = empathy_account_settings_get_string (priv->settings, param_name);
401       gtk_entry_set_text (GTK_ENTRY (widget), str ? str : "");
402
403       if (!tp_strdiff (param_name, "account"))
404         priv->param_account_widget = widget;
405       else if (!tp_strdiff (param_name, "password"))
406         priv->param_password_widget = widget;
407
408       if (strstr (param_name, "password"))
409         {
410           gtk_entry_set_visibility (GTK_ENTRY (widget), FALSE);
411         }
412
413       g_signal_connect (widget, "changed",
414           G_CALLBACK (account_widget_entry_changed_cb), self);
415     }
416   else if (GTK_IS_TOGGLE_BUTTON (widget))
417     {
418       gboolean value = FALSE;
419
420       value = empathy_account_settings_get_boolean (priv->settings,
421           param_name);
422       gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (widget), value);
423
424       g_signal_connect (widget, "toggled",
425           G_CALLBACK (account_widget_checkbutton_toggled_cb),
426           self);
427     }
428   else if (GTK_IS_COMBO_BOX (widget))
429     {
430       /* The combo box's model has to contain the param value in its first
431        * column (as a string) */
432       const gchar *str;
433       GtkTreeModel *model;
434       GtkTreeIter iter;
435       gboolean valid;
436
437       str = empathy_account_settings_get_string (priv->settings, param_name);
438       model = gtk_combo_box_get_model (GTK_COMBO_BOX (widget));
439
440       valid = gtk_tree_model_get_iter_first (model, &iter);
441       while (valid)
442         {
443           gchar *name;
444
445           gtk_tree_model_get (model, &iter, 0, &name, -1);
446           if (!tp_strdiff (name, str))
447             {
448               gtk_combo_box_set_active_iter (GTK_COMBO_BOX (widget), &iter);
449               valid = FALSE;
450             }
451           else
452             {
453               valid = gtk_tree_model_iter_next (model, &iter);
454             }
455
456           g_free (name);
457         }
458
459       g_signal_connect (widget, "changed",
460           G_CALLBACK (account_widget_combobox_changed_cb),
461           self);
462     }
463   else
464     {
465       DEBUG ("Unknown type of widget for param %s", param_name);
466     }
467 }
468
469 static gchar *
470 account_widget_generic_format_param_name (const gchar *param_name)
471 {
472   gchar *str;
473   gchar *p;
474
475   str = g_strdup (param_name);
476
477   if (str && g_ascii_isalpha (str[0]))
478     str[0] = g_ascii_toupper (str[0]);
479
480   while ((p = strchr (str, '-')) != NULL)
481     {
482       if (p[1] != '\0' && g_ascii_isalpha (p[1]))
483         {
484           p[0] = ' ';
485           p[1] = g_ascii_toupper (p[1]);
486         }
487
488       p++;
489     }
490
491   return str;
492 }
493
494 static void
495 accounts_widget_generic_setup (EmpathyAccountWidget *self,
496     GtkWidget *table_common_settings,
497     GtkWidget *table_advanced_settings)
498 {
499   TpConnectionManagerParam *params, *param;
500   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
501
502   params = empathy_account_settings_get_tp_params (priv->settings);
503
504   for (param = params; param != NULL && param->name != NULL; param++)
505     {
506       GtkWidget       *table_settings;
507       guint            n_rows = 0;
508       GtkWidget       *widget = NULL;
509       gchar           *param_name_formatted;
510
511       if (param->flags & TP_CONN_MGR_PARAM_FLAG_REQUIRED)
512         table_settings = table_common_settings;
513       else if (priv->simple)
514         return;
515       else
516         table_settings = table_advanced_settings;
517
518       param_name_formatted = account_widget_generic_format_param_name
519         (param->name);
520       g_object_get (table_settings, "n-rows", &n_rows, NULL);
521       gtk_table_resize (GTK_TABLE (table_settings), ++n_rows, 2);
522
523       if (param->dbus_signature[0] == 's')
524         {
525           gchar *str;
526
527           str = g_strdup_printf (_("%s:"), param_name_formatted);
528           widget = gtk_label_new (str);
529           gtk_misc_set_alignment (GTK_MISC (widget), 0, 0.5);
530           g_free (str);
531
532           gtk_table_attach (GTK_TABLE (table_settings),
533               widget,
534               0, 1,
535               n_rows - 1, n_rows,
536               GTK_FILL, 0,
537               0, 0);
538           gtk_widget_show (widget);
539
540           widget = gtk_entry_new ();
541           if (strcmp (param->name, "account") == 0)
542             {
543               g_signal_connect (widget, "realize",
544                   G_CALLBACK (gtk_widget_grab_focus),
545                   NULL);
546             }
547           gtk_table_attach (GTK_TABLE (table_settings),
548               widget,
549               1, 2,
550               n_rows - 1, n_rows,
551               GTK_FILL | GTK_EXPAND, 0,
552               0, 0);
553           gtk_widget_show (widget);
554         }
555       /* int types: ynqiuxt. double type is 'd' */
556       else if (param->dbus_signature[0] == 'y' ||
557           param->dbus_signature[0] == 'n' ||
558           param->dbus_signature[0] == 'q' ||
559           param->dbus_signature[0] == 'i' ||
560           param->dbus_signature[0] == 'u' ||
561           param->dbus_signature[0] == 'x' ||
562           param->dbus_signature[0] == 't' ||
563           param->dbus_signature[0] == 'd')
564         {
565           gchar   *str = NULL;
566           gdouble  minint = 0;
567           gdouble  maxint = 0;
568           gdouble  step = 1;
569
570           switch (param->dbus_signature[0])
571             {
572             case 'y': minint = G_MININT8;  maxint = G_MAXINT8;   break;
573             case 'n': minint = G_MININT16; maxint = G_MAXINT16;  break;
574             case 'q': minint = 0;          maxint = G_MAXUINT16; break;
575             case 'i': minint = G_MININT32; maxint = G_MAXINT32;  break;
576             case 'u': minint = 0;          maxint = G_MAXUINT32; break;
577             case 'x': minint = G_MININT64; maxint = G_MAXINT64;  break;
578             case 't': minint = 0;          maxint = G_MAXUINT64; break;
579             case 'd': minint = G_MININT32; maxint = G_MAXINT32;
580               step = 0.1; break;
581             }
582
583           str = g_strdup_printf (_("%s:"), param_name_formatted);
584           widget = gtk_label_new (str);
585           gtk_misc_set_alignment (GTK_MISC (widget), 0, 0.5);
586           g_free (str);
587
588           gtk_table_attach (GTK_TABLE (table_settings),
589               widget,
590               0, 1,
591               n_rows - 1, n_rows,
592               GTK_FILL, 0,
593               0, 0);
594           gtk_widget_show (widget);
595
596           widget = gtk_spin_button_new_with_range (minint, maxint, step);
597           gtk_table_attach (GTK_TABLE (table_settings),
598               widget,
599               1, 2,
600               n_rows - 1, n_rows,
601               GTK_FILL | GTK_EXPAND, 0,
602               0, 0);
603           gtk_widget_show (widget);
604         }
605       else if (param->dbus_signature[0] == 'b')
606         {
607           widget = gtk_check_button_new_with_label (param_name_formatted);
608           gtk_table_attach (GTK_TABLE (table_settings),
609               widget,
610               0, 2,
611               n_rows - 1, n_rows,
612               GTK_FILL | GTK_EXPAND, 0,
613               0, 0);
614           gtk_widget_show (widget);
615         }
616       else
617         {
618           DEBUG ("Unknown signature for param %s: %s",
619               param_name_formatted, param->dbus_signature);
620         }
621
622       if (widget)
623         empathy_account_widget_setup_widget (self, widget, param->name);
624
625       g_free (param_name_formatted);
626     }
627 }
628
629 static void
630 account_widget_handle_params_valist (EmpathyAccountWidget *self,
631     const gchar *first_widget,
632     va_list args)
633 {
634   GObject *object;
635   const gchar *name;
636
637   for (name = first_widget; name; name = va_arg (args, const gchar *))
638     {
639       const gchar *param_name;
640
641       param_name = va_arg (args, const gchar *);
642       object = gtk_builder_get_object (self->ui_details->gui, name);
643
644       if (!object)
645         {
646           g_warning ("Builder is missing object '%s'.", name);
647           continue;
648         }
649
650       empathy_account_widget_setup_widget (self, GTK_WIDGET (object),
651           param_name);
652     }
653 }
654
655 static void
656 account_widget_cancel_clicked_cb (GtkWidget *button,
657     EmpathyAccountWidget *self)
658 {
659   g_signal_emit (self, signals[CANCELLED], 0);
660 }
661
662 static void
663 account_widget_account_enabled_cb (GObject *source_object,
664     GAsyncResult *res,
665     gpointer user_data)
666 {
667   GError *error = NULL;
668   TpAccount *account = TP_ACCOUNT (source_object);
669   EmpathyAccountWidget *widget = EMPATHY_ACCOUNT_WIDGET (user_data);
670   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
671   TpConnectionPresenceType presence;
672   gchar *message = NULL;
673   gchar *status = NULL;
674
675   tp_account_set_enabled_finish (account, res, &error);
676
677   if (error != NULL)
678     {
679       DEBUG ("Could not enable the account: %s", error->message);
680       g_error_free (error);
681     }
682   else
683     {
684       /* only force presence if presence was offline, unknown or unset */
685       presence = tp_account_get_requested_presence (account, NULL, NULL);
686       switch (presence)
687         {
688         case TP_CONNECTION_PRESENCE_TYPE_OFFLINE:
689         case TP_CONNECTION_PRESENCE_TYPE_UNKNOWN:
690         case TP_CONNECTION_PRESENCE_TYPE_UNSET:
691           presence = tp_account_manager_get_most_available_presence (
692               priv->account_manager, &status, &message);
693           tp_account_request_presence_async (account, presence,
694               status, NULL, NULL, NULL);
695           break;
696         default:
697           /* do nothing if the presence is not offline */
698           break;
699         }
700     }
701
702   /* unref widget - part of the workaround */
703   g_object_unref (widget);
704   g_free (message);
705   g_free (status);
706 }
707
708 static void
709 account_widget_applied_cb (GObject *source_object,
710     GAsyncResult *res,
711     gpointer user_data)
712 {
713   GError *error = NULL;
714   TpAccount *account;
715   EmpathyAccountSettings *settings = EMPATHY_ACCOUNT_SETTINGS (source_object);
716   EmpathyAccountWidget *widget = EMPATHY_ACCOUNT_WIDGET (user_data);
717   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
718
719   empathy_account_settings_apply_finish (settings, res, &error);
720
721   if (error != NULL)
722     {
723       DEBUG ("Could not apply changes to account: %s", error->message);
724       g_error_free (error);
725       return;
726     }
727
728   account = empathy_account_settings_get_account (priv->settings);
729
730   if (account != NULL)
731     {
732       if (priv->creating_account)
733         {
734           /* By default, when an account is created, we enable it. */
735
736           /* workaround to keep widget alive during async call */
737           g_object_ref (widget);
738
739           tp_account_set_enabled_async (account, TRUE,
740               account_widget_account_enabled_cb, widget);
741           g_signal_emit (widget, signals[ACCOUNT_CREATED], 0, account);
742         }
743       else if (priv->enabled_checkbox != NULL)
744         {
745           gboolean enabled_checked;
746
747           enabled_checked =
748 #ifdef HAVE_MOBLIN
749             nbtk_gtk_light_switch_get_active (
750                 NBTK_GTK_LIGHT_SWITCH (priv->enabled_checkbox));
751 #else
752             gtk_toggle_button_get_active (
753                 GTK_TOGGLE_BUTTON (priv->enabled_checkbox));
754 #endif
755
756           if (tp_account_is_enabled (account) && enabled_checked)
757             {
758               /* After having applied changes to a user account, we
759                * automatically reconnect it. This is done so the new
760                * information entered by the user is validated on the server. */
761               tp_account_reconnect_async (account, NULL, NULL);
762             }
763         }
764     }
765
766   if (!priv->destroyed)
767     account_widget_set_control_buttons_sensitivity (widget, FALSE);
768
769   priv->contains_pending_changes = FALSE;
770
771   /* unref the widget - part of the workaround */
772   g_object_unref (widget);
773 }
774
775 static void
776 account_widget_apply_clicked_cb (GtkWidget *button,
777     EmpathyAccountWidget *self)
778 {
779   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
780
781   if (priv->radiobutton_reuse != NULL)
782     {
783       gboolean reuse = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (
784             priv->radiobutton_reuse));
785
786       DEBUG ("Set register param: %d", !reuse);
787       empathy_account_settings_set_boolean (priv->settings, "register", !reuse);
788     }
789
790   /* workaround to keep widget alive during async call */
791   g_object_ref (self);
792   empathy_account_settings_apply_async (priv->settings,
793       account_widget_applied_cb, self);
794 }
795
796 static void
797 account_widget_setup_generic (EmpathyAccountWidget *self)
798 {
799   GtkWidget *table_common_settings;
800   GtkWidget *table_advanced_settings;
801
802   table_common_settings = GTK_WIDGET (gtk_builder_get_object
803       (self->ui_details->gui, "table_common_settings"));
804   table_advanced_settings = GTK_WIDGET (gtk_builder_get_object
805       (self->ui_details->gui, "table_advanced_settings"));
806
807   accounts_widget_generic_setup (self, table_common_settings,
808       table_advanced_settings);
809
810   g_object_unref (self->ui_details->gui);
811 }
812
813 static void
814 account_widget_settings_ready_cb (EmpathyAccountSettings *settings,
815     GParamSpec *pspec,
816     gpointer user_data)
817 {
818   EmpathyAccountWidget *self = user_data;
819   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
820
821   if (empathy_account_settings_is_ready (priv->settings))
822     account_widget_setup_generic (self);
823 }
824
825 static void
826 account_widget_build_generic (EmpathyAccountWidget *self,
827     const char *filename)
828 {
829   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
830   GtkWidget *expander_advanced;
831
832   self->ui_details->gui = empathy_builder_get_file (filename,
833       "table_common_settings", &priv->table_common_settings,
834       "vbox_generic_settings", &self->ui_details->widget,
835       "expander_advanced_settings", &expander_advanced,
836       NULL);
837
838   if (priv->simple)
839     gtk_widget_hide (expander_advanced);
840
841   g_object_ref (self->ui_details->gui);
842
843   if (empathy_account_settings_is_ready (priv->settings))
844     account_widget_setup_generic (self);
845   else
846     g_signal_connect (priv->settings, "notify::ready",
847         G_CALLBACK (account_widget_settings_ready_cb), self);
848 }
849
850 static void
851 account_widget_build_salut (EmpathyAccountWidget *self,
852     const char *filename)
853 {
854   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
855   GtkWidget *expander_advanced;
856
857   self->ui_details->gui = empathy_builder_get_file (filename,
858       "table_common_settings", &priv->table_common_settings,
859       "vbox_salut_settings", &self->ui_details->widget,
860       "expander_advanced_settings", &expander_advanced,
861       NULL);
862
863   empathy_account_widget_handle_params (self,
864       "entry_published", "published-name",
865       "entry_nickname", "nickname",
866       "entry_first_name", "first-name",
867       "entry_last_name", "last-name",
868       "entry_email", "email",
869       "entry_jid", "jid",
870       NULL);
871
872   if (priv->simple)
873     gtk_widget_hide (expander_advanced);
874
875   self->ui_details->default_focus = g_strdup ("entry_first_name");
876 }
877
878 static void
879 account_widget_build_irc (EmpathyAccountWidget *self,
880   const char *filename)
881 {
882   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
883   empathy_account_widget_irc_build (self, filename,
884     &priv->table_common_settings);
885 }
886
887 static void
888 account_widget_build_sip (EmpathyAccountWidget *self,
889   const char *filename)
890 {
891   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
892   empathy_account_widget_sip_build (self, filename,
893     &priv->table_common_settings);
894 }
895
896 static void
897 account_widget_build_msn (EmpathyAccountWidget *self,
898     const char *filename)
899 {
900   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
901
902   if (priv->simple)
903     {
904       self->ui_details->gui = empathy_builder_get_file (filename,
905           "vbox_msn_simple", &self->ui_details->widget,
906           NULL);
907
908       empathy_account_widget_handle_params (self,
909           "entry_id_simple", "account",
910           "entry_password_simple", "password",
911           NULL);
912
913       self->ui_details->default_focus = g_strdup ("entry_id_simple");
914     }
915   else
916     {
917       self->ui_details->gui = empathy_builder_get_file (filename,
918           "table_common_msn_settings", &priv->table_common_settings,
919           "vbox_msn_settings", &self->ui_details->widget,
920           NULL);
921
922       empathy_account_widget_handle_params (self,
923           "entry_id", "account",
924           "entry_password", "password",
925           "entry_server", "server",
926           "spinbutton_port", "port",
927           NULL);
928
929       self->ui_details->default_focus = g_strdup ("entry_id");
930       self->ui_details->add_forget = TRUE;
931     }
932 }
933
934 static gboolean
935 account_widget_is_gtalk (EmpathyAccountWidget *self)
936 {
937   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
938
939   return !tp_strdiff (empathy_account_settings_get_icon_name (priv->settings),
940       "im-google-talk");
941 }
942
943 static gboolean
944 account_widget_is_facebook (EmpathyAccountWidget *self)
945 {
946   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
947
948   return !tp_strdiff (empathy_account_settings_get_icon_name (priv->settings),
949       "im-facebook");
950 }
951
952 #define FACEBOOK_SUFFIX "@chat.facebook.com"
953
954 static void
955 facebook_id_widget_changed_cb (GtkWidget *entry,
956     EmpathyAccountWidget *self)
957 {
958   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
959   const gchar *account;
960
961   account_widget_entry_changed_common (self, GTK_ENTRY (entry), FALSE);
962
963   account = empathy_account_settings_get_string (priv->settings, "account");
964   if (!EMP_STR_EMPTY (account) &&
965       !g_str_has_suffix (account, FACEBOOK_SUFFIX))
966     {
967       gchar *tmp;
968
969       tmp = g_strdup_printf ("%s%s", account, FACEBOOK_SUFFIX);
970
971       DEBUG ("Change account from '%s' to '%s'", account, tmp);
972
973       empathy_account_settings_set_string (priv->settings, "account", tmp);
974       g_free (tmp);
975     }
976
977   empathy_account_widget_changed (self);
978 }
979
980 static gchar *
981 remove_facebook_suffix (const gchar *str)
982 {
983   if (!g_str_has_suffix (str, FACEBOOK_SUFFIX))
984     return g_strdup (str);
985
986   return g_strndup (str, strlen (str) - strlen (FACEBOOK_SUFFIX));
987 }
988
989 static void
990 setup_facebook_id_widget (EmpathyAccountWidget *self,
991     GtkWidget *widget)
992 {
993   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
994   const gchar *str = NULL;
995
996   g_object_set_data_full (G_OBJECT (widget), "param_name",
997       g_strdup ("account"), g_free);
998
999   str = empathy_account_settings_get_string (priv->settings, "account");
1000   if (str != NULL)
1001     {
1002       gchar *tmp;
1003
1004       tmp = remove_facebook_suffix (str);
1005       gtk_entry_set_text (GTK_ENTRY (widget), tmp);
1006       g_free (tmp);
1007     }
1008
1009   priv->param_account_widget = widget;
1010
1011   g_signal_connect (widget, "changed",
1012       G_CALLBACK (facebook_id_widget_changed_cb), self);
1013 }
1014
1015 static void
1016 account_widget_build_jabber (EmpathyAccountWidget *self,
1017     const char *filename)
1018 {
1019   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1020   GtkWidget *spinbutton_port;
1021   GtkWidget *checkbutton_ssl;
1022   GtkWidget *label_id, *label_password;
1023   GtkWidget *label_id_create, *label_password_create;
1024   GtkWidget *label_example_gtalk, *label_example_jabber, *label_example_fb;
1025   gboolean is_gtalk, is_facebook;
1026   GtkWidget *expander_advanced;
1027   GtkWidget *entry_id;
1028
1029   is_gtalk = account_widget_is_gtalk (self);
1030   is_facebook = account_widget_is_facebook (self);
1031
1032   if (priv->simple && !is_gtalk && !is_facebook)
1033     {
1034       /* Simple widget for XMPP */
1035       self->ui_details->gui = empathy_builder_get_file (filename,
1036           "vbox_jabber_simple", &self->ui_details->widget,
1037           "label_id_simple", &label_id,
1038           "label_id_create", &label_id_create,
1039           "label_password_simple", &label_password,
1040           "label_password_create", &label_password_create,
1041           NULL);
1042
1043       if (empathy_account_settings_get_boolean (priv->settings, "register"))
1044         {
1045           gtk_widget_hide (label_id);
1046           gtk_widget_hide (label_password);
1047           gtk_widget_show (label_id_create);
1048           gtk_widget_show (label_password_create);
1049         }
1050
1051       empathy_account_widget_handle_params (self,
1052           "entry_id_simple", "account",
1053           "entry_password_simple", "password",
1054           NULL);
1055
1056       self->ui_details->default_focus = g_strdup ("entry_id_simple");
1057     }
1058   else if (priv->simple && is_gtalk)
1059     {
1060       /* Simple widget for Google Talk */
1061       self->ui_details->gui = empathy_builder_get_file (filename,
1062           "vbox_gtalk_simple", &self->ui_details->widget,
1063           NULL);
1064
1065       empathy_account_widget_handle_params (self,
1066           "entry_id_g_simple", "account",
1067           "entry_password_g_simple", "password",
1068           NULL);
1069
1070       self->ui_details->default_focus = g_strdup ("entry_id_g_simple");
1071     }
1072   else if (priv->simple && is_facebook)
1073     {
1074       /* Simple widget for Facebook */
1075       self->ui_details->gui = empathy_builder_get_file (filename,
1076           "vbox_fb_simple", &self->ui_details->widget,
1077           "entry_id_fb_simple", &entry_id,
1078           NULL);
1079
1080       empathy_account_widget_handle_params (self,
1081           "entry_password_fb_simple", "password",
1082           NULL);
1083
1084       setup_facebook_id_widget (self, entry_id);
1085
1086       self->ui_details->default_focus = g_strdup ("entry_id_fb_simple");
1087     }
1088   else
1089     {
1090       /* Full widget for XMPP, Google Talk and Facebook*/
1091       self->ui_details->gui = empathy_builder_get_file (filename,
1092           "table_common_settings", &priv->table_common_settings,
1093           "vbox_jabber_settings", &self->ui_details->widget,
1094           "spinbutton_port", &spinbutton_port,
1095           "checkbutton_ssl", &checkbutton_ssl,
1096           "label_username_example", &label_example_jabber,
1097           "label_username_g_example", &label_example_gtalk,
1098           "label_username_f_example", &label_example_fb,
1099           "expander_advanced", &expander_advanced,
1100           "entry_id", &entry_id,
1101           NULL);
1102
1103       empathy_account_widget_handle_params (self,
1104           "entry_password", "password",
1105           "entry_resource", "resource",
1106           "entry_server", "server",
1107           "spinbutton_port", "port",
1108           "spinbutton_priority", "priority",
1109           "checkbutton_ssl", "old-ssl",
1110           "checkbutton_ignore_ssl_errors", "ignore-ssl-errors",
1111           "checkbutton_encryption", "require-encryption",
1112           NULL);
1113
1114       if (is_facebook)
1115         {
1116           /* Facebook special case the entry ID widget to hide the
1117            * "@chat.facebook.com" part */
1118           setup_facebook_id_widget (self, entry_id);
1119         }
1120       else
1121         {
1122           empathy_account_widget_setup_widget (self, entry_id, "account");
1123         }
1124
1125       self->ui_details->default_focus = g_strdup ("entry_id");
1126       self->ui_details->add_forget = TRUE;
1127       priv->spinbutton_port = spinbutton_port;
1128
1129       g_signal_connect (checkbutton_ssl, "toggled",
1130           G_CALLBACK (account_widget_jabber_ssl_toggled_cb),
1131           self);
1132
1133       if (is_gtalk)
1134         {
1135           gtk_widget_hide (label_example_jabber);
1136           gtk_widget_show (label_example_gtalk);
1137         }
1138       else if (is_facebook)
1139         {
1140           gtk_widget_hide (label_example_jabber);
1141           gtk_widget_show (label_example_fb);
1142           gtk_widget_hide (expander_advanced);
1143         }
1144     }
1145 }
1146
1147 static void
1148 account_widget_build_icq (EmpathyAccountWidget *self,
1149     const char *filename)
1150 {
1151   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1152   GtkWidget *spinbutton_port;
1153
1154   if (priv->simple)
1155     {
1156       self->ui_details->gui = empathy_builder_get_file (filename,
1157           "vbox_icq_simple", &self->ui_details->widget,
1158           NULL);
1159
1160       empathy_account_widget_handle_params (self,
1161           "entry_uin_simple", "account",
1162           "entry_password_simple", "password",
1163           NULL);
1164
1165       self->ui_details->default_focus = g_strdup ("entry_uin_simple");
1166     }
1167   else
1168     {
1169       self->ui_details->gui = empathy_builder_get_file (filename,
1170           "table_common_settings", &priv->table_common_settings,
1171           "vbox_icq_settings", &self->ui_details->widget,
1172           "spinbutton_port", &spinbutton_port,
1173           NULL);
1174
1175       empathy_account_widget_handle_params (self,
1176           "entry_uin", "account",
1177           "entry_password", "password",
1178           "entry_server", "server",
1179           "spinbutton_port", "port",
1180           "entry_charset", "charset",
1181           NULL);
1182
1183       self->ui_details->default_focus = g_strdup ("entry_uin");
1184       self->ui_details->add_forget = TRUE;
1185     }
1186 }
1187
1188 static void
1189 account_widget_build_aim (EmpathyAccountWidget *self,
1190     const char *filename)
1191 {
1192   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1193   GtkWidget *spinbutton_port;
1194
1195   if (priv->simple)
1196     {
1197       self->ui_details->gui = empathy_builder_get_file (filename,
1198           "vbox_aim_simple", &self->ui_details->widget,
1199           NULL);
1200
1201       empathy_account_widget_handle_params (self,
1202           "entry_screenname_simple", "account",
1203           "entry_password_simple", "password",
1204           NULL);
1205
1206       self->ui_details->default_focus = g_strdup ("entry_screenname_simple");
1207     }
1208   else
1209     {
1210       self->ui_details->gui = empathy_builder_get_file (filename,
1211           "table_common_settings", &priv->table_common_settings,
1212           "vbox_aim_settings", &self->ui_details->widget,
1213           "spinbutton_port", &spinbutton_port,
1214           NULL);
1215
1216       empathy_account_widget_handle_params (self,
1217           "entry_screenname", "account",
1218           "entry_password", "password",
1219           "entry_server", "server",
1220           "spinbutton_port", "port",
1221           NULL);
1222
1223       self->ui_details->default_focus = g_strdup ("entry_screenname");
1224       self->ui_details->add_forget = TRUE;
1225     }
1226 }
1227
1228 static void
1229 account_widget_build_yahoo (EmpathyAccountWidget *self,
1230     const char *filename)
1231 {
1232   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1233
1234   if (priv->simple)
1235     {
1236       self->ui_details->gui = empathy_builder_get_file (filename,
1237           "vbox_yahoo_simple", &self->ui_details->widget,
1238           NULL);
1239
1240       empathy_account_widget_handle_params (self,
1241           "entry_id_simple", "account",
1242           "entry_password_simple", "password",
1243           NULL);
1244
1245       self->ui_details->default_focus = g_strdup ("entry_id_simple");
1246     }
1247   else
1248     {
1249       self->ui_details->gui = empathy_builder_get_file (filename,
1250           "table_common_settings", &priv->table_common_settings,
1251           "vbox_yahoo_settings", &self->ui_details->widget,
1252           NULL);
1253
1254       empathy_account_widget_handle_params (self,
1255           "entry_id", "account",
1256           "entry_password", "password",
1257           "entry_server", "server",
1258           "entry_locale", "room-list-locale",
1259           "entry_charset", "charset",
1260           "spinbutton_port", "port",
1261           "checkbutton_yahoojp", "yahoojp",
1262           "checkbutton_ignore_invites", "ignore-invites",
1263           NULL);
1264
1265       self->ui_details->default_focus = g_strdup ("entry_id");
1266       self->ui_details->add_forget = TRUE;
1267     }
1268 }
1269
1270 static void
1271 account_widget_build_groupwise (EmpathyAccountWidget *self,
1272     const char *filename)
1273 {
1274   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1275
1276   if (priv->simple)
1277     {
1278       self->ui_details->gui = empathy_builder_get_file (filename,
1279           "vbox_groupwise_simple", &self->ui_details->widget,
1280           NULL);
1281
1282       empathy_account_widget_handle_params (self,
1283           "entry_id_simple", "account",
1284           "entry_password_simple", "password",
1285           NULL);
1286
1287       self->ui_details->default_focus = g_strdup ("entry_id_simple");
1288     }
1289   else
1290     {
1291       self->ui_details->gui = empathy_builder_get_file (filename,
1292           "table_common_groupwise_settings", &priv->table_common_settings,
1293           "vbox_groupwise_settings", &self->ui_details->widget,
1294           NULL);
1295
1296       empathy_account_widget_handle_params (self,
1297           "entry_id", "account",
1298           "entry_password", "password",
1299           "entry_server", "server",
1300           "spinbutton_port", "port",
1301           NULL);
1302
1303       self->ui_details->default_focus = g_strdup ("entry_id");
1304       self->ui_details->add_forget = TRUE;
1305     }
1306 }
1307
1308 static void
1309 account_widget_destroy_cb (GtkWidget *widget,
1310     EmpathyAccountWidget *self)
1311 {
1312   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1313   /* set the destroyed flag - workaround */
1314   priv->destroyed = TRUE;
1315
1316   g_object_unref (self);
1317 }
1318
1319 static void
1320 empathy_account_widget_enabled_cb (TpAccount *account,
1321       GParamSpec *spec,
1322       gpointer user_data)
1323 {
1324   EmpathyAccountWidget *widget = EMPATHY_ACCOUNT_WIDGET (user_data);
1325   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1326   gboolean enabled = tp_account_is_enabled (account);
1327
1328   if (priv->enabled_checkbox != NULL)
1329     {
1330 #ifdef HAVE_MOBLIN
1331       nbtk_gtk_light_switch_set_active (
1332           NBTK_GTK_LIGHT_SWITCH (priv->enabled_checkbox),
1333           enabled);
1334 #else
1335       gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->enabled_checkbox),
1336           enabled);
1337 #endif /* HAVE_MOBLIN */
1338     }
1339 }
1340
1341 static void
1342 #ifdef HAVE_MOBLIN
1343 account_widget_switch_flipped_cb (NbtkGtkLightSwitch *sw,
1344     gboolean state,
1345     gpointer user_data)
1346 #else
1347 account_widget_enabled_released_cb (GtkToggleButton *toggle_button,
1348     gpointer user_data)
1349 #endif /* HAVE_MOBLIN */
1350 {
1351   EmpathyAccountWidgetPriv *priv = GET_PRIV (user_data);
1352   TpAccount *account;
1353 #ifndef HAVE_MOBLIN
1354   gboolean state;
1355
1356   state = gtk_toggle_button_get_active (toggle_button);
1357 #endif
1358
1359   account = empathy_account_settings_get_account (priv->settings);
1360
1361   /* Enable the account according to the value of the "Enabled" checkbox */
1362   /* workaround to keep widget alive during async call */
1363   g_object_ref (user_data);
1364   tp_account_set_enabled_async (account, state,
1365       account_widget_account_enabled_cb, user_data);
1366 }
1367
1368 static void
1369 do_set_property (GObject *object,
1370     guint prop_id,
1371     const GValue *value,
1372     GParamSpec *pspec)
1373 {
1374   EmpathyAccountWidgetPriv *priv = GET_PRIV (object);
1375
1376   switch (prop_id)
1377     {
1378     case PROP_SETTINGS:
1379       priv->settings = g_value_dup_object (value);
1380       break;
1381     case PROP_SIMPLE:
1382       priv->simple = g_value_get_boolean (value);
1383       break;
1384     case PROP_CREATING_ACCOUNT:
1385       priv->creating_account = g_value_get_boolean (value);
1386       break;
1387     default:
1388       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1389     }
1390 }
1391
1392 static void
1393 do_get_property (GObject *object,
1394     guint prop_id,
1395     GValue *value,
1396     GParamSpec *pspec)
1397 {
1398   EmpathyAccountWidgetPriv *priv = GET_PRIV (object);
1399
1400   switch (prop_id)
1401     {
1402     case PROP_PROTOCOL:
1403       g_value_set_string (value,
1404         empathy_account_settings_get_protocol (priv->settings));
1405       break;
1406     case PROP_SETTINGS:
1407       g_value_set_object (value, priv->settings);
1408       break;
1409     case PROP_SIMPLE:
1410       g_value_set_boolean (value, priv->simple);
1411       break;
1412     case PROP_CREATING_ACCOUNT:
1413       g_value_set_boolean (value, priv->creating_account);
1414       break;
1415     default:
1416       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1417     }
1418 }
1419
1420 static void
1421 presence_changed_cb (TpAccountManager *manager,
1422     TpConnectionPresenceType state,
1423     const gchar *status,
1424     const gchar *message,
1425     EmpathyAccountWidget *self)
1426 {
1427   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1428
1429   if (priv->destroyed)
1430     return;
1431
1432   if (priv->apply_button == NULL)
1433     /* This button doesn't exist in 'simple' mode */
1434     return;
1435
1436   if (state > TP_CONNECTION_PRESENCE_TYPE_OFFLINE)
1437     {
1438       /* We are online, display a Login button */
1439       GtkWidget *image;
1440
1441       gtk_button_set_use_stock (GTK_BUTTON (priv->apply_button), FALSE);
1442       gtk_button_set_label (GTK_BUTTON (priv->apply_button), _("L_og in"));
1443
1444       image = gtk_image_new_from_stock (GTK_STOCK_CONNECT,
1445           GTK_ICON_SIZE_BUTTON);
1446       gtk_button_set_image (GTK_BUTTON (priv->apply_button), image);
1447     }
1448   else
1449     {
1450       /* We are offline, display a Save button */
1451       gtk_button_set_image (GTK_BUTTON (priv->apply_button), NULL);
1452       gtk_button_set_use_stock (GTK_BUTTON (priv->apply_button), TRUE);
1453       gtk_button_set_label (GTK_BUTTON (priv->apply_button), GTK_STOCK_SAVE);
1454     }
1455 }
1456
1457 static void
1458 account_manager_ready_cb (GObject *source_object,
1459     GAsyncResult *result,
1460     gpointer user_data)
1461 {
1462   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (user_data);
1463   TpAccountManager *account_manager = TP_ACCOUNT_MANAGER (source_object);
1464   GError *error = NULL;
1465   TpConnectionPresenceType state;
1466
1467   if (!tp_account_manager_prepare_finish (account_manager, result, &error))
1468     {
1469       DEBUG ("Failed to prepare account manager: %s", error->message);
1470       g_error_free (error);
1471       goto out;
1472     }
1473
1474   state = tp_account_manager_get_most_available_presence (account_manager, NULL,
1475       NULL);
1476
1477   /* simulate a presence change so the apply button will be changed
1478    * if needed */
1479   presence_changed_cb (account_manager, state, NULL, NULL, self);
1480
1481 out:
1482   g_object_unref (self);
1483 }
1484
1485 #define WIDGET(cm, proto) \
1486   { #cm, #proto, "empathy-account-widget-"#proto".ui", \
1487     account_widget_build_##proto }
1488
1489 static void
1490 add_enable_checkbox (EmpathyAccountWidget *self,
1491     TpAccount *account)
1492 {
1493   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1494 #ifdef HAVE_MOBLIN
1495   GtkWidget *w;
1496 #else
1497   GtkWidget *vbox = self->ui_details->widget;
1498 #endif
1499   guint nb_rows, nb_columns;
1500   gboolean is_enabled;
1501
1502   /* handle the "Enabled" checkbox. We only add it when modifying an account */
1503   if (priv->creating_account || priv->table_common_settings == NULL)
1504     return;
1505
1506   is_enabled = tp_account_is_enabled (account);
1507
1508 #ifdef HAVE_MOBLIN
1509   w = gtk_label_new (_("Account:"));
1510   gtk_misc_set_alignment (GTK_MISC (w), 0, 0.5);
1511
1512   priv->enabled_checkbox = nbtk_gtk_light_switch_new ();
1513
1514   nbtk_gtk_light_switch_set_active (
1515       NBTK_GTK_LIGHT_SWITCH (priv->enabled_checkbox), is_enabled);
1516
1517   gtk_widget_show (w);
1518 #else
1519   priv->enabled_checkbox =
1520       gtk_check_button_new_with_label (_("Enabled"));
1521
1522   gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->enabled_checkbox),
1523       is_enabled);
1524 #endif /* HAVE_MOBLIN */
1525
1526   g_object_get (priv->table_common_settings, "n-rows", &nb_rows,
1527       "n-columns", &nb_columns, NULL);
1528
1529   gtk_table_resize (GTK_TABLE (priv->table_common_settings), ++nb_rows,
1530       nb_columns);
1531
1532 #ifdef HAVE_MOBLIN
1533   gtk_table_attach (GTK_TABLE (priv->table_common_settings),
1534       w,
1535       0, 1, nb_rows - 1, nb_rows,
1536       GTK_FILL, 0, 0, 0);
1537   gtk_table_attach (GTK_TABLE (priv->table_common_settings),
1538       priv->enabled_checkbox,
1539       1, nb_columns, nb_rows - 1, nb_rows,
1540       GTK_EXPAND | GTK_FILL, 0, 0, 0);
1541 #else
1542   gtk_box_pack_start (GTK_BOX (vbox), priv->enabled_checkbox, FALSE, FALSE, 0);
1543   gtk_box_reorder_child (GTK_BOX (vbox), priv->enabled_checkbox, 0);
1544 #endif /* HAVE_MOBLIN */
1545
1546   gtk_widget_show (priv->enabled_checkbox);
1547
1548 #ifdef HAVE_MOBLIN
1549   g_signal_connect (G_OBJECT (priv->enabled_checkbox), "switch-flipped",
1550       G_CALLBACK (account_widget_switch_flipped_cb), self);
1551 #else
1552   g_signal_connect (G_OBJECT (priv->enabled_checkbox), "released",
1553       G_CALLBACK (account_widget_enabled_released_cb), self);
1554 #endif /* HAVE_MOBLIN */
1555 }
1556
1557 #ifndef HAVE_MOBLIN
1558 /* Moblin doesn't support registration */
1559 static void
1560 add_register_buttons (EmpathyAccountWidget *self,
1561     TpAccount *account)
1562 {
1563   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1564   const TpConnectionManagerProtocol *protocol;
1565   GtkWidget *radiobutton_register;
1566   GtkWidget *vbox = self->ui_details->widget;
1567
1568   if (!priv->creating_account)
1569     return;
1570
1571   protocol = empathy_account_settings_get_tp_protocol (priv->settings);
1572   if (protocol == NULL)
1573     return;
1574
1575   if (!tp_connection_manager_protocol_can_register (protocol))
1576     return;
1577
1578   if (account_widget_is_gtalk (self) || account_widget_is_facebook (self))
1579     return;
1580
1581   if (priv->simple)
1582     return;
1583
1584   priv->radiobutton_reuse = gtk_radio_button_new_with_label (NULL,
1585       _("This account already exists on the server"));
1586   radiobutton_register = gtk_radio_button_new_with_label (
1587       gtk_radio_button_get_group (GTK_RADIO_BUTTON (priv->radiobutton_reuse)),
1588       _("Create a new account on the server"));
1589
1590   gtk_box_pack_start (GTK_BOX (vbox), priv->radiobutton_reuse, FALSE, FALSE, 0);
1591   gtk_box_pack_start (GTK_BOX (vbox), radiobutton_register, FALSE, FALSE, 0);
1592   gtk_box_reorder_child (GTK_BOX (vbox), priv->radiobutton_reuse, 0);
1593   gtk_box_reorder_child (GTK_BOX (vbox), radiobutton_register, 1);
1594   gtk_widget_show (priv->radiobutton_reuse);
1595   gtk_widget_show (radiobutton_register);
1596 }
1597 #endif
1598
1599 static void
1600 do_constructed (GObject *obj)
1601 {
1602   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1603   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1604   TpAccount *account;
1605   const gchar *protocol, *cm_name;
1606   guint i = 0;
1607   struct {
1608     const gchar *cm_name;
1609     const gchar *protocol;
1610     const char *file;
1611     void (*func)(EmpathyAccountWidget *self, const gchar *filename);
1612   } widgets [] = {
1613     { "salut", "local-xmpp", "empathy-account-widget-local-xmpp.ui",
1614         account_widget_build_salut },
1615     WIDGET (gabble, jabber),
1616     WIDGET (butterfly, msn),
1617     WIDGET (haze, icq),
1618     WIDGET (haze, aim),
1619     WIDGET (haze, yahoo),
1620     WIDGET (haze, groupwise),
1621     WIDGET (idle, irc),
1622     WIDGET (sofiasip, sip),
1623   };
1624
1625   cm_name = empathy_account_settings_get_cm (priv->settings);
1626   protocol = empathy_account_settings_get_protocol (priv->settings);
1627
1628   for (i = 0 ; i < G_N_ELEMENTS (widgets); i++)
1629     {
1630       if (!tp_strdiff (widgets[i].cm_name, cm_name) &&
1631           !tp_strdiff (widgets[i].protocol, protocol))
1632         {
1633           gchar *filename;
1634
1635           filename = empathy_file_lookup (widgets[i].file,
1636               "libempathy-gtk");
1637           widgets[i].func (self, filename);
1638           g_free (filename);
1639
1640           break;
1641         }
1642     }
1643
1644   if (i == G_N_ELEMENTS (widgets))
1645     {
1646       gchar *filename = empathy_file_lookup (
1647           "empathy-account-widget-generic.ui", "libempathy-gtk");
1648       account_widget_build_generic (self, filename);
1649       g_free (filename);
1650     }
1651
1652   /* handle default focus */
1653   if (self->ui_details->default_focus != NULL)
1654     {
1655       GObject *default_focus_entry;
1656
1657       default_focus_entry = gtk_builder_get_object
1658         (self->ui_details->gui, self->ui_details->default_focus);
1659       g_signal_connect (default_focus_entry, "realize",
1660           G_CALLBACK (gtk_widget_grab_focus),
1661           NULL);
1662     }
1663
1664   /* handle forget button */
1665   if (self->ui_details->add_forget)
1666     {
1667       const gchar *password = NULL;
1668
1669       priv->button_forget = GTK_WIDGET (gtk_builder_get_object
1670           (self->ui_details->gui, "button_forget"));
1671       priv->entry_password = GTK_WIDGET (gtk_builder_get_object
1672           (self->ui_details->gui, "entry_password"));
1673
1674       password = empathy_account_settings_get_string (priv->settings,
1675           "password");
1676       gtk_widget_set_sensitive (priv->button_forget,
1677           !EMP_STR_EMPTY (password));
1678
1679       g_signal_connect (priv->button_forget, "clicked",
1680           G_CALLBACK (account_widget_forget_clicked_cb),
1681           self);
1682       g_signal_connect (priv->entry_password, "changed",
1683           G_CALLBACK (account_widget_password_changed_cb),
1684           self);
1685     }
1686
1687   /* dup and init the account-manager */
1688   priv->account_manager = tp_account_manager_dup ();
1689
1690   g_object_ref (self);
1691   tp_account_manager_prepare_async (priv->account_manager, NULL,
1692       account_manager_ready_cb, self);
1693
1694   /* handle apply and cancel button */
1695   if (!priv->simple)
1696     {
1697       GtkWidget *hbox = gtk_hbox_new (TRUE, 3);
1698
1699       priv->cancel_button = gtk_button_new_from_stock (GTK_STOCK_CANCEL);
1700
1701       if (priv->creating_account)
1702         {
1703           /* Assumre we are offline, display a Save button. We'll update
1704            * it once the account manager is ready if needed */
1705           priv->apply_button = gtk_button_new_from_stock (GTK_STOCK_SAVE);
1706
1707           empathy_signal_connect_weak (priv->account_manager,
1708               "most-available-presence-changed",
1709               G_CALLBACK (presence_changed_cb), obj);
1710         }
1711       else
1712         {
1713           /* We are editing an existing account, display an Apply button */
1714           priv->apply_button = gtk_button_new_from_stock (GTK_STOCK_APPLY);
1715         }
1716
1717       gtk_box_pack_end (GTK_BOX (hbox), priv->apply_button, TRUE,
1718           TRUE, 3);
1719       gtk_box_pack_end (GTK_BOX (hbox), priv->cancel_button, TRUE,
1720           TRUE, 3);
1721
1722       gtk_box_pack_end (GTK_BOX (self->ui_details->widget), hbox, FALSE,
1723           FALSE, 3);
1724
1725       g_signal_connect (priv->cancel_button, "clicked",
1726           G_CALLBACK (account_widget_cancel_clicked_cb),
1727           self);
1728       g_signal_connect (priv->apply_button, "clicked",
1729           G_CALLBACK (account_widget_apply_clicked_cb),
1730           self);
1731       gtk_widget_show_all (hbox);
1732
1733       if (priv->creating_account)
1734         /* When creating an account, the user might have nothing to enter.
1735          * That means that no control interaction might occur,
1736          * so we update the control button sensitivity manually.
1737          */
1738         account_widget_handle_control_buttons_sensitivity (self);
1739       else
1740         account_widget_set_control_buttons_sensitivity (self, FALSE);
1741     }
1742
1743   account = empathy_account_settings_get_account (priv->settings);
1744
1745   if (account != NULL)
1746     {
1747       g_signal_connect (account, "notify::enabled",
1748           G_CALLBACK (empathy_account_widget_enabled_cb), self);
1749     }
1750
1751 #ifndef HAVE_MOBLIN
1752   add_register_buttons (self, account);
1753 #endif
1754   add_enable_checkbox (self, account);
1755
1756   /* hook up to widget destruction to unref ourselves */
1757   g_signal_connect (self->ui_details->widget, "destroy",
1758       G_CALLBACK (account_widget_destroy_cb), self);
1759
1760   empathy_builder_unref_and_keep_widget (self->ui_details->gui,
1761       self->ui_details->widget);
1762   self->ui_details->gui = NULL;
1763 }
1764
1765 static void
1766 do_dispose (GObject *obj)
1767 {
1768   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1769   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1770
1771   if (priv->dispose_run)
1772     return;
1773
1774   priv->dispose_run = TRUE;
1775
1776   empathy_account_settings_is_ready (priv->settings);
1777
1778   if (priv->settings != NULL)
1779     {
1780       TpAccount *account;
1781       account = empathy_account_settings_get_account (priv->settings);
1782
1783       if (account != NULL)
1784         {
1785           g_signal_handlers_disconnect_by_func (account,
1786               empathy_account_widget_enabled_cb, self);
1787         }
1788
1789       g_object_unref (priv->settings);
1790       priv->settings = NULL;
1791     }
1792
1793   if (priv->account_manager != NULL)
1794     {
1795       g_object_unref (priv->account_manager);
1796       priv->account_manager = NULL;
1797     }
1798
1799   if (G_OBJECT_CLASS (empathy_account_widget_parent_class)->dispose != NULL)
1800     G_OBJECT_CLASS (empathy_account_widget_parent_class)->dispose (obj);
1801 }
1802
1803 static void
1804 do_finalize (GObject *obj)
1805 {
1806   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1807
1808   g_free (self->ui_details->default_focus);
1809   g_slice_free (EmpathyAccountWidgetUIDetails, self->ui_details);
1810
1811   if (G_OBJECT_CLASS (empathy_account_widget_parent_class)->finalize != NULL)
1812     G_OBJECT_CLASS (empathy_account_widget_parent_class)->finalize (obj);
1813 }
1814
1815 static void
1816 empathy_account_widget_class_init (EmpathyAccountWidgetClass *klass)
1817 {
1818   GObjectClass *oclass = G_OBJECT_CLASS (klass);
1819   GParamSpec *param_spec;
1820
1821   oclass->get_property = do_get_property;
1822   oclass->set_property = do_set_property;
1823   oclass->constructed = do_constructed;
1824   oclass->dispose = do_dispose;
1825   oclass->finalize = do_finalize;
1826
1827   param_spec = g_param_spec_string ("protocol",
1828       "protocol", "The protocol of the account",
1829       NULL,
1830       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
1831   g_object_class_install_property (oclass, PROP_PROTOCOL, param_spec);
1832
1833   param_spec = g_param_spec_object ("settings",
1834       "settings", "The settings of the account",
1835       EMPATHY_TYPE_ACCOUNT_SETTINGS,
1836       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1837   g_object_class_install_property (oclass, PROP_SETTINGS, param_spec);
1838
1839   param_spec = g_param_spec_boolean ("simple",
1840       "simple", "Whether the account widget is a simple or an advanced one",
1841       FALSE,
1842       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1843   g_object_class_install_property (oclass, PROP_SIMPLE, param_spec);
1844
1845   param_spec = g_param_spec_boolean ("creating-account",
1846       "creating-account",
1847       "TRUE if we're creating an account, FALSE if we're modifying it",
1848       FALSE,
1849       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1850   g_object_class_install_property (oclass, PROP_CREATING_ACCOUNT, param_spec);
1851
1852   signals[HANDLE_APPLY] =
1853     g_signal_new ("handle-apply", G_TYPE_FROM_CLASS (klass),
1854         G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1855         g_cclosure_marshal_VOID__BOOLEAN,
1856         G_TYPE_NONE,
1857         1, G_TYPE_BOOLEAN);
1858
1859   /* This signal is emitted when an account has been created and enabled. */
1860   signals[ACCOUNT_CREATED] =
1861       g_signal_new ("account-created", G_TYPE_FROM_CLASS (klass),
1862           G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1863           g_cclosure_marshal_VOID__POINTER,
1864           G_TYPE_NONE,
1865           1, G_TYPE_OBJECT);
1866
1867   signals[CANCELLED] =
1868       g_signal_new ("cancelled", G_TYPE_FROM_CLASS (klass),
1869           G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1870           g_cclosure_marshal_VOID__VOID,
1871           G_TYPE_NONE,
1872           0);
1873
1874   g_type_class_add_private (klass, sizeof (EmpathyAccountWidgetPriv));
1875 }
1876
1877 static void
1878 empathy_account_widget_init (EmpathyAccountWidget *self)
1879 {
1880   EmpathyAccountWidgetPriv *priv =
1881     G_TYPE_INSTANCE_GET_PRIVATE ((self), EMPATHY_TYPE_ACCOUNT_WIDGET,
1882         EmpathyAccountWidgetPriv);
1883
1884   self->priv = priv;
1885   priv->dispose_run = FALSE;
1886
1887   self->ui_details = g_slice_new0 (EmpathyAccountWidgetUIDetails);
1888 }
1889
1890 /* public methods */
1891
1892 void
1893 empathy_account_widget_discard_pending_changes
1894     (EmpathyAccountWidget *widget)
1895 {
1896   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1897
1898   empathy_account_settings_discard_changes (priv->settings);
1899   priv->contains_pending_changes = FALSE;
1900 }
1901
1902 gboolean
1903 empathy_account_widget_contains_pending_changes (EmpathyAccountWidget *widget)
1904 {
1905   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1906
1907   return priv->contains_pending_changes;
1908 }
1909
1910 void
1911 empathy_account_widget_handle_params (EmpathyAccountWidget *self,
1912     const gchar *first_widget,
1913     ...)
1914 {
1915   va_list args;
1916
1917   va_start (args, first_widget);
1918   account_widget_handle_params_valist (self, first_widget, args);
1919   va_end (args);
1920 }
1921
1922 GtkWidget *
1923 empathy_account_widget_get_widget (EmpathyAccountWidget *widget)
1924 {
1925   return widget->ui_details->widget;
1926 }
1927
1928 EmpathyAccountWidget *
1929 empathy_account_widget_new_for_protocol (EmpathyAccountSettings *settings,
1930     gboolean simple)
1931 {
1932   EmpathyAccountWidget *self;
1933
1934   g_return_val_if_fail (EMPATHY_IS_ACCOUNT_SETTINGS (settings), NULL);
1935
1936   self = g_object_new
1937     (EMPATHY_TYPE_ACCOUNT_WIDGET,
1938         "settings", settings, "simple", simple,
1939         "creating-account",
1940         empathy_account_settings_get_account (settings) == NULL,
1941         NULL);
1942
1943   return self;
1944 }
1945
1946 gchar *
1947 empathy_account_widget_get_default_display_name (EmpathyAccountWidget *self)
1948 {
1949   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1950   const gchar *login_id;
1951   const gchar *protocol, *p;
1952   gchar *default_display_name;
1953
1954   login_id = empathy_account_settings_get_string (priv->settings, "account");
1955   protocol = empathy_account_settings_get_protocol (priv->settings);
1956
1957   if (login_id != NULL)
1958     {
1959       /* TODO: this should be done in empathy-account-widget-irc */
1960       if (!tp_strdiff (protocol, "irc"))
1961         {
1962           const gchar* server;
1963           server = empathy_account_settings_get_string (priv->settings,
1964               "server");
1965
1966           /* To translators: The first parameter is the login id and the
1967            * second one is the server. The resulting string will be something
1968            * like: "MyUserName on chat.freenode.net".
1969            * You should reverse the order of these arguments if the
1970            * server should come before the login id in your locale.*/
1971           default_display_name = g_strdup_printf (_("%1$s on %2$s"),
1972               login_id, server);
1973         }
1974       else if (account_widget_is_facebook (self))
1975         {
1976           gchar *tmp;
1977
1978           tmp = remove_facebook_suffix (login_id);
1979           default_display_name = g_strdup_printf ("Facebook (%s)", tmp);
1980           g_free (tmp);
1981         }
1982       else
1983         {
1984           default_display_name = g_strdup (login_id);
1985         }
1986
1987       return default_display_name;
1988     }
1989
1990   if ((p = empathy_protocol_name_to_display_name (protocol)) != NULL)
1991     protocol = p;
1992
1993   if (protocol != NULL)
1994     {
1995       /* To translators: The parameter is the protocol name. The resulting
1996        * string will be something like: "Jabber Account" */
1997       default_display_name = g_strdup_printf (_("%s Account"), protocol);
1998     }
1999   else
2000     {
2001       default_display_name = g_strdup (_("New account"));
2002     }
2003
2004   return default_display_name;
2005 }
2006
2007 /* Used by subclass to indicate that widget contains pending changes */
2008 void
2009 empathy_account_widget_changed (EmpathyAccountWidget *self)
2010 {
2011   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
2012
2013   account_widget_handle_control_buttons_sensitivity (self);
2014   priv->contains_pending_changes = TRUE;
2015 }
2016
2017 void
2018 empathy_account_widget_set_account_param (EmpathyAccountWidget *self,
2019     const gchar *account)
2020 {
2021   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
2022
2023   if (priv->param_account_widget == NULL)
2024     return;
2025
2026   gtk_entry_set_text (GTK_ENTRY (priv->param_account_widget), account);
2027 }
2028
2029 void
2030 empathy_account_widget_set_password_param (EmpathyAccountWidget *self,
2031     const gchar *account)
2032 {
2033   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
2034
2035   if (priv->param_password_widget == NULL)
2036     return;
2037
2038   gtk_entry_set_text (GTK_ENTRY (priv->param_password_widget), account);
2039 }