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