summaryrefslogtreecommitdiff
path: root/src/times.cc
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2009-02-21 02:08:49 -0400
committerJohn Wiegley <johnw@newartisans.com>2009-02-21 02:10:02 -0400
commita93111470d54e6345617b51fcc939f306854a524 (patch)
tree8a514e7f3d696ba6134907f7296c370c55a1811e /src/times.cc
parent4ff8087815c9d93447ddb1651fa3ec8b82bba9be (diff)
downloadfork-ledger-a93111470d54e6345617b51fcc939f306854a524.tar.gz
fork-ledger-a93111470d54e6345617b51fcc939f306854a524.tar.bz2
fork-ledger-a93111470d54e6345617b51fcc939f306854a524.zip
Greatly improved the way "weeks" are iterated
Diffstat (limited to 'src/times.cc')
-rw-r--r--src/times.cc52
1 files changed, 45 insertions, 7 deletions
diff --git a/src/times.cc b/src/times.cc
index df760d10..b10d9884 100644
--- a/src/times.cc
+++ b/src/times.cc
@@ -33,6 +33,7 @@
namespace ledger {
+int start_of_week = 0;
optional<std::string> input_date_format;
std::string output_datetime_format = "%Y-%m-%d %H:%M:%S";
std::string output_date_format = "%Y-%m-%d";
@@ -58,9 +59,7 @@ namespace {
"%Y",
NULL
};
-}
-namespace {
bool parse_date_mask(const char * date_str, std::tm& result)
{
if (input_date_format) {
@@ -106,6 +105,27 @@ namespace {
}
}
+int string_to_day_of_week(const std::string& str)
+{
+ if (str == "sun" || str == "sunday" || str == "0")
+ return 0;
+ else if (str == "mon" || str == "monday" || str == "1")
+ return 1;
+ else if (str == "tue" || str == "tuesday" || str == "2")
+ return 2;
+ else if (str == "wed" || str == "wednesday" || str == "3")
+ return 3;
+ else if (str == "thu" || str == "thursday" || str == "4")
+ return 4;
+ else if (str == "fri" || str == "friday" || str == "5")
+ return 5;
+ else if (str == "sat" || str == "saturday" || str == "6")
+ return 6;
+
+ assert(false);
+ return -1;
+}
+
datetime_t parse_datetime(const char * str, int)
{
std::tm when;
@@ -130,7 +150,19 @@ date_t interval_t::first(const optional<date_t>& moment)
// a date early enough that the range will be correct, but late enough
// that we don't spend hundreds of thousands of loops skipping through
// time.
- begin = date_t(moment->year(), gregorian::Jan, 1);
+ assert(moment);
+
+ if (months > 0 || years > 0) {
+ begin = date_t(moment->year(), gregorian::Jan, 1);
+ } else {
+ begin = date_t(*moment - gregorian::days(400));
+
+ // jww (2009-02-21): Add support for starting a week on any day
+ if (weekly) { // move it to a Sunday
+ while (begin.day_of_week() != start_of_week)
+ begin += gregorian::days(1);
+ }
+ }
}
date_t quant(begin);
@@ -281,12 +313,14 @@ void interval_t::parse(std::istream& in)
if (word == "every") {
read_lower_word(in, word);
if (std::isdigit(word[0])) {
- int quantity = std::atol(word.c_str());
+ int quantity = lexical_cast<int>(word);
read_lower_word(in, word);
if (word == "days")
days = quantity;
- else if (word == "weeks")
+ else if (word == "weeks") {
days = 7 * quantity;
+ weekly = true;
+ }
else if (word == "months")
months = quantity;
else if (word == "quarters")
@@ -296,8 +330,10 @@ void interval_t::parse(std::istream& in)
}
else if (word == "day")
days = 1;
- else if (word == "week")
+ else if (word == "week") {
days = 7;
+ weekly = true;
+ }
else if (word == "month")
months = 1;
else if (word == "quarter")
@@ -307,8 +343,10 @@ void interval_t::parse(std::istream& in)
}
else if (word == "daily")
days = 1;
- else if (word == "weekly")
+ else if (word == "weekly") {
days = 7;
+ weekly = true;
+ }
else if (word == "biweekly")
days = 14;
else if (word == "monthly")