/* This file is part of KMail Copyright (C) 1999 Waldo Bastian (bastian@kde.org) Copyright (c) 2003 Zack Rusin This library 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. This software 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 library; see the file COPYING. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "undostack.h" #include "kmmainwin.h" #include "kmkernel.h" #include #include #include #include #include #include namespace KMail { UndoStack::UndoStack(int size) : QObject(0), mSize(size), mLastId(0), mCachedInfo(0) { } UndoStack::~UndoStack() { clear(); } void UndoStack::clear() { qDeleteAll( mStack ); mStack.clear(); } QString UndoStack::undoInfo() const { if (!mStack.isEmpty()) { UndoInfo *info = mStack.first(); return info->moveToTrash ? i18n("Move To Trash") : i18np("Move Message", "Move Messages", info->items.count()); } else { return QString(); } } int UndoStack::newUndoAction( const Akonadi::Collection &srcFolder, const Akonadi::Collection &destFolder ) { UndoInfo *info = new UndoInfo; info->id = ++mLastId; info->srcFolder = srcFolder; info->destFolder = destFolder; info->moveToTrash = (destFolder == CommonKernel->trashCollectionFolder()); if ((int) mStack.count() == mSize) { delete mStack.last(); mStack.removeLast(); } mStack.prepend( info ); emit undoStackChanged(); return info->id; } void UndoStack::addMsgToAction( int undoId, const Akonadi::Item &item ) { if ( !mCachedInfo || mCachedInfo->id != undoId ) { QList::const_iterator itr = mStack.constBegin(); while ( itr != mStack.constEnd() ) { if ( (*itr)->id == undoId ) { mCachedInfo = (*itr); break; } ++itr; } } Q_ASSERT( mCachedInfo ); mCachedInfo->items.append( item ); } void UndoStack::undo() { if ( !mStack.isEmpty() ) { UndoInfo *info = mStack.takeFirst(); emit undoStackChanged(); Akonadi::ItemMoveJob * job = new Akonadi::ItemMoveJob( info->items, info->srcFolder, this ); connect( job, SIGNAL(result(KJob*)), this, SLOT(slotMoveResult(KJob*)) ); delete info; } else { // Sorry.. stack is empty.. KMessageBox::sorry( kmkernel->mainWin(), i18n("There is nothing to undo.")); } } void UndoStack::slotMoveResult( KJob *job ) { if ( job->error() ) KMessageBox::sorry( kmkernel->mainWin(), i18n("Cannot move message. %1", job->errorString() ) ); } void UndoStack::pushSingleAction(const Akonadi::Item &item, const Akonadi::Collection &folder, const Akonadi::Collection &destFolder) { int id = newUndoAction( folder, destFolder ); addMsgToAction( id, item ); } void UndoStack::msgDestroyed( const Akonadi::Item & /*msg*/) { /* for (UndoInfo *info = mStack.first(); info; ) { if (info->msgIdMD5 == msg->msgIdMD5()) { mStack.removeRef( info ); info = mStack.current(); } else info = mStack.next(); } */ } void UndoStack::folderDestroyed( const Akonadi::Collection &folder) { QList::iterator it = mStack.begin(); while ( it != mStack.end() ) { UndoInfo *info = *it; if ( info && ( (info->srcFolder == folder) || (info->destFolder == folder) ) ) { delete info; it = mStack.erase( it ); } else ++it; } emit undoStackChanged(); } }