diff --git a/kabc/scripts/addressee.src.cpp b/kabc/scripts/addressee.src.cpp index aacbfd9d2..d8b85440d 100644 --- a/kabc/scripts/addressee.src.cpp +++ b/kabc/scripts/addressee.src.cpp @@ -1,1085 +1,1086 @@ /* This file is part of libkabc. Copyright (c) 2001 Cornelius Schumacher Copyright (c) 2003 Carsten Pfeiffer Copyright (c) 2005 Ingo Kloecker This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include #include #include #include #include #include "addresseehelper.h" #include "field.h" #include "resource.h" #include "sortmode.h" #include "addressee.h" using namespace KABC; static bool matchBinaryPattern( int value, int pattern ); template static bool listEquals( const QList&, const QList& ); static bool listEquals( const QStringList&, const QStringList& ); static bool emailsEquals( const QStringList&, const QStringList& ); class Addressee::Private : public QSharedData { public: Private() : mUid( KRandom::randomString( 10 ) ), mResource( 0 ), mEmpty( true ), mChanged( false ) { } Private( const Private &other ) : QSharedData( other ), mResource( 0 ) { mUid = other.mUid; --VARIABLES_ASSIGNMENT-- mPhoneNumbers = other.mPhoneNumbers; mAddresses = other.mAddresses; mKeys = other.mKeys; mEmails = other.mEmails; mCategories = other.mCategories; mCustom = other.mCustom; mResource = other.mResource; mEmpty = other.mEmpty; mChanged = other.mChanged; } ~Private() { } QString mUid; --VARIABLES-- PhoneNumber::List mPhoneNumbers; Address::List mAddresses; Key::List mKeys; QStringList mEmails; QStringList mCategories; QStringList mCustom; Resource *mResource; bool mEmpty :1; bool mChanged :1; static KABC::SortMode *mSortMode; }; KABC::SortMode *Addressee::Private::mSortMode = 0; Addressee::Addressee() : d( new Private ) { } Addressee::~Addressee() { } Addressee::Addressee( const Addressee &other ) : d( other.d ) { } Addressee& Addressee::operator=( const Addressee &other ) { if ( this != &other ) d = other.d; return *this; } bool Addressee::operator==( const Addressee &a ) const { if ( d->mUid != a.d->mUid ) { kDebug() << "uid differs"; return false; } --EQUALSTEST-- if ( ( d->mUrl.isValid() || a.d->mUrl.isValid() ) && ( d->mUrl != a.d->mUrl ) ) { kDebug() << "url differs"; return false; } if ( !listEquals( d->mPhoneNumbers, a.d->mPhoneNumbers ) ) { kDebug() << "phoneNumbers differs"; return false; } if ( !listEquals( d->mAddresses, a.d->mAddresses ) ) { kDebug() << "addresses differs"; return false; } if ( !listEquals( d->mKeys, a.d->mKeys ) ) { kDebug() << "keys differs"; return false; } if ( !emailsEquals( d->mEmails, a.d->mEmails ) ) { kDebug() << "emails differs"; return false; } if ( !listEquals( d->mCategories, a.d->mCategories ) ) { kDebug() << "categories differs"; return false; } if ( !listEquals( d->mCustom, a.d->mCustom ) ) { kDebug() << "custom differs"; return false; } return true; } bool Addressee::operator!=( const Addressee &a ) const { return !( a == *this ); } bool Addressee::isEmpty() const { return d->mEmpty; } void Addressee::setUid( const QString &id ) { if ( id == d->mUid ) return; d->mEmpty = false; d->mUid = id; } QString Addressee::uid() const { return d->mUid; } QString Addressee::uidLabel() { return i18n("Unique Identifier"); } --DEFINITIONS-- void Addressee::setNameFromString( const QString &s ) { QString str = s; //remove enclosing quotes from string if ( str.length() > 1 && s[ 0 ] == '"' && s[ s.length() - 1 ] == '"' ) str = s.mid( 1, s.length() - 2 ); setFormattedName( str ); setName( str ); // clear all name parts setPrefix( QString() ); setGivenName( QString() ); setAdditionalName( QString() ); setFamilyName( QString() ); setSuffix( QString() ); if ( str.isEmpty() ) return; static QString spaceStr = " "; static QString emptyStr = ""; AddresseeHelper *helper = AddresseeHelper::self(); int i = str.indexOf( ',' ); if ( i < 0 ) { QStringList parts = str.split( spaceStr ); int leftOffset = 0; int rightOffset = parts.count() - 1; QString suffix; while ( rightOffset >= 0 ) { if ( helper->containsSuffix( parts[ rightOffset ] ) ) { suffix.prepend(parts[ rightOffset ] + (suffix.isEmpty() ? emptyStr : spaceStr)); rightOffset--; } else break; } setSuffix( suffix ); if ( rightOffset < 0 ) return; if ( rightOffset - 1 >= 0 && helper->containsPrefix( parts[ rightOffset - 1 ].toLower() ) ) { setFamilyName( parts[ rightOffset - 1 ] + spaceStr + parts[ rightOffset ] ); rightOffset--; } else { if ( helper->tradeAsFamilyName() ) setFamilyName( parts[ rightOffset ] ); else setGivenName( parts[ rightOffset ] ); } QString prefix; while ( leftOffset < rightOffset ) { if ( helper->containsTitle( parts[ leftOffset ] ) ) { prefix.append( (prefix.isEmpty() ? emptyStr : spaceStr) + parts[ leftOffset ] ); leftOffset++; } else break; } setPrefix( prefix ); if ( leftOffset < rightOffset ) { setGivenName( parts[ leftOffset ] ); leftOffset++; } QString additionalName; while ( leftOffset < rightOffset ) { additionalName.append( (additionalName.isEmpty() ? emptyStr : spaceStr) + parts[ leftOffset ] ); leftOffset++; } setAdditionalName( additionalName ); } else { QString part1 = str.left( i ); QString part2 = str.mid( i + 1 ); QStringList parts = part1.split( spaceStr ); int leftOffset = 0; int rightOffset = parts.count() - 1; if ( parts.count() > 0 ) { QString suffix; while ( rightOffset >= 0 ) { if ( helper->containsSuffix( parts[ rightOffset ] ) ) { suffix.prepend( parts[ rightOffset ] + (suffix.isEmpty() ? emptyStr : spaceStr) ); rightOffset--; } else break; } setSuffix( suffix ); if ( rightOffset - 1 >= 0 && helper->containsPrefix( parts[ rightOffset - 1 ].toLower() ) ) { setFamilyName( parts[ rightOffset - 1 ] + spaceStr + parts[ rightOffset ] ); rightOffset--; } else setFamilyName( parts[ rightOffset ] ); QString prefix; while ( leftOffset < rightOffset ) { if ( helper->containsTitle( parts[ leftOffset ] ) ) { prefix.append( (prefix.isEmpty() ? emptyStr : spaceStr) + parts[ leftOffset ] ); leftOffset++; } else break; } } else { setPrefix( "" ); setFamilyName( "" ); setSuffix( "" ); } parts = part2.split( spaceStr ); leftOffset = 0; rightOffset = parts.count(); if ( parts.count() > 0 ) { QString prefix; while ( leftOffset < rightOffset ) { if ( helper->containsTitle( parts[ leftOffset ] ) ) { prefix.append( (prefix.isEmpty() ? emptyStr : spaceStr) + parts[ leftOffset ] ); leftOffset++; } else break; } setPrefix( prefix ); if ( leftOffset < rightOffset ) { setGivenName( parts[ leftOffset ] ); leftOffset++; } QString additionalName; while ( leftOffset < rightOffset ) { additionalName.append( (additionalName.isEmpty() ? emptyStr : spaceStr) + parts[ leftOffset ] ); leftOffset++; } setAdditionalName( additionalName ); } else { setGivenName( "" ); setAdditionalName( "" ); } } } QString Addressee::realName() const { QString n( formattedName() ); if ( !n.isEmpty() ) return n; n = assembledName(); if ( !n.isEmpty() ) return n; n = name(); if ( !n.isEmpty() ) return n; return organization(); } QString Addressee::assembledName() const { QString name = prefix() + ' ' + givenName() + ' ' + additionalName() + ' ' + familyName() + ' ' + suffix(); return name.simplified(); } QString Addressee::fullEmail( const QString &email ) const { QString e; if ( email.isNull() ) { e = preferredEmail(); } else { e = email; } if ( e.isEmpty() ) return QString(); QString text; if ( realName().isEmpty() ) text = e; else { QRegExp needQuotes( "[^ 0-9A-Za-z\\x0080-\\xFFFF]" ); if ( realName().indexOf( needQuotes ) != -1 ) { QString name = realName(); name.replace( "\"", "\\\"" ); text = "\"" + name + "\" <" + e + '>'; } else text = realName() + " <" + e + '>'; } return text; } void Addressee::insertEmail( const QString &email, bool preferred ) { if ( email.simplified().isEmpty() ) return; if ( d->mEmails.contains( email ) ) { if ( !preferred || d->mEmails.first() == email ) return; d->mEmails.removeAll( email ); d->mEmails.prepend( email ); } else { + d->mEmpty = false; if ( preferred ) { d->mEmails.prepend( email ); } else { d->mEmails.append( email ); } } } void Addressee::removeEmail( const QString &email ) { if ( d->mEmails.contains( email ) ) { d->mEmails.removeAll( email ); } } QString Addressee::preferredEmail() const { if ( d->mEmails.count() == 0 ) return QString(); else return d->mEmails.first(); } QStringList Addressee::emails() const { return d->mEmails; } void Addressee::setEmails( const QStringList& emails ) { d->mEmails = emails; } void Addressee::insertPhoneNumber( const PhoneNumber &phoneNumber ) { d->mEmpty = false; PhoneNumber::List::Iterator it; for ( it = d->mPhoneNumbers.begin(); it != d->mPhoneNumbers.end(); ++it ) { if ( (*it).id() == phoneNumber.id() ) { *it = phoneNumber; return; } } if ( !phoneNumber.number().simplified().isEmpty() ) d->mPhoneNumbers.append( phoneNumber ); } void Addressee::removePhoneNumber( const PhoneNumber &phoneNumber ) { PhoneNumber::List::Iterator it; for ( it = d->mPhoneNumbers.begin(); it != d->mPhoneNumbers.end(); ++it ) { if ( (*it).id() == phoneNumber.id() ) { d->mPhoneNumbers.erase( it ); return; } } } PhoneNumber Addressee::phoneNumber( PhoneNumber::Type type ) const { PhoneNumber phoneNumber( "", type ); PhoneNumber::List::ConstIterator it; for ( it = d->mPhoneNumbers.constBegin(); it != d->mPhoneNumbers.constEnd(); ++it ) { if ( matchBinaryPattern( (*it).type(), type ) ) { if ( (*it).type() & PhoneNumber::Pref ) return (*it); else if ( phoneNumber.number().isEmpty() ) phoneNumber = (*it); } } return phoneNumber; } PhoneNumber::List Addressee::phoneNumbers() const { return d->mPhoneNumbers; } PhoneNumber::List Addressee::phoneNumbers( PhoneNumber::Type type ) const { PhoneNumber::List list; PhoneNumber::List::ConstIterator it; for ( it = d->mPhoneNumbers.constBegin(); it != d->mPhoneNumbers.constEnd(); ++it ) { if ( matchBinaryPattern( (*it).type(), type ) ) { list.append( *it ); } } return list; } PhoneNumber Addressee::findPhoneNumber( const QString &id ) const { PhoneNumber::List::ConstIterator it; for ( it = d->mPhoneNumbers.constBegin(); it != d->mPhoneNumbers.constEnd(); ++it ) { if ( (*it).id() == id ) { return *it; } } return PhoneNumber(); } void Addressee::insertKey( const Key &key ) { d->mEmpty = false; Key::List::Iterator it; for ( it = d->mKeys.begin(); it != d->mKeys.end(); ++it ) { if ( (*it).id() == key.id() ) { *it = key; return; } } d->mKeys.append( key ); } void Addressee::removeKey( const Key &key ) { Key::List::Iterator it; for ( it = d->mKeys.begin(); it != d->mKeys.end(); ++it ) { if ( (*it).id() == key.id() ) { d->mKeys.removeAll( key ); return; } } } Key Addressee::key( Key::Type type, QString customTypeString ) const { Key::List::ConstIterator it; for ( it = d->mKeys.constBegin(); it != d->mKeys.constEnd(); ++it ) { if ( (*it).type() == type ) { if ( type == Key::Custom ) { if ( customTypeString.isEmpty() ) { return *it; } else { if ( (*it).customTypeString() == customTypeString ) return (*it); } } else { return *it; } } } return Key( QString(), type ); } void Addressee::setKeys( const Key::List& list ) { d->mKeys = list; } Key::List Addressee::keys() const { return d->mKeys; } Key::List Addressee::keys( Key::Type type, QString customTypeString ) const { Key::List list; Key::List::ConstIterator it; for ( it = d->mKeys.constBegin(); it != d->mKeys.constEnd(); ++it ) { if ( (*it).type() == type ) { if ( type == Key::Custom ) { if ( customTypeString.isEmpty() ) { list.append( *it ); } else { if ( (*it).customTypeString() == customTypeString ) list.append( *it ); } } else { list.append( *it ); } } } return list; } Key Addressee::findKey( const QString &id ) const { Key::List::ConstIterator it; for ( it = d->mKeys.constBegin(); it != d->mKeys.constEnd(); ++it ) { if ( (*it).id() == id ) { return *it; } } return Key(); } QString Addressee::toString() const { QString str; str += QString( "Addressee {\n" ); str += QString( " Uid: %1\n" ).arg( uid() ); --DEBUG-- str += QString( " Emails {\n" ); const QStringList e = emails(); QStringList::ConstIterator it; for ( it = e.begin(); it != e.end(); ++it ) { str += QString( " %1\n" ).arg( *it ); } str += QString( " }\n" ); str += QString( " PhoneNumbers {\n" ); const PhoneNumber::List p = phoneNumbers(); PhoneNumber::List::ConstIterator it2; for ( it2 = p.begin(); it2 != p.end(); ++it2 ) { str += (*it2).toString(); } str += QString( " }\n" ); str += QString( " Addresses {\n" ); const Address::List a = addresses(); Address::List::ConstIterator it3; for ( it3 = a.begin(); it3 != a.end(); ++it3 ) { str += (*it3).toString(); } str += QString( " }\n" ); str += QString( " Keys {\n" ); const Key::List k = keys(); Key::List::ConstIterator it4; for ( it4 = k.begin(); it4 != k.end(); ++it4 ) { str += (*it4).toString(); } str += QString( " }\n" ); str += QString( "}\n" ); return str; } void Addressee::insertAddress( const Address &address ) { if ( address.isEmpty() ) return; d->mEmpty = false; Address::List::Iterator it; for ( it = d->mAddresses.begin(); it != d->mAddresses.end(); ++it ) { if ( (*it).id() == address.id() ) { *it = address; return; } } d->mAddresses.append( address ); } void Addressee::removeAddress( const Address &address ) { Address::List::Iterator it; for ( it = d->mAddresses.begin(); it != d->mAddresses.end(); ++it ) { if ( (*it).id() == address.id() ) { d->mAddresses.erase( it ); return; } } } Address Addressee::address( Address::Type type ) const { Address address( type ); Address::List::ConstIterator it; for ( it = d->mAddresses.constBegin(); it != d->mAddresses.constEnd(); ++it ) { if ( matchBinaryPattern( (*it).type(), type ) ) { if ( (*it).type() & Address::Pref ) return (*it); else if ( address.isEmpty() ) address = (*it); } } return address; } Address::List Addressee::addresses() const { return d->mAddresses; } Address::List Addressee::addresses( Address::Type type ) const { Address::List list; Address::List::ConstIterator it; for ( it = d->mAddresses.constBegin(); it != d->mAddresses.constEnd(); ++it ) { if ( matchBinaryPattern( (*it).type(), type ) ) { list.append( *it ); } } return list; } Address Addressee::findAddress( const QString &id ) const { Address::List::ConstIterator it; for ( it = d->mAddresses.constBegin(); it != d->mAddresses.constEnd(); ++it ) { if ( (*it).id() == id ) { return *it; } } return Address(); } void Addressee::insertCategory( const QString &c ) { d->mEmpty = false; if ( d->mCategories.contains( c ) ) return; d->mCategories.append( c ); } void Addressee::removeCategory( const QString &category ) { if ( d->mCategories.contains( category ) ) { d->mCategories.removeAll( category ); } } bool Addressee::hasCategory( const QString &category ) const { return d->mCategories.contains( category ); } void Addressee::setCategories( const QStringList &c ) { d->mEmpty = false; d->mCategories = c; } QStringList Addressee::categories() const { return d->mCategories; } void Addressee::insertCustom( const QString &app, const QString &name, const QString &value ) { if ( value.isEmpty() || name.isEmpty() || app.isEmpty() ) return; d->mEmpty = false; QString qualifiedName = app + '-' + name + ':'; QStringList::Iterator it; for ( it = d->mCustom.begin(); it != d->mCustom.end(); ++it ) { if ( (*it).startsWith( qualifiedName ) ) { (*it) = qualifiedName + value; return; } } d->mCustom.append( qualifiedName + value ); } void Addressee::removeCustom( const QString &app, const QString &name ) { const QString qualifiedName = app + '-' + name + ':'; QStringList::Iterator it; for ( it = d->mCustom.begin(); it != d->mCustom.end(); ++it ) { if ( (*it).startsWith( qualifiedName ) ) { d->mCustom.erase( it ); return; } } } QString Addressee::custom( const QString &app, const QString &name ) const { QString qualifiedName = app + '-' + name + ':'; QString value; QStringList::ConstIterator it; for ( it = d->mCustom.constBegin(); it != d->mCustom.constEnd(); ++it ) { if ( (*it).startsWith( qualifiedName ) ) { value = (*it).mid( (*it).indexOf( ":" ) + 1 ); break; } } return value; } void Addressee::setCustoms( const QStringList &l ) { d->mEmpty = false; d->mCustom = l; } QStringList Addressee::customs() const { return d->mCustom; } void Addressee::parseEmailAddress( const QString &rawEmail, QString &fullName, QString &email ) { // This is a simplified version of KPIM::splitAddress(). fullName = ""; email = ""; if ( rawEmail.isEmpty() ) return; // KPIM::AddressEmpty; // The code works on 8-bit strings, so convert the input to UTF-8. QByteArray address = rawEmail.toUtf8(); QByteArray displayName; QByteArray addrSpec; QByteArray comment; // The following is a primitive parser for a mailbox-list (cf. RFC 2822). // The purpose is to extract a displayable string from the mailboxes. // Comments in the addr-spec are not handled. No error checking is done. enum { TopLevel, InComment, InAngleAddress } context = TopLevel; bool inQuotedString = false; int commentLevel = 0; bool stop = false; for ( char* p = address.data(); *p && !stop; ++p ) { switch ( context ) { case TopLevel : { switch ( *p ) { case '"' : inQuotedString = !inQuotedString; displayName += *p; break; case '(' : if ( !inQuotedString ) { context = InComment; commentLevel = 1; } else displayName += *p; break; case '<' : if ( !inQuotedString ) { context = InAngleAddress; } else displayName += *p; break; case '\\' : // quoted character displayName += *p; ++p; // skip the '\' if ( *p ) displayName += *p; else //return KPIM::UnexpectedEnd; goto ABORT_PARSING; break; case ',' : if ( !inQuotedString ) { //if ( allowMultipleAddresses ) // stop = true; //else // return KPIM::UnexpectedComma; goto ABORT_PARSING; } else displayName += *p; break; default : displayName += *p; } break; } case InComment : { switch ( *p ) { case '(' : ++commentLevel; comment += *p; break; case ')' : --commentLevel; if ( commentLevel == 0 ) { context = TopLevel; comment += ' '; // separate the text of several comments } else comment += *p; break; case '\\' : // quoted character comment += *p; ++p; // skip the '\' if ( *p ) comment += *p; else //return KPIM::UnexpectedEnd; goto ABORT_PARSING; break; default : comment += *p; } break; } case InAngleAddress : { switch ( *p ) { case '"' : inQuotedString = !inQuotedString; addrSpec += *p; break; case '>' : if ( !inQuotedString ) { context = TopLevel; } else addrSpec += *p; break; case '\\' : // quoted character addrSpec += *p; ++p; // skip the '\' if ( *p ) addrSpec += *p; else //return KPIM::UnexpectedEnd; goto ABORT_PARSING; break; default : addrSpec += *p; } break; } } // switch ( context ) } ABORT_PARSING: displayName = displayName.trimmed(); comment = comment.trimmed(); addrSpec = addrSpec.trimmed(); fullName = QString::fromUtf8( displayName ); email = QString::fromUtf8( addrSpec ); // check for errors if ( inQuotedString ) return; // KPIM::UnbalancedQuote; if ( context == InComment ) return; // KPIM::UnbalancedParens; if ( context == InAngleAddress ) return; // KPIM::UnclosedAngleAddr; if ( addrSpec.isEmpty() ) { if ( displayName.isEmpty() ) return; // KPIM::NoAddressSpec; else { //addrSpec = displayName; //displayName.truncate( 0 ); // Address of the form "foo@bar" or "foo@bar (Name)". email = fullName; fullName = QString::fromUtf8( comment ); } } // Check that we do not have any extra characters on the end of the // strings unsigned int len = fullName.length(); if ( fullName[ 0 ] == '"' && fullName[ len - 1 ] == '"' ) fullName = fullName.mid( 1, len - 2 ); } void Addressee::setResource( Resource *resource ) { d->mResource = resource; } Resource *Addressee::resource() const { return d->mResource; } void Addressee::setChanged( bool value ) { d->mChanged = value; } bool Addressee::changed() const { return d->mChanged; } void Addressee::setSortMode( KABC::SortMode *mode ) { Private::mSortMode = mode; } bool Addressee::operator< ( const Addressee &addr ) const { if ( !Private::mSortMode ) return false; else return Private::mSortMode->lesser( *this, addr ); } QDataStream &KABC::operator<<( QDataStream &s, const Addressee &a ) { s << a.d->mUid; --STREAMOUT-- s << a.d->mPhoneNumbers; s << a.d->mAddresses; s << a.d->mEmails; s << a.d->mCategories; s << a.d->mCustom; s << a.d->mKeys; return s; } QDataStream &KABC::operator>>( QDataStream &s, Addressee &a ) { s >> a.d->mUid; --STREAMIN-- s >> a.d->mPhoneNumbers; s >> a.d->mAddresses; s >> a.d->mEmails; s >> a.d->mCategories; s >> a.d->mCustom; s >> a.d->mKeys; a.d->mEmpty = false; return s; } bool matchBinaryPattern( int value, int pattern ) { /** We want to match all telephonnumbers/addresses which have the bits in the pattern set. More are allowed. if pattern == 0 we have a special handling, then we want only those with exactly no bit set. */ if ( pattern == 0 ) return ( value == 0 ); else return ( pattern == ( pattern & value ) ); } template bool listEquals( const QList &list, const QList &pattern ) { if ( list.count() != pattern.count() ) return false; for ( int i = 0; i < list.count(); ++i ) if ( !pattern.contains( list[ i ] ) ) return false; return true; } bool listEquals( const QStringList &list, const QStringList &pattern ) { if ( list.count() != pattern.count() ) return false; for ( int i = 0; i < list.count(); ++i ) if ( !pattern.contains( list[ i ] ) ) return false; return true; } bool emailsEquals( const QStringList &list, const QStringList &pattern ) { if ( list.count() != pattern.count() ) return false; if ( list.isEmpty() ) return true; if ( list.first() != pattern.first() ) return false; QStringList::ConstIterator it; for ( it = list.begin(); it != list.end(); ++it ) if ( !pattern.contains( *it ) ) return false; return true; }