mirror of
https://bitbucket.org/smil3y/kde-playground.git
synced 2025-02-24 19:02:51 +00:00
579 lines
22 KiB
C++
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 ¤tFolder )
|
|
{
|
|
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 ¤tFolder )
|
|
{
|
|
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;
|
|
}
|