gnucash/libgnucash/app-utils/date-utilities.scm
2018-03-25 09:36:21 -07:00

1073 lines
34 KiB
Scheme

;; date-utilities.scm -- date utility functions.
;; Bryan Larsen (blarsen@ada-works.com)
;; Revised by Christopher Browne
;; Improvement to financial year support by Yves-Eric Martin
;;
;; This program is free software; you can redistribute it and/or
;; modify it under the terms of the GNU General Public License as
;; published by the Free Software Foundation; either version 2 of
;; the License, or (at your option) any later version.
;;
;; This program 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 General Public License for more details.
;;
;; You should have received a copy of the GNU General Public License
;; along with this program; if not, contact:
;;
;; Free Software Foundation Voice: +1-617-542-5942
;; 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652
;; Boston, MA 02110-1301, USA gnu@gnu.org
(use-modules (gnucash core-utils)
(gnucash gettext))
(define gnc:reldate-list '())
;; get stuff from localtime date vector
(define (gnc:date-get-year datevec)
(+ 1900 (tm:year datevec)))
(define (gnc:date-get-quarter datevec)
(+ (quotient (tm:mon datevec) 3) 1))
(define (gnc:date-get-month-day datevec)
(tm:mday datevec))
;; get month with january==1
(define (gnc:date-get-month datevec)
(+ (tm:mon datevec) 1))
(define (gnc:date-get-week-day datevec)
(+ (tm:wday datevec) 1))
;; jan 1 == 1
(define (gnc:date-get-week datevec)
(gnc:date-to-week (gnc:time64-start-day-time
(gnc-mktime datevec))))
(define (gnc:date-get-year-day datevec)
(+ (tm:yday datevec) 1))
(define (gnc:time64-get-year t64)
(gnc:date-get-year (gnc-localtime t64)))
(define (gnc:time64-get-quarter t64)
(gnc:date-get-quarter (gnc-localtime t64)))
(define (gnc:time64-get-month-day t64)
(gnc:date-get-month-day (gnc-localtime t64)))
(define (gnc:time64-get-month t64)
(gnc:date-get-month (gnc-localtime t64)))
(define (gnc:time64-get-week-day t64)
(gnc:date-get-week-day (gnc-localtime t64)))
(define (gnc:time64-get-week t64)
(gnc:date-get-week (gnc-localtime t64)))
(define (gnc:time64-get-year-day t64)
(gnc:date-get-year-day (gnc-localtime t64)))
(define (gnc:date-get-year-string datevec)
(gnc-locale-to-utf8 (strftime "%Y" datevec)))
(define (gnc:date-get-quarter-string datevec)
(format #f "Q~d" (gnc:date-get-quarter datevec)))
(define (gnc:date-get-quarter-year-string datevec)
(string-append
(gnc:date-get-quarter-string datevec)
" "
(gnc:date-get-year-string datevec)))
(define (gnc:date-get-month-string datevec)
(gnc-locale-to-utf8 (strftime "%B" datevec)))
(define (gnc:date-get-month-year-string datevec)
(gnc-locale-to-utf8 (strftime "%B %Y" datevec)))
(define (gnc:date-get-week-year-string datevec)
(let* ((beginweekt64 (* (gnc:time64-get-week
(gnc-mktime datevec))
604800))
(begin-string (qof-print-date (+ beginweekt64 345600)))
(end-string (qof-print-date (+ beginweekt64 864000))))
(format #f (_ "~a to ~a") begin-string end-string)))
;; is leap year?
(define (gnc:leap-year? year)
(or (and (zero? (remainder year 4))
(not (zero? (remainder year 100))))
(zero? (remainder year 400))))
;; number of days in year
(define (gnc:days-in-year year)
(if (gnc:leap-year? year) 366 365))
;; number of days in month
(define (gnc:days-in-month month year)
(case month
((1 3 5 7 8 10 12) 31)
((4 6 9 11) 30)
((2) (if (gnc:leap-year? year) 29 28))))
;; convert a date in seconds since 1970 into # of years since 1970 as
;; a fraction.
(define (gnc:date-to-year-fraction caltime)
(let ((lt (gnc-localtime caltime)))
(+ (- (gnc:date-get-year lt) 1970)
(/ (- (gnc:date-get-year-day lt) 1.0)
(* 1.0 (gnc:days-in-year (gnc:date-get-year lt)))))))
;; return the number of years (in floating point format) between two dates.
(define (gnc:date-year-delta caltime1 caltime2)
(let* ((lt1 (gnc-localtime caltime1))
(lt2 (gnc-localtime caltime2))
(day1 (gnc:date-get-year-day lt1))
(day2 (gnc:date-get-year-day lt2))
(year1 (gnc:date-get-year lt1))
(year2 (gnc:date-get-year lt2))
(dayadj1 (if (and (not (gnc:leap-year? year1))
(>= day1 59))
(+ day1 1)
day1))
(dayadj2 (if (and (not (gnc:leap-year? year2))
(>= day2 59))
(+ day2 1)
day2)))
(+ (- (gnc:date-get-year lt2) (gnc:date-get-year lt1))
(/ (- dayadj2 dayadj1)
366.0))))
;; convert a date in seconds since 1970 into # of 1/2 years since 1970 as
;; a fraction (doubling date-to-year-fraction)
(define (gnc:date-to-halfyear-fraction caltime)
(* (gnc:date-to-year-fraction caltime) 2))
;; convert a date in seconds since 1970 into # of quarters since 1970
;; (assuming quarter = 3 months and using 1/3 of date-to-month-fraction)
(define (gnc:date-to-quarter-fraction caltime)
(/ (gnc:date-to-month-fraction caltime) 3))
;; convert a date in seconds since 1970 into # of months since 1970
(define (gnc:date-to-month-fraction caltime)
(let ((lt (gnc-localtime caltime)))
(+ (* 12 (- (gnc:date-get-year lt) 1970.0))
(gnc:date-get-month lt) -1
(/ (- (gnc:date-get-month-day lt) 1.0) (gnc:days-in-month
(gnc:date-get-month lt)
(gnc:date-get-year lt))))))
;; convert a date in seconds since 1970 into # of two-week periods since
;; Jan 4, 1970 ignoring leap-seconds (just halfing date-to-week-fraction)
(define (gnc:date-to-twoweek-fraction caltime)
(/ (gnc:date-to-week-fraction caltime) 2))
;; convert a date in seconds since 1970 into # of weeks since Jan 4, 1970
;; ignoring leap-seconds
(define (gnc:date-to-week-fraction caltime)
(/ (- (/ (/ caltime 3600.0) 24) 3) 7))
(define (gnc:date-to-week caltime)
(floor (/ (- (/ caltime 86400) 3) 7)))
;; convert a date in seconds since 1970 into # of days since Feb 28, 1970
;; ignoring leap-seconds
(define (gnc:date-to-day-fraction caltime)
(- (/ (/ caltime 3600.0) 24) 59))
;; Returns the function that converts a date into a fraction of
;; {year,month,week,day} according to the given symbol, or #f if the
;; symbol was unknown
(define (gnc:date-get-fraction-func interval)
(case interval
((YearDelta) gnc:date-to-year-fraction)
((HalfYearDelta) gnc:date-to-halfyear-fraction)
((QuarterDelta) gnc:date-to-quarter-fraction)
((MonthDelta) gnc:date-to-month-fraction)
((TwoWeekDelta) gnc:date-to-twoweek-fraction)
((WeekDelta) gnc:date-to-week-fraction)
((DayDelta) gnc:date-to-day-fraction)
(else #f)))
;; Modify a date
(define (moddate op adate delta)
(let ((newtm (gnc-localtime adate)))
(begin
(set-tm:sec newtm (op (tm:sec newtm) (tm:sec delta)))
(set-tm:min newtm (op (tm:min newtm) (tm:min delta)))
(set-tm:hour newtm (op (tm:hour newtm) (tm:hour delta)))
(set-tm:mday newtm (op (tm:mday newtm) (tm:mday delta)))
(set-tm:mon newtm (op (tm:mon newtm) (tm:mon delta)))
(set-tm:year newtm (op (tm:year newtm) (tm:year delta)))
(set-tm:isdst newtm -1)
(gnc-mktime newtm))))
;; Add or subtract time from a date
(define (decdate adate delta) (moddate - adate delta ))
(define (incdate adate delta) (moddate + adate delta ))
;; date-granularity comparison functions.
(define (gnc:time64-le-date t1 t2)
(<= (time64CanonicalDayTime t1)
(time64CanonicalDayTime t2)))
(define (gnc:time64-ge-date t1 t2)
(gnc:time64-le-date t2 t1))
;; Build a list of time intervals.
;;
;; Note that the last interval will be shorter than <incr> if
;; (<curd>-<endd>) is not an integer multiple of <incr>. If you don't
;; want that you'll have to write another function.
(define (gnc:make-date-interval-list current-date end-date increment)
(if (< current-date end-date)
(let ((next-date (incdate current-date increment)))
(if (< next-date end-date)
(cons (list current-date (decdate next-date SecDelta) '())
(gnc:make-date-interval-list next-date end-date increment))
(cons (list current-date end-date '())
'())))
'()))
;; Build a list of times. The dates are evenly spaced with the
;; stepsize 'incr'. If the difference of 'startdate' and 'enddate' is
;; not an integer multiple of 'incr', 'enddate' will be added as the
;; last element of the list, thus making the last interval smaller
;; than 'incr'.
(define (gnc:make-date-list startdate enddate incr)
(if (< startdate enddate)
(cons startdate
(gnc:make-date-list (incdate startdate incr)
enddate incr))
(list enddate)))
; A reference zero date - the Beginning Of The Epoch
; Note: use of eval is evil... by making this a generator function,
; each delta function gets its own instance of Zero Date
(define (make-zdate)
(let ((zd (gnc-localtime (current-time))))
(set-tm:hour zd 0)
(set-tm:min zd 0)
(set-tm:sec zd 0)
(set-tm:mday zd 0)
(set-tm:mon zd 0)
(set-tm:year zd 0)
(set-tm:yday zd 0)
(set-tm:wday zd 0)
(set-tm:isdst zd 0)
zd))
(define SecDelta
(let ((ddt (make-zdate)))
(set-tm:sec ddt 1)
ddt))
(define DayDelta
(let ((ddt (make-zdate)))
(set-tm:mday ddt 1)
ddt))
(define WeekDelta
(let ((ddt (make-zdate)))
(set-tm:mday ddt 7)
ddt))
(define TwoWeekDelta
(let ((ddt (make-zdate)))
(set-tm:mday ddt 14)
ddt))
(define MonthDelta
(let ((ddt (make-zdate)))
(set-tm:mon ddt 1)
ddt))
(define QuarterDelta
(let ((ddt (make-zdate)))
(set-tm:mon ddt 3)
ddt))
(define HalfYearDelta
(let ((ddt (make-zdate)))
(set-tm:mon ddt 6)
ddt))
(define YearDelta
(let ((ddt (make-zdate)))
(set-tm:year ddt 1)
ddt))
(define ThirtyDayDelta
(let ((ddt (make-zdate)))
(set-tm:mday ddt 30)
ddt))
(define NinetyDayDelta
(let ((ddt (make-zdate)))
(set-tm:mday ddt 90)
ddt))
;; if you add any more FooDeltas, add to this list!!!
(define deltalist
(list (cons 'SecDelta SecDelta)
(cons 'DayDelta DayDelta)
(cons 'WeekDelta WeekDelta)
(cons 'TwoWeekDelta TwoWeekDelta)
(cons 'MonthDelta MonthDelta)
(cons 'QuarterDelta QuarterDelta)
(cons 'HalfYearDelta HalfYearDelta)
(cons 'YearDelta YearDelta)
(cons 'ThirtyDayDelta ThirtyDayDelta)
(cons 'NinetyDayDelta NinetyDayDelta)))
(define (gnc:deltasym-to-delta ds)
(let ((retval (assq ds deltalist)))
(if (pair? retval)
(cdr retval)
#f)))
;; given a time64 time on a certain day (local time)
;; converts it to be midday that day.
(define (gnc:time64-start-day-time t64)
(let ((bdt (gnc-localtime t64)))
(set-tm:sec bdt 0)
(set-tm:min bdt 0)
(set-tm:hour bdt 0)
(set-tm:isdst bdt -1)
(gnc-mktime bdt)))
(define (gnc:time64-end-day-time t64)
(let ((bdt (gnc-localtime t64)))
(set-tm:sec bdt 59)
(set-tm:min bdt 59)
(set-tm:hour bdt 23)
(set-tm:isdst bdt -1)
(gnc-mktime bdt)))
(define (gnc:time64-previous-day t64)
(decdate t64 DayDelta))
(define (gnc:time64-next-day t64)
(incdate t64 DayDelta))
(define (gnc:reldate-get-symbol x) (vector-ref x 0))
(define (gnc:reldate-get-string x) (vector-ref x 1))
(define (gnc:reldate-get-desc x) (vector-ref x 2))
(define (gnc:reldate-get-fn x) (vector-ref x 3))
(define (gnc:make-reldate-hash hash reldate-list)
(map (lambda (reldate) (hash-set!
hash
(gnc:reldate-get-symbol reldate)
reldate))
reldate-list))
(define gnc:reldate-string-db (gnc:make-string-database))
(define gnc:relative-date-values '())
(define gnc:relative-date-hash (make-hash-table 23))
(define (gnc:get-absolute-from-relative-date date-symbol)
(let ((rel-date-data (hash-ref gnc:relative-date-hash date-symbol)))
(if rel-date-data
((gnc:reldate-get-fn rel-date-data))
(gnc:error "Tried to look up an undefined date symbol"))))
(define (gnc:get-relative-date-strings date-symbol)
(let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
(cons (gnc:reldate-get-string rel-date-info)
(gnc:relate-get-desc rel-date-info))))
(define (gnc:get-relative-date-string date-symbol)
(let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
(gnc:reldate-get-string rel-date-info)))
(define (gnc:get-relative-date-desc date-symbol)
(let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
(gnc:reldate-get-desc rel-date-info)))
(define (gnc:get-start-cal-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:mon now 0)
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-cal-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(set-tm:mday now 31)
(set-tm:mon now 11)
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-prev-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:mon now 0)
(set-tm:year now (- (tm:year now) 1))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-prev-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(set-tm:mday now 31)
(set-tm:mon now 11)
(set-tm:year now (- (tm:year now) 1))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-next-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:mon now 0)
(set-tm:year now (+ (tm:year now) 1))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-next-year)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(set-tm:mday now 31)
(set-tm:mon now 11)
(set-tm:year now (+ (tm:year now) 1))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-accounting-period)
(gnc-accounting-period-fiscal-start))
(define (gnc:get-end-accounting-period)
(gnc-accounting-period-fiscal-end))
(define (gnc:get-start-this-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-this-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-prev-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(if (= (tm:mon now) 0)
(begin
(set-tm:mon now 11)
(set-tm:year now (- (tm:year now) 1)))
(set-tm:mon now (- (tm:mon now) 1)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-prev-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(if (= (tm:mon now) 0)
(begin
(set-tm:mon now 11)
(set-tm:year now (- (tm:year now) 1)))
(set-tm:mon now (- (tm:mon now) 1)))
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-next-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(if (= (tm:mon now) 11)
(begin
(set-tm:mon now 0)
(set-tm:year now (+ (tm:year now) 1)))
(set-tm:mon now (+ (tm:mon now) 1)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-next-month)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(if (= (tm:mon now) 11)
(begin
(set-tm:mon now 0)
(set-tm:year now (+ (tm:year now) 1)))
(set-tm:mon now (+ (tm:mon now) 1)))
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-current-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:mon now (- (tm:mon now) (modulo (tm:mon now) 3)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-current-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(set-tm:mon now (+ (tm:mon now)
(- 2 (modulo (tm:mon now) 3))))
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-prev-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(set-tm:mon now (- (tm:mon now) (modulo (tm:mon now) 3)))
(if (= (tm:mon now) 0)
(begin
(set-tm:mon now 9)
(set-tm:year now (- (tm:year now) 1)))
(set-tm:mon now (- (tm:mon now) 3)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-prev-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(if (< (tm:mon now) 3)
(begin
(set-tm:mon now 11)
(set-tm:year now (- (tm:year now) 1)))
(set-tm:mon now (- (tm:mon now)
(+ 1 (modulo (tm:mon now) 3)))))
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-start-next-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 0)
(set-tm:min now 0)
(set-tm:hour now 0)
(set-tm:mday now 1)
(if (> (tm:mon now) 8)
(begin
(set-tm:mon now 0)
(set-tm:year now (+ (tm:year now) 1)))
(set-tm:mon now (+ (tm:mon now) (- 3 (modulo (tm:mon now) 3)))))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-end-next-quarter)
(let ((now (gnc-localtime (current-time))))
(set-tm:sec now 59)
(set-tm:min now 59)
(set-tm:hour now 23)
(if (> (tm:mon now) 8)
(begin
(set-tm:mon now 2)
(set-tm:year now (+ (tm:year now) 1)))
(set-tm:mon now (+ (tm:mon now)
(+ 1 (modulo (tm:mon now) 3)))))
(set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900)))
(set-tm:isdst now -1)
(gnc-mktime now)))
(define (gnc:get-today)
(current-time))
(define (gnc:get-one-month-ago)
(let ((now (gnc-localtime (current-time))))
(if (= (tm:mon now) 0)
(begin
(set-tm:mon now 11)
(set-tm:year now (- (tm:year now) 1)))
(set-tm:mon now (- (tm:mon now) 1)))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-three-months-ago)
(let ((now (gnc-localtime (current-time))))
(if (< (tm:mon now) 3)
(begin
(set:tm-mon now (+ (tm:mon now) 12))
(set:tm-year now (- (tm:year now) 1))))
(set:tm-mon now (- (tm:mon now) 3))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-six-months-ago)
(let ((now (gnc-localtime (current-time))))
(if (< (tm:mon now) 6)
(begin
(set:tm-mon now (+ (tm:mon now) 12))
(set:tm-year now (- (tm:year now) 1))))
(set:tm-mon now (- (tm:mon now) 6))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-one-year-ago)
(let ((now (gnc-localtime (current-time))))
(set:tm-year now (- (tm:year now) 1))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-one-month-ahead)
(let ((now (gnc-localtime (current-time))))
(if (= (tm:mon now) 11)
(begin
(set-tm:mon now 0)
(set-tm:year now (+ (tm:year now) 1)))
(set-tm:mon now (+ (tm:mon now) 1)))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-three-months-ahead)
(let ((now (gnc-localtime (current-time))))
(if (> (tm:mon now) 8)
(begin
(set:tm-mon now (- (tm:mon now) 9))
(set:tm-year now (+ (tm:year now) 1))
(set:tm-mon now (+ (tm:mon now) 3))))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-six-months-ahead)
(let ((now (gnc-localtime (current-time))))
(if (> (tm:mon now) 5)
(begin
(set:tm-mon now (- (tm:mon now) 6))
(set:tm-year now (+ (tm:year now) 1))
(set:tm-mon now (+ (tm:mon now) 6))))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
(define (gnc:get-one-year-ahead)
(let ((now (gnc-localtime (current-time))))
(set:tm-year now (+ (tm:year now) 1))
(let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
(+ (tm:year now) 1900))))
(if (< month-days (tm:mday now))
(set-tm:mday now month-days))
(set-tm:isdst now -1)
(gnc-mktime now))))
;; There is no GNC:RELATIVE-DATES list like the one mentioned in
;; gnucash-design.info, is there? Here are the currently defined
;; items, loosely grouped.
;;today
;;start-cal-year end-cal-year start-prev-year end-prev-year
;;start-this-month end-this-month start-prev-month end-prev-month
;;start-current-quarter end-current-quarter start-prev-quarter
;;end-prev-quarter
;;one-month-ago three-months-ago six-months-ago one-year-ago
;;start-cur-fin-year start-prev-fin-year end-prev-fin-year
(define (gnc:reldate-initialize)
(gnc:reldate-string-db
'store 'start-cal-year-string
(N_ "Start of this year"))
(gnc:reldate-string-db
'store 'start-cal-year-desc
(N_ "First day of the current calendar year."))
(gnc:reldate-string-db
'store 'end-cal-year-string
(N_ "End of this year"))
(gnc:reldate-string-db
'store 'end-cal-year-desc
(N_ "Last day of the current calendar year."))
(gnc:reldate-string-db
'store 'start-prev-year-string
(N_ "Start of previous year"))
(gnc:reldate-string-db
'store 'start-prev-year-desc
(N_ "First day of the previous calendar year."))
(gnc:reldate-string-db
'store 'end-prev-year-string
(N_ "End of previous year"))
(gnc:reldate-string-db
'store 'end-prev-year-desc
(N_ "Last day of the previous calendar year."))
(gnc:reldate-string-db
'store 'start-next-year-string
(N_ "Start of next year"))
(gnc:reldate-string-db
'store 'start-next-year-desc
(N_ "First day of the next calendar year."))
(gnc:reldate-string-db
'store 'end-next-year-string
(N_ "End of next year"))
(gnc:reldate-string-db
'store 'end-next-year-desc
(N_ "Last day of the next calendar year."))
(gnc:reldate-string-db
'store 'start-accounting-period-string
(N_ "Start of accounting period"))
(gnc:reldate-string-db
'store 'start-accounting-period-desc
(N_ "First day of the accounting period, as set in the global preferences."))
(gnc:reldate-string-db
'store 'end-accounting-period-string
(N_ "End of accounting period"))
(gnc:reldate-string-db
'store 'end-accounting-period-desc
(N_ "Last day of the accounting period, as set in the global preferences."))
(gnc:reldate-string-db
'store 'start-this-month-string
(N_ "Start of this month"))
(gnc:reldate-string-db
'store 'start-this-month-desc
(N_ "First day of the current month."))
(gnc:reldate-string-db
'store 'end-this-month-string
(N_ "End of this month"))
(gnc:reldate-string-db
'store 'end-this-month-desc
(N_ "Last day of the current month."))
(gnc:reldate-string-db
'store 'start-prev-month-string
(N_ "Start of previous month"))
(gnc:reldate-string-db
'store 'start-prev-month-desc
(N_ "First day of the previous month."))
(gnc:reldate-string-db
'store 'end-prev-month-string
(N_ "End of previous month"))
(gnc:reldate-string-db
'store 'end-prev-month-desc
(N_ "Last day of previous month."))
(gnc:reldate-string-db
'store 'start-next-month-string
(N_ "Start of next month"))
(gnc:reldate-string-db
'store 'start-next-month-desc
(N_ "First day of the next month."))
(gnc:reldate-string-db
'store 'end-next-month-string
(N_ "End of next month"))
(gnc:reldate-string-db
'store 'end-next-month-desc
(N_ "Last day of next month."))
(gnc:reldate-string-db
'store 'start-current-quarter-string
(N_ "Start of current quarter"))
(gnc:reldate-string-db
'store 'start-current-quarter-desc
(N_ "First day of the current quarterly accounting period."))
(gnc:reldate-string-db
'store 'end-current-quarter-string
(N_ "End of current quarter"))
(gnc:reldate-string-db
'store 'end-current-quarter-desc
(N_ "Last day of the current quarterly accounting period."))
(gnc:reldate-string-db
'store 'start-prev-quarter-string
(N_ "Start of previous quarter"))
(gnc:reldate-string-db
'store 'start-prev-quarter-desc
(N_ "First day of the previous quarterly accounting period."))
(gnc:reldate-string-db
'store 'end-prev-quarter-string
(N_ "End of previous quarter"))
(gnc:reldate-string-db
'store 'end-prev-quarter-desc
(N_ "Last day of previous quarterly accounting period."))
(gnc:reldate-string-db
'store 'start-next-quarter-string
(N_ "Start of next quarter"))
(gnc:reldate-string-db
'store 'start-next-quarter-desc
(N_ "First day of the next quarterly accounting period."))
(gnc:reldate-string-db
'store 'end-next-quarter-string
(N_ "End of next quarter"))
(gnc:reldate-string-db
'store 'end-next-quarter-desc
(N_ "Last day of next quarterly accounting period."))
(gnc:reldate-string-db
'store 'today-string
(N_ "Today"))
(gnc:reldate-string-db
'store 'today-desc (N_ "The current date."))
(gnc:reldate-string-db
'store 'one-month-ago-string
(N_ "One Month Ago"))
(gnc:reldate-string-db
'store 'one-month-ago-desc (N_ "One Month Ago."))
(gnc:reldate-string-db
'store 'one-week-ago-string
(N_ "One Week Ago"))
(gnc:reldate-string-db
'store 'one-week-ago-desc (N_ "One Week Ago."))
(gnc:reldate-string-db
'store 'three-months-ago-string
(N_ "Three Months Ago"))
(gnc:reldate-string-db
'store 'three-months-ago-desc (N_ "Three Months Ago."))
(gnc:reldate-string-db
'store 'six-months-ago-string
(N_ "Six Months Ago"))
(gnc:reldate-string-db
'store 'six-months-ago-desc (N_ "Six Months Ago."))
(gnc:reldate-string-db
'store 'one-year-ago-string (N_ "One Year Ago"))
(gnc:reldate-string-db
'store 'one-year-ago-desc (N_ "One Year Ago."))
(gnc:reldate-string-db
'store 'one-month-ahead-string
(N_ "One Month Ahead"))
(gnc:reldate-string-db
'store 'one-month-ahead-desc (N_ "One Month Ahead."))
(gnc:reldate-string-db
'store 'one-week-ahead-string
(N_ "One Week Ahead"))
(gnc:reldate-string-db
'store 'one-week-ahead-desc (N_ "One Week Ahead."))
(gnc:reldate-string-db
'store 'three-months-ahead-string
(N_ "Three Months Ahead"))
(gnc:reldate-string-db
'store 'three-months-ahead-desc (N_ "Three Months Ahead."))
(gnc:reldate-string-db
'store 'six-months-ahead-string
(N_ "Six Months Ahead"))
(gnc:reldate-string-db
'store 'six-months-ahead-desc (N_ "Six Months Ahead."))
(gnc:reldate-string-db
'store 'one-year-ahead-string (N_ "One Year Ahead"))
(gnc:reldate-string-db
'store 'one-year-ahead-desc (N_ "One Year Ahead."))
(set! gnc:relative-date-values
(list
(vector 'start-cal-year
(gnc:reldate-string-db 'lookup 'start-cal-year-string)
(gnc:reldate-string-db 'lookup 'start-cal-year-desc)
gnc:get-start-cal-year)
(vector 'end-cal-year
(gnc:reldate-string-db 'lookup 'end-cal-year-string)
(gnc:reldate-string-db 'lookup 'end-cal-year-desc)
gnc:get-end-cal-year)
(vector 'start-prev-year
(gnc:reldate-string-db 'lookup 'start-prev-year-string)
(gnc:reldate-string-db 'lookup 'start-prev-year-desc)
gnc:get-start-prev-year)
(vector 'start-next-year
(gnc:reldate-string-db 'lookup 'start-next-year-string)
(gnc:reldate-string-db 'lookup 'start-next-year-desc)
gnc:get-start-next-year)
(vector 'end-prev-year
(gnc:reldate-string-db 'lookup 'end-prev-year-string)
(gnc:reldate-string-db 'lookup 'end-prev-year-desc)
gnc:get-end-prev-year)
(vector 'end-next-year
(gnc:reldate-string-db 'lookup 'end-next-year-string)
(gnc:reldate-string-db 'lookup 'end-next-year-desc)
gnc:get-end-next-year)
(vector 'start-accounting-period
(gnc:reldate-string-db 'lookup 'start-accounting-period-string)
(gnc:reldate-string-db 'lookup 'start-accounting-period-desc)
gnc:get-start-accounting-period)
(vector 'end-accounting-period
(gnc:reldate-string-db 'lookup 'end-accounting-period-string)
(gnc:reldate-string-db 'lookup 'end-accounting-period-desc)
gnc:get-end-accounting-period)
(vector 'start-this-month
(gnc:reldate-string-db 'lookup 'start-this-month-string)
(gnc:reldate-string-db 'lookup 'start-this-month-desc)
gnc:get-start-this-month)
(vector 'end-this-month
(gnc:reldate-string-db 'lookup 'end-this-month-string)
(gnc:reldate-string-db 'lookup 'end-this-month-desc)
gnc:get-end-this-month)
(vector 'start-prev-month
(gnc:reldate-string-db 'lookup 'start-prev-month-string)
(gnc:reldate-string-db 'lookup 'start-prev-month-desc)
gnc:get-start-prev-month)
(vector 'end-prev-month
(gnc:reldate-string-db 'lookup 'end-prev-month-string)
(gnc:reldate-string-db 'lookup 'end-prev-month-desc)
gnc:get-end-prev-month)
(vector 'start-next-month
(gnc:reldate-string-db 'lookup 'start-next-month-string)
(gnc:reldate-string-db 'lookup 'start-next-month-desc)
gnc:get-start-next-month)
(vector 'end-next-month
(gnc:reldate-string-db 'lookup 'end-next-month-string)
(gnc:reldate-string-db 'lookup 'end-next-month-desc)
gnc:get-end-next-month)
(vector 'start-current-quarter
(gnc:reldate-string-db 'lookup 'start-current-quarter-string)
(gnc:reldate-string-db 'lookup 'start-current-quarter-desc)
gnc:get-start-current-quarter)
(vector 'end-current-quarter
(gnc:reldate-string-db 'lookup 'end-current-quarter-string)
(gnc:reldate-string-db 'lookup 'end-current-quarter-desc)
gnc:get-end-current-quarter)
(vector 'start-prev-quarter
(gnc:reldate-string-db 'lookup 'start-prev-quarter-string)
(gnc:reldate-string-db 'lookup 'start-prev-quarter-desc)
gnc:get-start-prev-quarter)
(vector 'end-prev-quarter
(gnc:reldate-string-db 'lookup 'end-prev-quarter-string)
(gnc:reldate-string-db 'lookup 'end-prev-quarter-desc)
gnc:get-end-prev-quarter)
(vector 'start-next-quarter
(gnc:reldate-string-db 'lookup 'start-next-quarter-string)
(gnc:reldate-string-db 'lookup 'start-next-quarter-desc)
gnc:get-start-next-quarter)
(vector 'end-next-quarter
(gnc:reldate-string-db 'lookup 'end-next-quarter-string)
(gnc:reldate-string-db 'lookup 'end-next-quarter-desc)
gnc:get-end-next-quarter)
(vector 'today
(gnc:reldate-string-db 'lookup 'today-string)
(gnc:reldate-string-db 'lookup 'today-desc)
gnc:get-today)
(vector 'one-month-ago
(gnc:reldate-string-db 'lookup 'one-month-ago-string)
(gnc:reldate-string-db 'lookup 'one-month-ago-desc)
gnc:get-one-month-ago)
(vector 'three-months-ago
(gnc:reldate-string-db 'lookup 'three-months-ago-string)
(gnc:reldate-string-db 'lookup 'three-months-ago-desc)
gnc:get-three-months-ago)
(vector 'six-months-ago
(gnc:reldate-string-db 'lookup 'six-months-ago-string)
(gnc:reldate-string-db 'lookup 'six-months-ago-desc)
gnc:get-three-months-ago)
(vector 'one-year-ago
(gnc:reldate-string-db 'lookup 'one-year-ago-string)
(gnc:reldate-string-db 'lookup 'one-year-ago-desc)
gnc:get-one-year-ago)
(vector 'one-month-ahead
(gnc:reldate-string-db 'lookup 'one-month-ahead-string)
(gnc:reldate-string-db 'lookup 'one-month-ahead-desc)
gnc:get-one-month-ahead)
(vector 'three-months-ahead
(gnc:reldate-string-db 'lookup 'three-months-ahead-string)
(gnc:reldate-string-db 'lookup 'three-months-ahead-desc)
gnc:get-three-months-ahead)
(vector 'six-months-ahead
(gnc:reldate-string-db 'lookup 'six-months-ahead-string)
(gnc:reldate-string-db 'lookup 'six-months-ahead-desc)
gnc:get-three-months-ahead)
(vector 'one-year-ahead
(gnc:reldate-string-db 'lookup 'one-year-ahead-string)
(gnc:reldate-string-db 'lookup 'one-year-ahead-desc)
gnc:get-one-year-ahead)))
(gnc:make-reldate-hash gnc:relative-date-hash gnc:relative-date-values)
(set! gnc:reldate-list
(map (lambda (x) (vector-ref x 0)) gnc:relative-date-values)))