kdelibs/kdecore/date/kcalendarsystemgregorian.cpp

566 lines
20 KiB
C++

/*
Copyright 2009, 2010 John Layt <john@layt.net>
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.
*/
// Derived gregorian kde calendar class
#include "kcalendarsystemgregorian_p.h"
#include "kcalendarsystemgregorianprivate_p.h"
#include "kcalendarera_p.h"
#include "kdebug.h"
#include "klocale.h"
#include "kglobal.h"
#include "kconfiggroup.h"
#include <QtCore/qdatetime.h>
#include <QtCore/qstring.h>
// Shared d pointer base class definitions
KCalendarSystemGregorianPrivate::KCalendarSystemGregorianPrivate(KCalendarSystemGregorian *q)
: KCalendarSystemPrivate(q)
{
}
KCalendarSystemGregorianPrivate::~KCalendarSystemGregorianPrivate()
{
}
KLocale::CalendarSystem KCalendarSystemGregorianPrivate::calendarSystem() const
{
return KLocale::GregorianCalendar;
}
// Dummy version using Gregorian as an example
// This method MUST be re-implemented in any new Calendar System
void KCalendarSystemGregorianPrivate::loadDefaultEraList()
{
QString name, shortName, format;
KConfigGroup cg(config(), QString::fromLatin1("KCalendarSystem %1").arg(q->calendarType(q->calendarSystem())));
m_useCommonEra = cg.readEntry("UseCommonEra", false);
if (m_useCommonEra) {
name = i18nc("Calendar Era: Gregorian Common Era, years < 0, LongFormat", "Before Common Era");
shortName = i18nc("Calendar Era: Gregorian Common Era, years < 0, ShortFormat", "BCE");
} else {
name = i18nc("Calendar Era: Gregorian Christian Era, years < 0, LongFormat", "Before Christ");
shortName = i18nc("Calendar Era: Gregorian Christian Era, years < 0, ShortFormat", "BC");
}
format = i18nc("(kdedt-format) Gregorian, BC, full era year format used for %EY, e.g. 2000 BC", "%Ey %EC");
addEra('-', 1, q->epoch().addDays(-1), -1, q->earliestValidDate(), name, shortName, format);
if (m_useCommonEra) {
name = i18nc("Calendar Era: Gregorian Common Era, years > 0, LongFormat", "Common Era");
shortName = i18nc("Calendar Era: Gregorian Common Era, years > 0, ShortFormat", "CE");
} else {
name = i18nc("Calendar Era: Gregorian Christian Era, years > 0, LongFormat", "Anno Domini");
shortName = i18nc("Calendar Era: Gregorian Christian Era, years > 0, ShortFormat", "AD");
}
format = i18nc("(kdedt-format) Gregorian, AD, full era year format used for %EY, e.g. 2000 AD", "%Ey %EC");
addEra('+', 1, q->epoch(), 1, q->latestValidDate(), name, shortName, format);
}
int KCalendarSystemGregorianPrivate::monthsInYear(int year) const
{
Q_UNUSED(year)
return 12;
}
int KCalendarSystemGregorianPrivate::daysInMonth(int year, int month) const
{
if (month == 2) {
if (isLeapYear(year)) {
return 29;
} else {
return 28;
}
}
if (month == 4 || month == 6 || month == 9 || month == 11) {
return 30;
}
return 31;
}
int KCalendarSystemGregorianPrivate::daysInYear(int year) const
{
if (isLeapYear(year)) {
return 366;
} else {
return 365;
}
}
int KCalendarSystemGregorianPrivate::daysInWeek() const
{
return 7;
}
bool KCalendarSystemGregorianPrivate::isLeapYear(int year) const
{
if (!hasYearZero() && year < 1) {
year = year + 1;
}
if (year % 4 == 0) {
if (year % 100 != 0) {
return true;
} else if (year % 400 == 0) {
return true;
}
}
return false;
}
bool KCalendarSystemGregorianPrivate::hasLeapMonths() const
{
return false;
}
bool KCalendarSystemGregorianPrivate::hasYearZero() const
{
return false;
}
int KCalendarSystemGregorianPrivate::maxDaysInWeek() const
{
return 7;
}
int KCalendarSystemGregorianPrivate::maxMonthsInYear() const
{
return 12;
}
int KCalendarSystemGregorianPrivate::earliestValidYear() const
{
return -4713;
}
int KCalendarSystemGregorianPrivate::latestValidYear() const
{
return 9999;
}
QString KCalendarSystemGregorianPrivate::monthName(int month, int year, KLocale::DateTimeComponentFormat format, bool possessive) const
{
Q_UNUSED(year);
if (format == KLocale::NarrowName) {
switch (month) {
case 1:
return ki18nc("Gregorian month 1 - KLocale::NarrowName", "J").toString(locale());
case 2:
return ki18nc("Gregorian month 2 - KLocale::NarrowName", "F").toString(locale());
case 3:
return ki18nc("Gregorian month 3 - KLocale::NarrowName", "M").toString(locale());
case 4:
return ki18nc("Gregorian month 4 - KLocale::NarrowName", "A").toString(locale());
case 5:
return ki18nc("Gregorian month 5 - KLocale::NarrowName", "M").toString(locale());
case 6:
return ki18nc("Gregorian month 6 - KLocale::NarrowName", "J").toString(locale());
case 7:
return ki18nc("Gregorian month 7 - KLocale::NarrowName", "J").toString(locale());
case 8:
return ki18nc("Gregorian month 8 - KLocale::NarrowName", "A").toString(locale());
case 9:
return ki18nc("Gregorian month 9 - KLocale::NarrowName", "S").toString(locale());
case 10:
return ki18nc("Gregorian month 10 - KLocale::NarrowName", "O").toString(locale());
case 11:
return ki18nc("Gregorian month 11 - KLocale::NarrowName", "N").toString(locale());
case 12:
return ki18nc("Gregorian month 12 - KLocale::NarrowName", "D").toString(locale());
default:
return QString();
}
}
if (format == KLocale::ShortName && possessive) {
switch (month) {
case 1:
return ki18nc("Gregorian month 1 - KLocale::ShortName Possessive", "of Jan").toString(locale());
case 2:
return ki18nc("Gregorian month 2 - KLocale::ShortName Possessive", "of Feb").toString(locale());
case 3:
return ki18nc("Gregorian month 3 - KLocale::ShortName Possessive", "of Mar").toString(locale());
case 4:
return ki18nc("Gregorian month 4 - KLocale::ShortName Possessive", "of Apr").toString(locale());
case 5:
return ki18nc("Gregorian month 5 - KLocale::ShortName Possessive", "of May").toString(locale());
case 6:
return ki18nc("Gregorian month 6 - KLocale::ShortName Possessive", "of Jun").toString(locale());
case 7:
return ki18nc("Gregorian month 7 - KLocale::ShortName Possessive", "of Jul").toString(locale());
case 8:
return ki18nc("Gregorian month 8 - KLocale::ShortName Possessive", "of Aug").toString(locale());
case 9:
return ki18nc("Gregorian month 9 - KLocale::ShortName Possessive", "of Sep").toString(locale());
case 10:
return ki18nc("Gregorian month 10 - KLocale::ShortName Possessive", "of Oct").toString(locale());
case 11:
return ki18nc("Gregorian month 11 - KLocale::ShortName Possessive", "of Nov").toString(locale());
case 12:
return ki18nc("Gregorian month 12 - KLocale::ShortName Possessive", "of Dec").toString(locale());
default:
return QString();
}
}
if (format == KLocale::ShortName && !possessive) {
switch (month) {
case 1:
return ki18nc("Gregorian month 1 - KLocale::ShortName", "Jan").toString(locale());
case 2:
return ki18nc("Gregorian month 2 - KLocale::ShortName", "Feb").toString(locale());
case 3:
return ki18nc("Gregorian month 3 - KLocale::ShortName", "Mar").toString(locale());
case 4:
return ki18nc("Gregorian month 4 - KLocale::ShortName", "Apr").toString(locale());
case 5:
return ki18nc("Gregorian month 5 - KLocale::ShortName", "May").toString(locale());
case 6:
return ki18nc("Gregorian month 6 - KLocale::ShortName", "Jun").toString(locale());
case 7:
return ki18nc("Gregorian month 7 - KLocale::ShortName", "Jul").toString(locale());
case 8:
return ki18nc("Gregorian month 8 - KLocale::ShortName", "Aug").toString(locale());
case 9:
return ki18nc("Gregorian month 9 - KLocale::ShortName", "Sep").toString(locale());
case 10:
return ki18nc("Gregorian month 10 - KLocale::ShortName", "Oct").toString(locale());
case 11:
return ki18nc("Gregorian month 11 - KLocale::ShortName", "Nov").toString(locale());
case 12:
return ki18nc("Gregorian month 12 - KLocale::ShortName", "Dec").toString(locale());
default:
return QString();
}
}
if (format == KLocale::LongName && possessive) {
switch (month) {
case 1:
return ki18nc("Gregorian month 1 - KLocale::LongName Possessive", "of January").toString(locale());
case 2:
return ki18nc("Gregorian month 2 - KLocale::LongName Possessive", "of February").toString(locale());
case 3:
return ki18nc("Gregorian month 3 - KLocale::LongName Possessive", "of March").toString(locale());
case 4:
return ki18nc("Gregorian month 4 - KLocale::LongName Possessive", "of April").toString(locale());
case 5:
return ki18nc("Gregorian month 5 - KLocale::LongName Possessive", "of May").toString(locale());
case 6:
return ki18nc("Gregorian month 6 - KLocale::LongName Possessive", "of June").toString(locale());
case 7:
return ki18nc("Gregorian month 7 - KLocale::LongName Possessive", "of July").toString(locale());
case 8:
return ki18nc("Gregorian month 8 - KLocale::LongName Possessive", "of August").toString(locale());
case 9:
return ki18nc("Gregorian month 9 - KLocale::LongName Possessive", "of September").toString(locale());
case 10:
return ki18nc("Gregorian month 10 - KLocale::LongName Possessive", "of October").toString(locale());
case 11:
return ki18nc("Gregorian month 11 - KLocale::LongName Possessive", "of November").toString(locale());
case 12:
return ki18nc("Gregorian month 12 - KLocale::LongName Possessive", "of December").toString(locale());
default:
return QString();
}
}
// Default to LongName
switch (month) {
case 1:
return ki18nc("Gregorian month 1 - KLocale::LongName", "January").toString(locale());
case 2:
return ki18nc("Gregorian month 2 - KLocale::LongName", "February").toString(locale());
case 3:
return ki18nc("Gregorian month 3 - KLocale::LongName", "March").toString(locale());
case 4:
return ki18nc("Gregorian month 4 - KLocale::LongName", "April").toString(locale());
case 5:
return ki18nc("Gregorian month 5 - KLocale::LongName", "May").toString(locale());
case 6:
return ki18nc("Gregorian month 6 - KLocale::LongName", "June").toString(locale());
case 7:
return ki18nc("Gregorian month 7 - KLocale::LongName", "July").toString(locale());
case 8:
return ki18nc("Gregorian month 8 - KLocale::LongName", "August").toString(locale());
case 9:
return ki18nc("Gregorian month 9 - KLocale::LongName", "September").toString(locale());
case 10:
return ki18nc("Gregorian month 10 - KLocale::LongName", "October").toString(locale());
case 11:
return ki18nc("Gregorian month 11 - KLocale::LongName", "November").toString(locale());
case 12:
return ki18nc("Gregorian month 12 - KLocale::LongName", "December").toString(locale());
default:
return QString();
}
}
QString KCalendarSystemGregorianPrivate::weekDayName(int weekDay, KLocale::DateTimeComponentFormat format) const
{
if (format == KLocale::NarrowName) {
switch (weekDay) {
case 1:
return ki18nc("Gregorian weekday 1 - KLocale::NarrowName ", "M").toString(locale());
case 2:
return ki18nc("Gregorian weekday 2 - KLocale::NarrowName ", "T").toString(locale());
case 3:
return ki18nc("Gregorian weekday 3 - KLocale::NarrowName ", "W").toString(locale());
case 4:
return ki18nc("Gregorian weekday 4 - KLocale::NarrowName ", "T").toString(locale());
case 5:
return ki18nc("Gregorian weekday 5 - KLocale::NarrowName ", "F").toString(locale());
case 6:
return ki18nc("Gregorian weekday 6 - KLocale::NarrowName ", "S").toString(locale());
case 7:
return ki18nc("Gregorian weekday 7 - KLocale::NarrowName ", "S").toString(locale());
default:
return QString();
}
}
if (format == KLocale::ShortName || format == KLocale:: ShortNumber) {
switch (weekDay) {
case 1:
return ki18nc("Gregorian weekday 1 - KLocale::ShortName", "Mon").toString(locale());
case 2:
return ki18nc("Gregorian weekday 2 - KLocale::ShortName", "Tue").toString(locale());
case 3:
return ki18nc("Gregorian weekday 3 - KLocale::ShortName", "Wed").toString(locale());
case 4:
return ki18nc("Gregorian weekday 4 - KLocale::ShortName", "Thu").toString(locale());
case 5:
return ki18nc("Gregorian weekday 5 - KLocale::ShortName", "Fri").toString(locale());
case 6:
return ki18nc("Gregorian weekday 6 - KLocale::ShortName", "Sat").toString(locale());
case 7:
return ki18nc("Gregorian weekday 7 - KLocale::ShortName", "Sun").toString(locale());
default: return QString();
}
}
switch (weekDay) {
case 1:
return ki18nc("Gregorian weekday 1 - KLocale::LongName", "Monday").toString(locale());
case 2:
return ki18nc("Gregorian weekday 2 - KLocale::LongName", "Tuesday").toString(locale());
case 3:
return ki18nc("Gregorian weekday 3 - KLocale::LongName", "Wednesday").toString(locale());
case 4:
return ki18nc("Gregorian weekday 4 - KLocale::LongName", "Thursday").toString(locale());
case 5:
return ki18nc("Gregorian weekday 5 - KLocale::LongName", "Friday").toString(locale());
case 6:
return ki18nc("Gregorian weekday 6 - KLocale::LongName", "Saturday").toString(locale());
case 7:
return ki18nc("Gregorian weekday 7 - KLocale::LongName", "Sunday").toString(locale());
default:
return QString();
}
}
KCalendarSystemGregorian::KCalendarSystemGregorian(const KLocale *locale)
: KCalendarSystem(*new KCalendarSystemGregorianPrivate(this), KSharedConfig::Ptr(), locale)
{
d_ptr->loadConfig(calendarType());
}
KCalendarSystemGregorian::KCalendarSystemGregorian(const KSharedConfig::Ptr config,
const KLocale *locale)
: KCalendarSystem(*new KCalendarSystemGregorianPrivate(this), config, locale)
{
d_ptr->loadConfig(calendarType());
}
KCalendarSystemGregorian::KCalendarSystemGregorian(KCalendarSystemGregorianPrivate &dd,
const KSharedConfig::Ptr config,
const KLocale *locale)
: KCalendarSystem(dd, config, locale)
{
d_ptr->loadConfig(calendarType());
}
KCalendarSystemGregorian::~KCalendarSystemGregorian()
{
}
QString KCalendarSystemGregorian::calendarType() const
{
return QLatin1String("gregorian-proleptic");
}
QDate KCalendarSystemGregorian::epoch() const
{
return QDate::fromJulianDay(1721426);
}
QDate KCalendarSystemGregorian::earliestValidDate() const
{
// Gregorian 1 Jan 4713 BC, no year zero
return QDate::fromJulianDay(38);
}
QDate KCalendarSystemGregorian::latestValidDate() const
{
// Set to last day of year 9999 until confirm date formats & widgets support > 9999
// In Gregorian this is 9999-12-31, which is is jd 5373484
// Can't call setDate( 9999, 12, 31 ) as it creates circular reference!
return QDate::fromJulianDay(5373484);
}
bool KCalendarSystemGregorian::isValid(int year, int month, int day) const
{
return KCalendarSystem::isValid(year, month, day);
}
bool KCalendarSystemGregorian::isValid(const QDate &date) const
{
return KCalendarSystem::isValid(date);
}
bool KCalendarSystemGregorian::isLeapYear(int year) const
{
return KCalendarSystem::isLeapYear(year);
}
bool KCalendarSystemGregorian::isLeapYear(const QDate &date) const
{
return KCalendarSystem::isLeapYear(date);
}
QString KCalendarSystemGregorian::monthName(int month, int year, MonthNameFormat format) const
{
return KCalendarSystem::monthName(month, year, format);
}
QString KCalendarSystemGregorian::monthName(const QDate &date, MonthNameFormat format) const
{
return KCalendarSystem::monthName(date, format);
}
QString KCalendarSystemGregorian::weekDayName(int weekDay, WeekDayNameFormat format) const
{
return KCalendarSystem::weekDayName(weekDay, format);
}
QString KCalendarSystemGregorian::weekDayName(const QDate &date, WeekDayNameFormat format) const
{
return KCalendarSystem::weekDayName(date, format);
}
int KCalendarSystemGregorian::yearStringToInteger(const QString &sNum, int &iLength) const
{
return KCalendarSystem::yearStringToInteger(sNum, iLength);
}
int KCalendarSystemGregorian::weekDayOfPray() const
{
return 7; // sunday
}
bool KCalendarSystemGregorian::isLunar() const
{
return false;
}
bool KCalendarSystemGregorian::isLunisolar() const
{
return false;
}
bool KCalendarSystemGregorian::isSolar() const
{
return true;
}
bool KCalendarSystemGregorian::isProleptic() const
{
return true;
}
bool KCalendarSystemGregorian::julianDayToDate(int jd, int &year, int &month, int &day) const
{
Q_D(const KCalendarSystemGregorian);
// Formula from The Calendar FAQ by Claus Tondering
// http://www.tondering.dk/claus/cal/node3.html#SECTION003161000000000000000
// NOTE: Coded from scratch from mathematical formulas, not copied from
// the Boost licensed source code
int a = jd + 32044;
int b = ((4 * a) + 3) / 146097;
int c = a - ((146097 * b) / 4);
int dd = ((4 * c) + 3) / 1461;
int e = c - ((1461 * dd) / 4);
int m = ((5 * e) + 2) / 153;
day = e - (((153 * m) + 2) / 5) + 1;
month = m + 3 - (12 * (m / 10));
year = (100 * b) + dd - 4800 + (m / 10);
// If year is -ve then is BC. In Gregorian there is no year 0, but the maths
// is easier if we pretend there is, so internally year of 0 = 1BC = -1 outside
// Check for Year 0 support as some Gregorian based calendars do have it, e.g. Thai and ISO
if (!d->hasYearZero() && year < 1) {
year = year - 1;
}
return true;
}
bool KCalendarSystemGregorian::dateToJulianDay(int year, int month, int day, int &jd) const
{
Q_D(const KCalendarSystemGregorian);
// Formula from The Calendar FAQ by Claus Tondering
// http://www.tondering.dk/claus/cal/node3.html#SECTION003161000000000000000
// NOTE: Coded from scratch from mathematical formulas, not copied from
// the Boost licensed source code
// If year is -ve then is BC. In Gregorian there is no year 0, but the maths
// is easier if we pretend there is, so internally year of -1 = 1BC = 0 internally
// Check for Year 0 support as some Gregorian based calendars do have it, e.g. Thai and ISO
int y;
if (!d->hasYearZero() && year < 1) {
y = year + 1;
} else {
y = year;
}
int a = (14 - month) / 12;
y = y + 4800 - a;
int m = month + (12 * a) - 3;
jd = day
+ (((153 * m) + 2) / 5)
+ (365 * y)
+ (y / 4)
- (y / 100)
+ (y / 400)
- 32045;
return true;
}