/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Authors: Jeffrey Stedfast <fejj@helixcode.com>
*
* Copyright 2001 Helix Code, Inc. (www.helixcode.com)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA.
*
*/
#include "mail-account-editor.h"
#include "mail-session.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <camel/camel-url.h>
static void mail_account_editor_class_init (MailAccountEditorClass *class);
static void mail_account_editor_init (MailAccountEditor *editor);
static void mail_account_editor_finalise (GtkObject *obj);
static GnomeDialogClass *parent_class;
GtkType
mail_account_editor_get_type ()
{
static GtkType type = 0;
if (!type) {
GtkTypeInfo type_info = {
"MailAccountEditor",
sizeof (MailAccountEditor),
sizeof (MailAccountEditorClass),
(GtkClassInitFunc) mail_account_editor_class_init,
(GtkObjectInitFunc) mail_account_editor_init,
(GtkArgSetFunc) NULL,
(GtkArgGetFunc) NULL
};
type = gtk_type_unique (gnome_dialog_get_type (), &type_info);
}
return type;
}
static void
mail_account_editor_class_init (MailAccountEditorClass *class)
{
GtkObjectClass *object_class;
object_class = (GtkObjectClass *) class;
parent_class = gtk_type_class (gnome_dialog_get_type ());
object_class->finalize = mail_account_editor_finalise;
/* override methods */
}
static void
mail_account_editor_init (MailAccountEditor *o)
{
;
}
static void
mail_account_editor_finalise (GtkObject *obj)
{
MailAccountEditor *editor = (MailAccountEditor *) obj;
gtk_object_unref (GTK_OBJECT (editor->gui));
((GtkObjectClass *)(parent_class))->finalize (obj);
}
/* callbacks */
static void
entry_changed (GtkEntry *entry, gpointer data)
{
MailAccountEditor *editor = data;
char *account_name, *name, *address;
gboolean sensitive;
account_name = gtk_entry_get_text (editor->account_name);
name = gtk_entry_get_text (editor->name);
address = gtk_entry_get_text (editor->email);
sensitive = account_name && *account_name && name && *name && address && *address;
gnome_dialog_set_sensitive (GNOME_DIALOG (editor), 0, sensitive);
gnome_dialog_set_sensitive (GNOME_DIALOG (editor), 1, sensitive);
}
static gboolean
apply_changes (MailAccountEditor *editor)
{
MailConfigAccount *account;
char *host, *pport, *auth;
CamelURL *url;
int port;
account = (MailConfigAccount *) editor->account;
/* account name */
g_free (account->name);
account->name = g_strdup (gtk_entry_get_text (editor->account_name));
/* identity info */
g_free (account->id->name);
account->id->name = g_strdup (gtk_entry_get_text (editor->name));
g_free (account->id->address);
account->id->address = g_strdup (gtk_entry_get_text (editor->email));
g_free (account->id->reply_to);
account->id->reply_to = g_strdup (gtk_entry_get_text (editor->reply_to));
g_free (account->id->organization);
account->id->organization = g_strdup (gtk_entry_get_text (editor->organization));
g_free (account->id->signature);
account->id->signature = gnome_file_entry_get_full_path (editor->signature, TRUE);
/* source */
url = camel_url_new (account->source->url, NULL);
g_free (url->user);
url->user = g_strdup (gtk_entry_get_text (editor->source_user));
g_free (url->passwd);
url->passwd = g_strdup (gtk_entry_get_text (editor->source_passwd));
g_free (url->authmech);
auth = gtk_object_get_data (GTK_OBJECT (editor), "source_authmech");
url->authmech = auth && *auth ? g_strdup (auth) : NULL;
g_free (url->host);
host = g_strdup (gtk_entry_get_text (editor->source_host));
if (host && (pport = strchr (host, ':'))) {
*pport = '\0';
port = atoi (pport + 1);
} else {
port = 0;
}
url->host = host;
url->port = port;
g_free (url->path);
url->path = g_strdup (gtk_entry_get_text (editor->source_path));
account->source->save_passwd = GTK_TOGGLE_BUTTON (editor->save_passwd)->active;
account->source->keep_on_server = GTK_TOGGLE_BUTTON (editor->keep_on_server)->active;
/* check to make sure the source works */
if (!mail_config_check_service (url, CAMEL_PROVIDER_STORE, NULL)) {
camel_url_free (url);
account_destroy (account);
return FALSE;
}
if (account->source->save_passwd) {
char *string;
string = camel_url_to_string (url, FALSE);
mail_session_set_password (string, url->passwd);
mail_session_remember_password (string);
g_free (string);
}
/* now that we know this url works, set it */
g_free (account->source->url);
account->source->url = camel_url_to_string (url, FALSE);
camel_url_free (url);
/* transport */
url = g_new0 (CamelURL, 1);
url->protocol = g_strdup (editor->transport->protocol);
auth = gtk_object_get_data (GTK_OBJECT (editor), "transport_authmech");
url->authmech = auth && *auth ? g_strdup (auth) : NULL;
host = g_strdup (gtk_entry_get_text (editor->transport_host));
if (host && (pport = strchr (host, ':'))) {
*pport = '\0';
port = atoi (pport + 1);
} else {
port = 0;
}
url->host = host;
url->port = port;
/* check to make sure the transport works */
if (!mail_config_check_service (url, CAMEL_PROVIDER_TRANSPORT, NULL)) {
camel_url_free (url);
account_destroy (account);
return FALSE;
}
/* now that we know this url works, set it */
g_free (account->transport->url);
account->transport->url = camel_url_to_string (url, FALSE);
camel_url_free (url);
mail_config_write ();
return TRUE;
}
static void
apply_clicked (GtkWidget *widget, gpointer data)
{
MailAccountEditor *editor = data;
apply_changes (editor);
}
static void
ok_clicked (GtkWidget *widget, gpointer data)
{
MailAccountEditor *editor = data;
if (apply_changes (editor))
gtk_widget_destroy (GTK_WIDGET (editor));
}
static void
cancel_clicked (GtkWidget *widget, gpointer data)
{
MailAccountEditor *editor = data;
gtk_widget_destroy (GTK_WIDGET (editor));
}
static void
source_auth_type_changed (GtkWidget *widget, gpointer user_data)
{
MailAccountEditor *editor = user_data;
CamelServiceAuthType *authtype;
gboolean sensitive;
authtype = gtk_object_get_data (GTK_OBJECT (widget), "authtype");
gtk_object_set_data (GTK_OBJECT (editor), "source_authmech", authtype->authproto);
if (authtype->need_password)
sensitive = TRUE;
else
sensitive = FALSE;
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_passwd), sensitive);
gtk_widget_set_sensitive (GTK_WIDGET (editor->save_passwd), sensitive);
}
static void
source_auth_init (MailAccountEditor *editor, CamelURL *url)
{
GtkWidget *menu, *item, *authmech = NULL;
CamelServiceAuthType *authtype;
GList *authtypes = NULL;
guint i = 0, history = 0;
menu = gtk_menu_new ();
gtk_option_menu_remove_menu (editor->source_auth);
/* If we can't connect, don't let them continue. */
if (!mail_config_check_service (url, CAMEL_PROVIDER_STORE, &authtypes)) {
gtk_option_menu_set_menu (editor->source_auth, menu);
return;
}
if (authtypes) {
GList *l;
l = authtypes;
while (l) {
authtype = l->data;
item = gtk_menu_item_new_with_label (authtype->name);
gtk_object_set_data (GTK_OBJECT (item), "authtype", authtype);
gtk_signal_connect (GTK_OBJECT (item), "activate",
GTK_SIGNAL_FUNC (source_auth_type_changed),
editor);
gtk_menu_append (GTK_MENU (menu), item);
gtk_widget_show (item);
if (url->authmech && !g_strcasecmp (authtype->authproto, url->authmech)) {
authmech = item;
history = i;
}
l = l->next;
i++;
}
if (authmech) {
gtk_signal_emit_by_name (GTK_OBJECT (authmech), "activate", editor);
gtk_option_menu_set_history (editor->source_auth, history);
}
}
gtk_option_menu_set_menu (editor->source_auth, menu);
}
static void
transport_auth_type_changed (GtkWidget *widget, gpointer user_data)
{
MailAccountEditor *editor = user_data;
CamelServiceAuthType *authtype;
authtype = gtk_object_get_data (GTK_OBJECT (widget), "authtype");
gtk_object_set_data (GTK_OBJECT (editor), "transport_authmech", authtype->authproto);
}
static void
transport_construct_authmenu (MailAccountEditor *editor, CamelURL *url)
{
GtkWidget *first = NULL, *preferred = NULL;
GtkWidget *menu, *item;
CamelServiceAuthType *authtype;
GList *authtypes = NULL;
guint i = 0, fhist = 0, phist = 0;
menu = gtk_menu_new ();
gtk_option_menu_remove_menu (editor->transport_auth);
if (!mail_config_check_service (url, CAMEL_PROVIDER_TRANSPORT, &authtypes)) {
gtk_option_menu_set_menu (editor->transport_auth, menu);
return;
}
menu = gtk_menu_new ();
if (authtypes) {
GList *l;
l = authtypes;
while (l) {
authtype = l->data;
item = gtk_menu_item_new_with_label (authtype->name);
gtk_object_set_data (GTK_OBJECT (item), "authtype", authtype);
gtk_signal_connect (GTK_OBJECT (item), "activate",
GTK_SIGNAL_FUNC (transport_auth_type_changed),
editor);
gtk_menu_append (GTK_MENU (menu), item);
gtk_widget_show (item);
if (!first) {
first = item;
fhist = i;
}
if (url->authmech && !g_strcasecmp (authtype->authproto, url->authmech)) {
preferred = item;
phist = i;
}
l = l->next;
i++;
}
}
gtk_option_menu_set_menu (editor->transport_auth, menu);
if (preferred) {
gtk_signal_emit_by_name (GTK_OBJECT (preferred), "activate", editor);
gtk_option_menu_set_history (editor->transport_auth, phist);
} else if (first) {
gtk_signal_emit_by_name (GTK_OBJECT (first), "activate", editor);
gtk_option_menu_set_history (editor->transport_auth, phist);
}
}
static void
transport_type_changed (GtkWidget *widget, gpointer user_data)
{
MailAccountEditor *editor = user_data;
CamelProvider *provider;
GtkWidget *label;
provider = gtk_object_get_data (GTK_OBJECT (widget), "provider");
editor->transport = provider;
/* hostname */
label = glade_xml_get_widget (editor->gui, "lblTransportHost");
if (provider->url_flags & CAMEL_URL_ALLOW_HOST) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->transport_host), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->transport_host), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
/* auth */
label = glade_xml_get_widget (editor->gui, "lblTransportAuth");
if (provider->url_flags & CAMEL_URL_ALLOW_AUTH) {
CamelURL *url;
gtk_widget_set_sensitive (GTK_WIDGET (editor->transport_auth), TRUE);
gtk_widget_set_sensitive (label, TRUE);
/* regen the auth list */
url = g_new0 (CamelURL, 1);
url->protocol = g_strdup (provider->protocol);
url->host = g_strdup (gtk_entry_get_text (editor->transport_host));
transport_construct_authmenu (editor, url);
camel_url_free (url);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->transport_auth), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
}
static void
transport_type_init (MailAccountEditor *editor, CamelURL *url)
{
GtkWidget *menu, *xport = NULL;
GList *providers, *l;
guint i = 0, history = 0;
menu = gtk_menu_new ();
gtk_option_menu_remove_menu (editor->transport_auth);
providers = camel_session_list_providers (session, TRUE);
l = providers;
while (l) {
CamelProvider *provider = l->data;
if (strcmp (provider->domain, "mail")) {
l = l->next;
continue;
}
if (provider->object_types[CAMEL_PROVIDER_TRANSPORT]) {
GtkWidget *item;
item = gtk_menu_item_new_with_label (provider->name);
gtk_object_set_data (GTK_OBJECT (item), "provider", provider);
gtk_signal_connect (GTK_OBJECT (item), "activate",
GTK_SIGNAL_FUNC (transport_type_changed),
editor);
gtk_menu_append (GTK_MENU (menu), item);
gtk_widget_show (item);
if (!xport && !g_strcasecmp (provider->protocol, url->protocol)) {
xport = item;
history = i;
}
i++;
}
l = l->next;
}
gtk_option_menu_set_menu (editor->transport_type, menu);
if (xport) {
gtk_signal_emit_by_name (GTK_OBJECT (xport), "activate", editor);
gtk_option_menu_set_history (editor->transport_type, history);
}
}
static void
source_check (MailAccountEditor *editor, CamelURL *url)
{
GList *providers, *l;
providers = camel_session_list_providers (session, TRUE);
l = providers;
while (l) {
CamelProvider *provider = l->data;
if (strcmp (provider->domain, "mail")) {
l = l->next;
continue;
}
if (provider->object_types[CAMEL_PROVIDER_STORE] && provider->flags & CAMEL_PROVIDER_IS_SOURCE) {
if (!g_strcasecmp (provider->protocol, url->protocol)) {
GtkWidget *label;
/* keep-on-server */
if (!(provider->flags & CAMEL_PROVIDER_IS_STORAGE))
gtk_widget_set_sensitive (GTK_WIDGET (editor->keep_on_server), TRUE);
else
gtk_widget_set_sensitive (GTK_WIDGET (editor->keep_on_server), FALSE);
/* host */
label = glade_xml_get_widget (editor->gui, "lblSourceHost");
if (provider->url_flags & CAMEL_URL_ALLOW_HOST) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_host), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_host), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
/* user */
label = glade_xml_get_widget (editor->gui, "lblSourceUser");
if (provider->url_flags & CAMEL_URL_ALLOW_USER) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_user), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_user), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
/* path */
label = glade_xml_get_widget (editor->gui, "lblSourcePath");
if (provider->url_flags & CAMEL_URL_ALLOW_PATH) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_path), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_path), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
/* auth */
label = glade_xml_get_widget (editor->gui, "lblSourceAuth");
if (provider->url_flags & CAMEL_URL_ALLOW_AUTH) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_auth), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_auth), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
/* passwd */
label = glade_xml_get_widget (editor->gui, "lblSourcePasswd");
if (provider->url_flags & CAMEL_URL_ALLOW_PASSWORD) {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_passwd), TRUE);
gtk_widget_set_sensitive (GTK_WIDGET (editor->save_passwd), TRUE);
gtk_widget_set_sensitive (label, TRUE);
} else {
gtk_widget_set_sensitive (GTK_WIDGET (editor->source_passwd), FALSE);
gtk_widget_set_sensitive (GTK_WIDGET (editor->save_passwd), FALSE);
gtk_widget_set_sensitive (label, FALSE);
}
break;
}
}
l = l->next;
}
}
static void
construct (MailAccountEditor *editor, const MailConfigAccount *account)
{
GladeXML *gui;
GtkWidget *notebook, *entry;
CamelURL *url;
gui = glade_xml_new (EVOLUTION_GLADEDIR "/mail-config.glade", "mail-account-editor");
editor->gui = gui;
/* get our toplevel widget */
notebook = glade_xml_get_widget (gui, "notebook");
/* reparent */
gtk_widget_reparent (notebook, GNOME_DIALOG (editor)->vbox);
/* give our dialog an OK button and title */
gtk_window_set_title (GTK_WINDOW (editor), _("Evolution Account Editor"));
gtk_window_set_policy (GTK_WINDOW (editor), FALSE, TRUE, TRUE);
gtk_window_set_modal (GTK_WINDOW (editor), TRUE);
gnome_dialog_append_buttons (GNOME_DIALOG (editor),
GNOME_STOCK_BUTTON_OK,
GNOME_STOCK_BUTTON_APPLY,
GNOME_STOCK_BUTTON_CANCEL,
NULL);
gnome_dialog_button_connect (GNOME_DIALOG (editor), 0 /* OK */,
GTK_SIGNAL_FUNC (ok_clicked),
editor);
gnome_dialog_button_connect (GNOME_DIALOG (editor), 1 /* APPLY */,
GTK_SIGNAL_FUNC (apply_clicked),
editor);
gnome_dialog_button_connect (GNOME_DIALOG (editor), 2 /* CANCEL */,
GTK_SIGNAL_FUNC (cancel_clicked),
editor);
/* General */
editor->account_name = GTK_ENTRY (glade_xml_get_widget (gui, "txtAccountName"));
gtk_entry_set_text (editor->account_name, account->name);
gtk_signal_connect (GTK_OBJECT (editor->account_name), "changed", entry_changed, editor);
editor->name = GTK_ENTRY (glade_xml_get_widget (gui, "txtName"));
gtk_entry_set_text (editor->name, account->id->name);
gtk_signal_connect (GTK_OBJECT (editor->name), "changed", entry_changed, editor);
editor->email = GTK_ENTRY (glade_xml_get_widget (gui, "txtAddress"));
gtk_entry_set_text (editor->email, account->id->address);
gtk_signal_connect (GTK_OBJECT (editor->email), "changed", entry_changed, editor);
editor->reply_to = GTK_ENTRY (glade_xml_get_widget (gui, "txtReplyTo"));
gtk_entry_set_text (editor->reply_to, account->id->reply_to ? account->id->reply_to : "");
editor->organization = GTK_ENTRY (glade_xml_get_widget (gui, "txtOrganization"));
gtk_entry_set_text (editor->organization, account->id->organization);
editor->signature = GNOME_FILE_ENTRY (glade_xml_get_widget (gui, "fileSignature"));
entry = gnome_file_entry_gtk_entry (editor->signature);
gtk_entry_set_text (GTK_ENTRY (entry), account->id->signature);
/* Servers */
url = camel_url_new (account->source->url, NULL);
editor->source_type = GTK_ENTRY (glade_xml_get_widget (gui, "txtSourceType"));
gtk_entry_set_text (editor->source_type, url->protocol);
editor->source_host = GTK_ENTRY (glade_xml_get_widget (gui, "txtSourceHost"));
gtk_entry_set_text (editor->source_host, url->host ? url->host : "");
if (url->port) {
char port[10];
g_snprintf (port, 9, ":%d", url->port);
gtk_entry_append_text (editor->source_host, port);
}
editor->source_user = GTK_ENTRY (glade_xml_get_widget (gui, "txtSourceUser"));
gtk_entry_set_text (editor->source_user, url->user ? url->user : "");
editor->source_passwd = GTK_ENTRY (glade_xml_get_widget (gui, "txtSourcePasswd"));
gtk_entry_set_text (editor->source_passwd, url->passwd ? url->passwd : "");
editor->source_path = GTK_ENTRY (glade_xml_get_widget (gui, "txtSourcePath"));
gtk_entry_set_text (editor->source_path, url->path);
editor->save_passwd = GTK_CHECK_BUTTON (glade_xml_get_widget (gui, "chkSavePasswd"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (editor->save_passwd), account->source->save_passwd);
editor->source_auth = GTK_OPTION_MENU (glade_xml_get_widget (gui, "omenuSourceAuth"));
editor->source_ssl = GTK_CHECK_BUTTON (glade_xml_get_widget (gui, "chkSourceSSL"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (editor->source_ssl), account->source->use_ssl);
editor->keep_on_server = GTK_CHECK_BUTTON (glade_xml_get_widget (gui, "chkKeepMailOnServer"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (editor->keep_on_server), account->source->keep_on_server);
source_check (editor, url);
source_auth_init (editor, url);
camel_url_free (url);
/* Transport */
url = camel_url_new (account->transport->url, NULL);
editor->transport_type = GTK_OPTION_MENU (glade_xml_get_widget (gui, "omenuTransportType"));
editor->transport_host = GTK_ENTRY (glade_xml_get_widget (gui, "txtTransportHost"));
gtk_entry_set_text (editor->transport_host, url->host ? url->host : "");
if (url->port) {
char port[10];
g_snprintf (port, 9, ":%d", url->port);
gtk_entry_append_text (editor->transport_host, port);
}
editor->transport_auth = GTK_OPTION_MENU (glade_xml_get_widget (gui, "omenuTransportAuth"));
editor->transport_ssl = GTK_CHECK_BUTTON (glade_xml_get_widget (gui, "chkTransportSSL"));
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (editor->transport_ssl), account->transport->use_ssl);
transport_type_init (editor, url);
camel_url_free (url);
editor->account = account;
}
MailAccountEditor *
mail_account_editor_new (const MailConfigAccount *account)
{
MailAccountEditor *new;
new = (MailAccountEditor *) gtk_type_new (mail_account_editor_get_type ());
construct (new, account);
return new;
}