mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2024-11-29 07:58:28 +00:00
1158 lines
52 KiB
EmacsLisp
1158 lines
52 KiB
EmacsLisp
;;; cal-hebrew.el --- calendar functions for the Hebrew calendar -*- lexical-binding: t; -*-
|
||
|
||
;; Copyright (C) 1995, 1997, 2001-2024 Free Software Foundation, Inc.
|
||
|
||
;; Author: Nachum Dershowitz <nachum@cs.uiuc.edu>
|
||
;; Edward M. Reingold <reingold@cs.uiuc.edu>
|
||
;; Maintainer: emacs-devel@gnu.org
|
||
;; Keywords: calendar
|
||
;; Human-Keywords: Hebrew calendar, calendar, diary
|
||
;; Package: calendar
|
||
|
||
;; This file is part of GNU Emacs.
|
||
|
||
;; GNU Emacs 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 3 of the License, or
|
||
;; (at your option) any later version.
|
||
|
||
;; GNU Emacs 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 GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
||
|
||
;;; Commentary:
|
||
|
||
;; See calendar.el.
|
||
|
||
;;; Code:
|
||
|
||
(require 'calendar)
|
||
|
||
(defcustom diary-hebrew-sabbath-candles-minutes 18
|
||
"Number of minutes before sunset for sabbath candle lighting.
|
||
Used by `diary-hebrew-sabbath-candles'."
|
||
:group 'diary
|
||
:type 'integer
|
||
:version "21.1")
|
||
|
||
;; End of user options.
|
||
|
||
(defun calendar-hebrew-leap-year-p (year)
|
||
"Non-nil if YEAR is a Hebrew calendar leap year."
|
||
(< (% (1+ (* 7 year)) 19) 7))
|
||
|
||
(defun calendar-hebrew-last-month-of-year (year)
|
||
"The last month of the Hebrew calendar YEAR."
|
||
(if (calendar-hebrew-leap-year-p year)
|
||
13
|
||
12))
|
||
|
||
(defun calendar-hebrew-elapsed-days (year)
|
||
"Days to mean conjunction of Tishri of Hebrew YEAR.
|
||
Measured from Sunday before start of Hebrew calendar."
|
||
(let* ((months-elapsed
|
||
(+ (* 235 (/ (1- year) 19)) ; months in complete cycles so far
|
||
(* 12 (% (1- year) 19)) ; regular months in this cycle
|
||
(/ (1+ (* 7 (% (1- year) 19))) 19))) ; leap months this cycle
|
||
(parts-elapsed (+ 204 (* 793 (% months-elapsed 1080))))
|
||
(hours-elapsed (+ 5
|
||
(* 12 months-elapsed)
|
||
(* 793 (/ months-elapsed 1080))
|
||
(/ parts-elapsed 1080)))
|
||
(parts ; conjunction parts
|
||
(+ (* 1080 (% hours-elapsed 24)) (% parts-elapsed 1080)))
|
||
(day ; conjunction day
|
||
(+ 1 (* 29 months-elapsed) (/ hours-elapsed 24)))
|
||
(alternative-day
|
||
(if (or (>= parts 19440) ; if the new moon is at or after midday
|
||
(and (= (% day 7) 2) ; ...or is on a Tuesday...
|
||
(>= parts 9924) ; at 9 hours, 204 parts or later...
|
||
;; of a common year...
|
||
(not (calendar-hebrew-leap-year-p year)))
|
||
(and (= (% day 7) 1) ; ...or is on a Monday...
|
||
(>= parts 16789) ; at 15 hours, 589 parts or later...
|
||
;; at the end of a leap year.
|
||
(calendar-hebrew-leap-year-p (1- year))))
|
||
;; Then postpone Rosh HaShanah one day.
|
||
(1+ day)
|
||
;; Else:
|
||
day)))
|
||
;; If Rosh HaShanah would occur on Sunday, Wednesday, or Friday
|
||
(if (memq (% alternative-day 7) (list 0 3 5))
|
||
;; Then postpone it one (more) day and return.
|
||
(1+ alternative-day)
|
||
;; Else return.
|
||
alternative-day)))
|
||
|
||
(defun calendar-hebrew-days-in-year (year)
|
||
"Number of days in Hebrew YEAR."
|
||
(- (calendar-hebrew-elapsed-days (1+ year))
|
||
(calendar-hebrew-elapsed-days year)))
|
||
|
||
(defun calendar-hebrew-long-heshvan-p (year)
|
||
"Non-nil if Heshvan is long in Hebrew YEAR."
|
||
(= (% (calendar-hebrew-days-in-year year) 10) 5))
|
||
|
||
(defun calendar-hebrew-short-kislev-p (year)
|
||
"Non-nil if Kislev is short in Hebrew YEAR."
|
||
(= (% (calendar-hebrew-days-in-year year) 10) 3))
|
||
|
||
(defun calendar-hebrew-last-day-of-month (month year)
|
||
"The last day of MONTH in YEAR."
|
||
(if (or (memq month (list 2 4 6 10 13))
|
||
(and (= month 12) (not (calendar-hebrew-leap-year-p year)))
|
||
(and (= month 8) (not (calendar-hebrew-long-heshvan-p year)))
|
||
(and (= month 9) (calendar-hebrew-short-kislev-p year)))
|
||
29
|
||
30))
|
||
|
||
(defun calendar-hebrew-to-absolute (date)
|
||
"Absolute date of Hebrew DATE.
|
||
The absolute date is the number of days elapsed since the (imaginary)
|
||
Gregorian date Sunday, December 31, 1 BC."
|
||
(let ((month (calendar-extract-month date))
|
||
(day (calendar-extract-day date))
|
||
(year (calendar-extract-year date)))
|
||
(+ day ; days so far this month
|
||
(if (< month 7) ; before Tishri
|
||
;; Then add days in prior months this year before and after Nisan.
|
||
(+ (calendar-sum
|
||
m 7 (<= m (calendar-hebrew-last-month-of-year year))
|
||
(calendar-hebrew-last-day-of-month m year))
|
||
(calendar-sum
|
||
m 1 (< m month)
|
||
(calendar-hebrew-last-day-of-month m year)))
|
||
;; Else add days in prior months this year.
|
||
(calendar-sum
|
||
m 7 (< m month)
|
||
(calendar-hebrew-last-day-of-month m year)))
|
||
(calendar-hebrew-elapsed-days year) ; days in prior years
|
||
-1373429))) ; days elapsed before absolute date 1
|
||
|
||
(defun calendar-hebrew-from-absolute (date)
|
||
"Compute the Hebrew date (month day year) corresponding to absolute DATE.
|
||
The absolute date is the number of days elapsed since the (imaginary)
|
||
Gregorian date Sunday, December 31, 1 BC."
|
||
(let* ((greg-date (calendar-gregorian-from-absolute date))
|
||
(year (+ 3760 (calendar-extract-year greg-date)))
|
||
(month (aref [9 10 11 12 1 2 3 4 7 7 7 8]
|
||
(1- (calendar-extract-month greg-date))))
|
||
(length (progn
|
||
(while (>= date (calendar-hebrew-to-absolute
|
||
(list 7 1 (1+ year))))
|
||
(setq year (1+ year)))
|
||
(calendar-hebrew-last-month-of-year year)))
|
||
day)
|
||
(while (> date
|
||
(calendar-hebrew-to-absolute
|
||
(list month
|
||
(calendar-hebrew-last-day-of-month month year)
|
||
year)))
|
||
(setq month (1+ (% month length))))
|
||
(setq day (1+
|
||
(- date (calendar-hebrew-to-absolute (list month 1 year)))))
|
||
(list month day year)))
|
||
|
||
(defconst calendar-hebrew-month-name-array-common-year
|
||
["Nisan" "Iyar" "Sivan" "Tammuz" "Av" "Elul" "Tishri"
|
||
"Heshvan" "Kislev" "Teveth" "Shevat" "Adar"]
|
||
"Array of strings giving the names of the Hebrew months in a common year.")
|
||
|
||
(defconst calendar-hebrew-month-name-array-leap-year
|
||
["Nisan" "Iyar" "Sivan" "Tammuz" "Av" "Elul" "Tishri"
|
||
"Heshvan" "Kislev" "Teveth" "Shevat" "Adar I" "Adar II"]
|
||
"Array of strings giving the names of the Hebrew months in a leap year.")
|
||
|
||
;;;###cal-autoload
|
||
(defun calendar-hebrew-date-string (&optional date)
|
||
"String of Hebrew date before sunset of Gregorian DATE.
|
||
Defaults to today's date if DATE is not given.
|
||
Driven by the variable `calendar-date-display-form'."
|
||
(let* ((hebrew-date (calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian
|
||
(or date (calendar-current-date)))))
|
||
(calendar-month-name-array
|
||
(if (calendar-hebrew-leap-year-p (calendar-extract-year hebrew-date))
|
||
calendar-hebrew-month-name-array-leap-year
|
||
calendar-hebrew-month-name-array-common-year)))
|
||
(calendar-date-string hebrew-date nil t)))
|
||
|
||
;;;###cal-autoload
|
||
(defun calendar-hebrew-print-date ()
|
||
"Show the Hebrew calendar equivalent of the date under the cursor."
|
||
(interactive)
|
||
(message "Hebrew date (until sunset): %s"
|
||
(calendar-hebrew-date-string (calendar-cursor-to-date t))))
|
||
|
||
(defun calendar-hebrew-yahrzeit (death-date year)
|
||
"Absolute date of the anniversary of Hebrew DEATH-DATE in Hebrew YEAR."
|
||
(let ((death-day (calendar-extract-day death-date))
|
||
(death-month (calendar-extract-month death-date))
|
||
(death-year (calendar-extract-year death-date)))
|
||
(cond
|
||
;; If it's Heshvan 30 it depends on the first anniversary; if
|
||
;; that was not Heshvan 30, use the day before Kislev 1.
|
||
((and (= death-month 8)
|
||
(= death-day 30)
|
||
(not (calendar-hebrew-long-heshvan-p (1+ death-year))))
|
||
(1- (calendar-hebrew-to-absolute (list 9 1 year))))
|
||
;; If it's Kislev 30 it depends on the first anniversary; if that
|
||
;; was not Kislev 30, use the day before Teveth 1.
|
||
((and (= death-month 9)
|
||
(= death-day 30)
|
||
(calendar-hebrew-short-kislev-p (1+ death-year)))
|
||
(1- (calendar-hebrew-to-absolute (list 10 1 year))))
|
||
;; If it's Adar II, use the same day in last month of year (Adar
|
||
;; or Adar II).
|
||
((= death-month 13)
|
||
(calendar-hebrew-to-absolute
|
||
(list (calendar-hebrew-last-month-of-year year) death-day year)))
|
||
;; If it's the 30th in Adar I and year is not a leap year (so
|
||
;; Adar has only 29 days), use the last day in Shevat.
|
||
((and (= death-day 30)
|
||
(= death-month 12)
|
||
(not (calendar-hebrew-leap-year-p year)))
|
||
(calendar-hebrew-to-absolute (list 11 30 year)))
|
||
;; In all other cases, use the normal anniversary of the date of death.
|
||
(t (calendar-hebrew-to-absolute
|
||
(list death-month death-day year))))))
|
||
|
||
(defun calendar-hebrew-read-date ()
|
||
"Interactively read the arguments for a Hebrew date command.
|
||
Reads a year, month, and day."
|
||
(let* ((today (calendar-current-date))
|
||
(year (calendar-read-sexp
|
||
"Hebrew calendar year (>3760)"
|
||
(lambda (x) (> x 3760))
|
||
(calendar-extract-year
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian today)))))
|
||
(month-array (if (calendar-hebrew-leap-year-p year)
|
||
calendar-hebrew-month-name-array-leap-year
|
||
calendar-hebrew-month-name-array-common-year))
|
||
(completion-ignore-case t)
|
||
(month (cdr (assoc-string
|
||
(completing-read
|
||
"Hebrew calendar month name: "
|
||
(mapcar 'list (append month-array nil))
|
||
(if (= year 3761)
|
||
(lambda (x)
|
||
(let ((m (cdr
|
||
(assoc-string
|
||
(car x)
|
||
(calendar-make-alist month-array)
|
||
t))))
|
||
(< 0
|
||
(calendar-hebrew-to-absolute
|
||
(list m
|
||
(calendar-hebrew-last-day-of-month
|
||
m year)
|
||
year))))))
|
||
t)
|
||
(calendar-make-alist month-array 1) t)))
|
||
(last (calendar-hebrew-last-day-of-month month year))
|
||
(first (if (and (= year 3761) (= month 10))
|
||
18 1))
|
||
(day (calendar-read-sexp
|
||
"Hebrew calendar day (%d-%d)"
|
||
(lambda (x) (and (<= first x) (<= x last)))
|
||
nil
|
||
first last)))
|
||
(list (list month day year))))
|
||
|
||
;;;###cal-autoload
|
||
(defun calendar-hebrew-goto-date (date &optional noecho)
|
||
"Move cursor to Hebrew DATE; echo Hebrew date unless NOECHO is non-nil."
|
||
(interactive (calendar-hebrew-read-date))
|
||
(calendar-goto-date (calendar-gregorian-from-absolute
|
||
(calendar-hebrew-to-absolute date)))
|
||
(or noecho (calendar-hebrew-print-date)))
|
||
|
||
(defvar displayed-month) ; from calendar-generate
|
||
|
||
(defun calendar-hebrew-date-is-visible-p (month day)
|
||
"Return non-nil if Hebrew MONTH DAY is visible in the calendar window.
|
||
Returns the corresponding Gregorian date."
|
||
;; This test is only to speed things up a bit; it works fine without it.
|
||
(if (memq displayed-month
|
||
;; What this is doing is equivalent to +1,2,3,4,5 modulo 12, ie:
|
||
;; (mapcar (lambda (n) (let ((x (mod n 12)))
|
||
;; (if (zerop x) 12
|
||
;; x)))
|
||
;; (number-sequence (1+ month) (+ 5 month)))
|
||
;; Ie it makes a list:
|
||
;; 2 3 4 5 6 when month = 1
|
||
;; 3 4 5 6 7 when month = 2
|
||
;; ...
|
||
;; 8 9 10 11 12 when month = 7
|
||
;; 9 10 11 12 1 when month = 8
|
||
;; ...
|
||
;; 12 1 2 3 4 when month = 11
|
||
;; 1 2 3 4 5 when month = 12
|
||
;; This implies that hebrew month N cannot occur outside
|
||
;; Gregorian months N:N+6 (the calendar shows
|
||
;; displayed-month +/- 1 at any time).
|
||
;; So to put it another way:
|
||
;; (calendar-interval month 1 displayed-month
|
||
;; (if (> month displayed-month) 2 1))
|
||
;; must be >= 1 and <= 5. This could be expanded to:
|
||
;; (if (> month displayed-month) (+ 12 (- displayed-month month))
|
||
;; (- displayed-month month)
|
||
(list
|
||
(if (< 11 month) (- month 11) (+ month 1))
|
||
(if (< 10 month) (- month 10) (+ month 2))
|
||
(if (< 9 month) (- month 9) (+ month 3))
|
||
(if (< 8 month) (- month 8) (+ month 4))
|
||
(if (< 7 month) (- month 7) (+ month 5))))
|
||
(calendar-nongregorian-visible-p
|
||
month day 'calendar-hebrew-to-absolute
|
||
'calendar-hebrew-from-absolute
|
||
;; Hebrew new year is start of month 7.
|
||
;; If hmonth >= 7, choose the higher year.
|
||
(lambda (m) (> m 6)))))
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew (month day string)
|
||
"Holiday on MONTH, DAY (Hebrew) called STRING.
|
||
If MONTH, DAY (Hebrew) is visible, the value returned is corresponding
|
||
Gregorian date in the form of the list (((month day year) STRING)). Returns
|
||
nil if it is not visible in the current calendar window."
|
||
(let ((gdate (calendar-hebrew-date-is-visible-p month day)))
|
||
(if gdate (list (list gdate string)))))
|
||
|
||
;; h-r-h-e should be called from holidays code.
|
||
(declare-function holiday-filter-visible-calendar "holidays" (l))
|
||
|
||
(defvar displayed-year)
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew-rosh-hashanah (&optional all)
|
||
"List of dates related to Rosh Hashanah, as visible in calendar window.
|
||
Shows only the major holidays, unless `calendar-hebrew-all-holidays-flag'
|
||
or ALL is non-nil."
|
||
(when (memq displayed-month '(8 9 10 11))
|
||
(let ((abs-r-h (calendar-hebrew-to-absolute
|
||
(list 7 1 (+ displayed-year 3761)))))
|
||
(holiday-filter-visible-calendar
|
||
(append
|
||
(list
|
||
(list (calendar-gregorian-from-absolute abs-r-h)
|
||
(format "Rosh HaShanah %d" (+ 3761 displayed-year)))
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 9))
|
||
"Yom Kippur")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 14))
|
||
"Sukkot")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 21))
|
||
"Shemini Atzeret")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 22))
|
||
"Simchat Torah"))
|
||
(when (or all calendar-hebrew-all-holidays-flag)
|
||
(list
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (- abs-r-h 4)))
|
||
"Selichot (night)")
|
||
(list (calendar-gregorian-from-absolute (1- abs-r-h))
|
||
"Erev Rosh HaShanah")
|
||
(list (calendar-gregorian-from-absolute (1+ abs-r-h))
|
||
"Rosh HaShanah (second day)")
|
||
(list (calendar-gregorian-from-absolute
|
||
(+ abs-r-h (if (= (% abs-r-h 7) 4) 3 2)))
|
||
"Tzom Gedaliah")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (+ 7 abs-r-h)))
|
||
"Shabbat Shuvah")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 8))
|
||
"Erev Yom Kippur")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 13))
|
||
"Erev Sukkot")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 15))
|
||
"Sukkot (second day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 16))
|
||
"Hol Hamoed Sukkot (first day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 17))
|
||
"Hol Hamoed Sukkot (second day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 18))
|
||
"Hol Hamoed Sukkot (third day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 19))
|
||
"Hol Hamoed Sukkot (fourth day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-r-h 20))
|
||
"Hoshanah Rabbah"))))))))
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew-hanukkah (&optional all)
|
||
"List of dates related to Hanukkah, as visible in calendar window.
|
||
Shows only Hanukkah, unless `calendar-hebrew-all-holidays-flag' or ALL
|
||
is non-nil."
|
||
;; This test is only to speed things up a bit, it works fine without it.
|
||
(when (memq displayed-month '(10 11 12 1 2))
|
||
(let* ((m displayed-month)
|
||
(y displayed-year)
|
||
(h-y (progn
|
||
(calendar-increment-month m y 1)
|
||
(calendar-extract-year
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian
|
||
(list m (calendar-last-day-of-month m y) y))))))
|
||
(abs-h (calendar-hebrew-to-absolute (list 9 25 h-y)))
|
||
(ord ["first" "second" "third" "fourth" "fifth" "sixth"
|
||
"seventh" "eighth"]))
|
||
(holiday-filter-visible-calendar
|
||
(if (or all calendar-hebrew-all-holidays-flag)
|
||
(append
|
||
(list
|
||
(list (calendar-gregorian-from-absolute (1- abs-h))
|
||
"Erev Hanukkah"))
|
||
(let (han)
|
||
(dotimes (i 8)
|
||
(push (list
|
||
(calendar-gregorian-from-absolute (+ abs-h i))
|
||
(format "Hanukkah (%s day)" (aref ord i)))
|
||
han))
|
||
(nreverse han)))
|
||
(list (list (calendar-gregorian-from-absolute abs-h) "Hanukkah")))))))
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew-passover (&optional all)
|
||
"List of dates related to Passover, as visible in calendar window.
|
||
Shows only the major holidays, unless `calendar-hebrew-all-holidays-flag'
|
||
or ALL is non-nil."
|
||
(when (< displayed-month 8)
|
||
(let ((abs-p (calendar-hebrew-to-absolute
|
||
(list 1 15 (+ displayed-year 3760)))))
|
||
(holiday-filter-visible-calendar
|
||
;; The first two are out of order when the others are added.
|
||
(append
|
||
(list
|
||
(list (calendar-gregorian-from-absolute abs-p) "Passover")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 50))
|
||
"Shavuot"))
|
||
(when (or all calendar-hebrew-all-holidays-flag)
|
||
(let ((wday (% abs-p 7)))
|
||
(list
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (- abs-p 43)))
|
||
"Shabbat Shekalim")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (- abs-p 30)))
|
||
"Shabbat Zachor")
|
||
(list (calendar-gregorian-from-absolute
|
||
(- abs-p (if (= wday 2) 33 31)))
|
||
"Fast of Esther")
|
||
(list (calendar-gregorian-from-absolute (- abs-p 31))
|
||
"Erev Purim")
|
||
(list (calendar-gregorian-from-absolute (- abs-p 30))
|
||
"Purim")
|
||
(list (calendar-gregorian-from-absolute
|
||
(- abs-p (if (zerop wday) 28 29)))
|
||
"Shushan Purim")
|
||
(list (calendar-gregorian-from-absolute
|
||
(- (calendar-dayname-on-or-before 6 (- abs-p 14)) 7))
|
||
"Shabbat Parah")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (- abs-p 14)))
|
||
"Shabbat HaHodesh")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (1- abs-p)))
|
||
"Shabbat HaGadol")
|
||
(list (calendar-gregorian-from-absolute (1- abs-p))
|
||
"Erev Passover")
|
||
(list (calendar-gregorian-from-absolute (1+ abs-p))
|
||
"Passover (second day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 2))
|
||
"Hol Hamoed Passover (first day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 3))
|
||
"Hol Hamoed Passover (second day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 4))
|
||
"Hol Hamoed Passover (third day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 5))
|
||
"Hol Hamoed Passover (fourth day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 6))
|
||
"Passover (seventh day)")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 7))
|
||
"Passover (eighth day)")
|
||
(list (calendar-gregorian-from-absolute
|
||
(+ abs-p (if (zerop (% (+ abs-p 12) 7))
|
||
13
|
||
12)))
|
||
"Yom HaShoah")
|
||
(list (calendar-gregorian-from-absolute
|
||
(+ abs-p
|
||
;; If falls on Sat or Fri, moves to preceding Thurs.
|
||
;; If falls on Mon, moves to Tues (since 2004).
|
||
(cond ((zerop wday) 18) ; Sat
|
||
((= wday 6) 19) ; Fri
|
||
((= wday 2) 21) ; Mon
|
||
(t 20))))
|
||
"Yom HaAtzma'ut")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 33))
|
||
"Lag BaOmer")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 43))
|
||
"Yom Yerushalaim")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 49))
|
||
"Erev Shavuot")
|
||
(list (calendar-gregorian-from-absolute (+ abs-p 51))
|
||
"Shavuot (second day)")))))))))
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew-tisha-b-av ()
|
||
"List of dates around Tisha B'Av, as visible in calendar window."
|
||
(when (memq displayed-month '(5 6 7 8 9))
|
||
(let* ((abs-t-a (calendar-hebrew-to-absolute
|
||
(list 5 9 (+ displayed-year 3760))))
|
||
(wday (% abs-t-a 7)))
|
||
(holiday-filter-visible-calendar
|
||
(list
|
||
(list (calendar-gregorian-from-absolute
|
||
(- abs-t-a (if (= wday 6) 20 21)))
|
||
"Tzom Tammuz")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 abs-t-a))
|
||
"Shabbat Hazon")
|
||
(list (calendar-gregorian-from-absolute
|
||
(if (= wday 6) (1+ abs-t-a) abs-t-a))
|
||
"Tisha B'Av")
|
||
(list (calendar-gregorian-from-absolute
|
||
(calendar-dayname-on-or-before 6 (+ abs-t-a 7)))
|
||
"Shabbat Nahamu"))))))
|
||
|
||
(autoload 'holiday-julian "cal-julian")
|
||
|
||
;;;###holiday-autoload
|
||
(defun holiday-hebrew-misc ()
|
||
"Miscellaneous Hebrew holidays, if visible in calendar window.
|
||
Includes: Tal Umatar, Tzom Teveth, Tu B'Shevat, Shabbat Shirah, and
|
||
Kiddush HaHamah."
|
||
(let ((m displayed-month)
|
||
(y displayed-year)
|
||
year h-year)
|
||
(append
|
||
(holiday-julian
|
||
11
|
||
(progn
|
||
(calendar-increment-month m y -1)
|
||
(setq year (calendar-extract-year
|
||
(calendar-julian-from-absolute
|
||
(calendar-absolute-from-gregorian (list m 1 y)))))
|
||
(if (zerop (% (1+ year) 4))
|
||
22
|
||
21)) "\"Tal Umatar\" (evening)")
|
||
(holiday-hebrew
|
||
10
|
||
(progn
|
||
(setq h-year (calendar-extract-year
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian
|
||
(list displayed-month 28 displayed-year)))))
|
||
(if (= 6 (% (calendar-hebrew-to-absolute (list 10 10 h-year))
|
||
7))
|
||
11 10))
|
||
"Tzom Teveth")
|
||
(holiday-hebrew 11 15 "Tu B'Shevat")
|
||
(holiday-hebrew
|
||
11
|
||
(progn
|
||
(setq m displayed-month
|
||
y displayed-year
|
||
h-year (progn
|
||
(calendar-increment-month m y 1)
|
||
(calendar-extract-year
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian
|
||
(list m (calendar-last-day-of-month m y) y))))))
|
||
(calendar-extract-day
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-dayname-on-or-before
|
||
6 (calendar-hebrew-to-absolute
|
||
(list 11
|
||
(if (= 6
|
||
(% (calendar-hebrew-to-absolute
|
||
(list 7 1 h-year))
|
||
7))
|
||
17 16) h-year))))))
|
||
"Shabbat Shirah")
|
||
(and (progn
|
||
(setq m displayed-month
|
||
y displayed-year
|
||
year (progn
|
||
(calendar-increment-month m y -1)
|
||
(calendar-extract-year
|
||
(calendar-julian-from-absolute
|
||
(calendar-absolute-from-gregorian (list m 1 y))))))
|
||
(= 21 (% year 28)))
|
||
(holiday-julian 3 26 "Kiddush HaHamah")))))
|
||
|
||
|
||
(autoload 'diary-list-entries-1 "diary-lib")
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-list-entries ()
|
||
"Add any Hebrew date entries from the diary file to `diary-entries-list'.
|
||
Hebrew date diary entries must be prefaced by `diary-hebrew-entry-symbol'
|
||
\(normally an `H'). The same diary date forms govern the style
|
||
of the Hebrew calendar entries, except that the Hebrew month
|
||
names cannot be abbreviated. The Hebrew months are numbered
|
||
from 1 to 13 with Nisan being 1, 12 being Adar I and 13 being
|
||
Adar II; you must use \"Adar I\" if you want Adar of a common
|
||
Hebrew year. If a Hebrew date diary entry begins with
|
||
`diary-nonmarking-symbol', the entry will appear in the diary
|
||
listing, but will not be marked in the calendar. This function
|
||
is provided for use with `diary-nongregorian-listing-hook'."
|
||
(diary-list-entries-1 calendar-hebrew-month-name-array-leap-year
|
||
diary-hebrew-entry-symbol
|
||
'calendar-hebrew-from-absolute))
|
||
|
||
(autoload 'calendar-mark-complex "diary-lib")
|
||
|
||
;;;###diary-autoload
|
||
(defun calendar-hebrew-mark-date-pattern (month day year &optional color)
|
||
"Mark dates in calendar window that conform to Hebrew date MONTH/DAY/YEAR.
|
||
A value of 0 in any position is a wildcard. Optional argument COLOR is
|
||
passed to `calendar-mark-visible-date' as MARK."
|
||
;; FIXME not the same as the Bahá’í and Islamic cases, so can't use
|
||
;; calendar-mark-1.
|
||
(with-current-buffer calendar-buffer
|
||
(if (and (not (zerop month)) (not (zerop day)))
|
||
(if (not (zerop year))
|
||
;; Fully specified Hebrew date.
|
||
(let ((date (calendar-gregorian-from-absolute
|
||
(calendar-hebrew-to-absolute
|
||
(list month day year)))))
|
||
(if (calendar-date-is-visible-p date)
|
||
(calendar-mark-visible-date date color)))
|
||
;; Month and day in any year.
|
||
(let ((gdate (calendar-hebrew-date-is-visible-p month day)))
|
||
(if gdate (calendar-mark-visible-date gdate color))))
|
||
(calendar-mark-complex month day year
|
||
'calendar-hebrew-from-absolute color))))
|
||
|
||
(autoload 'diary-mark-entries-1 "diary-lib")
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-mark-entries ()
|
||
"Mark days in the calendar window that have Hebrew date diary entries.
|
||
Marks each entry in `diary-file' (or included files) visible in the calendar
|
||
window. See `list-hebrew-diary-entries' for more information."
|
||
(diary-mark-entries-1 'calendar-hebrew-mark-date-pattern
|
||
calendar-hebrew-month-name-array-leap-year
|
||
diary-hebrew-entry-symbol
|
||
'calendar-hebrew-from-absolute))
|
||
|
||
(autoload 'diary-insert-entry-1 "diary-lib")
|
||
|
||
;;;###cal-autoload
|
||
(defun diary-hebrew-insert-entry (arg)
|
||
"Insert a diary entry for the Hebrew date at point.
|
||
Prefix argument ARG makes the entry nonmarking."
|
||
(interactive "P")
|
||
(diary-insert-entry-1 nil arg calendar-hebrew-month-name-array-leap-year
|
||
diary-hebrew-entry-symbol
|
||
'calendar-hebrew-from-absolute))
|
||
|
||
;;;###cal-autoload
|
||
(defun diary-hebrew-insert-monthly-entry (arg)
|
||
"Insert a monthly diary entry.
|
||
For the day of the Hebrew month corresponding to the date indicated by point.
|
||
Prefix argument ARG makes the entry nonmarking."
|
||
(interactive "P")
|
||
(diary-insert-entry-1 'monthly arg calendar-hebrew-month-name-array-leap-year
|
||
diary-hebrew-entry-symbol
|
||
'calendar-hebrew-from-absolute))
|
||
|
||
;;;###cal-autoload
|
||
(defun diary-hebrew-insert-yearly-entry (arg)
|
||
"Insert an annual diary entry.
|
||
For the day of the Hebrew year corresponding to the date indicated by point.
|
||
Prefix argument ARG makes the entry nonmarking."
|
||
(interactive "P")
|
||
(diary-insert-entry-1 'yearly arg calendar-hebrew-month-name-array-leap-year
|
||
diary-hebrew-entry-symbol
|
||
'calendar-hebrew-from-absolute))
|
||
|
||
;;;###autoload
|
||
(defun calendar-hebrew-list-yahrzeits (death-date start-year end-year)
|
||
"List Yahrzeit dates for *Gregorian* DEATH-DATE from START-YEAR to END-YEAR.
|
||
When called interactively from the calendar window, the date of death is taken
|
||
from the cursor position."
|
||
(interactive
|
||
(let* ((death-date
|
||
(if (equal (current-buffer) (get-buffer calendar-buffer))
|
||
(calendar-cursor-to-date t)
|
||
(let* ((today (calendar-current-date))
|
||
(year (calendar-read-sexp
|
||
"Year of death (>0)"
|
||
(lambda (x) (> x 0))
|
||
(calendar-extract-year today)))
|
||
(month-array calendar-month-name-array)
|
||
(completion-ignore-case t)
|
||
(month (cdr (assoc-string
|
||
(completing-read
|
||
"Month of death (name): "
|
||
(mapcar 'list (append month-array nil))
|
||
nil t)
|
||
(calendar-make-alist month-array 1) t)))
|
||
(last (calendar-last-day-of-month month year))
|
||
(day (calendar-read-sexp
|
||
"Day of death (1-%d)"
|
||
(lambda (x) (and (< 0 x) (<= x last)))
|
||
nil
|
||
last)))
|
||
(list month day year))))
|
||
(death-year (calendar-extract-year death-date))
|
||
(start-year (calendar-read-sexp
|
||
"Starting year of Yahrzeit table (>%d)"
|
||
(lambda (x) (> x death-year))
|
||
(1+ death-year)
|
||
death-year))
|
||
(end-year (calendar-read-sexp
|
||
"Ending year of Yahrzeit table (>=%d)"
|
||
(lambda (x) (>= x start-year))
|
||
nil
|
||
start-year)))
|
||
(list death-date start-year end-year)))
|
||
(message "Computing Yahrzeits...")
|
||
(let* ((h-date (calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian death-date)))
|
||
(h-year (calendar-extract-year h-date))
|
||
(i (1- start-year)))
|
||
(calendar-in-read-only-buffer calendar-hebrew-yahrzeit-buffer
|
||
(calendar-set-mode-line
|
||
(format "Yahrzeit dates for %s = %s"
|
||
(calendar-date-string death-date)
|
||
(let ((calendar-month-name-array
|
||
(if (calendar-hebrew-leap-year-p h-year)
|
||
calendar-hebrew-month-name-array-leap-year
|
||
calendar-hebrew-month-name-array-common-year)))
|
||
(calendar-date-string h-date nil t))))
|
||
(while (<= (setq i (1+ i)) end-year)
|
||
(insert
|
||
(calendar-date-string
|
||
(calendar-gregorian-from-absolute
|
||
(calendar-hebrew-yahrzeit
|
||
h-date
|
||
(calendar-extract-year
|
||
(calendar-hebrew-from-absolute
|
||
(calendar-absolute-from-gregorian (list 1 1 i))))))) "\n"))))
|
||
(message "Computing Yahrzeits...done"))
|
||
|
||
(defun calendar-hebrew-birthday (date year)
|
||
"Absolute date of the anniversary of Hebrew birth DATE, in Hebrew YEAR."
|
||
(let ((b-day (calendar-extract-day date))
|
||
(b-month (calendar-extract-month date))
|
||
(b-year (calendar-extract-year date)))
|
||
;; If it's Adar in a normal Hebrew year or Adar II in a Hebrew leap year...
|
||
(if (= b-month (calendar-hebrew-last-month-of-year b-year))
|
||
;; ...then use the same day in last month of Hebrew year.
|
||
(calendar-hebrew-to-absolute
|
||
(list (calendar-hebrew-last-month-of-year year) b-day year))
|
||
;; Else use the normal anniversary of the birth date,
|
||
;; or the corresponding day in years without that date.
|
||
(+ (calendar-hebrew-to-absolute (list b-month 1 year)) b-day -1))))
|
||
|
||
;; The function below is designed to be used in sexp diary entries,
|
||
;; and may be present in users' diary files, so suppress the warning
|
||
;; about this prefix-less dynamic variable. It's called from
|
||
;; `diary-list-sexp-entries', which binds the variable.
|
||
(with-suppressed-warnings ((lexical date))
|
||
(defvar date))
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-date ()
|
||
"Hebrew calendar equivalent of date diary entry."
|
||
(format "Hebrew date (until sunset): %s" (calendar-hebrew-date-string date)))
|
||
|
||
;; `diary-hebrew-birthday' can be used in users' diary files, and
|
||
;; `entry' has to be dynamically bound when that is used.
|
||
(with-suppressed-warnings ((lexical entry))
|
||
(defvar entry))
|
||
(declare-function diary-ordinal-suffix "diary-lib" (n))
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-birthday (month day year &optional after-sunset)
|
||
"Hebrew birthday diary entry.
|
||
Entry applies if date is birthdate (MONTH DAY YEAR), or the day before.
|
||
The order of the input parameters changes according to
|
||
`calendar-date-style' (e.g. to DAY MONTH YEAR in the European style).
|
||
|
||
Assumes the associated diary entry is the name of the person.
|
||
|
||
Although the date of birth is specified by the *civil* calendar,
|
||
this function determines the proper Hebrew calendar birthday.
|
||
If the optional argument AFTER-SUNSET is non-nil, this means the
|
||
birth occurred after local sunset on the given civil date.
|
||
In this case, the following civil date corresponds to the Hebrew birthday."
|
||
(let* ((h-date (calendar-hebrew-from-absolute
|
||
(+ (calendar-absolute-from-gregorian
|
||
(diary-make-date month day year))
|
||
(if after-sunset 1 0))))
|
||
(h-year (calendar-extract-year h-date)) ; birth-day
|
||
(d (calendar-absolute-from-gregorian date)) ; today
|
||
(h-yr (calendar-extract-year (calendar-hebrew-from-absolute d)))
|
||
(age (- h-yr h-year)) ; current H year - birth H-year
|
||
(b-date (calendar-hebrew-birthday h-date h-yr)))
|
||
(and (> age 0) (memq b-date (list d (1+ d)))
|
||
(format "%s's %d%s Hebrew birthday%s" entry age
|
||
(diary-ordinal-suffix age)
|
||
(if (= b-date d) "" " (evening)")))))
|
||
|
||
(defvar diary-hebrew-omer-sefirot
|
||
["Hesed" "Gevurah" "Tiferet" "Netzach" "Hod" "Yesod" "Malchut"]
|
||
"The order of Sefirot for counting the Omer.
|
||
See https://opensiddur.org/prayers/solilunar/solar-cycles/sefirat-haomer/the-order-of-counting-the-omer-in-the-spring/")
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-omer (&optional mark)
|
||
"Omer count diary entry.
|
||
Entry applies if date is within 50 days after Passover.
|
||
|
||
An optional parameter MARK specifies a face or single-character string to
|
||
use when highlighting the day in the calendar."
|
||
(let* ((passover
|
||
(calendar-hebrew-to-absolute
|
||
(list 1 15 (+ (calendar-extract-year date) 3760))))
|
||
(omer (- (calendar-absolute-from-gregorian date) passover))
|
||
(week (/ omer 7))
|
||
(day (% omer 7)))
|
||
(if (and (> omer 0) (< omer 50))
|
||
(cons mark
|
||
(format "Day %d%s of the omer (until sunset) %s she'be'%s"
|
||
omer
|
||
(if (zerop week)
|
||
""
|
||
(format ", that is, %d week%s%s"
|
||
week
|
||
(if (= week 1) "" "s")
|
||
(if (zerop day)
|
||
""
|
||
(format " and %d day%s"
|
||
day (if (= day 1) "" "s")))))
|
||
(aref diary-hebrew-omer-sefirot (% (+ 6 day) 7))
|
||
(aref diary-hebrew-omer-sefirot
|
||
(+ (if (zerop day) -1 0) week)))))))
|
||
|
||
(autoload 'diary-make-date "diary-lib")
|
||
|
||
(declare-function diary-ordinal-suffix "diary-lib" (n))
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-yahrzeit (death-month death-day death-year
|
||
&optional mark after-sunset)
|
||
"Yahrzeit diary entry--entry applies if date is Yahrzeit or the day before.
|
||
Parameters are DEATH-MONTH, DEATH-DAY, DEATH-YEAR; the diary
|
||
entry is assumed to be the name of the person. Although the date
|
||
of death is specified by the civil calendar, the proper Hebrew
|
||
calendar Yahrzeit is determined.
|
||
|
||
If the death occurred after local sunset on the given civil date,
|
||
the following civil date corresponds to the Hebrew date of
|
||
death--set the optional parameter AFTER-SUNSET non-nil in this case.
|
||
|
||
The order of the input parameters changes according to `calendar-date-style'
|
||
\(e.g. to DEATH-DAY, DEATH-MONTH, DEATH-YEAR in the European style).
|
||
|
||
An optional parameter MARK specifies a face or single-character string to
|
||
use when highlighting the day in the calendar."
|
||
(let* ((h-date (calendar-hebrew-from-absolute
|
||
(+ (calendar-absolute-from-gregorian
|
||
(diary-make-date death-month death-day death-year))
|
||
(if after-sunset 1 0))))
|
||
(h-year (calendar-extract-year h-date))
|
||
(d (calendar-absolute-from-gregorian date))
|
||
(yr (calendar-extract-year (calendar-hebrew-from-absolute d)))
|
||
(diff (- yr h-year))
|
||
(y (calendar-hebrew-yahrzeit h-date yr)))
|
||
(if (and (> diff 0) (or (= y d) (= y (1+ d))))
|
||
(cons mark
|
||
(format "Yahrzeit of %s%s: %d%s anniversary"
|
||
entry
|
||
(if (= y d) "" " (evening)")
|
||
diff
|
||
(diary-ordinal-suffix diff))))))
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-rosh-hodesh (&optional mark)
|
||
"Rosh Hodesh diary entry.
|
||
Entry applies if date is Rosh Hodesh, the day before, or the Saturday before.
|
||
|
||
An optional parameter MARK specifies a face or single-character string to
|
||
use when highlighting the day in the calendar."
|
||
(let* ((d (calendar-absolute-from-gregorian date))
|
||
(h-date (calendar-hebrew-from-absolute d))
|
||
(h-month (calendar-extract-month h-date))
|
||
(h-day (calendar-extract-day h-date))
|
||
(h-year (calendar-extract-year h-date))
|
||
(leap-year (calendar-hebrew-leap-year-p h-year))
|
||
(last-day (calendar-hebrew-last-day-of-month h-month h-year))
|
||
(h-month-names
|
||
(if leap-year
|
||
calendar-hebrew-month-name-array-leap-year
|
||
calendar-hebrew-month-name-array-common-year))
|
||
(this-month (aref h-month-names (1- h-month)))
|
||
(h-yesterday (calendar-extract-day
|
||
(calendar-hebrew-from-absolute (1- d)))))
|
||
(if (or (= h-day 30) (and (= h-day 1) (/= h-month 7)))
|
||
(cons mark
|
||
(format
|
||
"Rosh Hodesh %s"
|
||
(if (= h-day 30)
|
||
(format
|
||
"%s (first day)"
|
||
;; Next month must be in the same year since this
|
||
;; month can't be the last month of the year since
|
||
;; it has 30 days
|
||
(aref h-month-names h-month))
|
||
(if (= h-yesterday 30)
|
||
(format "%s (second day)" this-month)
|
||
this-month))))
|
||
(if (= (% d 7) 6) ; Saturday--check for Shabbat Mevarchim
|
||
(cond ((and (> h-day 22) (/= h-month 6) (= 29 last-day))
|
||
(cons mark
|
||
(format "Mevarchim Rosh Hodesh %s (%s)"
|
||
(aref h-month-names
|
||
(if (= h-month
|
||
(calendar-hebrew-last-month-of-year
|
||
h-year))
|
||
0 h-month))
|
||
(aref calendar-day-name-array (- 29 h-day)))))
|
||
((and (< h-day 30) (> h-day 22) (= 30 last-day))
|
||
(cons mark
|
||
(format "Mevarchim Rosh Hodesh %s (%s-%s)"
|
||
(aref h-month-names h-month)
|
||
(if (= h-day 29)
|
||
"tomorrow"
|
||
(aref calendar-day-name-array (- 29 h-day)))
|
||
(aref calendar-day-name-array
|
||
(% (- 30 h-day) 7))))))
|
||
(if (and (= h-day 29) (/= h-month 6))
|
||
(cons mark
|
||
(format "Erev Rosh Hodesh %s"
|
||
(aref h-month-names
|
||
(if (= h-month
|
||
(calendar-hebrew-last-month-of-year
|
||
h-year))
|
||
0 h-month)))))))))
|
||
|
||
(defconst calendar-hebrew-parashiot-names
|
||
["Bereshith" "Noah" "Lech L'cha" "Vayera" "Hayei Sarah" "Toledoth"
|
||
"Vayetze" "Vayishlah" "Vayeshev" "Mikketz" "Vayiggash" "Vayhi"
|
||
"Shemoth" "Vaera" "Bo" "Beshallah" "Yithro" "Mishpatim"
|
||
"Terumah" "Tetzavveh" "Ki Tissa" "Vayakhel" "Pekudei" "Vayikra"
|
||
"Tzav" "Shemini" "Tazria" "Metzora" "Aharei Moth" "Kedoshim"
|
||
"Emor" "Behar" "Behukkotai" "Bemidbar" "Naso" "Behaalot'cha"
|
||
"Shelah L'cha" "Korah" "Hukkath" "Balak" "Pinhas" "Mattoth"
|
||
"Masei" "Devarim" "Vaethanan" "Ekev" "Reeh" "Shofetim"
|
||
"Ki Tetze" "Ki Tavo" "Nitzavim" "Vayelech" "Haazinu"]
|
||
"The names of the parashiot in the Torah.")
|
||
|
||
(defun calendar-hebrew-parasha-name (p)
|
||
"Name(s) corresponding to parasha P."
|
||
(if (arrayp p) ; combined parasha
|
||
(format "%s/%s"
|
||
(aref calendar-hebrew-parashiot-names (aref p 0))
|
||
(aref calendar-hebrew-parashiot-names (aref p 1)))
|
||
(aref calendar-hebrew-parashiot-names p)))
|
||
|
||
;; Following 14 constants are used in diary-parasha (intern).
|
||
|
||
;; The seven ordinary year types (keviot).
|
||
(defconst calendar-hebrew-year-Saturday-incomplete-Sunday
|
||
[nil 52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22]
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 34 35 36 37 38 39 40 [41 42]
|
||
43 44 45 46 47 48 49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year starts on Saturday, is `incomplete' (Heshvan and Kislev each have
|
||
29 days), and has Passover start on Sunday.")
|
||
|
||
(defconst calendar-hebrew-year-Saturday-complete-Tuesday
|
||
[nil 52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22]
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 34 35 36 37 38 39 40 [41 42]
|
||
43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Saturday, is `complete' (Heshvan and Kislev each
|
||
have 30 days), and has Passover start on Tuesday.")
|
||
|
||
(defconst calendar-hebrew-year-Monday-incomplete-Tuesday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22]
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 34 35 36 37 38 39 40 [41 42]
|
||
43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Monday, is `incomplete' (Heshvan and Kislev each
|
||
have 29 days), and has Passover start on Tuesday.")
|
||
|
||
(defconst calendar-hebrew-year-Monday-complete-Thursday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22]
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 (nil . 34) (34 . 35) (35 . 36)
|
||
(36 . 37) (37 . 38) ([38 39] . 39) 40 [41 42] 43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Monday, is `complete' (Heshvan and Kislev each have
|
||
30 days), and has Passover start on Thursday.")
|
||
|
||
(defconst calendar-hebrew-year-Tuesday-regular-Thursday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22]
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 (nil . 34) (34 . 35) (35 . 36)
|
||
(36 . 37) (37 . 38) ([38 39] . 39) 40 [41 42] 43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Tuesday, is `regular' (Heshvan has 29 days and
|
||
Kislev has 30 days), and has Passover start on Thursday.")
|
||
|
||
(defconst calendar-hebrew-year-Thursday-regular-Saturday
|
||
[52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 [21 22] 23
|
||
24 nil (nil . 25) (25 . [26 27]) ([26 27] . [28 29]) ([28 29] . 30)
|
||
(30 . 31) ([31 32] . 32) 33 34 35 36 37 38 39 40 [41 42] 43 44 45 46 47 48
|
||
49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Thursday, is `regular' (Heshvan has 29 days and
|
||
Kislev has 30 days), and has Passover start on Saturday.")
|
||
|
||
(defconst calendar-hebrew-year-Thursday-complete-Sunday
|
||
[52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 nil 25 [26 27] [28 29] 30 [31 32] 33 34 35 36 37 38 39 40 [41 42]
|
||
43 44 45 46 47 48 49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Thursday, is `complete' (Heshvan and Kislev each
|
||
have 30 days), and has Passover start on Sunday.")
|
||
|
||
;; The seven leap year types (keviot).
|
||
(defconst calendar-hebrew-year-Saturday-incomplete-Tuesday
|
||
[nil 52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 nil 28 29 30 31 32 33 34 35 36 37 38 39 40 [41 42]
|
||
43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Saturday, is `incomplete' (Heshvan and Kislev each
|
||
have 29 days), and has Passover start on Tuesday.")
|
||
|
||
(defconst calendar-hebrew-year-Saturday-complete-Thursday
|
||
[nil 52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 nil 28 29 30 31 32 33 (nil . 34) (34 . 35) (35 . 36)
|
||
(36 . 37) (37 . 38) ([38 39] . 39) 40 [41 42] 43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Saturday, is `complete' (Heshvan and Kislev each
|
||
have 30 days), and has Passover start on Thursday.")
|
||
|
||
(defconst calendar-hebrew-year-Monday-incomplete-Thursday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 nil 28 29 30 31 32 33 (nil . 34) (34 . 35) (35 . 36)
|
||
(36 . 37) (37 . 38) ([38 39] . 39) 40 [41 42] 43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Monday, is `incomplete' (Heshvan and Kislev each
|
||
have 29 days), and has Passover start on Thursday.")
|
||
|
||
(defconst calendar-hebrew-year-Monday-complete-Saturday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 nil (nil . 28) (28 . 29) (29 . 30) (30 . 31) (31 . 32)
|
||
(32 . 33) (33 . 34) (34 . 35) (35 . 36) (36 . 37) (37 . 38) (38 . 39)
|
||
(39 . 40) (40 . 41) ([41 42] . 42) 43 44 45 46 47 48 49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Monday, is `complete' (Heshvan and Kislev each have
|
||
30 days), and has Passover start on Saturday.")
|
||
|
||
(defconst calendar-hebrew-year-Tuesday-regular-Saturday
|
||
[51 52 nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 nil (nil . 28) (28 . 29) (29 . 30) (30 . 31) (31 . 32)
|
||
(32 . 33) (33 . 34) (34 . 35) (35 . 36) (36 . 37) (37 . 38) (38 . 39)
|
||
(39 . 40) (40 . 41) ([41 42] . 42) 43 44 45 46 47 48 49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Tuesday, is `regular' (Heshvan has 29 days and
|
||
Kislev has 30 days), and has Passover start on Saturday.")
|
||
|
||
(defconst calendar-hebrew-year-Thursday-incomplete-Sunday
|
||
[52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 28 nil 29 30 31 32 33 34 35 36 37 38 39 40 41 42
|
||
43 44 45 46 47 48 49 50]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Thursday, is `incomplete' (Heshvan and Kislev both
|
||
have 29 days), and has Passover start on Sunday.")
|
||
|
||
(defconst calendar-hebrew-year-Thursday-complete-Tuesday
|
||
[52 nil nil 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
||
23 24 25 26 27 28 nil 29 30 31 32 33 34 35 36 37 38 39 40 41 42
|
||
43 44 45 46 47 48 49 [50 51]]
|
||
"The structure of the parashiot.
|
||
Hebrew year that starts on Thursday, is `complete' (Heshvan and Kislev both
|
||
have 30 days), and has Passover start on Tuesday.")
|
||
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-parasha (&optional mark)
|
||
"Parasha diary entry--entry applies if date is a Saturday.
|
||
An optional parameter MARK specifies a face or single-character string to
|
||
use when highlighting the day in the calendar."
|
||
(let ((d (calendar-absolute-from-gregorian date)))
|
||
(if (= (% d 7) 6) ; Saturday
|
||
(let* ((h-year (calendar-extract-year
|
||
(calendar-hebrew-from-absolute d)))
|
||
(rosh-hashanah
|
||
(calendar-hebrew-to-absolute (list 7 1 h-year)))
|
||
(passover
|
||
(calendar-hebrew-to-absolute (list 1 15 h-year)))
|
||
(rosh-hashanah-day
|
||
(aref calendar-day-name-array (% rosh-hashanah 7)))
|
||
(passover-day
|
||
(aref calendar-day-name-array (% passover 7)))
|
||
(long-h (calendar-hebrew-long-heshvan-p h-year))
|
||
(short-k (calendar-hebrew-short-kislev-p h-year))
|
||
(type (cond ((and long-h (not short-k)) "complete")
|
||
((and (not long-h) short-k) "incomplete")
|
||
(t "regular")))
|
||
(year-format
|
||
(symbol-value
|
||
(intern (format "calendar-hebrew-year-%s-%s-%s" ; keviah
|
||
rosh-hashanah-day type passover-day))))
|
||
(first-saturday ; of Hebrew year
|
||
(calendar-dayname-on-or-before 6 (+ 6 rosh-hashanah)))
|
||
(saturday ; which Saturday of the Hebrew year
|
||
(/ (- d first-saturday) 7))
|
||
(parasha (aref year-format saturday)))
|
||
(if parasha
|
||
(cons mark
|
||
(format
|
||
"Parashat %s"
|
||
(if (listp parasha) ; Israel differs from diaspora
|
||
(if (car parasha)
|
||
(format "%s (diaspora), %s (Israel)"
|
||
(calendar-hebrew-parasha-name
|
||
(car parasha))
|
||
(calendar-hebrew-parasha-name
|
||
(cdr parasha)))
|
||
(format "%s (Israel)"
|
||
(calendar-hebrew-parasha-name
|
||
(cdr parasha))))
|
||
(calendar-hebrew-parasha-name parasha)))))))))
|
||
|
||
|
||
(declare-function solar-setup "solar" ())
|
||
(declare-function solar-sunrise-sunset "solar" (date))
|
||
(defvar calendar-latitude)
|
||
(defvar calendar-longitude)
|
||
(defvar calendar-time-zone)
|
||
|
||
|
||
;; To be called from diary-list-sexp-entries, where DATE is bound.
|
||
;;;###diary-autoload
|
||
(defun diary-hebrew-sabbath-candles (&optional mark)
|
||
"Local time of candle lighting diary entry--applies if date is a Friday.
|
||
No diary entry if there is no sunset on that date. Uses
|
||
`diary-hebrew-sabbath-candles-minutes'.
|
||
|
||
An optional parameter MARK specifies a face or single-character string to
|
||
use when highlighting the day in the calendar."
|
||
(require 'solar)
|
||
(or (and calendar-latitude calendar-longitude calendar-time-zone)
|
||
(solar-setup))
|
||
(if (= (% (calendar-absolute-from-gregorian date) 7) 5) ; Friday
|
||
(let ((sunset (cadr (solar-sunrise-sunset date))))
|
||
(if sunset
|
||
(cons mark (format
|
||
"%s Sabbath candle lighting"
|
||
(apply 'solar-time-string
|
||
(cons (- (car sunset)
|
||
(/ diary-hebrew-sabbath-candles-minutes
|
||
60.0))
|
||
(cdr sunset)))))))))
|
||
|
||
|
||
(provide 'cal-hebrew)
|
||
|
||
;;; cal-hebrew.el ends here
|