* src/modest-tny-account-store.[ch]:
[modest] / src / maemo / modest-connection-specific-smtp-window.c
1 /* Copyright (c) 2006, Nokia Corporation
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  * * Redistributions of source code must retain the above copyright
9  *   notice, this list of conditions and the following disclaimer.
10  * * Redistributions in binary form must reproduce the above copyright
11  *   notice, this list of conditions and the following disclaimer in the
12  *   documentation and/or other materials provided with the distribution.
13  * * Neither the name of the Nokia Corporation nor the names of its
14  *   contributors may be used to endorse or promote products derived from
15  *   this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
18  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
20  * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
21  * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
24  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
25  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
26  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29
30 #include "modest-connection-specific-smtp-window.h"
31 #include "modest-connection-specific-smtp-edit-window.h"
32 #include <modest-account-mgr-helpers.h>
33 #include "widgets/modest-ui-constants.h"
34
35 #include <modest-runtime.h>
36 #include <tny-maemo-conic-device.h>
37
38 #include <gtk/gtktreeview.h>
39 #include <gtk/gtkcellrenderertext.h>
40 #include <gtk/gtkliststore.h>
41 #include <gtk/gtkscrolledwindow.h>
42 #include <gtk/gtkbutton.h>
43 #include <gtk/gtkhbox.h>
44 #include <gtk/gtkvbox.h>
45 #include <gtk/gtkstock.h>
46
47 #include "modest-hildon-includes.h"
48 #include "modest-platform.h"
49 #include "modest-maemo-utils.h"
50
51 #include <glib/gi18n.h>
52 #include <string.h>
53
54 G_DEFINE_TYPE (ModestConnectionSpecificSmtpWindow, modest_connection_specific_smtp_window,
55                GTK_TYPE_DIALOG);
56
57 #define CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE(o) \
58         (G_TYPE_INSTANCE_GET_PRIVATE ((o), MODEST_TYPE_CONNECTION_SPECIFIC_SMTP_WINDOW, ModestConnectionSpecificSmtpWindowPrivate))
59
60 typedef struct _ModestConnectionSpecificSmtpWindowPrivate ModestConnectionSpecificSmtpWindowPrivate;
61
62 struct _ModestConnectionSpecificSmtpWindowPrivate
63 {
64         GtkTreeView *treeview;
65         GtkTreeModel *model;
66         GtkWidget *button_edit;
67         
68         ModestAccountMgr *account_manager;
69 };
70
71 static gboolean on_key_pressed (GtkWidget *self, GdkEventKey *event, gpointer user_data);
72
73 static void
74 modest_connection_specific_smtp_window_get_property (GObject *object, guint property_id,
75                                                                                                                         GValue *value, GParamSpec *pspec)
76 {
77         switch (property_id) {
78         default:
79                 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
80         }
81 }
82
83 static void
84 modest_connection_specific_smtp_window_set_property (GObject *object, guint property_id,
85                                                                                                                         const GValue *value, GParamSpec *pspec)
86 {
87         switch (property_id) {
88         default:
89                 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
90         }
91 }
92
93 static void
94 modest_connection_specific_smtp_window_dispose (GObject *object)
95 {
96         if (G_OBJECT_CLASS (modest_connection_specific_smtp_window_parent_class)->dispose)
97                 G_OBJECT_CLASS (modest_connection_specific_smtp_window_parent_class)->dispose (object);
98 }
99
100 enum MODEL_COLS {
101         MODEL_COL_NAME = 0, /* libconic IAP Name: a string */
102         MODEL_COL_ID = 1, /* libconic IAP ID: a string */
103         MODEL_COL_SERVER_ACCOUNT_NAME = 2, /* a string */
104         MODEL_COL_SERVER_NAME = 3, /* a string */
105         MODEL_COL_SERVER_ACCOUNT_DATA = 4 /* a gpointer */
106 };
107
108
109 void update_model_server_names (ModestConnectionSpecificSmtpWindow *self);
110
111 static void
112 modest_connection_specific_smtp_window_finalize (GObject *object)
113 {
114         ModestConnectionSpecificSmtpWindowPrivate *priv = CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (object);
115         
116         /* Free all the data items from the treemodel: */
117         GtkTreeIter iter;
118         gboolean valid = gtk_tree_model_get_iter_first (priv->model, &iter);
119         while (valid) {
120                 ModestServerAccountData *data = NULL;
121                 
122                 gtk_tree_model_get (priv->model, &iter, 
123                                     MODEL_COL_SERVER_ACCOUNT_DATA, &data,
124                                     -1);
125                                  
126                 if (data)
127                         modest_account_mgr_free_server_account_data (priv->account_manager, data);
128                         
129                 /* Get next row: */
130                 valid = gtk_tree_model_iter_next (priv->model, &iter);
131         }
132         
133         g_object_unref (G_OBJECT (priv->model));
134         
135         G_OBJECT_CLASS (modest_connection_specific_smtp_window_parent_class)->finalize (object);
136 }
137
138 static void
139 modest_connection_specific_smtp_window_class_init (ModestConnectionSpecificSmtpWindowClass *klass)
140 {
141         GObjectClass *object_class = G_OBJECT_CLASS (klass);
142
143         g_type_class_add_private (klass, sizeof (ModestConnectionSpecificSmtpWindowPrivate));
144
145         object_class->get_property = modest_connection_specific_smtp_window_get_property;
146         object_class->set_property = modest_connection_specific_smtp_window_set_property;
147         object_class->dispose = modest_connection_specific_smtp_window_dispose;
148         object_class->finalize = modest_connection_specific_smtp_window_finalize;
149 }
150
151 /* libconic does not return a list of connections in scratchbox,
152  * so enable this to put a fake row in the list,
153  * so we can test other parts of the code. */
154 /* #define DEBUG_WITHOUT_LIBCONIC 1 */
155
156 void
157 modest_connection_specific_smtp_window_fill_with_connections (ModestConnectionSpecificSmtpWindow *self, ModestAccountMgr *account_manager)
158 {
159         ModestConnectionSpecificSmtpWindowPrivate *priv = 
160                 CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
161         priv->account_manager = account_manager;
162         
163         GtkListStore *liststore = GTK_LIST_STORE (priv->model);
164         
165         TnyDevice *device = modest_runtime_get_device ();
166         g_assert (TNY_IS_MAEMO_CONIC_DEVICE (device));
167         
168         /* Get the list of Internet Access Points: */
169         #ifdef DEBUG_WITHOUT_LIBCONIC
170         GSList *list_iaps = g_slist_append(NULL, (gpointer)1);
171         #else
172         TnyMaemoConicDevice *maemo_device = TNY_MAEMO_CONIC_DEVICE (device);
173         GSList *list_iaps = tny_maemo_conic_device_get_iap_list (maemo_device);
174         #endif
175         
176         /* printf("debug: list_iaps=%p, list_iaps size = %d\n", list_iaps, g_slist_length(list_iaps)); */
177         
178         GSList* iter = list_iaps;
179         while (iter) {
180                 ConIcIap *iap = (ConIcIap*)iter->data;
181                 if (iap) {
182                         #ifdef DEBUG_WITHOUT_LIBCONIC
183                         const gchar *connection_name = "debug name";
184                         const gchar *connection_id = "debug id";
185                         #else
186                         const gchar *connection_name = con_ic_iap_get_name (iap);
187                         const gchar *connection_id = con_ic_iap_get_id (iap);
188                         #endif
189                         
190                         printf ("debug: iac name=%s, id=%s\n", connection_name, connection_id);
191                         
192                         /* Get any already-associated connection-specific server account: */
193                         gchar *server_account_name = NULL;
194                         server_account_name = modest_account_mgr_get_connection_specific_smtp (
195                                 priv->account_manager, connection_name);
196                                         
197                         /* Add the row to the model: */
198                         GtkTreeIter iter;
199                         gtk_list_store_append (liststore, &iter);
200                         gtk_list_store_set(liststore, &iter, 
201                                 MODEL_COL_ID, connection_id, 
202                                 MODEL_COL_NAME, connection_name,
203                                 MODEL_COL_SERVER_ACCOUNT_NAME, server_account_name,
204                                 -1);
205                                 
206                         g_free (server_account_name);
207                 }
208                 
209                 iter = g_slist_next (iter);     
210         }
211                 
212         #ifndef DEBUG_WITHOUT_LIBCONIC
213         if (list_iaps)
214                 tny_maemo_conic_device_free_iap_list (maemo_device, list_iaps);
215         #endif
216                 
217         update_model_server_names (self);
218 }
219         
220 static void
221 on_button_edit (GtkButton *button, gpointer user_data)
222 {
223         ModestConnectionSpecificSmtpWindow *self = MODEST_CONNECTION_SPECIFIC_SMTP_WINDOW (user_data);
224         ModestConnectionSpecificSmtpWindowPrivate *priv = CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
225         
226         gchar *id = NULL;
227         gchar *connection_name = NULL;
228         gchar *server_account_name = NULL;
229         ModestServerAccountData *data = NULL;
230         GtkTreeSelection *sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->treeview));
231         GtkTreeIter iter;
232         GtkTreeModel *model = 0;
233         if (gtk_tree_selection_get_selected (sel, &model, &iter)) {
234                 gtk_tree_model_get (priv->model, &iter, 
235                                     MODEL_COL_ID, &id, 
236                                     MODEL_COL_NAME, &connection_name, 
237                                     MODEL_COL_SERVER_ACCOUNT_NAME, &server_account_name,
238                                     MODEL_COL_SERVER_ACCOUNT_DATA, &data,
239                                     -1);
240         
241                 /* printf("DEBUG: %s: BEFORE: connection-specific server_account_name=%s\n", __FUNCTION__, server_account_name); */
242                 /* TODO: Is 0 an allowed libconic IAP ID? 
243                  * If not then we should check for it. */
244                 
245                 /* Get existing server account data if a server account is already specified: */
246                 gboolean data_was_retrieved = FALSE;
247                 if (server_account_name && !data) {
248                         data = modest_account_mgr_get_server_account_data (priv->account_manager, 
249                                 server_account_name);
250                         if (data)
251                                 data_was_retrieved = TRUE;
252                 }
253                 
254                 GtkWidget * window = GTK_WIDGET (modest_connection_specific_smtp_edit_window_new ());
255                 modest_connection_specific_smtp_edit_window_set_connection (
256                         MODEST_CONNECTION_SPECIFIC_SMTP_EDIT_WINDOW (window), id, connection_name, data);
257                         
258                 /* Delete data, unless it was data from the rowmodel: */
259                 if (data_was_retrieved) {
260                         modest_account_mgr_free_server_account_data (priv->account_manager, data);
261                         data = NULL;
262                 }
263                         
264                 gtk_window_set_transient_for (GTK_WINDOW (self), GTK_WINDOW (window));
265                 
266                 gboolean dialog_finished = FALSE;
267                 while (!dialog_finished)
268                 {
269                         gint response = gtk_dialog_run (GTK_DIALOG (window));
270                         if (response == GTK_RESPONSE_OK) {
271                                 gtk_widget_hide (window);
272                                 dialog_finished = TRUE;
273                                 /* Delete any previous data for this row: */
274                                 if (data) 
275                                 {
276                                         modest_account_mgr_free_server_account_data (priv->account_manager, data);
277                                         data = NULL;
278                                 }
279                                 
280                                 /* Get the new account data and save it in the row for later:
281                                  * We free this in finalize(),
282                                  * and save it to our configuration in 
283                                  * modest_connection_specific_smtp_window_save_server_accounts(). */
284                                 data = modest_connection_specific_smtp_edit_window_get_settings (
285                                                         MODEST_CONNECTION_SPECIFIC_SMTP_EDIT_WINDOW (window), 
286                                                         priv->account_manager);
287                                 
288                                 if (data) {
289                                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
290                                                             MODEL_COL_SERVER_ACCOUNT_DATA, data,
291                                                             MODEL_COL_SERVER_NAME, data->hostname,
292                                                             -1);
293                                 } else {
294                                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
295                                                             MODEL_COL_SERVER_ACCOUNT_DATA, NULL,
296                                                             MODEL_COL_SERVER_NAME, NULL,
297                                                             MODEL_COL_SERVER_ACCOUNT_NAME, NULL,
298                                                             -1);
299                                 }
300                         }
301                         else
302                         {
303                                 if (!modest_connection_specific_smtp_edit_window_is_dirty(
304                                                 MODEST_CONNECTION_SPECIFIC_SMTP_EDIT_WINDOW(window)))
305                                 {
306                                         gtk_widget_hide(window);
307                                         dialog_finished = TRUE;
308                                 }
309                                 else
310                                 {
311                                                 
312                                         gint response;
313                                         response = modest_platform_run_confirmation_dialog (GTK_WINDOW (window), 
314                                                                             _("imum_nc_wizard_confirm_lose_changes"));                   
315                                         if (response == GTK_RESPONSE_OK)
316                                         {
317                                                 gtk_widget_hide(window);
318                                                 dialog_finished = TRUE;
319                                         }
320                                 }
321                         }
322                 }
323         }
324         g_free (connection_name);
325         g_free (id);
326         g_free (server_account_name);
327         update_model_server_names (self);
328 }
329
330 static void
331 on_button_cancel (GtkButton *button, gpointer user_data)
332 {
333         ModestConnectionSpecificSmtpWindow *self = MODEST_CONNECTION_SPECIFIC_SMTP_WINDOW (user_data);
334
335         /* Hide the window.
336          * The code that showed it will respond to the hide signal. */  
337         gtk_widget_hide (GTK_WIDGET (self));
338 }
339
340 static void
341 on_selection_changed (GtkTreeSelection *sel, ModestConnectionSpecificSmtpWindow *self)
342 {
343         ModestConnectionSpecificSmtpWindowPrivate *priv = 
344                 CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
345
346         GtkTreeModel *model = NULL;
347         GtkTreeIter iter;
348         const gboolean has_selection =
349                 gtk_tree_selection_get_selected (sel, &model, &iter);
350
351         gtk_widget_set_sensitive (priv->button_edit, has_selection);
352 }
353
354 static void
355 modest_connection_specific_smtp_window_init (ModestConnectionSpecificSmtpWindow *self)
356 {
357         ModestWindowMgr *mgr;
358
359         /* Specify a default size, because the GtkTreeView's default requested size  
360          * is not big enough: */
361         gtk_window_set_default_size (GTK_WINDOW (self), 500, 300);
362         
363         /* This seems to be necessary to make the window show at the front with decoration.
364          * If we use property type=GTK_WINDOW_TOPLEVEL instead of the default GTK_WINDOW_POPUP+decoration, 
365          * then the window will be below the others. */
366         gtk_window_set_type_hint (GTK_WINDOW (self),
367                             GDK_WINDOW_TYPE_HINT_DIALOG);
368                             
369         ModestConnectionSpecificSmtpWindowPrivate *priv = 
370                 CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
371
372         /* Create a tree model for the tree view:
373          * with a string for the name, a string for the server name, and an int for the ID.
374          * This must match our MODEL_COLS enum constants.
375          */
376         priv->model = GTK_TREE_MODEL (gtk_list_store_new (5, 
377                 G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER));
378
379         /* Setup the tree view: */
380         priv->treeview = GTK_TREE_VIEW (gtk_tree_view_new_with_model (priv->model));
381
382         /* Show the column headers,
383          * which does not seem to be the default on Maemo.
384          */                     
385         gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(priv->treeview), TRUE);
386         
387         /* name column:
388          * The ID model column in not shown in the view. */
389         GtkTreeViewColumn *view_column = gtk_tree_view_column_new ();
390         GtkCellRenderer *renderer = gtk_cell_renderer_text_new ();
391         gtk_tree_view_column_pack_start(view_column, renderer, TRUE);
392         gtk_tree_view_column_set_attributes (view_column, renderer, 
393         "text", MODEL_COL_NAME, NULL);
394         gtk_tree_view_column_set_title (view_column, _("mcen_ia_optionalsmtp_connection_name"));
395         gtk_tree_view_append_column (priv->treeview, view_column);
396
397         
398         /* server name column: */
399         view_column = gtk_tree_view_column_new ();
400         renderer = gtk_cell_renderer_text_new ();
401         gtk_tree_view_column_pack_start(view_column, renderer, TRUE);
402         gtk_tree_view_column_set_attributes (view_column, renderer, 
403         "text", MODEL_COL_SERVER_NAME, NULL);
404         gtk_tree_view_column_set_title (view_column, _("mcen_ia_optionalsmtp_servername"));
405         gtk_tree_view_append_column (priv->treeview, view_column);
406         
407         /* The application must call modest_connection_specific_smtp_window_fill_with_connections(). */
408         
409         GtkWidget *vbox = GTK_DIALOG(self)->vbox;
410         //gtk_vbox_new (FALSE, MODEST_MARGIN_DEFAULT);
411
412         /* Introductory note: */
413         /* TODO: For some reason this label does not wrap. It is truncated. */
414         GtkWidget *label = gtk_label_new(_("mcen_ia_optionalsmtp_note"));
415         gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
416         /* So that it is shown without being truncated: */
417         gtk_label_set_max_width_chars (GTK_LABEL (label), 20);
418         /* The documentation for gtk_label_set_line_wrap() says that we must 
419          * call gtk_widget_set_size_request() with a hard-coded width, 
420          * though I wonder why gtk_label_set_max_width_chars() isn't enough. */
421         gtk_widget_set_size_request (label, 400, -1);
422         gtk_widget_show (label);
423         gtk_box_pack_start (GTK_BOX (vbox), label, FALSE, FALSE, MODEST_MARGIN_HALF);
424         
425         /* Put the treeview in a scrolled window and add it to the box: */
426         GtkWidget *scrolled_window = gtk_scrolled_window_new (NULL, NULL);
427         gtk_container_set_border_width (GTK_CONTAINER (scrolled_window), MODEST_MARGIN_DEFAULT);
428         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_NEVER, 
429                 GTK_POLICY_AUTOMATIC);
430         gtk_widget_show (scrolled_window);
431         gtk_container_add (GTK_CONTAINER (scrolled_window), GTK_WIDGET (priv->treeview));
432         gtk_box_pack_start (GTK_BOX (vbox), GTK_WIDGET (scrolled_window), TRUE, TRUE, MODEST_MARGIN_HALF);
433         gtk_widget_show (GTK_WIDGET (priv->treeview));
434         
435         /* Add the buttons: */
436         GtkWidget *hbox = gtk_hbox_new (FALSE, MODEST_MARGIN_HALF);
437         gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, MODEST_MARGIN_HALF);
438         gtk_widget_show (hbox);
439         
440         priv->button_edit = gtk_button_new_from_stock (_("mcen_bd_edit"));
441         gtk_box_pack_start (GTK_BOX (hbox), priv->button_edit, TRUE, FALSE, MODEST_MARGIN_HALF);
442         gtk_widget_show (priv->button_edit);
443         g_signal_connect (G_OBJECT (priv->button_edit), "clicked",
444                 G_CALLBACK (on_button_edit), self);
445         
446         GtkWidget *button_cancel = gtk_button_new_from_stock (_("mcen_bd_close"));
447         gtk_box_pack_start (GTK_BOX (hbox), button_cancel, TRUE, FALSE, MODEST_MARGIN_HALF);
448         gtk_widget_show (button_cancel);
449         g_signal_connect (G_OBJECT (button_cancel), "clicked",
450                 G_CALLBACK (on_button_cancel), self);
451         
452         //gtk_container_add (GTK_CONTAINER (self), GTK_WIDGET (vbox));
453         gtk_widget_show (vbox);
454         
455         /* Disable the Edit button when nothing is selected: */
456         GtkTreeSelection *sel = gtk_tree_view_get_selection (priv->treeview);
457         g_signal_connect (sel, "changed",
458                           G_CALLBACK(on_selection_changed), self);
459         on_selection_changed (sel, self);
460         
461         /* When this window is shown, hibernation should not be possible, 
462          * because there is no sensible way to save the state: */
463         mgr = modest_runtime_get_window_mgr ();
464         modest_window_mgr_prevent_hibernation_while_window_is_shown (mgr, 
465                                                                      GTK_WINDOW (self)); 
466
467         /* Set window title */
468         gtk_window_set_title (GTK_WINDOW (self), _("mcen_ti_optionalsmtp_servers"));
469
470         /* Track key presses to close the window if the Escape is pressed */
471         g_signal_connect (G_OBJECT (self), 
472                           "key-press-event", 
473                           G_CALLBACK (on_key_pressed), NULL);
474         
475         hildon_help_dialog_help_enable (GTK_DIALOG(self),
476                                         "applications_email_connectionsspecificsmtpconf",
477                                         modest_maemo_utils_get_osso_context());
478 }
479
480 ModestConnectionSpecificSmtpWindow*
481 modest_connection_specific_smtp_window_new (void)
482 {
483         return g_object_new (MODEST_TYPE_CONNECTION_SPECIFIC_SMTP_WINDOW, NULL);
484 }
485
486 /** The application should call this when the user changes should be saved.
487  */
488 gboolean
489 modest_connection_specific_smtp_window_save_server_accounts (ModestConnectionSpecificSmtpWindow *self)
490 {
491         ModestConnectionSpecificSmtpWindowPrivate *priv = 
492                 CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
493         
494         
495         /* Get the first iter in the list */
496         GtkTreeIter iter;
497         gboolean valid = gtk_tree_model_get_iter_first (priv->model, &iter);
498
499         /* Walk through the list, reading each row */
500         while (valid) {
501                 gchar *id = NULL;
502                 gchar *connection_name = NULL;
503                 gchar *server_account_name = NULL;
504                 gchar *server_name = NULL;
505                 ModestServerAccountData *data = NULL;
506                 
507                 gtk_tree_model_get (priv->model, &iter, 
508                                     MODEL_COL_ID, &id, 
509                                     MODEL_COL_NAME, &connection_name, 
510                                     MODEL_COL_SERVER_NAME, &server_name,
511                                     MODEL_COL_SERVER_ACCOUNT_NAME, &server_account_name,
512                                     MODEL_COL_SERVER_ACCOUNT_DATA, &data,
513                                     -1);
514                                  
515                 gboolean success = TRUE;   
516                 if (id && data) { /* The presence of data suggests that there is something to save. */
517                         if (!server_account_name) {
518                                 /* Add a new server account, building a (non-human-visible) name: */
519                                 gchar *name_start = g_strdup_printf("specific_%s", connection_name);
520                                 server_account_name = modest_account_mgr_get_unused_account_name (
521                                         priv->account_manager, name_start, TRUE /* server account. */);
522                                 g_assert (server_account_name);
523                                 g_free (name_start);
524                                 
525                                 success = modest_account_mgr_add_server_account (priv->account_manager,
526                                                                                  server_account_name,
527                                                                                  data->hostname, 0,
528                                                                                  data->username, data->password,
529                                                                                  MODEST_PROTOCOL_TRANSPORT_SMTP,
530                                                                                  data->security,
531                                                                                  data->secure_auth);
532                                 if (success) {
533                                         TnyAccount *account = TNY_ACCOUNT (modest_tny_account_store_new_connection_specific_transport_account 
534                                                                            (modest_runtime_get_account_store (),
535                                                                             server_account_name));
536                                         if (account)
537                                                 g_object_unref (account);
538                                 }
539                                         
540                                 /* associate the specific server account with this connection for this account: */
541                                 success = success && modest_account_mgr_set_connection_specific_smtp (
542                                         priv->account_manager, connection_name, server_account_name);
543         
544                                 /* Save the new name in the treemodel, so it can be edited again later: */
545                                 gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
546                                         MODEL_COL_SERVER_ACCOUNT_NAME, server_account_name, -1);
547                                 
548                         } else {
549                                 /* Change an existing server account: */
550                                 modest_account_mgr_set_server_account_hostname (priv->account_manager, server_account_name, 
551                                                                                 data->hostname);
552                                                 
553                                 modest_account_mgr_set_server_account_username (priv->account_manager, server_account_name,
554                                         data->username);
555                                                         
556                                 modest_account_mgr_set_server_account_password (priv->account_manager, server_account_name,
557                                         data->password);
558                                                 
559                                 modest_account_mgr_set_server_account_secure_auth (priv->account_manager, server_account_name, 
560                                         data->secure_auth);
561                                                 
562                                 modest_account_mgr_set_server_account_security (priv->account_manager, server_account_name, 
563                                         data->security);
564
565                                 modest_account_mgr_set_server_account_port (priv->account_manager, server_account_name, data->port);
566                         }
567                 } else if (connection_name && server_name && 
568                            !strcmp (server_name, _("mcen_ia_optionalsmtp_notdefined"))) {
569                         modest_account_mgr_remove_connection_specific_smtp (priv->account_manager, 
570                                                                             connection_name);
571                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
572                                             MODEL_COL_SERVER_ACCOUNT_NAME, NULL, -1);
573                 }
574                 
575                 g_free (connection_name);
576                 g_free (id);
577                 g_free (server_account_name);
578                 g_free (server_name);
579                 
580                 if (!success)
581                         return FALSE;
582                         
583                 /* Get next row: */
584                 valid = gtk_tree_model_iter_next (priv->model, &iter);
585         }
586         
587         update_model_server_names (self);
588         
589         return TRUE;
590 }
591
592 void update_model_server_names (ModestConnectionSpecificSmtpWindow *self)
593 {
594         ModestConnectionSpecificSmtpWindowPrivate *priv = CONNECTION_SPECIFIC_SMTP_WINDOW_GET_PRIVATE (self);
595
596         GtkTreeIter iter;
597         gboolean valid = gtk_tree_model_get_iter_first (priv->model, &iter);
598         while (valid) {
599                 
600                 gchar *server_account_name = NULL;
601                 ModestServerAccountData *data = NULL;
602                 gtk_tree_model_get (priv->model, &iter, 
603                                     MODEL_COL_SERVER_ACCOUNT_NAME, &server_account_name,
604                                     MODEL_COL_SERVER_ACCOUNT_DATA, &data,
605                                     -1);
606                 if (server_account_name) {
607                         
608                         /* Get the server hostname and show it in the treemodel: */     
609                         gchar *hostname = modest_account_mgr_get_server_account_hostname (priv->account_manager, 
610                                                                                           server_account_name);
611                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
612                                             MODEL_COL_SERVER_NAME, hostname,
613                                             -1);
614                         g_free (hostname);
615                 } else if (data && data->hostname && (data->hostname[0] != '\0')) {
616                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, 
617                                             MODEL_COL_SERVER_NAME, data->hostname,
618                                             -1);
619                 } else {
620                         gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter,
621                                             MODEL_COL_SERVER_NAME, _("mcen_ia_optionalsmtp_notdefined"),
622                                             -1);
623                 }
624                         
625                 /* Get next row: */
626                 valid = gtk_tree_model_iter_next (priv->model, &iter);
627         }
628 }
629
630 static gboolean
631 on_key_pressed (GtkWidget *self,
632                 GdkEventKey *event,
633                 gpointer user_data)
634 {
635         if (event->keyval == GDK_Escape) {
636                 /* Simulate a press on Cancel to close the dialog */
637                 on_button_cancel (NULL, self);
638         }
639         
640         return FALSE;
641 }