Support hourly repeat cookies.

* org.el (org-repeat-re)
(org-clone-subtree-with-time-shift, org-auto-repeat-maybe)
(org-deadline, org-schedule, org-matcher-time)
(org-time-stamp, org-read-date, org-read-date-get-relative)
(org-display-custom-time, org-get-wdays)
(org-time-string-to-absolute, org-closest-date)
(org-timestamp-change): Allow to set hourly repeat cookie.
Send an error when an hourly repeat cookie is set and no hour
is specified in the timestamp.

* org.texi (Repeated tasks): Document repeat cookies for
years, months, weeks, days and hours.

* org-agenda.el (org-agenda-get-timestamps): Match hourly
repeat cookies.

* org-icalendar.el (org-print-icalendar-entries): Handle
hourly repeat cookies.

Thanks a lot to Takafumi Arakaki for this idea and for a preliminary
version of this patch.
This commit is contained in:
Bastien Guerry 2012-04-20 16:55:48 +02:00
parent 66ae794a66
commit ec921a2a68
4 changed files with 27 additions and 24 deletions

View File

@ -5848,9 +5848,10 @@ or plain timestamp. In the following example
@noindent @noindent
the @code{+1m} is a repeater; the intended interpretation is that the task the @code{+1m} is a repeater; the intended interpretation is that the task
has a deadline on <2005-10-01> and repeats itself every (one) month starting has a deadline on <2005-10-01> and repeats itself every (one) month starting
from that time. If you need both a repeater and a special warning period in from that time. You can use yearly, monthly, weekly, daily and hourly repeat
a deadline entry, the repeater should come first and the warning period last: cookies by using the @code{y/w/m/d/h} letters. If you need both a repeater
@code{DEADLINE: <2005-10-01 Sat +1m -3d>}. and a special warning period in a deadline entry, the repeater should come
first and the warning period last: @code{DEADLINE: <2005-10-01 Sat +1m -3d>}.
@vindex org-todo-repeat-to-state @vindex org-todo-repeat-to-state
Deadlines and scheduled items produce entries in the agenda when they are Deadlines and scheduled items produce entries in the agenda when they are

View File

@ -4962,7 +4962,7 @@ This function is invoked if `org-agenda-todo-ignore-deadlines',
(apply 'encode-time ; DATE bound by calendar (apply 'encode-time ; DATE bound by calendar
(list 0 0 0 (nth 1 date) (car date) (nth 2 date)))) (list 0 0 0 (nth 1 date) (car date) (nth 2 date))))
1 11)) 1 11))
"\\|\\(<[0-9]+-[0-9]+-[0-9]+[^>\n]+?\\+[0-9]+[dwmy]>\\)" "\\|\\(<[0-9]+-[0-9]+-[0-9]+[^>\n]+?\\+[0-9]+[hdwmy]>\\)"
"\\|\\(<%%\\(([^>\n]+)\\)>\\)")) "\\|\\(<%%\\(([^>\n]+)\\)>\\)"))
marker hdmarker deadlinep scheduledp clockp closedp inactivep marker hdmarker deadlinep scheduledp clockp closedp inactivep
donep tmp priority category org-category-pos ee txt timestr tags donep tmp priority category org-category-pos ee txt timestr tags

View File

@ -403,12 +403,12 @@ When COMBINE is non nil, add the category to each line."
(if (or (string-match org-tr-regexp hd) (if (or (string-match org-tr-regexp hd)
(string-match org-ts-regexp hd)) (string-match org-ts-regexp hd))
(setq hd (replace-match "" t t hd))) (setq hd (replace-match "" t t hd)))
(if (string-match "\\+\\([0-9]+\\)\\([dwmy]\\)>" ts) (if (string-match "\\+\\([0-9]+\\)\\([hdwmy]\\)>" ts)
(setq rrule (setq rrule
(concat "\nRRULE:FREQ=" (concat "\nRRULE:FREQ="
(cdr (assoc (cdr (assoc
(match-string 2 ts) (match-string 2 ts)
'(("d" . "DAILY")("w" . "WEEKLY") '(("h" . "HOURLY")("d" . "DAILY")("w" . "WEEKLY")
("m" . "MONTHLY")("y" . "YEARLY")))) ("m" . "MONTHLY")("y" . "YEARLY"))))
";INTERVAL=" (match-string 1 ts))) ";INTERVAL=" (match-string 1 ts)))
(setq rrule "")) (setq rrule ""))

View File

@ -694,7 +694,7 @@ An entry can be toggled between QUOTE and normal with
:type 'string) :type 'string)
(defconst org-repeat-re (defconst org-repeat-re
"<[0-9]\\{4\\}-[0-9][0-9]-[0-9][0-9] [^>\n]*?\\([.+]?\\+[0-9]+[dwmy]\\(/[0-9]+[dwmy]\\)?\\)" "<[0-9]\\{4\\}-[0-9][0-9]-[0-9][0-9] [^>\n]*?\\([.+]?\\+[0-9]+[hdwmy]\\(/[0-9]+[hdwmy]\\)?\\)"
"Regular expression for specifying repeated events. "Regular expression for specifying repeated events.
After a match, group 1 contains the repeat expression.") After a match, group 1 contains the repeat expression.")
@ -7967,7 +7967,7 @@ and still retain the repeater to cover future instances of the task."
(if (not (and (integerp n) (> n 0))) (if (not (and (integerp n) (> n 0)))
(error "Invalid number of replications %s" n)) (error "Invalid number of replications %s" n))
(if (and (setq doshift (and (stringp shift) (string-match "\\S-" shift))) (if (and (setq doshift (and (stringp shift) (string-match "\\S-" shift)))
(not (string-match "\\`[ \t]*\\+?\\([0-9]+\\)\\([dwmy]\\)[ \t]*\\'" (not (string-match "\\`[ \t]*\\+?\\([0-9]+\\)\\([hdwmy]\\)[ \t]*\\'"
shift))) shift)))
(error "Invalid shift specification %s" shift)) (error "Invalid shift specification %s" shift))
(when doshift (when doshift
@ -7985,7 +7985,7 @@ and still retain the repeater to cover future instances of the task."
(setq end (point)) (setq end (point))
(setq template (buffer-substring beg end)) (setq template (buffer-substring beg end))
(when (and doshift (when (and doshift
(string-match "<[^<>\n]+ [.+]?\\+[0-9]+[dwmy][^<>\n]*>" template)) (string-match "<[^<>\n]+ [.+]?\\+[0-9]+[hdwmy][^<>\n]*>" template))
(delete-region beg end) (delete-region beg end)
(setq end beg) (setq end beg)
(setq nmin 0 nmax (1+ nmax) n-no-remove nmax)) (setq nmin 0 nmax (1+ nmax) n-no-remove nmax))
@ -8016,7 +8016,7 @@ and still retain the repeater to cover future instances of the task."
(while (re-search-forward org-ts-regexp nil t) (while (re-search-forward org-ts-regexp nil t)
(save-excursion (save-excursion
(goto-char (match-beginning 0)) (goto-char (match-beginning 0))
(if (looking-at "<[^<>\n]+\\( +[.+]?\\+[0-9]+[dwmy]\\)") (if (looking-at "<[^<>\n]+\\( +[.+]?\\+[0-9]+[hdwmy]\\)")
(delete-region (match-beginning 1) (match-end 1))))))) (delete-region (match-beginning 1) (match-end 1)))))))
(setq task (buffer-string))) (setq task (buffer-string)))
(insert task)) (insert task))
@ -11936,7 +11936,7 @@ This function is run automatically after each state change to a DONE state."
(aa (assoc org-last-state org-todo-kwd-alist)) (aa (assoc org-last-state org-todo-kwd-alist))
(interpret (nth 1 aa)) (interpret (nth 1 aa))
(head (nth 2 aa)) (head (nth 2 aa))
(whata '(("d" . day) ("m" . month) ("y" . year))) (whata '(("h" . hour) ("d" . day) ("m" . month) ("y" . year)))
(msg "Entry repeats: ") (msg "Entry repeats: ")
(org-log-done nil) (org-log-done nil)
(org-todo-log-states nil) (org-todo-log-states nil)
@ -11972,10 +11972,12 @@ This function is run automatically after each state change to a DONE state."
(setq type (if (match-end 1) org-scheduled-string (setq type (if (match-end 1) org-scheduled-string
(if (match-end 3) org-deadline-string "Plain:")) (if (match-end 3) org-deadline-string "Plain:"))
ts (match-string (if (match-end 2) 2 (if (match-end 4) 4 0)))) ts (match-string (if (match-end 2) 2 (if (match-end 4) 4 0))))
(when (string-match "\\([.+]\\)?\\(\\+[0-9]+\\)\\([dwmy]\\)" ts) (when (string-match "\\([.+]\\)?\\(\\+[0-9]+\\)\\([hdwmy]\\)" ts)
(setq n (string-to-number (match-string 2 ts)) (setq n (string-to-number (match-string 2 ts))
what (match-string 3 ts)) what (match-string 3 ts))
(if (equal what "w") (setq n (* n 7) what "d")) (if (equal what "w") (setq n (* n 7) what "d"))
(if (and (equal what "h") (not (string-match "[0-9]\\{1,2\\}:[0-9]\\{2\\}" ts)))
(error "Cannot repeat in Repeat in %d hour(s) because no hour has been set" n))
;; Preparation, see if we need to modify the start date for the change ;; Preparation, see if we need to modify the start date for the change
(when (match-end 1) (when (match-end 1)
(setq time (save-match-data (org-time-string-to-time ts))) (setq time (save-match-data (org-time-string-to-time ts)))
@ -12001,7 +12003,7 @@ This function is run automatically after each state change to a DONE state."
;; rematch, so that we have everything in place for the real shift ;; rematch, so that we have everything in place for the real shift
(org-at-timestamp-p t) (org-at-timestamp-p t)
(setq ts (match-string 1)) (setq ts (match-string 1))
(string-match "\\([.+]\\)?\\(\\+[0-9]+\\)\\([dwmy]\\)" ts)))) (string-match "\\([.+]\\)?\\(\\+[0-9]+\\)\\([hdwmy]\\)" ts))))
(org-timestamp-change n (cdr (assoc what whata))) (org-timestamp-change n (cdr (assoc what whata)))
(setq msg (concat msg type " " org-last-changed-timestamp " ")))) (setq msg (concat msg type " " org-last-changed-timestamp " "))))
(setq org-log-post-message msg) (setq org-log-post-message msg)
@ -12048,7 +12050,7 @@ can either be an Org date like \"2011-07-24\" or a delta like \"+2d\"."
(let* ((old-date (org-entry-get nil "DEADLINE")) (let* ((old-date (org-entry-get nil "DEADLINE"))
(repeater (and old-date (repeater (and old-date
(string-match (string-match
"\\([.+-]+[0-9]+[dwmy]\\(?:[/ ][-+]?[0-9]+[dwmy]\\)?\\) ?" "\\([.+-]+[0-9]+[hdwmy]\\(?:[/ ][-+]?[0-9]+[hdwmy]\\)?\\) ?"
old-date) old-date)
(match-string 1 old-date)))) (match-string 1 old-date))))
(if remove (if remove
@ -12096,7 +12098,7 @@ either be an Org date like \"2011-07-24\" or a delta like \"+2d\"."
(let* ((old-date (org-entry-get nil "SCHEDULED")) (let* ((old-date (org-entry-get nil "SCHEDULED"))
(repeater (and old-date (repeater (and old-date
(string-match (string-match
"\\([.+-]+[0-9]+[dwmy]\\(?:[/ ][-+]?[0-9]+[dwmy]\\)?\\) ?" "\\([.+-]+[0-9]+[hdwmy]\\(?:[/ ][-+]?[0-9]+[hdwmy]\\)?\\) ?"
old-date) old-date)
(match-string 1 old-date)))) (match-string 1 old-date))))
(if remove (if remove
@ -13250,7 +13252,7 @@ epoch to the beginning of today (00:00)."
((string= s "<today>") (org-time-today)) ((string= s "<today>") (org-time-today))
((string= s "<tomorrow>") (+ 86400.0 (org-time-today))) ((string= s "<tomorrow>") (+ 86400.0 (org-time-today)))
((string= s "<yesterday>") (- (org-time-today) 86400.0)) ((string= s "<yesterday>") (- (org-time-today) 86400.0))
((string-match "^<\\([-+][0-9]+\\)\\([dwmy]\\)>$" s) ((string-match "^<\\([-+][0-9]+\\)\\([hdwmy]\\)>$" s)
(+ (org-time-today) (+ (org-time-today)
(* (string-to-number (match-string 1 s)) (* (string-to-number (match-string 1 s))
(cdr (assoc (match-string 2 s) (cdr (assoc (match-string 2 s)
@ -14946,7 +14948,7 @@ at the cursor, it will be modified."
(save-match-data (save-match-data
(beginning-of-line) (beginning-of-line)
(when (re-search-forward (when (re-search-forward
"\\([.+-]+[0-9]+[dwmy] ?\\)+" ;;\\(?:[/ ][-+]?[0-9]+[dwmy]\\)?\\) ?" "\\([.+-]+[0-9]+[hdwmy] ?\\)+" ;;\\(?:[/ ][-+]?[0-9]+[hdwmy]\\)?\\) ?"
(save-excursion (progn (end-of-line) (point))) t) (save-excursion (progn (end-of-line) (point))) t)
(match-string 0))))) (match-string 0)))))
org-time-was-given org-end-time-was-given time) org-time-was-given org-end-time-was-given time)
@ -15048,7 +15050,7 @@ mean next year. For details, see the manual. A few examples:
etc. etc.
Furthermore you can specify a relative date by giving, as the *first* thing Furthermore you can specify a relative date by giving, as the *first* thing
in the input: a plus/minus sign, a number and a letter [dwmy] to indicate in the input: a plus/minus sign, a number and a letter [hdwmy] to indicate
change in days weeks, months, years. change in days weeks, months, years.
With a single plus or minus, the date is relative to today. With a double With a single plus or minus, the date is relative to today. With a double
plus or minus, it is relative to the date in DEFAULT-TIME. E.g. plus or minus, it is relative to the date in DEFAULT-TIME. E.g.
@ -15466,7 +15468,7 @@ DEF-FLAG is t when a double ++ or -- indicates shift relative to
(concat (concat
"\\`[ \t]*\\([-+]\\{0,2\\}\\)" "\\`[ \t]*\\([-+]\\{0,2\\}\\)"
"\\([0-9]+\\)?" "\\([0-9]+\\)?"
"\\([dwmy]\\|\\(" (mapconcat 'car parse-time-weekdays "\\|") "\\)\\)?" "\\([hdwmy]\\|\\(" (mapconcat 'car parse-time-weekdays "\\|") "\\)\\)?"
"\\([ \t]\\|$\\)") s) "\\([ \t]\\|$\\)") s)
(or (> (match-end 1) (match-beginning 1)) (match-end 4))) (or (> (match-end 1) (match-beginning 1)) (match-end 4)))
(let* ((dir (if (> (match-end 1) (match-beginning 1)) (let* ((dir (if (> (match-end 1) (match-beginning 1))
@ -15583,7 +15585,7 @@ The command returns the inserted time stamp."
t1 w1 with-hm tf time str w2 (off 0)) t1 w1 with-hm tf time str w2 (off 0))
(save-match-data (save-match-data
(setq t1 (org-parse-time-string ts t)) (setq t1 (org-parse-time-string ts t))
(if (string-match "\\(-[0-9]+:[0-9]+\\)?\\( [.+]?\\+[0-9]+[dwmy]\\(/[0-9]+[dwmy]\\)?\\)?\\'" ts) (if (string-match "\\(-[0-9]+:[0-9]+\\)?\\( [.+]?\\+[0-9]+[hdwmy]\\(/[0-9]+[hdwmy]\\)?\\)?\\'" ts)
(setq off (- (match-end 0) (match-beginning 0))))) (setq off (- (match-end 0) (match-beginning 0)))))
(setq end (- end off)) (setq end (- end off))
(setq w1 (- end beg) (setq w1 (- end beg)
@ -15654,7 +15656,7 @@ Don't touch the rest."
((<= org-deadline-warning-days 0) ((<= org-deadline-warning-days 0)
;; 0 or negative, enforce this value no matter what ;; 0 or negative, enforce this value no matter what
(- org-deadline-warning-days)) (- org-deadline-warning-days))
((string-match "-\\([0-9]+\\)\\([dwmy]\\)\\(\\'\\|>\\| \\)" ts) ((string-match "-\\([0-9]+\\)\\([hdwmy]\\)\\(\\'\\|>\\| \\)" ts)
;; lead time is specified. ;; lead time is specified.
(floor (* (string-to-number (match-string 1 ts)) (floor (* (string-to-number (match-string 1 ts))
(cdr (assoc (match-string 2 ts) (cdr (assoc (match-string 2 ts)
@ -15845,7 +15847,7 @@ The variable date is bound by the calendar when this is called."
(if (org-diary-sexp-entry (match-string 1 s) "" date) (if (org-diary-sexp-entry (match-string 1 s) "" date)
daynr daynr
(+ daynr 1000))) (+ daynr 1000)))
((and daynr (string-match "\\+[0-9]+[dwmy]" s)) ((and daynr (string-match "\\+[0-9]+[hdwmy]" s))
(org-closest-date s (if (and (boundp 'daynr) (integerp daynr)) daynr (org-closest-date s (if (and (boundp 'daynr) (integerp daynr)) daynr
(time-to-days (current-time))) (match-string 0 s) (time-to-days (current-time))) (match-string 0 s)
prefer show-all)) prefer show-all))
@ -15956,7 +15958,7 @@ When SHOW-ALL is nil, only return the current occurrence of a time stamp."
(if (<= cday sday) (throw 'exit sday)) (if (<= cday sday) (throw 'exit sday))
(if (string-match "\\(\\+[0-9]+\\)\\([dwmy]\\)" change) (if (string-match "\\(\\+[0-9]+\\)\\([hdwmy]\\)" change)
(setq dn (string-to-number (match-string 1 change)) (setq dn (string-to-number (match-string 1 change))
dw (cdr (assoc (match-string 2 change) a1))) dw (cdr (assoc (match-string 2 change) a1)))
(error "Invalid change specifier: %s" change)) (error "Invalid change specifier: %s" change))
@ -16140,7 +16142,7 @@ in the timestamp determines what will be changed."
ts (match-string 0)) ts (match-string 0))
(replace-match "") (replace-match "")
(if (string-match (if (string-match
"\\(\\(-[012][0-9]:[0-5][0-9]\\)?\\( +[.+]?[-+][0-9]+[dwmy]\\(/[0-9]+[dwmy]\\)?\\)*\\)[]>]" "\\(\\(-[012][0-9]:[0-5][0-9]\\)?\\( +[.+]?[-+][0-9]+[hdwmy]\\(/[0-9]+[hdwmy]\\)?\\)*\\)[]>]"
ts) ts)
(setq extra (match-string 1 ts))) (setq extra (match-string 1 ts)))
(if (string-match "^.\\{10\\}.*?[0-9]+:[0-9][0-9]" ts) (if (string-match "^.\\{10\\}.*?[0-9]+:[0-9][0-9]" ts)