/* Copyright 2009, 2010 John Layt 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 #include // 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; }