/* This file is part of libkcal. Copyright (c) 2002 Cornelius Schumacher <schumacher@kde.org> Copyright (C) 2003-2004 Reinhold Kainhofer <reinhold@kainhofer.com> 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 "compat.h" #include <kdebug.h> #include <tqregexp.h> #include "incidence.h" using namespace KCal; Compat *CompatFactory::createCompat( const TQString &productId ) { // kdDebug(5800) << "CompatFactory::createCompat(): '" << productId << "'" // << endl; Compat *compat = 0; int korg = productId.find( "KOrganizer" ); int outl9 = productId.find( "Outlook 9.0" ); // int kcal = productId.find( "LibKCal" ); // TODO: Use the version of LibKCal to determine the compat class... if ( korg >= 0 ) { int versionStart = productId.find( " ", korg ); if ( versionStart >= 0 ) { int versionStop = productId.find( TQRegExp( "[ /]" ), versionStart + 1 ); if ( versionStop >= 0 ) { TQString version = productId.mid( versionStart + 1, versionStop - versionStart - 1 ); // kdDebug(5800) << "Found KOrganizer version: " << version << endl; int versionNum = version.section( ".", 0, 0 ).toInt() * 10000 + version.section( ".", 1, 1 ).toInt() * 100 + version.section( ".", 2, 2 ).toInt(); int releaseStop = productId.find( "/", versionStop ); TQString release; if ( releaseStop > versionStop ) { release = productId.mid( versionStop+1, releaseStop-versionStop-1 ); } // kdDebug(5800) << "KOrganizer release: \"" << release << "\"" << endl; // kdDebug(5800) << "Numerical version: " << versionNum << endl; if ( versionNum < 30100 ) { compat = new CompatPre31; } else if ( versionNum < 30200 ) { compat = new CompatPre32; } else if ( versionNum == 30200 && release == "pre" ) { kdDebug(5800) << "Generating compat for KOrganizer 3.2 pre " << endl; compat = new Compat32PrereleaseVersions; } else if ( versionNum < 30400 ) { compat = new CompatPre34; } else if ( versionNum < 30500 ) { compat = new CompatPre35; } } } } else if ( outl9 >= 0 ) { kdDebug(5800) << "Generating compat for Outlook < 2000 (Outlook 9.0)" << endl; compat = new CompatOutlook9; } if ( !compat ) compat = new Compat; return compat; } void Compat::fixEmptySummary( Incidence *incidence ) { // some stupid vCal exporters ignore the standard and use Description // instead of Summary for the default field. Correct for this: Copy the // first line of the description to the summary (if summary is just one // line, move it) if (incidence->summary().isEmpty() && !(incidence->description().isEmpty())) { TQString oldDescription = incidence->description().stripWhiteSpace(); TQString newSummary( oldDescription ); newSummary.remove( TQRegExp("\n.*") ); incidence->setSummary( newSummary ); if ( oldDescription == newSummary ) incidence->setDescription(""); } } void Compat::fixRecurrence( Incidence */*incidence*/ ) { // Prevent use of compatibility mode during subsequent changes by the application // incidence->recurrence()->setCompatVersion(); } /** Before kde 3.5, the start date was not automatically a recurring date. So if the start date doesn't match the recurrence rule, we need to add an ex date for the date start. If a duration was given, the DTSTART was only counted if it matched, so by accident this was already the correct behavior, so we don't need to adjust the duration... */ void CompatPre35::fixRecurrence( Incidence *incidence ) { Recurrence* recurrence = incidence->recurrence(); if (recurrence ) { TQDateTime start( incidence->dtStart() ); // kde < 3.5 only had one rrule, so no need to loop over all RRULEs. RecurrenceRule *r = recurrence->defaultRRule(); if ( r && !r->dateMatchesRules( start ) ) { recurrence->addExDateTime( start ); } } // Call base class method now that everything else is done Compat::fixRecurrence( incidence ); } int CompatPre34::fixPriority( int prio ) { if ( 0<prio && prio<6 ) { // adjust 1->1, 2->3, 3->5, 4->7, 5->9 return 2*prio - 1; } else return prio; } /** The recurrence has a specified number of repetitions. Pre-3.2, this was extended by the number of exception dates. This is also rfc 2445-compliant. The duration of an RRULE also counts events that are later excluded via EXDATE or EXRULE. */ void CompatPre32::fixRecurrence( Incidence *incidence ) { Recurrence* recurrence = incidence->recurrence(); if ( recurrence->doesRecur() && recurrence->duration() > 0 ) { recurrence->setDuration( recurrence->duration() + incidence->recurrence()->exDates().count() ); } // Call base class method now that everything else is done CompatPre35::fixRecurrence( incidence ); } /** Before kde 3.1, floating events (events without a date) had 0:00 of their last day as the end date. E.g. 28.5.2005 0:00 until 28.5.2005 0:00 for an event that lasted the whole day on May 28, 2005. According to RFC 2445, the end date for such an event needs to be 29.5.2005 0:00. Update: We misunderstood rfc 2445 in this regard. For all-day events, the DTEND is the last day of the event. See a mail from the Author or rfc 2445: http://www.imc.org/ietf-calendar/archive1/msg03648.html However, as all other applications also got this wrong, we'll just leave it as it is and use the wrong interpretation (was also discussed on ietf-calsify)*/ void CompatPre31::fixFloatingEnd( TQDate &endDate ) { endDate = endDate.addDays( 1 ); } void CompatPre31::fixRecurrence( Incidence *incidence ) { CompatPre32::fixRecurrence( incidence ); Recurrence *recur = incidence->recurrence(); RecurrenceRule *r = 0; if ( recur ) r = recur->defaultRRule(); if ( recur && r ) { int duration = r->duration(); if ( duration > 0 ) { // Backwards compatibility for KDE < 3.1. // rDuration was set to the number of time periods to recur, // with week start always on a Monday. // Convert this to the number of occurrences. r->setDuration( -1 ); TQDate end( r->startDt().date() ); bool doNothing = false; // # of periods: int tmp = ( duration - 1 ) * r->frequency(); switch ( r->recurrenceType() ) { case RecurrenceRule::rWeekly: { end = end.addDays( tmp * 7 + 7 - end.dayOfWeek() ); break; } case RecurrenceRule::rMonthly: { int month = end.month() - 1 + tmp; end.setYMD( end.year() + month / 12, month % 12 + 1, 31 ); break; } case RecurrenceRule::rYearly: { end.setYMD( end.year() + tmp, 12, 31); break; } default: doNothing = true; break; } if ( !doNothing ) { duration = r->durationTo( TQDateTime( end, TQTime( 0, 0, 0 ) ) ); r->setDuration( duration ); } } /* addYearlyNum */ // Dates were stored as day numbers, with a fiddle to take account of leap years. // Convert the day number to a month. TQValueList<int> days = r->byYearDays(); if ( !days.isEmpty() ) { TQValueList<int> months = r->byMonths(); for ( TQValueListConstIterator<int> it = days.begin(); it != days.end(); ++it ) { int newmonth = TQDate( r->startDt().date().year(), 1, 1).addDays( (*it) - 1 ).month(); if ( !months.contains( newmonth ) ) months.append( newmonth ); } r->setByMonths( months ); days.clear(); r->setByYearDays( days ); } } } /** In Outlook 9, alarms have the wrong sign. I.e. RFC 2445 says that negative values for the trigger are before the event's start. Outlook/exchange, however used positive values. */ void CompatOutlook9::fixAlarms( Incidence *incidence ) { if ( !incidence ) return; Alarm::List alarms = incidence->alarms(); Alarm::List::Iterator it; for ( it = alarms.begin(); it != alarms.end(); ++it ) { Alarm *al = *it; if ( al && al->hasStartOffset() ) { Duration offsetDuration = al->startOffset(); int offs = offsetDuration.asSeconds(); if ( offs>0 ) offsetDuration = Duration( -offs ); al->setStartOffset( offsetDuration ); } } }