/dendroam.el Secret
Last active
April 25, 2021 10:27
Implementing dendron-like hierarchies in org-roam
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(defvar org-roam-utils-capture-templates '()) | |
;; This is for correct title formatting | |
(setq org-roam-capture-templates | |
'(("d" "default" plain | |
"%?" | |
:if-new (file+head "${slug}.org" | |
"#+title: ${hierarchy-title}\n") | |
:immediate-finish t | |
:unnarrowed t))) | |
;;This allows dailies to integrate with hierarchies | |
(setq org-roam-dailies-capture-templates | |
'(("d" "default" entry | |
"* %?" | |
:if-new (file+head "journal.daily.%<%Y.%m.%d>.org" | |
"#+title: %<%Y-%m-%d>\n")))) | |
;; This templates are used to populate time notes, scratch notes | |
;; an other functions that you can create and are useful for you. | |
(setq org-roam-utils-capture-templates | |
'(("o" "OKRs" entry | |
"* %?" | |
:if-new (file+head "journal.okr.%<%Y.%m.%d>.org" | |
"#+title: %<%Y-%m-%d>\n")) | |
("p" "PPP" entry | |
"* %?" | |
:if-new (file+head "journal.ppp.%<%Y.%m.%V>.org" | |
"#+title: %<%Y-%m-%d>\n")) | |
("t" "Time note" entry | |
"* %?" | |
:if-new (file+head "${current-file}.%<%Y.%m.%d.%.%M%S%3N>.org" | |
"#+title: %<%Y-%m-%d>\n")) | |
("s" "Scratch note" entry | |
"* %?" | |
:if-new (file+head "scratch.%<%Y.%m.%d.%.%M%S%3N>.org" | |
"#+title: %<%M%S%3N>\n")) | |
)) | |
;;Node custom getters | |
(cl-defmethod org-roam-node-current-file (node) | |
"Gets node file-name-base by file name" | |
(file-name-base (org-roam-node-file node))) | |
(cl-defmethod org-roam-node-hierarchy-title (node) | |
"Gets node title excluding the hierarchy and capitalize it" | |
(capitalize | |
(car | |
(last | |
(split-string | |
(org-roam-node-title node) | |
"\\."))))) | |
(defun org-roam-format-hierarchy (file) | |
"Formats node's path, to get the hierarchy whithout the title | |
where title will be the last child of the hierarchy: | |
from the filename this.is.a.hierarchy.note-title.org | |
returns this.is.a.hierarchy" | |
(let* ((base-name (file-name-base file)) | |
(hierarchy-no-title (file-name-base base-name))) | |
hierarchy-no-title)) | |
(cl-defmethod org-roam-node-hierarchy (node) | |
"Gets node hierarchy by file name" | |
(funcall 'org-roam-format-hierarchy (org-roam-node-file node))) | |
(cl-defmethod org-roam-node-current-file (node) | |
(file-name-base (buffer-file-name))) | |
;; Refactor functions | |
(defun org-roam-fetch-same-hierarchy-files (hierarchy) | |
"Gets all the nodes that share the same HIERARCHY totally or parcially" | |
(let ((files | |
(mapcar #'car (org-roam-db-query [:select [file] | |
:from nodes | |
:where (like file $r1)] | |
(concat "%" hierarchy "%"))))) | |
files)) | |
(defun org-roam-refactor-hierarchy (&optional current) | |
"Prompts the user to change the hierarchy of the current file node | |
and updates its hierarchy and the hierarchy of all the nodes that have it" | |
(interactive) | |
(let* | |
((initial-file | |
(file-name-nondirectory (buffer-file-name))) | |
(initial-slug | |
(file-name-base initial-file)) | |
(new-slug (file-name-base | |
(read-string "Refactor: " initial-slug))) | |
(initial-slug-no-title | |
(file-name-base initial-slug)) | |
(files-to-upd (if current | |
`(,initial-file) | |
(vic/org-roam-fetch-same-hierarchy-files | |
initial-slug-no-title)))) | |
(dolist (file files-to-upd) | |
(let ((new-file | |
(replace-regexp-in-string initial-slug-no-title new-slug file))) | |
(rename-file file new-file) | |
(if (equal buffer-file-name file) | |
(progn | |
(kill-current-buffer) | |
(find-file new-file))))))) | |
(defun org-roam-refactor-file () | |
(interactive) | |
(let* ((initial-file (buffer-file-name)) | |
(initial-slug (file-name-base initial-file)) | |
(new-slug (read-string "Refactor: " initial-slug)) | |
(new-file (concat | |
(expand-file-name new-slug org-roam-directory) | |
".org"))) | |
(rename-file initial-file new-file) | |
(kill-current-buffer) | |
(find-file new-file))) | |
;; Useful notes functions | |
(defun org-roam-insert-time-note(&optional goto) | |
"Creates a time note in the current level of the hierarchy. | |
Time notes have the format: current.Y.m.d.MS3N | |
The file is created using a template from `org-roam-utils-capture-templates'" | |
(interactive "P") | |
(org-roam-capture- :goto (when goto '(4)) | |
:node (org-roam-node-create) | |
:templates org-roam-utils-capture-templates | |
:keys "t" | |
:props (list :default-time (current-time)))) | |
(defun org-roam-insert-scratch-note(&optional goto) | |
"Creates a time note in the current level of the hierarchy. | |
Time notes have the format: current.Y.m.d.MS3N | |
The file is created using a template from `org-roam-utils-capture-templates'" | |
(interactive "P") | |
(org-roam-capture- :goto (when goto '(4)) | |
:node (org-roam-node-create) | |
:templates org-roam-utils-capture-templates | |
:keys "s" | |
:props (list :default-time (current-time)))) | |
;; Org roam overrides to allow these features | |
(eval-after-load "org-roam" | |
'(cl-defmethod org-roam-node-slug ((node org-roam-node)) | |
"Override. Generates a dendron-like slug from *title* | |
this expects an input like: lang.elisp.what is nil | |
and will create a file wih the name: lang.elisp.what-is-nil" | |
(let ((title (org-roam-node-title node))) | |
(cl-flet* ((nonspacing-mark-p (char) | |
(memq char org-roam-slug-trim-chars)) | |
(strip-nonspacing-marks (s) | |
(ucs-normalize-NFC-string | |
(apply #'string (seq-remove #'nonspacing-mark-p | |
(ucs-normalize-NFD-string s))))) | |
(cl-replace (title pair) | |
(replace-regexp-in-string (car pair) (cdr pair) title))) | |
(let* ((pairs `(("[^[:alnum:][:digit:]_.]" . "-") ;; convert anything not alphanumeric except "." | |
(" " . "-") ;; remove whitespaces | |
("__*" . "-") ;; remove sequential underscores | |
("^_" . "") ;; remove starting underscore | |
("_$" . ""))) ;; remove ending underscore | |
(slug (-reduce-from #'cl-replace (strip-nonspacing-marks title) pairs))) | |
(downcase slug)))))) | |
;; Some notes functions that are useful for me | |
(defun org-roam-insert-ppp (&optional goto) | |
"Capture an entry in a daily-note for TIME, creating it if necessary. | |
When GOTO is non-nil, go the note without creating an entry." | |
(interactive "P") | |
(org-roam-capture- :goto (when goto '(4)) | |
:node (org-roam-node-create) | |
:templates org-roam-utils-capture-templates | |
:keys "p" | |
:props (list :default-time (current-time)))) | |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
This is not a package, is only tested as part of my doom emacs config