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.
480 lines
13 KiB
480 lines
13 KiB
// -*- mode: C++; c-file-style: "gnu" -*- |
|
// kmfiltermgr.cpp |
|
|
|
#ifdef HAVE_CONFIG_H |
|
#include <config.h> |
|
#endif |
|
|
|
// my header |
|
#include "kmfiltermgr.h" |
|
|
|
// other kmail headers |
|
#include "filterlog.h" |
|
using KMail::FilterLog; |
|
#include "kmfilterdlg.h" |
|
#include "kmfolderindex.h" |
|
#include "messageproperty.h" |
|
using KMail::MessageProperty; |
|
#include "kmfoldermgr.h" |
|
|
|
// other KDE headers |
|
#include <kdebug.h> |
|
#include <klocale.h> |
|
#include <kconfig.h> |
|
|
|
// other Qt headers |
|
#include <qregexp.h> |
|
|
|
// other headers |
|
#include <assert.h> |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
KMFilterMgr::KMFilterMgr( bool popFilter ) |
|
: mEditDialog( 0 ), |
|
bPopFilter( popFilter ), |
|
mShowLater( false ), |
|
mDirtyBufferedFolderTarget( true ), |
|
mBufferedFolderTarget( true ), |
|
mRefCount( 0 ) |
|
{ |
|
if (bPopFilter) |
|
kdDebug(5006) << "pPopFilter set" << endl; |
|
connect( kmkernel, SIGNAL( folderRemoved( KMFolder* ) ), |
|
this, SLOT( slotFolderRemoved( KMFolder* ) ) ); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
KMFilterMgr::~KMFilterMgr() |
|
{ |
|
deref( true ); |
|
writeConfig( false ); |
|
clear(); |
|
} |
|
|
|
void KMFilterMgr::clear() |
|
{ |
|
mDirtyBufferedFolderTarget = true; |
|
qDeleteAll( mFilters ); |
|
mFilters.clear(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::readConfig(void) |
|
{ |
|
KConfig* config = KMKernel::config(); |
|
int numFilters; |
|
QString grpName; |
|
|
|
clear(); |
|
|
|
KConfigGroup group(config, "General"); |
|
|
|
if (bPopFilter) { |
|
numFilters = group.readEntry( "popfilters", 0 ); |
|
mShowLater = group.readEntry( "popshowDLmsgs", 0 ); |
|
} else { |
|
numFilters = group.readEntry( "filters", 0 ); |
|
} |
|
|
|
for ( int i=0 ; i < numFilters ; ++i ) { |
|
grpName.sprintf("%s #%d", (bPopFilter ? "PopFilter" : "Filter") , i); |
|
KConfigGroup group(config, grpName); |
|
KMFilter * filter = new KMFilter(group, bPopFilter); |
|
filter->purify(); |
|
if ( filter->isEmpty() ) { |
|
#ifndef NDEBUG |
|
kdDebug(5006) << "KMFilter::readConfig: filter\n" << filter->asString() |
|
<< "is empty!" << endl; |
|
#endif |
|
delete filter; |
|
} else |
|
mFilters.append(filter); |
|
} |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::writeConfig(bool withSync) |
|
{ |
|
KConfig* config = KMKernel::config(); |
|
|
|
// first, delete all groups: |
|
QStringList filterGroups = |
|
config->groupList().grep( QRegExp( bPopFilter ? "PopFilter #\\d+" : "Filter #\\d+" ) ); |
|
for ( QStringList::Iterator it = filterGroups.begin() ; |
|
it != filterGroups.end() ; ++it ) |
|
config->deleteGroup( *it ); |
|
|
|
// Now, write out the new stuff: |
|
int i = 0; |
|
QString grpName; |
|
for ( QList<KMFilter*>::const_iterator it = mFilters.begin() ; |
|
it != mFilters.end() ; ++it ) { |
|
if ( !(*it)->isEmpty() ) { |
|
if ( bPopFilter ) |
|
grpName.sprintf("PopFilter #%d", i); |
|
else |
|
grpName.sprintf("Filter #%d", i); |
|
KConfigGroup group(config, grpName); |
|
(*it)->writeConfig( group ); |
|
++i; |
|
} |
|
} |
|
|
|
KConfigGroup group(config, "General"); |
|
if (bPopFilter) { |
|
group.writeEntry("popfilters", i); |
|
group.writeEntry("popshowDLmsgs", mShowLater); |
|
} else |
|
group.writeEntry("filters", i); |
|
|
|
if (withSync) config->sync(); |
|
} |
|
|
|
|
|
int KMFilterMgr::processPop( KMMessage * msg ) const { |
|
for ( QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
it != mFilters.end() ; ++it ) |
|
if ( (*it)->pattern()->matches( msg ) ) |
|
return (*it)->action(); |
|
return NoAction; |
|
} |
|
|
|
bool KMFilterMgr::beginFiltering(KMMsgBase *msgBase) const |
|
{ |
|
if (MessageProperty::filtering( msgBase )) |
|
return false; |
|
MessageProperty::setFiltering( msgBase, true ); |
|
MessageProperty::setFilterFolder( msgBase, 0 ); |
|
if ( FilterLog::instance()->isLogging() ) { |
|
FilterLog::instance()->addSeparator(); |
|
} |
|
return true; |
|
} |
|
|
|
int KMFilterMgr::moveMessage(KMMessage *msg) const |
|
{ |
|
if (MessageProperty::filterFolder(msg)->moveMsg( msg ) == 0) { |
|
if ( kmkernel->folderIsTrash( MessageProperty::filterFolder( msg ))) |
|
KMFilterAction::sendMDN( msg, KMime::MDN::Deleted ); |
|
} else { |
|
kdDebug(5006) << "KMfilterAction - couldn't move msg" << endl; |
|
return 2; |
|
} |
|
return 0; |
|
} |
|
|
|
void KMFilterMgr::endFiltering(KMMsgBase *msgBase) const |
|
{ |
|
KMFolder *parent = msgBase->parent(); |
|
if ( parent ) { |
|
if ( parent == MessageProperty::filterFolder( msgBase ) ) { |
|
parent->take( parent->find( msgBase ) ); |
|
} |
|
else if ( ! MessageProperty::filterFolder( msgBase ) ) { |
|
int index = parent->find( msgBase ); |
|
KMMessage *msg = parent->getMsg( index ); |
|
parent->take( index ); |
|
parent->addMsgKeepUID( msg ); |
|
} |
|
} |
|
MessageProperty::setFiltering( msgBase, false ); |
|
} |
|
|
|
int KMFilterMgr::process( KMMessage * msg, const KMFilter * filter ) { |
|
bool stopIt = false; |
|
int result = 1; |
|
|
|
if ( !msg || !filter || !beginFiltering( msg )) |
|
return 1; |
|
|
|
if ( isMatching( msg, filter ) ) { |
|
if (filter->execActions( msg, stopIt ) == KMFilter::CriticalError) |
|
return 2; |
|
|
|
KMFolder *folder = MessageProperty::filterFolder( msg ); |
|
|
|
endFiltering( msg ); |
|
if (folder) { |
|
tempOpenFolder( folder ); |
|
result = folder->moveMsg( msg ); |
|
} |
|
} else { |
|
endFiltering( msg ); |
|
result = 1; |
|
} |
|
return result; |
|
} |
|
|
|
int KMFilterMgr::process( KMMessage * msg, FilterSet set, |
|
bool account, uint accountId ) { |
|
if ( bPopFilter ) |
|
return processPop( msg ); |
|
|
|
if ( set == NoSet ) { |
|
kdDebug(5006) << "KMFilterMgr: process() called with not filter set selected" |
|
<< endl; |
|
return 1; |
|
} |
|
|
|
bool stopIt = false; |
|
bool atLeastOneRuleMatched = false; |
|
|
|
if (!beginFiltering( msg )) |
|
return 1; |
|
for ( QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
!stopIt && it != mFilters.end() ; ++it ) { |
|
|
|
if ( ( ( (set&Inbound) && (*it)->applyOnInbound() ) && |
|
( !account || |
|
( account && (*it)->applyOnAccount( accountId ) ) ) ) || |
|
( (set&Outbound) && (*it)->applyOnOutbound() ) || |
|
( (set&Explicit) && (*it)->applyOnExplicit() ) ) { |
|
// filter is applicable |
|
|
|
if ( isMatching( msg, (*it) ) ) { |
|
// filter matches |
|
atLeastOneRuleMatched = true; |
|
// execute actions: |
|
if ( (*it)->execActions(msg, stopIt) == KMFilter::CriticalError ) |
|
return 2; |
|
} |
|
} |
|
} |
|
|
|
KMFolder *folder = MessageProperty::filterFolder( msg ); |
|
/* endFilter does a take() and addButKeepUID() to ensure the changed |
|
* message is on disk. This is unnessecary if nothing matched, so just |
|
* reset state and don't update the listview at all. */ |
|
if ( atLeastOneRuleMatched ) |
|
endFiltering( msg ); |
|
else |
|
MessageProperty::setFiltering( msg, false ); |
|
if (folder) { |
|
tempOpenFolder( folder ); |
|
folder->moveMsg(msg); |
|
return 0; |
|
} |
|
return 1; |
|
} |
|
|
|
bool KMFilterMgr::isMatching( KMMessage * msg, const KMFilter * filter ) |
|
{ |
|
bool result = false; |
|
if ( FilterLog::instance()->isLogging() ) { |
|
QString logText( i18n( "<b>Evaluating filter rules:</b> " ) ); |
|
logText.append( filter->pattern()->asString() ); |
|
FilterLog::instance()->add( logText, FilterLog::patternDesc ); |
|
} |
|
if ( filter->pattern()->matches( msg ) ) { |
|
if ( FilterLog::instance()->isLogging() ) { |
|
FilterLog::instance()->add( i18n( "<b>Filter rules have matched.</b>" ), |
|
FilterLog::patternResult ); |
|
} |
|
result = true; |
|
} |
|
return result; |
|
} |
|
|
|
bool KMFilterMgr::atLeastOneFilterAppliesTo( unsigned int accountID ) const |
|
{ |
|
QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
for ( ; it != mFilters.end() ; ++it ) { |
|
if ( (*it)->applyOnAccount( accountID ) ) { |
|
return true; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
bool KMFilterMgr::atLeastOneIncomingFilterAppliesTo( unsigned int accountID ) const |
|
{ |
|
QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
for ( ; it != mFilters.end() ; ++it ) { |
|
if ( (*it)->applyOnInbound() && (*it)->applyOnAccount( accountID ) ) { |
|
return true; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
bool KMFilterMgr::atLeastOneOnlineImapFolderTarget() |
|
{ |
|
if (!mDirtyBufferedFolderTarget) |
|
return mBufferedFolderTarget; |
|
|
|
mDirtyBufferedFolderTarget = false; |
|
|
|
QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
for ( ; it != mFilters.end() ; ++it ) { |
|
KMFilter *filter = *it; |
|
QList<KMFilterAction*>::const_iterator jt = filter->actions()->begin(); |
|
const QList<KMFilterAction*>::const_iterator jtend = filter->actions()->end(); |
|
for ( ; jt != jtend ; ++jt ) { |
|
KMFilterActionWithFolder *f = dynamic_cast<KMFilterActionWithFolder*>(*jt); |
|
if (!f) |
|
continue; |
|
QString name = f->argsAsString(); |
|
KMFolder *folder = kmkernel->imapFolderMgr()->findIdString( name ); |
|
if (folder) { |
|
mBufferedFolderTarget = true; |
|
return true; |
|
} |
|
} |
|
} |
|
mBufferedFolderTarget = false; |
|
return false; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::ref(void) |
|
{ |
|
mRefCount++; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::deref(bool force) |
|
{ |
|
if (!force) |
|
mRefCount--; |
|
if (mRefCount < 0) |
|
mRefCount = 0; |
|
if (mRefCount && !force) |
|
return; |
|
QVector< KMFolder *>::const_iterator it; |
|
for ( it = mOpenFolders.begin(); it != mOpenFolders.end(); ++it ) |
|
(*it)->close(); |
|
mOpenFolders.clear(); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
int KMFilterMgr::tempOpenFolder(KMFolder* aFolder) |
|
{ |
|
assert( aFolder ); |
|
|
|
int rc = aFolder->open(); |
|
if (rc) return rc; |
|
|
|
mOpenFolders.append( aFolder ); |
|
return 0; |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::openDialog( QWidget *, bool checkForEmptyFilterList ) |
|
{ |
|
if( !mEditDialog ) |
|
{ |
|
// |
|
// We can't use the parent as long as the dialog is modeless |
|
// and there is one shared dialog for all top level windows. |
|
// |
|
mEditDialog = new KMFilterDlg( 0, "filterdialog", bPopFilter, |
|
checkForEmptyFilterList ); |
|
} |
|
mEditDialog->show(); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::createFilter( const QByteArray & field, const QString & value ) |
|
{ |
|
openDialog( 0, false ); |
|
mEditDialog->createFilter( field, value ); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
const QString KMFilterMgr::createUniqueName( const QString & name ) |
|
{ |
|
QString uniqueName = name; |
|
int counter = 0; |
|
bool found = true; |
|
|
|
while ( found ) { |
|
found = false; |
|
for ( QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
it != mFilters.end(); ++it ) { |
|
if ( !( (*it)->name().compare( uniqueName ) ) ) { |
|
found = true; |
|
++counter; |
|
uniqueName = name; |
|
uniqueName += QString( " (" ) + QString::number( counter ) |
|
+ QString( ")" ); |
|
break; |
|
} |
|
} |
|
} |
|
return uniqueName; |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::appendFilters( const QList<KMFilter*> &filters, |
|
bool replaceIfNameExists ) |
|
{ |
|
mDirtyBufferedFolderTarget = true; |
|
beginUpdate(); |
|
if ( replaceIfNameExists ) { |
|
QList<KMFilter*>::const_iterator it1 = filters.begin(); |
|
for ( ; it1 != filters.end() ; ++it1 ) { |
|
QList<KMFilter*>::const_iterator it2 = mFilters.begin(); |
|
for ( ; it2 != mFilters.end() ; ++it2 ) { |
|
if ( (*it1)->name() == (*it2)->name() ) { |
|
mFilters.remove( (*it2) ); |
|
it2 = mFilters.constBegin(); |
|
} |
|
} |
|
} |
|
} |
|
mFilters += filters; |
|
writeConfig( true ); |
|
endUpdate(); |
|
} |
|
|
|
void KMFilterMgr::setFilters( const QList<KMFilter*> &filters ) |
|
{ |
|
clear(); |
|
mFilters = filters; |
|
} |
|
|
|
void KMFilterMgr::slotFolderRemoved( KMFolder * aFolder ) |
|
{ |
|
folderRemoved( aFolder, 0 ); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
bool KMFilterMgr::folderRemoved(KMFolder* aFolder, KMFolder* aNewFolder) |
|
{ |
|
mDirtyBufferedFolderTarget = true; |
|
bool rem = false; |
|
QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
for ( ; it != mFilters.end() ; ++it ) |
|
if ( (*it)->folderRemoved(aFolder, aNewFolder) ) |
|
rem = true; |
|
|
|
return rem; |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
#ifndef NDEBUG |
|
void KMFilterMgr::dump(void) const |
|
{ |
|
QList<KMFilter*>::const_iterator it = mFilters.begin(); |
|
for ( ; it != mFilters.end() ; ++it ) { |
|
kdDebug(5006) << (*it)->asString() << endl; |
|
} |
|
} |
|
#endif |
|
|
|
//----------------------------------------------------------------------------- |
|
void KMFilterMgr::endUpdate(void) |
|
{ |
|
emit filterListUpdated(); |
|
} |
|
|
|
#include "kmfiltermgr.moc"
|
|
|