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.
 
 
 

315 lines
10 KiB

/**
* Copyright (c) 2004 Carsten Burghardt <burghardt@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; version 2 of the License
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* In addition, as a special exception, the copyright holders give
* permission to link the code of this program with any edition of
* the Qt library by Trolltech AS, Norway (or with modified versions
* of Qt that use the same license as Qt), and distribute linked
* combinations including the two. You must obey the GNU General
* Public License in all respects for all of the code used other than
* Qt. If you modify this file, you may extend this exception to
* your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from
* your version.
*/
#include "renamejob.h"
#include "kmfolderimap.h"
#include "kmfoldercachedimap.h"
#include "folderstorage.h"
#include "kmfolder.h"
#include "kmfolderdir.h"
#include "kmfoldertype.h"
#include "kmfoldermgr.h"
#include "imapaccountbase.h"
#include "kmacctimap.h"
#include "kmacctcachedimap.h"
#include "kmcommands.h"
#include "kmmsgbase.h"
#include "undostack.h"
#include <kdebug.h>
#include <kurl.h>
#include <kio/scheduler.h>
#include <kio/job.h>
#include <kio/global.h>
#include <klocale.h>
#include <config.h>
#include <qmap.h>
using namespace KMail;
RenameJob::RenameJob( FolderStorage* storage, const QString& newName,
KMFolderDir* newParent )
: FolderJob( 0, tOther, (storage ? storage->folder() : 0) ),
mStorage( storage ), mNewParent( newParent ),
mNewName( newName ), mNewFolder( 0 )
{
mOldName = storage->name();
if ( storage->folderType() == KMFolderTypeImap ) {
mOldImapPath = static_cast<KMFolderImap*>(storage)->imapPath();
} else if ( storage->folderType() == KMFolderTypeCachedImap ) {
mOldImapPath = static_cast<KMFolderCachedImap*>(storage)->imapPath();
}
}
RenameJob::~RenameJob()
{
}
void RenameJob::execute()
{
if ( mNewParent )
{
// move the folder to a different parent
KMFolderType type = mStorage->folderType();
if ( ( type == KMFolderTypeMbox || type == KMFolderTypeMaildir ) &&
mNewParent->type() == KMStandardDir )
{
// local folders can handle this on their own
mStorage->rename( mNewName, mNewParent );
emit renameDone( mNewName, true );
deleteLater();
return;
}
// first create the new folder
KMFolderMgr* folderMgr = kmkernel->folderMgr();
if ( mNewParent->type() == KMImapDir ) {
folderMgr = kmkernel->imapFolderMgr();
} else if ( mNewParent->type() == KMDImapDir ) {
folderMgr = kmkernel->dimapFolderMgr();
}
// get the default mailbox type
KConfig *config = KMKernel::config();
KConfigGroupSaver saver(config, "General");
int deftype = config->readNumEntry("default-mailbox-format", 1);
if ( deftype < 0 || deftype > 1 ) deftype = 1;
// the type of the new folder
KMFolderType typenew =
( deftype == 0 ) ? KMFolderTypeMbox : KMFolderTypeMaildir;
if ( mNewParent->owner() )
typenew = mNewParent->owner()->folderType();
mNewFolder = folderMgr->createFolder( mNewName, false, typenew, mNewParent );
if ( !mNewFolder )
{
kdWarning(5006) << k_funcinfo << "could not create folder" << endl;
emit renameDone( mNewName, false );
deleteLater();
return;
}
kdDebug(5006)<< "RenameJob::rename - " << mStorage->folder()->idString()
<< " |=> " << mNewFolder->idString() << endl;
if ( mNewParent->type() == KMImapDir )
{
// online imap
// create it on the server and wait for the folderAdded signal
connect( kmkernel->imapFolderMgr(), SIGNAL( changed() ),
this, SLOT( slotMoveMessages() ) );
KMFolderImap* imapFolder =
static_cast<KMFolderImap*>(mNewParent->owner()->storage());
imapFolder->createFolder( mNewName );
} else if ( mNewParent->type() == KMDImapDir )
{
KMFolderCachedImap* newStorage = static_cast<KMFolderCachedImap*>(mNewFolder->storage());
KMFolderCachedImap* owner = static_cast<KMFolderCachedImap*>(mNewParent->owner()->storage());
newStorage->initializeFrom( owner );
slotMoveMessages();
} else
{
// local
slotMoveMessages();
}
} else
{
// only rename the folder
if ( mStorage->folderType() != KMFolderTypeImap )
{
// local and dimap folder handle this directly
mStorage->rename( mNewName );
emit renameDone( mNewName, true );
deleteLater();
return;
}
if ( mOldImapPath.isEmpty() )
{
// sanity
emit renameDone( mNewName, false );
deleteLater();
return;
} else if ( mOldName == mNewName || mOldImapPath == "/INBOX/" ) {
emit renameDone( mNewName, true ); // noop
deleteLater();
return;
}
ImapAccountBase* account = static_cast<KMFolderImap*>(mStorage)->account();
// first rename it on the server
mNewImapPath = mOldImapPath;
mNewImapPath = mNewImapPath.replace( mOldName, mNewName );
KURL src( account->getUrl() );
src.setPath( mOldImapPath );
KURL dst( account->getUrl() );
dst.setPath( mNewImapPath );
KIO::SimpleJob *job = KIO::rename( src, dst, true );
kdDebug(5006)<< "RenameJob::rename - " << src.prettyURL()
<< " |=> " << dst.prettyURL() << endl;
ImapAccountBase::jobData jd( src.url() );
account->insertJob( job, jd );
KIO::Scheduler::assignJobToSlave( account->slave(), job );
connect( job, SIGNAL(result(KIO::Job*)),
SLOT(slotRenameResult(KIO::Job*)) );
}
}
void RenameJob::slotRenameResult( KIO::Job *job )
{
ImapAccountBase* account = static_cast<KMFolderImap*>(mStorage)->account();
ImapAccountBase::JobIterator it = account->findJob(job);
if ( it == account->jobsEnd() )
{
emit renameDone( mNewName, false );
deleteLater();
return;
}
if ( job->error() )
{
account->handleJobError( job, i18n("Error while renaming a folder.") );
emit renameDone( mNewName, false );
deleteLater();
return;
}
account->removeJob(it);
// set the new path
if ( mStorage->folderType() == KMFolderTypeImap )
static_cast<KMFolderImap*>(mStorage)->setImapPath( mNewImapPath );
// unsubscribe old (we don't want ghosts)
account->changeSubscription( false, mOldImapPath );
// subscribe new
account->changeSubscription( true, mNewImapPath );
// local part (will set the new name)
mStorage->rename( mNewName );
emit renameDone( mNewName, true );
deleteLater();
}
void RenameJob::slotMoveMessages()
{
kdDebug(5006) << k_funcinfo << endl;
disconnect( kmkernel->imapFolderMgr(), SIGNAL( changed() ),
this, SLOT( slotMoveMessages() ) );
mStorage->blockSignals( true );
// move all messages to the new folder
QPtrList<KMMsgBase> msgList;
for ( int i = 0; i < mStorage->count(); i++ )
{
KMMsgBase* msgBase = mStorage->getMsgBase( i );
assert( msgBase );
msgList.append( msgBase );
}
if ( msgList.count() == 0 )
{
slotMoveCompleted( 0 );
} else
{
KMCommand *command = new KMMoveCommand( mNewFolder, msgList );
connect( command, SIGNAL( completed( KMCommand * ) ),
this, SLOT( slotMoveCompleted( KMCommand * ) ) );
command->start();
}
}
void RenameJob::slotMoveCompleted( KMCommand* command )
{
kdDebug(5006) << k_funcinfo << (command?command->result():0) << endl;
disconnect( command, SIGNAL( completed( KMCommand * ) ),
this, SLOT( slotMoveCompleted( KMCommand * ) ) );
if ( !command || command->result() == KMCommand::OK )
{
kdDebug(5006) << "deleting old folder" << endl;
// move complete or not necessary
// save our settings
QString oldconfig = "Folder-" + mStorage->folder()->idString();
KConfig* config = KMKernel::config();
QMap<QString, QString> entries = config->entryMap( oldconfig );
KConfigGroupSaver saver(config, "Folder-" + mNewFolder->idString());
for ( QMap<QString, QString>::Iterator it = entries.begin();
it != entries.end(); ++it )
{
if ( it.key() == "Id" || it.key() == "ImapPath" ||
it.key() == "UidValidity" )
continue;
config->writeEntry( it.key(), it.data() );
}
mNewFolder->readConfig( config );
// delete the old folder
mStorage->blockSignals( false );
if ( mStorage->folderType() == KMFolderTypeImap )
{
kmkernel->imapFolderMgr()->remove( mStorage->folder() );
} else if ( mStorage->folderType() == KMFolderTypeCachedImap )
{
// tell the account (see KMFolderCachedImap::listDirectory2)
KMAcctCachedImap* acct = static_cast<KMFolderCachedImap*>(mStorage)->account();
if ( acct )
acct->addDeletedFolder( mOldImapPath );
kmkernel->dimapFolderMgr()->remove( mStorage->folder() );
} else if ( mStorage->folderType() == KMFolderTypeSearch )
{
// invalid
kdWarning(5006) << k_funcinfo << "cannot remove a search folder" << endl;
} else {
kmkernel->folderMgr()->remove( mStorage->folder() );
}
emit renameDone( mNewName, true );
} else
{
kdDebug(5006) << "rollback - deleting folder" << endl;
// move failed - rollback the last transaction
kmkernel->undoStack()->undo();
// .. and delete the new folder
if ( mNewFolder->folderType() == KMFolderTypeImap )
{
kmkernel->imapFolderMgr()->remove( mNewFolder );
} else if ( mNewFolder->folderType() == KMFolderTypeCachedImap )
{
// tell the account (see KMFolderCachedImap::listDirectory2)
KMFolderCachedImap* folder = static_cast<KMFolderCachedImap*>(mNewFolder->storage());
KMAcctCachedImap* acct = folder->account();
if ( acct )
acct->addDeletedFolder( folder->imapPath() );
kmkernel->dimapFolderMgr()->remove( mNewFolder );
} else if ( mNewFolder->folderType() == KMFolderTypeSearch )
{
// invalid
kdWarning(5006) << k_funcinfo << "cannot remove a search folder" << endl;
} else {
kmkernel->folderMgr()->remove( mNewFolder );
}
emit renameDone( mNewName, false );
}
deleteLater();
}
#include "renamejob.moc"