kde-playground/kdepim/messageviewer/utils/util.cpp
2015-04-14 21:49:29 +00:00

579 lines
22 KiB
C++

/*******************************************************************************
**
** Filename : util
** Created on : 03 April, 2005
** Copyright : (c) 2005 Till Adam
** Email : <adam@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.
**
** It 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
**
** 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 "utils/util.h"
#include "utils/iconnamecache.h"
#include "viewer/nodehelper.h"
#include "messagecore/settings/globalsettings.h"
#include "messagecore/helpers/nodehelper.h"
#include "messagecore/utils/stringutil.h"
#include "pimcommon/widgets/renamefiledialog.h"
#include <akonadi/item.h>
#include <kmbox/mbox.h>
#include <KMime/Message>
#include <kcharsets.h>
#include <KFileDialog>
#include <kglobal.h>
#include <klocale.h>
#include <kmessagebox.h>
#include <kio/netaccess.h>
#include <kdebug.h>
#include <KMimeType>
#include <KTemporaryFile>
#include <ktoolinvocation.h>
#include <KAction>
#include <QTextCodec>
#include <QWidget>
#include <QDBusInterface>
#include <QDBusConnectionInterface>
#include <QActionGroup>
#include <QPointer>
using namespace MessageViewer;
bool Util::checkOverwrite( const KUrl &url, QWidget *w )
{
if ( KIO::NetAccess::exists( url, KIO::NetAccess::DestinationSide, w ) ) {
if ( KMessageBox::Cancel == KMessageBox::warningContinueCancel(
w,
i18n( "A file named \"%1\" already exists. "
"Are you sure you want to overwrite it?", url.prettyUrl() ),
i18n( "Overwrite File?" ),
KStandardGuiItem::overwrite() ) )
return false;
}
return true;
}
QString Util::fileNameForMimetype( const QString &mimeType, int iconSize,
const QString &fallbackFileName1,
const QString &fallbackFileName2 )
{
QString fileName;
QString tMimeType = mimeType;
// convert non-registered types to registered types
if ( mimeType == QLatin1String( "application/x-vnd.kolab.contact" ) ) {
tMimeType = QLatin1String( "text/x-vcard" );
} else if ( mimeType == QLatin1String( "application/x-vnd.kolab.event" ) ) {
tMimeType = QLatin1String( "application/x-vnd.akonadi.calendar.event" );
} else if ( mimeType == QLatin1String( "application/x-vnd.kolab.task" ) ) {
tMimeType = QLatin1String( "application/x-vnd.akonadi.calendar.todo" );
} else if ( mimeType == QLatin1String( "application/x-vnd.kolab.journal" ) ) {
tMimeType = QLatin1String( "application/x-vnd.akonadi.calendar.journal" );
} else if ( mimeType == QLatin1String( "application/x-vnd.kolab.note" ) ) {
tMimeType = QLatin1String( "application/x-vnd.akonadi.note" );
}
KMimeType::Ptr mime = KMimeType::mimeType( tMimeType, KMimeType::ResolveAliases );
if ( mime ) {
fileName = mime->iconName();
} else {
fileName = QLatin1String( "unknown" );
if ( !tMimeType.isEmpty() ) {
kWarning() << "unknown mimetype" << tMimeType;
}
}
//WorkAround for #199083
if(fileName == QLatin1String("text-vcard")) {
fileName = QLatin1String("text-x-vcard");
}
if ( fileName.isEmpty() ) {
fileName = fallbackFileName1;
if ( fileName.isEmpty() ) {
fileName = fallbackFileName2;
}
if ( !fileName.isEmpty() ) {
fileName = KMimeType::findByPath( QLatin1String("/tmp/") + fileName, 0, true )->iconName();
}
}
return IconNameCache::instance()->iconPath( fileName, iconSize );
}
#if defined Q_WS_WIN || defined Q_WS_MACX
#include <QDesktopServices>
#endif
bool Util::handleUrlWithQDesktopServices( const KUrl& url )
{
#if defined Q_WS_WIN || defined Q_WS_MACX
QDesktopServices::openUrl( url );
return true;
#else
Q_UNUSED( url );
return false;
#endif
}
QList<KMime::Content*> Util::allContents( const KMime::Content *message )
{
KMime::Content::List result;
KMime::Content *child = MessageCore::NodeHelper::firstChild( message );
if ( child ) {
result += child;
result += allContents( child );
}
KMime::Content *next = MessageCore::NodeHelper::nextSibling( message );
if ( next ) {
result += next;
result += allContents( next );
}
return result;
}
QList<KMime::Content*> Util::extractAttachments( const KMime::Message *message )
{
const KMime::Content::List contents = allContents( message );
KMime::Content::List result;
for ( KMime::Content::List::const_iterator it = contents.constBegin();
it != contents.constEnd(); ) {
KMime::Content* content = *it;
if ( content->contentDisposition()->filename().trimmed().isEmpty() &&
( content->contentType()->name().trimmed().isEmpty() ||
content == message ) ) {
++it;
} else {
result <<( *it );
++it;
}
}
return result;
}
bool Util::saveContents( QWidget *parent, const QList<KMime::Content*> &contents, KUrl &currentFolder )
{
KUrl url, dirUrl;
const bool multiple = (contents.count() > 1);
if ( multiple ) {
// get the dir
dirUrl = KFileDialog::getExistingDirectoryUrl( KUrl( QLatin1String("kfiledialog:///saveAttachment") ),
parent,
i18n( "Save Attachments To" ) );
if ( !dirUrl.isValid() ) {
return false;
}
// we may not get a slash-terminated url out of KFileDialog
dirUrl.adjustPath( KUrl::AddTrailingSlash );
currentFolder = dirUrl;
} else {
// only one item, get the desired filename
KMime::Content *content = contents.first();
QString fileName = NodeHelper::fileName( content );
fileName = MessageCore::StringUtil::cleanFileName( fileName );
if ( fileName.isEmpty() ) {
fileName = i18nc( "filename for an unnamed attachment", "attachment.1" );
}
KUrl pathUrl = KUrl( QLatin1String("kfiledialog:///saveAttachment/"));
pathUrl.addPath(fileName);
url = KFileDialog::getSaveUrl( pathUrl ,
QString(),
parent,
i18n( "Save Attachment" ) );
if ( url.isEmpty() ) {
return false;
}
currentFolder = KUrl(url.upUrl());
}
QMap< QString, int > renameNumbering;
bool globalResult = true;
int unnamedAtmCount = 0;
PimCommon::RenameFileDialog::RenameFileDialogResult result = PimCommon::RenameFileDialog::RENAMEFILE_IGNORE;
foreach( KMime::Content *content, contents ) {
KUrl curUrl;
if ( !dirUrl.isEmpty() ) {
curUrl = dirUrl;
QString fileName = MessageViewer::NodeHelper::fileName( content );
fileName = MessageCore::StringUtil::cleanFileName( fileName );
if ( fileName.isEmpty() ) {
++unnamedAtmCount;
fileName = i18nc( "filename for the %1-th unnamed attachment",
"attachment.%1", unnamedAtmCount );
}
curUrl.setFileName( fileName );
} else {
curUrl = url;
}
if ( !curUrl.isEmpty() ) {
//Bug #312954
if (multiple && (curUrl.fileName() == QLatin1String("smime.p7s")) ) {
continue;
}
// Rename the file if we have already saved one with the same name:
// try appending a number before extension (e.g. "pic.jpg" => "pic_2.jpg")
QString origFile = curUrl.fileName();
QString file = origFile;
while ( renameNumbering.contains(file) ) {
file = origFile;
int num = renameNumbering[file] + 1;
int dotIdx = file.lastIndexOf(QLatin1Char('.'));
file = file.insert( (dotIdx>=0) ? dotIdx : file.length(), QLatin1String("_") + QString::number(num) );
}
curUrl.setFileName(file);
// Increment the counter for both the old and the new filename
if ( !renameNumbering.contains(origFile))
renameNumbering[origFile] = 1;
else
renameNumbering[origFile]++;
if ( file != origFile ) {
if ( !renameNumbering.contains(file))
renameNumbering[file] = 1;
else
renameNumbering[file]++;
}
if( !(result == PimCommon::RenameFileDialog::RENAMEFILE_OVERWRITEALL ||
result == PimCommon::RenameFileDialog::RENAMEFILE_IGNOREALL )) {
if ( KIO::NetAccess::exists( curUrl, KIO::NetAccess::DestinationSide, parent ) ) {
PimCommon::RenameFileDialog *dlg = new PimCommon::RenameFileDialog(curUrl, multiple, parent);
result = static_cast<PimCommon::RenameFileDialog::RenameFileDialogResult>(dlg->exec());
if ( result == PimCommon::RenameFileDialog::RENAMEFILE_IGNORE ||
result == PimCommon::RenameFileDialog::RENAMEFILE_IGNOREALL)
{
delete dlg;
continue;
}
else if ( result == PimCommon::RenameFileDialog::RENAMEFILE_RENAME )
{
curUrl = dlg->newName();
}
delete dlg;
}
}
// save
if( result != PimCommon::RenameFileDialog::RENAMEFILE_IGNOREALL ) {
const bool result = saveContent( parent, content, curUrl );
if ( !result )
globalResult = result;
}
}
}
return globalResult;
}
bool Util::saveContent( QWidget *parent, KMime::Content* content, const KUrl& url )
{
// FIXME: This is all horribly broken. First of all, creating a NodeHelper and then immediatley
// reading out the encryption/signature state will not work at all.
// Then, topLevel() will not work for attachments that are inside encrypted parts.
// What should actually be done is either passing in an ObjectTreeParser that has already
// parsed the message, or creating an OTP here (which would have the downside that the
// password dialog for decrypting messages is shown twice)
#if 0 // totally broken
KMime::Content *topContent = content->topLevel();
MessageViewer::NodeHelper *mNodeHelper = new MessageViewer::NodeHelper;
bool bSaveEncrypted = false;
bool bEncryptedParts = mNodeHelper->encryptionState( content ) != MessageViewer::KMMsgNotEncrypted;
if( bEncryptedParts )
if( KMessageBox::questionYesNo( parent,
i18n( "The part %1 of the message is encrypted. Do you want to keep the encryption when saving?",
url.fileName() ),
i18n( "KMail Question" ), KGuiItem(i18n("Keep Encryption")), KGuiItem(i18n("Do Not Keep")) ) ==
KMessageBox::Yes )
bSaveEncrypted = true;
bool bSaveWithSig = true;
if(mNodeHelper->signatureState( content ) != MessageViewer::KMMsgNotSigned )
if( KMessageBox::questionYesNo( parent,
i18n( "The part %1 of the message is signed. Do you want to keep the signature when saving?",
url.fileName() ),
i18n( "KMail Question" ), KGuiItem(i18n("Keep Signature")), KGuiItem(i18n("Do Not Keep")) ) !=
KMessageBox::Yes )
bSaveWithSig = false;
QByteArray data;
if( bSaveEncrypted || !bEncryptedParts) {
KMime::Content *dataNode = content;
QByteArray rawDecryptedBody;
bool gotRawDecryptedBody = false;
if ( !bSaveWithSig ) {
if ( topContent->contentType()->mimeType() == "multipart/signed" ) {
// carefully look for the part that is *not* the signature part:
if ( ObjectTreeParser::findType( topContent, "application/pgp-signature", true, false ) ) {
dataNode = ObjectTreeParser::findTypeNot( topContent, "application", "pgp-signature", true, false );
} else if ( ObjectTreeParser::findType( topContent, "application/pkcs7-mime" , true, false ) ) {
dataNode = ObjectTreeParser::findTypeNot( topContent, "application", "pkcs7-mime", true, false );
} else {
dataNode = ObjectTreeParser::findTypeNot( topContent, "multipart", "", true, false );
}
} else {
EmptySource emptySource;
ObjectTreeParser otp( &emptySource, 0, 0, false, false );
// process this node and all it's siblings and descendants
mNodeHelper->setNodeUnprocessed( dataNode, true );
otp.parseObjectTree( dataNode );
rawDecryptedBody = otp.rawDecryptedBody();
gotRawDecryptedBody = true;
}
}
QByteArray cstr = gotRawDecryptedBody
? rawDecryptedBody
: dataNode->decodedContent();
data = KMime::CRLFtoLF( cstr );
}
#else
const QByteArray data = content->decodedContent();
kWarning() << "Port the encryption/signature handling when saving a KMime::Content.";
#endif
QDataStream ds;
QFile file;
KTemporaryFile tf;
if ( url.isLocalFile() )
{
// save directly
file.setFileName( url.toLocalFile() );
if ( !file.open( QIODevice::WriteOnly ) )
{
KMessageBox::error( parent,
i18nc( "1 = file name, 2 = error string",
"<qt>Could not write to the file<br><filename>%1</filename><br><br>%2",
file.fileName(),
file.errorString() ),
i18n( "Error saving attachment" ) );
return false;
}
const int permissions = MessageViewer::Util::getWritePermissions();
if ( permissions >= 0 )
fchmod( file.handle(), permissions );
ds.setDevice( &file );
} else
{
// tmp file for upload
tf.open();
ds.setDevice( &tf );
}
const int bytesWritten = ds.writeRawData( data.data(), data.size() );
if ( bytesWritten != data.size() ) {
QFile *f = static_cast<QFile *>( ds.device() );
KMessageBox::error( parent,
i18nc( "1 = file name, 2 = error string",
"<qt>Could not write to the file<br><filename>%1</filename><br><br>%2",
f->fileName(),
f->errorString() ),
i18n( "Error saving attachment" ) );
// Remove the newly created empty or partial file
f->remove();
return false;
}
if ( !url.isLocalFile() )
{
// QTemporaryFile::fileName() is only defined while the file is open
QString tfName = tf.fileName();
tf.close();
if ( !KIO::NetAccess::upload( tfName, url, parent ) )
{
KMessageBox::error( parent,
i18nc( "1 = file name, 2 = error string",
"<qt>Could not write to the file<br><filename>%1</filename><br><br>%2",
url.prettyUrl(),
KIO::NetAccess::lastErrorString() ),
i18n( "Error saving attachment" ) );
return false;
}
}
else
file.close();
#if 0
mNodeHelper->removeTempFiles();
delete mNodeHelper;
#endif
return true;
}
int Util::getWritePermissions()
{
// #79685, #232001 by default use the umask the user defined, but let it be configurable
if ( MessageCore::GlobalSettings::self()->disregardUmask() ) {
return S_IRUSR | S_IWUSR;
} else {
return -1;
}
}
bool Util::saveAttachments( const KMime::Content::List& contents, QWidget *parent, KUrl &currentFolder )
{
if ( contents.isEmpty() ) {
KMessageBox::information( parent, i18n( "Found no attachments to save." ) );
return false;
}
return Util::saveContents( parent, contents, currentFolder );
}
bool Util::saveMessageInMbox( const QList<Akonadi::Item>& retrievedMsgs, QWidget *parent, bool appendMessages )
{
QString fileName;
if ( retrievedMsgs.isEmpty() )
return true;
const Akonadi::Item msgBase = retrievedMsgs.first();
if( msgBase.hasPayload<KMime::Message::Ptr>() )
fileName = MessageCore::StringUtil::cleanFileName(MessageViewer::NodeHelper::cleanSubject ( msgBase.payload<KMime::Message::Ptr>().get() ).trimmed() );
else
fileName = i18n("message");
if ( !fileName.endsWith( QLatin1String( ".mbox" ) ) )
fileName += QLatin1String(".mbox");
const QString filter = i18n( "*.mbox|email messages (*.mbox)\n*|all files (*)" );
QPointer<KFileDialog> dlg = new KFileDialog(KUrl::fromPath( fileName ), filter, parent);
dlg->setCaption(i18np("Save Message", "Save Messages", retrievedMsgs.count()));
dlg->setMode(KFile::File|KFile::LocalOnly);
dlg->setOperationMode(KFileDialog::Saving);
if( !appendMessages )
dlg->setConfirmOverwrite(true);
if (dlg->exec()) {
KUrl url = dlg->selectedUrl();
if ( url.isEmpty() ) {
delete dlg;
return true;
}
const QString localFileName = url.toLocalFile();
if ( localFileName.isEmpty() ) {
delete dlg;
return true;
}
if( !appendMessages ) {
QFile::remove(localFileName);
}
KMBox::MBox mbox;
if ( !mbox.load( localFileName ) ) {
if( appendMessages ) {
KMessageBox::error( parent, i18n("File %1 could not be loaded.",localFileName) , i18n( "Error loading message" ) );
} else {
KMessageBox::error( parent, i18n("File %1 could not be created.",localFileName) , i18n( "Error saving message" ) );
}
delete dlg;
return false;
}
foreach ( const Akonadi::Item &item, retrievedMsgs ) {
if ( item.hasPayload<KMime::Message::Ptr>() ) {
mbox.appendMessage( item.payload<KMime::Message::Ptr>() );
}
}
if ( !mbox.save() ) {
KMessageBox::error( parent, i18n("We cannot save message.") , i18n( "Error saving message" ) );
delete dlg;
return false;
}
}
delete dlg;
return true;
}
bool Util::speakSelectedText( const QString& text, QWidget *parent)
{
if(text.isEmpty())
return false;
// If KTTSD not running, start it.
if (!QDBusConnection::sessionBus().interface()->isServiceRegistered(QLatin1String("org.kde.kttsd")))
{
QString error;
if (KToolInvocation::startServiceByDesktopName(QLatin1String("kttsd"), QStringList(), &error))
{
KMessageBox::error(parent, i18n( "Starting Jovie Text-to-Speech Service Failed"), error );
return false;
}
}
QDBusInterface ktts(QLatin1String("org.kde.kttsd"), QLatin1String("/KSpeech"), QLatin1String("org.kde.KSpeech"));
ktts.asyncCall(QLatin1String("say"), text, 0);
return true;
}
KAction* Util::createAppAction(const KService::Ptr& service, bool singleOffer, QActionGroup *actionGroup, QObject *parent )
{
QString actionName(service->name().replace(QLatin1Char('&'), QLatin1String("&&")));
if (singleOffer) {
actionName = i18n("Open &with %1", actionName);
} else {
actionName = i18nc("@item:inmenu Open With, %1 is application name", "%1", actionName);
}
KAction *act = new KAction(parent);
act->setIcon(KIcon(service->icon()));
act->setText(actionName);
actionGroup->addAction( act );
act->setData(QVariant::fromValue(service));
return act;
}
KMimeType::Ptr Util::mimetype(const QString& name)
{
// consider the filename if mimetype cannot be found by content-type
KMimeType::Ptr mimeType = KMimeType::findByPath( name, 0, true /* no disk access */ );
if ( mimeType->name() == QLatin1String("application/octet-stream") ) {
// consider the attachment's contents if neither the Content-Type header
// nor the filename give us a clue
mimeType = KMimeType::findByFileContent( name );
}
return mimeType;
}