/*
addressbooklink . cpp - Manages operations involving the KDE Address Book
Copyright ( c ) 2005 Will Stephenson < lists @ stevello . free - online . co . uk >
Kopete ( c ) 2002 - 2004 by the Kopete developers < kopete - devel @ kde . org >
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* *
* This library is free software ; you can redistribute it and / or *
* modify it under the terms of the GNU Lesser General Public *
* License as published by the Free Software Foundation ; either *
* version 2 of the License , or ( at your option ) any later version . *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*/
# include <tqstring.h>
# include <tqtimer.h>
# include <kabc/addressbook.h>
# include <kabc/addressee.h>
# include <kabc/resource.h>
# include <kabc/stdaddressbook.h>
// UI related includes used for importing from KABC
# include <kdialogbase.h>
# include <klocale.h>
# include <kmessagebox.h>
# include "accountselector.h"
# include "kopeteuiglobal.h"
# include <kstaticdeleter.h>
# include "kopeteaccount.h"
# include "kopeteaccountmanager.h"
# include "kopetecontact.h"
# include "kopetemetacontact.h"
# include "kopetepluginmanager.h"
# include "kopeteprotocol.h"
# include "kabcpersistence.h"
namespace Kopete
{
/**
* utility function to merge two TQStrings containing individual elements separated by 0xE000
*/
static TQString unionContents ( TQString arg1 , TQString arg2 )
{
TQChar separator ( 0xE000 ) ;
TQStringList outList = TQStringList : : split ( separator , arg1 ) ;
TQStringList arg2List = TQStringList : : split ( separator , arg2 ) ;
for ( TQStringList : : iterator it = arg2List . begin ( ) ; it ! = arg2List . end ( ) ; + + it )
if ( ! outList . contains ( * it ) )
outList . append ( * it ) ;
TQString out = outList . join ( separator ) ;
return out ;
}
KABCPersistence : : KABCPersistence ( TQObject * parent , const char * name ) : TQObject ( parent , name )
{
s_pendingResources . setAutoDelete ( false ) ;
}
KABCPersistence : : ~ KABCPersistence ( )
{
}
KABCPersistence * KABCPersistence : : s_self = 0L ;
bool KABCPersistence : : s_addrBookWritePending = false ;
TQPtrList < KABC : : Resource > KABCPersistence : : s_pendingResources ;
KABC : : AddressBook * KABCPersistence : : s_addressBook = 0 ;
KABCPersistence * KABCPersistence : : self ( )
{
static KStaticDeleter < KABCPersistence > deleter ;
if ( ! s_self )
deleter . setObject ( s_self , new KABCPersistence ( ) ) ;
return s_self ;
}
KABC : : AddressBook * KABCPersistence : : addressBook ( )
{
if ( s_addressBook = = 0L )
{
s_addressBook = KABC : : StdAddressBook : : self ( ) ;
KABC : : StdAddressBook : : setAutomaticSave ( false ) ;
}
return s_addressBook ;
}
void KABCPersistence : : write ( MetaContact * mc )
{
// Save any changes in each contact's addressBookFields to KABC
KABC : : AddressBook * ab = addressBook ( ) ;
kdDebug ( 14010 ) < < k_funcinfo < < " looking up Addressee for " < < mc - > displayName ( ) < < " ... " < < endl ;
// Look up the address book entry
KABC : : Addressee theAddressee = ab - > findByUid ( mc - > metaContactId ( ) ) ;
// Check that if addressee is not deleted or if the link is spurious
// (inherited from Kopete < 0.8, where all metacontacts had random ids)
if ( theAddressee . isEmpty ( ) )
{
// not found in currently enabled addressbooks - may be in a disabled resource...
return ;
}
else
{
// collate the instant messaging data to be inserted into the address book
TQMap < TQString , TQStringList > addressMap ;
TQPtrList < Contact > contacts = mc - > contacts ( ) ;
TQPtrListIterator < Contact > cIt ( contacts ) ;
while ( Contact * c = cIt . current ( ) )
{
TQStringList addresses = addressMap [ c - > protocol ( ) - > addressBookIndexField ( ) ] ;
addresses . append ( c - > contactId ( ) ) ;
addressMap . insert ( c - > protocol ( ) - > addressBookIndexField ( ) , addresses ) ;
+ + cIt ;
}
// insert a custom field for each protocol
TQMap < TQString , TQStringList > : : ConstIterator it = addressMap . begin ( ) ;
for ( ; it ! = addressMap . end ( ) ; + + it )
{
// read existing data for this key
TQString currentCustomForProtocol = theAddressee . custom ( it . key ( ) , TQString : : fromLatin1 ( " All " ) ) ;
// merge without duplicating
TQString toWrite = unionContents ( currentCustomForProtocol , it . data ( ) . join ( TQChar ( 0xE000 ) ) ) ;
// Note if nothing ends up in the KABC data, this is because insertCustom does nothing if any param is empty.
kdDebug ( 14010 ) < < k_funcinfo < < " Writing: " < < it . key ( ) < < " , " < < " All " < < " , " < < toWrite < < endl ;
theAddressee . insertCustom ( it . key ( ) , TQString : : fromLatin1 ( " All " ) , toWrite ) ;
TQString check = theAddressee . custom ( it . key ( ) , TQString : : fromLatin1 ( " All " ) ) ;
}
ab - > insertAddressee ( theAddressee ) ;
//kdDebug( 14010 ) << k_funcinfo << "dumping addressbook before write " << endl;
//dumpAB();
writeAddressBook ( theAddressee . resource ( ) ) ;
//theAddressee.dump();
}
/* // Wipe out the existing addressBook entries
d - > addressBook . clear ( ) ;
// This causes each Kopete::Protocol subclass to serialise its contacts' data into the metacontact's plugin data and address book data
emit aboutToSave ( this ) ;
kdDebug ( 14010 ) < < k_funcinfo < < " ...FOUND ONE! " < < endl ;
// Store address book fields
TQMap < TQString , TQMap < TQString , TQString > > : : ConstIterator appIt = d - > addressBook . begin ( ) ;
for ( ; appIt ! = d - > addressBook . end ( ) ; + + appIt )
{
TQMap < TQString , TQString > : : ConstIterator addrIt = appIt . data ( ) . begin ( ) ;
for ( ; addrIt ! = appIt . data ( ) . end ( ) ; + + addrIt )
{
// read existing data for this key
TQString currentCustom = theAddressee . custom ( appIt . key ( ) , addrIt . key ( ) ) ;
// merge without duplicating
TQString toWrite = unionContents ( currentCustom , addrIt . data ( ) ) ;
// write the result
// Note if nothing ends up in the KABC data, this is because insertCustom does nothing if any param is empty.
kdDebug ( 14010 ) < < k_funcinfo < < " Writing: " < < appIt . key ( ) < < " , " < < addrIt . key ( ) < < " , " < < toWrite < < endl ;
theAddressee . insertCustom ( appIt . key ( ) , addrIt . key ( ) , toWrite ) ;
}
}
ab - > insertAddressee ( theAddressee ) ;
writeAddressBook ( ) ;
} */
}
void KABCPersistence : : writeAddressBook ( const KABC : : Resource * res )
{
if ( ! s_pendingResources . containsRef ( res ) )
s_pendingResources . append ( res ) ;
if ( ! s_addrBookWritePending )
{
s_addrBookWritePending = true ;
TQTimer : : singleShot ( 2000 , this , TQT_SLOT ( slotWriteAddressBook ( ) ) ) ;
}
}
void KABCPersistence : : slotWriteAddressBook ( )
{
//kdDebug( 14010 ) << k_funcinfo << endl;
KABC : : AddressBook * ab = addressBook ( ) ;
TQPtrListIterator < KABC : : Resource > it ( s_pendingResources ) ;
for ( ; it . current ( ) ; + + it )
{
//kdDebug( 14010 ) << "Writing resource " << it.current()->resourceName() << endl;
KABC : : Ticket * ticket = ab - > requestSaveTicket ( it . current ( ) ) ;
if ( ! ticket )
kdWarning ( 14010 ) < < " WARNING: Resource is locked by other application! " < < endl ;
else
{
if ( ! ab - > save ( ticket ) )
{
kdWarning ( 14010 ) < < " ERROR: Saving failed! " < < endl ;
ab - > releaseSaveTicket ( ticket ) ;
}
}
//kdDebug( 14010 ) << "Finished writing KABC" << endl;
}
s_pendingResources . clear ( ) ;
s_addrBookWritePending = false ;
}
void KABCPersistence : : removeKABC ( MetaContact * )
{
/* // remove any data this KMC has written to the KDE address book
// Save any changes in each contact's addressBookFields to KABC
KABC : : AddressBook * ab = addressBook ( ) ;
// Wipe out the existing addressBook entries
d - > addressBook . clear ( ) ;
// This causes each Kopete::Protocol subclass to serialise its contacts' data into the metacontact's plugin data and address book data
emit aboutToSave ( this ) ;
// If the metacontact is linked to a kabc entry
if ( ! d - > metaContactId . isEmpty ( ) )
{
//kdDebug( 14010 ) << k_funcinfo << "looking up Addressee for " << displayName() << "..." << endl;
// Look up the address book entry
KABC : : Addressee theAddressee = ab - > findByUid ( metaContactId ( ) ) ;
if ( theAddressee . isEmpty ( ) )
{
// remove the link
//kdDebug( 14010 ) << k_funcinfo << "...not found." << endl;
d - > metaContactId = TQString ( ) ;
}
else
{
//kdDebug( 14010 ) << k_funcinfo << "...FOUND ONE!" << endl;
// Remove address book fields
TQMap < TQString , TQMap < TQString , TQString > > : : ConstIterator appIt = d - > addressBook . begin ( ) ;
for ( ; appIt ! = d - > addressBook . end ( ) ; + + appIt )
{
TQMap < TQString , TQString > : : ConstIterator addrIt = appIt . data ( ) . begin ( ) ;
for ( ; addrIt ! = appIt . data ( ) . end ( ) ; + + addrIt )
{
// FIXME: This assumes Kopete is the only app writing these fields
kdDebug ( 14010 ) < < k_funcinfo < < " Removing: " < < appIt . key ( ) < < " , " < < addrIt . key ( ) < < endl ;
theAddressee . removeCustom ( appIt . key ( ) , addrIt . key ( ) ) ;
}
}
ab - > insertAddressee ( theAddressee ) ;
writeAddressBook ( ) ;
}
}
// kdDebug(14010) << k_funcinfo << kdBacktrace() <<endl;*/
}
bool KABCPersistence : : syncWithKABC ( MetaContact * mc )
{
kdDebug ( 14010 ) < < k_funcinfo < < endl ;
bool contactAdded = false ;
// check whether the dontShowAgain was checked
KABC : : AddressBook * ab = addressBook ( ) ;
KABC : : Addressee addr = ab - > findByUid ( mc - > metaContactId ( ) ) ;
if ( ! addr . isEmpty ( ) ) // if we are associated with KABC
{
// load the set of addresses from KABC
TQStringList customs = addr . customs ( ) ;
TQStringList : : ConstIterator it ;
for ( it = customs . begin ( ) ; it ! = customs . end ( ) ; + + it )
{
TQString app , name , value ;
splitField ( * it , app , name , value ) ;
kdDebug ( 14010 ) < < " app= " < < app < < " name= " < < name < < " value= " < < value < < endl ;
if ( app . startsWith ( TQString : : fromLatin1 ( " messaging/ " ) ) )
{
if ( name = = TQString : : fromLatin1 ( " All " ) )
{
kdDebug ( 14010 ) < < " syncing \" " < < app < < " : " < < name < < " with contactlist " < < endl ;
// Get the protocol name from the custom field
// by chopping the 'messaging/' prefix from the custom field app name
TQString protocolName = app . right ( app . length ( ) - 10 ) ;
// munge Jabber hack
if ( protocolName = = TQString : : fromLatin1 ( " xmpp " ) )
protocolName = TQString : : fromLatin1 ( " jabber " ) ;
// Check Kopete supports it
Protocol * proto = dynamic_cast < Protocol * > ( PluginManager : : self ( ) - > loadPlugin ( TQString : : fromLatin1 ( " kopete_ " ) + protocolName ) ) ;
if ( ! proto )
{
KMessageBox : : queuedMessageBox ( Kopete : : UI : : Global : : mainWidget ( ) , KMessageBox : : Sorry ,
i18n ( " <qt> \" %1 \" is not supported by Kopete.</qt> " ) . arg ( protocolName ) ,
i18n ( " Could Not Sync with KDE Address Book " ) ) ;
continue ;
}
// See if we need to add each contact in this protocol
TQStringList addresses = TQStringList : : split ( TQChar ( 0xE000 ) , value ) ;
TQStringList : : iterator end = addresses . end ( ) ;
for ( TQStringList : : iterator it = addresses . begin ( ) ; it ! = end ; + + it )
{
// check whether each one is present in Kopete
// Is it in the contact list?
// First discard anything after an 0xE120, this is used by IRC to separate nick and server group name, but
// IRC doesn't support this properly yet, so the user will have to select an appropriate account manually
int separatorPos = ( * it ) . find ( TQChar ( 0xE120 ) ) ;
if ( separatorPos ! = - 1 )
* it = ( * it ) . left ( separatorPos ) ;
TQDict < Kopete : : Account > accounts = Kopete : : AccountManager : : self ( ) - > accounts ( proto ) ;
TQDictIterator < Kopete : : Account > acs ( accounts ) ;
Kopete : : MetaContact * otherMc = 0 ;
for ( acs . toFirst ( ) ; acs . current ( ) ; + + acs )
{
Kopete : : Contact * c = acs . current ( ) - > contacts ( ) [ * it ] ;
if ( c )
{
otherMc = c - > metaContact ( ) ;
break ;
}
}
if ( otherMc ) // Is it in another metacontact?
{
// Is it already in this metacontact? If so, we needn't do anything
if ( otherMc = = mc )
{
kdDebug ( 14010 ) < < * it < < " already a child of this metacontact. " < < endl ;
continue ;
}
kdDebug ( 14010 ) < < * it < < " already exists in OTHER metacontact, move here? " < < endl ;
// find the Kopete::Contact and attempt to move it to this metacontact.
otherMc - > findContact ( proto - > pluginId ( ) , TQString ( ) , * it ) - > setMetaContact ( mc ) ;
}
else
{
// if not, prompt to add it
kdDebug ( 14010 ) < < proto - > pluginId ( ) < < " :// " < < * it < < " was not found in the contact list. Prompting to add... " < < endl ;
if ( KMessageBox : : Yes = = KMessageBox : : questionYesNo ( Kopete : : UI : : Global : : mainWidget ( ) ,
i18n ( " <qt>An address was added to this contact by another application.<br>Would you like to use it in Kopete?<br><b>Protocol:</b> %1<br><b>Address:</b> %2</qt> " ) . arg ( proto - > displayName ( ) ) . arg ( * it ) , i18n ( " Import Address From Address Book " ) , i18n ( " Use " ) , i18n ( " Do Not Use " ) , TQString : : fromLatin1 ( " ImportFromKABC " ) ) )
{
// Check the accounts for this protocol are all connected
// Most protocols do not allow you to add contacts while offline
// Would be better to have a virtual bool Kopete::Account::readyToAddContact()
bool allAccountsConnected = true ;
for ( acs . toFirst ( ) ; acs . current ( ) ; + + acs )
if ( ! acs . current ( ) - > isConnected ( ) )
{ allAccountsConnected = false ;
break ;
}
if ( ! allAccountsConnected )
{
KMessageBox : : queuedMessageBox ( Kopete : : UI : : Global : : mainWidget ( ) , KMessageBox : : Sorry ,
i18n ( " <qt>One or more of your accounts using %1 are offline. Most systems have to be connected to add contacts. Please connect these accounts and try again.</qt> " ) . arg ( protocolName ) ,
i18n ( " Not Connected " ) ) ;
continue ;
}
// we have got a contact to add, our accounts are connected, so add it.
// Do we need to choose an account
Kopete : : Account * chosen = 0 ;
if ( accounts . count ( ) > 1 )
{ // if we have >1 account in this protocol, prompt for the protocol.
KDialogBase * chooser = new KDialogBase ( 0 , " chooser " , true ,
i18n ( " Choose Account " ) , KDialogBase : : Ok | KDialogBase : : Cancel ,
KDialogBase : : Ok , false ) ;
AccountSelector * accSelector = new AccountSelector ( proto , chooser ,
" accSelector " ) ;
chooser - > setMainWidget ( accSelector ) ;
if ( chooser - > exec ( ) = = TQDialog : : Rejected )
continue ;
chosen = accSelector - > selectedItem ( ) ;
delete chooser ;
}
else if ( accounts . isEmpty ( ) )
{
KMessageBox : : queuedMessageBox ( Kopete : : UI : : Global : : mainWidget ( ) , KMessageBox : : Sorry ,
i18n ( " <qt>You do not have an account configured for <b>%1</b> yet. Please create an account, connect it, and try again.</qt> " ) . arg ( protocolName ) ,
i18n ( " No Account Found " ) ) ;
continue ;
}
else // if we have 1 account in this protocol, choose it
{
chosen = acs . toFirst ( ) ;
}
// add the contact to the chosen account
if ( chosen )
{
kdDebug ( 14010 ) < < " Adding " < < * it < < " to " < < chosen - > accountId ( ) < < endl ;
if ( chosen - > addContact ( * it , mc ) )
contactAdded = true ;
else
KMessageBox : : queuedMessageBox ( Kopete : : UI : : Global : : mainWidget ( ) , KMessageBox : : Sorry ,
i18n ( " <qt>It was not possible to add the contact.</qt> " ) ,
i18n ( " Could Not Add Contact " ) ) ;
}
}
else
kdDebug ( 14010 ) < < " user declined to add " < < * it < < " to contactlist " < < endl ;
}
}
kdDebug ( 14010 ) < < " all " < < addresses . count ( ) < < " contacts in " < < proto - > pluginId ( ) < < " checked " < < endl ;
}
else
kdDebug ( 14010 ) < < " not interested in name= " < < name < < endl ;
}
else
kdDebug ( 14010 ) < < " not interested in app= " < < app < < endl ;
}
}
return contactAdded ;
return false ;
}
// FIXME: Remove when IM address API is in KABC (KDE 4)
void KABCPersistence : : splitField ( const TQString & str , TQString & app , TQString & name , TQString & value )
{
int colon = str . find ( ' : ' ) ;
if ( colon ! = - 1 ) {
TQString tmp = str . left ( colon ) ;
value = str . mid ( colon + 1 ) ;
int dash = tmp . find ( ' - ' ) ;
if ( dash ! = - 1 ) {
app = tmp . left ( dash ) ;
name = tmp . mid ( dash + 1 ) ;
}
}
}
void KABCPersistence : : dumpAB ( )
{
KABC : : AddressBook * ab = addressBook ( ) ;
kdDebug ( 14010 ) < < k_funcinfo < < " DUMPING ADDRESSBOOK " < < endl ;
KABC : : AddressBook : : ConstIterator dumpit = ab - > begin ( ) ;
for ( ; dumpit ! = ab - > end ( ) ; + + dumpit )
{
( * dumpit ) . dump ( ) ;
}
}
} // end namespace Kopete
// dump addressbook contents
# include "kabcpersistence.moc"