summaryrefslogtreecommitdiffstats
path: root/kresources/newexchange/exchangeconvertercalendar.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kresources/newexchange/exchangeconvertercalendar.cpp')
-rw-r--r--kresources/newexchange/exchangeconvertercalendar.cpp820
1 files changed, 820 insertions, 0 deletions
diff --git a/kresources/newexchange/exchangeconvertercalendar.cpp b/kresources/newexchange/exchangeconvertercalendar.cpp
new file mode 100644
index 000000000..bc0481daf
--- /dev/null
+++ b/kresources/newexchange/exchangeconvertercalendar.cpp
@@ -0,0 +1,820 @@
+/*
+ This file is part of the exchange resource.
+ Copyright (c) 2004 Reinhold Kainhofer <[email protected]>
+ Parts are derived from the old libkpimexchange library:
+ Copyright (c) 2002 Jan-Pascal van Best <[email protected]>
+
+ 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., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+*/
+
+#include "exchangeconvertercalendar.h"
+#include <webdavhandler.h>
+#include <libkcal/incidence.h>
+#include <libkcal/event.h>
+#include <libkcal/journal.h>
+#include <libkcal/todo.h>
+#include <libkcal/icalformat.h>
+#include <libemailfunctions/email.h>
+
+#include <kdebug.h>
+
+using namespace KCal;
+
+#define TaskNamespace1 "http://schemas.microsoft.com/mapi/id/{00062003-0000-0000-C000-000000000046}/"
+#define TaskProp_Status "0x00008101"
+#define TaskProp_PercentCompleted "0x00008102"
+#define TaskProp_DtStart "0x00008104"
+#define TaskProp_DtDue "0x00008105"
+#define TaskProp_Duration "0x00008106"
+#define TaskProp_CompletionDate "0x0000810f"
+#define TaskProp_IsCompleted "0x0000811C"
+#define TaskProp_Owner "0x0000811F"
+#define TaskProp_DoesRecur "0x00008126"
+
+#define TaskNamespace2 "http://schemas.microsoft.com/mapi/id/{00062008-0000-0000-C000-000000000046}/"
+#define TaskProp_ReminderTime "0x00008502"
+#define TaskProp_ReminderSet "0x00008503"
+#define TaskProp_ReminderPlaySound "0x0000851E"
+#define TaskProp_ReminderSoundFile "0x0000851F"
+#define TaskProp_ContactNames "0x0000853A"
+
+
+ExchangeConverterCalendar::ExchangeConverterCalendar()
+{
+}
+
+void ExchangeConverterCalendar::setTimeZone( const QString &id )
+{
+ // kdDebug() << "Setting timezone to: " << id << endl;
+ mFormat.setTimeZone( id, true );
+}
+
+#define propertyDAV( prop ) \
+ WebdavHandler::addElement( doc, root, "d:"prop )
+#define propertyNS( ns, prop ) \
+ WebdavHandler::addElementNS( doc, root, ns, prop )
+#define propertyCalendar( prop ) \
+ WebdavHandler::addElement( doc, root, "c:"prop )
+#define propertyHTTPMail( prop ) \
+ WebdavHandler::addElement( doc, root, "m:"prop )
+#define propertyMailHeader( prop ) \
+ WebdavHandler::addElement( doc, root, "h:"prop )
+#define property( prop ) \
+ WebdavHandler::addElement( doc, root, prop )
+
+void ExchangeConverterCalendar::createRequestIncidence( QDomDocument &doc, QDomElement &root )
+{
+ propertyDAV( "contentclass" );
+ propertyDAV( "getcontenttype" );
+ propertyNS( "http://schemas.microsoft.com/exchange/", "outlookmessageclass" );
+ propertyDAV( "getetag" );
+ propertyDAV( "href" );
+ propertyDAV( "isreadonly" );
+ propertyNS( "http://schemas.microsoft.com/repl/", "repl-uid" );
+ propertyHTTPMail( "subject" );
+ propertyHTTPMail( "textdescription" );
+ propertyHTTPMail( "date" );
+ propertyDAV( "comment" );
+ propertyNS( "urn:schemas-microsoft-com:office:office", "Keywords" );
+ propertyNS( "http://schemas.microsoft.com/exchange/", "sensitivity" );
+ propertyHTTPMail( "priority" );
+ propertyHTTPMail( "from" );
+ propertyHTTPMail( "to" );
+ propertyHTTPMail( "cc" );
+ propertyHTTPMail( "bcc" );
+ propertyHTTPMail( "hasattachment" );
+}
+
+void ExchangeConverterCalendar::createRequestAppointment( QDomDocument &doc, QDomElement &root )
+{
+ createRequestIncidence( doc, root );
+ QDomAttr att_c = doc.createAttribute( "xmlns:c" );
+ att_c.setValue( "urn:schemas:calendar:" );
+ doc.documentElement().setAttributeNode( att_c );
+ propertyCalendar( "uid" );
+ propertyCalendar( "created" );
+ propertyCalendar( "lastmodified" );
+ propertyCalendar( "dtstamp" );
+ propertyCalendar( "sequence" );
+ propertyCalendar( "location" );
+ propertyCalendar( "busystatus" );
+ propertyCalendar( "transparent" );
+ propertyCalendar( "timezone" );
+ propertyCalendar( "alldayevent" );
+ propertyCalendar( "dtstart" );
+ propertyCalendar( "dtend" );
+ propertyCalendar( "duration" );
+ propertyCalendar( "rrule" );
+ propertyCalendar( "rdate" );
+ propertyCalendar( "exrule" );
+ propertyCalendar( "exdate" );
+ propertyCalendar( "recurrenceid" );
+ propertyCalendar( "instancetype" );
+ propertyCalendar( "reminderoffset" );
+ propertyCalendar( "resources" );
+}
+
+#define propertyTask1( prop ) \
+ WebdavHandler::addElement( doc, props, "t1:"prop )
+#define propertyTask2( prop ) \
+ WebdavHandler::addElement( doc, props, "t2:"prop )
+
+void ExchangeConverterCalendar::createRequestTask( QDomDocument &doc, QDomElement &props )
+{
+ createRequestIncidence( doc, props );
+
+ QDomElement root = doc.documentElement();
+
+ QDomAttr att_t1 = doc.createAttribute( "xmlns:t1" );
+ att_t1.setValue( TaskNamespace1 );
+ root.setAttributeNode( att_t1 );
+
+ QDomAttr att_t2 = doc.createAttribute( "xmlns:t2" );
+ att_t2.setValue( TaskNamespace2 );
+ root.setAttributeNode( att_t2 );
+
+ // TODO: Insert the correct namespaces here:
+// propertyTask1( TaskProp_UID );
+ propertyDAV( "creationdate" );
+ propertyDAV( "getlastmodified" );
+ propertyTask1( TaskProp_Owner );
+ propertyTask2( TaskProp_ContactNames );
+ propertyTask1( TaskProp_DtStart );
+ propertyTask1( TaskProp_DtDue );
+ propertyTask1( TaskProp_Duration );
+ propertyTask1( TaskProp_IsCompleted );
+ propertyTask1( TaskProp_PercentCompleted );
+ propertyTask1( TaskProp_CompletionDate );
+ propertyTask1( TaskProp_DoesRecur );
+ // What to do about recurrence rules?
+ propertyTask2( TaskProp_ReminderSet );
+ propertyTask2( TaskProp_ReminderTime );
+ propertyTask2( TaskProp_ReminderPlaySound );
+ propertyTask2( TaskProp_ReminderSoundFile );
+ propertyTask1( TaskProp_Status );
+}
+#undef propertyTask1
+#undef propertyTask2
+
+void ExchangeConverterCalendar::createRequestJournal( QDomDocument &doc, QDomElement &root )
+{
+ createRequestIncidence( doc, root );
+ propertyDAV( "uid" );
+ propertyDAV( "creationdate" );
+ propertyDAV( "getlastmodified" );
+}
+#undef propertyDAV
+#undef propertyNS
+#undef propertyCalendar
+#undef propertyHTTPMail
+#undef propertyMailHeader
+#undef property
+
+bool ExchangeConverterCalendar::readTZ( const QDomElement &node, Incidence */*incidence*/ )
+{
+ QString timezoneid;
+ if ( WebdavHandler::extractString( node, "timezoneid", timezoneid ) ) {
+ // kdDebug() << "DEBUG: timezoneid = " << timezoneid << endl;
+ }
+
+ QString timezone;
+ if ( WebdavHandler::extractString( node, "timezone", timezone ) ) {
+ // kdDebug() << "DEBUG: timezone = " << timezone << endl;
+ }
+
+ // TODO:
+/* // mFormat is used for parsing recurrence rules.
+ QString localTimeZoneId;
+ if ( mCalendar ) {
+ mFormat.setTimeZone( mCalendar->timeZoneId(), !mCalendar->isLocalTime() );
+ localTimeZoneId = mCalendar->timeZoneId();
+ } else {
+ localTimeZoneId = "UTC";
+ // If no mCalendar, stay in UTC
+ }
+*/
+ return true;
+}
+
+bool ExchangeConverterCalendar::readIncidence( const QDomElement &node, Incidence *incidence )
+{
+kdDebug()<<"ExchangeConverterCalendar::readIncidencd"<<endl;
+ QDateTime tmpdt;
+ bool tmpbool;
+ QString tmpstr;
+ long tmplng;
+ QStringList tmplst;
+
+ readTZ( node, incidence );
+
+ if ( WebdavHandler::extractString( node, "getetag", tmpstr ) )
+ incidence->setCustomProperty( "KDEPIM-Exchange-Resource", "fingerprint", tmpstr );
+ if ( WebdavHandler::extractString( node, "href", tmpstr ) )
+ incidence->setCustomProperty( "KDEPIM-Exchange-Resource", "href", tmpstr );
+
+ // FIXME: use repl-uid as scheduling id?
+ if ( WebdavHandler::extractString( node, "textdescription", tmpstr ) )
+ incidence->setDescription( tmpstr );
+ if ( WebdavHandler::extractString( node, "subject", tmpstr ) )
+ incidence->setSummary( tmpstr );
+ if ( WebdavHandler::extractStringList( node, "Keywords", tmplst ) )
+ incidence->setCategories( tmplst );
+
+ // Use "created" or "creationdate"?
+ if ( WebdavHandler::extractBool( node, "isreadonly" , tmpbool ) )
+ incidence->setReadOnly( tmpbool );
+
+ // FIXME: Ignore the comment for now
+
+ // Exchange sentitivity values:
+ // 0 None, 1 Personal, 2 Private, 3 Company Confidential
+ if ( WebdavHandler::extractLong( node, "sensitivity", tmplng ) ) {
+ switch( tmplng ) {
+ case 0: incidence->setSecrecy( KCal::Incidence::SecrecyPublic ); break;
+ case 1:
+ case 2: incidence->setSecrecy( KCal::Incidence::SecrecyPrivate ); break;
+ case 3: incidence->setSecrecy( KCal::Incidence::SecrecyConfidential ); break;
+ default: kdWarning() << "Unknown sensitivity: " << tmplng << endl;
+ }
+ }
+
+ if ( WebdavHandler::extractBool( node, "hasattachment", tmpbool ) && tmpbool ) {
+ // FIXME: Extract attachments...
+ }
+
+ if ( WebdavHandler::extractLong( node, "priority", tmplng ) )
+ incidence->setPriority( tmplng );
+
+ // FIXME: Use the urn:schemes:httpmail:date property for what?
+
+ // Organizer, required and optional Attendees:
+ if ( WebdavHandler::extractString( node, "from", tmpstr ) )
+ incidence->setOrganizer( tmpstr );
+ if ( WebdavHandler::extractString( node, "to", tmpstr ) ) {
+ QStringList atts( KPIM::splitEmailAddrList( tmpstr ) );
+ for ( QStringList::Iterator it = atts.begin(); it != atts.end(); ++it ) {
+ QString name, email;
+ KPIM::getNameAndMail( *it, name, email );
+ Attendee *att = new Attendee( name, email );
+ att->setRole( KCal::Attendee::ReqParticipant );
+ // FIXME: Retrieve the other attendee properties somehow...
+ // urn:schemas:calendar:method
+ // urn:schemas:calendar:responserequested
+ // urn:schemas:calendar:meetingstatus
+ // urn:schemas:calendar:replytime
+ incidence->addAttendee( att );
+ }
+ }
+ if ( WebdavHandler::extractString( node, "cc", tmpstr ) ) {
+ QStringList atts( KPIM::splitEmailAddrList( tmpstr ) );
+ for ( QStringList::Iterator it = atts.begin(); it != atts.end(); ++it ) {
+ QString name, email;
+ KPIM::getNameAndMail( *it, name, email );
+ Attendee *att = new Attendee( name, email );
+ att->setRole( KCal::Attendee::OptParticipant );
+ // FIXME: Retrieve the other attendee properties somehow...
+ // urn:schemas:calendar:method
+ // urn:schemas:calendar:responserequested
+ // urn:schemas:calendar:meetingstatus
+ // urn:schemas:calendar:replytime
+ incidence->addAttendee( att );
+ }
+ }
+
+ return true;
+}
+
+/* FIXME: Handle recurrences
+void ExchangeDownload::handleRecurrence( QString uid )
+{
+ // kdDebug() << "Handling recurrence info for uid=" << uid << endl;
+ QString query =
+ "SELECT \"DAV:href\", \"urn:schemas:calendar:instancetype\"\r\n"
+ "FROM Scope('shallow traversal of \"\"')\r\n"
+ "WHERE \"urn:schemas:calendar:uid\" = '" + uid + "'\r\n"
+ " AND (\"urn:schemas:calendar:instancetype\" = 1)\r\n";
+// " OR \"urn:schemas:calendar:instancetype\" = 3)\r\n" // FIXME: exception are not handled
+
+ // kdDebug() << "Exchange master query: " << endl << query << endl;
+
+ KIO::DavJob* job = KIO::davSearch( mAccount->calendarURL(), "DAV:", "sql",
+ query, false );
+ KIO::Scheduler::scheduleJob( job );
+ job->setWindow( mWindow );
+ connect( job, SIGNAL( result( KIO::Job * ) ),
+ SLOT( slotMasterResult( KIO::Job * ) ) );
+}
+*/
+
+bool ExchangeConverterCalendar::readEvent( const QDomElement &node, Event *event )
+{
+ if ( !readIncidence( node, event ) ) return false;
+kdDebug()<<"ExchangeConverterCalendar::readEvent"<<endl;
+
+ QDateTime tmpdt;
+ QString tmpstr;
+ long tmplng;
+ bool tmpbool;
+
+ // The UID is absolutely required!
+ if ( WebdavHandler::extractString( node, "uid", tmpstr ) ) {
+ event->setUid( tmpstr );
+ } else {
+kdDebug()<<"ExchangeConverterCalendar::readIncidence: ERROR: No UID given"<<endl;
+ return false;
+ }
+ if ( WebdavHandler::extractDateTime( node, "created", tmpdt ) )
+ event->setCreated( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ if ( WebdavHandler::extractDateTime( node, "lastmodified", tmpdt ) )
+ event->setLastModified( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ // FIXME: Retrieve time zone: "timezone"
+ // FIXME: Use the "recurrenceid" prop for the recurrenceId of the event (which is protected!)
+
+ // FIXME: Retrieve MICROSOFT-CDO-* tags first
+
+ if ( WebdavHandler::extractLong( node, "sequence", tmplng ) )
+ event->setRevision( tmplng );
+
+ if ( WebdavHandler::extractString( node, "location", tmpstr ) )
+ event->setLocation( tmpstr );
+
+ // FIXME: Use "organizer" here instead of the From: person?
+/* if ( WebdavHandler::extractString( node, "organizer", tmpstr ) )
+ incidence->setOrganizer( tmpstr );*/
+
+ if ( WebdavHandler::extractDateTime( node, "dtstart", tmpdt ) ) {
+ event->setDtStart( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ }
+
+ bool allDay = false;
+ if ( WebdavHandler::extractBool( node, "alldayevent", allDay ) )
+ event->setFloats( allDay );
+
+ if ( WebdavHandler::extractLong( node, "duration", tmplng ) ) {
+ if (allDay)
+ tmplng--; // Otherwise event extends into next day
+ event->setDuration( tmplng );
+ // kdDebug() << "DURATION " << tmplng << "\n";
+ } else if ( WebdavHandler::extractDateTime( node, "dtend", tmpdt ) ) {
+ event->setDtEnd( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ }
+
+ // FIXME: Here we have two different props for the same thing?!?!?
+ if ( WebdavHandler::extractLong( node, "transparent", tmplng ) )
+ event->setTransparency( tmplng>0 ? Event::Transparent : Event::Opaque );
+ if ( WebdavHandler::extractString( node, "busystatus", tmpstr ) ) {
+ if ( tmpstr == "FREE" )
+ event->setTransparency( KCal::Event::Transparent );
+ if ( tmpstr == "BUSY" )
+ event->setTransparency( KCal::Event::Opaque );
+ }
+
+ if ( WebdavHandler::extractLong( node, "reminderoffset", tmplng ) ) {
+ // Duration before event in seconds
+ KCal::Duration offset( -tmplng );
+ KCal::Alarm *alarm = event->newAlarm();
+ alarm->setStartOffset( offset );
+ alarm->setEnabled( true );
+ alarm->setType( KCal::Alarm::Display);
+ // TODO: multiple alarms;
+ }
+
+
+ if ( WebdavHandler::extractString( node, "rrule", tmpstr ) && !tmpstr.isEmpty() ) {
+ kdDebug() << "Got rrule: " << tmpstr << endl;
+ // Timezone should be handled automatically
+ // because we used mFormat.setTimeZone() earlier
+ // FIXME: Implement this using the format!
+ RecurrenceRule *rrule = event->recurrence()->defaultRRule( true );
+ if ( ! mFormat.fromString( rrule, tmpstr ) ) {
+ kdError() << "ERROR parsing rrule " << tmpstr << endl;
+ event->recurrence()->addRRule( rrule );
+ }
+ }
+
+ QStringList tmplst;
+ if ( WebdavHandler::extractStringList( node, "exdate", tmplst ) ) {
+ QStringList::Iterator it = tmplst.begin();
+ KCal::DateList exdates;
+ for ( ; it != tmplst.end(); ++it ) {
+ exdates.append( /*utcAsZone(*/ QDateTime::fromString( *it, Qt::ISODate )/*,
+ localTimeZoneId )*/.date() );
+ }
+ event->recurrence()->setExDates( exdates );
+ }
+ // FIXME: use rdate and exrule!
+/* FIXME: Recurring events, they are split up
+ QDomElement instancetypeElement = prop.namedItem( "instancetype" ).toElement();
+ if ( instancetypeElement.isNull() ) {
+ kdError() << "Error: no instance type in Exchange server reply" << endl;
+ continue;
+ }
+ int instanceType = instancetypeElement.text().toInt();
+ //kdDebug() << "Instance type: " << instanceType << endl;
+
+ if ( recurrence && instanceType > 0 ) {
+ QDomElement uidElement = prop.namedItem( "uid" ).toElement();
+ if ( uidElement.isNull() ) {
+ kdError() << "Error: no uid in Exchange server reply" << endl;
+ continue;
+ }
+ QString uid = uidElement.text();
+ if ( ! m_uids.contains( uid ) ) {
+ m_uids[uid] = 1;
+ handleRecurrence(uid);
+ successCount++;
+ }
+ continue;
+ }
+*/
+
+
+ // FIXME: read the resources from the "resources" tag
+
+ // FIXME: Custom fields not yet implemented
+ return true;
+}
+
+bool ExchangeConverterCalendar::readTodo( const QDomElement &node, Todo *todo )
+{
+ if ( !readIncidence( node, todo ) ) return false;
+kdDebug()<<"ExchangeConverterCalendar::readTodo"<<endl;
+
+ // FIXME: Retrieve time zone: "timezone"
+ // FIXME: What to with TaskProp_Owner and TaskProp_ContactNames?
+
+ QDateTime tmpdt;
+ float tmpfloat;
+ long tmplong;
+ bool tmpbool;
+ QString tmpstr;
+
+ // The UID is absolutely required!
+ // FIXME: Which field shall be used as uid???
+/* if ( WebdavHandler::extractString( node, "uid", tmpstr ) ) {
+ todo->setUid( tmpstr );
+ } else {
+kdDebug()<<"ExchangeConverterCalendar::readIncidence: ERROR: No UID given"<<endl;
+ return false;
+ }*/
+// if ( WebdavHandler::extractDateTime( node, "created", tmpdt ) )
+/* FIXME: creation and last modification dates:
+ if ( WebdavHandler::extractDateTime( node, "creationdate", tmpdt ) )
+ incidence->setCreated( tmpdt );
+ if ( WebdavHandler::extractDateTime( node, "getlastmodified", tmpdt ) )
+ incidence->setLastModified( tmpdt );*/
+
+ if ( WebdavHandler::extractDateTime( node, TaskProp_DtStart, tmpdt ) )
+ todo->setDtStart( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ if ( WebdavHandler::extractDateTime( node, TaskProp_DtDue, tmpdt ) )
+ todo->setDtDue( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ if ( WebdavHandler::extractLong( node, TaskProp_Duration, tmplong ) )
+ todo->setDuration( tmplong );
+
+ if ( WebdavHandler::extractBool( node, TaskProp_IsCompleted, tmpbool ) && tmpbool ) {
+ todo->setCompleted( tmpbool );
+ if ( tmpbool && WebdavHandler::extractDateTime( node, TaskProp_CompletionDate, tmpdt ) ) {
+ todo->setCompleted( WebdavHandler::utcAsZone( tmpdt, mFormat.timeZoneId() ) );
+ }
+ }
+ if ( WebdavHandler::extractFloat( node, TaskProp_PercentCompleted, tmpfloat ) )
+ todo->setPercentComplete( (int)(tmpfloat*100) );
+
+ // FIXME: Recurrence, using TaskProp_DoesRecur
+ // What to do about recurrence rules?
+
+ // FIXME: Reminders, use TaskProp_ReminderSet, TaskProp_ReminderTime,
+ // TaskProp_ReminderPlaySound, TaskProp_ReminderSoundFile, TaskProp_Status
+ // But how do I get the offset?
+
+ return true;
+}
+
+bool ExchangeConverterCalendar::readJournal( const QDomElement &node, Journal *journal )
+{
+ if ( !readIncidence( node, journal ) ) return false;
+kdDebug()<<"ExchangeConverterCalendar::readJournal"<<endl;
+ QDateTime tmpdt;
+ QString tmpstr;
+ // The UID is absolutely required!
+ // FIXME: Which field shall be used as UID?
+ if ( WebdavHandler::extractString( node, "uid", tmpstr ) ) {
+ journal->setUid( tmpstr );
+ } else {
+kdDebug()<<"ExchangeConverterCalendar::readJournal: ERROR: No UID given"<<endl;
+ return false;
+ }
+/* FIXME: creation and last modification times:
+ if ( WebdavHandler::extractDateTime( node, "created", tmpdt ) )
+ incidence->setCreated( tmpdt );
+ if ( WebdavHandler::extractDateTime( node, "lastmodified", tmpdt ) )
+ incidence->setLastModified( tmpdt );*/
+
+ if ( WebdavHandler::extractDateTime( node, "date", tmpdt ) )
+ journal->setDtStart( tmpdt );
+ return true;
+}
+
+Incidence::List ExchangeConverterCalendar::parseWebDAV( const QDomDocument& davdata )
+{
+ QDomElement prop = davdata.documentElement().namedItem( "response" )
+ .namedItem( "propstat" ).namedItem( "prop" ).toElement();
+ if ( prop.isNull() ) return Incidence::List();
+
+ QString contentclass;
+ bool success = WebdavHandler::extractString( prop, "contentclass", contentclass );
+ if ( !success ) return Incidence::List();
+
+ Incidence *incidence = 0;
+ success = false;
+ if ( contentclass == "urn:content-classes:appointment" ) {
+ Event *event = new Event();
+ success = readEvent( prop, event );
+ incidence = event;
+ } else if ( contentclass == "urn:content-classes:task" ) {
+ Todo *todo = new Todo();
+ success = readTodo( prop, todo );
+ incidence = todo;
+ } else if ( contentclass == "urn:content-classes:journal" ||
+ contentclass == "urn:content-classes:message" ) {
+ Journal *journal = new Journal();
+ success = readJournal( prop, journal );
+ incidence = journal;
+ }
+
+ Incidence::List list;
+ if ( success ) {
+ list.append( incidence );
+ }
+ return list;
+}
+
+
+#define domDavProperty( name, value ) \
+ WebdavHandler::addElement( mDoc, mElement, "d:"name, value )
+#define domProperty( NS, name, value ) \
+ WebdavHandler::addElementNS( mDoc, mElement, NS, name, value )
+#define domCalendarProperty( name, value ) \
+ WebdavHandler::addElement( mDoc, mElement, "c:"name, value )
+#define domHTTPMailProperty( name, value ) \
+ WebdavHandler::addElement( mDoc, mElement, "m:"name, value )
+#define domMailHeaderProperty( name, value ) \
+ WebdavHandler::addElement( mDoc, mElement, "h:"name, value )
+
+
+
+
+class ExchangeConverterCalendar::createWebDAVVisitor : public IncidenceBase::Visitor
+{
+ public:
+ createWebDAVVisitor() : Visitor() {}
+ bool act( QDomDocument doc, QDomElement el, IncidenceBase *incidence, const QString &timeZoneId )
+ {
+ mDoc = doc;
+ mElement = el;
+ mTimeZoneId = timeZoneId;
+ return incidence->accept( *this );
+ }
+ protected:
+ void addBoolProp( QDomElement &el ) { el.setAttribute( "b:dt", "boolean" ); }
+ void addDateProp( QDomElement &el ) { el.setAttribute( "b:dt", "dateTime.tz" ); }
+ void addFloatProp( QDomElement &el ) { el.setAttribute( "b:dt", "float" ); }
+ void addIntProp( QDomElement &el ) { el.setAttribute( "b:dt", "int" ); }
+ QString timePropString( const QDateTime &dt ) { return dt.toString( Qt::ISODate )+"Z"; }
+
+ bool visitIncidence( Incidence *incidence )
+ {
+ QString tmpstr;
+ domDavProperty( "isreadonly", (incidence->isReadOnly())?"1":"0" );
+ // FIXME: scheduling ID
+// domProperty( "http://schemas.microsoft.com/repl/", "repl-uid", ??? );
+ domHTTPMailProperty( "subject", incidence->summary() );
+ domHTTPMailProperty( "textdescription", incidence->description() );
+ // FIXME: timestampt, comments and categories
+// domHTTPMailProperty( "date", ??? ); // timestamp not available in libkcal
+// domDavProperty( "comment", incidence->comments() ); // libkcal has a QStringlist, not one string
+// domProperty( "urn:schemas-microsoft-com:office:office", "Keywords", ??? ); // It's a <v>entyr1</v><v>entry2</v> String list!
+ tmpstr = QString::null;
+ switch ( incidence->secrecy() ) {
+ case KCal::Incidence::SecrecyPublic: tmpstr = "0"; break;
+ case KCal::Incidence::SecrecyPrivate: tmpstr = "2"; break;
+ case KCal::Incidence::SecrecyConfidential: tmpstr = "3"; break;
+ default: break;
+ }
+ if ( !tmpstr.isEmpty() )
+ domProperty( "http://schemas.microsoft.com/exchange/", "sensitivity", tmpstr );
+
+ domHTTPMailProperty( "priority", QString::number(incidence->priority()) );
+
+ domMailHeaderProperty( "from", incidence->organizer().fullName() );
+
+ // Attendees:
+ tmpstr = QString::null;
+ QStringList reqattnames;
+ QStringList optattnames;
+ Attendee::List atts = incidence->attendees();
+ for ( Attendee::List::Iterator it = atts.begin(); it != atts.end(); ++it ) {
+ switch ( (*it)->role() ) {
+ case KCal::Attendee::Chair:
+ case KCal::Attendee::ReqParticipant:
+ reqattnames << (*it)->fullName();
+ break;
+ case KCal::Attendee::OptParticipant:
+ case KCal::Attendee::NonParticipant:
+ optattnames << (*it)->fullName();
+ break;
+ default: break;
+ }
+ }
+
+ domMailHeaderProperty( "to", reqattnames.join(", ") );
+
+ domMailHeaderProperty( "cc", optattnames.join(", ") );
+
+ // FIXME: Attachments: propertyHTTPMail( "hasattachment" );
+
+ return true;
+ }
+ bool visit( Event *event )
+ {
+ if ( !visitIncidence(event) ) return false;
+
+ QDomAttr att_c = mDoc.createAttribute( "xmlns:c" );
+ att_c.setValue( "urn:schemas:calendar:" );
+ mDoc.documentElement().setAttributeNode( att_c );
+
+ domDavProperty( "contentclass", "urn:content-classes:appointment" );
+ domProperty( "http://schemas.microsoft.com/exchange/",
+ "outlookmessageclass", "IPM.Appointment" );
+ domCalendarProperty( "uid", event->uid() );
+ QDomElement el = domCalendarProperty( "created", timePropString( WebdavHandler::zoneAsUtc( event->created(), mTimeZoneId ) ) );
+ addDateProp( el );
+ el = domCalendarProperty( "lastmodified", timePropString( WebdavHandler::zoneAsUtc( event->lastModified(), mTimeZoneId ) ) );
+ addDateProp( el );
+ // FIXME: domCalendarProperty( "dtstamp", ??);
+// FIXME: domCalendarProperty( "sequence", event->sequence() );
+ domCalendarProperty( "location", event->location() );
+
+ QString tmpstr( QString::null );
+ switch ( event->transparency() ) {
+ case KCal::Event::Transparent: tmpstr = "FREE"; break;
+ case KCal::Event::Opaque: tmpstr = "BUSY"; break;
+ }
+ if ( !tmpstr.isEmpty() )
+ domCalendarProperty( "busystatus", tmpstr );
+// FIXME: What do do with the "transparent" property?
+// FIXME: Use the "timezone" property...
+ domCalendarProperty( "alldayevent", event->doesFloat()?"1":"0" );
+ el = domCalendarProperty( "dtstart", timePropString( WebdavHandler::zoneAsUtc( event->dtStart(), mTimeZoneId ) ) );
+ addDateProp( el );
+ if ( event->hasEndDate() ) {
+ el = domCalendarProperty( "dtend", timePropString( WebdavHandler::zoneAsUtc( event->dtEnd(), mTimeZoneId ) ) );
+ addDateProp( el );
+ } else {
+ domCalendarProperty( "duration", QString::number( event->duration() ) );
+ }
+ // FIXME: Convert the recurrence rule to a string:
+ if ( event->doesRecur() ) {
+// tmpstr = event->recurrence().....
+// domCalendarProperty( "rrule", tmpstr );
+ // FIXME: Use "rdate" and "exrule"
+ // FIXME: Use "exdate", what's the syntax?
+ // FIXME: use the "instancetype" property
+ }
+ // FIXME: RecurrenceID is protected!
+// domCalendarProperty( "recurrenceid", event->recurrenceId() );
+ // FIXME: "reminderoffset"
+ // FIXME: "resources" Alarm::List alarms = event->alarms();
+
+ Alarm::List::ConstIterator it;
+ Alarm::List alarms = event->alarms();
+ for( it = alarms.begin(); it != alarms.end(); ++it ) {
+ if ((*it)->hasStartOffset()) {
+ domCalendarProperty( "reminderoffset", QString::number( (*it)->startOffset().asSeconds() * -1 ) );
+ } else {
+ kdDebug() << "ExchangeConverterCalendar::createWebDAVVisitor: Alarm type not supported\n";
+ }
+ }
+
+ return true;
+ }
+ bool visit( Todo *todo )
+ {
+ if ( !visitIncidence(todo) ) return false;
+
+ QDomAttr att_t1 = mDoc.createAttribute( "xmlns:t1" );
+ att_t1.setValue( TaskNamespace1 );
+ mDoc.documentElement().setAttributeNode( att_t1 );
+
+ QDomAttr att_t2 = mDoc.createAttribute( "xmlns:t2" );
+ att_t2.setValue( TaskNamespace2 );
+ mDoc.documentElement().setAttributeNode( att_t2 );
+
+
+ domDavProperty( "contentclass", "urn:content-classes:task" );
+ domProperty( "http://schemas.microsoft.com/exchange/",
+ "outlookmessageclass", "IPM.Task" );
+
+/* FIXME:
+ domCalendarProperty( "uid", todo->uid() );
+ domCalendarProperty( "created", todo->created().toString( Qt::ISODate ) );
+ domCalendarProperty( "lastmodified", todo->lastModified().toString( Qt::ISODate ) );*/
+ // TODO
+/*propertyTask1( TaskProp_Owner );
+ propertyTask2( TaskProp_ContactNames );
+ propertyTask1( TaskProp_DtStart );
+ propertyTask1( TaskProp_DtDue );
+ propertyTask1( TaskProp_Duration );
+ propertyTask1( TaskProp_IsCompleted );
+ propertyTask1( TaskProp_PercentCompleted );
+ propertyTask1( TaskProp_CompetionDate );
+ propertyTask1( TaskProp_DoesRecur );
+ // What to do about recurrence rules?
+ propertyTask2( TaskProp_ReminderSet );
+ propertyTask2( TaskProp_ReminderTime );
+ propertyTask2( TaskProp_ReminderPlaySound );
+ propertyTask2( TaskProp_ReminderSoundFile );
+ propertyTask1( TaskProp_Status );*/
+ return true;
+ }
+ bool visit( Journal *journal )
+ {
+ if ( !visitIncidence(journal) ) return false;
+ domDavProperty( "contentclass", "urn:content-classes:journal" );
+ domProperty( "http://schemas.microsoft.com/exchange/",
+ "outlookmessageclass", "IPM.Journal" );
+/* FIXME:
+ domCalendarProperty( "uid", todo->uid() );
+ domCalendarProperty( "created", todo->created().toString( Qt::ISODate ) );
+ domCalendarProperty( "lastmodified", todo->lastModified().toString( Qt::ISODate ) );*/
+ // TODO
+ return true;
+ }
+
+ protected:
+ QDomDocument mDoc;
+ QDomElement mElement;
+ QString mTimeZoneId;
+};
+
+// Prefixes for the namespaces:
+// d... DAV:
+// b... urn:schemas-microsoft-com:datatypes
+// c... calendar
+// m... httpmail
+// h... httpheader
+// p... mapi
+// o... office
+//
+
+QDomDocument ExchangeConverterCalendar::createWebDAV( Incidence *incidence )
+{
+ // TODO
+ QDomDocument doc;
+ QDomElement root = WebdavHandler::addDavElement( doc, doc, "d:propertyupdate" );
+ QDomElement set = WebdavHandler::addElement( doc, root, "d:set" );
+ QDomElement prop = WebdavHandler::addElement( doc, set, "d:prop" );
+
+ QDomAttr att_b = doc.createAttribute( "xmlns:b" );
+ att_b.setValue( "urn:schemas-microsoft-com:datatypes" );
+ root.setAttributeNode( att_b );
+
+ QDomAttr att_h = doc.createAttribute( "xmlns:h" );
+ att_h.setValue( "urn:schemas:mailheader:" );
+ root.setAttributeNode( att_h );
+
+ QDomAttr att_m = doc.createAttribute( "xmlns:m" );
+ att_m.setValue( "urn:schemas:httpmail:" );
+ root.setAttributeNode( att_m );
+
+// QDomAttr att1 = doc.createAttributeNS( "do:whatever:you:like", "x:attname");
+// att1.setValue( "value" );
+// prop.setAttributeNodeNS( att1 );
+// root.setAttributeNodeNS( att1 );
+// set.setAttributeNode( att1 );
+// // prop.setAttributeNS ( "xmlns:b", "xmlns:b", "urn:schemas-microsoft-com:datatypes" );
+
+ ExchangeConverterCalendar::createWebDAVVisitor v;
+ v.act( doc, prop, incidence, mFormat.timeZoneId() );
+ kdDebug() << "ExchangeConverterCalendar::createWebDAVVisitor=" << doc.toString() << "\n";
+
+ return doc;
+}
+#undef domDavProperty
+#undef domProperty
+#undef domCalendarProperty
+#undef domHTTPMailProperty
+#undef domMailHeaderProperty