-
Notifications
You must be signed in to change notification settings - Fork 4
/
org-brain-export.el
220 lines (191 loc) · 8.67 KB
/
org-brain-export.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
;;; org-brain-export.el --- Export org-brain to other formats -*- lexical-binding: t; -*-
;; Copyright (C) 2020 Erik Sjöstrand
;; MIT License
;; Author: Erik Sjöstrand <[email protected]>
;; URL: http://github.com/Kungsgeten/org-brain-export
;; Keywords: outlines hypermedia
;; Package-Requires: ((emacs "25.1") (org-brain "0.9") (a "0.1.1") (xmlgen "0.5"))
;; Version: 0.1
;;; Commentary:
;; Package for exporting `org-brain' data to other formats.
;;; Code:
(require 'org-brain)
(require 'dash)
(require 'a)
(require 'xmlgen)
;;; General
(defvar org-brain-export-directory (expand-file-name "export" org-brain-path)
"Directory where exported files will end up.")
(defun org-brain-export--relation-data (from-entry to-entry)
"Get relationship data FROM-ENTRY TO-ENTRY.
TO-ENTRY is the main entry, while TO-ENTRY is the relationship entry."
(a-list :id (org-brain-entry-identifier to-entry)
:annotation (org-brain-get-edge-annotation from-entry to-entry)
:title (org-brain-title to-entry)))
(defun org-brain-export-generate-data (entry)
"Generate data representation of `org-brain' ENTRY.
Represented as an alist."
(a-list
:id (org-brain-entry-identifier entry)
:type (if (org-brain-filep entry) 'file 'headline)
:title (org-brain-title entry)
:text (org-brain-text entry)
:children (mapcar (lambda (child)
(org-brain-export--relation-data entry child))
(org-brain-children entry))
:parents (mapcar (lambda (parent)
(org-brain-export--relation-data entry parent))
(org-brain-parents entry))
:friends (mapcar (lambda (friend)
(org-brain-export--relation-data entry friend))
(org-brain-friends entry))))
;;; GraphViz
(defvar org-brain-export-dot-file (expand-file-name "brain.dot" org-brain-export-directory)
"Where the output of `org-brain-export-dot' goes.")
(defun org-brain-export--dot-id (ob-data)
"Convert id of OB-DATA into a dot readable format."
(concat "ob"
(replace-regexp-in-string
"[-/]" "" (alist-get :id ob-data))))
(defun org-brain-export--dot-node-def (ob-data)
"Get node entry line (a string) of OB-DATA."
(format "%s [label=\"%s\"];\n"
(org-brain-export--dot-id ob-data)
(alist-get :title ob-data)))
(defun org-brain-export--dot-children (ob-data &optional pool)
"Get children relations in OB-DATA as a dot string.
Only get children who's :id is a member of POOL.
If POOL is nil or omitted, get all children."
(mapconcat
(lambda (child)
(if (and pool (member (alist-get :id child) pool))
(format "%s -> %s;\n"
(org-brain-export--dot-id ob-data)
(org-brain-export--dot-id child))
""))
(alist-get :children ob-data)
""))
(defun org-brain-export--dot-friends (ob-data &optional pool)
"Get friend relations in OB-DATA as a dot string.
Only get friends who's :id is a member of POOL.
If POOL is nil or omitted, get all friends."
(mapconcat
(lambda (friend)
(if (and pool (member (alist-get :id friend) pool))
(format "%s -> %s [dir=none color=\"blue\"];\n"
(org-brain-export--dot-id ob-data)
(org-brain-export--dot-id friend))
""))
(alist-get :friends ob-data)
""))
(defun org-brain-export-dot-save (entries file)
"Export ENTRIES to GraphViz dot FILE.
Removes duplicates from ENTRIES."
(make-directory (file-name-directory file) t)
(let* ((data (mapcar #'org-brain-export-generate-data (-distinct entries)))
(pool (mapcar (lambda (x) (alist-get :id x)) data)))
(with-temp-file file
(insert "digraph G {\nconcentrate=true;\n")
(dolist (ob-data data)
(insert (org-brain-export--dot-node-def ob-data)
(org-brain-export--dot-children ob-data pool)
(org-brain-export--dot-friends ob-data pool)))
(insert "}"))))
(defun org-brain-export-dot (file)
"Export your `org-brain' to Graphviz dot FILE."
(interactive "F")
(message "Starting dot export...")
(org-brain-export-dot-save (append (org-brain-files t)
(org-brain-headline-entries))
file)
(message "Dot export finished!"))
(defun org-brain-export-dot-entry (entry file &optional parent-max-level children-max-level exclude-siblings)
"Export `org-brain' ENTRY to GraphViz dot FILE.
If run interactively, get ENTRY from context and prompt for FILE.
PARENT-MAX-LEVEL and CHILDREN-MAX-LEVEL determines how many
levels (grand)parents/children that'll be included.
If EXCLUDE-SIBLINGS is non-nil, the siblings of ENTRY will not be included.
If run interactively in `org-brain-visualize', PARENT-MAX-LEVEL
and CHILDREN-MAX-LEVEL respects the ancestors and decendants of
the current entry in `org-brain-visualize-mind-map' mode."
(interactive (list (org-brain-entry-at-pt)
(read-file-name "Output file: ")
org-brain-mind-map-parent-level
org-brain-mind-map-child-level))
(message "Starting dot export...")
(let ((entries (append (list entry) (org-brain-friends entry))))
(org-brain-recursive-parents entry (or parent-max-level 1)
(lambda (x) (push x entries)))
(org-brain-recursive-children entry (or children-max-level 2)
(lambda (x) (push x entries)))
(unless exclude-siblings
(dolist (parent (org-brain-siblings entry))
(dolist (sibling (cdr parent))
(push sibling entries))))
(org-brain-export-dot-save entries file))
(message "Dot export finished!"))
;;; HTML
(defvar org-brain-export-html-file (expand-file-name "brain.html" org-brain-export-directory)
"Where the output of `org-brain-export-html' goes.")
(defun org-brain-export--html-relation-list-item (relation)
"Generate the HTML <li> to RELATION."
`(li ,@(remove nil (list
`(a :href ,(concat "#" (alist-get :id relation)) ,(alist-get :title relation))
(when-let ((annotation (alist-get :annotation relation)))
'(span :class "ob-edge" annotation))))))
(defun org-brain-export--html-parents (ob-data)
"Generate HTML representation of parents in OB-DATA."
(when-let ((parents (alist-get :parents ob-data)))
`(section :class "ob-parents"
(h2 "Parents")
(ul :class "ob-parent-list"
,@(mapcar #'org-brain-export--html-relation-list-item parents)))))
(defun org-brain-export--html-friends (ob-data)
"Generate HTML representation of parents in OB-DATA."
(when-let ((friends (alist-get :friends ob-data)))
`(section :class "ob-friends"
(h2 "Friends")
(ul :class "ob-friend-list"
,@(mapcar #'org-brain-export--html-relation-list-item friends)))))
(defun org-brain-export--html-children (ob-data)
"Generate HTML representation of children in OB-DATA."
(when-let ((children (alist-get :children ob-data)))
`(section :class "ob-children"
(h2 "Children")
(ul :class "ob-children-list"
,@(mapcar #'org-brain-export--html-relation-list-item children)))))
(defun org-brain-export--generate-html (ob-data)
"Generate HTML representation of OB-DATA."
`(article :class "ob-entry" :id ,(alist-get :id ob-data)
,@(remove nil (list
`(h1 ,(alist-get :title ob-data))
(org-brain-export--html-parents ob-data)
(org-brain-export--html-children ob-data)
(org-brain-export--html-friends ob-data)
;; The %s here is sent to format during
;; org-brain-export-html, since raw HTML doesn't play well
;; with xmlgen
`(section :class "ob-entry-text" "%s")))))
(defun org-brain-export-html ()
"Export your `org-brain' to HTML."
(interactive)
(make-directory (file-name-directory org-brain-export-html-file) t)
(message "Starting HTML export...")
(let ((data-rep (mapcar #'org-brain-export-generate-data
(append (org-brain-files t)
(org-brain-headline-entries)))))
(with-temp-file org-brain-export-html-file
(insert
(apply #'format
(xmlgen
`(html
(head)
(body
,@(mapcar #'org-brain-export--generate-html data-rep))))
(mapcar (lambda (x)
(or (ignore-errors (org-export-string-as (alist-get :text x) 'html t))
"<i>Error during parsing of entry text...</i>"))
data-rep)))))
(message "HTML export finished!"))
(provide 'org-brain-export)
;;; org-brain-export.el ends here