// -*- mode: C++; c-file-style: "gnu" -*- // kmfiltermgr.cpp #ifdef HAVE_CONFIG_H #include #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 #include #include // other Qt headers #include // other headers #include //----------------------------------------------------------------------------- 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; while (!mFilters.isEmpty()) delete mFilters.takeFirst(); } //----------------------------------------------------------------------------- void KMFilterMgr::readConfig(void) { KConfig* config = KMKernel::config(); int numFilters; QString grpName; clear(); KConfigGroup group(config, "General"); if (bPopFilter) { numFilters = group.readNumEntry("popfilters",0); mShowLater = group.readNumEntry("popshowDLmsgs",0); } else { numFilters = group.readNumEntry("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(config, 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::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(config); ++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::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 ) { if ( !msg || !filter || !beginFiltering( msg )) return 1; bool stopIt = false; int result = 1; if ( FilterLog::instance()->isLogging() ) { QString logText( i18n( "Evaluating filter rules: " ) ); logText.append( filter->pattern()->asString() ); FilterLog::instance()->add( logText, FilterLog::patternDesc ); } if (filter->pattern()->matches( msg )) { if ( FilterLog::instance()->isLogging() ) { FilterLog::instance()->add( i18n( "Filter rules have matched." ), FilterLog::patternResult ); } 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::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 ( FilterLog::instance()->isLogging() ) { QString logText( i18n( "Evaluating filter rules: " ) ); logText.append( (*it)->pattern()->asString() ); FilterLog::instance()->add( logText, FilterLog::patternDesc ); } if ( (*it)->pattern()->matches( msg ) ) { // filter matches if ( FilterLog::instance()->isLogging() ) { FilterLog::instance()->add( i18n( "Filter rules have matched." ), FilterLog::patternResult ); } 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::atLeastOneFilterAppliesTo( unsigned int accountID ) const { QList::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::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::const_iterator it = mFilters.begin(); for ( ; it != mFilters.end() ; ++it ) { KMFilter *filter = *it; QList::const_iterator jt = filter->actions()->begin(); const QList::const_iterator jtend = filter->actions()->end(); for ( ; jt != jtend ; ++jt ) { KMFilterActionWithFolder *f = dynamic_cast(*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::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 &filters, bool replaceIfNameExists ) { mDirtyBufferedFolderTarget = true; beginUpdate(); if ( replaceIfNameExists ) { QList::const_iterator it1 = filters.begin(); for ( ; it1 != filters.end() ; ++it1 ) { QList::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 &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::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::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"