/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* * Copyright (C) 2005 Novell, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * 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 Place - Suite 330, * Boston, MA 02111-1307, USA. * * Author: Michael Zucchi */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include "evolution-mail-session.h" #include #include "evolution-mail-store.h" #include "e-corba-utils.h" #include #define PARENT_TYPE bonobo_object_get_type () static BonoboObjectClass *parent_class = NULL; #define _PRIVATE(o) (g_type_instance_get_private ((GTypeInstance *)o, evolution_mail_session_get_type())) struct _EvolutionMailSessionPrivate { EAccountList *accounts; GList *stores; /* FIXME: locking */ EDList listeners; guint account_added; guint account_changed; guint account_removed; }; static int is_storage(EAccount *ea) { const char *uri; int ok = FALSE; uri = e_account_get_string(ea, E_ACCOUNT_SOURCE_URL); if (uri && uri[0]) { CamelProvider *prov; CamelException ex = { 0 }; prov = camel_provider_get(uri, &ex); camel_exception_clear(&ex); ok = prov && (prov->flags & CAMEL_PROVIDER_IS_STORAGE); } return ok; } /* GObject methods */ static void impl_dispose (GObject *object) { struct _EvolutionMailSessionPrivate *p = _PRIVATE(object); if (p->stores) { /* FIXME: free stores */ } /* FIXME: Free accounts */ /* FIXME: free listners */ (* G_OBJECT_CLASS (parent_class)->dispose) (object); } static void impl_finalize (GObject *object) { g_warning("EvolutionMailStore is finalised!\n"); (* G_OBJECT_CLASS (parent_class)->finalize) (object); } /* Evolution.Mail.Session */ static CORBA_boolean impl_getProperties(PortableServer_Servant _servant, const Evolution_Mail_PropertyNames* names, Evolution_Mail_Properties **props, CORBA_Environment * ev) { EvolutionMailSession *ems = (EvolutionMailSession *)bonobo_object_from_servant(_servant); ems = ems; return CORBA_TRUE; } static Evolution_Mail_StoreInfos * impl_getStores(PortableServer_Servant _servant, const CORBA_char * pattern, const Evolution_Mail_StoreListener listener, CORBA_Environment * ev) { EvolutionMailSession *ems = (EvolutionMailSession *)bonobo_object_from_servant(_servant); struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); Evolution_Mail_StoreInfos *seq; int i, len; GList *l; seq = Evolution_Mail_StoreInfos__alloc(); /* FIXME: pattern? */ len = g_list_length(p->stores); seq->_length = len; seq->_maximum = len; seq->_buffer = Evolution_Mail_StoreInfos_allocbuf(seq->_length); CORBA_sequence_set_release(seq, TRUE); l = p->stores; for (i=0;l && idata; evolution_mail_store_addlistener(store, listener); e_mail_storeinfo_set_store(&seq->_buffer[i], store); l = g_list_next(l); } return seq; } static void impl_addListener(PortableServer_Servant _servant, const Evolution_Mail_SessionListener listener, CORBA_Environment * ev) { EvolutionMailSession *ems = (EvolutionMailSession *)bonobo_object_from_servant(_servant); struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); e_mail_listener_add(&p->listeners, listener); } static void impl_removeListener(PortableServer_Servant _servant, const Evolution_Mail_SessionListener listener, CORBA_Environment * ev) { EvolutionMailSession *ems = (EvolutionMailSession *)bonobo_object_from_servant(_servant); struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); d(printf("Removing listener from session\n")); if (!e_mail_listener_remove(&p->listeners, listener)) { d(printf("no more listeners, could shut down session?\n")); } } /* Initialization */ static void evolution_mail_session_class_init (EvolutionMailSessionClass *klass) { POA_Evolution_Mail_Session__epv *epv = &klass->epv; GObjectClass *object_class = G_OBJECT_CLASS (klass); parent_class = g_type_class_peek_parent (klass); epv->getProperties = impl_getProperties; epv->getStores = impl_getStores; epv->addListener = impl_addListener; epv->removeListener = impl_removeListener; object_class->dispose = impl_dispose; object_class->finalize = impl_finalize; g_type_class_add_private(klass, sizeof(struct _EvolutionMailSessionPrivate)); } static void ems_set_changes(Evolution_Mail_SessionChange *change, Evolution_Mail_ChangeType how, EvolutionMailStore *store) { change->type = how; change->stores._length = 1; change->stores._maximum = 1; change->stores._buffer = Evolution_Mail_StoreInfos_allocbuf(change->stores._maximum); CORBA_sequence_set_release(&change->stores, TRUE); e_mail_storeinfo_set_store(&change->stores._buffer[0], store); } static void ems_listener_session_event(EvolutionMailSession *ems, Evolution_Mail_ChangeType how, EvolutionMailStore *store) { Evolution_Mail_SessionChanges *changes; /* NB: we only ever create 1 changetype at the moment */ changes = Evolution_Mail_SessionChanges__alloc(); changes->_maximum = 1; changes->_length = 1; changes->_buffer = Evolution_Mail_SessionChanges_allocbuf(1); CORBA_sequence_set_release(changes, TRUE); ems_set_changes(&changes->_buffer[0], how, store); evolution_mail_session_changed(ems, changes); CORBA_free(changes); } static void ems_account_added(EAccountList *eal, EAccount *ea, EvolutionMailSession *ems) { struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); if (ea->enabled && is_storage(ea)) { EvolutionMailStore *store; d(printf("Account added %s\n", ea->uid)); store = evolution_mail_store_new(ems, ea); p->stores = g_list_append(p->stores, store); ems_listener_session_event(ems, Evolution_Mail_ADDED, store); } } static void ems_account_changed(EAccountList *eal, EAccount *ea, EvolutionMailSession *ems) { struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); EvolutionMailStore *store = NULL; GList *l; for (l = p->stores;l;l=l->next) { if (((EvolutionMailStore *)l->data)->account == ea) { store = l->data; break; } } if (store) { /* account has been disabled? */ if (!ea->enabled) { d(printf("Account changed, now disabled %s\n", ea->uid)); p->stores = g_list_remove(p->stores, store); ems_listener_session_event(ems, Evolution_Mail_REMOVED, store); g_object_unref(store); } else { d(printf("Account changed, dont know how %s\n", ea->uid)); ems_listener_session_event(ems, Evolution_Mail_CHANGED, store); } } else if (ea->enabled && is_storage(ea)) { d(printf("Account changed, now added %s\n", ea->uid)); store = evolution_mail_store_new(ems, ea); p->stores = g_list_append(p->stores, store); ems_listener_session_event(ems, Evolution_Mail_ADDED, store); } } static void ems_account_removed(EAccountList *eal, EAccount *ea, EvolutionMailSession *ems) { struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); GList *l; /* for accounts we dont have, we dont care */ for (l = p->stores;l;l=l->next) { EvolutionMailStore *store = l->data; if (store->account == ea) { d(printf("Account removed %s\n", ea->uid)); p->stores = g_list_remove(p->stores, store); ems_listener_session_event(ems, Evolution_Mail_REMOVED, store); g_object_unref(store); break; } } } static void evolution_mail_session_init (EvolutionMailSession *ems, EvolutionMailSessionClass *klass) { GConfClient *gconf = gconf_client_get_default(); struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); EIterator *iter; d(printf("EvolutionMailSession.init\n")); e_dlist_init(&p->listeners); /* local store first */ p->stores = g_list_append(p->stores, evolution_mail_store_new(ems, NULL)); p->accounts = e_account_list_new(gconf); iter = e_list_get_iterator((EList *)p->accounts); while (e_iterator_is_valid (iter)) { EAccount *ea; if ((ea = (EAccount *)e_iterator_get(iter)) && ea->enabled && is_storage(ea)) { p->stores = g_list_append(p->stores, evolution_mail_store_new(ems, ea)); } e_iterator_next(iter); } g_object_unref(iter); g_object_unref(gconf); p->account_added = g_signal_connect(p->accounts, "account_added", G_CALLBACK(ems_account_added), ems); p->account_changed = g_signal_connect(p->accounts, "account_changed", G_CALLBACK(ems_account_changed), ems); p->account_removed = g_signal_connect(p->accounts, "account_removed", G_CALLBACK(ems_account_removed), ems); ems->session = mail_component_peek_session(NULL); bonobo_object_set_immortal((BonoboObject *)ems, TRUE); } void evolution_mail_session_changed(EvolutionMailSession *ems, Evolution_Mail_SessionChanges *changes) { struct _EvolutionMailSessionPrivate *p = _PRIVATE(ems); if (!e_mail_listener_emit(&p->listeners, (EMailListenerChanged)Evolution_Mail_SessionListener_changed, bonobo_object_corba_objref((BonoboObject *)ems), changes)) { w(printf("No more listeners for store, could dispose session object now?\n")); } } BONOBO_TYPE_FUNC_FULL (EvolutionMailSession, Evolution_Mail_Session, PARENT_TYPE, evolution_mail_session)