You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

539 lines
16 KiB

/* -*- mode: C++; c-file-style: "gnu" -*-
This file is part of KMail, the KDE mail client.
Copyright (c) 2009 Montel Laurent <montel@kde.org>
KMail is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License, version 2, as
published by the Free Software Foundation.
KMail 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.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "foldercollection.h"
#include <kdebug.h>
#include "kmkernel.h"
#include "util.h"
#include "imapsettings.h"
#include <kpimidentities/identitymanager.h>
#include <kpimidentities/identity.h>
#include <akonadi/itemfetchjob.h>
#include <akonadi/itemfetchscope.h>
#include <akonadi/collectiondeletejob.h>
#include <kio/jobuidelegate.h>
#include "kmcommands.h"
#include "expirejob.h"
#include "compactionjob.h"
#include <QMutex>
#include <QMutexLocker>
#include <QWeakPointer>
using namespace Akonadi;
static QMutex mapMutex;
static QMap<Collection::Id,QWeakPointer<FolderCollection> > fcMap;
QSharedPointer<FolderCollection> FolderCollection::forCollection( const Akonadi::Collection& coll )
{
QMutexLocker lock( &mapMutex );
QSharedPointer<FolderCollection> sptr = fcMap.value( coll.id() ).toStrongRef();
if ( !sptr ) {
sptr = QSharedPointer<FolderCollection>( new FolderCollection( coll, true ) );
fcMap.insert( coll.id(), sptr );
}
return sptr;
}
FolderCollection::FolderCollection( const Akonadi::Collection & col, bool writeconfig )
: mCollection( col ),
mPutRepliesInSameFolder( false ),
mHideInSelectionDialog( false ),
mIgnoreNewMail( false ),
mExpireMessages( false ),
mWriteConfig( writeconfig ),
mUnreadExpireAge( 28 ),
mReadExpireAge( 14 ),
mUnreadExpireUnits( expireNever ),
mReadExpireUnits( expireNever ),
mExpireAction( ExpireDelete )
{
mIdentity = KMKernel::self()->identityManager()->defaultIdentity().uoid();
readConfig();
connect( KMKernel::self()->identityManager(), SIGNAL( changed() ),
this, SLOT( slotIdentitiesChanged() ) );
}
FolderCollection::~FolderCollection()
{
// kDebug()<<" FolderCollection::~FolderCollection";
if ( mWriteConfig )
writeConfig();
}
QString FolderCollection::name() const
{
return mCollection.name();
}
bool FolderCollection::isSystemFolder() const
{
return KMKernel::self()->isSystemFolderCollection( mCollection );
}
bool FolderCollection::noContent() const
{
// FIXME: This is wrong, no content means something else, it is a special IMAP property
return ( count() <= 0 );
}
bool FolderCollection::isReadOnly() const
{
return mCollection.rights() & Akonadi::Collection::ReadOnly;
}
bool FolderCollection::canDeleteMessages() const
{
return mCollection.rights() & Akonadi::Collection::CanDeleteItem;
}
qint64 FolderCollection::count() const
{
return mCollection.statistics().count();
}
Akonadi::Collection::Rights FolderCollection::rights() const
{
return mCollection.rights();
}
Akonadi::CollectionStatistics FolderCollection::statistics() const
{
return mCollection.statistics();
}
Akonadi::Collection FolderCollection::collection()
{
return mCollection;
}
void FolderCollection::slotIdentitiesChanged()
{
uint defaultIdentity = KMKernel::self()->identityManager()->defaultIdentity().uoid();
// The default identity may have changed, therefore set it again
// if necessary
if ( mUseDefaultIdentity )
mIdentity = defaultIdentity;
// Fall back to the default identity if the one used currently is invalid
if ( KMKernel::self()->identityManager()->identityForUoid( mIdentity ).isNull() ) {
mIdentity = defaultIdentity;
mUseDefaultIdentity = true;
}
}
QString FolderCollection::configGroupName() const
{
return "Folder-"+ QString::number( mCollection.id() );
}
void FolderCollection::readConfig()
{
KConfigGroup configGroup( KMKernel::config(), configGroupName() );
mExpireMessages = configGroup.readEntry( "ExpireMessages", false );
mReadExpireAge = configGroup.readEntry( "ReadExpireAge", 3 );
mReadExpireUnits = (ExpireUnits)configGroup.readEntry( "ReadExpireUnits", (int)expireMonths );
mUnreadExpireAge = configGroup.readEntry( "UnreadExpireAge", 12 );
mUnreadExpireUnits = (ExpireUnits)
configGroup.readEntry( "UnreadExpireUnits", (int)expireNever );
mExpireAction = configGroup.readEntry( "ExpireAction", "Delete") == "Move" ? ExpireMove : ExpireDelete;
mExpireToFolderId = configGroup.readEntry( "ExpireToFolder" );
mMailingListEnabled = configGroup.readEntry( "MailingListEnabled", false );
mMailingList.readConfig( configGroup );
mUseDefaultIdentity = configGroup.readEntry( "UseDefaultIdentity", true );
uint defaultIdentity = KMKernel::self()->identityManager()->defaultIdentity().uoid();
mIdentity = configGroup.readEntry("Identity", defaultIdentity );
slotIdentitiesChanged();
setUserWhoField( configGroup.readEntry( "WhoField" ), false );
uint savedId = configGroup.readEntry( "Id", 0 );
#if 0 //TODO ???
// make sure that we don't overwrite a valid id
if ( savedId != 0 && mId == 0 )
mId = savedId;
#endif
mPutRepliesInSameFolder = configGroup.readEntry( "PutRepliesInSameFolder", false );
mHideInSelectionDialog = configGroup.readEntry( "HideInSelectionDialog", false );
mIgnoreNewMail = configGroup.readEntry( "IgnoreNewMail", false );
QString shortcut( configGroup.readEntry( "Shortcut" ) );
if ( !shortcut.isEmpty() ) {
KShortcut sc( shortcut );
setShortcut( sc );
}
configGroup.sync();
}
bool FolderCollection::isValid() const
{
return mCollection.isValid();
}
QString FolderCollection::idString() const
{
return QString::number( mCollection.id() );
}
void FolderCollection::writeConfig() const
{
KConfigGroup configGroup( KMKernel::config(), configGroupName() );
configGroup.writeEntry("ExpireMessages", mExpireMessages);
configGroup.writeEntry("ReadExpireAge", mReadExpireAge);
configGroup.writeEntry("ReadExpireUnits", (int)mReadExpireUnits);
configGroup.writeEntry("UnreadExpireAge", mUnreadExpireAge);
configGroup.writeEntry("UnreadExpireUnits", (int)mUnreadExpireUnits);
configGroup.writeEntry("ExpireAction", mExpireAction == ExpireDelete ? "Delete" : "Move");
configGroup.writeEntry("ExpireToFolder", mExpireToFolderId);
configGroup.writeEntry("MailingListEnabled", mMailingListEnabled);
mMailingList.writeConfig( configGroup );
configGroup.writeEntry( "UseDefaultIdentity", mUseDefaultIdentity );
if ( !mUseDefaultIdentity ) {
int identityId = -1;
OrgKdeAkonadiImapSettingsInterface *imapSettingsInterface = KMail::Util::createImapSettingsInterface( mCollection.resource() );
if ( imapSettingsInterface->isValid() ) {
QDBusReply<int> reply = imapSettingsInterface->accountIdentity();
if ( reply.isValid() ) {
identityId = reply;
}
}
delete imapSettingsInterface;
if ( identityId != -1 && mIdentity != identityId )
configGroup.writeEntry("Identity", mIdentity);
else
configGroup.deleteEntry( "Identity" );
}
else
configGroup.deleteEntry("Identity");
configGroup.writeEntry("WhoField", mUserWhoField);
#if 0 //TODO ????
configGroup.writeEntry("Id", mId);
#endif
configGroup.writeEntry( "PutRepliesInSameFolder", mPutRepliesInSameFolder );
configGroup.writeEntry( "HideInSelectionDialog", mHideInSelectionDialog );
configGroup.writeEntry( "IgnoreNewMail", mIgnoreNewMail );
if ( !mShortcut.isEmpty() )
configGroup.writeEntry( "Shortcut", mShortcut.toString() );
else
configGroup.deleteEntry( "Shortcut" );
}
void FolderCollection::setShortcut( const KShortcut &sc )
{
if ( mShortcut != sc ) {
mShortcut = sc;
emit shortcutChanged( mCollection );
}
}
void FolderCollection::setUserWhoField( const QString& whoField, bool _writeConfig )
{
if ( mUserWhoField == whoField && !whoField.isEmpty() )
return;
if ( whoField.isEmpty() )
{
// default setting
const KPIMIdentities::Identity & identity =
kmkernel->identityManager()->identityForUoidOrDefault( mIdentity );
if ( kmkernel->isSystemFolderCollection(mCollection) && !kmkernel->isImapFolder( mCollection ) ) {
// local system folders
if ( mCollection == kmkernel->inboxCollectionFolder() ||
mCollection == kmkernel->trashCollectionFolder() )
mWhoField = "From";
if ( mCollection == kmkernel->outboxCollectionFolder() ||
mCollection == kmkernel->sentCollectionFolder() ||
mCollection == kmkernel->templatesCollectionFolder() ||
mCollection == kmkernel->draftsCollectionFolder() )
mWhoField = "To";
} else if ( identity.drafts() == idString() ||
identity.templates() == idString() ||
identity.fcc() == idString() )
// drafts, templates or sent of the identity
mWhoField = "To";
else
mWhoField = "From";
} else if ( whoField == "From" || whoField == "To" )
// set the whoField according to the user-setting
mWhoField = whoField;
else {
// this should not happen...
kDebug() << "Illegal setting" << whoField << "for userWhoField!";
return; // don't use the value
}
mUserWhoField = whoField;
if (_writeConfig)
writeConfig();
//TODO fixme !!!
emit viewConfigChanged();
}
void FolderCollection::setUseDefaultIdentity( bool useDefaultIdentity )
{
mUseDefaultIdentity = useDefaultIdentity;
if ( mUseDefaultIdentity )
mIdentity = KMKernel::self()->identityManager()->defaultIdentity().uoid();
KMKernel::self()->slotRequestConfigSync();
}
void FolderCollection::setIdentity( uint identity )
{
mIdentity = identity;
KMKernel::self()->slotRequestConfigSync();
}
uint FolderCollection::identity() const
{
if ( mUseDefaultIdentity ) {
int identityId = -1;
OrgKdeAkonadiImapSettingsInterface *imapSettingsInterface = KMail::Util::createImapSettingsInterface( mCollection.resource() );
if ( imapSettingsInterface->isValid() ) {
QDBusReply<int> reply = imapSettingsInterface->accountIdentity();
if ( reply.isValid() ) {
identityId = reply;
}
}
delete imapSettingsInterface;
if ( identityId != -1 )
return identityId;
}
return mIdentity;
}
void FolderCollection::setAutoExpire( bool enabled )
{
if( enabled != mExpireMessages ) {
mExpireMessages = enabled;
writeConfig();
}
}
void FolderCollection::setUnreadExpireAge( int age )
{
if( age >= 0 && age != mUnreadExpireAge ) {
mUnreadExpireAge = age;
writeConfig();
}
}
void FolderCollection::setUnreadExpireUnits( ExpireUnits units )
{
if (units >= expireNever && units < expireMaxUnits) {
mUnreadExpireUnits = units;
writeConfig();
}
}
void FolderCollection::setReadExpireAge( int age )
{
if( age >= 0 && age != mReadExpireAge ) {
mReadExpireAge = age;
writeConfig();
}
}
void FolderCollection::setReadExpireUnits( ExpireUnits units )
{
if (units >= expireNever && units <= expireMaxUnits) {
mReadExpireUnits = units;
writeConfig();
}
}
QString FolderCollection::mailingListPostAddress() const
{
if ( mMailingList.features() & MailingList::Post ) {
KUrl::List::const_iterator it;
KUrl::List post = mMailingList.postURLS();
for( it = post.constBegin(); it != post.constEnd(); ++it ) {
// We check for isEmpty because before 3.3 postAddress was just an
// email@kde.org and that leaves protocol() field in the kurl class
if ( (*it).protocol() == "mailto" || (*it).protocol().isEmpty() )
return (*it).path();
}
}
return QString();
}
void FolderCollection::setExpireAction( ExpireAction a )
{
if ( a != mExpireAction ) {
mExpireAction = a;
writeConfig();
}
}
void FolderCollection::setExpireToFolderId( const QString& id )
{
if ( id != mExpireToFolderId ) {
mExpireToFolderId = id;
writeConfig();
}
}
void FolderCollection::setMailingListEnabled( bool enabled )
{
mMailingListEnabled = enabled;
writeConfig();
}
void FolderCollection::setMailingList( const MailingList& mlist )
{
mMailingList = mlist;
writeConfig();
}
static int daysToExpire( int number, ExpireUnits units )
{
switch (units) {
case expireDays: // Days
return number;
case expireWeeks: // Weeks
return number * 7;
case expireMonths: // Months - this could be better rather than assuming 31day months.
return number * 31;
default: // this avoids a compiler warning (not handled enumeration values)
;
}
return -1;
}
void FolderCollection::daysToExpire(int& unreadDays, int& readDays) {
unreadDays = ::daysToExpire( getUnreadExpireAge(), getUnreadExpireUnits() );
readDays = ::daysToExpire( getReadExpireAge(), getReadExpireUnits() );
}
void FolderCollection::markNewAsUnread()
{
if ( mCollection.isValid() ) {
Akonadi::ItemFetchJob *job = new Akonadi::ItemFetchJob( mCollection,this );
connect( job, SIGNAL( result( KJob* ) ), this, SLOT( slotMarkNewAsUnreadfetchDone( KJob* ) ) );
}
}
void FolderCollection::slotMarkNewAsUnreadfetchDone( KJob * job )
{
if ( job->error() )
return;
Akonadi::ItemFetchJob *fjob = dynamic_cast<Akonadi::ItemFetchJob*>( job );
Q_ASSERT( fjob );
Akonadi::Item::List items;
foreach( const Akonadi::Item &item, fjob->items() ) {
MessageStatus status;
status.setStatusFromFlags( item.flags() );
if ( !status.isNew() ) {
items.append( item );
}
}
if ( items.empty() )
return;
KMCommand *command = new KMSetStatusCommand( MessageStatus::statusUnread(), items );
command->start();
}
void FolderCollection::markUnreadAsRead()
{
if ( mCollection.isValid() ) {
Akonadi::ItemFetchJob *job = new Akonadi::ItemFetchJob( mCollection,this );
connect( job, SIGNAL( result( KJob* ) ), this, SLOT( slotMarkNewAsReadfetchDone( KJob* ) ) );
}
}
void FolderCollection::slotMarkNewAsReadfetchDone( KJob * job)
{
if ( job->error() )
return;
Akonadi::ItemFetchJob *fjob = dynamic_cast<Akonadi::ItemFetchJob*>( job );
Q_ASSERT( fjob );
Akonadi::Item::List items;
foreach( const Akonadi::Item &item, fjob->items() ) {
MessageStatus status;
status.setStatusFromFlags( item.flags() );
if (status.isNew() || status.isUnread()) {
items.append( item );
}
}
if ( items.empty() )
return;
KMCommand *command = new KMSetStatusCommand( MessageStatus::statusRead(), items );
command->start();
}
void FolderCollection::removeCollection()
{
Akonadi::CollectionDeleteJob *job = new Akonadi::CollectionDeleteJob( mCollection );
connect( job, SIGNAL( result( KJob* ) ), this, SLOT( slotDeletionCollectionResult( KJob* ) ) );
}
void FolderCollection::slotDeletionCollectionResult( KJob * job )
{
if ( job->error() ) {
// handle errors
static_cast<KIO::Job*>(job)->ui()->showErrorMessage();
}
}
void FolderCollection::expireOldMessages( bool immediate )
{
kDebug() << "AKONADI PORT: port it " << Q_FUNC_INFO;
KMail::ScheduledExpireTask* task = new KMail::ScheduledExpireTask(mCollection, immediate);
#if 0
kmkernel->jobScheduler()->registerTask( task );
if ( immediate ) {
// #82259: compact after expiring.
compact( CompactLater );
}
#endif
}
void FolderCollection::compact( FolderCollection::CompactOptions options )
{
kDebug() << "AKONADI PORT: Disabled code in " << Q_FUNC_INFO;
if ( options == CompactLater ) {
KMail::ScheduledCompactionTask* task = new KMail::ScheduledCompactionTask(mCollection, false);
#if 0
kmkernel->jobScheduler()->registerTask( task );
#endif
} else {
#if 0
mStorage->compact( options == CompactSilentlyNow );
#endif
}
}