This repository has been archived by the owner on Sep 8, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
org-collection.el
521 lines (457 loc) · 20.9 KB
/
org-collection.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
;;; org-collection.el --- Less global Org setup -*- lexical-binding: t; sentence-end-double-space: t -*-
;; Copyright © 2020 Gustav Wikström <[email protected]>
;; Author: Gustav Wikström <[email protected]>
;; URL: https://github.com/whil-/org-collection
;; Keywords: org-mode, collection, project
;; Version: 0.2
;; Package-Requires: ((emacs "27.1") (org "9.3"))
;; This file is NOT part of GNU Emacs.
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 3, or (at your option)
;; any later version.
;;
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;;
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to the
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
;; Boston, MA 02110-1301, USA.
;;; Commentary:
;;
;; This library aims at making Org mode things local to folders and
;; collections of Org mode documents. It provides a global-minor-mode
;; `org-collection-mode' that tracks which collection is active, if
;; any. And it provides commands to switch between collection.
;; Switching between collections is manual and
;; `org-collection-register' is the entry. When registered
;; collections can be enabled with `org-collection-load'.
;;
;; A bunch of auxiliary functions are provided as well to simplify
;; working with collections. Like listing files and going to their
;; roots.
;;
;;; Code:
;;;; Dependencies
(require 'org)
(require 'org-id)
(require 'map)
;;;; Customizations
(defconst org-collection-file ".org-collection"
"Filename for a collection.
To reside in the directory root of a collection.")
(defgroup org-collection nil
"Options concerning configuring Org collection."
:tag "Org Collection"
:group 'org)
(defcustom org-collection-list-file (locate-user-emacs-file "org-collections")
"File in which to save the list of known collections."
:type 'file)
(defcustom org-collection-default-id-locations-file ".org-id-locations"
"Default customization of `org-id-locations-file' within a collection.
The path is relative to the location of the collection unless
declared as an absolut path."
:group 'org-collection
:type 'file)
;(defcustom org-collection-default-id-locations-file-relative t
; "Override value of `org-id-locations-file-relative' for Org collection.
;For collections to be portable everything should be relative to
;the collection path by default. This customization sets that
;default. If set to nil, use value from
;`org-id-locations-file-relative' instead."
; :group 'org-collection
; :type 'boolean)
(defcustom org-collection-variables '((":path:" . (lambda (dir) dir)))
"List of variables to be used in an org-collection-file.
The variables are literal values that will be replaced with the
output from the associated function before the configuration is
read and evaluated.
This is configured using an alist with keys as the literals to
replace and functions for determining the value that is to be put
instead. The functions will be called with one argument: the
directory of the collection."
:group 'org-collection
:type '(alist :key-type string :value-type: function
:tag "Alist of variable and function"))
(defcustom org-collection-files-extension "org"
"The extension for org files in a collection."
:group 'org-collection
:type '(string))
(defcustom org-collection-scan-directories-recursively t
"Determines if subdirectories are included in the collection.
Decide if subdirectories inside `org-collection' are considered
part of the collection or not."
:group 'org-collection
:type '(boolean))
;;;; Management variables
(defvar org-collection-mode)
(defvar org-collection nil
"Variable used for determining the active org-collection.
Wen set, `org-collection' will be a collection object, as
returned from `org-collection--try-get-collection'.")
(defvar org-collection-active nil
"Variable used for determining if a collection is active.")
(defvar org-collection-list nil
"Variable used to cache the content of `org-collection-file'.
Used mainly to reduce filesystem access.")
(defvar org-collection-defaults-plist nil
"Plist of modified variables and their defaults.
This variable is used internally to keep track of changes that
has been made, to be able to reset them when the mode is
disabled.")
;;;; Functions working with collections
;;;;; Internals
(defun org-collection--set (collection)
"Configure customization for the loaded Org collection.
Set some Org mode properties for a COLLECTION, for it to work well
with the rest of Org mode, for things not relying on buffer local
configurations."
(let* ((location (plist-get collection ':location))
(customization-alist (plist-get collection ':customization))
(require-alist (plist-get collection ':require))
(id-file (expand-file-name org-collection-default-id-locations-file
location)))
;; Some things will always be set, no matter what the collection contains.
;; Otherwise org-collection will not work correctly
(org-collection--set-variables `((org-directory ,location)
(org-id-locations-file ,id-file)
(org-id-track-globally t)
(org-id-locations-file-relative t)))
(dolist (package require-alist)
(condition-case nil
(unless (featurep package)
(require package))
(file-missing (warn "Could not load %s" (symbol-name package)))))
(org-collection--set-variables customization-alist)
;; Ignore the error message that org-id-locations-load may
;; display if a global org-id file is missing. It's noise at this location.
;; TODO this message inhibitation doesn't seem to work... Maybe fix someday.
(let ((inhibit-message t))
(org-id-locations-load))
(setq org-collection collection)))
(defun org-collection--unset ()
"Unset Org collection custiomization.
This will reset Org mode customizations to their default, as set
before the org-collection was loaded.
Some variables are expected to be reset elsewhere. Notably
`org-collection-defaults-plist' which is traversed and emptied in
`org-collection--unset-variables'."
(org-collection--unset-variables)
;; Ignore the error message that org-id-locations-load may
;; display if a global org-id file is missing. It's noise at this location.
(let ((inhibit-message t))
(org-id-locations-load))
(setq org-collection nil))
(defun org-collection--set-variables (property-alist)
"Set values for org collection.
If a key in PROPERTY-ALIST match a predefined key within this
function, the value of that property is set for the key."
(dolist (property property-alist)
(let* ((symbol (car property))
(value (cadr property))
(name (symbol-name symbol)))
(when (and (or (string-prefix-p "org-" name)
(string-prefix-p "ol-" name)
(string-prefix-p "ox-" name))
(custom-variable-p symbol)
(not (equal (default-value symbol) value)))
(setq org-collection-defaults-plist
(plist-put org-collection-defaults-plist symbol (default-value symbol)))
(set-default symbol value)))))
(defun org-collection--unset-variables ()
"Reset properties.
Goes through plist `org-collection-defaults-plist' and (re)sets symbols
to their default value."
(while org-collection-defaults-plist
(let ((symbol (pop org-collection-defaults-plist))
(value (pop org-collection-defaults-plist)))
(set-default symbol value))))
(defun org-collection--try-load-list-file (&optional force)
"If the collection list have not been loaded from file, load it.
When FORCE is non-nil reload file even if already loaded."
(when (and (or force (not org-collection-list))
(file-readable-p org-collection-list-file))
(with-temp-buffer
(insert-file-contents org-collection-list-file)
(condition-case-unless-debug nil
(let ((read-circle nil))
(setq org-collection-list (read (current-buffer))))
(end-of-file nil)))))
(defun org-collection--maybe-update-list (collection)
"Update `org-collection-list' if needed."
(let* ((name (plist-get collection ':name))
(location (plist-get collection ':location))
(location-existing (lax-plist-get org-collection-list name)))
(cond ((and (stringp location-existing)
(stringp location)
(not (equal (expand-file-name location-existing)
(expand-file-name location))))
;; Might add some user-interaction here later... Right now
;; just overwrite existing with new.
(setq org-collection-list
(lax-plist-put org-collection-list name location))
(org-collection--try-persist org-collection-list))
((not location-existing)
(setq org-collection-list
(lax-plist-put org-collection-list name location))
(org-collection--try-persist org-collection-list)))))
(defun org-collection--try-persist (c-list)
"Persist collection if possible.
If collection is not listed in `org-collection-list-file'
already, persist it there."
(when org-collection-list-file
(save-excursion
(condition-case nil
(progn
(set-buffer (find-file-noselect org-collection-list-file t))
(erase-buffer)
(insert ";; Org collection list file.
;; This contains a plist of known org collections with collection
;; `:name' as key and filesystem location as value.\n")
(pp c-list (current-buffer))
(insert "\n")
(condition-case nil
(save-buffer 0)
(error
(message "File %s could not be saved."
org-collection-list-file)))
(kill-buffer (current-buffer)))
(error
(message "File %s could not be read." org-collection-list-file))))))
(defun org-collection--lax-plist-delete (plist property)
"Delete PROPERTY from PLIST with equal as comparison operator.
This is in contrast to merely setting it to 0. Based on `org-plist-delete'."
(let (p)
(while plist
(if (not (equal property (car plist)))
(setq p (plist-put p (car plist) (nth 1 plist))))
(setq plist (cddr plist)))
p))
(defun org-collection--try-get-collection (dir)
"Return an Org collection given a directory, if it exists and works.
If `org-collection' already is set this is returned if the
location of that collection matches `default-directory'.
If `org-collection-scan-directories-recursively' is not nil then
a collection can also be returned for paths closer to the root of
the filesystem-tree. The deepest path takes precedence."
(when-let* ((c-dir (if org-collection-scan-directories-recursively
(locate-dominating-file dir org-collection-file)
dir))
(c-file (expand-file-name org-collection-file c-dir)))
(let ((org-collection-location (plist-get org-collection ':location))
collection)
(cond ((and org-collection-location
(equal (expand-file-name org-collection-location)
(expand-file-name c-dir)))
org-collection)
((file-readable-p c-file)
(with-temp-buffer
(insert-file-contents c-file)
(dolist (config org-collection-variables)
(let ((from-str (car config))
(to-str (funcall (cdr config) c-dir)))
(save-excursion
(while (search-forward from-str nil t)
(replace-match to-str)))))
(condition-case-unless-debug nil
(let ((read-circle nil))
(setq collection (cdr (read (current-buffer)))))
(end-of-file nil)))
(when collection
(plist-put collection :location c-dir)))))))
(defun org-collection-files (&optional relative collection)
"Get all org files.
If RELATIVE is t, then return relative paths and remove file
extension. Uses `org-collection' if `collection' is nil.
Ignores dotfiles."
(let ((path (plist-get (or collection org-collection) ':location)))
(if relative
(mapcar #'org-collection-path-entry-name (org-collection-files))
(if org-collection-scan-directories-recursively
(directory-files-recursively
path (format "^[^.].*\\.%s$" org-collection-files-extension))
(directory-files
path t (format "^[^.].*\\.%s$" org-collection-files-extension))))))
(defun org-collection-update-id-locations ()
"Scan `org-collection-files' using `org-id-update-id-locations'."
(interactive)
(org-id-update-id-locations (org-collection-files)))
(defun org-collection-path-entry-name (path &optional collection)
"Get PATH as an entry name."
(let ((collection-path (plist-get (or collection org-collection) ':location)))
(string-remove-suffix (concat "." org-collection-files-extension)
(file-relative-name (expand-file-name path)
(expand-file-name collection-path)))))
;;;;; Interactive stuff
(defun org-collection-create (dir name)
"Create a template configuration file and register it."
(interactive "DCreate collection at location: \nsName of collection: ")
(let ((settings-file (expand-file-name org-collection-file dir)))
;; Either warn that something's wrong, or execute!
(cond
;; 1. No existing collection with that name exist
((lax-plist-get org-collection-list name)
(message "Collection name already registered."))
;; 2. No existing ocllection exist at that location since before
((file-exists-p settings-file)
(message "Collection settings-file already exist."))
(t (let ((settings-buffer (find-file-noselect settings-file)))
(with-current-buffer settings-buffer
(insert (format "; -*- mode: emacs-lisp -*-
;; template org-collection
(org-collection-definition
:name \"%s\"
:require (org)
:customization
((org-agenda-files (\":path:\"))))"
name))
(save-buffer)
(emacs-lisp-mode))
(org-collection-register dir)
(switch-to-buffer settings-buffer))))))
(defun org-collection-goto (collection-name)
"Goto a collection."
(interactive (list (org-completing-read "Goto collection: " (map-keys org-collection-list))))
(let ((dir (lax-plist-get org-collection-list collection-name)))
(find-file dir)))
(defun org-collection-visit-home ()
"Goto the home of a collection, if it exist."
(interactive)
(if-let* ((file (plist-get org-collection ':home))
(base-path (plist-get org-collection ':location))
(file-fullname (expand-file-name file base-path)))
(find-file file-fullname)
(message "No :home is set in current collection.")))
(defun org-collection-visit-file (collection-file-no-extension)
"Visit a file in a collection.
`collection-file-no-extension' is a path relative to the current
collection without file-extension."
(interactive (list (org-completing-read "Visit: " (org-collection-files t))))
(when-let* ((base-path (plist-get org-collection ':location))
(file-fullname (expand-file-name (format "%s.%s" collection-file-no-extension
org-collection-files-extension)
base-path)))
(find-file file-fullname)))
(defun org-collection-register (directory)
"Opens a collection and registers it for easier future use."
(interactive "Dorg-track directory: ")
(let ((collection (org-collection--try-get-collection directory)))
(cond (collection
(org-collection--maybe-update-list collection)
(org-collection-load (plist-get collection ':name)))
(t
(error
(message "No collection found at given location. Could not load or register"))))))
(defun org-collection-unregister (collection-name)
"Removes a collection from the persisted list of collections."
(interactive (progn
(org-collection--try-load-list-file 'force)
(list (org-completing-read "Unregister collection: " (map-keys org-collection-list)))))
(setq org-collection-list
(org-collection--lax-plist-delete org-collection-list collection-name))
(org-collection--try-persist org-collection-list))
(defun org-collection-visit-settings (collection-name)
"Open collection settings file."
(interactive (list (org-completing-read "Visit settings for collection: " (map-keys org-collection-list))))
(let ((dir (lax-plist-get org-collection-list collection-name)))
(find-file (expand-file-name org-collection-file dir))))
(defun org-collection-load (collection-name)
"Load customizations for a collection."
(interactive (list (org-completing-read "Load collection: " (map-keys org-collection-list))))
(if (not org-collection-mode)
(message "Cannot load a collection unless org-collection-mode is turned on.")
(when org-collection-active
(org-collection-unload))
(let* ((dir (lax-plist-get org-collection-list collection-name))
(collection (org-collection--try-get-collection dir)))
;; Unset existing collection
(org-collection--unset)
;; Enforce the whole collection to be global
(org-collection--set collection)
;; Mark the loaded collection in the mode-line
(org-collection-update-mode-line t)
(setq org-collection-active t))))
(defun org-collection-unload ()
"Unload the loaded org collection."
(interactive)
(if (not org-collection-mode)
(message "Org collection mode is not active, nothing to unload.")
;; Unset existing globals
(org-collection--unset)
(setq org-collection-active nil)
(org-collection-update-mode-line)))
;;;; Keymaps
(defvar org-collection-mode-map
(let ((map (make-sparse-keymap))
(pmap (make-sparse-keymap)))
(define-key pmap "q" 'org-collection-mode)
(define-key pmap "g" 'org-collection-goto)
(define-key pmap "c" 'org-collection-create)
(define-key pmap "r" 'org-collection-register)
(define-key pmap "R" 'org-collection-unregister)
(define-key pmap "l" 'org-collection-load)
(define-key pmap "L" 'org-collection-unload)
(define-key pmap "s" 'org-collection-visit-settings)
(define-key pmap "f" 'org-collection-visit-file)
(define-key pmap "h" 'org-collection-visit-home)
;; bind our submap into map
(define-key map "\C-cz" pmap)
map)
"Keymap used in org collection global minor mode.")
;;;; Mode line stuff
(defcustom org-collection--mode-line-prefix " OC"
"Mode line lighter prefix for org collection."
:group 'org-collection
:type 'string)
(defvar org-collection--mode-line org-collection--mode-line-prefix
"String displayed in the mode line when Org collection global
mode is turned on.")
(defun org-collection-mode-line (collection)
"Report collection name in the modeline."
(let* ((name (plist-get collection ':name)))
(format "%s%s"
org-collection--mode-line-prefix
(if name (format ":%s" name) ""))))
(defun org-collection-update-mode-line (&optional loaded)
"Set `org-collection--mode-line'.
When optional parameter LOADED is not nil, the loaded collection name
is added to the mode-line."
(let ((mode-line (org-collection-mode-line org-collection)))
(cond (loaded
(setq org-collection--mode-line mode-line)
;; Also set the default value, but make sure it's reset
;; when unloaded or the mode is disabled.
(setq org-collection-defaults-plist
(plist-put org-collection-defaults-plist
'org-collection--mode-line
org-collection--mode-line-prefix))
(setq-default org-collection--mode-line mode-line))
(t
(setq org-collection--mode-line mode-line)))
(force-mode-line-update)
mode-line))
;;;; Minor mode declarations
;;;###autoload
(define-minor-mode org-collection-mode
"Comment."
:init-value nil
:lighter org-collection--mode-line
:keymap org-collection-mode-map
:global t
:version "27.1"
(cond (org-collection-mode
;; Mode was turned on.
;; Try to load the list file before enabling the event. Saves
;; one unwind-protect!
(org-collection--try-load-list-file))
(t
;; Unset after hook is removed. Saves one unwind-protect!
(org-collection--unset)
(setq org-collection-list nil))))
;;;; Connect to emacs project and module provisioning
(add-hook 'project-find-functions #'org-collection--try-get-collection)
(provide 'org-collection)
;;; org-collection.el ends here