2015-12-24 17:10:02 -05:00
|
|
|
// Copyright (c) 2014-2016 Thomas Fussell
|
2015-12-24 15:51:11 -05:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, WRISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE
|
|
|
|
//
|
|
|
|
// @license: http://www.opensource.org/licenses/mit-license.php
|
|
|
|
// @author: see AUTHORS file
|
2015-11-19 22:54:54 -05:00
|
|
|
#include <cmath>
|
|
|
|
#include <ctime>
|
|
|
|
|
|
|
|
#include <xlnt/utils/date.hpp>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
std::tm safe_localtime(std::time_t raw_time)
|
|
|
|
{
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
std::tm result;
|
|
|
|
localtime_s(&result, &raw_time);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
#else
|
|
|
|
return *localtime(&raw_time);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace xlnt {
|
|
|
|
|
2015-12-24 15:51:11 -05:00
|
|
|
date::date(int year_, int month_, int day_) : year(year_), month(month_), day(day_)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-11-19 22:54:54 -05:00
|
|
|
date date::from_number(int days_since_base_year, calendar base_date)
|
|
|
|
{
|
|
|
|
date result(0, 0, 0);
|
|
|
|
|
|
|
|
if (base_date == calendar::mac_1904)
|
|
|
|
{
|
|
|
|
days_since_base_year += 1462;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (days_since_base_year == 60)
|
|
|
|
{
|
|
|
|
result.day = 29;
|
|
|
|
result.month = 2;
|
|
|
|
result.year = 1900;
|
2016-07-16 19:40:20 -04:00
|
|
|
|
|
|
|
return result;
|
2015-11-19 22:54:54 -05:00
|
|
|
}
|
|
|
|
else if (days_since_base_year < 60)
|
|
|
|
{
|
|
|
|
days_since_base_year++;
|
|
|
|
}
|
|
|
|
|
|
|
|
int l = days_since_base_year + 68569 + 2415019;
|
|
|
|
int n = int((4 * l) / 146097);
|
|
|
|
l = l - int((146097 * n + 3) / 4);
|
|
|
|
int i = int((4000 * (l + 1)) / 1461001);
|
|
|
|
l = l - int((1461 * i) / 4) + 31;
|
|
|
|
int j = int((80 * l) / 2447);
|
|
|
|
result.day = l - int((2447 * j) / 80);
|
|
|
|
l = int(j / 11);
|
|
|
|
result.month = j + 2 - (12 * l);
|
|
|
|
result.year = 100 * (n - 49) + i + l;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool date::operator==(const date &comparand) const
|
|
|
|
{
|
|
|
|
return year == comparand.year && month == comparand.month && day == comparand.day;
|
|
|
|
}
|
|
|
|
|
|
|
|
int date::to_number(calendar base_date) const
|
|
|
|
{
|
|
|
|
if (day == 29 && month == 2 && year == 1900)
|
|
|
|
{
|
|
|
|
return 60;
|
|
|
|
}
|
|
|
|
|
|
|
|
int days_since_1900 = int((1461 * (year + 4800 + int((month - 14) / 12))) / 4) +
|
|
|
|
int((367 * (month - 2 - 12 * ((month - 14) / 12))) / 12) -
|
|
|
|
int((3 * (int((year + 4900 + int((month - 14) / 12)) / 100))) / 4) + day - 2415019 - 32075;
|
|
|
|
|
|
|
|
if (days_since_1900 <= 60)
|
|
|
|
{
|
|
|
|
days_since_1900--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (base_date == calendar::mac_1904)
|
|
|
|
{
|
|
|
|
return days_since_1900 - 1462;
|
|
|
|
}
|
|
|
|
|
|
|
|
return days_since_1900;
|
|
|
|
}
|
|
|
|
|
|
|
|
date date::today()
|
|
|
|
{
|
|
|
|
std::tm now = safe_localtime(std::time(0));
|
|
|
|
return date(1900 + now.tm_year, now.tm_mon + 1, now.tm_mday);
|
|
|
|
}
|
|
|
|
|
2016-07-20 19:04:44 -04:00
|
|
|
int date::weekday() const
|
|
|
|
{
|
|
|
|
auto year_temp = (month == 1 || month == 2) ? year - 1 : year;
|
|
|
|
auto month_temp = month == 1 ? 13 : month == 2 ? 14 : month;
|
|
|
|
auto day_temp = day + 1;
|
|
|
|
|
|
|
|
auto days = day_temp + static_cast<int>(13 * (month_temp + 1) / 5.0)
|
|
|
|
+ (year_temp % 100) + static_cast<int>((year_temp % 100) / 4.0);
|
|
|
|
auto gregorian = days + static_cast<int>(year_temp / 400.0) - 2 * year_temp / 100;
|
|
|
|
auto julian = days + 5 - year_temp / 100;
|
|
|
|
|
|
|
|
int weekday = (year_temp > 1582 ? gregorian : julian) % 7;
|
|
|
|
|
|
|
|
return weekday == 0 ? 7 : weekday;
|
|
|
|
}
|
|
|
|
|
2015-11-19 22:54:54 -05:00
|
|
|
} // namespace xlnt
|