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.
924 lines
32 KiB
924 lines
32 KiB
/* |
|
* kmail: KDE mail client |
|
* Copyright (c) 1996-1998 Stefan Taferner <taferner@kde.org> |
|
* Copyright (c) 2001 Aaron J. Seigo <aseigo@kde.org> |
|
* |
|
* This program is free software; you can redistribute it and/or modify |
|
* it under the terms of the GNU General Public License as published by |
|
* the Free Software Foundation; either version 2 of the License, or |
|
* (at your option) any later version. |
|
* |
|
* This program 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 <config.h> |
|
#include "kmcommands.h" |
|
#include "searchwindow.h" |
|
#include "kmmainwidget.h" |
|
#include "kmmsgdict.h" |
|
#include "kmmsgpart.h" |
|
#include "kmfolderimap.h" |
|
#include "kmfoldermgr.h" |
|
#include "kmfoldersearch.h" |
|
#include "kmfoldertree.h" |
|
#include "kmheaders.h" |
|
#include "kmsearchpatternedit.h" |
|
#include "kmsearchpattern.h" |
|
#include "folderrequester.h" |
|
#include "messagecopyhelper.h" |
|
#include "textsource.h" |
|
|
|
#include <kapplication.h> |
|
#include <kdebug.h> |
|
#include <kstatusbar.h> |
|
#include <kwin.h> |
|
#include <kconfig.h> |
|
#include <kstdaction.h> |
|
#include <kiconloader.h> |
|
|
|
#include <qcheckbox.h> |
|
#include <qlayout.h> |
|
#include <klineedit.h> |
|
#include <qpushbutton.h> |
|
#include <qradiobutton.h> |
|
#include <qbuttongroup.h> |
|
#include <qcombobox.h> |
|
#include <qobjectlist.h> //for mPatternEdit->queryList( 0, "mRuleField" )->first(); |
|
#include <qcursor.h> |
|
#include <qpopupmenu.h> |
|
|
|
#include <maillistdrag.h> |
|
using namespace KPIM; |
|
|
|
#include <mimelib/enum.h> |
|
#include <mimelib/boyermor.h> |
|
|
|
#include <assert.h> |
|
#include <stdlib.h> |
|
|
|
namespace KMail { |
|
|
|
const int SearchWindow::MSGID_COLUMN = 4; |
|
|
|
// KListView sub-class for dnd support |
|
class MatchListView : public KListView |
|
{ |
|
public: |
|
MatchListView( QWidget *parent, SearchWindow* sw, const char* name = 0 ) : |
|
KListView( parent, name ), |
|
mSearchWindow( sw ) |
|
{} |
|
|
|
protected: |
|
virtual QDragObject* dragObject() |
|
{ |
|
KMMessageList list = mSearchWindow->selectedMessages(); |
|
MailList mailList; |
|
for ( KMMsgBase* msg = list.first(); msg; msg = list.next() ) { |
|
if ( !msg ) |
|
continue; |
|
MailSummary mailSummary( msg->getMsgSerNum(), msg->msgIdMD5(), |
|
msg->subject(), msg->fromStrip(), |
|
msg->toStrip(), msg->date() ); |
|
mailList.append( mailSummary ); |
|
} |
|
MailListDrag *d = new MailListDrag( mailList, viewport(), new KMTextSource ); |
|
|
|
QPixmap pixmap; |
|
if( mailList.count() == 1 ) |
|
pixmap = QPixmap( DesktopIcon("message", KIcon::SizeSmall) ); |
|
else |
|
pixmap = QPixmap( DesktopIcon("kmultiple", KIcon::SizeSmall) ); |
|
|
|
d->setPixmap( pixmap ); |
|
return d; |
|
} |
|
|
|
private: |
|
SearchWindow* mSearchWindow; |
|
}; |
|
|
|
//----------------------------------------------------------------------------- |
|
SearchWindow::SearchWindow(KMMainWidget* w, const char* name, |
|
KMFolder *curFolder, bool modal): |
|
KDialogBase(0, name, modal, i18n("Find Messages"), |
|
User1 | User2 | Close, User1, false, |
|
KGuiItem( i18n("&Search"), "find" ), |
|
KStdGuiItem::stop()), |
|
mStopped(false), |
|
mCloseRequested(false), |
|
mSortColumn(0), |
|
mSortOrder(Ascending), |
|
mFolder(0), |
|
mTimer(new QTimer(this, "mTimer")), |
|
mLastFocus(0), |
|
mKMMainWidget(w) |
|
{ |
|
#if !KDE_IS_VERSION( 3, 2, 91 ) |
|
// HACK - KWin keeps all dialogs on top of their mainwindows, but that's probably |
|
// wrong (#76026), and should be done only for modals. CVS HEAD should get |
|
// proper fix in KWin (l.lunak@kde.org) |
|
XDeleteProperty( qt_xdisplay(), winId(), XA_WM_TRANSIENT_FOR ); |
|
#endif |
|
KWin::setIcons(winId(), kapp->icon(), kapp->miniIcon()); |
|
|
|
KConfig* config = KMKernel::config(); |
|
config->setGroup("SearchDialog"); |
|
|
|
QWidget* searchWidget = new QWidget(this); |
|
QVBoxLayout *vbl = new QVBoxLayout( searchWidget, 0, spacingHint(), "kmfs_vbl" ); |
|
|
|
QButtonGroup * radioGroup = new QButtonGroup( searchWidget ); |
|
radioGroup->hide(); |
|
|
|
mChkbxAllFolders = new QRadioButton(i18n("Search in &all local folders"), searchWidget); |
|
vbl->addWidget( mChkbxAllFolders ); |
|
radioGroup->insert( mChkbxAllFolders ); |
|
|
|
QHBoxLayout *hbl = new QHBoxLayout( vbl, spacingHint(), "kmfs_hbl" ); |
|
mChkbxSpecificFolders = new QRadioButton(i18n("Search &only in:"), searchWidget); |
|
hbl->addWidget(mChkbxSpecificFolders); |
|
mChkbxSpecificFolders->setChecked(true); |
|
radioGroup->insert( mChkbxSpecificFolders ); |
|
|
|
mCbxFolders = new FolderRequester( searchWidget, |
|
kmkernel->getKMMainWidget()->folderTree() ); |
|
mCbxFolders->setMustBeReadWrite( false ); |
|
mCbxFolders->setFolder(curFolder); |
|
hbl->addWidget(mCbxFolders); |
|
|
|
mChkSubFolders = new QCheckBox(i18n("I&nclude sub-folders"), searchWidget); |
|
mChkSubFolders->setChecked(true); |
|
hbl->addWidget(mChkSubFolders); |
|
|
|
QWidget *spacer = new QWidget( searchWidget, "spacer" ); |
|
spacer->setMinimumHeight( 2 ); |
|
vbl->addWidget( spacer ); |
|
|
|
mPatternEdit = new KMSearchPatternEdit( "", searchWidget , "spe", false, true ); |
|
mPatternEdit->setFrameStyle( QFrame::NoFrame | QFrame::Plain ); |
|
mPatternEdit->setInsideMargin( 0 ); |
|
mSearchPattern = new KMSearchPattern(); |
|
KMFolderSearch *searchFolder = 0; |
|
if (curFolder) |
|
searchFolder = dynamic_cast<KMFolderSearch*>(curFolder->storage()); |
|
if (searchFolder) { |
|
KConfig config(curFolder->location()); |
|
KMFolder *root = searchFolder->search()->root(); |
|
config.setGroup("Search Folder"); |
|
mSearchPattern->readConfig(&config); |
|
if (root) { |
|
mChkbxSpecificFolders->setChecked(true); |
|
mCbxFolders->setFolder(root); |
|
mChkSubFolders->setChecked(searchFolder->search()->recursive()); |
|
} else { |
|
mChkbxAllFolders->setChecked(true); |
|
} |
|
mFolder = searchFolder; |
|
} |
|
mPatternEdit->setSearchPattern( mSearchPattern ); |
|
QObjectList *list = mPatternEdit->queryList( 0, "mRuleField" ); |
|
QObject *object = 0; |
|
if ( list ) |
|
object = list->first(); |
|
delete list; |
|
if (!searchFolder && object && ::qt_cast<QComboBox*>(object)) |
|
static_cast<QComboBox*>(object)->setCurrentText("Subject"); |
|
|
|
vbl->addWidget( mPatternEdit ); |
|
|
|
// enable/disable widgets depending on radio buttons: |
|
connect( mChkbxSpecificFolders, SIGNAL(toggled(bool)), |
|
mCbxFolders, SLOT(setEnabled(bool)) ); |
|
connect( mChkbxSpecificFolders, SIGNAL(toggled(bool)), |
|
mChkSubFolders, SLOT(setEnabled(bool)) ); |
|
connect( mChkbxAllFolders, SIGNAL(toggled(bool)), |
|
this, SLOT(setEnabledSearchButton(bool)) ); |
|
|
|
mLbxMatches = new MatchListView(searchWidget, this, "Find Messages"); |
|
|
|
/* |
|
Default is to sort by date. TODO: Unfortunately this sorts *while* |
|
inserting, which looks rather strange - the user cannot read |
|
the results so far as they are constantly re-sorted --dnaber |
|
|
|
Sorting is now disabled when a search is started and reenabled |
|
when it stops. Items are appended to the list. This not only |
|
solves the above problem, but speeds searches with many hits |
|
up considerably. - till |
|
|
|
TODO: subclass KListViewItem and do proper (and performant) |
|
comapare functions |
|
*/ |
|
mLbxMatches->setSorting(2, false); |
|
mLbxMatches->setShowSortIndicator(true); |
|
mLbxMatches->setAllColumnsShowFocus(true); |
|
mLbxMatches->setSelectionModeExt(KListView::Extended); |
|
mLbxMatches->addColumn(i18n("Subject"), |
|
config->readNumEntry("SubjectWidth", 150)); |
|
mLbxMatches->addColumn(i18n("Sender/Receiver"), |
|
config->readNumEntry("SenderWidth", 120)); |
|
mLbxMatches->addColumn(i18n("Date"), |
|
config->readNumEntry("DateWidth", 120)); |
|
mLbxMatches->addColumn(i18n("Folder"), |
|
config->readNumEntry("FolderWidth", 100)); |
|
|
|
mLbxMatches->addColumn(""); // should be hidden |
|
mLbxMatches->setColumnWidthMode( MSGID_COLUMN, QListView::Manual ); |
|
mLbxMatches->setColumnWidth(MSGID_COLUMN, 0); |
|
mLbxMatches->header()->setResizeEnabled(false, MSGID_COLUMN); |
|
|
|
mLbxMatches->setDragEnabled( true ); |
|
|
|
connect(mLbxMatches, SIGNAL(doubleClicked(QListViewItem *)), |
|
this, SLOT(slotShowMsg(QListViewItem *))); |
|
connect( mLbxMatches, SIGNAL( contextMenuRequested( QListViewItem*, const QPoint &, int )), |
|
this, SLOT( slotContextMenuRequested( QListViewItem*, const QPoint &, int ))); |
|
vbl->addWidget(mLbxMatches); |
|
|
|
QHBoxLayout *hbl2 = new QHBoxLayout( vbl, spacingHint(), "kmfs_hbl2" ); |
|
mSearchFolderLbl = new QLabel(i18n("Search folder &name:"), searchWidget); |
|
hbl2->addWidget(mSearchFolderLbl); |
|
mSearchFolderEdt = new KLineEdit(searchWidget); |
|
if (searchFolder) |
|
mSearchFolderEdt->setText(searchFolder->folder()->name()); |
|
else |
|
mSearchFolderEdt->setText(i18n("Last Search")); |
|
|
|
mSearchFolderLbl->setBuddy(mSearchFolderEdt); |
|
hbl2->addWidget(mSearchFolderEdt); |
|
mSearchFolderBtn = new QPushButton(i18n("&Rename"), searchWidget); |
|
mSearchFolderBtn->setEnabled(false); |
|
hbl2->addWidget(mSearchFolderBtn); |
|
mSearchFolderOpenBtn = new QPushButton(i18n("Op&en"), searchWidget); |
|
mSearchFolderOpenBtn->setEnabled(false); |
|
hbl2->addWidget(mSearchFolderOpenBtn); |
|
connect( mSearchFolderEdt, SIGNAL( textChanged( const QString &)), |
|
this, SLOT( updateCreateButton( const QString & ))); |
|
connect( mSearchFolderBtn, SIGNAL( clicked() ), |
|
this, SLOT( renameSearchFolder() )); |
|
connect( mSearchFolderOpenBtn, SIGNAL( clicked() ), |
|
this, SLOT( openSearchFolder() )); |
|
mStatusBar = new KStatusBar(searchWidget); |
|
mStatusBar->insertFixedItem(i18n("AMiddleLengthText..."), 0, true); |
|
mStatusBar->changeItem(i18n("Ready."), 0); |
|
mStatusBar->setItemAlignment(0, AlignLeft | AlignVCenter); |
|
mStatusBar->insertItem(QString::null, 1, 1, true); |
|
mStatusBar->setItemAlignment(1, AlignLeft | AlignVCenter); |
|
vbl->addWidget(mStatusBar); |
|
|
|
int mainWidth = config->readNumEntry("SearchWidgetWidth", 0); |
|
int mainHeight = config->readNumEntry("SearchWidgetHeight", 0); |
|
|
|
if (mainWidth || mainHeight) |
|
resize(mainWidth, mainHeight); |
|
|
|
setMainWidget(searchWidget); |
|
setButtonBoxOrientation(QWidget::Vertical); |
|
|
|
mBtnSearch = actionButton(KDialogBase::User1); |
|
mBtnStop = actionButton(KDialogBase::User2); |
|
mBtnStop->setEnabled(false); |
|
|
|
connect(this, SIGNAL(user1Clicked()), SLOT(slotSearch())); |
|
connect(this, SIGNAL(user2Clicked()), SLOT(slotStop())); |
|
connect(this, SIGNAL(finished()), this, SLOT(deleteLater())); |
|
|
|
// give focus to the value field of the first search rule |
|
object = mPatternEdit->child( "regExpLineEdit" ); |
|
if ( object && object->isWidgetType() ) { |
|
static_cast<QWidget*>(object)->setFocus(); |
|
//kdDebug(5006) << "SearchWindow: focus has been given to widget " |
|
// << object->name() << endl; |
|
} |
|
else |
|
kdDebug(5006) << "SearchWindow: regExpLineEdit not found" << endl; |
|
|
|
//set up actions |
|
KActionCollection *ac = actionCollection(); |
|
ac->setWidget( this ); |
|
mReplyAction = new KAction( i18n("&Reply..."), "mail_reply", 0, this, |
|
SLOT(slotReplyToMsg()), ac, "search_reply" ); |
|
mReplyAllAction = new KAction( i18n("Reply to &All..."), "mail_replyall", |
|
0, this, SLOT(slotReplyAllToMsg()), |
|
ac, "search_reply_all" ); |
|
mReplyListAction = new KAction( i18n("Reply to Mailing-&List..."), |
|
"mail_replylist", 0, this, |
|
SLOT(slotReplyListToMsg()), ac, |
|
"search_reply_list" ); |
|
mForwardActionMenu = new KActionMenu( i18n("Message->","&Forward"), |
|
"mail_forward", ac, |
|
"search_message_forward" ); |
|
connect( mForwardActionMenu, SIGNAL(activated()), this, |
|
SLOT(slotForwardInlineMsg()) ); |
|
mForwardAttachedAction = new KAction( i18n("Message->Forward->","As &Attachment..."), |
|
"mail_forward", 0, this, |
|
SLOT(slotForwardAttachedMsg()), ac, |
|
"search_message_forward_as_attachment" ); |
|
mForwardInlineAction = new KAction( i18n("&Inline..."), |
|
"mail_forward", 0, this, |
|
SLOT(slotForwardInlineMsg()), ac, |
|
"search_message_forward_inline" ); |
|
if ( GlobalSettings::self()->forwardingInlineByDefault() ) { |
|
mForwardActionMenu->insert( mForwardInlineAction ); |
|
mForwardActionMenu->insert( mForwardAttachedAction ); |
|
} else { |
|
mForwardActionMenu->insert( mForwardAttachedAction ); |
|
mForwardActionMenu->insert( mForwardInlineAction ); |
|
} |
|
|
|
mForwardDigestAction = new KAction( i18n("Message->Forward->","As Di&gest..."), |
|
"mail_forward", 0, this, |
|
SLOT(slotForwardDigestMsg()), ac, |
|
"search_message_forward_as_digest" ); |
|
mForwardActionMenu->insert( mForwardDigestAction ); |
|
mRedirectAction = new KAction( i18n("Message->Forward->","&Redirect..."), |
|
"mail_forward", 0, this, |
|
SLOT(slotRedirectMsg()), ac, |
|
"search_message_forward_redirect" ); |
|
mForwardActionMenu->insert( mRedirectAction ); |
|
mSaveAsAction = KStdAction::saveAs( this, SLOT(slotSaveMsg()), ac, "search_file_save_as" ); |
|
mSaveAtchAction = new KAction( i18n("Save Attachments..."), "attach", 0, |
|
this, SLOT(slotSaveAttachments()), ac, "search_save_attachments" ); |
|
|
|
mPrintAction = KStdAction::print( this, SLOT(slotPrintMsg()), ac, "search_print" ); |
|
mClearAction = new KAction( i18n("Clear Selection"), 0, 0, this, |
|
SLOT(slotClearSelection()), ac, "search_clear_selection" ); |
|
|
|
mCopyAction = KStdAction::copy( this, SLOT(slotCopyMsgs()), ac, "search_copy_messages" ); |
|
mCutAction = KStdAction::cut( this, SLOT(slotCutMsgs()), ac, "search_cut_messages" ); |
|
|
|
connect(mTimer, SIGNAL(timeout()), this, SLOT(updStatus())); |
|
connect(kmkernel->searchFolderMgr(), SIGNAL(folderInvalidated(KMFolder*)), |
|
this, SLOT(folderInvalidated(KMFolder*))); |
|
|
|
connect(mCbxFolders, SIGNAL(folderChanged(KMFolder*)), |
|
this, SLOT(slotFolderActivated())); |
|
|
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
SearchWindow::~SearchWindow() |
|
{ |
|
QValueListIterator<QGuardedPtr<KMFolder> > fit; |
|
for ( fit = mFolders.begin(); fit != mFolders.end(); ++fit ) { |
|
if (!(*fit)) |
|
continue; |
|
(*fit)->close("searchwindow"); |
|
} |
|
|
|
KConfig* config = KMKernel::config(); |
|
config->setGroup("SearchDialog"); |
|
config->writeEntry("SubjectWidth", mLbxMatches->columnWidth(0)); |
|
config->writeEntry("SenderWidth", mLbxMatches->columnWidth(1)); |
|
config->writeEntry("DateWidth", mLbxMatches->columnWidth(2)); |
|
config->writeEntry("FolderWidth", mLbxMatches->columnWidth(3)); |
|
config->writeEntry("SearchWidgetWidth", width()); |
|
config->writeEntry("SearchWidgetHeight", height()); |
|
config->sync(); |
|
} |
|
|
|
void SearchWindow::setEnabledSearchButton(bool) |
|
{ |
|
//Make sure that button is enable |
|
//Before when we selected a folder == "Local Folder" as that it was not a folder |
|
//search button was disable, and when we select "Search in all local folder" |
|
//Search button was never enabled :( |
|
mBtnSearch->setEnabled( true ); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::updStatus(void) |
|
{ |
|
QString genMsg, detailMsg, procMsg; |
|
int numMatches = 0, numProcessed = 0; |
|
KMSearch const *search = (mFolder) ? (mFolder->search()) : 0; |
|
QString folderName; |
|
if (search) { |
|
numMatches = search->foundCount(); |
|
numProcessed = search->searchCount(); |
|
folderName = search->currentFolder(); |
|
} |
|
|
|
if (search && !search->running()) { |
|
procMsg = i18n("%n message searched", "%n messages searched", |
|
numProcessed); |
|
if(!mStopped) { |
|
genMsg = i18n("Done."); |
|
detailMsg = i18n("%n match in %1", "%n matches in %1", |
|
numMatches).arg(procMsg); |
|
} else { |
|
genMsg = i18n("Search canceled."); |
|
detailMsg = i18n("%n match so far in %1", "%n matches so far in %1", |
|
numMatches).arg(procMsg); |
|
} |
|
} else { |
|
procMsg = i18n("%n message", "%n messages", numProcessed); |
|
genMsg = i18n("%n match", "%n matches", numMatches); |
|
detailMsg = i18n("Searching in %1. %2 searched so far") |
|
.arg(folderName).arg(procMsg); |
|
} |
|
|
|
mStatusBar->changeItem(genMsg, 0); |
|
mStatusBar->changeItem(detailMsg, 1); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::keyPressEvent(QKeyEvent *evt) |
|
{ |
|
KMSearch const *search = (mFolder) ? mFolder->search() : 0; |
|
bool searching = (search) ? search->running() : false; |
|
if (evt->key() == Key_Escape && searching) { |
|
mFolder->stopSearch(); |
|
return; |
|
} |
|
|
|
KDialogBase::keyPressEvent(evt); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotFolderActivated() |
|
{ |
|
mChkbxSpecificFolders->setChecked(true); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::activateFolder(KMFolder *curFolder) |
|
{ |
|
mChkbxSpecificFolders->setChecked(true); |
|
mCbxFolders->setFolder(curFolder); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotSearch() |
|
{ |
|
mLastFocus = focusWidget(); |
|
mBtnSearch->setFocus(); // set focus so we don't miss key event |
|
|
|
mStopped = false; |
|
mFetchingInProgress = 0; |
|
|
|
mSearchFolderOpenBtn->setEnabled(true); |
|
mBtnSearch->setEnabled(false); |
|
mBtnStop->setEnabled(true); |
|
|
|
mLbxMatches->clear(); |
|
|
|
mSortColumn = mLbxMatches->sortColumn(); |
|
mSortOrder = mLbxMatches->sortOrder(); |
|
mLbxMatches->setSorting(-1); |
|
mLbxMatches->setShowSortIndicator(false); |
|
|
|
// If we haven't openend an existing search folder, find or |
|
// create one. |
|
if (!mFolder) { |
|
KMFolderMgr *mgr = kmkernel->searchFolderMgr(); |
|
if (mSearchFolderEdt->text().isEmpty()) |
|
mSearchFolderEdt->setText(i18n("Last Search")); |
|
QString baseName = mSearchFolderEdt->text(); |
|
QString fullName = baseName; |
|
int count = 0; |
|
KMFolder *folder; |
|
while ((folder = mgr->find(fullName))) { |
|
if (folder->storage()->inherits("KMFolderSearch")) |
|
break; |
|
fullName = QString("%1 %2").arg(baseName).arg(++count); |
|
} |
|
|
|
if (!folder) |
|
folder = mgr->createFolder(fullName, false, KMFolderTypeSearch, |
|
&mgr->dir()); |
|
|
|
mFolder = dynamic_cast<KMFolderSearch*>( folder->storage() ); |
|
} |
|
mFolder->stopSearch(); |
|
disconnect(mFolder, SIGNAL(msgAdded(int)), |
|
this, SLOT(slotAddMsg(int))); |
|
disconnect(mFolder, SIGNAL(msgRemoved(KMFolder*, Q_UINT32)), |
|
this, SLOT(slotRemoveMsg(KMFolder*, Q_UINT32))); |
|
connect(mFolder, SIGNAL(msgAdded(int)), |
|
this, SLOT(slotAddMsg(int))); |
|
connect(mFolder, SIGNAL(msgRemoved(KMFolder*, Q_UINT32)), |
|
this, SLOT(slotRemoveMsg(KMFolder*, Q_UINT32))); |
|
KMSearch *search = new KMSearch(); |
|
connect(search, SIGNAL(finished(bool)), |
|
this, SLOT(searchDone())); |
|
if (mChkbxAllFolders->isChecked()) { |
|
search->setRecursive(true); |
|
} else { |
|
search->setRoot(mCbxFolders->folder()); |
|
search->setRecursive(mChkSubFolders->isChecked()); |
|
} |
|
|
|
mPatternEdit->updateSearchPattern(); |
|
KMSearchPattern *searchPattern = new KMSearchPattern(); |
|
*searchPattern = *mSearchPattern; //deep copy |
|
searchPattern->purify(); |
|
search->setSearchPattern(searchPattern); |
|
mFolder->setSearch(search); |
|
enableGUI(); |
|
|
|
if (mFolder && !mFolders.contains(mFolder.operator->()->folder())) { |
|
mFolder->open("searchwindow"); |
|
mFolders.append(mFolder.operator->()->folder()); |
|
} |
|
mTimer->start(200); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::searchDone() |
|
{ |
|
mTimer->stop(); |
|
updStatus(); |
|
|
|
QTimer::singleShot(0, this, SLOT(enableGUI())); |
|
if(mLastFocus) |
|
mLastFocus->setFocus(); |
|
if (mCloseRequested) |
|
close(); |
|
|
|
mLbxMatches->setSorting(mSortColumn, mSortOrder == Ascending); |
|
mLbxMatches->setShowSortIndicator(true); |
|
} |
|
|
|
void SearchWindow::slotAddMsg(int idx) |
|
{ |
|
if (!mFolder) |
|
return; |
|
bool unget = !mFolder->isMessage(idx); |
|
KMMessage *msg = mFolder->getMsg(idx); |
|
QString from, fName; |
|
KMFolder *pFolder = msg->parent(); |
|
if (!mFolders.contains(pFolder)) { |
|
mFolders.append(pFolder); |
|
pFolder->open("searchwindow"); |
|
} |
|
if(pFolder->whoField() == "To") |
|
from = msg->to(); |
|
else |
|
from = msg->from(); |
|
if (pFolder->isSystemFolder()) |
|
fName = i18n(pFolder->name().utf8()); |
|
else |
|
fName = pFolder->name(); |
|
|
|
(void)new KListViewItem(mLbxMatches, mLbxMatches->lastItem(), |
|
msg->subject(), from, msg->dateIsoStr(), |
|
fName, |
|
QString::number(mFolder->serNum(idx))); |
|
if (unget) |
|
mFolder->unGetMsg(idx); |
|
} |
|
|
|
void SearchWindow::slotRemoveMsg(KMFolder *, Q_UINT32 serNum) |
|
{ |
|
if (!mFolder) |
|
return; |
|
QListViewItemIterator it(mLbxMatches); |
|
while (it.current()) { |
|
QListViewItem *item = *it; |
|
if (serNum == (*it)->text(MSGID_COLUMN).toUInt()) { |
|
delete item; |
|
return; |
|
} |
|
++it; |
|
} |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotStop() |
|
{ |
|
if (mFolder) |
|
mFolder->stopSearch(); |
|
mStopped = true; |
|
mBtnStop->setEnabled(false); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotClose() |
|
{ |
|
accept(); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::closeEvent(QCloseEvent *e) |
|
{ |
|
if (mFolder && mFolder->search() && mFolder->search()->running()) { |
|
mCloseRequested = true; |
|
//Cancel search in progress by setting the search folder search to |
|
//the null search |
|
mFolder->setSearch(new KMSearch()); |
|
QTimer::singleShot(0, this, SLOT(slotClose())); |
|
} else { |
|
KDialogBase::closeEvent(e); |
|
} |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::updateCreateButton( const QString &s) |
|
{ |
|
mSearchFolderBtn->setEnabled(s != i18n("Last Search") && mSearchFolderOpenBtn->isEnabled()); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::renameSearchFolder() |
|
{ |
|
if (mFolder && (mFolder->folder()->name() != mSearchFolderEdt->text())) { |
|
int i = 1; |
|
QString name = mSearchFolderEdt->text(); |
|
while (i < 100) { |
|
if (!kmkernel->searchFolderMgr()->find( name )) { |
|
mFolder->rename( name ); |
|
kmkernel->searchFolderMgr()->contentsChanged(); |
|
break; |
|
} |
|
name.setNum( i ); |
|
name = mSearchFolderEdt->text() + " " + name; |
|
++i; |
|
} |
|
} |
|
} |
|
|
|
void SearchWindow::openSearchFolder() |
|
{ |
|
renameSearchFolder(); |
|
mKMMainWidget->slotSelectFolder( mFolder->folder() ); |
|
slotClose(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::folderInvalidated(KMFolder *folder) |
|
{ |
|
if (folder->storage() == mFolder) { |
|
mLbxMatches->clear(); |
|
if (mFolder->search()) |
|
connect(mFolder->search(), SIGNAL(finished(bool)), |
|
this, SLOT(searchDone())); |
|
mTimer->start(200); |
|
enableGUI(); |
|
} |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
bool SearchWindow::slotShowMsg(QListViewItem *item) |
|
{ |
|
if(!item) |
|
return false; |
|
|
|
KMFolder* folder; |
|
int msgIndex; |
|
KMMsgDict::instance()->getLocation(item->text(MSGID_COLUMN).toUInt(), |
|
&folder, &msgIndex); |
|
|
|
if (!folder || msgIndex < 0) |
|
return false; |
|
|
|
mKMMainWidget->slotSelectFolder(folder); |
|
KMMessage* message = folder->getMsg(msgIndex); |
|
if (!message) |
|
return false; |
|
|
|
mKMMainWidget->slotSelectMessage(message); |
|
return true; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::enableGUI() |
|
{ |
|
KMSearch const *search = (mFolder) ? (mFolder->search()) : 0; |
|
bool searching = (search) ? (search->running()) : false; |
|
actionButton(KDialogBase::Close)->setEnabled(!searching); |
|
mCbxFolders->setEnabled(!searching); |
|
mChkSubFolders->setEnabled(!searching); |
|
mChkbxAllFolders->setEnabled(!searching); |
|
mChkbxSpecificFolders->setEnabled(!searching); |
|
mPatternEdit->setEnabled(!searching); |
|
mBtnSearch->setEnabled(!searching); |
|
mBtnStop->setEnabled(searching); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
KMMessageList SearchWindow::selectedMessages() |
|
{ |
|
KMMessageList msgList; |
|
KMFolder* folder = 0; |
|
int msgIndex = -1; |
|
for (QListViewItemIterator it(mLbxMatches); it.current(); it++) |
|
if (it.current()->isSelected()) { |
|
KMMsgDict::instance()->getLocation((*it)->text(MSGID_COLUMN).toUInt(), |
|
&folder, &msgIndex); |
|
if (folder && msgIndex >= 0) |
|
msgList.append(folder->getMsgBase(msgIndex)); |
|
} |
|
return msgList; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
KMMessage* SearchWindow::message() |
|
{ |
|
QListViewItem *item = mLbxMatches->currentItem(); |
|
KMFolder* folder = 0; |
|
int msgIndex = -1; |
|
if (!item) |
|
return 0; |
|
KMMsgDict::instance()->getLocation(item->text(MSGID_COLUMN).toUInt(), |
|
&folder, &msgIndex); |
|
if (!folder || msgIndex < 0) |
|
return 0; |
|
|
|
return folder->getMsg(msgIndex); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::moveSelectedToFolder( int menuId ) |
|
{ |
|
KMFolder *dest = mMenuToFolder[menuId]; |
|
if (!dest) |
|
return; |
|
|
|
KMMessageList msgList = selectedMessages(); |
|
KMCommand *command = new KMMoveCommand( dest, msgList ); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::copySelectedToFolder( int menuId ) |
|
{ |
|
KMFolder *dest = mMenuToFolder[menuId]; |
|
if (!dest) |
|
return; |
|
|
|
KMMessageList msgList = selectedMessages(); |
|
KMCommand *command = new KMCopyCommand( dest, msgList ); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::updateContextMenuActions() |
|
{ |
|
int count = selectedMessages().count(); |
|
bool single_actions = count == 1; |
|
mReplyAction->setEnabled( single_actions ); |
|
mReplyAllAction->setEnabled( single_actions ); |
|
mReplyListAction->setEnabled( single_actions ); |
|
mPrintAction->setEnabled( single_actions ); |
|
mForwardDigestAction->setEnabled( !single_actions ); |
|
mRedirectAction->setEnabled( single_actions ); |
|
mCopyAction->setEnabled( count > 0 ); |
|
mCutAction->setEnabled( count > 0 ); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotContextMenuRequested( QListViewItem *lvi, const QPoint &, int ) |
|
{ |
|
if (!lvi) |
|
return; |
|
mLbxMatches->setSelected( lvi, true ); |
|
mLbxMatches->setCurrentItem( lvi ); |
|
// FIXME is this ever unGetMsg()'d? |
|
if (!message()) |
|
return; |
|
QPopupMenu *menu = new QPopupMenu(this); |
|
updateContextMenuActions(); |
|
|
|
mMenuToFolder.clear(); |
|
QPopupMenu *msgMoveMenu = new QPopupMenu(menu); |
|
mKMMainWidget->folderTree()->folderToPopupMenu( KMFolderTree::MoveMessage, |
|
this, &mMenuToFolder, msgMoveMenu ); |
|
QPopupMenu *msgCopyMenu = new QPopupMenu(menu); |
|
mKMMainWidget->folderTree()->folderToPopupMenu( KMFolderTree::CopyMessage, |
|
this, &mMenuToFolder, msgCopyMenu ); |
|
|
|
// show most used actions |
|
mReplyAction->plug(menu); |
|
mReplyAllAction->plug(menu); |
|
mReplyListAction->plug(menu); |
|
mForwardActionMenu->plug(menu); |
|
menu->insertSeparator(); |
|
mCopyAction->plug(menu); |
|
mCutAction->plug(menu); |
|
menu->insertItem(i18n("&Copy To"), msgCopyMenu); |
|
menu->insertItem(i18n("&Move To"), msgMoveMenu); |
|
menu->insertSeparator(); |
|
mSaveAsAction->plug(menu); |
|
mSaveAtchAction->plug(menu); |
|
mPrintAction->plug(menu); |
|
menu->insertSeparator(); |
|
mClearAction->plug(menu); |
|
menu->exec (QCursor::pos(), 0); |
|
delete menu; |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotClearSelection() |
|
{ |
|
mLbxMatches->clearSelection(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotReplyToMsg() |
|
{ |
|
KMCommand *command = new KMReplyToCommand(this, message()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotReplyAllToMsg() |
|
{ |
|
KMCommand *command = new KMReplyToAllCommand(this, message()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotReplyListToMsg() |
|
{ |
|
KMCommand *command = new KMReplyListCommand(this, message()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotForwardInlineMsg() |
|
{ |
|
KMCommand *command = new KMForwardInlineCommand(this, selectedMessages()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotForwardAttachedMsg() |
|
{ |
|
KMCommand *command = new KMForwardAttachedCommand(this, selectedMessages()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotForwardDigestMsg() |
|
{ |
|
KMCommand *command = new KMForwardDigestCommand(this, selectedMessages()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotRedirectMsg() |
|
{ |
|
KMCommand *command = new KMRedirectCommand(this, message()); |
|
command->start(); |
|
} |
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotSaveMsg() |
|
{ |
|
KMSaveMsgCommand *saveCommand = new KMSaveMsgCommand(this, |
|
selectedMessages()); |
|
if (saveCommand->url().isEmpty()) |
|
delete saveCommand; |
|
else |
|
saveCommand->start(); |
|
} |
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotSaveAttachments() |
|
{ |
|
KMSaveAttachmentsCommand *saveCommand = new KMSaveAttachmentsCommand(this, |
|
selectedMessages()); |
|
saveCommand->start(); |
|
} |
|
|
|
|
|
//----------------------------------------------------------------------------- |
|
void SearchWindow::slotPrintMsg() |
|
{ |
|
KMCommand *command = new KMPrintCommand(this, message()); |
|
command->start(); |
|
} |
|
|
|
void SearchWindow::slotCopyMsgs() |
|
{ |
|
QValueList<Q_UINT32> list = MessageCopyHelper::serNumListFromMsgList( selectedMessages() ); |
|
mKMMainWidget->headers()->setCopiedMessages( list, false ); |
|
} |
|
|
|
void SearchWindow::slotCutMsgs() |
|
{ |
|
QValueList<Q_UINT32> list = MessageCopyHelper::serNumListFromMsgList( selectedMessages() ); |
|
mKMMainWidget->headers()->setCopiedMessages( list, true ); |
|
} |
|
|
|
|
|
void SearchWindow::setSearchPattern( const KMSearchPattern& pattern ) |
|
{ |
|
*mSearchPattern = pattern; |
|
mPatternEdit->setSearchPattern( mSearchPattern ); |
|
} |
|
|
|
} // namespace KMail |
|
#include "searchwindow.moc"
|
|
|