2017-07-16 18:07:00 +02:00
|
|
|
|
;;; org.el -- Daniel's org mode configuration
|
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Basic Setup
|
|
|
|
|
|
2017-08-14 08:18:55 +02:00
|
|
|
|
(require 'org)
|
|
|
|
|
|
2017-07-16 18:07:00 +02:00
|
|
|
|
(setq org-deadline-warning-days 14
|
|
|
|
|
org-read-date-popup-calendar t
|
|
|
|
|
org-insert-heading-respect-content t
|
|
|
|
|
org-list-description-max-indent 5
|
|
|
|
|
org-adapt-indentation nil
|
|
|
|
|
org-edit-timestamp-down-means-later t
|
|
|
|
|
org-archive-location "%s_archive.gpg::"
|
|
|
|
|
org-image-actual-width nil
|
2017-08-14 08:18:55 +02:00
|
|
|
|
org-footnote-section nil
|
|
|
|
|
org-log-into-drawer "LOGBOOK"
|
|
|
|
|
org-log-reschedule 'time
|
|
|
|
|
org-clone-delete-id t
|
|
|
|
|
org-catch-invisible-edits 'error)
|
|
|
|
|
|
|
|
|
|
(bind-key [remap org-return] 'org-return-indent org-mode-map)
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(setq org-todo-keywords
|
|
|
|
|
'((sequence "TODO(t)" "CONT(n!)" "|" "DONE(d@)")
|
|
|
|
|
(sequence "GOTO(g)" "ATTN(a)" "|" "DONE(d@)")
|
|
|
|
|
(sequence "READ(r)" "CONT(n!)" "|" "DONE(d@)")
|
|
|
|
|
(sequence "DELG(e@/!)" "WAIT(w@/!)" "HOLD(h@/!)" "|" "CANC(c@/!)" "PHONE" "MEETING")))
|
|
|
|
|
|
|
|
|
|
(setq org-todo-state-tags-triggers
|
|
|
|
|
'(("WAIT" ("WAIT" . t))
|
|
|
|
|
("HOLD" ("HOLD" . t))
|
2017-07-27 13:06:54 +02:00
|
|
|
|
(done ("HOLD") ("WAIT") ("DATE") ("NO_EXPORT" . t))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
("TODO" ("HOLD") ("WAIT") ("NO_EXPORT"))
|
|
|
|
|
("READ" ("READ" . t) ("HOLD") ("WAIT"))
|
|
|
|
|
("GOTO" ("DATE" . t) ("HOLD") ("WAIT"))
|
|
|
|
|
("CONT" ("HOLD") ("WAIT"))
|
|
|
|
|
("ATTN" ("HOLD") ("WAIT"))))
|
|
|
|
|
|
|
|
|
|
(setq org-tag-alist
|
|
|
|
|
'((:startgroup . nil)
|
|
|
|
|
("WORK" . ?w)
|
|
|
|
|
("HOME" . ?h)
|
|
|
|
|
("FUN" . ?f)
|
|
|
|
|
("UNTAGGED" . ?u)
|
|
|
|
|
(:endgroup . nil)
|
|
|
|
|
("NOTE" . ?n)))
|
|
|
|
|
|
|
|
|
|
(setq org-treat-S-cursor-todo-selection-as-state-change nil
|
|
|
|
|
org-fast-tag-selection-single-key 'expert)
|
|
|
|
|
|
|
|
|
|
(setq org-global-properties
|
|
|
|
|
'(("Effort_ALL" . "0:00 0:15 0:30 0:45 1:00 2:00 3:00 4:00 5:00 6:00")))
|
|
|
|
|
|
|
|
|
|
(setq org-columns-default-format
|
|
|
|
|
"%80ITEM(Task) %10Effort(Effort){:} %10CLOCKSUM")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Faces
|
|
|
|
|
|
|
|
|
|
(setq org-todo-keyword-faces
|
|
|
|
|
'(("TODO" :foreground "red" :weight normal)
|
|
|
|
|
("GOTO" :foreground "red" :weight normal)
|
|
|
|
|
("READ" :foreground "red" :weight normal)
|
|
|
|
|
("CONT" :foreground "DeepSkyBlue" :weight normal)
|
|
|
|
|
("ATTN" :foreground "DeepSkyBlue" :weight normal)
|
|
|
|
|
("DONE" :foreground "forest green" :weight normal)
|
|
|
|
|
("DELG" :foreground "dark orange" :weight normal)
|
|
|
|
|
("WAIT" :foreground "orange" :weight normal)
|
|
|
|
|
("HOLD" :foreground "magenta" :weight normal)
|
|
|
|
|
("CANC" :foreground "lime green" :weight normal)
|
|
|
|
|
("MEETING" :foreground "forest green" :weight normal)
|
|
|
|
|
("PHONE" :foreground "forest green" :weight normal)
|
|
|
|
|
("REPEAT" :foreground "indian red" :weight normal)))
|
|
|
|
|
|
|
|
|
|
(setq org-fontify-done-headline nil)
|
|
|
|
|
|
|
|
|
|
(setq org-priority-faces
|
|
|
|
|
'((?A . (:foreground "Red" :weight bold))
|
|
|
|
|
(?B . (:foreground "firebrick"))
|
|
|
|
|
(?C . (:foreground "tomato"))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Clocking
|
|
|
|
|
|
|
|
|
|
(setq org-clock-history-length 23
|
|
|
|
|
org-clock-in-resume t
|
|
|
|
|
org-clock-into-drawer t
|
|
|
|
|
org-clock-idle-time nil
|
|
|
|
|
org-clock-out-remove-zero-time-clocks t
|
|
|
|
|
org-clock-out-when-done '("DONE" "CANC" "WAIT" "HOLD")
|
|
|
|
|
org-clock-auto-clock-resolution 'when-no-clock-is-running
|
|
|
|
|
org-clock-mode-line-total 'auto
|
2017-09-02 13:33:57 +02:00
|
|
|
|
org-clock-report-include-clocking-task t
|
2017-07-16 18:07:00 +02:00
|
|
|
|
org-clock-in-switch-to-state (lambda (kw)
|
|
|
|
|
(when (and (not
|
|
|
|
|
(and (boundp 'org-capture-mode)
|
|
|
|
|
org-capture-mode)))
|
|
|
|
|
(cond
|
|
|
|
|
((member (org-get-todo-state)
|
|
|
|
|
(list "TODO" "READ"))
|
|
|
|
|
"CONT")
|
|
|
|
|
((member (org-get-todo-state)
|
|
|
|
|
(list "GOTO"))
|
2017-08-14 08:18:55 +02:00
|
|
|
|
"ATTN")))))
|
|
|
|
|
|
|
|
|
|
(org-clock-persistence-insinuate)
|
|
|
|
|
|
|
|
|
|
(setq org-clock-persist t
|
|
|
|
|
org-clock-persist-query-resume nil)
|
|
|
|
|
|
|
|
|
|
(setq org-time-clocksum-format '(:hours "%d"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
:require-hours t
|
|
|
|
|
:minutes ":%02d"
|
|
|
|
|
:require-minutes t)
|
2017-08-14 08:18:55 +02:00
|
|
|
|
org-time-stamp-rounding-minutes '(1 1))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
;; Default Tasks for Working, Home, Breaks
|
|
|
|
|
|
2017-09-02 13:34:22 +02:00
|
|
|
|
(defvar org-working-task-id nil)
|
|
|
|
|
(defvar org-break-task-id nil)
|
|
|
|
|
(defvar org-home-task-id nil)
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(add-hook 'org-clock-in-hook ; mark current default task
|
|
|
|
|
(lambda ()
|
|
|
|
|
(let ((current-id (org-id-get org-clock-marker)))
|
|
|
|
|
(when (member current-id (list org-working-task-id
|
|
|
|
|
org-home-task-id))
|
|
|
|
|
(org-clock-mark-default-task)))))
|
|
|
|
|
|
|
|
|
|
;; Clock in default task if no other task is given
|
|
|
|
|
|
|
|
|
|
(defun db/find-parent-task ()
|
|
|
|
|
;; http://doc.norang.ca/org-mode.html#Clocking
|
|
|
|
|
"Return point of the nearest parent task, and NIL if no such task exists."
|
|
|
|
|
(save-mark-and-excursion
|
|
|
|
|
(save-restriction
|
|
|
|
|
(widen)
|
|
|
|
|
(let ((parent-task nil))
|
|
|
|
|
(or (org-at-heading-p)
|
|
|
|
|
(org-back-to-heading t))
|
|
|
|
|
(while (and (not parent-task)
|
|
|
|
|
(org-up-heading-safe))
|
|
|
|
|
(let ((tags (nth 5 (org-heading-components))))
|
|
|
|
|
(unless (and tags (member "NOP" (split-string tags ":" t)))
|
|
|
|
|
(setq parent-task (point)))))
|
|
|
|
|
parent-task))))
|
|
|
|
|
|
|
|
|
|
(defun db/ensure-running-clock ()
|
|
|
|
|
"Clocks in into the parent task, if it exists, or the default task."
|
|
|
|
|
(when (and (not org-clock-clocking-in)
|
|
|
|
|
(not org-clock-resolving-clocks-due-to-idleness))
|
|
|
|
|
(let ((parent-task (db/find-parent-task)))
|
|
|
|
|
(save-mark-and-excursion
|
|
|
|
|
(cond
|
|
|
|
|
(parent-task
|
|
|
|
|
;; found parent task
|
|
|
|
|
(org-with-point-at parent-task
|
|
|
|
|
(org-clock-in)))
|
|
|
|
|
((and (markerp org-clock-default-task)
|
|
|
|
|
(marker-buffer org-clock-default-task))
|
|
|
|
|
;; default task is set
|
|
|
|
|
(org-with-point-at org-clock-default-task
|
|
|
|
|
(org-clock-in)))
|
|
|
|
|
(t
|
|
|
|
|
(org-clock-in '(4))))))))
|
|
|
|
|
|
|
|
|
|
(add-hook 'org-clock-out-hook #'db/ensure-running-clock 'append)
|
|
|
|
|
|
|
|
|
|
;; clock-in helpers
|
|
|
|
|
|
|
|
|
|
(defun clock-in-task-by-id (task-id)
|
|
|
|
|
"Clock in org mode task as given by TASK-ID."
|
|
|
|
|
(org-with-point-at (org-id-find task-id 'marker)
|
|
|
|
|
(org-clock-in))
|
|
|
|
|
(org-save-all-org-buffers))
|
|
|
|
|
|
|
|
|
|
(defun clock-out-task-by-id (task-id)
|
|
|
|
|
"Clock out org mode task as given by TASK-ID."
|
|
|
|
|
(org-with-point-at (org-id-find task-id 'marker)
|
|
|
|
|
(org-clock-out))
|
|
|
|
|
(org-save-all-org-buffers))
|
|
|
|
|
|
|
|
|
|
(defun db/org-clock-in-last-task (&optional arg)
|
|
|
|
|
;; from doc.norang.ca, originally bh/clock-in-last-task
|
|
|
|
|
"Clock in the interrupted task if there is one.
|
|
|
|
|
|
|
|
|
|
Skip the default task and get the next one. If ARG is given,
|
|
|
|
|
forces clocking in of the default task."
|
|
|
|
|
(interactive "p")
|
|
|
|
|
(let ((clock-in-to-task
|
|
|
|
|
(cond
|
|
|
|
|
((eq arg 4) org-clock-default-task)
|
|
|
|
|
((and (org-clock-is-active)
|
|
|
|
|
(equal org-clock-default-task (cadr org-clock-history)))
|
|
|
|
|
(caddr org-clock-history))
|
|
|
|
|
((org-clock-is-active) (cadr org-clock-history))
|
|
|
|
|
((equal org-clock-default-task (car org-clock-history))
|
|
|
|
|
(cadr org-clock-history))
|
|
|
|
|
(t (car org-clock-history)))))
|
|
|
|
|
(widen)
|
|
|
|
|
(org-with-point-at clock-in-to-task
|
|
|
|
|
(org-clock-in nil))))
|
|
|
|
|
|
|
|
|
|
;; Communicate the currently clocked in task to the outside world
|
|
|
|
|
|
|
|
|
|
(defvar db/org-clock-current-task-file
|
|
|
|
|
"~/.org-current-task")
|
|
|
|
|
|
|
|
|
|
(defun db/org-current-task ()
|
|
|
|
|
"Format currently clocked task and write it to
|
|
|
|
|
`db/org-clock-current-task-file'."
|
|
|
|
|
(with-temp-file db/org-clock-current-task-file
|
|
|
|
|
(let ((clock-buffer (marker-buffer org-clock-marker)))
|
|
|
|
|
(if (null clock-buffer)
|
|
|
|
|
(insert "No running clock"))
|
|
|
|
|
(let ((current-task org-clock-heading)
|
|
|
|
|
(current-file (file-name-sans-extension
|
|
|
|
|
(file-name-nondirectory
|
|
|
|
|
(buffer-file-name clock-buffer)))))
|
|
|
|
|
(insert current-file "/" current-task)))))
|
|
|
|
|
|
|
|
|
|
(add-hook 'org-clock-in-hook #'db/org-current-task)
|
|
|
|
|
|
|
|
|
|
(defun db/select-clocking-task ()
|
|
|
|
|
"Select task from recent clocked-in tasks."
|
|
|
|
|
(interactive)
|
|
|
|
|
(org-clock-in '(4)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Agenda Customization
|
|
|
|
|
|
|
|
|
|
(defvar db/org-default-work-file
|
|
|
|
|
"~/Documents/uni/admin/misc/work.org")
|
|
|
|
|
|
|
|
|
|
(defvar db/org-default-home-file
|
2017-08-04 22:08:43 +02:00
|
|
|
|
"~/Documents/home/admin/home.org")
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(defvar db/org-default-notes-file
|
2017-08-04 22:08:43 +02:00
|
|
|
|
"~/Documents/home/misc/notes.org")
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(defvar db/org-default-refile-file
|
|
|
|
|
"~/Desktop/refile.org")
|
|
|
|
|
|
|
|
|
|
(defvar db/org-default-pensieve-file
|
2017-08-04 22:08:43 +02:00
|
|
|
|
"~/Documents/home/misc/pensieve.org.gpg")
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(setq org-agenda-files (list db/org-default-home-file
|
|
|
|
|
db/org-default-work-file
|
|
|
|
|
db/org-default-refile-file
|
|
|
|
|
db/org-default-notes-file))
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-include-diary nil
|
|
|
|
|
org-agenda-span 1
|
|
|
|
|
org-agenda-diary-file db/org-default-refile-file
|
|
|
|
|
org-agenda-insert-diary-strategy 'top-level
|
|
|
|
|
org-catch-invisible-edits 'show
|
|
|
|
|
org-ellipsis "⤵"
|
|
|
|
|
org-agenda-sorting-strategy '((agenda time-up habit-up priority-down)
|
|
|
|
|
(todo category-keep)
|
|
|
|
|
(tags category-keep)
|
|
|
|
|
(search category-keep)))
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-window-setup 'current-window
|
|
|
|
|
org-agenda-restore-windows-after-quit t
|
|
|
|
|
org-agenda-compact-blocks nil)
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-todo-ignore-with-date nil
|
|
|
|
|
org-agenda-todo-ignore-deadlines nil
|
|
|
|
|
org-agenda-todo-ignore-scheduled nil
|
|
|
|
|
org-agenda-todo-ignore-timestamp nil
|
|
|
|
|
org-agenda-skip-deadline-if-done t
|
|
|
|
|
org-agenda-skip-scheduled-if-done t
|
|
|
|
|
org-agenda-skip-timestamp-if-done t
|
|
|
|
|
org-agenda-skip-scheduled-if-deadline-is-shown 'not-today
|
|
|
|
|
org-agenda-tags-todo-honor-ignore-options t
|
|
|
|
|
org-agenda-start-with-log-mode nil
|
|
|
|
|
org-agenda-log-mode-items '(closed state)
|
|
|
|
|
org-agenda-remove-tags t
|
|
|
|
|
org-agenda-sticky nil
|
|
|
|
|
org-agenda-inhibit-startup t
|
|
|
|
|
org-agenda-tags-todo-honor-ignore-options t
|
|
|
|
|
org-agenda-dim-blocked-tasks nil
|
|
|
|
|
org-enforce-todo-checkbox-dependencies t
|
|
|
|
|
org-enforce-todo-dependencies t
|
|
|
|
|
org-agenda-use-time-grid t
|
|
|
|
|
org-agenda-persistent-filter t
|
|
|
|
|
org-agenda-search-headline-for-time nil)
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-clock-consistency-checks
|
|
|
|
|
'(:max-duration 999999999
|
|
|
|
|
:min-duration 0
|
|
|
|
|
:max-gap 0
|
|
|
|
|
:gap-ok-around nil
|
|
|
|
|
:default-face ((:background "DarkRed") (:foreground "white"))
|
|
|
|
|
:overlap-face nil :gap-face nil :no-end-time-face nil
|
|
|
|
|
:long-face nil :short-face nil))
|
|
|
|
|
|
|
|
|
|
(add-hook 'org-agenda-mode-hook #'hl-line-mode 'append)
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-clockreport-parameter-plist
|
|
|
|
|
'(:link t :maxlevel 4 :compact t :narrow 60 :fileskip0 t))
|
|
|
|
|
|
|
|
|
|
(setq org-stuck-projects
|
|
|
|
|
'("-REGULAR-HOLD-NOTE+TODO=\"\""
|
|
|
|
|
("CONT" "TODO" "READ" "WAIT" "GOTO" "DELG")
|
|
|
|
|
("DATE" "NOP" "HOLD")
|
|
|
|
|
""))
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-prefix-format
|
|
|
|
|
'((agenda . " %11s%-4c %4e %?-12t")
|
|
|
|
|
(timeline . "% s")
|
|
|
|
|
(todo . " %-4c ")
|
|
|
|
|
(tags . " %-4c ")
|
|
|
|
|
(search . " %-4c ")))
|
|
|
|
|
|
|
|
|
|
(defun db/org-agenda-list-deadlines (&optional match)
|
|
|
|
|
;; XXX org-agenda-later does not work, fix this
|
|
|
|
|
"Prepare agenda view that only lists upcoming deadlines.
|
|
|
|
|
|
|
|
|
|
Ignores MATCH."
|
|
|
|
|
(interactive "P")
|
|
|
|
|
(catch 'exit
|
|
|
|
|
(org-agenda-prepare "Deadlines")
|
|
|
|
|
(org-compile-prefix-format 'agenda)
|
|
|
|
|
(org-set-sorting-strategy 'agenda)
|
|
|
|
|
|
|
|
|
|
(let* ((today (org-today))
|
|
|
|
|
(thefiles (org-agenda-files nil 'ifmode))
|
|
|
|
|
(inhibit-redisplay (not debug-on-error))
|
|
|
|
|
s rtn rtnall file files date start-pos end-pos)
|
|
|
|
|
|
|
|
|
|
;; headline
|
|
|
|
|
(unless org-agenda-compact-blocks
|
|
|
|
|
(setq s (point))
|
|
|
|
|
(if org-agenda-overriding-header
|
|
|
|
|
(insert (org-add-props (copy-sequence org-agenda-overriding-header)
|
|
|
|
|
nil 'face 'org-agenda-structure) "\n"))
|
|
|
|
|
(org-agenda-mark-header-line s))
|
|
|
|
|
|
|
|
|
|
;; actual content
|
|
|
|
|
(setq date (calendar-gregorian-from-absolute today)
|
|
|
|
|
s (point)
|
|
|
|
|
start-pos (point)
|
|
|
|
|
files thefiles
|
|
|
|
|
rtnall nil)
|
|
|
|
|
(while (setq file (pop files))
|
|
|
|
|
(catch 'nextfile
|
|
|
|
|
(org-check-agenda-file file)
|
|
|
|
|
(setq rtn (apply 'org-agenda-get-day-entries
|
|
|
|
|
file date
|
|
|
|
|
'(:deadline)))
|
|
|
|
|
(setq rtnall (append rtnall rtn)))) ;; all entries
|
|
|
|
|
(when rtnall
|
|
|
|
|
(insert (org-agenda-finalize-entries rtnall 'agenda)
|
|
|
|
|
"\n"))
|
|
|
|
|
|
|
|
|
|
;; finalize
|
|
|
|
|
(goto-char (point-min))
|
|
|
|
|
(or org-agenda-multi (org-agenda-fit-window-to-buffer))
|
|
|
|
|
(unless (and (pos-visible-in-window-p (point-min))
|
|
|
|
|
(pos-visible-in-window-p (point-max)))
|
|
|
|
|
(goto-char (1- (point-max)))
|
|
|
|
|
(recenter -1)
|
|
|
|
|
(if (not (pos-visible-in-window-p (or start-pos 1)))
|
|
|
|
|
(progn
|
|
|
|
|
(goto-char (or start-pos 1))
|
|
|
|
|
(recenter 1))))
|
|
|
|
|
(goto-char (or start-pos 1))
|
|
|
|
|
(add-text-properties
|
|
|
|
|
(point-min) (point-max)
|
|
|
|
|
`(org-agenda-type agenda
|
|
|
|
|
org-redo-cmd
|
|
|
|
|
(db/org-agenda-list-deadlines ,match)))
|
|
|
|
|
(org-agenda-finalize)
|
|
|
|
|
(setq buffer-read-only t)
|
|
|
|
|
(message ""))))
|
|
|
|
|
|
|
|
|
|
(defun db/org-agenda-skip-tag (tag &optional others)
|
|
|
|
|
;; https://stackoverflow.com/questions/10074016/org-mode-filter-on-tag-in-agenda-view
|
|
|
|
|
"Skip all entries that correspond to TAG.
|
|
|
|
|
|
|
|
|
|
If OTHERS is true, skip all entries that do not correspond to TAG."
|
|
|
|
|
(let* ((next-headline (save-mark-and-excursion
|
|
|
|
|
(or (outline-next-heading) (point-max))))
|
|
|
|
|
(current-headline (or (and (org-at-heading-p)
|
|
|
|
|
(point))
|
|
|
|
|
(save-mark-and-excursion
|
|
|
|
|
;; remember to also consider invisible headings
|
|
|
|
|
(org-back-to-heading t))))
|
|
|
|
|
(has-tag (member tag (org-get-tags-at current-headline))))
|
|
|
|
|
(if (or (and others (not has-tag))
|
|
|
|
|
(and (not others) has-tag))
|
|
|
|
|
next-headline
|
|
|
|
|
nil)))
|
|
|
|
|
|
|
|
|
|
(defun db/cmp-date-property (prop)
|
|
|
|
|
;; https://emacs.stackexchange.com/questions/26351/custom-sorting-for-agenda
|
|
|
|
|
"Compare two `org-mode' agenda entries, `A' and `B', by some date property.
|
|
|
|
|
|
|
|
|
|
If a is before b, return -1. If a is after b, return 1. If they
|
|
|
|
|
are equal return nil."
|
|
|
|
|
(lexical-let ((prop prop))
|
|
|
|
|
#'(lambda (a b)
|
|
|
|
|
(let* ((a-pos (get-text-property 0 'org-marker a))
|
|
|
|
|
(b-pos (get-text-property 0 'org-marker b))
|
|
|
|
|
(a-date (or (org-entry-get a-pos prop)
|
|
|
|
|
(format "<%s>" (org-read-date t nil "now"))))
|
|
|
|
|
(b-date (or (org-entry-get b-pos prop)
|
|
|
|
|
(format "<%s>" (org-read-date t nil "now"))))
|
|
|
|
|
(cmp (compare-strings a-date nil nil b-date nil nil)))
|
|
|
|
|
(if (eq cmp t) nil (signum cmp))))))
|
|
|
|
|
|
|
|
|
|
(setq org-agenda-custom-commands
|
|
|
|
|
`(("A" "Main List"
|
|
|
|
|
((agenda
|
|
|
|
|
""
|
|
|
|
|
((org-agenda-entry-types '(:timestamp :sexp :scheduled :deadline))
|
|
|
|
|
(org-deadline-warning-days 0)
|
|
|
|
|
(org-agenda-files (append org-agenda-files
|
|
|
|
|
(mapcar #'second db/ical-org-links)))))
|
|
|
|
|
(db/org-agenda-list-deadlines
|
|
|
|
|
""
|
|
|
|
|
((org-agenda-overriding-header "Deadlines")
|
|
|
|
|
(org-agenda-sorting-strategy '(deadline-up priority-down))
|
|
|
|
|
(org-agenda-skip-deadline-prewarning-if-scheduled t)
|
|
|
|
|
(org-deadline-warning-days 30)))
|
|
|
|
|
(tags-todo "-NOAGENDA/WAIT|DELG"
|
|
|
|
|
((org-agenda-overriding-header "Waiting-fors")
|
|
|
|
|
(org-agenda-todo-ignore-deadlines t)
|
|
|
|
|
(org-agenda-todo-ignore-scheduled t)))
|
|
|
|
|
(tags "REFILE"
|
|
|
|
|
((org-agenda-files (list db/org-default-refile-file))
|
|
|
|
|
(org-agenda-overriding-header "Things to refile")))))
|
|
|
|
|
("R" "Reading List"
|
|
|
|
|
((tags-todo "READ/-DONE-CANC"
|
|
|
|
|
((org-agenda-overriding-header "To Read (unscheduled)")
|
|
|
|
|
(org-agenda-cmp-user-defined (db/cmp-date-property "CREATED"))
|
|
|
|
|
(org-agenda-sorting-strategy '(user-defined-up))
|
|
|
|
|
(org-agenda-todo-ignore-scheduled t)))))
|
|
|
|
|
("E" "Everything"
|
|
|
|
|
((tags-todo "/WAIT"
|
|
|
|
|
((org-agenda-overriding-header "Tasks requiring response/input")))
|
|
|
|
|
(tags-todo "-HOLD-READ-SOMEWHEN/-DONE"
|
|
|
|
|
((org-agenda-overriding-header "Things not being scheduled or deadlined")
|
|
|
|
|
(org-tags-match-list-sublevels t)
|
|
|
|
|
(org-agenda-todo-ignore-with-date t)
|
|
|
|
|
(org-agenda-sorting-strategy
|
|
|
|
|
'(priority-down time-up category-keep))))
|
|
|
|
|
(stuck ""
|
|
|
|
|
((org-agenda-overriding-header "Stuck Tasks")))))
|
|
|
|
|
("S" "Somewhen"
|
|
|
|
|
((tags "SOMEWHEN/-CANC-DONE"
|
2017-08-26 11:39:00 +02:00
|
|
|
|
((org-agenda-overriding-header "Things to do somewhen")
|
|
|
|
|
(org-tags-match-list-sublevels nil)))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
(tags-todo "/HOLD"
|
|
|
|
|
((org-agenda-overriding-header "Tasks on Hold")))))
|
|
|
|
|
("W" "Weekly Review"
|
|
|
|
|
((agenda ""
|
|
|
|
|
((org-agenda-span 7)
|
|
|
|
|
(org-agenda-archives-mode t)
|
|
|
|
|
(org-agenda-dim-blocked-tasks nil)
|
|
|
|
|
(org-agenda-skip-deadline-prewarning-if-scheduled t)))))
|
|
|
|
|
("M" "Monthly Preview"
|
|
|
|
|
((db/org-agenda-list-deadlines
|
|
|
|
|
""
|
|
|
|
|
((org-agenda-overriding-header "Deadlines")
|
|
|
|
|
(org-agenda-sorting-strategy '(deadline-up priority-down))
|
|
|
|
|
(org-deadline-warning-days 90)))
|
|
|
|
|
(agenda ""
|
|
|
|
|
((org-agenda-span 'month)
|
|
|
|
|
(org-agenda-dim-blocked-tasks nil)
|
|
|
|
|
(org-deadline-warning-days 0) ; covered by display above
|
|
|
|
|
))))
|
|
|
|
|
("N" "Notes" tags "NOTE"
|
|
|
|
|
((org-agenda-overriding-header "Notes")
|
|
|
|
|
(org-use-tag-inheritance nil)
|
|
|
|
|
(org-agenda-prefix-format '((tags . " ")))))))
|
|
|
|
|
|
|
|
|
|
(defun db/org-add-clocking-time ()
|
2017-08-26 11:38:44 +02:00
|
|
|
|
"Add \"CLOCK:\" line to the task under point in the current org-mode file.
|
|
|
|
|
|
|
|
|
|
Start and end time will be queried interactively."
|
|
|
|
|
(interactive)
|
|
|
|
|
;; FIXME: make this function to take two optional arguments
|
|
|
|
|
(if (not (eq major-mode 'org-mode))
|
|
|
|
|
(user-error "Must be in org mode")
|
|
|
|
|
(let* ((starting-time (org-read-date 4 'totime nil
|
|
|
|
|
"Start:" (current-time) nil t))
|
|
|
|
|
(ending-time (org-read-date 4 'totime nil
|
|
|
|
|
"End:" (current-time) nil t)))
|
|
|
|
|
(save-mark-and-excursion
|
|
|
|
|
(org-clock-find-position nil)
|
|
|
|
|
(open-line 1)
|
|
|
|
|
(indent-according-to-mode)
|
|
|
|
|
(insert "CLOCK: ")
|
|
|
|
|
(org-insert-time-stamp starting-time t t)
|
|
|
|
|
(insert "--")
|
|
|
|
|
(org-insert-time-stamp ending-time t t)
|
|
|
|
|
(org-clock-update-time-maybe)))))
|
|
|
|
|
|
|
|
|
|
(bind-key "C-c C-x C-a" #'db/org-add-clocking-time org-mode-map)
|
|
|
|
|
|
|
|
|
|
(defun db/org-agenda-add-clocking-time ()
|
2017-07-16 18:07:00 +02:00
|
|
|
|
"Add \"CLOCK:\" line to the task under point in the agenda.
|
|
|
|
|
|
|
|
|
|
Start and end time will be queried interactively."
|
|
|
|
|
(interactive)
|
|
|
|
|
;; FIXME: make this function to take two optional arguments
|
2017-08-26 11:38:44 +02:00
|
|
|
|
(if (not (eq major-mode 'org-agenda-mode))
|
|
|
|
|
(user-error "Must be in org agenda mode")
|
2017-07-16 18:07:00 +02:00
|
|
|
|
(save-window-excursion
|
|
|
|
|
(org-agenda-switch-to)
|
2017-08-26 11:38:44 +02:00
|
|
|
|
(db/org-add-clocking-time))))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
(eval-after-load 'org-agenda
|
2017-09-02 13:34:09 +02:00
|
|
|
|
'(bind-key "V u" #'db/org-agenda-add-clocking-time org-agenda-mode-map))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
;; A Hydra for changing agenda appearance
|
|
|
|
|
;; http://oremacs.com/2016/04/04/hydra-doc-syntax/
|
|
|
|
|
|
|
|
|
|
(defun db/org-agenda-span ()
|
|
|
|
|
"Return the display span of the current shown agenda."
|
|
|
|
|
(let ((args (get-text-property
|
|
|
|
|
(min (1- (point-max)) (point))
|
|
|
|
|
'org-last-args)))
|
|
|
|
|
(nth 2 args)))
|
|
|
|
|
|
|
|
|
|
(defhydra hydra-org-agenda-view (:hint none)
|
|
|
|
|
"
|
|
|
|
|
_d_: ?d? day _g_: time grid=?g? _a_: arch-trees
|
|
|
|
|
_w_: ?w? week _[_: inactive _A_: arch-files
|
|
|
|
|
_t_: ?t? fortnight _F_: follow=?F? _r_: report=?r?
|
|
|
|
|
_m_: ?m? month _e_: entry =?e? _D_: diary=?D?
|
|
|
|
|
_y_: ?y? year _q_: quit _L__l__c_: ?l?
|
|
|
|
|
|
|
|
|
|
"
|
|
|
|
|
("SPC" org-agenda-reset-view)
|
|
|
|
|
("d" org-agenda-day-view
|
|
|
|
|
(if (eq 'day (db/org-agenda-span))
|
|
|
|
|
"[x]" "[ ]"))
|
|
|
|
|
("w" org-agenda-week-view
|
|
|
|
|
(if (eq 'week (db/org-agenda-span))
|
|
|
|
|
"[x]" "[ ]"))
|
|
|
|
|
("t" org-agenda-fortnight-view
|
|
|
|
|
(if (eq 'fortnight (db/org-agenda-span))
|
|
|
|
|
"[x]" "[ ]"))
|
|
|
|
|
("m" org-agenda-month-view
|
|
|
|
|
(if (eq 'month (db/org-agenda-span)) "[x]" "[ ]"))
|
|
|
|
|
("y" org-agenda-year-view
|
|
|
|
|
(if (eq 'year (db/org-agenda-span)) "[x]" "[ ]"))
|
|
|
|
|
("l" org-agenda-log-mode
|
|
|
|
|
(format "% -3S" org-agenda-show-log))
|
|
|
|
|
("L" (org-agenda-log-mode '(4)))
|
|
|
|
|
("c" (org-agenda-log-mode 'clockcheck))
|
|
|
|
|
("F" org-agenda-follow-mode
|
|
|
|
|
(format "% -3S" org-agenda-follow-mode))
|
|
|
|
|
("a" org-agenda-archives-mode)
|
|
|
|
|
("A" (org-agenda-archives-mode 'files))
|
|
|
|
|
("r" org-agenda-clockreport-mode
|
|
|
|
|
(format "% -3S" org-agenda-clockreport-mode))
|
|
|
|
|
("e" org-agenda-entry-text-mode
|
|
|
|
|
(format "% -3S" org-agenda-entry-text-mode))
|
|
|
|
|
("g" org-agenda-toggle-time-grid
|
|
|
|
|
(format "% -3S" org-agenda-use-time-grid))
|
|
|
|
|
("D" org-agenda-toggle-diary
|
|
|
|
|
(format "% -3S" org-agenda-include-diary))
|
|
|
|
|
("!" org-agenda-toggle-deadlines)
|
|
|
|
|
("["
|
|
|
|
|
(let ((org-agenda-include-inactive-timestamps t))
|
|
|
|
|
(org-agenda-check-type t 'timeline 'agenda)
|
|
|
|
|
(org-agenda-redo)))
|
|
|
|
|
("q" (message "Abort") :exit t))
|
|
|
|
|
|
|
|
|
|
(eval-after-load 'org-agenda
|
|
|
|
|
'(bind-key "v" #'hydra-org-agenda-view/body org-agenda-mode-map))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Capturing
|
|
|
|
|
|
|
|
|
|
;; disable usage of helm for `org-capture'
|
|
|
|
|
(eval-after-load 'helm-mode
|
|
|
|
|
'(add-to-list 'helm-completing-read-handlers-alist '(org-capture . nil)))
|
|
|
|
|
|
2017-07-18 18:14:58 +02:00
|
|
|
|
(setq org-capture-use-agenda-date nil)
|
|
|
|
|
|
2017-07-16 18:07:00 +02:00
|
|
|
|
(setq org-capture-templates
|
2017-07-18 18:14:58 +02:00
|
|
|
|
`(("t" "Todo"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* TODO %^{What}\n"
|
|
|
|
|
"SCHEDULED: %(org-insert-time-stamp (org-read-date nil t \"+0d\"))\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
2017-07-31 13:21:19 +02:00
|
|
|
|
"%?"))
|
2017-07-18 18:14:58 +02:00
|
|
|
|
("n" "Note"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
entry
|
|
|
|
|
(file+olp db/org-default-notes-file "Notes")
|
2017-07-27 13:08:05 +02:00
|
|
|
|
"* %^{About} :NOTE:\n%T\n%?"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
:clock-in t :clock-resume t)
|
2017-07-18 18:14:58 +02:00
|
|
|
|
("d" "Date"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
2017-08-14 20:36:18 +02:00
|
|
|
|
"* GOTO %^{What} :DATE:\n%^{When}t\n%a")
|
2017-07-18 18:14:53 +02:00
|
|
|
|
("i" "Interruptions")
|
|
|
|
|
("in" "Interruption now"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
2017-07-27 13:08:05 +02:00
|
|
|
|
"* DONE %^{What}\n\n%?"
|
2017-07-18 18:14:53 +02:00
|
|
|
|
:clock-in t :clock-resume t)
|
|
|
|
|
("ip" "Interruption previously" ; bad English vs mnemonics
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* DONE %^{What}\n"
|
|
|
|
|
":LOGBOOK:\n"
|
|
|
|
|
"%(db/read-clockline)\n" ; evaluated before above prompt?
|
2017-07-27 13:08:05 +02:00
|
|
|
|
":END:\n"
|
2017-07-18 18:14:53 +02:00
|
|
|
|
"%?"))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
("j" "journal entry"
|
|
|
|
|
plain
|
|
|
|
|
(file+datetree db/org-default-pensieve-file)
|
2017-07-22 19:16:18 +02:00
|
|
|
|
"\n%i%U\n\n%?\n")
|
2017-07-16 18:07:00 +02:00
|
|
|
|
("r" "respond"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* TODO E-Mail: %:subject (%:from) :EMAIL:\n"
|
|
|
|
|
"SCHEDULED: %^{Reply when?}t\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
|
|
|
|
"\n%a")
|
|
|
|
|
:immediate-finish t)
|
|
|
|
|
("R" "read"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* READ %:subject :READ:\n"
|
|
|
|
|
;; "DEADLINE: <%(org-read-date nil nil \"+1m\")>\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
2017-08-14 20:36:18 +02:00
|
|
|
|
"\n%a"))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
("U" "Read current content of clipboard"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* READ %^{Description} :READ:\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
2017-08-14 20:36:18 +02:00
|
|
|
|
"\n%(current-kill 0)"))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
("m" "Meeting"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* MEETING with %^{With}: %^{What} :MEETING:\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
|
|
|
|
"\n%?")
|
|
|
|
|
:clock-in t :clock-resume t)
|
|
|
|
|
("p" "Phone call"
|
|
|
|
|
entry
|
|
|
|
|
(file db/org-default-refile-file)
|
|
|
|
|
,(concat "* PHONE %^{Calling} :PHONE:\n"
|
|
|
|
|
":PROPERTIES:\n:CREATED: %U\n:END:\n"
|
|
|
|
|
"\n%?")
|
|
|
|
|
:clock-in t :clock-resume t)
|
|
|
|
|
("w" "Weekly Summary"
|
|
|
|
|
entry
|
|
|
|
|
(file+datetree db/org-default-pensieve-file)
|
|
|
|
|
"* Weekly Review\n\n%?")))
|
|
|
|
|
|
2017-07-18 18:14:53 +02:00
|
|
|
|
(defun db/org-timestamp-difference (stamp-1 stamp-2)
|
|
|
|
|
"Returns time difference between two given org-mode timestamps."
|
|
|
|
|
;; Things copied from `org-clock-update-time-maybe’
|
|
|
|
|
(let* ((s (-
|
|
|
|
|
(float-time
|
|
|
|
|
(apply #'encode-time (org-parse-time-string stamp-2 nil t)))
|
|
|
|
|
(float-time
|
|
|
|
|
(apply #'encode-time (org-parse-time-string stamp-1 nil t)))))
|
|
|
|
|
(neg (< s 0))
|
|
|
|
|
(s (abs s))
|
|
|
|
|
(h (floor (/ s 3600)))
|
|
|
|
|
(m (floor (/ (- s (* 3600 h)) 60))))
|
|
|
|
|
(format (if neg "-%d:%02d" "%2d:%02d") h m)))
|
|
|
|
|
|
|
|
|
|
(defun db/read-clockline ()
|
|
|
|
|
"Read starting and ending time from user and return org mode
|
|
|
|
|
clock line."
|
|
|
|
|
(let* ((now (format-time-string "%H:%M"))
|
|
|
|
|
(starting (format "[%s]" (org-read-date t nil nil "Started: "
|
|
|
|
|
(current-time)
|
|
|
|
|
now)))
|
|
|
|
|
(ending (format "[%s]" (org-read-date t nil nil "Ended: "
|
|
|
|
|
(current-time)
|
|
|
|
|
now)))
|
|
|
|
|
(difference (db/org-timestamp-difference starting ending)))
|
|
|
|
|
(format "CLOCK: %s--%s => %s" starting ending difference)))
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
;; (add-to-list 'display-buffer-alist
|
|
|
|
|
;; `(,(rx bos "*Capture*" eos)
|
|
|
|
|
;; (display-buffer-in-side-display . window-buffer-reuse-window)
|
|
|
|
|
;; (reusable-frames . visible)
|
|
|
|
|
;; (side . bottom)
|
|
|
|
|
;; (window-height . 0.4)))
|
|
|
|
|
|
|
|
|
|
;; TODO: bind pop-to-buffer-same-window to pop-to-buffer in
|
|
|
|
|
;; org-capture-place-template
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Refiling
|
|
|
|
|
|
|
|
|
|
;; Refiling targets include this file and any file contributing to the agenda,
|
|
|
|
|
;; up to 9 levels deep
|
|
|
|
|
(setq org-refile-targets '((org-agenda-files . (:maxlevel . 9))))
|
|
|
|
|
|
|
|
|
|
;; Use full outline paths for refile targets
|
|
|
|
|
(setq org-refile-use-outline-path 'file)
|
|
|
|
|
|
|
|
|
|
(setq org-refile-allow-creating-parent-nodes 'confirm)
|
|
|
|
|
(setq org-indirect-buffer-display 'current-window)
|
|
|
|
|
(setq org-outline-path-complete-in-steps nil)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Babel
|
|
|
|
|
|
|
|
|
|
(setq org-structure-template-alist
|
|
|
|
|
'(("s" "#+begin_src ?\n\n#+end_src" "<src lang=\"?\">\n\n</src>")
|
|
|
|
|
("e" "#+begin_example\n?\n#+end_example" "<example>\n?\n</example>")
|
|
|
|
|
("q" "#+begin_quote\n?\n#+end_quote" "<quote>\n?\n</quote>")
|
|
|
|
|
("Q" "#+begin_equation\n?\n#+end_equation"
|
|
|
|
|
"<equation>\n?\n</equation>")
|
|
|
|
|
("v" "#+begin_verse\n?\n#+end_verse" "<verse>\n?\n</verse>")
|
|
|
|
|
("V" "#+begin_verbatim\n?\n#+end_verbatim" "<verbatim>\n?\n</verbatim>")
|
|
|
|
|
("c" "#+begin_center\n?\n#+end_center" "<center>\n?\n</center>")
|
|
|
|
|
("l" "#+begin_latex\n?\n#+end_latex"
|
|
|
|
|
"<literal style=\"latex\">\n?\n</literal>")
|
|
|
|
|
("l" "#+latex: " "<literal style=\"latex\">?</literal>")
|
|
|
|
|
("h" "#+begin_html\n?\n#+end_html"
|
|
|
|
|
"<literal style=\"html\">\n?\n</literal>")
|
|
|
|
|
("h" "#+html: " "<literal style=\"html\">?</literal>")
|
|
|
|
|
("a" "#+begin_ascii\n?\n#+end_ascii" "")
|
|
|
|
|
("a" "#+ascii: " "")
|
|
|
|
|
("i" "#+index: ?" "#+index: ?")
|
|
|
|
|
("i" "#+include: %file ?"
|
|
|
|
|
"<include file=%file markup=\"?\">")))
|
|
|
|
|
|
|
|
|
|
(setq org-src-fontify-natively t
|
|
|
|
|
org-src-preserve-indentation t)
|
|
|
|
|
|
|
|
|
|
|
2017-08-12 11:25:04 +02:00
|
|
|
|
;;; Reset checklists
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
2017-08-12 11:25:04 +02:00
|
|
|
|
;; from `org-checklist’ by James TD Smith (@ ahktenzero (. mohorovi cc)),
|
|
|
|
|
;; version: 1.0
|
|
|
|
|
|
|
|
|
|
(defun org-reset-checkbox-state-maybe ()
|
|
|
|
|
"Reset all checkboxes in an entry if the `RESET_CHECK_BOXES' property is set"
|
|
|
|
|
(interactive "*")
|
|
|
|
|
(if (org-entry-get (point) "RESET_CHECK_BOXES")
|
|
|
|
|
(org-reset-checkbox-state-subtree)))
|
|
|
|
|
|
|
|
|
|
(add-hook 'org-after-todo-state-change-hook 'org-reset-checkbox-state-maybe)
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Calendar
|
|
|
|
|
|
|
|
|
|
(use-package ox-icalendar
|
|
|
|
|
:commands (org-icalendar-combine-agenda-files)
|
|
|
|
|
:config (progn
|
|
|
|
|
(setq org-icalendar-include-body nil
|
|
|
|
|
org-icalendar-store-UID t
|
|
|
|
|
org-icalendar-use-deadline nil
|
|
|
|
|
org-icalendar-use-scheduled nil
|
|
|
|
|
org-icalendar-include-todo nil
|
|
|
|
|
org-icalendar-exclude-tags '("NO_EXPORT")
|
|
|
|
|
org-icalendar-combined-agenda-file "~/Public/daniel.ics")))
|
|
|
|
|
|
|
|
|
|
(defun db/export-diary ()
|
|
|
|
|
"Export diary.org as ics file to ~/Public."
|
|
|
|
|
(interactive)
|
|
|
|
|
(require 'ox-icalendar)
|
|
|
|
|
(org-save-all-org-buffers)
|
|
|
|
|
(let ((org-agenda-files (list db/org-default-home-file
|
|
|
|
|
db/org-default-work-file))
|
|
|
|
|
(org-agenda-new-buffers nil))
|
|
|
|
|
;; check whether we need to do something
|
|
|
|
|
(when (some (lambda (org-file)
|
|
|
|
|
(file-newer-than-file-p org-file
|
|
|
|
|
org-icalendar-combined-agenda-file))
|
|
|
|
|
org-agenda-files)
|
|
|
|
|
(message "Exporting diary ...")
|
|
|
|
|
;; open files manually to avoid polluting `org-agenda-new-buffers’; we don’t
|
|
|
|
|
;; want these buffers to be closed after exporting
|
|
|
|
|
(mapc #'find-file-noselect org-agenda-files)
|
|
|
|
|
;; actual export; calls `org-release-buffers’ and may thus close buffers
|
|
|
|
|
;; we want to keep around … which is why we set `org-agenda-new-buffers’
|
|
|
|
|
;; to nil
|
|
|
|
|
(org-icalendar-combine-agenda-files)
|
|
|
|
|
(message "Exporting diary ... done."))))
|
|
|
|
|
|
|
|
|
|
(defun db/ical-to-org (ical-file-name org-file-name category filetags)
|
|
|
|
|
"Convert ICAL-FILE-NAME to ORG-FILE-NAME using ical2org.
|
|
|
|
|
|
|
|
|
|
CATEGORY and FILETAGS specify the category and the filetags of
|
|
|
|
|
the resulting org mode file, respectively."
|
|
|
|
|
(when (string-match "^https?://" ical-file-name)
|
|
|
|
|
(let ((tmp-file (make-temp-file "/tmp/emacs-ical-")))
|
|
|
|
|
(url-copy-file ical-file-name tmp-file t)
|
|
|
|
|
(setq ical-file-name tmp-file)))
|
2017-08-11 09:07:19 +02:00
|
|
|
|
(unless (zerop (call-process "ical2org"
|
2017-07-16 18:07:00 +02:00
|
|
|
|
ical-file-name
|
|
|
|
|
`(:file ,org-file-name)
|
|
|
|
|
nil
|
|
|
|
|
"-c" category
|
|
|
|
|
"-f" filetags))
|
|
|
|
|
(error (concat "Error in converting ical file «%s» into org file;"
|
|
|
|
|
" see «%s» for more information")
|
|
|
|
|
ical-file-name org-file-name)))
|
|
|
|
|
|
|
|
|
|
(defvar db/ical-org-links nil
|
|
|
|
|
"List of ical-file-names and their linked org mode files.")
|
|
|
|
|
|
|
|
|
|
(defun db/add-ical-org-link (ical-file-name org-file-name category filetags)
|
|
|
|
|
"Add a new link from ICAL-FILE-NAME to ORG-FILE-NAME.
|
|
|
|
|
|
|
|
|
|
Resulting org mode file will have CATEGORY and FILETAGS set."
|
|
|
|
|
(cl-pushnew (list ical-file-name org-file-name category filetags)
|
|
|
|
|
db/ical-org-links
|
|
|
|
|
:test #'equal))
|
|
|
|
|
|
|
|
|
|
(defun db/update-ical-org-files ()
|
|
|
|
|
"Update all org mode files that are linked to some iCal file."
|
|
|
|
|
(interactive)
|
|
|
|
|
(dolist (entry db/ical-org-links)
|
|
|
|
|
(cl-destructuring-bind (url target category filetags) entry
|
|
|
|
|
(condition-case ex
|
|
|
|
|
(progn
|
|
|
|
|
;; update TARGET
|
|
|
|
|
(db/ical-to-org url target category filetags)
|
|
|
|
|
;; revert buffers visiting TARGET
|
|
|
|
|
(with-current-buffer (find-buffer-visiting target)
|
|
|
|
|
(let ((revert-without-query (list (regexp-quote (file-truename target)))))
|
|
|
|
|
(revert-buffer))))
|
|
|
|
|
(error
|
|
|
|
|
(warn "Error in generating %s: %s" target (cadr ex)))))))
|
|
|
|
|
|
|
|
|
|
(defun db/sync-all-iCal ()
|
|
|
|
|
"Sync all iCal sources."
|
|
|
|
|
(interactive)
|
|
|
|
|
(db/export-diary)
|
|
|
|
|
(db/update-ical-org-files))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Fixes
|
|
|
|
|
|
|
|
|
|
(defun endless/org-ispell ()
|
|
|
|
|
"Configure `ispell-skip-region-alist' for `org-mode'."
|
|
|
|
|
(make-local-variable 'ispell-skip-region-alist)
|
|
|
|
|
(add-to-list 'ispell-skip-region-alist '(org-property-drawer-re))
|
|
|
|
|
(add-to-list 'ispell-skip-region-alist '("~" "~"))
|
|
|
|
|
(add-to-list 'ispell-skip-region-alist '("=" "="))
|
|
|
|
|
(add-to-list 'ispell-skip-region-alist '("^#\\+BEGIN_SRC" . "^#\\+END_SRC")))
|
|
|
|
|
|
|
|
|
|
(add-hook 'org-mode-hook #'endless/org-ispell)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Exporting
|
|
|
|
|
|
|
|
|
|
(eval-after-load 'ox
|
|
|
|
|
'(progn
|
|
|
|
|
(add-to-list 'org-latex-classes
|
|
|
|
|
'("scrartcl" "\\documentclass[11pt]{scrartcl}\n\\usepackage{babel}\n"
|
|
|
|
|
("\\section{%s}" . "\\section*{%s}")
|
|
|
|
|
("\\subsection{%s}" . "\\subsection*{%s}")
|
|
|
|
|
("\\subsubsection{%s}" . "\\subsubsection*{%s}")
|
|
|
|
|
("\\paragraph{%s}" . "\\paragraph*{%s}")
|
|
|
|
|
("\\subparagraph{%s}" . "\\subparagraph*{%s}")))
|
|
|
|
|
|
|
|
|
|
(require 'ox-md)))
|
|
|
|
|
|
2017-08-06 13:56:14 +02:00
|
|
|
|
|
|
|
|
|
;;; Hydra
|
|
|
|
|
|
|
|
|
|
(require 'hydra)
|
|
|
|
|
|
|
|
|
|
(defhydra hydra-org-clock (:color blue)
|
|
|
|
|
"
|
|
|
|
|
Current Task: %`org-clock-current-task; "
|
|
|
|
|
("w" (lambda ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(clock-in-task-by-id org-working-task-id)))
|
|
|
|
|
("h" (lambda ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(clock-in-task-by-id org-home-task-id)))
|
|
|
|
|
("b" (lambda ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(clock-in-task-by-id org-break-task-id)))
|
|
|
|
|
("i" (lambda ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(org-clock-in '(4))))
|
|
|
|
|
("a" counsel-org-goto-all)
|
|
|
|
|
("o" org-clock-out)
|
|
|
|
|
("l" db/org-clock-in-last-task)
|
|
|
|
|
("p" db/play-playlist)
|
|
|
|
|
("d" (lambda ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(when (org-clock-is-active)
|
|
|
|
|
(save-window-excursion
|
|
|
|
|
(org-clock-goto)
|
|
|
|
|
(let ((org-inhibit-logging 'note))
|
|
|
|
|
(org-todo 'done)
|
|
|
|
|
(org-save-all-org-buffers)))))))
|
|
|
|
|
|
2017-07-16 18:07:00 +02:00
|
|
|
|
|
|
|
|
|
;;; End
|
|
|
|
|
|
|
|
|
|
(provide 'db-org)
|
|
|
|
|
|
|
|
|
|
;;; db-org.el ends here
|