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.
 
 
 

1813 lines
65 KiB

/*
This file is part of KMail.
Copyright (c) 2003 Steffen Hansen <steffen@klaralvdalens-datakonsult.se>
Copyright (c) 2003 - 2004 Bo Thorsen <bo@sonofthor.dk>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. 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.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include "kmailicalifaceimpl.h"
#include "kmfoldertree.h"
#include "kmfolderdir.h"
#include "kmgroupware.h"
#include "kmfoldermgr.h"
#include "kmcommands.h"
#include "kmfolderindex.h"
#include "kmmsgdict.h"
#include "kmmsgpart.h"
#include "kmfolderimap.h"
#include "globalsettings.h"
#include "kmacctmgr.h"
#include <mimelib/enum.h>
#include <mimelib/utility.h>
#include <mimelib/body.h>
#include <mimelib/mimepp.h>
#include <kdebug.h>
#include <kiconloader.h>
#include <dcopclient.h>
#include <kmessagebox.h>
#include <kconfig.h>
#include <kurl.h>
#include <qmap.h>
#include <ktempfile.h>
#include <qfile.h>
#include <qdom.h>
#include "kmfoldercachedimap.h"
// Local helper methods
static void vPartMicroParser( const QString& str, QString& s );
static void reloadFolderTree();
// The index in this array is the KMail::FolderContentsType enum
static const struct {
const char* contentsTypeStr; // the string used in the DCOP interface
const char* mimetype;
KFolderTreeItem::Type treeItemType;
const char* annotation;
} s_folderContentsType[] = {
{ "Mail", "application/x-vnd.kolab.mail", KFolderTreeItem::Other, "mail"},
{ "Calendar", "application/x-vnd.kolab.event", KFolderTreeItem::Calendar, "event" },
{ "Contact", "application/x-vnd.kolab.contact", KFolderTreeItem::Contacts, "contact" },
{ "Note", "application/x-vnd.kolab.note", KFolderTreeItem::Notes, "note" },
{ "Task", "application/x-vnd.kolab.task", KFolderTreeItem::Tasks, "task" },
{ "Journal", "application/x-vnd.kolab.journal", KFolderTreeItem::Journals, "journal" }
};
static QString folderContentsType( KMail::FolderContentsType type )
{
return s_folderContentsType[type].contentsTypeStr;
}
static QString folderKolabMimeType( KMail::FolderContentsType type )
{
return s_folderContentsType[type].mimetype;
}
static KMail::FolderContentsType folderContentsType( const QString& type )
{
for ( uint i = 0 ; i < sizeof s_folderContentsType / sizeof *s_folderContentsType; ++i )
if ( type == s_folderContentsType[i].contentsTypeStr )
return static_cast<KMail::FolderContentsType>( i );
return KMail::ContentsTypeMail;
}
const char* KMailICalIfaceImpl::annotationForContentsType( KMail::FolderContentsType type )
{
return s_folderContentsType[type].annotation;
}
/*
This interface has three parts to it - libkcal interface;
kmail interface; and helper functions.
The libkcal interface and the kmail interface have the same three
methods: add, delete and refresh. The only difference is that the
libkcal interface is used from the IMAP resource in libkcal and
the kmail interface is used from the groupware object in kmail.
*/
KMailICalIfaceImpl::KMailICalIfaceImpl()
: DCOPObject( "KMailICalIface" ), QObject( 0, "KMailICalIfaceImpl" ),
mContacts( 0 ), mCalendar( 0 ), mNotes( 0 ), mTasks( 0 ), mJournals( 0 ),
mFolderLanguage( 0 ), mUseResourceIMAP( false ), mResourceQuiet( false ),
mHideFolders( true )
{
// Listen to config changes
connect( kmkernel, SIGNAL( configChanged() ), this, SLOT( readConfig() ) );
connect( kmkernel, SIGNAL( folderRemoved( KMFolder* ) ),
this, SLOT( slotFolderRemoved( KMFolder* ) ) );
mExtraFolders.setAutoDelete( true );
}
// Receive an iCal or vCard from the resource
bool KMailICalIfaceImpl::addIncidence( const QString& type,
const QString& folder,
const QString& uid,
const QString& ical )
{
kdDebug(5006) << "KMailICalIfaceImpl::addIncidence( " << type << ", "
<< uid << ", " << ical << " )" << endl;
if( !mUseResourceIMAP )
return false;
// Find the folder
KMFolder* f = folderFromType( type, folder );
if( !f ) {
kdError(5006) << "addIncidence(" << type << "," << folder << ") : Not an IMAP resource folder" << endl;
return false;
}
if ( storageFormat( f ) != StorageIcalVcard ) {
kdError(5006) << "addIncidence(" << type << "," << folder << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return false;
}
bool rc = false;
bool quiet = mResourceQuiet;
mResourceQuiet = true;
// Make a new message for the incidence
KMMessage* msg = new KMMessage();
msg->initHeader();
msg->setType( DwMime::kTypeText );
if( f == mContacts ) {
msg->setSubtype( DwMime::kSubtypeXVCard );
msg->setHeaderField( "Content-Type", "Text/X-VCard; charset=\"utf-8\"" );
msg->setSubject( "vCard " + uid );
} else {
msg->setSubtype( DwMime::kSubtypeVCal );
msg->setHeaderField("Content-Type",
"text/calendar; method=REQUEST; charset=\"utf-8\"");
msg->setSubject( "iCal " + uid );
}
msg->setBodyEncoded( ical.utf8() );
// Mark the message as read and store it in the folder
msg->touch();
f->addMsg( msg );
rc = true;
addFolderChange( f, Contents );
mResourceQuiet = quiet;
return rc;
}
// Helper function to find an attachment of a given mimetype
// Can't use KMMessage::findDwBodyPart since it only works with known mimetypes.
static DwBodyPart* findBodyPartByMimeType( const KMMessage& msg, const char* sType, const char* sSubtype )
{
// quickly searching for our message part: since Kolab parts are
// top-level parts we do *not* have to travel into embedded multiparts
DwBodyPart* part = msg.getFirstDwBodyPart();
while( part ){
// kdDebug() << part->Headers().ContentType().TypeStr().c_str() << " "
// << part->Headers().ContentType().SubtypeStr().c_str() << endl;
if ( part->hasHeaders()
&& part->Headers().HasContentType()
&& part->Headers().ContentType().TypeStr() == sType
&& part->Headers().ContentType().SubtypeStr() == sSubtype)
return part;
part = part->Next();
}
return 0;
}
// Helper function to find an attachment with a given filename
static DwBodyPart* findBodyPart( const KMMessage& msg, const QString& attachmentName )
{
// quickly searching for our message part: since Kolab parts are
// top-level parts we do *not* have to travel into embedded multiparts
for ( DwBodyPart* part = msg.getFirstDwBodyPart(); part; part = part->Next() ) {
//kdDebug(5006) << "findBodyPart: - " << part->Headers().ContentDisposition().Filename().c_str() << endl;
if ( part->hasHeaders()
&& attachmentName == part->Headers().ContentDisposition().Filename().c_str() )
return part;
}
return 0;
}
#if 0
static void debugBodyParts( const char* foo, const KMMessage& msg )
{
kdDebug(5006) << "--debugBodyParts " << foo << "--" << endl;
for ( DwBodyPart* part = msg.getFirstDwBodyPart(); part; part = part->Next() ) {
if ( part->hasHeaders() ) {
kdDebug(5006) << " bodypart: " << part << endl;
kdDebug(5006) << " " << part->Headers().AsString().c_str() << endl;
}
else
kdDebug(5006) << " part " << part << " has no headers" << endl;
}
}
#else
inline static void debugBodyParts( const char*, const KMMessage& ) {}
#endif
// Add (or overwrite, resp.) an attachment in an existing mail,
// attachments must be local files, they are identified by their names.
// If lookupByName if false the attachment to replace is looked up by mimetype.
// return value: wrong if attachment could not be added/updated
bool KMailICalIfaceImpl::updateAttachment( KMMessage& msg,
const QString& attachmentURL,
const QString& attachmentName,
const QString& attachmentMimetype,
bool lookupByName )
{
kdDebug(5006) << "KMailICalIfaceImpl::updateAttachment( " << attachmentURL << " )" << endl;
bool bOK = false;
KURL url( attachmentURL );
if ( url.isValid() && url.isLocalFile() ) {
const QString fileName( url.path() );
QFile file( fileName );
if( file.open( IO_ReadOnly ) ) {
QByteArray rawData = file.readAll();
file.close();
// create the new message part with data read from temp file
KMMessagePart msgPart;
msgPart.setName( attachmentName );
const int iSlash = attachmentMimetype.find('/');
const QCString sType = attachmentMimetype.left( iSlash ).latin1();
const QCString sSubtype = attachmentMimetype.mid( iSlash+1 ).latin1();
msgPart.setTypeStr( sType );
msgPart.setSubtypeStr( sSubtype );
QCString ctd("attachment;\n filename=\"");
ctd.append( attachmentName.latin1() );
ctd.append("\"");
msgPart.setContentDisposition( ctd );
QValueList<int> dummy;
msgPart.setBodyAndGuessCte( rawData, dummy );
msgPart.setPartSpecifier( fileName );
DwBodyPart* newPart = msg.createDWBodyPart( &msgPart );
// This whole method is a bit special. We mix code for writing and code for reading.
// E.g. we need to parse the content-disposition again for ContentDisposition().Filename()
// to work later on.
newPart->Headers().ContentDisposition().Parse();
DwBodyPart* part = lookupByName ? findBodyPart( msg, attachmentName )
: findBodyPartByMimeType( msg, sType, sSubtype );
if ( part ) {
// Make sure the replacing body part is pointing
// to the same next part as the original body part.
newPart->SetNext( part->Next() );
// call DwBodyPart::operator =
// which calls DwEntity::operator =
*part = *newPart;
delete newPart;
msg.setNeedsAssembly();
kdDebug(5006) << "Attachment " << attachmentName << " updated." << endl;
} else {
msg.addDwBodyPart( newPart );
kdDebug(5006) << "Attachment " << attachmentName << " added." << endl;
}
bOK = true;
}else{
kdDebug(5006) << "Attachment " << attachmentURL << " can not be read." << endl;
}
}else{
kdDebug(5006) << "Attachment " << attachmentURL << " not a local file." << endl;
}
return bOK;
}
// Look for the attachment with the right mimetype
bool KMailICalIfaceImpl::kolabXMLFoundAndDecoded( const KMMessage& msg, const QString& mimetype, QString& s )
{
const int iSlash = mimetype.find('/');
const QCString sType = mimetype.left( iSlash ).latin1();
const QCString sSubtype = mimetype.mid( iSlash+1 ).latin1();
DwBodyPart* part = findBodyPartByMimeType( msg, sType, sSubtype );
if ( part ) {
KMMessagePart msgPart;
KMMessage::bodyPart(part, &msgPart);
s = msgPart.bodyToUnicode();
return true;
}
return false;
}
// Delete an attachment in an existing mail.
// return value: wrong if attachment could not be deleted
//
// This code could be optimized: for now we just replace
// the attachment by an empty dummy attachment since Mimelib
// does not provide an option for deleting attachments yet.
bool KMailICalIfaceImpl::deleteAttachment( KMMessage& msg,
const QString& attachmentName )
{
kdDebug(5006) << "KMailICalIfaceImpl::deleteAttachment( " << attachmentName << " )" << endl;
bool bOK = false;
// quickly searching for our message part: since Kolab parts are
// top-level parts we do *not* have to travel into embedded multiparts
DwBodyPart* part = findBodyPart( msg, attachmentName );
if ( part ) {
msg.getTopLevelPart()->Body().RemoveBodyPart( part );
delete part;
msg.setNeedsAssembly();
kdDebug(5006) << "Attachment deleted." << endl;
bOK = true;
}
if( !bOK ){
kdDebug(5006) << "Attachment " << attachmentName << " not found." << endl;
}
return bOK;
}
// Store a new entry that was received from the resource
Q_UINT32 KMailICalIfaceImpl::addIncidenceKolab( KMFolder& folder,
const QString& subject,
const QStringList& attachmentURLs,
const QStringList& attachmentNames,
const QStringList& attachmentMimetypes )
{
kdDebug(5006) << "KMailICalIfaceImpl::addIncidenceKolab( " << attachmentNames << " )" << endl;
Q_UINT32 sernum = 0;
bool bAttachOK = true;
// Make a new message for the incidence
KMMessage* msg = new KMMessage();
msg->initHeader();
msg->setType( DwMime::kTypeMultipart );
msg->setSubtype( DwMime::kSubtypeMixed );
msg->headers().ContentType().CreateBoundary( 0 );
msg->headers().ContentType().Assemble();
msg->setSubject( subject );
msg->setAutomaticFields( true );
// add a first body part to be displayed by all mailer
// than can NOT display Kolab data: no matter if these
// mailers are MIME compliant or not
KMMessagePart firstPart;
firstPart.setType( DwMime::kTypeText );
firstPart.setSubtype( DwMime::kSubtypePlain );
const char * firstPartTextUntranslated = I18N_NOOP(
"This is a Kolab Groupware object.\nTo view this object you"
" will need an email client that can understand the Kolab"
" Groupware format.\nFor a list of such email clients please"
" visit\nhttp://www.kolab.org/kolab2-clients.html");
QString firstPartText = i18n( firstPartTextUntranslated );
if ( firstPartText != firstPartTextUntranslated ) {
firstPartText.append("\n\n-----------------------------------------------------\n\n");
firstPartText.append( firstPartTextUntranslated );
}
firstPart.setBodyFromUnicode( firstPartText );
msg->addBodyPart( &firstPart );
Q_ASSERT( attachmentMimetypes.count() == attachmentURLs.count() );
Q_ASSERT( attachmentNames.count() == attachmentURLs.count() );
// Add all attachments by reading them from their temp. files
QStringList::ConstIterator itmime = attachmentMimetypes.begin();
QStringList::ConstIterator iturl = attachmentURLs.begin();
for( QStringList::ConstIterator itname = attachmentNames.begin();
itname != attachmentNames.end()
&& itmime != attachmentMimetypes.end()
&& iturl != attachmentURLs.end();
++itname, ++iturl, ++itmime ){
bool bymimetype = (*itmime).startsWith( "application/x-vnd.kolab." );
if( !updateAttachment( *msg, *iturl, *itname, *itmime, !bymimetype ) ){
kdWarning(5006) << "Attachment error, can not add Incidence." << endl;
bAttachOK = false;
break;
}
if ( bymimetype )
msg->setHeaderField( "X-Kolab-Type", *itmime );
}
if( bAttachOK ){
// Mark the message as read and store it in the folder
msg->cleanupHeader();
//debugBodyParts( "after cleanup", *msg );
msg->touch();
if ( folder.addMsg( msg ) == 0 )
// Message stored
sernum = msg->getMsgSerNum();
kdDebug(5006) << "addIncidenceKolab(): Message done and saved. Sernum: "
<< sernum << endl;
//debugBodyParts( "after addMsg", *msg );
} else
kdError(5006) << "addIncidenceKolab(): Message *NOT* saved!\n";
return sernum;
}
// The resource orders a deletion
bool KMailICalIfaceImpl::deleteIncidence( const QString& type,
const QString& folder,
const QString& uid )
{
if( !mUseResourceIMAP )
return false;
kdDebug(5006) << "KMailICalIfaceImpl::deleteIncidence( " << type << ", "
<< uid << " )" << endl;
// Find the folder
KMFolder* f = folderFromType( type, folder );
if( !f ) {
kdError(5006) << "deleteIncidence(" << type << "," << folder << ") : Not an IMAP resource folder" << endl;
return false;
}
if ( storageFormat( f ) != StorageIcalVcard ) {
kdError(5006) << "deleteIncidence(" << type << "," << folder << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return false;
}
bool rc = false;
bool quiet = mResourceQuiet;
mResourceQuiet = true;
KMMessage* msg = findMessageByUID( uid, f );
if( msg ) {
// Message found - delete it and return happy
deleteMsg( msg );
rc = true;
} else {
kdDebug(5006) << "Message not found, cannot remove id " << uid << endl;
}
mResourceQuiet = quiet;
return rc;
}
bool KMailICalIfaceImpl::deleteIncidenceKolab( const QString& resource,
Q_UINT32 sernum )
{
// Find the message from the serial number and delete it.
if( !mUseResourceIMAP )
return false;
kdDebug(5006) << "KMailICalIfaceImpl::deleteIncidenceKolab( "
<< resource << ", " << sernum << ")\n";
// Find the folder
KMFolder* f = findResourceFolder( resource );
if( !f ) {
kdError(5006) << "deleteIncidenceKolab(" << resource << ") : Not an IMAP resource folder" << endl;
return false;
}
if ( storageFormat( f ) != StorageXML ) {
kdError(5006) << "deleteIncidenceKolab(" << resource << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return false;
}
bool rc = false;
KMMessage* msg = findMessageBySerNum( sernum, f );
if( msg ) {
// Message found - delete it and return happy
deleteMsg( msg );
rc = true;
} else {
kdDebug(5006) << "Message not found, cannot remove serNum " << sernum << endl;
}
return rc;
}
// The resource asks for a full list of incidences
QStringList KMailICalIfaceImpl::incidences( const QString& type,
const QString& folder )
{
if( !mUseResourceIMAP )
return QStringList();
kdDebug(5006) << "KMailICalIfaceImpl::incidences( " << type << ", "
<< folder << " )" << endl;
QStringList ilist;
KMFolder* f = folderFromType( type, folder );
if( !f ) {
kdError(5006) << "incidences(" << type << "," << folder << ") : Not an IMAP resource folder" << endl;
return ilist;
}
if ( storageFormat( f ) != StorageIcalVcard ) {
kdError(5006) << "incidences(" << type << "," << folder << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return ilist;
}
f->open();
QString s;
for( int i=0; i<f->count(); ++i ) {
bool unget = !f->isMessage(i);
if( vPartFoundAndDecoded( f->getMsg( i ), s ) )
ilist << s;
if( unget ) f->unGetMsg(i);
}
return ilist;
}
QMap<Q_UINT32, QString> KMailICalIfaceImpl::incidencesKolab( const QString& mimetype,
const QString& resource )
{
/// Get the mimetype attachments from this folder. Returns a
/// QMap with serialNumber/attachment pairs.
/// (serial numbers of the mail are provided for easier later update)
QMap<Q_UINT32, QString> aMap;
if( !mUseResourceIMAP )
return aMap;
kdDebug(5006) << "KMailICalIfaceImpl::incidencesKolab( " << mimetype << ", "
<< resource << " )" << endl;
KMFolder* f = findResourceFolder( resource );
if( !f ) {
kdError(5006) << "incidencesKolab(" << resource << ") : Not an IMAP resource folder" << endl;
return aMap;
}
if ( storageFormat( f ) != StorageXML ) {
kdError(5006) << "incidencesKolab(" << resource << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return aMap;
}
f->open();
QString s;
for( int i=0; i<f->count(); ++i ) {
bool unget = !f->isMessage(i);
KMMessage* msg = f->getMsg( i );
if( msg ){
const int iSlash = mimetype.find('/');
const QCString sType = mimetype.left( iSlash ).latin1();
const QCString sSubtype = mimetype.mid( iSlash+1 ).latin1();
if ( sType.isEmpty() || sSubtype.isEmpty() ) {
kdError(5006) << mimetype << " not an type/subtype combination" << endl;
} else {
DwBodyPart* dwPart = findBodyPartByMimeType( *msg, sType, sSubtype );
if ( dwPart ) {
KMMessagePart msgPart;
KMMessage::bodyPart(dwPart, &msgPart);
aMap.insert(msg->getMsgSerNum(), msgPart.bodyToUnicode());
} else {
// This is *not* an error: it may be that not all of the messages
// have a message part that is matching the wanted MIME type
}
}
if( unget ) f->unGetMsg(i);
}
}
return aMap;
}
QStringList KMailICalIfaceImpl::subresources( const QString& type )
{
QStringList lst;
// Add the default one
KMFolder* f = folderFromType( type, QString::null );
if ( f && storageFormat( f ) == StorageIcalVcard )
lst << f->location();
// Add the extra folders
KMail::FolderContentsType t = folderContentsType( type );
QDictIterator<ExtraFolder> it( mExtraFolders );
for ( ; it.current(); ++it ) {
f = it.current()->folder;
if ( f && f->storage()->contentsType() == t
&& storageFormat( f ) == StorageIcalVcard )
lst << f->location();
}
return lst;
}
QValueList<KMailICalIfaceImpl::SubResource> KMailICalIfaceImpl::subresourcesKolab( const QString& contentsType )
{
QValueList<SubResource> subResources;
// Add the default one
KMFolder* f = folderFromType( contentsType, QString::null );
if ( f && storageFormat( f ) == StorageXML ) {
subResources.append( SubResource( f->location(), f->prettyURL(), !f->isReadOnly() ) );
kdDebug(5006) << "Adding(1) folder " << f->location() << " " <<
( f->isReadOnly() ? "readonly" : "" ) << endl;
}
// get the extra ones
const KMail::FolderContentsType t = folderContentsType( contentsType );
QDictIterator<ExtraFolder> it( mExtraFolders );
for ( ; it.current(); ++it ){
f = it.current()->folder;
if ( f && f->storage()->contentsType() == t
&& storageFormat( f ) == StorageXML ) {
subResources.append( SubResource( f->location(), f->prettyURL(), !f->isReadOnly() ) );
kdDebug(5006) << "Adding(2) folder " << f->location() << " " <<
( f->isReadOnly() ? "readonly" : "" ) << endl;
}
}
if ( subResources.isEmpty() )
kdDebug(5006) << "subresourcesKolab: No folder found for " << contentsType << endl;
return subResources;
}
bool KMailICalIfaceImpl::isWritableFolder( const QString& type,
const QString& resource )
{
KMFolder* f = folderFromType( type, resource );
if ( !f )
// Definitely not writable
return false;
return !f->isReadOnly();
}
bool KMailICalIfaceImpl::update( const QString& type, const QString& folder,
const QStringList& entries )
{
if( !mUseResourceIMAP )
return false;
if( entries.count() % 2 == 1 )
// Something is wrong - an odd amount of strings should not happen
return false;
QStringList::ConstIterator it = entries.begin();
while( true ) {
// Read them in pairs and call the single update method
QString uid, entry;
if( it == entries.end() )
break;
uid = *it;
++it;
if( it == entries.end() )
break;
entry = *it;
++it;
if( !update( type, folder, uid, entry ) )
// Some error happened
return false;
}
return true;
}
bool KMailICalIfaceImpl::update( const QString& type, const QString& folder,
const QString& uid, const QString& entry )
{
if( !mUseResourceIMAP )
return false;
kdDebug(5006) << "Update( " << type << ", " << folder << ", " << uid << ")\n";
// Find the folder and the incidence in it
KMFolder* f = folderFromType( type, folder );
if( !f ) {
kdError(5006) << "update(" << type << "," << folder << ") : Not an IMAP resource folder" << endl;
return false;
}
if ( storageFormat( f ) != StorageIcalVcard ) {
kdError(5006) << "update(" << type << "," << folder << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return false;
}
bool rc = true;
bool quiet = mResourceQuiet;
mResourceQuiet = true;
KMMessage* msg = findMessageByUID( uid, f );
if( msg ) {
// Message found - update it
deleteMsg( msg );
addIncidence( type, folder, uid, entry );
rc = true;
} else {
kdDebug(5006) << type << " not found, cannot update uid " << uid << endl;
// Since it doesn't seem to be there, save it instead
addIncidence( type, folder, uid, entry );
}
mResourceQuiet = quiet;
return rc;
}
Q_UINT32 KMailICalIfaceImpl::update( const QString& resource,
Q_UINT32 sernum,
const QString& subject,
const QStringList& attachmentURLs,
const QStringList& attachmentMimetypes,
const QStringList& attachmentNames,
const QStringList& deletedAttachments )
{
Q_UINT32 rc = 0;
// This finds the message with serial number "sernum", sets the
// xml attachments to hold the contents of "xml", and updates all
// attachments.
// The mail can have additional attachments, and these are not
// touched! They belong to other clients - like Outlook
// So we delete all the attachments listed in the
// "deletedAttachments" arg, and then update/add all the attachments
// given by the urllist attachments.
// If the mail does not already exist, id will not be a valid serial
// number, and the mail is just added instead. In this case
// the deletedAttachments can be forgotten.
if( !mUseResourceIMAP )
return rc;
Q_ASSERT( !resource.isEmpty() );
kdDebug(5006) << "KMailICalIfaceImpl::update( " << resource << ", " << sernum << " )\n";
kdDebug(5006) << attachmentURLs << "\n";
kdDebug(5006) << attachmentMimetypes << "\n";
kdDebug(5006) << attachmentNames << "\n";
kdDebug(5006) << "deleted attachments:" << deletedAttachments << "\n";
// Find the folder
KMFolder* f = findResourceFolder( resource );
if( !f ) {
kdError(5006) << "update(" << resource << ") : Not an IMAP resource folder" << endl;
return rc;
}
if ( storageFormat( f ) != StorageXML ) {
kdError(5006) << "update(" << resource << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return rc;
}
f->open();
kdDebug(5006) << "Updating in folder " << f << " " << f->location() << endl;
KMMessage* msg = 0;
if ( sernum != 0 )
msg = findMessageBySerNum( sernum, f );
if ( msg ) {
// Message found - make a copy and update it:
KMMessage* newMsg = new KMMessage( *msg );
newMsg->setSubject( subject );
newMsg->setParent( 0 ); // workaround strange line in KMMsgBase::assign. newMsg is not in any folder yet.
// Delete some attachments according to list
for( QStringList::ConstIterator it = deletedAttachments.begin();
it != deletedAttachments.end();
++it ){
if( !deleteAttachment( *newMsg, *it ) ){
// Note: It is _not_ an error if an attachment was already deleted.
}
}
// Add all attachments by reading them from their temp. files
QStringList::ConstIterator iturl = attachmentURLs.begin();
QStringList::ConstIterator itmime = attachmentMimetypes.begin();
QStringList::ConstIterator itname = attachmentNames.begin();
for( ;
iturl != attachmentURLs.end()
&& itmime != attachmentMimetypes.end()
&& itname != attachmentNames.end();
++iturl, ++itname, ++itmime ){
bool bymimetype = (*itname).startsWith( "application/x-vnd.kolab." );
if( !updateAttachment( *newMsg, *iturl, *itname, *itmime, bymimetype ) ){
kdDebug(5006) << "Attachment error, can not update attachment " << *iturl << endl;
break;
}
}
//debugBodyParts( "in update, before cleanup", *newMsg );
// This is necessary for the headers to be readable later on
newMsg->cleanupHeader();
//debugBodyParts( "in update, after cleanup", *newMsg );
deleteMsg( msg );
if ( f->addMsg( newMsg ) == 0 ) {
// Message stored
rc = newMsg->getMsgSerNum();
kdDebug(5006) << "forget about " << sernum << ", it's " << rc << " now" << endl;
}
} else {
// Message not found - store it newly
rc = addIncidenceKolab( *f, subject,
attachmentURLs,
attachmentNames,
attachmentMimetypes );
}
f->close();
addFolderChange( f, Contents );
return rc;
}
KURL KMailICalIfaceImpl::getAttachment( const QString& resource,
Q_UINT32 sernum,
const QString& filename )
{
// This finds the attachment with the filename, saves it to a
// temp file and returns a URL to it. It's up to the resource
// to delete the tmp file later.
if( !mUseResourceIMAP )
return KURL();
kdDebug(5006) << "KMailICalIfaceImpl::getAttachment( "
<< resource << ", " << sernum << ", " << filename << " )\n";
// Find the folder
KMFolder* f = findResourceFolder( resource );
if( !f ) {
kdError(5006) << "getAttachment(" << resource << ") : Not an IMAP resource folder" << endl;
return KURL();
}
if ( storageFormat( f ) != StorageXML ) {
kdError(5006) << "getAttachment(" << resource << ") : Folder has wrong storage format " << storageFormat( f ) << endl;
return KURL();
}
KURL url;
bool bOK = false;
bool quiet = mResourceQuiet;
mResourceQuiet = true;
KMMessage* msg = findMessageBySerNum( sernum, f );
if( msg ) {
// Message found - look for the attachment:
DwBodyPart* part = findBodyPart( *msg, filename );
if ( part ) {
// Save the contents of the attachment.
KMMessagePart aPart;
msg->bodyPart( part, &aPart );
QByteArray rawData( aPart.bodyDecodedBinary() );
KTempFile file;
file.file()->writeBlock( rawData.data(), rawData.size() );
url.setPath( file.name() );
bOK = true;
}
if( !bOK ){
kdDebug(5006) << "Attachment " << filename << " not found." << endl;
}
}else{
kdDebug(5006) << "Message not found." << endl;
}
mResourceQuiet = quiet;
return url;
}
void KMailICalIfaceImpl::slotFolderRemoved( KMFolder* folder )
{
// pretend the folder just changed back to the mail type, which
// does the right thing, namely remove resource
folderContentsTypeChanged( folder, KMail::ContentsTypeMail );
}
// KMail added a file to one of the groupware folders
void KMailICalIfaceImpl::slotIncidenceAdded( KMFolder* folder,
Q_UINT32 sernum )
{
if( mResourceQuiet || !mUseResourceIMAP )
return;
QString type = folderContentsType( folder->storage()->contentsType() );
if( !type.isEmpty() ) {
// Get the index of the mail
int i = 0;
KMFolder* aFolder = 0;
kmkernel->msgDict()->getLocation( sernum, &aFolder, &i );
assert( folder == aFolder );
bool unget = !folder->isMessage( i );
QString s;
bool ok = false;
KMMessage* msg = folder->getMsg( i );
StorageFormat format = storageFormat( folder );
switch( format ) {
case StorageIcalVcard:
// Read the iCal or vCard
ok = vPartFoundAndDecoded( msg, s );
break;
case StorageXML:
// Read the XML from the attachment with the given mimetype
ok = kolabXMLFoundAndDecoded( *msg, folderKolabMimeType( folder->storage()->contentsType() ), s );
break;
}
if ( ok ) {
kdDebug(5006) << "Emitting DCOP signal incidenceAdded( " << type
<< ", " << folder->location() << ", " << s << " )" << endl;
incidenceAdded( type, folder->location(), s );
incidenceAdded( type, folder->location(), sernum, format, s );
}
if( unget ) folder->unGetMsg(i);
} else
kdError(5006) << "Not an IMAP resource folder" << endl;
}
// KMail deleted a file
void KMailICalIfaceImpl::slotIncidenceDeleted( KMFolder* folder,
Q_UINT32 sernum )
{
if( mResourceQuiet || !mUseResourceIMAP )
return;
QString type = folderContentsType( folder->storage()->contentsType() );
kdDebug(5006) << folder << " " << type << " " << sernum << endl;
if( !type.isEmpty() ) {
// Get the index of the mail
int i = 0;
KMFolder* aFolder = 0;
kmkernel->msgDict()->getLocation( sernum, &aFolder, &i );
assert( folder == aFolder );
// Read the iCal or vCard
bool unget = !folder->isMessage( i );
QString s;
bool ok = false;
KMMessage* msg = folder->getMsg( i );
QString uid( "UID" );
switch( storageFormat( folder ) ) {
case StorageIcalVcard:
if( vPartFoundAndDecoded( msg, s ) ) {
vPartMicroParser( s, uid );
ok = true;
}
break;
case StorageXML:
if ( kolabXMLFoundAndDecoded( *msg, folderKolabMimeType( folder->storage()->contentsType() ), s ) ) {
QDomDocument doc;
if ( doc.setContent( s ) ) {
QDomElement top = doc.documentElement();
for ( QDomNode n = top.firstChild(); !n.isNull(); n = n.nextSibling() ) {
QDomElement e = n.toElement();
if ( e.tagName() == "uid" ) {
uid = e.text();
ok = true;
break;
}
}
}
}
break;
}
if ( ok ) {
kdDebug(5006) << "Emitting DCOP signal incidenceDeleted( "
<< type << ", " << folder->location() << ", " << uid
<< " )" << endl;
incidenceDeleted( type, folder->location(), uid );
}
if( unget ) folder->unGetMsg(i);
} else
kdError(5006) << "Not a groupware folder" << endl;
}
// KMail orders a refresh
void KMailICalIfaceImpl::slotRefresh( const QString& type )
{
if( mUseResourceIMAP ) {
signalRefresh( type, QString::null /* PENDING(bo) folder->location() */ );
kdDebug(5006) << "Emitting DCOP signal signalRefresh( " << type << " )" << endl;
}
}
// This is among other things called when an expunge of a folder happens
void KMailICalIfaceImpl::slotRefreshFolder( KMFolder* folder)
{
// TODO: The resources would of course be better off, if only this
// folder would need refreshing. Currently it just orders a reload of
// the type of the folder
if( mUseResourceIMAP && folder ) {
if( folder == mCalendar || folder == mContacts
|| folder == mNotes || folder == mTasks
|| folder == mJournals || mExtraFolders.find( folder->location() ) ) {
// Refresh the folder of this type
KMail::FolderContentsType ct = folder->storage()->contentsType();
slotRefresh( s_folderContentsType[ct].contentsTypeStr );
}
}
}
/****************************
* The folder and message stuff code
*/
KMFolder* KMailICalIfaceImpl::folderFromType( const QString& type,
const QString& folder )
{
if( mUseResourceIMAP ) {
KMFolder* f = 0;
if ( !folder.isEmpty() ) {
f = extraFolder( type, folder );
if ( f )
return f;
}
if( type == "Calendar" ) f = mCalendar;
else if( type == "Contact" ) f = mContacts;
else if( type == "Note" ) f = mNotes;
else if( type == "Task" || type == "Todo" ) f = mTasks;
else if( type == "Journal" ) f = mJournals;
if ( f && ( folder.isEmpty() || folder == f->location() ) )
return f;
kdError(5006) << "No folder ( " << type << ", " << folder << " )\n";
}
return 0;
}
// Returns true if folder is a resource folder. If the resource isn't enabled
// this always returns false
bool KMailICalIfaceImpl::isResourceImapFolder( KMFolder* folder ) const
{
return mUseResourceIMAP && folder &&
( isStandardResourceFolder( folder ) || mExtraFolders.find( folder->location() )!=0 );
}
bool KMailICalIfaceImpl::isStandardResourceFolder( KMFolder* folder ) const
{
return ( folder == mCalendar || folder == mTasks || folder == mJournals ||
folder == mNotes || folder == mContacts );
}
bool KMailICalIfaceImpl::hideResourceImapFolder( KMFolder* folder ) const
{
return mHideFolders && isResourceImapFolder( folder );
}
KFolderTreeItem::Type KMailICalIfaceImpl::folderType( KMFolder* folder ) const
{
if( mUseResourceIMAP && folder ) {
if( folder == mCalendar || folder == mContacts
|| folder == mNotes || folder == mTasks
|| folder == mJournals || mExtraFolders.find( folder->location() ) ) {
KMail::FolderContentsType ct = folder->storage()->contentsType();
return s_folderContentsType[ct].treeItemType;
}
}
return KFolderTreeItem::Other;
}
// Global tables of foldernames is different languages
// For now: 0->English, 1->German, 2->French, 3->Dutch
static QMap<KFolderTreeItem::Type,QString> folderNames[4];
QString KMailICalIfaceImpl::folderName( KFolderTreeItem::Type type, int language ) const
{
// With the XML storage, folders are always (internally) named in English
if ( GlobalSettings::theIMAPResourceStorageFormat() == GlobalSettings::EnumTheIMAPResourceStorageFormat::XML )
language = 0;
static bool folderNamesSet = false;
if( !folderNamesSet ) {
folderNamesSet = true;
/* NOTE: If you add something here, you also need to update
GroupwarePage in configuredialog.cpp */
// English
folderNames[0][KFolderTreeItem::Calendar] = QString::fromLatin1("Calendar");
folderNames[0][KFolderTreeItem::Tasks] = QString::fromLatin1("Tasks");
folderNames[0][KFolderTreeItem::Journals] = QString::fromLatin1("Journal");
folderNames[0][KFolderTreeItem::Contacts] = QString::fromLatin1("Contacts");
folderNames[0][KFolderTreeItem::Notes] = QString::fromLatin1("Notes");
// German
folderNames[1][KFolderTreeItem::Calendar] = QString::fromLatin1("Kalender");
folderNames[1][KFolderTreeItem::Tasks] = QString::fromLatin1("Aufgaben");
folderNames[1][KFolderTreeItem::Journals] = QString::fromLatin1("Journal");
folderNames[1][KFolderTreeItem::Contacts] = QString::fromLatin1("Kontakte");
folderNames[1][KFolderTreeItem::Notes] = QString::fromLatin1("Notizen");
// French
folderNames[2][KFolderTreeItem::Calendar] = QString::fromLatin1("Calendrier");
folderNames[2][KFolderTreeItem::Tasks] = QString::fromLatin1("T<EFBFBD>ches");
folderNames[2][KFolderTreeItem::Journals] = QString::fromLatin1("Journal");
folderNames[2][KFolderTreeItem::Contacts] = QString::fromLatin1("Contacts");
folderNames[2][KFolderTreeItem::Notes] = QString::fromLatin1("Notes");
// Dutch
folderNames[3][KFolderTreeItem::Calendar] = QString::fromLatin1("Agenda");
folderNames[3][KFolderTreeItem::Tasks] = QString::fromLatin1("Taken");
folderNames[3][KFolderTreeItem::Journals] = QString::fromLatin1("Logboek");
folderNames[3][KFolderTreeItem::Contacts] = QString::fromLatin1("Contactpersonen");
folderNames[3][KFolderTreeItem::Notes] = QString::fromLatin1("Notities");
}
if( language < 0 || language > 3 ) {
return folderNames[mFolderLanguage][type];
}
else {
return folderNames[language][type];
}
}
// Find message matching a given UID
KMMessage *KMailICalIfaceImpl::findMessageByUID( const QString& uid, KMFolder* folder )
{
if( !folder ) return 0;
for( int i=0; i<folder->count(); ++i ) {
bool unget = !folder->isMessage(i);
KMMessage* msg = folder->getMsg( i );
if( msg ) {
QString vCal;
if( vPartFoundAndDecoded( msg, vCal ) ) {
QString msgUid( "UID" );
vPartMicroParser( vCal, msgUid );
if( msgUid == uid )
return msg;
}
}
if( unget ) folder->unGetMsg(i);
}
return 0;
}
// Find message matching a given serial number
KMMessage *KMailICalIfaceImpl::findMessageBySerNum( Q_UINT32 serNum, KMFolder* folder )
{
if( !folder ) return 0;
KMMessage *message = 0;
KMFolder* aFolder = 0;
int index;
kmkernel->msgDict()->getLocation( serNum, &aFolder, &index );
if( aFolder && aFolder != folder ){
kdWarning(5006) << "findMessageBySerNum( " << serNum << " ) found it in folder " << aFolder->location() << ", expected " << folder->location() << endl;
}else{
if( aFolder )
message = aFolder->getMsg( index );
if (!message)
kdWarning(5006) << "findMessageBySerNum( " << serNum << " ) invalid serial number\n" << endl;
}
return message;
}
void KMailICalIfaceImpl::deleteMsg( KMMessage *msg )
{
if( !msg ) return;
// Commands are now delayed; can't use that anymore, we need immediate deletion
//( new KMDeleteMsgCommand( msg->parent(), msg ) )->start();
KMFolder *srcFolder = msg->parent();
int idx = srcFolder->find(msg);
assert(idx != -1);
srcFolder->removeMsg(idx);
delete msg;
addFolderChange( srcFolder, Contents );
}
void KMailICalIfaceImpl::folderContentsTypeChanged( KMFolder* folder,
KMail::FolderContentsType contentsType )
{
if ( !mUseResourceIMAP )
return;
kdDebug(5006) << "folderContentsTypeChanged( " << folder->name()
<< ", " << contentsType << ")\n";
// The builtins can't change type
if ( isStandardResourceFolder( folder ) )
return;
// Check if already know that 'extra folder'
const QString location = folder->location();
ExtraFolder* ef = mExtraFolders.find( location );
if ( ef && ef->folder ) {
// Notify that the old folder resource is no longer available
subresourceDeleted(folderContentsType( folder->storage()->contentsType() ), location );
if ( contentsType == 0 ) {
// Delete the old entry, stop listening and stop here
mExtraFolders.remove( location );
folder->disconnect( this );
return;
}
// So the type changed to another groupware type, ok.
} else {
if ( ef && !ef->folder ) // deleted folder, clean up
mExtraFolders.remove( location );
if ( contentsType == 0 )
return;
kdDebug(5006) << "registering " << location << " as extra folder" << endl;
// Make a new entry for the list
ef = new ExtraFolder( folder );
mExtraFolders.insert( location, ef );
StorageFormat format= GlobalSettings::theIMAPResourceStorageFormat()
== GlobalSettings::EnumTheIMAPResourceStorageFormat::XML ? StorageXML : StorageIcalVcard;
FolderInfo info( format, NoChange );
mFolderInfoMap.insert( folder, info );
// avoid multiple connections
disconnect( folder, SIGNAL( msgAdded( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceAdded( KMFolder*, Q_UINT32 ) ) );
disconnect( folder, SIGNAL( msgRemoved( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceDeleted( KMFolder*, Q_UINT32 ) ) );
disconnect( folder, SIGNAL( expunged( KMFolder* ) ),
this, SLOT( slotRefreshFolder( KMFolder* ) ) );
// And listen to changes from it
connect( folder, SIGNAL( msgAdded( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceAdded( KMFolder*, Q_UINT32 ) ) );
connect( folder, SIGNAL( msgRemoved( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceDeleted( KMFolder*, Q_UINT32 ) ) );
connect( folder, SIGNAL( expunged( KMFolder* ) ),
this, SLOT( slotRefreshFolder( KMFolder* ) ) );
}
// Tell about the new resource
/* FIXME merge once we are back in HEAD. IMAP Resource still uses the other one. */
subresourceAdded( folderContentsType( contentsType ), location, folder->prettyURL() );
subresourceAdded( folderContentsType( contentsType ), location );
}
KMFolder* KMailICalIfaceImpl::extraFolder( const QString& type,
const QString& folder )
{
// If an extra folder exists that matches the type and folder location,
// use that
int t = folderContentsType( type );
if ( t < 1 || t > 5 )
return 0;
ExtraFolder* ef = mExtraFolders.find( folder );
if ( ef && ef->folder && ef->folder->storage()->contentsType() == t )
return ef->folder;
return 0;
}
KMailICalIfaceImpl::StorageFormat KMailICalIfaceImpl::storageFormat( KMFolder* folder ) const
{
FolderInfoMap::ConstIterator it = mFolderInfoMap.find( folder );
if ( it != mFolderInfoMap.end() )
return (*it).mStorageFormat;
return GlobalSettings::theIMAPResourceStorageFormat() == GlobalSettings::EnumTheIMAPResourceStorageFormat::XML ? StorageXML : StorageIcalVcard;
}
void KMailICalIfaceImpl::setStorageFormat( KMFolder* folder, StorageFormat format )
{
FolderInfoMap::Iterator it = mFolderInfoMap.find( folder );
if ( it != mFolderInfoMap.end() ) {
(*it).mStorageFormat = format;
} else {
FolderInfo info( format, NoChange );
mFolderInfoMap.insert( folder, info );
}
KConfigGroup configGroup( kmkernel->config(), "GroupwareFolderInfo" );
configGroup.writeEntry( folder->idString() + "-storageFormat",
format == StorageXML ? "xml" : "icalvcard" );
}
void KMailICalIfaceImpl::addFolderChange( KMFolder* folder, FolderChanges changes )
{
FolderInfoMap::Iterator it = mFolderInfoMap.find( folder );
if ( it != mFolderInfoMap.end() ) {
(*it).mChanges = static_cast<FolderChanges>( (*it).mChanges | changes );
} else { // Otherwise, well, it's a folder we don't care about.
kdDebug(5006) << "addFolderChange: nothing known about folder " << folder->location() << endl;
}
KConfigGroup configGroup( kmkernel->config(), "GroupwareFolderInfo" );
configGroup.writeEntry( folder->idString() + "-changes", (*it).mChanges );
}
void KMailICalIfaceImpl::folderSynced( KMFolder* folder, const KURL& folderURL )
{
FolderInfoMap::Iterator it = mFolderInfoMap.find( folder );
if ( it != mFolderInfoMap.end() && (*it).mChanges ) {
handleFolderSynced( folder, folderURL, (*it).mChanges );
(*it).mChanges = NoChange;
}
}
void KMailICalIfaceImpl::handleFolderSynced( KMFolder* folder,
const KURL& folderURL,
int _changes )
{
// This is done here instead of in the resource, because
// there could be 0, 1, or N kolab resources at this point.
// We can hack the N case, but not the 0 case.
// So the idea of a DCOP signal for this wouldn't work.
if ( ( _changes & KMailICalIface::Contents ) ||
( _changes & KMailICalIface::ACL ) ) {
if ( storageFormat( folder ) == StorageXML && folder->storage()->contentsType() == KMail::ContentsTypeCalendar )
triggerKolabFreeBusy( folderURL );
}
}
void KMailICalIfaceImpl::triggerKolabFreeBusy( const KURL& folderURL )
{
/* Steffen said: you must issue an authenticated HTTP GET request to
https://kolabserver/freebusy/trigger/user@domain/Folder/NestedFolder.pfb
(replace .pfb with .xpfb for extended fb lists). */
KURL httpURL( folderURL );
// Keep username ("user@domain"), pass, and host from the imap url
httpURL.setProtocol( "https" );
httpURL.setPort( 0 ); // remove imap port
// IMAP path is either /INBOX/<path> or /user/someone/<path>
QString path = folderURL.path( -1 );
Q_ASSERT( path.startsWith( "/" ) );
int secondSlash = path.find( '/', 1 );
if ( secondSlash == -1 ) {
kdWarning() << "KCal::ResourceKolab::fromKMailFolderSynced path is too short: " << path << endl;
return;
}
if ( path.startsWith( "/INBOX/", false ) ) {
// If INBOX, replace it with the username (which is user@domain)
path = path.mid( secondSlash );
path.prepend( folderURL.user() );
} else {
// If user, just remove it. So we keep the IMAP-returned username.
// This assumes it's a known user on the same domain.
path = path.mid( secondSlash );
}
httpURL.setPath( "/freebusy/trigger/" + path + ".pfb" );
httpURL.setQuery( QString::null );
kdDebug() << "Triggering PFB update for " << folderURL << " : getting " << httpURL << endl;
// "Fire and forget". No need for error handling, nor for explicit deletion.
// Maybe we should try to prevent launching it if it's already running (for this URL) though.
/*KIO::Job* job =*/ KIO::get( httpURL, false, false /*no progress info*/ );
}
KMFolder* KMailICalIfaceImpl::findResourceFolder( const QString& resource )
{
// Try the standard folders
if( mCalendar && mCalendar->location() == resource )
return mCalendar;
if ( mContacts && mContacts->location() == resource )
return mContacts;
if ( mNotes && mNotes->location() == resource )
return mNotes;
if ( mTasks && mTasks->location() == resource )
return mTasks;
if ( mJournals && mJournals->location() == resource )
return mJournals;
// No luck. Try the extrafolders
ExtraFolder* ef = mExtraFolders.find( resource );
if ( ef )
return ef->folder;
// No luck at all
return 0;
}
/****************************
* The config stuff
*/
void KMailICalIfaceImpl::readConfig()
{
bool enabled = GlobalSettings::theIMAPResourceEnabled();
if( !enabled ) {
if( mUseResourceIMAP == true ) {
// Shutting down
mUseResourceIMAP = false;
cleanup();
reloadFolderTree();
}
return;
}
// Read remaining options
const bool hideFolders = GlobalSettings::hideGroupwareFolders();
QString parentName = GlobalSettings::theIMAPResourceFolderParent();
// Find the folder parent
KMFolderDir* folderParentDir;
KMFolderType folderType;
KMFolder* folderParent = kmkernel->findFolderById( parentName );
if( folderParent == 0 ) {
// Parent folder not found. It was probably deleted. The user will have to
// configure things again.
kdDebug(5006) << "Groupware folder " << parentName << " not found. Groupware functionality disabled" << endl;
// Or maybe the inbox simply wasn't created on the first startup
KMAccount* account = kmkernel->acctMgr()->find( GlobalSettings::theIMAPResourceAccount() );
Q_ASSERT( account );
if ( account ) {
// just in case we were connected already
disconnect( account, SIGNAL( finishedCheck( bool, CheckStatus ) ),
this, SLOT( slotCheckDone() ) );
connect( account, SIGNAL( finishedCheck( bool, CheckStatus ) ),
this, SLOT( slotCheckDone() ) );
}
mUseResourceIMAP = false;
// We can't really call cleanup(), if those folders were completely deleted.
mCalendar = 0;
mTasks = 0;
mJournals = 0;
mContacts = 0;
mNotes = 0;
return;
} else {
folderParentDir = folderParent->createChildFolder();
folderType = folderParent->folderType();
}
// Make sure the folder parent has the subdirs
bool makeSubFolders = false;
KMFolder* folder;
folder = findStandardResourceFolder( folderParentDir, KMail::ContentsTypeCalendar );
if( !folder ) {
kdDebug(5006) << "Calendar folder not found in " << parentName << endl;
makeSubFolders = true;
mCalendar = 0;
}
folder = findStandardResourceFolder( folderParentDir, KMail::ContentsTypeTask );
if( !folder ) {
kdDebug(5006) << "Tasks folder not found in " << parentName << endl;
makeSubFolders = true;
mTasks = 0;
}
folder = findStandardResourceFolder( folderParentDir, KMail::ContentsTypeJournal );
if( !folder ) {
kdDebug(5006) << "Journals folder not found in " << parentName << endl;
makeSubFolders = true;
mJournals = 0;
}
folder = findStandardResourceFolder( folderParentDir, KMail::ContentsTypeContact );
if( !folder ) {
kdDebug(5006) << "Contacts folder not found in " << parentName << endl;
makeSubFolders = true;
mContacts = 0;
}
folder = findStandardResourceFolder( folderParentDir, KMail::ContentsTypeNote );
if( !folder ) {
kdDebug(5006) << "Notes folder not found in " << parentName << endl;
makeSubFolders = true;
mNotes = 0;
}
if( makeSubFolders ) {
// Not all subfolders were there, so ask if we can make them
if( KMessageBox::questionYesNo( 0, i18n("KMail will now create the required folders for the IMAP resource"
" as subfolders of %1; if you do not want this, press \"No\","
" and the IMAP resource will be disabled").arg(folderParent!=0?folderParent->name():folderParentDir->name()),
i18n("IMAP Resource Folders") ) == KMessageBox::No ) {
GlobalSettings::setTheIMAPResourceEnabled( false );
mUseResourceIMAP = false;
mFolderParentDir = 0;
reloadFolderTree();
return;
}
}
// Check if something changed
if( mUseResourceIMAP && !makeSubFolders && mFolderParentDir == folderParentDir
&& mFolderType == folderType ) {
// Nothing changed
if ( hideFolders != mHideFolders ) {
// Well, the folder hiding has changed
mHideFolders = hideFolders;
reloadFolderTree();
}
return;
}
// Make the new settings work
mUseResourceIMAP = true;
mFolderLanguage = GlobalSettings::theIMAPResourceFolderLanguage();
if( mFolderLanguage > 3 ) mFolderLanguage = 0;
mFolderParentDir = folderParentDir;
mFolderType = folderType;
mHideFolders = hideFolders;
// Close the previous folders
cleanup();
// Set the new folders
mCalendar = initFolder( "GCa", KMail::ContentsTypeCalendar );
mTasks = initFolder( "GTa", KMail::ContentsTypeTask );
mJournals = initFolder( "GTa", KMail::ContentsTypeJournal );
mContacts = initFolder( "GCo", KMail::ContentsTypeContact );
mNotes = initFolder( "GNo", KMail::ContentsTypeNote );
mCalendar->setLabel( i18n( "Calendar" ) );
mTasks->setLabel( i18n( "Tasks" ) );
mJournals->setLabel( i18n( "Journal" ) );
mContacts->setLabel( i18n( "Contacts" ) );
mNotes->setLabel( i18n( "Notes" ) );
// Store final annotation (with .default) so that we won't ask again on next startup
if ( mCalendar->folderType() == KMFolderTypeCachedImap )
static_cast<KMFolderCachedImap *>( mCalendar->storage() )->updateAnnotationFolderType();
if ( mTasks->folderType() == KMFolderTypeCachedImap )
static_cast<KMFolderCachedImap *>( mTasks->storage() )->updateAnnotationFolderType();
if ( mJournals->folderType() == KMFolderTypeCachedImap )
static_cast<KMFolderCachedImap *>( mJournals->storage() )->updateAnnotationFolderType();
if ( mContacts->folderType() == KMFolderTypeCachedImap )
static_cast<KMFolderCachedImap *>( mContacts->storage() )->updateAnnotationFolderType();
if ( mNotes->folderType() == KMFolderTypeCachedImap )
static_cast<KMFolderCachedImap *>( mNotes->storage() )->updateAnnotationFolderType();
kdDebug(5006) << k_funcinfo << "mCalendar=" << mCalendar << " " << mCalendar->location() << endl;
kdDebug(5006) << k_funcinfo << "mNotes=" << mNotes << " " << mNotes->location() << endl;
// Find all extra folders
QStringList folderNames;
QValueList<QGuardedPtr<KMFolder> > folderList;
kmkernel->dimapFolderMgr()->createFolderList(&folderNames, &folderList);
for(QValueList<QGuardedPtr<KMFolder> >::iterator it = folderList.begin();
it != folderList.end(); ++it)
{
FolderStorage* storage = (*it)->storage();
if ( storage->contentsType() != 0 ) {
folderContentsTypeChanged( *it, storage->contentsType() );
}
}
// If we just created them, they might have been registered as extra folders temporarily.
// -> undo that.
mExtraFolders.remove( mCalendar->location() );
mExtraFolders.remove( mTasks->location() );
mExtraFolders.remove( mJournals->location() );
mExtraFolders.remove( mContacts->location() );
mExtraFolders.remove( mNotes->location() );
// Make KOrganizer re-read everything
#if 0 // old way, not enough finegrained (and most resources don't call doOpen, so they miss the subresources anyway)
slotRefresh( "Calendar" );
slotRefresh( "Task" );
slotRefresh( "Journal" );
slotRefresh( "Contact" );
slotRefresh( "Notes" );
#else
subresourceAdded( folderContentsType( KMail::ContentsTypeCalendar ), mCalendar->location(), mCalendar->label() );
subresourceAdded( folderContentsType( KMail::ContentsTypeCalendar ), mCalendar->location() );
subresourceAdded( folderContentsType( KMail::ContentsTypeTask ), mTasks->location(), mTasks->label() );
subresourceAdded( folderContentsType( KMail::ContentsTypeTask ), mTasks->location() );
subresourceAdded( folderContentsType( KMail::ContentsTypeJournal ), mJournals->location(), mJournals->label() );
subresourceAdded( folderContentsType( KMail::ContentsTypeJournal ), mJournals->location() );
subresourceAdded( folderContentsType( KMail::ContentsTypeContact ), mContacts->location(), mContacts->label() );
subresourceAdded( folderContentsType( KMail::ContentsTypeContact ), mContacts->location() );
subresourceAdded( folderContentsType( KMail::ContentsTypeNote ), mNotes->location(), mNotes->label() );
subresourceAdded( folderContentsType( KMail::ContentsTypeNote ), mNotes->location() );
// This also shows that we might even get rid of the mCalendar etc. special
// case and just use ExtraFolder for all
#endif
reloadFolderTree();
}
void KMailICalIfaceImpl::slotCheckDone()
{
QString parentName = GlobalSettings::theIMAPResourceFolderParent();
KMFolder* folderParent = kmkernel->findFolderById( parentName );
kdDebug(5006) << k_funcinfo << " folderParent=" << folderParent << endl;
if ( folderParent ) // cool it exists now
{
KMAccount* account = kmkernel->acctMgr()->find( GlobalSettings::theIMAPResourceAccount() );
if ( account )
disconnect( account, SIGNAL( finishedCheck( bool, CheckStatus ) ),
this, SLOT( slotCheckDone() ) );
readConfig();
}
}
KMFolder* KMailICalIfaceImpl::initFolder( const char* typeString,
KMail::FolderContentsType contentsType )
{
// Figure out what type of folder this is supposed to be
KMFolderType type = mFolderType;
if( type == KMFolderTypeUnknown ) type = KMFolderTypeMaildir;
KFolderTreeItem::Type itemType = s_folderContentsType[contentsType].treeItemType;
//kdDebug(5006) << "KMailICalIfaceImpl::initFolder " << folderName( itemType ) << endl;
// Find the folder
KMFolder* folder = findStandardResourceFolder( mFolderParentDir, contentsType );
if( !folder && GlobalSettings::theIMAPResourceStorageFormat() == GlobalSettings::EnumTheIMAPResourceStorageFormat::XML ) {
// Maybe there's a folder with the right name - well, change its type then
KMFolderNode* node = mFolderParentDir->hasNamedFolder( folderName( itemType ) );
if ( node && !node->isDir() ) {
folder = static_cast<KMFolder *>( node );
folder->storage()->setContentsType( contentsType );
kdDebug(5006) << "Adjusted type of " << folder->location() << " to contentsType " << contentsType << endl;
}
}
if ( !folder ) {
// The folder isn't there yet - create it
folder =
mFolderParentDir->createFolder( folderName( itemType ), false, type );
if( mFolderType == KMFolderTypeImap )
static_cast<KMFolderImap*>( folder->storage() )->
createFolder( folderName( itemType ) );
// Groupware folder created, use the global setting for storage format
setStorageFormat( folder, GlobalSettings::theIMAPResourceStorageFormat() == GlobalSettings::EnumTheIMAPResourceStorageFormat::XML ? StorageXML : StorageIcalVcard );
} else {
KConfigGroup configGroup( kmkernel->config(), "GroupwareFolderInfo" );
QString str = configGroup.readEntry( folder->idString() + "-storageFormat", "icalvcard" );
FolderInfo info;
info.mStorageFormat = ( str == "xml" ) ? StorageXML : StorageIcalVcard;
info.mChanges = (FolderChanges) configGroup.readNumEntry( folder->idString() + "-changes" );
mFolderInfoMap.insert( folder, info );
//kdDebug(5006) << "Found existing folder type " << itemType << " : " << folder->location() << endl;
}
if( folder->canAccess() != 0 ) {
KMessageBox::sorry(0, i18n("You do not have read/write permission to your %1 folder.")
.arg( folderName( itemType ) ) );
return 0;
}
folder->setType( typeString );
folder->storage()->setContentsType( contentsType );
folder->setSystemFolder( true );
folder->open();
// avoid multiple connections
disconnect( folder, SIGNAL( msgAdded( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceAdded( KMFolder*, Q_UINT32 ) ) );
disconnect( folder, SIGNAL( msgRemoved( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceDeleted( KMFolder*, Q_UINT32 ) ) );
disconnect( folder, SIGNAL( expunged( KMFolder* ) ),
this, SLOT( slotRefreshFolder( KMFolder* ) ) );
// Setup the signals to listen for changes
connect( folder, SIGNAL( msgAdded( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceAdded( KMFolder*, Q_UINT32 ) ) );
connect( folder, SIGNAL( msgRemoved( KMFolder*, Q_UINT32 ) ),
this, SLOT( slotIncidenceDeleted( KMFolder*, Q_UINT32 ) ) );
connect( folder, SIGNAL( expunged( KMFolder* ) ),
this, SLOT( slotRefreshFolder( KMFolder* ) ) );
return folder;
}
static void cleanupFolder( KMFolder* folder, KMailICalIfaceImpl* _this )
{
if( folder ) {
folder->setType( "plain" );
folder->setSystemFolder( false );
folder->disconnect( _this );
folder->close();
}
}
void KMailICalIfaceImpl::cleanup()
{
cleanupFolder( mContacts, this );
cleanupFolder( mCalendar, this );
cleanupFolder( mNotes, this );
cleanupFolder( mTasks, this );
cleanupFolder( mJournals, this );
mContacts = mCalendar = mNotes = mTasks = mJournals = 0;
}
void KMailICalIfaceImpl::loadPixmaps() const
{
static bool pixmapsLoaded = false;
if( mUseResourceIMAP && !pixmapsLoaded ) {
pixmapsLoaded = true;
pixContacts = new QPixmap( UserIcon("kmgroupware_folder_contacts"));
pixCalendar = new QPixmap( UserIcon("kmgroupware_folder_calendar"));
pixNotes = new QPixmap( UserIcon("kmgroupware_folder_notes"));
pixTasks = new QPixmap( UserIcon("kmgroupware_folder_tasks"));
pixJournals = new QPixmap( UserIcon("kmgroupware_folder_journals"));
}
}
QString KMailICalIfaceImpl::folderPixmap( KFolderTreeItem::Type type ) const
{
if( !mUseResourceIMAP )
return QString::null;
if( type == KFolderTreeItem::Contacts )
return QString::fromLatin1( "kmgroupware_folder_contacts" );
else if( type == KFolderTreeItem::Calendar )
return QString::fromLatin1( "kmgroupware_folder_calendar" );
else if( type == KFolderTreeItem::Notes )
return QString::fromLatin1( "kmgroupware_folder_notes" );
else if( type == KFolderTreeItem::Tasks )
return QString::fromLatin1( "kmgroupware_folder_tasks" );
else if( type == KFolderTreeItem::Journals )
return QString::fromLatin1( "kmgroupware_folder_journals" );
return QString::null;
}
QPixmap* KMailICalIfaceImpl::pixContacts;
QPixmap* KMailICalIfaceImpl::pixCalendar;
QPixmap* KMailICalIfaceImpl::pixNotes;
QPixmap* KMailICalIfaceImpl::pixTasks;
QPixmap* KMailICalIfaceImpl::pixJournals;
static void reloadFolderTree()
{
// Make the folder tree show the icons or not
kmkernel->folderMgr()->contentsChanged();
}
// This is a very light-weight and fast 'parser' to retrieve
// a data entry from a vCal taking continuation lines
// into account
static void vPartMicroParser( const QString& str, QString& s )
{
QString line;
uint len = str.length();
for( uint i=0; i<len; ++i){
if( str[i] == '\r' || str[i] == '\n' ){
if( str[i] == '\r' )
++i;
if( i+1 < len && str[i+1] == ' ' ){
// found a continuation line, skip it's leading blanc
++i;
}else{
// found a logical line end, process the line
if( line.startsWith( s ) ) {
s = line.mid( s.length() + 1 );
return;
}
line = "";
}
} else {
line += str[i];
}
}
// Not found. Clear it
s.truncate(0);
}
KMFolder* KMailICalIfaceImpl::findStandardResourceFolder( KMFolderDir* folderParentDir, KMail::FolderContentsType contentsType )
{
if ( GlobalSettings::theIMAPResourceStorageFormat() == GlobalSettings::EnumTheIMAPResourceStorageFormat::XML )
{
// Look for a folder with an annotation like "event.default"
QPtrListIterator<KMFolderNode> it( *folderParentDir );
for ( ; it.current(); ++it ) {
if ( !it.current()->isDir() ) {
KMFolder* folder = static_cast<KMFolder *>( it.current() );
if ( folder->folderType() == KMFolderTypeCachedImap ) {
QString annotation = static_cast<KMFolderCachedImap*>( folder->storage() )->annotationFolderType();
//kdDebug(5006) << "findStandardResourceFolder: " << folder->name() << " has annotation " << annotation << endl;
if ( annotation == QString( s_folderContentsType[contentsType].annotation ) + ".default" )
return folder;
}
}
}
kdDebug(5006) << "findStandardResourceFolder: no standard resource folder for " << s_folderContentsType[contentsType].annotation << endl;
return 0;
}
else // icalvcard: look up standard resource folders by name
{
KFolderTreeItem::Type itemType = s_folderContentsType[contentsType].treeItemType;
unsigned int folderLanguage = GlobalSettings::theIMAPResourceFolderLanguage();
if( folderLanguage > 3 ) folderLanguage = 0;
KMFolderNode* node = folderParentDir->hasNamedFolder( folderName( itemType, folderLanguage ) );
if ( !node || node->isDir() )
return 0;
return static_cast<KMFolder*>( node );
}
}
#include "kmailicalifaceimpl.moc"