]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-account-widget.c
account-widget-salut: don't display advanced properties in simple mode
[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 void
944 account_widget_build_jabber (EmpathyAccountWidget *self,
945     const char *filename)
946 {
947   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
948   GtkWidget *spinbutton_port;
949   GtkWidget *checkbutton_ssl;
950   GtkWidget *label_id, *label_password;
951   GtkWidget *label_id_create, *label_password_create;
952   GtkWidget *label_example_gtalk, *label_example_jabber;
953   gboolean is_gtalk;
954
955   is_gtalk = account_widget_is_gtalk (self);
956
957   if (priv->simple && !is_gtalk)
958     {
959       self->ui_details->gui = empathy_builder_get_file (filename,
960           "vbox_jabber_simple", &self->ui_details->widget,
961           "label_id_simple", &label_id,
962           "label_id_create", &label_id_create,
963           "label_password_simple", &label_password,
964           "label_password_create", &label_password_create,
965           NULL);
966
967       if (empathy_account_settings_get_boolean (priv->settings, "register"))
968         {
969           gtk_widget_hide (label_id);
970           gtk_widget_hide (label_password);
971           gtk_widget_show (label_id_create);
972           gtk_widget_show (label_password_create);
973         }
974
975       empathy_account_widget_handle_params (self,
976           "entry_id_simple", "account",
977           "entry_password_simple", "password",
978           NULL);
979
980       self->ui_details->default_focus = g_strdup ("entry_id_simple");
981     }
982   else if (priv->simple && is_gtalk)
983     {
984       self->ui_details->gui = empathy_builder_get_file (filename,
985           "vbox_gtalk_simple", &self->ui_details->widget,
986           NULL);
987
988       empathy_account_widget_handle_params (self,
989           "entry_id_g_simple", "account",
990           "entry_password_g_simple", "password",
991           NULL);
992
993       self->ui_details->default_focus = g_strdup ("entry_id_g_simple");
994     }
995   else
996     {
997       self->ui_details->gui = empathy_builder_get_file (filename,
998           "table_common_settings", &priv->table_common_settings,
999           "vbox_jabber_settings", &self->ui_details->widget,
1000           "spinbutton_port", &spinbutton_port,
1001           "checkbutton_ssl", &checkbutton_ssl,
1002           "label_username_example", &label_example_jabber,
1003           "label_username_g_example", &label_example_gtalk,
1004           NULL);
1005
1006       empathy_account_widget_handle_params (self,
1007           "entry_id", "account",
1008           "entry_password", "password",
1009           "entry_resource", "resource",
1010           "entry_server", "server",
1011           "spinbutton_port", "port",
1012           "spinbutton_priority", "priority",
1013           "checkbutton_ssl", "old-ssl",
1014           "checkbutton_ignore_ssl_errors", "ignore-ssl-errors",
1015           "checkbutton_encryption", "require-encryption",
1016           NULL);
1017
1018       self->ui_details->default_focus = g_strdup ("entry_id");
1019       self->ui_details->add_forget = TRUE;
1020       priv->spinbutton_port = spinbutton_port;
1021
1022       g_signal_connect (checkbutton_ssl, "toggled",
1023           G_CALLBACK (account_widget_jabber_ssl_toggled_cb),
1024           self);
1025
1026       if (is_gtalk)
1027         {
1028           gtk_widget_hide (label_example_jabber);
1029           gtk_widget_show (label_example_gtalk);
1030         }
1031     }
1032 }
1033
1034 static void
1035 account_widget_build_icq (EmpathyAccountWidget *self,
1036     const char *filename)
1037 {
1038   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1039   GtkWidget *spinbutton_port;
1040
1041   if (priv->simple)
1042     {
1043       self->ui_details->gui = empathy_builder_get_file (filename,
1044           "vbox_icq_simple", &self->ui_details->widget,
1045           NULL);
1046
1047       empathy_account_widget_handle_params (self,
1048           "entry_uin_simple", "account",
1049           "entry_password_simple", "password",
1050           NULL);
1051
1052       self->ui_details->default_focus = g_strdup ("entry_uin_simple");
1053     }
1054   else
1055     {
1056       self->ui_details->gui = empathy_builder_get_file (filename,
1057           "table_common_settings", &priv->table_common_settings,
1058           "vbox_icq_settings", &self->ui_details->widget,
1059           "spinbutton_port", &spinbutton_port,
1060           NULL);
1061
1062       empathy_account_widget_handle_params (self,
1063           "entry_uin", "account",
1064           "entry_password", "password",
1065           "entry_server", "server",
1066           "spinbutton_port", "port",
1067           "entry_charset", "charset",
1068           NULL);
1069
1070       self->ui_details->default_focus = g_strdup ("entry_uin");
1071       self->ui_details->add_forget = TRUE;
1072     }
1073 }
1074
1075 static void
1076 account_widget_build_aim (EmpathyAccountWidget *self,
1077     const char *filename)
1078 {
1079   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1080   GtkWidget *spinbutton_port;
1081
1082   if (priv->simple)
1083     {
1084       self->ui_details->gui = empathy_builder_get_file (filename,
1085           "vbox_aim_simple", &self->ui_details->widget,
1086           NULL);
1087
1088       empathy_account_widget_handle_params (self,
1089           "entry_screenname_simple", "account",
1090           "entry_password_simple", "password",
1091           NULL);
1092
1093       self->ui_details->default_focus = g_strdup ("entry_screenname_simple");
1094     }
1095   else
1096     {
1097       self->ui_details->gui = empathy_builder_get_file (filename,
1098           "table_common_settings", &priv->table_common_settings,
1099           "vbox_aim_settings", &self->ui_details->widget,
1100           "spinbutton_port", &spinbutton_port,
1101           NULL);
1102
1103       empathy_account_widget_handle_params (self,
1104           "entry_screenname", "account",
1105           "entry_password", "password",
1106           "entry_server", "server",
1107           "spinbutton_port", "port",
1108           NULL);
1109
1110       self->ui_details->default_focus = g_strdup ("entry_screenname");
1111       self->ui_details->add_forget = TRUE;
1112     }
1113 }
1114
1115 static void
1116 account_widget_build_yahoo (EmpathyAccountWidget *self,
1117     const char *filename)
1118 {
1119   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1120
1121   if (priv->simple)
1122     {
1123       self->ui_details->gui = empathy_builder_get_file (filename,
1124           "vbox_yahoo_simple", &self->ui_details->widget,
1125           NULL);
1126
1127       empathy_account_widget_handle_params (self,
1128           "entry_id_simple", "account",
1129           "entry_password_simple", "password",
1130           NULL);
1131
1132       self->ui_details->default_focus = g_strdup ("entry_id_simple");
1133     }
1134   else
1135     {
1136       self->ui_details->gui = empathy_builder_get_file (filename,
1137           "table_common_settings", &priv->table_common_settings,
1138           "vbox_yahoo_settings", &self->ui_details->widget,
1139           NULL);
1140
1141       empathy_account_widget_handle_params (self,
1142           "entry_id", "account",
1143           "entry_password", "password",
1144           "entry_server", "server",
1145           "entry_locale", "room-list-locale",
1146           "entry_charset", "charset",
1147           "spinbutton_port", "port",
1148           "checkbutton_yahoojp", "yahoojp",
1149           "checkbutton_ignore_invites", "ignore-invites",
1150           NULL);
1151
1152       self->ui_details->default_focus = g_strdup ("entry_id");
1153       self->ui_details->add_forget = TRUE;
1154     }
1155 }
1156
1157 static void
1158 account_widget_build_groupwise (EmpathyAccountWidget *self,
1159     const char *filename)
1160 {
1161   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1162
1163   if (priv->simple)
1164     {
1165       self->ui_details->gui = empathy_builder_get_file (filename,
1166           "vbox_groupwise_simple", &self->ui_details->widget,
1167           NULL);
1168
1169       empathy_account_widget_handle_params (self,
1170           "entry_id_simple", "account",
1171           "entry_password_simple", "password",
1172           NULL);
1173
1174       self->ui_details->default_focus = g_strdup ("entry_id_simple");
1175     }
1176   else
1177     {
1178       self->ui_details->gui = empathy_builder_get_file (filename,
1179           "table_common_groupwise_settings", &priv->table_common_settings,
1180           "vbox_groupwise_settings", &self->ui_details->widget,
1181           NULL);
1182
1183       empathy_account_widget_handle_params (self,
1184           "entry_id", "account",
1185           "entry_password", "password",
1186           "entry_server", "server",
1187           "spinbutton_port", "port",
1188           NULL);
1189
1190       self->ui_details->default_focus = g_strdup ("entry_id");
1191       self->ui_details->add_forget = TRUE;
1192     }
1193 }
1194
1195 static void
1196 account_widget_destroy_cb (GtkWidget *widget,
1197     EmpathyAccountWidget *self)
1198 {
1199   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1200   /* set the destroyed flag - workaround */
1201   priv->destroyed = TRUE;
1202
1203   g_object_unref (self);
1204 }
1205
1206 static void
1207 empathy_account_widget_enabled_cb (TpAccount *account,
1208       GParamSpec *spec,
1209       gpointer user_data)
1210 {
1211   EmpathyAccountWidget *widget = EMPATHY_ACCOUNT_WIDGET (user_data);
1212   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1213   gboolean enabled = tp_account_is_enabled (account);
1214
1215   if (priv->enabled_checkbox != NULL)
1216     {
1217 #ifdef HAVE_MOBLIN
1218       nbtk_gtk_light_switch_set_active (
1219           NBTK_GTK_LIGHT_SWITCH (priv->enabled_checkbox),
1220           enabled);
1221 #else
1222       gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->enabled_checkbox),
1223           enabled);
1224 #endif /* HAVE_MOBLIN */
1225     }
1226 }
1227
1228 static void
1229 #ifdef HAVE_MOBLIN
1230 account_widget_switch_flipped_cb (NbtkGtkLightSwitch *sw,
1231     gboolean state,
1232     gpointer user_data)
1233 #else
1234 account_widget_enabled_released_cb (GtkToggleButton *toggle_button,
1235     gpointer user_data)
1236 #endif /* HAVE_MOBLIN */
1237 {
1238   EmpathyAccountWidgetPriv *priv = GET_PRIV (user_data);
1239   TpAccount *account;
1240 #ifndef HAVE_MOBLIN
1241   gboolean state;
1242
1243   state = gtk_toggle_button_get_active (toggle_button);
1244 #endif
1245
1246   account = empathy_account_settings_get_account (priv->settings);
1247
1248   /* Enable the account according to the value of the "Enabled" checkbox */
1249   /* workaround to keep widget alive during async call */
1250   g_object_ref (user_data);
1251   tp_account_set_enabled_async (account, state,
1252       account_widget_account_enabled_cb, user_data);
1253 }
1254
1255 static void
1256 do_set_property (GObject *object,
1257     guint prop_id,
1258     const GValue *value,
1259     GParamSpec *pspec)
1260 {
1261   EmpathyAccountWidgetPriv *priv = GET_PRIV (object);
1262
1263   switch (prop_id)
1264     {
1265     case PROP_SETTINGS:
1266       priv->settings = g_value_dup_object (value);
1267       break;
1268     case PROP_SIMPLE:
1269       priv->simple = g_value_get_boolean (value);
1270       break;
1271     case PROP_CREATING_ACCOUNT:
1272       priv->creating_account = g_value_get_boolean (value);
1273       break;
1274     default:
1275       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1276     }
1277 }
1278
1279 static void
1280 do_get_property (GObject *object,
1281     guint prop_id,
1282     GValue *value,
1283     GParamSpec *pspec)
1284 {
1285   EmpathyAccountWidgetPriv *priv = GET_PRIV (object);
1286
1287   switch (prop_id)
1288     {
1289     case PROP_PROTOCOL:
1290       g_value_set_string (value,
1291         empathy_account_settings_get_protocol (priv->settings));
1292       break;
1293     case PROP_SETTINGS:
1294       g_value_set_object (value, priv->settings);
1295       break;
1296     case PROP_SIMPLE:
1297       g_value_set_boolean (value, priv->simple);
1298       break;
1299     case PROP_CREATING_ACCOUNT:
1300       g_value_set_boolean (value, priv->creating_account);
1301       break;
1302     default:
1303       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1304     }
1305 }
1306
1307 static void
1308 presence_changed_cb (TpAccountManager *manager,
1309     TpConnectionPresenceType state,
1310     const gchar *status,
1311     const gchar *message,
1312     EmpathyAccountWidget *self)
1313 {
1314   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1315
1316   if (priv->destroyed)
1317     return;
1318
1319   if (priv->apply_button == NULL)
1320     /* This button doesn't exist in 'simple' mode */
1321     return;
1322
1323   if (state > TP_CONNECTION_PRESENCE_TYPE_OFFLINE)
1324     {
1325       /* We are online, display a Login button */
1326       GtkWidget *image;
1327
1328       gtk_button_set_use_stock (GTK_BUTTON (priv->apply_button), FALSE);
1329       gtk_button_set_label (GTK_BUTTON (priv->apply_button), _("L_og in"));
1330
1331       image = gtk_image_new_from_stock (GTK_STOCK_CONNECT,
1332           GTK_ICON_SIZE_BUTTON);
1333       gtk_button_set_image (GTK_BUTTON (priv->apply_button), image);
1334     }
1335   else
1336     {
1337       /* We are offline, display a Save button */
1338       gtk_button_set_image (GTK_BUTTON (priv->apply_button), NULL);
1339       gtk_button_set_use_stock (GTK_BUTTON (priv->apply_button), TRUE);
1340       gtk_button_set_label (GTK_BUTTON (priv->apply_button), GTK_STOCK_SAVE);
1341     }
1342 }
1343
1344 static void
1345 account_manager_ready_cb (GObject *source_object,
1346     GAsyncResult *result,
1347     gpointer user_data)
1348 {
1349   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (user_data);
1350   TpAccountManager *account_manager = TP_ACCOUNT_MANAGER (source_object);
1351   GError *error = NULL;
1352   TpConnectionPresenceType state;
1353
1354   if (!tp_account_manager_prepare_finish (account_manager, result, &error))
1355     {
1356       DEBUG ("Failed to prepare account manager: %s", error->message);
1357       g_error_free (error);
1358       goto out;
1359     }
1360
1361   state = tp_account_manager_get_most_available_presence (account_manager, NULL,
1362       NULL);
1363
1364   /* simulate a presence change so the apply button will be changed
1365    * if needed */
1366   presence_changed_cb (account_manager, state, NULL, NULL, self);
1367
1368 out:
1369   g_object_unref (self);
1370 }
1371
1372 #define WIDGET(cm, proto) \
1373   { #cm, #proto, "empathy-account-widget-"#proto".ui", \
1374     account_widget_build_##proto }
1375
1376 static void
1377 add_enable_checkbox (EmpathyAccountWidget *self,
1378     TpAccount *account)
1379 {
1380   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1381 #ifdef HAVE_MOBLIN
1382   GtkWidget *w;
1383 #else
1384   GtkWidget *vbox = self->ui_details->widget;
1385 #endif
1386   guint nb_rows, nb_columns;
1387   gboolean is_enabled;
1388
1389   /* handle the "Enabled" checkbox. We only add it when modifying an account */
1390   if (priv->creating_account || priv->table_common_settings == NULL)
1391     return;
1392
1393   is_enabled = tp_account_is_enabled (account);
1394
1395 #ifdef HAVE_MOBLIN
1396   w = gtk_label_new (_("Account:"));
1397   gtk_misc_set_alignment (GTK_MISC (w), 0, 0.5);
1398
1399   priv->enabled_checkbox = nbtk_gtk_light_switch_new ();
1400
1401   nbtk_gtk_light_switch_set_active (
1402       NBTK_GTK_LIGHT_SWITCH (priv->enabled_checkbox), is_enabled);
1403
1404   gtk_widget_show (w);
1405 #else
1406   priv->enabled_checkbox =
1407       gtk_check_button_new_with_label (_("Enabled"));
1408
1409   gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->enabled_checkbox),
1410       is_enabled);
1411 #endif /* HAVE_MOBLIN */
1412
1413   g_object_get (priv->table_common_settings, "n-rows", &nb_rows,
1414       "n-columns", &nb_columns, NULL);
1415
1416   gtk_table_resize (GTK_TABLE (priv->table_common_settings), ++nb_rows,
1417       nb_columns);
1418
1419 #ifdef HAVE_MOBLIN
1420   gtk_table_attach (GTK_TABLE (priv->table_common_settings),
1421       w,
1422       0, 1, nb_rows - 1, nb_rows,
1423       GTK_FILL, 0, 0, 0);
1424   gtk_table_attach (GTK_TABLE (priv->table_common_settings),
1425       priv->enabled_checkbox,
1426       1, nb_columns, nb_rows - 1, nb_rows,
1427       GTK_EXPAND | GTK_FILL, 0, 0, 0);
1428 #else
1429   gtk_box_pack_start (GTK_BOX (vbox), priv->enabled_checkbox, FALSE, FALSE, 0);
1430   gtk_box_reorder_child (GTK_BOX (vbox), priv->enabled_checkbox, 0);
1431 #endif /* HAVE_MOBLIN */
1432
1433   gtk_widget_show (priv->enabled_checkbox);
1434
1435 #ifdef HAVE_MOBLIN
1436   g_signal_connect (G_OBJECT (priv->enabled_checkbox), "switch-flipped",
1437       G_CALLBACK (account_widget_switch_flipped_cb), self);
1438 #else
1439   g_signal_connect (G_OBJECT (priv->enabled_checkbox), "released",
1440       G_CALLBACK (account_widget_enabled_released_cb), self);
1441 #endif /* HAVE_MOBLIN */
1442 }
1443
1444 #ifndef HAVE_MOBLIN
1445 /* Moblin doesn't support registration */
1446 static void
1447 add_register_buttons (EmpathyAccountWidget *self,
1448     TpAccount *account)
1449 {
1450   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1451   const TpConnectionManagerProtocol *protocol;
1452   GtkWidget *radiobutton_register;
1453   GtkWidget *vbox = self->ui_details->widget;
1454
1455   if (!priv->creating_account)
1456     return;
1457
1458   protocol = empathy_account_settings_get_tp_protocol (priv->settings);
1459   if (protocol == NULL)
1460     return;
1461
1462   if (!tp_connection_manager_protocol_can_register (protocol))
1463     return;
1464
1465   if (account_widget_is_gtalk (self))
1466     return;
1467
1468   if (priv->simple)
1469     return;
1470
1471   priv->radiobutton_reuse = gtk_radio_button_new_with_label (NULL,
1472       _("This account already exists on the server"));
1473   radiobutton_register = gtk_radio_button_new_with_label (
1474       gtk_radio_button_get_group (GTK_RADIO_BUTTON (priv->radiobutton_reuse)),
1475       _("Create a new account on the server"));
1476
1477   gtk_box_pack_start (GTK_BOX (vbox), priv->radiobutton_reuse, FALSE, FALSE, 0);
1478   gtk_box_pack_start (GTK_BOX (vbox), radiobutton_register, FALSE, FALSE, 0);
1479   gtk_box_reorder_child (GTK_BOX (vbox), priv->radiobutton_reuse, 0);
1480   gtk_box_reorder_child (GTK_BOX (vbox), radiobutton_register, 1);
1481   gtk_widget_show (priv->radiobutton_reuse);
1482   gtk_widget_show (radiobutton_register);
1483 }
1484 #endif
1485
1486 static void
1487 do_constructed (GObject *obj)
1488 {
1489   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1490   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1491   TpAccount *account;
1492   const gchar *protocol, *cm_name;
1493   guint i = 0;
1494   struct {
1495     const gchar *cm_name;
1496     const gchar *protocol;
1497     const char *file;
1498     void (*func)(EmpathyAccountWidget *self, const gchar *filename);
1499   } widgets [] = {
1500     { "salut", "local-xmpp", "empathy-account-widget-local-xmpp.ui",
1501         account_widget_build_salut },
1502     WIDGET (gabble, jabber),
1503     WIDGET (butterfly, msn),
1504     WIDGET (haze, icq),
1505     WIDGET (haze, aim),
1506     WIDGET (haze, yahoo),
1507     WIDGET (haze, groupwise),
1508     WIDGET (idle, irc),
1509     WIDGET (sofiasip, sip),
1510   };
1511
1512   cm_name = empathy_account_settings_get_cm (priv->settings);
1513   protocol = empathy_account_settings_get_protocol (priv->settings);
1514
1515   for (i = 0 ; i < G_N_ELEMENTS (widgets); i++)
1516     {
1517       if (!tp_strdiff (widgets[i].cm_name, cm_name) &&
1518           !tp_strdiff (widgets[i].protocol, protocol))
1519         {
1520           gchar *filename;
1521
1522           filename = empathy_file_lookup (widgets[i].file,
1523               "libempathy-gtk");
1524           widgets[i].func (self, filename);
1525           g_free (filename);
1526
1527           break;
1528         }
1529     }
1530
1531   if (i == G_N_ELEMENTS (widgets))
1532     {
1533       gchar *filename = empathy_file_lookup (
1534           "empathy-account-widget-generic.ui", "libempathy-gtk");
1535       account_widget_build_generic (self, filename);
1536       g_free (filename);
1537     }
1538
1539   /* handle default focus */
1540   if (self->ui_details->default_focus != NULL)
1541     {
1542       GObject *default_focus_entry;
1543
1544       default_focus_entry = gtk_builder_get_object
1545         (self->ui_details->gui, self->ui_details->default_focus);
1546       g_signal_connect (default_focus_entry, "realize",
1547           G_CALLBACK (gtk_widget_grab_focus),
1548           NULL);
1549     }
1550
1551   /* handle forget button */
1552   if (self->ui_details->add_forget)
1553     {
1554       const gchar *password = NULL;
1555
1556       priv->button_forget = GTK_WIDGET (gtk_builder_get_object
1557           (self->ui_details->gui, "button_forget"));
1558       priv->entry_password = GTK_WIDGET (gtk_builder_get_object
1559           (self->ui_details->gui, "entry_password"));
1560
1561       password = empathy_account_settings_get_string (priv->settings,
1562           "password");
1563       gtk_widget_set_sensitive (priv->button_forget,
1564           !EMP_STR_EMPTY (password));
1565
1566       g_signal_connect (priv->button_forget, "clicked",
1567           G_CALLBACK (account_widget_forget_clicked_cb),
1568           self);
1569       g_signal_connect (priv->entry_password, "changed",
1570           G_CALLBACK (account_widget_password_changed_cb),
1571           self);
1572     }
1573
1574   /* dup and init the account-manager */
1575   priv->account_manager = tp_account_manager_dup ();
1576
1577   g_object_ref (self);
1578   tp_account_manager_prepare_async (priv->account_manager, NULL,
1579       account_manager_ready_cb, self);
1580
1581   /* handle apply and cancel button */
1582   if (!priv->simple)
1583     {
1584       GtkWidget *hbox = gtk_hbox_new (TRUE, 3);
1585
1586       priv->cancel_button = gtk_button_new_from_stock (GTK_STOCK_CANCEL);
1587
1588       if (priv->creating_account)
1589         {
1590           /* Assumre we are offline, display a Save button. We'll update
1591            * it once the account manager is ready if needed */
1592           priv->apply_button = gtk_button_new_from_stock (GTK_STOCK_SAVE);
1593
1594           empathy_signal_connect_weak (priv->account_manager,
1595               "most-available-presence-changed",
1596               G_CALLBACK (presence_changed_cb), obj);
1597         }
1598       else
1599         {
1600           /* We are editing an existing account, display an Apply button */
1601           priv->apply_button = gtk_button_new_from_stock (GTK_STOCK_APPLY);
1602         }
1603
1604       gtk_box_pack_end (GTK_BOX (hbox), priv->apply_button, TRUE,
1605           TRUE, 3);
1606       gtk_box_pack_end (GTK_BOX (hbox), priv->cancel_button, TRUE,
1607           TRUE, 3);
1608
1609       gtk_box_pack_end (GTK_BOX (self->ui_details->widget), hbox, FALSE,
1610           FALSE, 3);
1611
1612       g_signal_connect (priv->cancel_button, "clicked",
1613           G_CALLBACK (account_widget_cancel_clicked_cb),
1614           self);
1615       g_signal_connect (priv->apply_button, "clicked",
1616           G_CALLBACK (account_widget_apply_clicked_cb),
1617           self);
1618       gtk_widget_show_all (hbox);
1619
1620       if (priv->creating_account)
1621         /* When creating an account, the user might have nothing to enter.
1622          * That means that no control interaction might occur,
1623          * so we update the control button sensitivity manually.
1624          */
1625         account_widget_handle_control_buttons_sensitivity (self);
1626       else
1627         account_widget_set_control_buttons_sensitivity (self, FALSE);
1628     }
1629
1630   account = empathy_account_settings_get_account (priv->settings);
1631
1632   if (account != NULL)
1633     {
1634       g_signal_connect (account, "notify::enabled",
1635           G_CALLBACK (empathy_account_widget_enabled_cb), self);
1636     }
1637
1638 #ifndef HAVE_MOBLIN
1639   add_register_buttons (self, account);
1640 #endif
1641   add_enable_checkbox (self, account);
1642
1643   /* hook up to widget destruction to unref ourselves */
1644   g_signal_connect (self->ui_details->widget, "destroy",
1645       G_CALLBACK (account_widget_destroy_cb), self);
1646
1647   empathy_builder_unref_and_keep_widget (self->ui_details->gui,
1648       self->ui_details->widget);
1649   self->ui_details->gui = NULL;
1650 }
1651
1652 static void
1653 do_dispose (GObject *obj)
1654 {
1655   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1656   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1657
1658   if (priv->dispose_run)
1659     return;
1660
1661   priv->dispose_run = TRUE;
1662
1663   empathy_account_settings_is_ready (priv->settings);
1664
1665   if (priv->settings != NULL)
1666     {
1667       TpAccount *account;
1668       account = empathy_account_settings_get_account (priv->settings);
1669
1670       if (account != NULL)
1671         {
1672           g_signal_handlers_disconnect_by_func (account,
1673               empathy_account_widget_enabled_cb, self);
1674         }
1675
1676       g_object_unref (priv->settings);
1677       priv->settings = NULL;
1678     }
1679
1680   if (priv->account_manager != NULL)
1681     {
1682       g_object_unref (priv->account_manager);
1683       priv->account_manager = NULL;
1684     }
1685
1686   if (G_OBJECT_CLASS (empathy_account_widget_parent_class)->dispose != NULL)
1687     G_OBJECT_CLASS (empathy_account_widget_parent_class)->dispose (obj);
1688 }
1689
1690 static void
1691 do_finalize (GObject *obj)
1692 {
1693   EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (obj);
1694
1695   g_free (self->ui_details->default_focus);
1696   g_slice_free (EmpathyAccountWidgetUIDetails, self->ui_details);
1697
1698   if (G_OBJECT_CLASS (empathy_account_widget_parent_class)->finalize != NULL)
1699     G_OBJECT_CLASS (empathy_account_widget_parent_class)->finalize (obj);
1700 }
1701
1702 static void
1703 empathy_account_widget_class_init (EmpathyAccountWidgetClass *klass)
1704 {
1705   GObjectClass *oclass = G_OBJECT_CLASS (klass);
1706   GParamSpec *param_spec;
1707
1708   oclass->get_property = do_get_property;
1709   oclass->set_property = do_set_property;
1710   oclass->constructed = do_constructed;
1711   oclass->dispose = do_dispose;
1712   oclass->finalize = do_finalize;
1713
1714   param_spec = g_param_spec_string ("protocol",
1715       "protocol", "The protocol of the account",
1716       NULL,
1717       G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
1718   g_object_class_install_property (oclass, PROP_PROTOCOL, param_spec);
1719
1720   param_spec = g_param_spec_object ("settings",
1721       "settings", "The settings of the account",
1722       EMPATHY_TYPE_ACCOUNT_SETTINGS,
1723       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1724   g_object_class_install_property (oclass, PROP_SETTINGS, param_spec);
1725
1726   param_spec = g_param_spec_boolean ("simple",
1727       "simple", "Whether the account widget is a simple or an advanced one",
1728       FALSE,
1729       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1730   g_object_class_install_property (oclass, PROP_SIMPLE, param_spec);
1731
1732   param_spec = g_param_spec_boolean ("creating-account",
1733       "creating-account",
1734       "TRUE if we're creating an account, FALSE if we're modifying it",
1735       FALSE,
1736       G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT_ONLY);
1737   g_object_class_install_property (oclass, PROP_CREATING_ACCOUNT, param_spec);
1738
1739   signals[HANDLE_APPLY] =
1740     g_signal_new ("handle-apply", G_TYPE_FROM_CLASS (klass),
1741         G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1742         g_cclosure_marshal_VOID__BOOLEAN,
1743         G_TYPE_NONE,
1744         1, G_TYPE_BOOLEAN);
1745
1746   /* This signal is emitted when an account has been created and enabled. */
1747   signals[ACCOUNT_CREATED] =
1748       g_signal_new ("account-created", G_TYPE_FROM_CLASS (klass),
1749           G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1750           g_cclosure_marshal_VOID__POINTER,
1751           G_TYPE_NONE,
1752           1, G_TYPE_OBJECT);
1753
1754   signals[CANCELLED] =
1755       g_signal_new ("cancelled", G_TYPE_FROM_CLASS (klass),
1756           G_SIGNAL_RUN_LAST, 0, NULL, NULL,
1757           g_cclosure_marshal_VOID__VOID,
1758           G_TYPE_NONE,
1759           0);
1760
1761   g_type_class_add_private (klass, sizeof (EmpathyAccountWidgetPriv));
1762 }
1763
1764 static void
1765 empathy_account_widget_init (EmpathyAccountWidget *self)
1766 {
1767   EmpathyAccountWidgetPriv *priv =
1768     G_TYPE_INSTANCE_GET_PRIVATE ((self), EMPATHY_TYPE_ACCOUNT_WIDGET,
1769         EmpathyAccountWidgetPriv);
1770
1771   self->priv = priv;
1772   priv->dispose_run = FALSE;
1773
1774   self->ui_details = g_slice_new0 (EmpathyAccountWidgetUIDetails);
1775 }
1776
1777 /* public methods */
1778
1779 void
1780 empathy_account_widget_discard_pending_changes
1781     (EmpathyAccountWidget *widget)
1782 {
1783   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1784
1785   empathy_account_settings_discard_changes (priv->settings);
1786   priv->contains_pending_changes = FALSE;
1787 }
1788
1789 gboolean
1790 empathy_account_widget_contains_pending_changes (EmpathyAccountWidget *widget)
1791 {
1792   EmpathyAccountWidgetPriv *priv = GET_PRIV (widget);
1793
1794   return priv->contains_pending_changes;
1795 }
1796
1797 void
1798 empathy_account_widget_handle_params (EmpathyAccountWidget *self,
1799     const gchar *first_widget,
1800     ...)
1801 {
1802   va_list args;
1803
1804   va_start (args, first_widget);
1805   account_widget_handle_params_valist (self, first_widget, args);
1806   va_end (args);
1807 }
1808
1809 GtkWidget *
1810 empathy_account_widget_get_widget (EmpathyAccountWidget *widget)
1811 {
1812   return widget->ui_details->widget;
1813 }
1814
1815 EmpathyAccountWidget *
1816 empathy_account_widget_new_for_protocol (EmpathyAccountSettings *settings,
1817     gboolean simple)
1818 {
1819   EmpathyAccountWidget *self;
1820
1821   g_return_val_if_fail (EMPATHY_IS_ACCOUNT_SETTINGS (settings), NULL);
1822
1823   self = g_object_new
1824     (EMPATHY_TYPE_ACCOUNT_WIDGET,
1825         "settings", settings, "simple", simple,
1826         "creating-account",
1827         empathy_account_settings_get_account (settings) == NULL,
1828         NULL);
1829
1830   return self;
1831 }
1832
1833 gchar *
1834 empathy_account_widget_get_default_display_name (EmpathyAccountWidget *self)
1835 {
1836   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1837   const gchar *login_id;
1838   const gchar *protocol, *p;
1839   gchar *default_display_name;
1840
1841   login_id = empathy_account_settings_get_string (priv->settings, "account");
1842   protocol = empathy_account_settings_get_protocol (priv->settings);
1843
1844   if (login_id != NULL)
1845     {
1846       /* TODO: this should be done in empathy-account-widget-irc */
1847       if (!tp_strdiff (protocol, "irc"))
1848         {
1849           const gchar* server;
1850           server = empathy_account_settings_get_string (priv->settings,
1851               "server");
1852
1853           /* To translators: The first parameter is the login id and the
1854            * second one is the server. The resulting string will be something
1855            * like: "MyUserName on chat.freenode.net".
1856            * You should reverse the order of these arguments if the
1857            * server should come before the login id in your locale.*/
1858           default_display_name = g_strdup_printf (_("%1$s on %2$s"),
1859               login_id, server);
1860         }
1861       else
1862         {
1863           default_display_name = g_strdup (login_id);
1864         }
1865
1866       return default_display_name;
1867     }
1868
1869   if ((p = empathy_protocol_name_to_display_name (protocol)) != NULL)
1870     protocol = p;
1871
1872   if (protocol != NULL)
1873     {
1874       /* To translators: The parameter is the protocol name. The resulting
1875        * string will be something like: "Jabber Account" */
1876       default_display_name = g_strdup_printf (_("%s Account"), protocol);
1877     }
1878   else
1879     {
1880       default_display_name = g_strdup (_("New account"));
1881     }
1882
1883   return default_display_name;
1884 }
1885
1886 /* Used by subclass to indicate that widget contains pending changes */
1887 void
1888 empathy_account_widget_changed (EmpathyAccountWidget *self)
1889 {
1890   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1891
1892   account_widget_handle_control_buttons_sensitivity (self);
1893   priv->contains_pending_changes = TRUE;
1894 }
1895
1896 void
1897 empathy_account_widget_set_account_param (EmpathyAccountWidget *self,
1898     const gchar *account)
1899 {
1900   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1901
1902   if (priv->param_account_widget == NULL)
1903     return;
1904
1905   gtk_entry_set_text (GTK_ENTRY (priv->param_account_widget), account);
1906 }
1907
1908 void
1909 empathy_account_widget_set_password_param (EmpathyAccountWidget *self,
1910     const gchar *account)
1911 {
1912   EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
1913
1914   if (priv->param_password_widget == NULL)
1915     return;
1916
1917   gtk_entry_set_text (GTK_ENTRY (priv->param_password_widget), account);
1918 }