emacs/lisp/calendar/icalendar.el
Eli Zaretskii 71f8b2c324 Fix Icalendar export with ISO dates
* lisp/calendar/icalendar.el (icalendar--datestring-to-isodate):
Accept dashes in ISO-style numeric dates.  Patch by Erwan Hingant
<erwan.hingant@mailo.com>. (Bug#69894)

* test/lisp/calendar/icalendar-tests.el
(icalendar--datestring-to-isodate): Add a test for dashes in
ISO-style numeric dates.
2024-04-13 10:52:55 +03:00

2564 lines
110 KiB
EmacsLisp

;;; icalendar.el --- iCalendar implementation -*- lexical-binding: t -*-
;; Copyright (C) 2002-2024 Free Software Foundation, Inc.
;; Author: Ulf Jasper <ulf.jasper@web.de>
;; Maintainer: emacs-devel@gnu.org
;; Created: August 2002
;; Keywords: calendar
;; Human-Keywords: calendar, diary, iCalendar, vCalendar
;; Old-Version: 0.19
;; 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:
;; This package is documented in the Emacs Manual.
;; Please note:
;; - Diary entries which have a start time but no end time are assumed to
;; last for one hour when they are exported.
;; - Weekly diary entries are assumed to occur the first time in the first
;; week of the year 2000 when they are exported.
;; - Yearly diary entries are assumed to occur the first time in the year
;; 1900 when they are exported.
;; - Float diary entries are assumed to occur the first time on the
;; day when they are exported.
;;; History:
;; 0.07 onwards: see commit logs and ../ChangeLog*.
;; 0.06: (2004-10-06)
;; - Bugfixes regarding icalendar-import-format-*.
;; - Fix in icalendar-export-file -- thanks to Philipp Grau.
;; 0.05: (2003-06-19)
;; - New import format scheme: Replaced icalendar-import-prefix-*,
;; icalendar-import-ignored-properties, and
;; icalendar-import-separator with icalendar-import-format(-*).
;; - icalendar-import-file and icalendar-export-file
;; have an extra parameter which should prevent them from
;; erasing their target files (untested!).
;; - Tested with Emacs 21.3.2
;; 0.04:
;; - Bugfix: import: double quoted param values did not work
;; - Read DURATION property when importing.
;; - Added parameter icalendar-duration-correction.
;; 0.03: (2003-05-07)
;; - Export takes care of european-calendar-style.
;; - Tested with Emacs 21.3.2 and XEmacs 21.4.12
;; 0.02:
;; - Should work in XEmacs now. Thanks to Len Trigg for the XEmacs patches!
;; - Added exporting from Emacs diary to ical.
;; - Some bugfixes, after testing with calendars from https://icalshare.com.
;; - Tested with Emacs 21.3.2 and XEmacs 21.4.12
;; 0.01: (2003-03-21)
;; - First published version. Trial version. Alpha version.
;; ======================================================================
;; To Do:
;; * Import from ical to diary:
;; + Need more properties for icalendar-import-format
;; (added all that Mozilla Calendar uses)
;; From iCal specifications (RFC2445: 4.8.1), icalendar.el lacks
;; ATTACH, CATEGORIES, COMMENT, GEO, PERCENT-COMPLETE (VTODO),
;; PRIORITY, RESOURCES) not considering date/time and time-zone
;; + check vcalendar version
;; + check (unknown) elements
;; + recurring events!
;; + works for european style calendars only! Does it?
;; + alarm
;; + exceptions in recurring events
;; + the parser is too soft
;; + error log is incomplete
;; + nice to have: #include "webcal://foo.com/some-calendar.ics"
;; + timezones probably still need some improvements.
;; * Export from diary to ical
;; + diary-date, diary-float, and self-made sexp entries are not
;; understood
;; * Other things
;; + clean up all those date/time parsing functions
;; + Handle todo items?
;; + Check iso 8601 for datetime and period
;; + Which chars to (un)escape?
;;; Code:
;; ======================================================================
;; Customizables
;; ======================================================================
(defgroup icalendar nil
"iCalendar support."
:prefix "icalendar-"
:group 'calendar)
(defcustom icalendar-import-format
"%s%d%l%o"
"Format for importing events from iCalendar into Emacs diary.
It defines how iCalendar events are inserted into diary file.
This may either be a string or a function.
In case of a formatting STRING the following specifiers can be used:
%c Class, see `icalendar-import-format-class'
%d Description, see `icalendar-import-format-description'
%l Location, see `icalendar-import-format-location'
%o Organizer, see `icalendar-import-format-organizer'
%s Summary, see `icalendar-import-format-summary'
%t Status, see `icalendar-import-format-status'
%u URL, see `icalendar-import-format-url'
%U UID, see `icalendar-import-format-uid'
A formatting FUNCTION will be called with a VEVENT as its only
argument. It must return a string. See
`icalendar-import-format-sample' for an example."
:type '(choice
(string :tag "String")
(function :tag "Function"))
:group 'icalendar)
(defcustom icalendar-import-format-summary
"%s"
"Format string defining how the summary element is formatted.
This applies only if the summary is not empty! `%s' is replaced
by the summary."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-description
"\n Desc: %s"
"Format string defining how the description element is formatted.
This applies only if the description is not empty! `%s' is
replaced by the description."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-location
"\n Location: %s"
"Format string defining how the location element is formatted.
This applies only if the location is not empty! `%s' is replaced
by the location."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-organizer
"\n Organizer: %s"
"Format string defining how the organizer element is formatted.
This applies only if the organizer is not empty! `%s' is
replaced by the organizer."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-url
"\n URL: %s"
"Format string defining how the URL element is formatted.
This applies only if the URL is not empty! `%s' is replaced by
the URL."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-uid
"\n UID: %s"
"Format string defining how the UID element is formatted.
This applies only if the UID is not empty! `%s' is replaced by
the UID."
:type 'string
:version "24.3"
:group 'icalendar)
(defcustom icalendar-import-format-status
"\n Status: %s"
"Format string defining how the status element is formatted.
This applies only if the status is not empty! `%s' is replaced by
the status."
:type 'string
:group 'icalendar)
(defcustom icalendar-import-format-class
"\n Class: %s"
"Format string defining how the class element is formatted.
This applies only if the class is not empty! `%s' is replaced by
the class."
:type 'string
:group 'icalendar)
(defcustom icalendar-recurring-start-year
2005
"Start year for recurring events.
Some calendar browsers only propagate recurring events for
several years beyond the start time. Set this string to a year
just before the start of your personal calendar."
:type 'integer
:group 'icalendar)
(defcustom icalendar-export-hidden-diary-entries
t
"Determines whether hidden diary entries are exported.
If non-nil hidden diary entries (starting with `&') get exported,
if nil they are ignored."
:type 'boolean
:group 'icalendar)
(defcustom icalendar-uid-format
"emacs%t%c"
"Format of unique ID code (UID) for each iCalendar object.
The following specifiers are available:
%c COUNTER, an integer value that is increased each time a uid is
generated. This may be necessary for systems which do not
provide time-resolution finer than a second.
%h HASH, a hash value of the diary entry,
%s DTSTART, the start date (excluding time) of the diary entry,
%t TIMESTAMP, a unique creation timestamp,
%u USERNAME, the variable `user-login-name'.
For example, a value of \"%s_%h@mydomain.com\" will generate a
UID code for each entry composed of the time of the event, a hash
code for the event, and your personal domain name."
:type 'string
:group 'icalendar)
(defcustom icalendar-export-sexp-enumeration-days
14
"Number of days over which a sexp diary entry is enumerated.
In general sexp entries cannot be translated to icalendar format.
They are therefore enumerated, i.e. explicitly evaluated for a
certain number of days, and then exported. The enumeration starts
on the current day and continues for the number of days given here.
See `icalendar-export-sexp-enumerate-all' for a list of sexp
entries which by default are NOT enumerated."
:version "25.1"
:type 'integer
:group 'icalendar)
(defcustom icalendar-export-sexp-enumerate-all
nil
"Determines whether ALL sexp diary entries are enumerated.
If non-nil all sexp diary entries are enumerated for
`icalendar-export-sexp-enumeration-days' days instead of
translating into an icalendar equivalent. This affects the
following sexp diary entries: `diary-anniversary',
`diary-cyclic', `diary-date', `diary-float', `diary-block'. All
other sexp entries are enumerated in any case."
:version "25.1"
:type 'boolean
:group 'icalendar)
(defcustom icalendar-export-alarms
nil
"Determine if and how alarms are included in exported diary events."
:version "25.1"
:type '(choice (const :tag "Do not include alarms in export"
nil)
(list :tag "Create alarms in exported diary entries"
(integer :tag "Advance time (minutes)"
:value 10)
(set :tag "Alarm type"
(list :tag "Audio"
(const :tag "Audio" audio))
(list :tag "Display"
(const :tag "Display" display))
(list :tag "Email"
(const email)
(repeat :tag "Attendees"
(string :tag "Email"))))))
:group 'icalendar)
(defvar icalendar-debug nil
"Enable icalendar debug messages.")
;; ======================================================================
;; NO USER SERVICEABLE PARTS BELOW THIS LINE
;; ======================================================================
(defconst icalendar--weekday-array ["SU" "MO" "TU" "WE" "TH" "FR" "SA"])
;; ======================================================================
;; all the other libs we need
;; ======================================================================
(require 'calendar)
(require 'diary-lib)
;; ======================================================================
;; misc
;; ======================================================================
(defun icalendar--dmsg (&rest args)
"Print message ARGS if `icalendar-debug' is non-nil."
(if icalendar-debug
(apply 'message args)))
;; ======================================================================
;; Core functionality
;; Functions for parsing icalendars, importing and so on
;; ======================================================================
(defun icalendar--get-unfolded-buffer (folded-ical-buffer)
"Return a new buffer containing the unfolded contents of a buffer.
Folding is the iCalendar way of wrapping long lines. In the
created buffer all occurrences of CR LF BLANK are replaced by the
empty string. Argument FOLDED-ICAL-BUFFER is the folded input
buffer."
(let ((unfolded-buffer (get-buffer-create " *icalendar-work*")))
(save-current-buffer
(set-buffer unfolded-buffer)
(erase-buffer)
(insert-buffer-substring folded-ical-buffer)
(icalendar--clean-up-line-endings)
(goto-char (point-min))
(while (re-search-forward "\r?\n[ \t]" nil t)
(replace-match "" nil nil)))
unfolded-buffer))
(defun icalendar--clean-up-line-endings ()
"Replace DOS- and MAC-like line endings with unix line endings.
All occurrences of (CR LF) and (LF CF) are replaced with LF in
the current buffer. This is necessary in buffers which contain a
mix of different line endings."
(save-excursion
(goto-char (point-min))
(while (re-search-forward "\r\n\\|\n\r" nil t)
(replace-match "\n" nil nil))))
(define-obsolete-function-alias 'icalendar--rris
'replace-regexp-in-string "27.1")
(defun icalendar--read-element (invalue inparams)
"Recursively read the next iCalendar element in the current buffer.
INVALUE gives the current iCalendar element we are reading.
INPARAMS gives the current parameters.....
This function calls itself recursively for each nested calendar element
it finds. The current buffer should be an unfolded buffer as returned
from `icalendar--get-unfolded-buffer'."
(let (element children line name params param param-name param-value
value
(continue t))
(setq children '())
(while (and continue
(re-search-forward "^\\([A-Za-z0-9-]+\\)[;:]" nil t))
(setq name (intern (match-string 1)))
(backward-char 1)
(setq params '())
(setq line '())
(while (looking-at ";")
(re-search-forward ";\\([A-Za-z0-9-]+\\)=" nil nil)
(setq param-name (intern (match-string 1)))
(re-search-forward "\\(\\([^;,:\"]+\\)\\|\"\\([^\"]+\\)\"\\)[;:]"
nil t)
(backward-char 1)
(setq param-value (or (match-string 2) (match-string 3)))
(setq param (list param-name param-value))
(while (looking-at ",")
(re-search-forward "\\(\\([^;,:]+\\)\\|\"\\([^\"]+\\)\"\\)"
nil t)
(if (match-string 2)
(setq param-value (match-string 2))
(setq param-value (match-string 3)))
(setq param (append param param-value)))
(setq params (append params param)))
(unless (looking-at ":")
(error "Oops"))
(forward-char 1)
(let ((start (point)))
(end-of-line)
(setq value (buffer-substring start (point))))
(setq line (list name params value))
(cond ((eq name 'BEGIN)
(setq children
(append children
(list (icalendar--read-element (intern value)
params)))))
((eq name 'END)
(setq continue nil))
(t
(setq element (append element (list line))))))
(if invalue
(list invalue inparams element children)
children)))
;; ======================================================================
;; helper functions for examining events
;; ======================================================================
;;(defsubst icalendar--get-all-event-properties (event)
;; "Return the list of properties in this EVENT."
;; (car (cddr event)))
(defun icalendar--get-event-property (event prop)
"For the given EVENT return the value of the first occurrence of PROP."
(catch 'found
(let ((props (car (cddr event))) pp)
(while props
(setq pp (car props))
(if (eq (car pp) prop)
(throw 'found (car (cddr pp))))
(setq props (cdr props))))
nil))
(defun icalendar--get-event-property-attributes (event prop)
"For the given EVENT return attributes of the first occurrence of PROP."
(catch 'found
(let ((props (car (cddr event))) pp)
(while props
(setq pp (car props))
(if (eq (car pp) prop)
(throw 'found (cadr pp)))
(setq props (cdr props))))
nil))
(defun icalendar--get-event-properties (event prop)
"For the given EVENT return a list of all values of the property PROP."
(let ((props (car (cddr event))) pp result)
(while props
(setq pp (car props))
(if (eq (car pp) prop)
(setq result (append (split-string (car (cddr pp)) ",") result)))
(setq props (cdr props)))
result))
;; (defun icalendar--set-event-property (event prop new-value)
;; "For the given EVENT set the property PROP to the value NEW-VALUE."
;; (catch 'found
;; (let ((props (car (cddr event))) pp)
;; (while props
;; (setq pp (car props))
;; (when (eq (car pp) prop)
;; (setcdr (cdr pp) new-value)
;; (throw 'found (car (cddr pp))))
;; (setq props (cdr props)))
;; (setq props (car (cddr event)))
;; (setcar (cddr event)
;; (append props (list (list prop nil new-value)))))))
(defun icalendar--get-children (node name)
"Return all children of the given NODE which have a name NAME.
For instance the VCALENDAR node can have VEVENT children as well as VTODO
children."
(let ((result nil)
(children (cadr (cddr node))))
(when (eq (car node) name)
(setq result node))
;;(message "%s" node)
(when children
(let ((subresult
(delq nil
(mapcar (lambda (n)
(icalendar--get-children n name))
children))))
(if subresult
(if result
(setq result (append result subresult))
(setq result subresult)))))
result))
;; private
(defun icalendar--all-events (icalendar)
"Return the list of all existing events in the given ICALENDAR."
(let ((result '()))
(mapc (lambda (elt)
(setq result (append (icalendar--get-children elt 'VEVENT)
result)))
(nreverse icalendar))
result))
(defun icalendar--split-value (value-string)
"Split VALUE-STRING at `;='."
(let ((result '())
param-name param-value)
(when value-string
(save-current-buffer
(set-buffer (get-buffer-create " *icalendar-work*"))
(set-buffer-modified-p nil)
(erase-buffer)
(insert value-string)
(goto-char (point-min))
(while
(re-search-forward
"\\([A-Za-z0-9-]+\\)=\\(\\([^;:]+\\)\\|\"\\([^\"]+\\)\"\\);?"
nil t)
(setq param-name (intern (match-string 1)))
(setq param-value (match-string 2))
(setq result
(append result (list (list param-name param-value)))))))
result))
(defun icalendar--convert-tz-offset (alist dst-p)
"Return a cons of two strings representing a timezone start.
ALIST is an alist entry from a VTIMEZONE, like STANDARD.
DST-P is non-nil if this is for daylight savings time.
The strings are suitable for assembling into a TZ variable."
(let* ((offsetto (car (cddr (assq 'TZOFFSETTO alist))))
(offsetfrom (car (cddr (assq 'TZOFFSETFROM alist))))
(rrule-value (car (cddr (assq 'RRULE alist))))
(rdate-p (and (assq 'RDATE alist) t))
(dtstart (car (cddr (assq 'DTSTART alist))))
(no-dst (or rdate-p (equal offsetto offsetfrom))))
;; FIXME: the presence of an RDATE is assumed to denote the first day of the year
(when (and offsetto dtstart (or rrule-value no-dst))
(let* ((rrule (icalendar--split-value rrule-value))
(freq (cadr (assq 'FREQ rrule)))
(bymonth (cadr (assq 'BYMONTH rrule)))
(byday (cadr (assq 'BYDAY rrule))))
;; FIXME: we don't correctly handle WKST here.
(if (or no-dst (and (string= freq "YEARLY") bymonth))
(cons
(concat
;; Fake a name.
(if dst-p "DST" "STD")
;; For TZ, OFFSET is added to the local time. So,
;; invert the values.
(if (eq (aref offsetto 0) ?-) "+" "-")
(substring offsetto 1 3)
":"
(substring offsetto 3 5))
;; The start time.
(let* ((day (if no-dst
1
(icalendar--get-weekday-number (substring byday -2))))
(week (if no-dst
"1"
(if (eq day -1)
byday
(substring byday 0 -2)))))
;; "Translate" the iCalendar way to specify the last
;; (sun|mon|...)day in month to the tzset way.
(if (string= week "-1") ; last day as iCalendar calls it
(setq week "5")) ; last day as tzset calls it
(when no-dst (setq bymonth "1"))
(concat "M" bymonth "." week "." (if (eq day -1) "0"
(int-to-string day))
;; Start time.
"/"
(substring dtstart -6 -4)
":"
(substring dtstart -4 -2)
":"
(substring dtstart -2)))))))))
(defun icalendar--parse-vtimezone (alist)
"Turn a VTIMEZONE ALIST into a cons (ID . TZ-STRING).
Consider only the most recent date specification.
Return nil if timezone cannot be parsed."
(let* ((tz-id (icalendar--convert-string-for-import
(icalendar--get-event-property alist 'TZID)))
(daylight (cadr (cdar (icalendar--get-most-recent-observance alist 'DAYLIGHT))))
(day (and daylight (icalendar--convert-tz-offset daylight t)))
(standard (cadr (cdar (icalendar--get-most-recent-observance alist 'STANDARD))))
(std (and standard (icalendar--convert-tz-offset standard nil))))
(if (and tz-id std)
(cons tz-id
(if day
(concat (car std) (car day)
"," (cdr day) "," (cdr std))
(car std))))))
(defun icalendar--get-most-recent-observance (alist sub-comp)
"Return the latest observance for SUB-COMP DAYLIGHT or STANDARD.
ALIST is a VTIMEZONE potentially containing historical records."
;FIXME?: "most recent" should be relative to a given date
(let ((components (icalendar--get-children alist sub-comp)))
(list
(car
(sort components
(lambda (a b)
(let* ((get-recent (lambda (n)
(car
(sort
(delq nil
(mapcar (lambda (p)
(and (memq (car p) '(DTSTART RDATE))
(car (cddr p))))
n))
'string-greaterp))))
(a-recent (funcall get-recent (car (cddr a))))
(b-recent (funcall get-recent (car (cddr b)))))
(string-greaterp a-recent b-recent))))))))
(defun icalendar--convert-all-timezones (icalendar)
"Convert all timezones in the ICALENDAR into an alist.
Each element of the alist is a cons (ID . TZ-STRING),
like `icalendar--parse-vtimezone'."
(let (result)
(dolist (zone (icalendar--get-children (car icalendar) 'VTIMEZONE))
(setq zone (icalendar--parse-vtimezone zone))
(if zone
(setq result (cons zone result))))
result))
(defun icalendar--find-time-zone (prop-list zone-map)
"Return a timezone string for the time zone in PROP-LIST, or nil if none.
ZONE-MAP is a timezone alist as returned by `icalendar--convert-all-timezones'."
(let ((id (plist-get prop-list 'TZID)))
(if id
(cdr (assoc id zone-map)))))
(defun icalendar--decode-isodatetime (isodatetimestring &optional day-shift
source-zone
result-zone)
"Return ISODATETIMESTRING in format like `decode-time'.
Converts from ISO-8601 to Emacs representation. If
ISODATETIMESTRING specifies UTC time (trailing letter Z) the
decoded time is given in the local time zone! If optional
parameter DAY-SHIFT is non-nil the result is shifted by DAY-SHIFT
days.
SOURCE-ZONE, if provided, is the timezone for decoding the time,
in any format understood by `encode-time'.
RESULT-ZONE, if provided, is the timezone for encoding the result
in any format understood by `decode-time'.
FIXME: multiple comma-separated values should be allowed!"
(icalendar--dmsg isodatetimestring)
(if isodatetimestring
;; day/month/year must be present
(let ((year (read (substring isodatetimestring 0 4)))
(month (read (substring isodatetimestring 4 6)))
(day (read (substring isodatetimestring 6 8)))
(hour 0)
(minute 0)
(second 0))
(when (> (length isodatetimestring) 12)
;; hour/minute present
(setq hour (read (substring isodatetimestring 9 11)))
(setq minute (read (substring isodatetimestring 11 13))))
(when (> (length isodatetimestring) 14)
;; seconds present
(setq second (read (substring isodatetimestring 13 15))))
;; FIXME: Support subseconds.
(when (> (length isodatetimestring) 15)
(pcase (aref isodatetimestring 15)
(?Z
(setq source-zone t))
((or ?- ?+)
(setq source-zone
(concat "UTC" (substring isodatetimestring 15))))))
;; shift if necessary
(if day-shift
(let ((mdy (calendar-gregorian-from-absolute
(+ (calendar-absolute-from-gregorian
(list month day year))
day-shift))))
(setq month (nth 0 mdy))
(setq day (nth 1 mdy))
(setq year (nth 2 mdy))))
;; create the decoded date-time
;; FIXME!?!
(let ((decoded-time (list second minute hour day month year
nil -1 source-zone)))
(condition-case nil
(decode-time (encode-time decoded-time) result-zone)
(error
(message "Cannot decode \"%s\"" isodatetimestring)
;; Hope for the best....
decoded-time))))
;; isodatetimestring == nil
nil))
(defun icalendar--decode-isoduration (isodurationstring
&optional duration-correction)
"Convert ISODURATIONSTRING into format provided by `decode-time'.
Converts from ISO-8601 to Emacs representation. If ISODURATIONSTRING
specifies UTC time (trailing letter Z) the decoded time is given in
the local time zone!
Optional argument DURATION-CORRECTION shortens result by one day.
FIXME: TZID-attributes are ignored....!
FIXME: multiple comma-separated values should be allowed!"
(if isodurationstring
(save-match-data
(string-match
(concat
"^P[+-]?\\("
"\\(\\([0-9]+\\)D\\)" ; days only
"\\|"
"\\(\\(\\([0-9]+\\)D\\)?T\\(\\([0-9]+\\)H\\)?" ; opt days
"\\(\\([0-9]+\\)M\\)?\\(\\([0-9]+\\)S\\)?\\)" ; mand. time
"\\|"
"\\(\\([0-9]+\\)W\\)" ; weeks only
"\\)$") isodurationstring)
(let ((seconds 0)
(minutes 0)
(hours 0)
(days 0)
(months 0)
(years 0))
(cond
((match-beginning 2) ;days only
(setq days (read (substring isodurationstring
(match-beginning 3)
(match-end 3))))
(when duration-correction
(setq days (1- days))))
((match-beginning 4) ;days and time
(if (match-beginning 5)
(setq days (read (substring isodurationstring
(match-beginning 6)
(match-end 6)))))
(if (match-beginning 7)
(setq hours (read (substring isodurationstring
(match-beginning 8)
(match-end 8)))))
(if (match-beginning 9)
(setq minutes (read (substring isodurationstring
(match-beginning 10)
(match-end 10)))))
;; FIXME: Support subseconds.
(if (match-beginning 11)
(setq seconds (read (substring isodurationstring
(match-beginning 12)
(match-end 12))))))
((match-beginning 13) ;weeks only
(setq days (* 7 (read (substring isodurationstring
(match-beginning 14)
(match-end 14)))))))
(list seconds minutes hours days months years)))
;; isodatetimestring == nil
nil))
(defun icalendar--add-decoded-times (time1 time2)
"Add TIME1 to TIME2.
Both times must be given in decoded form. One of these times must be
valid (year > 1900 or something)."
;; FIXME: does this function exist already? Can we use decoded-time-add?
(decode-time (encode-time
;; FIXME: Support subseconds.
(time-convert (time-add (decoded-time-second time1)
(decoded-time-second time2))
'integer)
(+ (decoded-time-minute time1) (decoded-time-minute time2))
(+ (decoded-time-hour time1) (decoded-time-hour time2))
(+ (decoded-time-day time1) (decoded-time-day time2))
(+ (decoded-time-month time1) (decoded-time-month time2))
(+ (decoded-time-year time1) (decoded-time-year time2))
nil
nil
;;(or (nth 6 time1) (nth 6 time2)) ;; FIXME?
)))
(defun icalendar--datetime-to-american-date (datetime &optional separator)
"Convert the decoded DATETIME to American-style format.
Optional argument SEPARATOR gives the separator between month,
day, and year. If nil a blank character is used as separator.
American format: \"month day year\"."
(if datetime
(format "%d%s%d%s%d" (nth 4 datetime) ;month
(or separator " ")
(nth 3 datetime) ;day
(or separator " ")
(nth 5 datetime)) ;year
;; datetime == nil
nil))
(defun icalendar--datetime-to-european-date (datetime &optional separator)
"Convert the decoded DATETIME to European format.
Optional argument SEPARATOR gives the separator between month,
day, and year. If nil a blank character is used as separator.
European format: (day month year).
FIXME"
(if datetime
(format "%d%s%d%s%d" (nth 3 datetime) ;day
(or separator " ")
(nth 4 datetime) ;month
(or separator " ")
(nth 5 datetime)) ;year
;; datetime == nil
nil))
(defun icalendar--datetime-to-iso-date (datetime &optional separator)
"Convert the decoded DATETIME to ISO format.
Optional argument SEPARATOR gives the separator between month,
day, and year. If nil a blank character is used as separator.
ISO format: (year month day)."
(if datetime
(format "%d%s%d%s%d" (nth 5 datetime) ;year
(or separator " ")
(nth 4 datetime) ;month
(or separator " ")
(nth 3 datetime)) ;day
;; datetime == nil
nil))
(defun icalendar--datetime-to-diary-date (datetime &optional separator)
"Convert the decoded DATETIME to diary format.
Optional argument SEPARATOR gives the separator between month,
day, and year. If nil a blank character is used as separator.
Call icalendar--datetime-to-*-date according to the current
calendar date style."
(funcall (intern-soft (format "icalendar--datetime-to-%s-date"
calendar-date-style))
datetime separator))
(defun icalendar--datetime-to-colontime (datetime)
"Extract the time part of a decoded DATETIME into 24-hour format.
Note that this silently ignores seconds."
(format "%02d:%02d" (nth 2 datetime) (nth 1 datetime)))
(defun icalendar--get-month-number (monthname)
"Return the month number for the given MONTHNAME."
(catch 'found
(let ((num 1)
(m (downcase monthname)))
(mapc (lambda (month)
(let ((mm (downcase month)))
(if (or (string-equal mm m)
(string-equal (substring mm 0 3) m))
(throw 'found num))
(setq num (1+ num))))
calendar-month-name-array))
;; Error:
-1))
(defun icalendar--get-weekday-number (abbrevweekday)
"Return the number for the ABBREVWEEKDAY."
(if abbrevweekday
(catch 'found
(let ((num 0)
(aw (downcase abbrevweekday)))
(mapc (lambda (day)
(let ((d (downcase day)))
(if (string-equal d aw)
(throw 'found num))
(setq num (1+ num))))
icalendar--weekday-array)))
;; Error:
-1))
(defun icalendar--get-weekday-numbers (abbrevweekdays)
"Return the list of numbers for the comma-separated ABBREVWEEKDAYS."
(when abbrevweekdays
(let* ((num -1)
(weekday-alist (mapcar (lambda (day)
(progn
(setq num (1+ num))
(cons (downcase day) num)))
icalendar--weekday-array)))
(delq nil
(mapcar (lambda (abbrevday)
(cdr (assoc abbrevday weekday-alist)))
(split-string (downcase abbrevweekdays) ","))))))
(defun icalendar--get-weekday-abbrev (weekday)
"Return the abbreviated WEEKDAY."
(catch 'found
(let ((num 0)
(w (downcase weekday)))
(mapc (lambda (day)
(let ((d (downcase day)))
(if (or (string-equal d w)
(string-equal (substring d 0 3) w))
(throw 'found (aref icalendar--weekday-array num)))
(setq num (1+ num))))
calendar-day-name-array))
;; Error:
nil))
(defun icalendar--date-to-isodate (date &optional day-shift)
"Convert DATE to iso-style date.
DATE must be a list of the form (month day year).
If DAY-SHIFT is non-nil, the result is shifted by DAY-SHIFT days."
(let ((mdy (calendar-gregorian-from-absolute
(+ (calendar-absolute-from-gregorian date)
(or day-shift 0)))))
(format "%04d%02d%02d" (nth 2 mdy) (nth 0 mdy) (nth 1 mdy))))
(defun icalendar--datestring-to-isodate (datestring &optional day-shift year-shift)
"Convert diary-style DATESTRING to iso-style date.
If DAY-SHIFT is non-nil, the result is shifted by DAY-SHIFT days
-- DAY-SHIFT must be either nil or an integer. If YEAR-SHIFT is
non-nil, the result is shifted by YEAR-SHIFT years -- YEAR-SHIFT
must be either nil or an integer. This function tries to figure
the date style from DATESTRING itself. If that is not possible
it uses the current calendar date style."
(let ((day -1) month year)
(save-match-data
(cond ( ;; iso-style numeric date
(string-match (concat "\\s-*"
"\\([0-9]\\{4\\}\\)[ \t/-]\\s-*"
"0?\\([1-9][0-9]?\\)[ \t/-]\\s-*"
"0?\\([1-9][0-9]?\\)")
datestring)
(setq year (read (substring datestring (match-beginning 1)
(match-end 1))))
(setq month (read (substring datestring (match-beginning 2)
(match-end 2))))
(setq day (read (substring datestring (match-beginning 3)
(match-end 3)))))
( ;; non-iso numeric date -- must rely on configured
;; calendar style
(string-match (concat "\\s-*"
"0?\\([1-9][0-9]?\\)[ \t/]\\s-*"
"0?\\([1-9][0-9]?\\),?[ \t/]\\s-*"
"\\([0-9]\\{4\\}\\)")
datestring)
(setq day (read (substring datestring (match-beginning 1)
(match-end 1))))
(setq month (read (substring datestring (match-beginning 2)
(match-end 2))))
(setq year (read (substring datestring (match-beginning 3)
(match-end 3))))
(if (eq calendar-date-style 'american)
(let ((x month))
(setq month day)
(setq day x))))
( ;; date contains month names -- iso style
(string-match (concat "\\s-*"
"\\([0-9]\\{4\\}\\)[ \t/]\\s-*"
"\\([A-Za-z][^ ]+\\)[ \t/]\\s-*"
"0?\\([123]?[0-9]\\)")
datestring)
(setq year (read (substring datestring (match-beginning 1)
(match-end 1))))
(setq month (icalendar--get-month-number
(substring datestring (match-beginning 2)
(match-end 2))))
(setq day (read (substring datestring (match-beginning 3)
(match-end 3)))))
( ;; date contains month names -- european style
(string-match (concat "\\s-*"
"0?\\([123]?[0-9]\\)[ \t/]\\s-*"
"\\([A-Za-z][^ ]+\\)[ \t/]\\s-*"
"\\([0-9]\\{4\\}\\)")
datestring)
(setq day (read (substring datestring (match-beginning 1)
(match-end 1))))
(setq month (icalendar--get-month-number
(substring datestring (match-beginning 2)
(match-end 2))))
(setq year (read (substring datestring (match-beginning 3)
(match-end 3)))))
( ;; date contains month names -- american style
(string-match (concat "\\s-*"
"\\([A-Za-z][^ ]+\\)[ \t/]\\s-*"
"0?\\([123]?[0-9]\\),?[ \t/]\\s-*"
"\\([0-9]\\{4\\}\\)")
datestring)
(setq day (read (substring datestring (match-beginning 2)
(match-end 2))))
(setq month (icalendar--get-month-number
(substring datestring (match-beginning 1)
(match-end 1))))
(setq year (read (substring datestring (match-beginning 3)
(match-end 3)))))
(t
nil)))
(when year-shift
(setq year (+ year year-shift)))
(if (> day 0)
(let ((mdy (calendar-gregorian-from-absolute
(+ (calendar-absolute-from-gregorian (list month day
year))
(or day-shift 0)))))
(icalendar--dmsg (format "%04d%02d%02d" (nth 2 mdy) (nth 0 mdy) (nth 1 mdy)))
(format "%04d%02d%02d" (nth 2 mdy) (nth 0 mdy) (nth 1 mdy)))
nil)))
(defun icalendar--diarytime-to-isotime (timestring ampmstring)
"Convert a time like 9:30pm to an iso-conform string like T213000.
In this example the TIMESTRING would be \"9:30\" and the
AMPMSTRING would be \"pm\". The minutes may be missing as long
as the colon is missing as well, i.e. \"9\" is allowed as
TIMESTRING and has the same result as \"9:00\"."
(if timestring
(let* ((parts (save-match-data (split-string timestring ":")))
(h (car parts))
(m (if (cdr parts) (cadr parts)
(if (> (length h) 2) "" "00")))
(starttimenum (read (concat h m))))
;; take care of am/pm style
;; Be sure *not* to convert 12:00pm - 12:59pm to 2400-2459
(if (and ampmstring (string= "pm" ampmstring) (< starttimenum 1200))
(setq starttimenum (+ starttimenum 1200)))
;; Similar effect with 12:00am - 12:59am (need to convert to 0000-0059)
(if (and ampmstring (string= "am" ampmstring) (>= starttimenum 1200))
(setq starttimenum (- starttimenum 1200)))
(format "T%04d00" starttimenum))
nil))
(defun icalendar--convert-string-for-export (string)
"Escape comma and other critical characters in STRING."
(string-replace "," "\\," string))
(defun icalendar--convert-string-for-import (string)
"Remove escape chars for comma, semicolon etc. from STRING."
(string-replace
"\\n" "\n " (string-replace
"\\\"" "\"" (string-replace
"\\;" ";" (string-replace
"\\," "," string)))))
;; ======================================================================
;; Export -- convert emacs-diary to iCalendar
;; ======================================================================
;;;###autoload
(defun icalendar-export-file (diary-filename ical-filename)
"Export diary file to iCalendar format.
All diary entries in the file DIARY-FILENAME are converted to iCalendar
format. The result is appended to the file ICAL-FILENAME."
(interactive "FExport diary data from file: \n\
Finto iCalendar file: ")
(save-current-buffer
(set-buffer (find-file diary-filename))
(icalendar-export-region (point-min) (point-max) ical-filename)))
(defvar icalendar--uid-count 0
"Auxiliary counter for creating unique ids.")
(defun icalendar--create-uid (entry-full contents)
"Construct a unique iCalendar UID for a diary entry.
ENTRY-FULL is the full diary entry string. CONTENTS is the
current iCalendar object, as a string. Increase
`icalendar--uid-count'. Returns the UID string."
(let ((uid icalendar-uid-format))
(if
;; Allow other apps (such as org-mode) to create its own uid
(get-text-property 0 'uid entry-full)
(setq uid (get-text-property 0 'uid entry-full))
(setq uid (replace-regexp-in-string
"%c"
(format "%d" icalendar--uid-count)
uid t t))
(setq icalendar--uid-count (1+ icalendar--uid-count))
(setq uid (replace-regexp-in-string
"%t"
(format-time-string "%s%N")
uid t t))
(setq uid (replace-regexp-in-string
"%h"
(format "%d" (abs (sxhash entry-full))) uid t t))
(setq uid (replace-regexp-in-string
"%u" (or user-login-name "UNKNOWN_USER") uid t t))
(let ((dtstart (if (string-match "^DTSTART[^:]*:\\([0-9]*\\)" contents)
(substring contents (match-beginning 1) (match-end 1))
"DTSTART")))
(setq uid (replace-regexp-in-string "%s" dtstart uid t t))))
;; Return the UID string
uid))
;;;###autoload
(defun icalendar-export-region (min max ical-filename)
"Export region in diary file to iCalendar format.
All diary entries in the region from MIN to MAX in the current buffer are
converted to iCalendar format. The result is appended to the file
ICAL-FILENAME.
This function attempts to return t if something goes wrong. In this
case an error string which describes all the errors and problems is
written into the buffer `*icalendar-errors*'."
(interactive "r
FExport diary data into iCalendar file: ")
(let ((result "")
(entry-main "")
(entry-rest "")
(entry-full "")
(header "")
(contents)
(alarm)
(found-error nil)
(nonmarker (concat "^" (regexp-quote diary-nonmarking-symbol)
"?"))
(other-elements nil)
(cns-cons-or-list nil))
;; prepare buffer with error messages
(save-current-buffer
(set-buffer (get-buffer-create "*icalendar-errors*"))
(erase-buffer))
;; here we go
(save-excursion
(goto-char min)
(while (re-search-forward
;; possibly ignore hidden entries beginning with "&"
(if icalendar-export-hidden-diary-entries
"^\\([^ \t\n#].+\\)\\(\\(\n[ \t].*\\)*\\)"
"^\\([^ \t\n&#].+\\)\\(\\(\n[ \t].*\\)*\\)")
max t)
(setq entry-main (match-string 1))
(if (match-beginning 2)
(setq entry-rest (match-string 2))
(setq entry-rest ""))
(setq entry-full (concat entry-main entry-rest))
(condition-case error-val
(progn
(setq cns-cons-or-list
(icalendar--convert-to-ical nonmarker entry-main))
(setq other-elements (icalendar--parse-summary-and-rest
entry-full))
(mapc (lambda (contents-n-summary)
(setq contents (concat (car contents-n-summary)
"\nSUMMARY:"
(cdr contents-n-summary)))
(let ((cla (cdr (assoc 'cla other-elements)))
(des (cdr (assoc 'des other-elements)))
(loc (cdr (assoc 'loc other-elements)))
(org (cdr (assoc 'org other-elements)))
(sta (cdr (assoc 'sta other-elements)))
;; (sum (cdr (assoc 'sum other-elements)))
(url (cdr (assoc 'url other-elements)))
(uid (cdr (assoc 'uid other-elements))))
(if cla
(setq contents (concat contents "\nCLASS:" cla)))
(if des
(setq contents (concat contents "\nDESCRIPTION:"
des)))
(if loc
(setq contents (concat contents "\nLOCATION:" loc)))
(if org
(setq contents (concat contents "\nORGANIZER:"
org)))
(if sta
(setq contents (concat contents "\nSTATUS:" sta)))
;;(if sum
;; (setq contents (concat contents "\nSUMMARY:" sum)))
(if url
(setq contents (concat contents "\nURL:" url)))
(setq header (concat "\nBEGIN:VEVENT\nUID:"
(or uid
(icalendar--create-uid
entry-full contents))))
(setq alarm (icalendar--create-ical-alarm
(cdr contents-n-summary))))
(setq result (concat result header contents alarm
"\nEND:VEVENT")))
(if (and (consp cns-cons-or-list)
(not (listp (cdr cns-cons-or-list))))
(list cns-cons-or-list)
cns-cons-or-list)))
;; handle errors
(error
(setq found-error t)
(save-current-buffer
(set-buffer (get-buffer-create "*icalendar-errors*"))
(insert (format-message "Error in line %d -- %s: `%s'\n"
(count-lines (point-min) (point))
error-val
entry-main))))))
;; we're done, insert everything into the file
(save-current-buffer
(let ((coding-system-for-write 'utf-8))
(set-buffer (find-file ical-filename))
(goto-char (point-max))
(insert "BEGIN:VCALENDAR")
(insert "\nPRODID:-//Emacs//NONSGML icalendar.el//EN")
(insert "\nVERSION:2.0")
(insert result)
(insert "\nEND:VCALENDAR\n")
;; save the diary file
(save-buffer)
(unless found-error
(bury-buffer)))))
found-error))
(defun icalendar--convert-to-ical (nonmarker entry-main)
"Convert a diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(or
(unless icalendar-export-sexp-enumerate-all
(or
;; anniversaries -- %%(diary-anniversary ...)
(icalendar--convert-anniversary-to-ical nonmarker entry-main)
;; cyclic events -- %%(diary-cyclic ...)
(icalendar--convert-cyclic-to-ical nonmarker entry-main)
;; diary-date -- %%(diary-date ...)
(icalendar--convert-date-to-ical nonmarker entry-main)
;; float events -- %%(diary-float ...)
(icalendar--convert-float-to-ical nonmarker entry-main)
;; block events -- %%(diary-block ...)
(icalendar--convert-block-to-ical nonmarker entry-main)))
;; other sexp diary entries
(icalendar--convert-sexp-to-ical nonmarker entry-main)
;; weekly by day -- Monday 8:30 Team meeting
(icalendar--convert-weekly-to-ical nonmarker entry-main)
;; yearly by day -- 1 May Tag der Arbeit
(icalendar--convert-yearly-to-ical nonmarker entry-main)
;; "ordinary" events, start and end time given
;; 1 Feb 2003 blah
(icalendar--convert-ordinary-to-ical nonmarker entry-main)
;; everything else
;; Oops! what's that?
(error "Could not parse entry")))
(defun icalendar--parse-summary-and-rest (summary-and-rest)
"Parse SUMMARY-AND-REST from a diary to fill iCalendar properties.
Returns an alist."
(save-match-data
(if (functionp icalendar-import-format)
;; can't do anything
nil
;; split summary-and-rest
(let* ((case-fold-search nil)
(s icalendar-import-format)
(p-cla (or (string-match "%c" icalendar-import-format) -1))
(p-des (or (string-match "%d" icalendar-import-format) -1))
(p-loc (or (string-match "%l" icalendar-import-format) -1))
(p-org (or (string-match "%o" icalendar-import-format) -1))
(p-sum (or (string-match "%s" icalendar-import-format) -1))
(p-sta (or (string-match "%t" icalendar-import-format) -1))
(p-url (or (string-match "%u" icalendar-import-format) -1))
(p-uid (or (string-match "%U" icalendar-import-format) -1))
(p-list (sort (list p-cla p-des p-loc p-org p-sta p-sum p-url p-uid) '<))
(ct 0)
pos-cla pos-des pos-loc pos-org pos-sta pos-url pos-uid) ;pos-sum
(dotimes (i (length p-list))
;; Use 'ct' to keep track of current position in list
(cond ((and (>= p-cla 0) (= (nth i p-list) p-cla))
(setq ct (+ ct 1))
(setq pos-cla (* 2 ct)))
((and (>= p-des 0) (= (nth i p-list) p-des))
(setq ct (+ ct 1))
(setq pos-des (* 2 ct)))
((and (>= p-loc 0) (= (nth i p-list) p-loc))
(setq ct (+ ct 1))
(setq pos-loc (* 2 ct)))
((and (>= p-org 0) (= (nth i p-list) p-org))
(setq ct (+ ct 1))
(setq pos-org (* 2 ct)))
((and (>= p-sta 0) (= (nth i p-list) p-sta))
(setq ct (+ ct 1))
(setq pos-sta (* 2 ct)))
((and (>= p-sum 0) (= (nth i p-list) p-sum))
(setq ct (+ ct 1))
;; (setq pos-sum (* 2 ct))
)
((and (>= p-url 0) (= (nth i p-list) p-url))
(setq ct (+ ct 1))
(setq pos-url (* 2 ct)))
((and (>= p-uid 0) (= (nth i p-list) p-uid))
(setq ct (+ ct 1))
(setq pos-uid (* 2 ct)))) )
(mapc (lambda (ij)
(setq s (replace-regexp-in-string (car ij) (cadr ij) s t t)))
(list
;; summary must be first! because of %s
(list "%s"
(concat "\\(" icalendar-import-format-summary "\\)??"))
(list "%c"
(concat "\\(" icalendar-import-format-class "\\)??"))
(list "%d"
(concat "\\(" icalendar-import-format-description "\\)??"))
(list "%l"
(concat "\\(" icalendar-import-format-location "\\)??"))
(list "%o"
(concat "\\(" icalendar-import-format-organizer "\\)??"))
(list "%t"
(concat "\\(" icalendar-import-format-status "\\)??"))
(list "%u"
(concat "\\(" icalendar-import-format-url "\\)??"))
(list "%U"
(concat "\\(" icalendar-import-format-uid "\\)??"))))
;; Need the \' regexp in order to detect multi-line items
(setq s (concat "\\`"
(replace-regexp-in-string "%s" "\\([^z-a]*?\\)" s nil t)
"\\'"))
(if (string-match s summary-and-rest)
(let (cla des loc org sta url uid) ;; sum
;; (if (and pos-sum (match-beginning pos-sum))
;; (setq sum (substring summary-and-rest
;; (match-beginning pos-sum)
;; (match-end pos-sum))))
(if (and pos-cla (match-beginning pos-cla))
(setq cla (substring summary-and-rest
(match-beginning pos-cla)
(match-end pos-cla))))
(if (and pos-des (match-beginning pos-des))
(setq des (substring summary-and-rest
(match-beginning pos-des)
(match-end pos-des))))
(if (and pos-loc (match-beginning pos-loc))
(setq loc (substring summary-and-rest
(match-beginning pos-loc)
(match-end pos-loc))))
(if (and pos-org (match-beginning pos-org))
(setq org (substring summary-and-rest
(match-beginning pos-org)
(match-end pos-org))))
(if (and pos-sta (match-beginning pos-sta))
(setq sta (substring summary-and-rest
(match-beginning pos-sta)
(match-end pos-sta))))
(if (and pos-url (match-beginning pos-url))
(setq url (substring summary-and-rest
(match-beginning pos-url)
(match-end pos-url))))
(if (and pos-uid (match-beginning pos-uid))
(setq uid (substring summary-and-rest
(match-beginning pos-uid)
(match-end pos-uid))))
(list (if cla (cons 'cla cla) nil)
(if des (cons 'des des) nil)
(if loc (cons 'loc loc) nil)
(if org (cons 'org org) nil)
(if sta (cons 'sta sta) nil)
;;(if sum (cons 'sum sum) nil)
(if url (cons 'url url) nil)
(if uid (cons 'uid uid) nil))))))))
(defun icalendar--create-ical-alarm (summary)
"Return VALARM blocks for the given SUMMARY."
(when icalendar-export-alarms
(let* ((advance-time (car icalendar-export-alarms))
(alarm-specs (cadr icalendar-export-alarms))
(fun (lambda (spec)
(icalendar--do-create-ical-alarm advance-time spec summary))))
(mapconcat fun alarm-specs ""))))
(defun icalendar--do-create-ical-alarm (advance-time alarm-spec summary)
"Return a VALARM block.
Argument ADVANCE-TIME is a number giving the time when the alarm
fires (minutes before the respective event). Argument ALARM-SPEC
is a list which must be one of (audio), (display) or
\(email (ADDRESS1 ...)), see `icalendar-export-alarms'. Argument
SUMMARY is a string which contains a short description for the
alarm."
(let* ((action (car alarm-spec))
(act (format "\nACTION:%s"
(cdr (assoc action '((audio . "AUDIO")
(display . "DISPLAY")
(email . "EMAIL"))))))
(tri (format "\nTRIGGER:-PT%dM" advance-time))
(des (if (memq action '(display email))
(format "\nDESCRIPTION:%s" summary)
""))
(sum (if (eq action 'email)
(format "\nSUMMARY:%s" summary)
""))
(att (if (eq action 'email)
(mapconcat (lambda (i)
(format "\nATTENDEE:MAILTO:%s" i))
(cadr alarm-spec) "")
"")))
(concat "\nBEGIN:VALARM" act tri des sum att "\nEND:VALARM")))
;; subroutines for icalendar-export-region
(defun icalendar--convert-ordinary-to-ical (nonmarker entry-main)
"Convert \"ordinary\" diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match
(concat nonmarker
"\\([^ /]+[ /]+[^ /]+[ /]+[^ ]+\\)\\s-*" ; date
"\\(\\([0-9][0-9]?\\(:[0-9][0-9]\\)?\\)\\([ap]m\\)?" ; start time
"\\("
"-\\([0-9][0-9]?\\(:[0-9][0-9]\\)?\\)\\([ap]m\\)?\\)?" ; end time
"\\)?"
"\\s-*\\(.*?\\) ?$")
entry-main)
(let* ((datetime (substring entry-main (match-beginning 1)
(match-end 1)))
(startisostring (icalendar--datestring-to-isodate
datetime))
(endisostring (icalendar--datestring-to-isodate
datetime 1))
(endisostring1)
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 3)
(substring entry-main
(match-beginning 3)
(match-end 3))
nil)
(if (match-beginning 5)
(substring entry-main
(match-beginning 5)
(match-end 5))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)
(if (match-beginning 9)
(substring entry-main
(match-beginning 9)
(match-end 9))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 10)
(match-end 10)))))
(icalendar--dmsg "ordinary %s" entry-main)
(unless startisostring
(error "Could not parse date"))
;; If only start-date is specified, then end-date is next day,
;; otherwise it is same day.
(setq endisostring1 (if starttimestring
startisostring
endisostring))
(when starttimestring
(unless endtimestring
(let ((time
(read (replace-regexp-in-string "^T0?" ""
starttimestring))))
(if (< time 230000)
;; Case: ends on same day
(setq endtimestring (format "T%06d"
(+ 10000 time)))
;; Case: ends on next day
(setq endtimestring (format "T%06d"
(- time 230000)))
(setq endisostring1 endisostring)) )))
(cons (concat "\nDTSTART;"
(if starttimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
startisostring
(or starttimestring "")
"\nDTEND;"
(if endtimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
endisostring1
(or endtimestring ""))
summary))
;; no match
nil))
(defun icalendar-first-weekday-of-year (abbrevweekday year)
"Find the first ABBREVWEEKDAY in a given YEAR.
Returns day number."
(let* ((day-of-week-jan01 (calendar-day-of-week (list 1 1 year)))
(result (+ 1
(- (icalendar--get-weekday-number abbrevweekday)
day-of-week-jan01))))
(cond ((<= result 0)
(setq result (+ result 7)))
((> result 7)
(setq result (- result 7))))
result))
(defun icalendar--convert-weekly-to-ical (nonmarker entry-main)
"Convert weekly diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (and (string-match (concat nonmarker
"\\([a-z]+\\)\\s-+"
"\\(\\([0-9][0-9]?:[0-9][0-9]\\)"
"\\([ap]m\\)?"
"\\(-"
"\\([0-9][0-9]?:[0-9][0-9]\\)"
"\\([ap]m\\)?\\)?"
"\\)?"
"\\s-*\\(.*?\\) ?$")
entry-main)
(icalendar--get-weekday-abbrev
(substring entry-main (match-beginning 1)
(match-end 1))))
(let* ((day (icalendar--get-weekday-abbrev
(substring entry-main (match-beginning 1)
(match-end 1))))
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 3)
(substring entry-main
(match-beginning 3)
(match-end 3))
nil)
(if (match-beginning 4)
(substring entry-main
(match-beginning 4)
(match-end 4))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 6)
(substring entry-main
(match-beginning 6)
(match-end 6))
nil)
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 8)
(match-end 8)))))
(icalendar--dmsg "weekly %s" entry-main)
(when starttimestring
(unless endtimestring
(let ((time (read
(replace-regexp-in-string "^T0?" ""
starttimestring))))
(setq endtimestring (format "T%06d"
(+ 10000 time))))))
(cons (concat "\nDTSTART;"
(if starttimestring
"VALUE=DATE-TIME:"
"VALUE=DATE:")
;; Find the first requested weekday of the
;; start year
(funcall 'format "%04d%02d%02d"
icalendar-recurring-start-year 1
(icalendar-first-weekday-of-year
day icalendar-recurring-start-year))
(or starttimestring "")
"\nDTEND;"
(if endtimestring
"VALUE=DATE-TIME:"
"VALUE=DATE:")
(funcall 'format "%04d%02d%02d"
;; end is non-inclusive!
icalendar-recurring-start-year 1
(+ (icalendar-first-weekday-of-year
day icalendar-recurring-start-year)
(if endtimestring 0 1)))
(or endtimestring "")
"\nRRULE:FREQ=WEEKLY;INTERVAL=1;BYDAY="
day)
summary))
;; no match
nil))
(defun icalendar--convert-yearly-to-ical (nonmarker entry-main)
"Convert yearly diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker
(if (eq calendar-date-style 'european)
"\\([0-9]+[0-9]?\\)\\s-+\\([a-z]+\\)\\s-+"
"\\([a-z]+\\)\\s-+\\([0-9]+[0-9]?\\)\\s-+")
"\\*?\\s-*"
"\\(\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?"
"\\("
"-\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?\\)?"
"\\)?"
"\\s-*\\([^0-9]+.*?\\) ?$" ; must not match years
)
entry-main)
(let* ((daypos (if (eq calendar-date-style 'european) 1 2))
(monpos (if (eq calendar-date-style 'european) 2 1))
(day (read (substring entry-main
(match-beginning daypos)
(match-end daypos))))
(month (icalendar--get-month-number
(substring entry-main
(match-beginning monpos)
(match-end monpos))))
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 4)
(substring entry-main
(match-beginning 4)
(match-end 4))
nil)
(if (match-beginning 5)
(substring entry-main
(match-beginning 5)
(match-end 5))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)
(if (match-beginning 8)
(substring entry-main
(match-beginning 8)
(match-end 8))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 9)
(match-end 9)))))
(icalendar--dmsg "yearly %s" entry-main)
(when starttimestring
(unless endtimestring
(let ((time (read
(replace-regexp-in-string "^T0?" ""
starttimestring))))
(setq endtimestring (format "T%06d"
(+ 10000 time))))))
(cons (concat "\nDTSTART;"
(if starttimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
(format "1900%02d%02d" month day)
(or starttimestring "")
"\nDTEND;"
(if endtimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
;; end is not included! shift by one day
(icalendar--date-to-isodate
(list month day 1900)
(if endtimestring 0 1))
(or endtimestring "")
"\nRRULE:FREQ=YEARLY;INTERVAL=1;BYMONTH="
(format "%d" month)
";BYMONTHDAY="
(format "%d" day))
summary))
;; no match
nil))
(defun icalendar--convert-sexp-to-ical (nonmarker entry-main &optional start)
"Convert sexp diary entry to iCalendar format.
Enumerate the evaluated sexp entry for the next
`icalendar-export-sexp-enumeration-days' days. NONMARKER is a
regular expression matching the start of non-marking entries.
ENTRY-MAIN is the first line of the diary entry.
Optional argument START determines the first day of the
enumeration, given as a Lisp time value -- used for test purposes."
(cond ((string-match (concat nonmarker
"%%(and \\(([^)]+)\\))\\(\\s-*.*?\\) ?$")
entry-main)
;; simple sexp entry as generated by icalendar.el: strip off the
;; unnecessary (and)
(icalendar--dmsg "diary-sexp from icalendar.el %s" entry-main)
(icalendar--convert-to-ical
nonmarker
(concat "%%"
(substring entry-main (match-beginning 1) (match-end 1))
(substring entry-main (match-beginning 2) (match-end 2)))))
((string-match (concat nonmarker
"%%\\(([^)]+)\\)\\s-*\\(.*\\)")
entry-main)
;; regular sexp entry
(icalendar--dmsg "diary-sexp %s" entry-main)
(let* ((entry-main (substring entry-main 2))
(res (read-from-string entry-main))
(p1 (prin1-to-string (car res)))
(p2 (substring entry-main (cdr res)))
(now (or start (current-time))))
(delete nil
(mapcar
(lambda (offset)
(let* ((day (decode-time (time-add now
(* 60 60 24 offset))))
(d (decoded-time-day day))
(m (decoded-time-month day))
(y (decoded-time-year day))
(se (diary-sexp-entry p1 p2 (list m d y)))
(see (cond ((stringp se) se)
((consp se) (cdr se))
(t nil))))
(cond ((null see)
nil)
((stringp see)
(let ((calendar-date-style 'iso))
(icalendar--convert-ordinary-to-ical
nonmarker (format "%4d/%02d/%02d %s" y m d see))))
(;TODO:
(error "Unsupported Sexp-entry: %s"
entry-main)))))
(number-sequence
0 (- icalendar-export-sexp-enumeration-days 1))))))
(t
;; no match
nil)))
(defun icalendar--convert-block-to-ical (nonmarker entry-main)
"Convert block diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker
"%%(diary-block \\([^ /]+[ /]+[^ /]+[ /]+[^ ]+\\)"
" +\\([^ /]+[ /]+[^ /]+[ /]+[^ ]+\\))\\s-*"
"\\(\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?"
"\\("
"-\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?\\)?"
"\\)?"
"\\s-*\\(.*?\\) ?$")
entry-main)
(let* ((startstring (substring entry-main
(match-beginning 1)
(match-end 1)))
(endstring (substring entry-main
(match-beginning 2)
(match-end 2)))
(startisostring (icalendar--datestring-to-isodate
startstring))
(endisostring (icalendar--datestring-to-isodate
endstring))
(endisostring+1 (icalendar--datestring-to-isodate
endstring 1))
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 4)
(substring entry-main
(match-beginning 4)
(match-end 4))
nil)
(if (match-beginning 5)
(substring entry-main
(match-beginning 5)
(match-end 5))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)
(if (match-beginning 8)
(substring entry-main
(match-beginning 8)
(match-end 8))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 9)
(match-end 9)))))
(icalendar--dmsg "diary-block %s" entry-main)
(when starttimestring
(unless endtimestring
(let ((time
(read (replace-regexp-in-string "^T0?" ""
starttimestring))))
(setq endtimestring (format "T%06d"
(+ 10000 time))))))
(if starttimestring
;; with time -> write rrule
(cons (concat "\nDTSTART;VALUE=DATE-TIME:"
startisostring
starttimestring
"\nDTEND;VALUE=DATE-TIME:"
startisostring
endtimestring
"\nRRULE:FREQ=DAILY;INTERVAL=1;UNTIL="
endisostring)
summary)
;; no time -> write long event
(cons (concat "\nDTSTART;VALUE=DATE:" startisostring
"\nDTEND;VALUE=DATE:" endisostring+1)
summary)))
;; no match
nil))
(defun icalendar--convert-float-to-ical (nonmarker entry-main)
"Convert float diary entry to iCalendar format -- partially unsupported!
FIXME! DAY from `diary-float' yet unimplemented.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker "%%\\((diary-float .+\\) ?$") entry-main)
(with-temp-buffer
(insert (match-string 1 entry-main))
(goto-char (point-min))
(let* ((sexp (read (current-buffer))) ;using `read' here
;easier than regexp
;matching, esp. with
;different forms of
;MONTH
(month (nth 1 sexp))
(dayname (nth 2 sexp))
(n (nth 3 sexp))
(day (nth 4 sexp))
(summary
(replace-regexp-in-string
"\\(^\s+\\|\s+$\\)" ""
(buffer-substring (point) (point-max)))))
(when day
(progn
(icalendar--dmsg "diary-float %s" entry-main)
(error "Don't know if or how to implement day in `diary-float'")))
(cons (concat
;;Start today (yes this is an arbitrary choice):
"\nDTSTART;VALUE=DATE:"
(format-time-string "%Y%m%d")
;;BUT remove today if `diary-float'
;;expression does not hold true for today:
(when
(null (calendar-dlet ((date (calendar-current-date))
(entry entry-main))
(diary-float month dayname n)))
(concat
"\nEXDATE;VALUE=DATE:"
(format-time-string "%Y%m%d")))
"\nRRULE:"
(if (or (numberp month) (listp month))
"FREQ=YEARLY;BYMONTH="
"FREQ=MONTHLY")
(when
(listp month)
(mapconcat
(lambda (m)
(number-to-string m))
(cadr month) ","))
(when
(numberp month)
(number-to-string month))
";BYDAY="
(number-to-string n)
(aref icalendar--weekday-array dayname))
summary)))
;; no match
nil))
(defun icalendar--convert-date-to-ical (nonmarker entry-main)
"Convert `diary-date' diary entry to iCalendar format -- unsupported!
FIXME!
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker
"%%(diary-date \\([^)]+\\))\\s-*\\(.*?\\) ?$")
entry-main)
(progn
(icalendar--dmsg "diary-date %s" entry-main)
(error "`diary-date' is not supported yet"))
;; no match
nil))
(defun icalendar--convert-cyclic-to-ical (nonmarker entry-main)
"Convert `diary-cyclic' diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker
"%%(diary-cyclic \\([^ ]+\\) +"
"\\([^ /]+[ /]+[^ /]+[ /]+[^ ]+\\))\\s-*"
"\\(\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?"
"\\("
"-\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?\\)?"
"\\)?"
"\\s-*\\(.*?\\) ?$")
entry-main)
(let* ((frequency (substring entry-main (match-beginning 1)
(match-end 1)))
(datetime (substring entry-main (match-beginning 2)
(match-end 2)))
(startisostring (icalendar--datestring-to-isodate
datetime))
(endisostring (icalendar--datestring-to-isodate
datetime))
(endisostring+1 (icalendar--datestring-to-isodate
datetime 1))
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 4)
(substring entry-main
(match-beginning 4)
(match-end 4))
nil)
(if (match-beginning 5)
(substring entry-main
(match-beginning 5)
(match-end 5))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)
(if (match-beginning 8)
(substring entry-main
(match-beginning 8)
(match-end 8))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 9)
(match-end 9)))))
(icalendar--dmsg "diary-cyclic %s" entry-main)
(when starttimestring
(unless endtimestring
(let ((time
(read (replace-regexp-in-string "^T0?" ""
starttimestring))))
(setq endtimestring (format "T%06d"
(+ 10000 time))))))
(cons (concat "\nDTSTART;"
(if starttimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
startisostring
(or starttimestring "")
"\nDTEND;"
(if endtimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
(if endtimestring endisostring endisostring+1)
(or endtimestring "")
"\nRRULE:FREQ=DAILY;INTERVAL=" frequency
;; strange: korganizer does not expect
;; BYSOMETHING here...
)
summary))
;; no match
nil))
(defun icalendar--convert-anniversary-to-ical (nonmarker entry-main)
"Convert `diary-anniversary' diary entry to iCalendar format.
NONMARKER is a regular expression matching the start of non-marking
entries. ENTRY-MAIN is the first line of the diary entry."
(if (string-match (concat nonmarker
"%%(diary-anniversary \\([^)]+\\))\\s-*"
"\\(\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?"
"\\("
"-\\([0-9][0-9]?:[0-9][0-9]\\)\\([ap]m\\)?\\)?"
"\\)?"
"\\s-*\\(.*?\\) ?$")
entry-main)
(let* ((datetime (substring entry-main (match-beginning 1)
(match-end 1)))
(startisostring (icalendar--datestring-to-isodate
datetime nil 1))
(endisostring (icalendar--datestring-to-isodate
datetime 1 1))
(starttimestring (icalendar--diarytime-to-isotime
(if (match-beginning 3)
(substring entry-main
(match-beginning 3)
(match-end 3))
nil)
(if (match-beginning 4)
(substring entry-main
(match-beginning 4)
(match-end 4))
nil)))
(endtimestring (icalendar--diarytime-to-isotime
(if (match-beginning 6)
(substring entry-main
(match-beginning 6)
(match-end 6))
nil)
(if (match-beginning 7)
(substring entry-main
(match-beginning 7)
(match-end 7))
nil)))
(summary (icalendar--convert-string-for-export
(substring entry-main (match-beginning 8)
(match-end 8)))))
(icalendar--dmsg "diary-anniversary %s" entry-main)
(when starttimestring
(unless endtimestring
(let ((time
(read (replace-regexp-in-string "^T0?" ""
starttimestring))))
(setq endtimestring (format "T%06d"
(+ 10000 time))))))
(cons (concat "\nDTSTART;"
(if starttimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
startisostring
(or starttimestring "")
"\nDTEND;"
(if endtimestring "VALUE=DATE-TIME:"
"VALUE=DATE:")
endisostring
(or endtimestring "")
"\nRRULE:FREQ=YEARLY;INTERVAL=1"
;; the following is redundant,
;; but korganizer seems to expect this... ;(
;; and evolution doesn't understand it... :(
;; so... who is wrong?!
";BYMONTH="
(substring startisostring 4 6)
";BYMONTHDAY="
(substring startisostring 6 8))
summary))
;; no match
nil))
;; ======================================================================
;; Import -- convert iCalendar to emacs-diary
;; ======================================================================
;;;###autoload
(defun icalendar-import-file (ical-filename diary-filename
&optional non-marking)
"Import an iCalendar file and append to a diary file.
Argument ICAL-FILENAME output iCalendar file.
Argument DIARY-FILENAME input `diary-file'.
Optional argument NON-MARKING determines whether events are created as
non-marking or not."
(interactive "fImport iCalendar data from file: \nFInto diary file: \nP")
;; clean up the diary file
(save-current-buffer
;; now load and convert from the ical file
(set-buffer (find-file ical-filename))
(icalendar-import-buffer diary-filename t non-marking)))
;;;###autoload
(defun icalendar-import-buffer (&optional diary-filename do-not-ask
non-marking)
"Extract iCalendar events from current buffer.
This function searches the current buffer for the first iCalendar
object, reads it and adds all VEVENT elements to the diary
DIARY-FILENAME.
It will ask for each appointment whether to add it to the diary
unless DO-NOT-ASK is non-nil. When called interactively,
DO-NOT-ASK is nil, so that you are asked for each event.
NON-MARKING determines whether diary events are created as
non-marking.
Return code t means that importing worked well, return code nil
means that an error has occurred. Error messages will be in the
buffer `*icalendar-errors*'."
(interactive)
(save-current-buffer
;; prepare ical
(message "Preparing iCalendar...")
(set-buffer (icalendar--get-unfolded-buffer (current-buffer)))
(goto-char (point-min))
(message "Preparing iCalendar...done")
(if (re-search-forward "^BEGIN:VCALENDAR\\s-*$" nil t)
(let (ical-contents ical-errors)
;; read ical
(message "Reading iCalendar...")
(beginning-of-line)
(setq ical-contents (icalendar--read-element nil nil))
(message "Reading iCalendar...done")
;; convert ical
(message "Converting iCalendar...")
(setq ical-errors (icalendar--convert-ical-to-diary
ical-contents
diary-filename do-not-ask non-marking))
(when diary-filename
;; save the diary file if it is visited already
(let ((b (find-buffer-visiting diary-filename)))
(when b
(save-current-buffer
(set-buffer b)
(save-buffer)))))
(message "Converting iCalendar...done")
;; return t if no error occurred
(not ical-errors))
(message
"Current buffer does not contain iCalendar contents!")
;; return nil, i.e. import did not work
nil)))
(defun icalendar--format-ical-event (event)
"Create a string representation of an iCalendar EVENT."
(if (functionp icalendar-import-format)
(funcall icalendar-import-format event)
(let ((string icalendar-import-format)
(case-fold-search nil)
(conversion-list
'(("%c" CLASS icalendar-import-format-class)
("%d" DESCRIPTION icalendar-import-format-description)
("%l" LOCATION icalendar-import-format-location)
("%o" ORGANIZER icalendar-import-format-organizer)
("%s" SUMMARY icalendar-import-format-summary)
("%t" STATUS icalendar-import-format-status)
("%u" URL icalendar-import-format-url)
("%U" UID icalendar-import-format-uid))))
;; convert the specifiers in the format string
(mapc (lambda (i)
(let* ((spec (car i))
(prop (cadr i))
(format (car (cddr i)))
(contents (icalendar--get-event-property event prop))
(formatted-contents ""))
(when (and contents (> (length contents) 0))
(setq formatted-contents
(replace-regexp-in-string "%s"
(icalendar--convert-string-for-import
contents)
(symbol-value format)
t t)))
(setq string (replace-regexp-in-string spec
formatted-contents
string
t t))))
conversion-list)
string)))
(defun icalendar--convert-ical-to-diary (ical-list diary-filename
&optional do-not-ask
non-marking)
"Convert iCalendar data to an Emacs diary file.
Import VEVENTS from the iCalendar object ICAL-LIST and saves them to a
DIARY-FILENAME. If DO-NOT-ASK is nil the user is asked for each event
whether to actually import it. NON-MARKING determines whether diary
events are created as non-marking.
This function attempts to return t if something goes wrong. In this
case an error string which describes all the errors and problems is
written into the buffer `*icalendar-errors*'."
(let* ((ev (icalendar--all-events ical-list))
(error-string "")
(event-ok t)
(found-error nil)
(zone-map (icalendar--convert-all-timezones ical-list))
e diary-string)
;; step through all events/appointments
(while ev
(setq e (car ev))
(setq ev (cdr ev))
(setq event-ok nil)
(condition-case error-val
(let* ((dtstart (icalendar--get-event-property e 'DTSTART))
(dtstart-zone (icalendar--find-time-zone
(icalendar--get-event-property-attributes
e 'DTSTART)
zone-map))
(dtstart-dec (icalendar--decode-isodatetime dtstart nil
dtstart-zone))
(start-d (icalendar--datetime-to-diary-date
dtstart-dec))
(start-t (and dtstart
(> (length dtstart) 8)
(icalendar--datetime-to-colontime dtstart-dec)))
(dtend (icalendar--get-event-property e 'DTEND))
(dtend-zone (icalendar--find-time-zone
(icalendar--get-event-property-attributes
e 'DTEND)
zone-map))
(dtend-dec (icalendar--decode-isodatetime dtend
nil dtend-zone))
(dtend-1-dec (icalendar--decode-isodatetime dtend -1
dtend-zone))
end-d
end-1-d
end-t
(summary (icalendar--convert-string-for-import
(or (icalendar--get-event-property e 'SUMMARY)
"No summary")))
(rrule (icalendar--get-event-property e 'RRULE))
(rdate (icalendar--get-event-property e 'RDATE))
(duration (icalendar--get-event-property e 'DURATION)))
(icalendar--dmsg "%s: `%s'" start-d summary)
;; check whether start-time is missing
(if (and dtstart
(string=
(cadr (icalendar--get-event-property-attributes
e 'DTSTART))
"DATE"))
(setq start-t nil))
(when duration
(let ((dtend-dec-d (icalendar--add-decoded-times
dtstart-dec
(icalendar--decode-isoduration duration)))
(dtend-1-dec-d (icalendar--add-decoded-times
dtstart-dec
(icalendar--decode-isoduration duration
t))))
(if (and dtend-dec (not (eq dtend-dec dtend-dec-d)))
(message "Inconsistent endtime and duration for %s"
summary))
(setq dtend-dec dtend-dec-d)
(setq dtend-1-dec dtend-1-dec-d)))
(setq end-d (if dtend-dec
(icalendar--datetime-to-diary-date dtend-dec)
start-d))
(setq end-1-d (if dtend-1-dec
(icalendar--datetime-to-diary-date dtend-1-dec)
start-d))
(setq end-t (if (and
dtend-dec
(not (string=
(cadr
(icalendar--get-event-property-attributes
e 'DTEND))
"DATE")))
(icalendar--datetime-to-colontime dtend-dec)))
(icalendar--dmsg "start-d: %s, end-d: %s" start-d end-d)
(cond
;; recurring event
(rrule
(setq diary-string
(icalendar--convert-recurring-to-diary e dtstart-dec start-t
end-t))
(setq event-ok t))
(rdate
(icalendar--dmsg "rdate event")
(setq diary-string "")
(mapc (lambda (_datestring)
(setq diary-string
(concat diary-string "......")))
(icalendar--split-value rdate)))
;; non-recurring event
;; all-day event
((not (string= start-d end-d))
(setq diary-string
(icalendar--convert-non-recurring-all-day-to-diary
start-d end-1-d))
(setq event-ok t))
;; not all-day
((and start-t (or (not end-t)
(not (string= start-t end-t))))
(setq diary-string
(icalendar--convert-non-recurring-not-all-day-to-diary
dtstart-dec start-t end-t))
(setq event-ok t))
;; all-day event
(t
(icalendar--dmsg "all day event")
(setq diary-string (icalendar--datetime-to-diary-date
dtstart-dec "/"))
(setq event-ok t)))
;; add all other elements unless the user doesn't want to have
;; them
(if event-ok
(progn
(setq diary-string
(concat diary-string " "
(icalendar--format-ical-event e)))
(if do-not-ask (setq summary nil))
;; add entry to diary and store actual name of diary
;; file (in case it was nil)
(setq diary-filename
(icalendar--add-diary-entry diary-string diary-filename
non-marking summary)))
;; event was not ok
(setq found-error t)
(setq error-string
(format "%s\nCannot handle this event:%s"
error-string e))))
;; FIXME: inform user about ignored event properties
;; handle errors
(error
(message "Ignoring event \"%s\"" e)
(setq found-error t)
(setq error-string (format "%s\n%s\nCannot handle this event: %s"
error-val error-string e))
(message "%s" error-string))))
;; insert final newline
(if diary-filename
(let ((b (find-buffer-visiting diary-filename)))
(when b
(save-current-buffer
(set-buffer b)
(goto-char (point-max))
(insert "\n")))))
(if found-error
(save-current-buffer
(set-buffer (get-buffer-create "*icalendar-errors*"))
(erase-buffer)
(insert error-string)))
(message "Converting iCalendar...done")
found-error))
;; subroutines for importing
(defun icalendar--convert-recurring-to-diary (e dtstart-dec start-t end-t)
"Convert recurring iCalendar event E to diary format.
DTSTART-DEC is the DTSTART property of E.
START-T is the event's start time in diary format.
END-T is the event's end time in diary format."
(icalendar--dmsg "recurring event")
(let* ((rrule (icalendar--get-event-property e 'RRULE))
(rrule-props (icalendar--split-value rrule))
(frequency (cadr (assoc 'FREQ rrule-props)))
(until (cadr (assoc 'UNTIL rrule-props)))
(count (cadr (assoc 'COUNT rrule-props)))
(interval (read (or (cadr (assoc 'INTERVAL rrule-props)) "1")))
(dtstart-conv (icalendar--datetime-to-diary-date dtstart-dec))
(until-conv (icalendar--datetime-to-diary-date
(icalendar--decode-isodatetime until)))
(until-1-conv (icalendar--datetime-to-diary-date
(icalendar--decode-isodatetime until -1)))
(result ""))
;; FIXME FIXME interval!!!!!!!!!!!!!
(when count
(if until
(message "Must not have UNTIL and COUNT -- ignoring COUNT element!")
(let ((until-1 0))
(cond ((string-equal frequency "DAILY")
(setq until (icalendar--add-decoded-times
dtstart-dec
(list 0 0 0 (* (read count) interval) 0 0)))
(setq until-1 (icalendar--add-decoded-times
dtstart-dec
(list 0 0 0 (* (- (read count) 1) interval)
0 0)))
)
((string-equal frequency "WEEKLY")
(setq until (icalendar--add-decoded-times
dtstart-dec
(list 0 0 0 (* (read count) 7 interval) 0 0)))
(setq until-1 (icalendar--add-decoded-times
dtstart-dec
(list 0 0 0 (* (- (read count) 1) 7
interval) 0 0)))
)
((string-equal frequency "MONTHLY")
(setq until (icalendar--add-decoded-times
dtstart-dec (list 0 0 0 0 (* (- (read count) 1)
interval) 0)))
(setq until-1 (icalendar--add-decoded-times
dtstart-dec (list 0 0 0 0 (* (- (read count) 1)
interval) 0)))
)
((string-equal frequency "YEARLY")
(setq until (icalendar--add-decoded-times
dtstart-dec (list 0 0 0 0 0 (* (- (read count) 1)
interval))))
(setq until-1 (icalendar--add-decoded-times
dtstart-dec
(list 0 0 0 0 0 (* (- (read count) 1)
interval))))
)
(t
(message "Cannot handle COUNT attribute for `%s' events."
frequency)))
(setq until-conv (icalendar--datetime-to-diary-date until))
(setq until-1-conv (icalendar--datetime-to-diary-date until-1))
))
)
(cond ((string-equal frequency "WEEKLY")
(let* ((byday (cadr (assoc 'BYDAY rrule-props)))
(weekdays
(icalendar--get-weekday-numbers byday))
(weekday-clause
(when (> (length weekdays) 1)
(format "(memq (calendar-day-of-week date) '%s) "
weekdays))))
(if (not start-t)
(progn
;; weekly and all-day
(icalendar--dmsg "weekly all-day")
(if until
(setq result
(format
(concat "%%%%(and "
"%s"
"(diary-block %s %s))")
(or weekday-clause
(format "(diary-cyclic %d %s) "
(* interval 7)
dtstart-conv))
dtstart-conv
(if count until-1-conv until-conv)
))
(setq result
(format "%%%%(and %s(diary-cyclic %d %s))"
(or weekday-clause "")
(if weekday-clause 1 (* interval 7))
dtstart-conv))))
;; weekly and not all-day
(icalendar--dmsg "weekly not-all-day")
(if until
(setq result
(format
(concat "%%%%(and "
"%s"
"(diary-block %s %s)) "
"%s%s%s")
(or weekday-clause
(format "(diary-cyclic %d %s) "
(* interval 7)
dtstart-conv))
dtstart-conv
until-conv
(or start-t "")
(if end-t "-" "") (or end-t "")))
;; no limit
;; FIXME!!!!
;; DTSTART;VALUE=DATE-TIME:20030919T090000
;; DTEND;VALUE=DATE-TIME:20030919T113000
(setq result
(format
"%%%%(and %s(diary-cyclic %d %s)) %s%s%s"
(or weekday-clause "")
(if weekday-clause 1 (* interval 7))
dtstart-conv
(or start-t "")
(if end-t "-" "") (or end-t "")))))))
;; yearly
((string-equal frequency "YEARLY")
(icalendar--dmsg "yearly")
(if until
(let ((day (nth 3 dtstart-dec))
(month (nth 4 dtstart-dec)))
(setq result (concat "%%(and (diary-date "
(cond ((eq calendar-date-style 'iso)
(format "t %d %d" month day))
((eq calendar-date-style 'european)
(format "%d %d t" day month))
((eq calendar-date-style 'american)
(format "%d %d t" month day)))
") (diary-block "
dtstart-conv
" "
until-conv
")) "
(or start-t "")
(if end-t "-" "")
(or end-t ""))))
(setq result (format
"%%%%(diary-anniversary %s) %s%s%s"
(let* ((year (nth 5 dtstart-dec))
(dtstart-1y-dec (copy-sequence dtstart-dec)))
(setf (nth 5 dtstart-1y-dec) (1- year))
(icalendar--datetime-to-diary-date dtstart-1y-dec))
(or start-t "")
(if end-t "-" "") (or end-t "")))))
;; monthly
((string-equal frequency "MONTHLY")
(icalendar--dmsg "monthly")
(let* ((byday (cadr (assoc 'BYDAY rrule-props)))
(count-weekday
(and byday
(save-match-data
(when (string-match "\\(-?[0-9]+\\)\\([A-Z][A-Z]\\)"
byday)
(cons (substring byday
(match-beginning 1)
(match-end 1))
(substring byday
(match-beginning 2)
(match-end 2)))))))
(rule-part
(if count-weekday
(let ((count (car count-weekday))
(weekdaynum (icalendar--get-weekday-number
(cdr count-weekday))))
;; FIXME: this is valid only for interval==1
(format "(diary-float t %s %s)" weekdaynum count))
(format "(diary-date %s)"
(let ((day (nth 3 dtstart-dec)))
(cond ((eq calendar-date-style 'iso)
(format "t t %d" day))
((eq calendar-date-style 'european)
(format "%d t t" day))
((eq calendar-date-style 'american)
(format "t %d t" day))))))))
(setq result
(format
"%%%%(and %s (diary-block %s %s)) %s%s%s"
rule-part
dtstart-conv
(if until
until-conv
(if (eq calendar-date-style 'iso) "9999 1 1" "1 1 9999")) ;; FIXME: should be unlimited
(or start-t "")
(if end-t "-" "") (or end-t "")))))
;; daily
((and (string-equal frequency "DAILY"))
(if until
(setq result
(format
(concat "%%%%(and (diary-cyclic %s %s) "
"(diary-block %s %s)) %s%s%s")
interval dtstart-conv dtstart-conv
(if count until-1-conv until-conv)
(or start-t "")
(if end-t "-" "") (or end-t "")))
(setq result
(format
"%%%%(and (diary-cyclic %s %s)) %s%s%s"
interval
dtstart-conv
(or start-t "")
(if end-t "-" "") (or end-t ""))))))
;; Handle exceptions from recurrence rules
(let ((ex-dates (icalendar--get-event-properties e 'EXDATE)))
(while ex-dates
(let* ((ex-start (icalendar--decode-isodatetime
(car ex-dates)))
(ex-d (icalendar--datetime-to-diary-date
ex-start)))
(setq result
(replace-regexp-in-string "^%%(\\(and \\)?"
(format
"%%%%(and (not (diary-date %s)) "
ex-d)
result)))
(setq ex-dates (cdr ex-dates))))
;; FIXME: exception rules are not recognized
(if (icalendar--get-event-property e 'EXRULE)
(setq result
(concat result
"\n Exception rules: "
(icalendar--get-event-properties
e 'EXRULE))))
result))
(defun icalendar--convert-non-recurring-all-day-to-diary (start-d end-d)
"Convert non-recurring iCalendar EVENT to diary format.
DTSTART is the decoded DTSTART property of E.
Argument START-D gives the first day.
Argument END-D gives the last day."
(icalendar--dmsg "non-recurring all-day event")
(format "%%%%(and (diary-block %s %s))" start-d end-d))
(defun icalendar--convert-non-recurring-not-all-day-to-diary (dtstart-dec
start-t
end-t)
"Convert recurring icalendar EVENT to diary format.
DTSTART-DEC is the decoded DTSTART property of E.
START-T is the event's start time in diary format.
END-T is the event's end time in diary format."
(icalendar--dmsg "not all day event")
(cond (end-t
(format "%s %s-%s"
(icalendar--datetime-to-diary-date
dtstart-dec "/")
start-t end-t))
(t
(format "%s %s"
(icalendar--datetime-to-diary-date
dtstart-dec "/")
start-t))))
(defun icalendar--add-diary-entry (string diary-filename non-marking
&optional summary)
"Add STRING to the diary file DIARY-FILENAME.
STRING must be a properly formatted valid diary entry. NON-MARKING
determines whether diary events are created as non-marking. If
SUMMARY is not nil it must be a string that gives the summary of the
entry. In this case the user will be asked whether he wants to insert
the entry."
(when (or (not summary)
(y-or-n-p (format-message "Add appointment for `%s' to diary? "
summary)))
(when summary
(setq non-marking
(y-or-n-p "Make appointment non-marking? ")))
(unless diary-filename
(setq diary-filename
(read-file-name "Add appointment to this diary file: ")))
(diary-make-entry string non-marking diary-filename t t))
;; return diary-filename in case it has been changed interactively
diary-filename)
;; ======================================================================
;; Examples
;; ======================================================================
(defun icalendar-import-format-sample (event)
"Example function for formatting an iCalendar EVENT."
(format (concat "SUMMARY='%s' DESCRIPTION='%s' LOCATION='%s' ORGANIZER='%s' "
"STATUS='%s' URL='%s' CLASS='%s'")
(or (icalendar--get-event-property event 'SUMMARY) "")
(or (icalendar--get-event-property event 'DESCRIPTION) "")
(or (icalendar--get-event-property event 'LOCATION) "")
(or (icalendar--get-event-property event 'ORGANIZER) "")
(or (icalendar--get-event-property event 'STATUS) "")
(or (icalendar--get-event-property event 'URL) "")
(or (icalendar--get-event-property event 'CLASS) "")))
;; Obsolete
(defconst icalendar-version "0.19" "Version number of icalendar.el.")
(make-obsolete-variable 'icalendar-version 'emacs-version "28.1")
(provide 'icalendar)
;;; icalendar.el ends here