kde-playground/kdepimlibs/kalarmcal/kacalendar.cpp
2015-04-14 21:49:29 +00:00

525 lines
17 KiB
C++

/*
* kacalendar.cpp - KAlarm kcal library calendar and event functions
* This file is part of kalarmcal library, which provides access to KAlarm
* calendar data.
* Copyright © 2001-2013 by David Jarvie <djarvie@kde.org>
*
* 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 "kacalendar.h"
#include "kaevent.h"
#include "version.h"
#ifndef KALARMCAL_USE_KRESOURCES
#include "collectionattribute.h"
#include <kcalcore/event.h>
#include <kcalcore/alarm.h>
#include <kcalcore/memorycalendar.h>
#include <kmessagebox.h>
#else
#include <kcal/event.h>
#include <kcal/alarm.h>
#include <kcal/calendarlocal.h>
#endif
#include <kglobal.h>
#include <klocalizedstring.h>
#include <kdebug.h>
#include <QMap>
#include <QFile>
#include <QFileInfo>
#include <QTextStream>
#ifndef KALARMCAL_USE_KRESOURCES
using namespace KCalCore;
using Akonadi::Collection;
#else
using namespace KCal;
#endif
static const KCatalogLoader loader(QLatin1String("libkalarmcal"));
namespace KAlarmCal
{
#ifndef KALARMCAL_USE_KRESOURCES
const QLatin1String MIME_BASE("application/x-vnd.kde.alarm");
const QLatin1String MIME_ACTIVE("application/x-vnd.kde.alarm.active");
const QLatin1String MIME_ARCHIVED("application/x-vnd.kde.alarm.archived");
const QLatin1String MIME_TEMPLATE("application/x-vnd.kde.alarm.template");
#endif
static const QByteArray VERSION_PROPERTY("VERSION"); // X-KDE-KALARM-VERSION VCALENDAR property
static bool isUTC(const QString& localFile);
class Private
{
public:
#ifndef KALARMCAL_USE_KRESOURCES
static int readKAlarmVersion(const FileStorage::Ptr&, QString& subVersion, QString& versionString);
#else
static int readKAlarmVersion(CalendarLocal&, const QString& localFile, QString& subVersion, QString& versionString);
#endif
static QByteArray mIcalProductId;
};
QByteArray Private::mIcalProductId;
//=============================================================================
namespace KACalendar
{
const QByteArray APPNAME("KALARM");
void setProductId(const QByteArray& progName, const QByteArray& progVersion)
{
Private::mIcalProductId = QByteArray("-//K Desktop Environment//NONSGML " + progName + " " + progVersion + "//EN");
}
QByteArray icalProductId()
{
return Private::mIcalProductId.isEmpty() ? QByteArray("-//K Desktop Environment//NONSGML //EN") : Private::mIcalProductId;
}
/******************************************************************************
* Set the X-KDE-KALARM-VERSION property in a calendar.
*/
#ifndef KALARMCAL_USE_KRESOURCES
void setKAlarmVersion(const Calendar::Ptr& calendar)
{
calendar->setCustomProperty(APPNAME, VERSION_PROPERTY, QString::fromLatin1(KAEvent::currentCalendarVersionString()));
}
#else
void setKAlarmVersion(CalendarLocal& calendar)
{
calendar.setCustomProperty(APPNAME, VERSION_PROPERTY, QString::fromLatin1(KAEvent::currentCalendarVersionString()));
}
#endif
/******************************************************************************
* Check the version of KAlarm which wrote a calendar file, and convert it in
* memory to the current KAlarm format if possible. The storage file is not
* updated. The compatibility of the calendar format is indicated by the return
* value.
*/
#ifndef KALARMCAL_USE_KRESOURCES
int updateVersion(const FileStorage::Ptr& fileStorage, QString& versionString)
#else
int updateVersion(CalendarLocal& calendar, const QString& localFile, QString& versionString)
#endif
{
QString subVersion;
#ifndef KALARMCAL_USE_KRESOURCES
const int version = Private::readKAlarmVersion(fileStorage, subVersion, versionString);
#else
const int version = Private::readKAlarmVersion(calendar, localFile, subVersion, versionString);
#endif
if (version == CurrentFormat)
return CurrentFormat; // calendar is in the current KAlarm format
if (version == IncompatibleFormat || version > KAEvent::currentCalendarVersion())
return IncompatibleFormat; // calendar was created by another program, or an unknown version of KAlarm
// Calendar was created by an earlier version of KAlarm.
// Convert it to the current format.
#ifndef KALARMCAL_USE_KRESOURCES
const QString localFile = fileStorage->fileName();
#endif
int ver = version;
if (version == KAlarmCal::Version(0,5,7) && !localFile.isEmpty())
{
// KAlarm version 0.5.7 - check whether times are stored in UTC, in which
// case it is the KDE 3.0.0 version, which needs adjustment of summer times.
if (isUTC(localFile))
ver = -version;
kDebug() << "KAlarm version 0.5.7 (" << (ver < 0 ? "" : "non-") << "UTC)";
}
else
kDebug() << "KAlarm version" << version;
// Convert events to current KAlarm format for when/if the calendar is saved
#ifndef KALARMCAL_USE_KRESOURCES
KAEvent::convertKCalEvents(fileStorage->calendar(), ver);
#else
KAEvent::convertKCalEvents(calendar, ver);
#endif
return version;
}
} // namespace KACalendar
/******************************************************************************
* Return the KAlarm version which wrote the calendar which has been loaded.
* The format is, for example, 000507 for 0.5.7.
* Reply = CurrentFormat if the calendar was created by the current version of KAlarm
* = IncompatibleFormat if it was created by KAlarm pre-0.3.5, or another program
* = version number if created by another KAlarm version.
*/
#ifndef KALARMCAL_USE_KRESOURCES
int Private::readKAlarmVersion(const FileStorage::Ptr& fileStorage, QString& subVersion, QString& versionString)
#else
int Private::readKAlarmVersion(CalendarLocal& calendar, const QString& localFile, QString& subVersion, QString& versionString)
#endif
{
subVersion.clear();
#ifndef KALARMCAL_USE_KRESOURCES
Calendar::Ptr calendar = fileStorage->calendar();
versionString = calendar->customProperty(KACalendar::APPNAME, VERSION_PROPERTY);
kDebug() << "File=" << fileStorage->fileName() << ", version=" << versionString;
#else
versionString = calendar.customProperty(KACalendar::APPNAME, VERSION_PROPERTY);
#endif
if (versionString.isEmpty())
{
// Pre-KAlarm 1.4 defined the KAlarm version number in the PRODID field.
// If another application has written to the file, this may not be present.
#ifndef KALARMCAL_USE_KRESOURCES
const QString prodid = calendar->productId();
#else
const QString prodid = calendar.productId();
#endif
if (prodid.isEmpty())
{
// Check whether the calendar file is empty, in which case
// it can be written to freely.
#ifndef KALARMCAL_USE_KRESOURCES
const QFileInfo fi(fileStorage->fileName());
#else
const QFileInfo fi(localFile);
#endif
if (!fi.size())
return KACalendar::CurrentFormat;
}
// Find the KAlarm identifier
QString progname = QLatin1String(" KAlarm ");
int i = prodid.indexOf(progname, 0, Qt::CaseInsensitive);
if (i < 0)
{
// Older versions used KAlarm's translated name in the product ID, which
// could have created problems using a calendar in different locales.
progname = QLatin1String(" ") + i18n("KAlarm") + QLatin1Char(' ');
i = prodid.indexOf(progname, 0, Qt::CaseInsensitive);
if (i < 0)
return KACalendar::IncompatibleFormat; // calendar wasn't created by KAlarm
}
// Extract the KAlarm version string
versionString = prodid.mid(i + progname.length()).trimmed();
i = versionString.indexOf(QLatin1Char('/'));
const int j = versionString.indexOf(QLatin1Char(' '));
if (j >= 0 && j < i)
i = j;
if (i <= 0)
return KACalendar::IncompatibleFormat; // missing version string
versionString = versionString.left(i); // 'versionString' now contains the KAlarm version string
}
if (versionString == QLatin1String(KAEvent::currentCalendarVersionString()))
return KACalendar::CurrentFormat; // the calendar is in the current KAlarm format
const int ver = KAlarmCal::getVersionNumber(versionString, &subVersion);
if (ver == KAEvent::currentCalendarVersion())
return KACalendar::CurrentFormat; // the calendar is in the current KAlarm format
return KAlarmCal::getVersionNumber(versionString, &subVersion);
}
/******************************************************************************
* Check whether the calendar file has its times stored as UTC times,
* indicating that it was written by the KDE 3.0.0 version of KAlarm 0.5.7.
* Reply = true if times are stored in UTC
* = false if the calendar is a vCalendar, times are not UTC, or any error occurred.
*/
bool isUTC(const QString& localFile)
{
// Read the calendar file into a string
QFile file(localFile);
if (!file.open(QIODevice::ReadOnly))
return false;
QTextStream ts(&file);
ts.setCodec("ISO 8859-1");
const QByteArray text = ts.readAll().toLocal8Bit();
file.close();
// Extract the CREATED property for the first VEVENT from the calendar
const QByteArray BEGIN_VCALENDAR("BEGIN:VCALENDAR");
const QByteArray BEGIN_VEVENT("BEGIN:VEVENT");
const QByteArray CREATED("CREATED:");
const QList<QByteArray> lines = text.split('\n');
for (int i = 0, end = lines.count(); i < end; ++i)
{
if (lines[i].startsWith(BEGIN_VCALENDAR))
{
while (++i < end)
{
if (lines[i].startsWith(BEGIN_VEVENT))
{
while (++i < end)
{
if (lines[i].startsWith(CREATED))
return lines[i].endsWith('Z');
}
}
}
break;
}
}
return false;
}
//=============================================================================
namespace CalEvent
{
// Struct to contain static strings, to allow use of K_GLOBAL_STATIC
// to delete them on program termination.
struct StaticStrings
{
StaticStrings()
: STATUS_PROPERTY("TYPE"),
ACTIVE_STATUS(QLatin1String("ACTIVE")),
TEMPLATE_STATUS(QLatin1String("TEMPLATE")),
ARCHIVED_STATUS(QLatin1String("ARCHIVED")),
DISPLAYING_STATUS(QLatin1String("DISPLAYING")),
ARCHIVED_UID(QLatin1String("-exp-")),
DISPLAYING_UID(QLatin1String("-disp-")),
TEMPLATE_UID(QLatin1String("-tmpl-"))
{}
// Event custom properties.
// Note that all custom property names are prefixed with X-KDE-KALARM- in the calendar file.
const QByteArray STATUS_PROPERTY; // X-KDE-KALARM-TYPE property
const QString ACTIVE_STATUS;
const QString TEMPLATE_STATUS;
const QString ARCHIVED_STATUS;
const QString DISPLAYING_STATUS;
// Event ID identifiers
const QString ARCHIVED_UID;
const QString DISPLAYING_UID;
// Old KAlarm format identifiers
const QString TEMPLATE_UID;
};
K_GLOBAL_STATIC(StaticStrings, staticStrings)
/******************************************************************************
* Convert a unique ID to indicate that the event is in a specified calendar file.
*/
QString uid(const QString& id, Type status)
{
QString result = id;
Type oldType;
int i, len;
if ((i = result.indexOf(staticStrings->ARCHIVED_UID)) > 0)
{
oldType = ARCHIVED;
len = staticStrings->ARCHIVED_UID.length();
}
else if ((i = result.indexOf(staticStrings->DISPLAYING_UID)) > 0)
{
oldType = DISPLAYING;
len = staticStrings->DISPLAYING_UID.length();
}
else
{
oldType = ACTIVE;
i = result.lastIndexOf(QLatin1Char('-'));
len = 1;
if (i < 0)
{
i = result.length();
len = 0;
}
else
len = 1;
}
if (status != oldType && i > 0)
{
QString part;
switch (status)
{
case ARCHIVED: part = staticStrings->ARCHIVED_UID; break;
case DISPLAYING: part = staticStrings->DISPLAYING_UID; break;
case ACTIVE:
case TEMPLATE:
case EMPTY:
default: part = QLatin1String("-"); break;
}
result.replace(i, len, part);
}
return result;
}
/******************************************************************************
* Check an event to determine its type - active, archived, template or empty.
* The default type is active if it contains alarms and there is nothing to
* indicate otherwise.
* Note that the mere fact that all an event's alarms have passed does not make
* an event archived, since it may be that they have not yet been able to be
* triggered. They will be archived once KAlarm tries to handle them.
* Do not call this function for the displaying alarm calendar.
*/
#ifndef KALARMCAL_USE_KRESOURCES
Type status(const Event::Ptr& event, QString* param)
#else
Type status(const Event* event, QString* param)
#endif
{
// Set up a static quick lookup for type strings
typedef QMap<QString, Type> PropertyMap;
static PropertyMap properties;
if (properties.isEmpty())
{
properties[staticStrings->ACTIVE_STATUS] = ACTIVE;
properties[staticStrings->TEMPLATE_STATUS] = TEMPLATE;
properties[staticStrings->ARCHIVED_STATUS] = ARCHIVED;
properties[staticStrings->DISPLAYING_STATUS] = DISPLAYING;
}
if (param)
param->clear();
if (!event)
return EMPTY;
const Alarm::List alarms = event->alarms();
if (alarms.isEmpty())
return EMPTY;
const QString property = event->customProperty(KACalendar::APPNAME, staticStrings->STATUS_PROPERTY);
if (!property.isEmpty())
{
// There's a X-KDE-KALARM-TYPE property.
// It consists of the event type, plus an optional parameter.
PropertyMap::ConstIterator it = properties.constFind(property);
if (it != properties.constEnd())
return it.value();
const int i = property.indexOf(QLatin1Char(';'));
if (i < 0)
return EMPTY;
it = properties.constFind(property.left(i));
if (it == properties.constEnd())
return EMPTY;
if (param)
*param = property.mid(i + 1);
return it.value();
}
// The event either wasn't written by KAlarm, or was written by a pre-2.0 version.
// Check first for an old KAlarm format, which indicated the event type in its UID.
const QString uid = event->uid();
if (uid.indexOf(staticStrings->ARCHIVED_UID) > 0)
return ARCHIVED;
if (uid.indexOf(staticStrings->TEMPLATE_UID) > 0)
return TEMPLATE;
// Otherwise, assume it's an active alarm
return ACTIVE;
}
/******************************************************************************
* Set the event's type - active, archived, template, etc.
* If a parameter is supplied, it will be appended as a second parameter to the
* custom property.
*/
#ifndef KALARMCAL_USE_KRESOURCES
void setStatus(const Event::Ptr& event, Type status, const QString& param)
#else
void setStatus(Event* event, Type status, const QString& param)
#endif
{
if (!event)
return;
QString text;
switch (status)
{
case ACTIVE: text = staticStrings->ACTIVE_STATUS; break;
case TEMPLATE: text = staticStrings->TEMPLATE_STATUS; break;
case ARCHIVED: text = staticStrings->ARCHIVED_STATUS; break;
case DISPLAYING: text = staticStrings->DISPLAYING_STATUS; break;
default:
event->removeCustomProperty(KACalendar::APPNAME, staticStrings->STATUS_PROPERTY);
return;
}
if (!param.isEmpty())
text += QLatin1Char(';') + param;
event->setCustomProperty(KACalendar::APPNAME, staticStrings->STATUS_PROPERTY, text);
}
#ifndef KALARMCAL_USE_KRESOURCES
Type type(const QString& mimeType)
{
if (mimeType == MIME_ACTIVE)
return ACTIVE;
if (mimeType == MIME_ARCHIVED)
return ARCHIVED;
if (mimeType == MIME_TEMPLATE)
return TEMPLATE;
return EMPTY;
}
Types types(const QStringList& mimeTypes)
{
Types types = 0;
foreach (const QString& type, mimeTypes)
{
if (type == MIME_ACTIVE)
types |= ACTIVE;
if (type == MIME_ARCHIVED)
types |= ARCHIVED;
if (type == MIME_TEMPLATE)
types |= TEMPLATE;
}
return types;
}
QString mimeType(Type type)
{
switch (type)
{
case ACTIVE: return MIME_ACTIVE;
case ARCHIVED: return MIME_ARCHIVED;
case TEMPLATE: return MIME_TEMPLATE;
default: return QString();
}
}
QStringList mimeTypes(Types types)
{
QStringList mimes;
for (int i = 1; types; i <<= 1)
{
if (types & i)
{
mimes += mimeType(Type(i));
types &= ~i;
}
}
return mimes;
}
#endif
} // namespace CalEvent
} // namespace KAlarmCal
// vim: et sw=4: