-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathpos-marker.lisp
399 lines (325 loc) · 12.8 KB
/
pos-marker.lisp
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
(in-package #:neomacs)
(sera:export-always
'(end-pos end-pos-p end-pos-node
text-pos text-pos-p text-pos-node text-pos-offset
node-after node-before node-containing
pos-left pos-right pos-next pos-prev
pos-up pos-down pos-down-last
pos-left-until pos-right-until pos-next-until pos-prev-until
pos-up-until
pos-left-ensure pos-right-ensure pos-next-ensure pos-prev-ensure
pos-up-ensure
npos-left npos-right npos-next npos-prev
npos-left-until npos-right-until npos-next-until npos-prev-until
npos-left-ensure npos-right-ensure npos-next-ensure npos-prev-ensure
marker with-marker pos advance-p))
;;; Positions
(eval-always
(defstruct (end-pos (:constructor end-pos (node)))
(node (alex:required-argument 'node) :type element))
(defstruct (text-pos (:constructor text-pos (node offset)))
(node (alex:required-argument 'node) :type text-node)
(offset (alex:required-argument 'offset) :type (integer 0))))
(deftype pos ()
'(or element text-pos end-pos null))
(defmethod host ((pos end-pos))
(host (end-pos-node pos)))
(defmethod host ((pos text-pos))
(host (text-pos-node pos)))
(defun resolve-marker (marker-or-pos)
(if (typep marker-or-pos 'marker)
(pos marker-or-pos)
marker-or-pos))
(defun node-after (marker-or-pos)
"Return the node after MARKER-OR-POS."
(let ((pos (resolve-marker marker-or-pos)))
(ematch pos
((element) pos)
((end-pos) nil)
((text-pos node offset) (aref (text node) offset))
(nil))))
(defun node-before (marker-or-pos)
"Return the node before MARKER-OR-POS."
(let ((pos (resolve-marker marker-or-pos)))
(labels ((handle-text-node (node)
(if (text-node-p node)
(last-elt (text node))
node)))
(ematch pos
((element) (handle-text-node (previous-sibling pos)))
((end-pos node) (handle-text-node (last-child node)))
((text-pos node offset)
(if (> offset 0)
(aref (text node) (1- offset))
(handle-text-node (previous-sibling node))))
(nil)))))
(defun node-containing (marker-or-pos)
"Return the node containing MARKER-OR-POS."
(let ((pos (resolve-marker marker-or-pos)))
(ematch pos
((element) (parent pos))
((end-pos node) node)
((text-pos node) (parent node))
(nil))))
(defun normalize-node-pos (node direction)
"Convert NODE to `text-pos' if necessary.
If DIRECTION is nil, convert to leftmost position;
otherwise, convert to rightmost position."
(if (text-node-p node)
(text-pos node (if direction (1- (length (text node))) 0))
node))
(defun pos-right (pos &key destructive)
"Return the position to the right of POS.
If DESTRUCTIVE is non-nil, POS might be mutated."
(setq pos (resolve-marker pos))
(labels ((node-right (node)
(or (normalize-node-pos (next-sibling node) nil)
(end-pos (parent node)))))
(ematch pos
((element) (node-right pos))
((end-pos) nil)
((text-pos node offset)
(if (< (1+ offset) (length (text node)))
(if destructive
(progn
(incf (text-pos-offset pos))
pos)
(text-pos node (1+ offset)))
(node-right node)))
(nil))))
(defun pos-left (pos &key destructive)
"Return the position to the left of POS.
If DESTRUCTIVE is non-nil, POS might be mutated."
(setq pos (resolve-marker pos))
(labels ((node-left (node)
(normalize-node-pos (previous-sibling node) t)))
(ematch pos
((element) (node-left pos))
((end-pos node)
(normalize-node-pos (last-child node) t))
((text-pos node offset)
(if (> offset 0)
(if destructive
(progn
(decf (text-pos-offset pos))
pos)
(text-pos node (1- offset)))
(node-left node)))
(nil))))
(defun pos-up (pos)
"Return the parent position of POS."
(setq pos (resolve-marker pos))
(when pos
(let ((node (ematch pos
((element) (parent pos))
((end-pos node) node)
((text-pos node) (parent node)))))
;; Avoid selecting document root
(when (parent node)
node))))
(defun pos-down (pos)
"Return the first child position of POS."
(setq pos (resolve-marker pos))
(etypecase pos
(element
(or (normalize-node-pos (first-child pos) nil)
(end-pos pos)))
((or text-pos end-pos) nil)
(null)))
(defun pos-down-last (pos)
"Return the last child position of POS."
(setq pos (resolve-marker pos))
(etypecase pos
(element (end-pos pos))
((or text-pos end-pos) nil)
(null)))
(defun pos-next (pos &key destructive)
"Return the next position of POS in preorder traversal.
If DESTRUCTIVE is non-nil, POS might be mutated."
(setq pos (resolve-marker pos))
(or (pos-down pos)
(pos-right pos :destructive destructive)
(when-let (up (pos-up pos))
(pos-right up))))
(defun pos-prev (pos &key destructive)
"Return the previous position of POS in preorder traversal.
If DESTRUCTIVE is non-nil, POS might be mutated."
(setq pos (resolve-marker pos))
(if-let (left (pos-left pos :destructive destructive))
(or (pos-down-last left) left)
(pos-up pos)))
(declaim (inline iterate-pos-until))
(defun iterate-pos-until (function pos predicate &rest args)
(iter (setq pos (apply function pos args))
(while pos)
(until (funcall predicate pos)))
pos)
(defun pos-next-until (pos predicate &key destructive)
"Return the first position after POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-until #'pos-next pos predicate :destructive destructive))
(defun pos-prev-until (pos predicate &key destructive)
"Return the last position before POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-until #'pos-prev pos predicate :destructive destructive))
(defun pos-right-until (pos predicate &key destructive)
"Return the first sibling after POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-until #'pos-right pos predicate :destructive destructive))
(defun pos-left-until (pos predicate &key destructive)
"Return the last sibling before POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-until #'pos-left pos predicate :destructive destructive))
(defun pos-up-until (pos predicate)
"Return the first parent of POS satisfying PREDICATE."
(iterate-pos-until #'pos-up pos predicate))
(declaim (inline iterate-pos-ensure))
(defun iterate-pos-ensure (function pos predicate &rest args)
(setq pos (resolve-marker pos))
(iter (until (funcall predicate pos))
(setq pos (apply function pos args))
(while pos))
pos)
(defun pos-next-ensure (pos predicate &key destructive)
"Return POS or the first position after POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-ensure #'pos-next pos predicate :destructive destructive))
(defun pos-prev-ensure (pos predicate &key destructive)
"Return POS or the last position before POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-ensure #'pos-prev pos predicate :destructive destructive))
(defun pos-right-ensure (pos predicate &key destructive)
"Return POS or the first sibling after POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-ensure #'pos-right pos predicate :destructive destructive))
(defun pos-left-ensure (pos predicate &key destructive)
"Return POS or the last sibling before POS satisfying PREDICATE.
If DESTRUCTIVE is non-nil, POS might be mutated."
(iterate-pos-ensure #'pos-left pos predicate :destructive destructive))
(defun pos-up-ensure (pos predicate)
"Return POS or the first parent of POS satisfying PREDICATE."
(iterate-pos-ensure #'pos-up pos predicate))
(macrolet ((define-npos (symbol &rest args)
(let ((npos (alex:symbolicate "N" symbol)))
`(progn
(declaim (inline ,npos))
(defun ,npos (,@args)
,(let ((*print-case* :downcase))
(format nil "Destructive version of `~a'."
symbol))
(,symbol ,@args :destructive t))))))
(define-npos pos-left pos)
(define-npos pos-right pos)
(define-npos pos-next pos)
(define-npos pos-prev pos)
(define-npos pos-next-until pos predicate)
(define-npos pos-prev-until pos predicate)
(define-npos pos-left-until pos predicate)
(define-npos pos-right-until pos predicate)
(define-npos pos-next-ensure pos predicate)
(define-npos pos-prev-ensure pos predicate)
(define-npos pos-left-ensure pos predicate)
(define-npos pos-right-ensure pos predicate))
;;; Marker
(eval-always
(defstruct (%start-pos (:constructor %start-pos (node)))
(node (alex:required-argument 'node) :type element))
(defstruct (%after-pos (:constructor %after-pos (before)))
(before (alex:required-argument 'before) :type pos))
(defclass marker ()
((pos :initarg :pos :type pos))))
(defmethod host ((pos %start-pos))
(host (%start-pos-node pos)))
(defmethod host ((pos %after-pos))
(host (%after-pos-before pos)))
(defmethod host ((m marker))
(host (slot-value m 'pos)))
(defun copy-marker (marker)
(make-instance 'marker :pos (slot-value marker 'pos)))
(defmethod initialize-instance :after ((m marker) &key)
(push m (markers (host (slot-value m 'pos)))))
(defun advance-p (marker)
"Returns t if MARKER advances, nil otherwise.
This place is setf-able."
(etypecase (slot-value marker 'pos)
((or element end-pos text-pos) t)
((or %start-pos %after-pos) nil)))
(defun (setf advance-p) (new-val marker)
(check-type new-val boolean)
(unless (eql new-val (advance-p marker))
(setf (slot-value marker 'pos)
(pos-to-advance-p (pos marker) new-val)))
new-val)
(defun pos-to-advance-p (pos advance-p)
(if advance-p pos
(if-let (left (npos-left pos))
(%after-pos left)
(%start-pos (node-containing pos)))))
(defun copy-pos (marker-or-pos)
(etypecase marker-or-pos
(marker (pos marker-or-pos))
(element marker-or-pos)
(end-pos (copy-end-pos marker-or-pos))
(text-pos (copy-text-pos marker-or-pos))))
(defmethod pos ((m marker))
(let ((pos (slot-value m 'pos)))
(ematch pos
((or (element) (end-pos) (text-pos))
(copy-pos pos))
((%start-pos node)
(or (normalize-node-pos (first-child node) nil)
(end-pos node)))
((%after-pos before) (pos-right before)))))
(defun focus-marker-p (marker)
"Test if MARKER is the focus marker of some buffer.
Return that buffer or nil otherwise."
(let ((host (host marker)))
(when (eq marker (focus-marker host))
host)))
(defvar *atomic-motion-markers* nil)
(defun call-with-atomic-motion (marker thunk)
(if (member marker *atomic-motion-markers*)
(funcall thunk)
(let ((saved (copy-marker marker))
success)
(unwind-protect
(multiple-value-prog1
(let ((*atomic-motion-markers*
(cons marker *atomic-motion-markers*)))
(funcall thunk))
(setq success t))
(unless success
(setf (slot-value marker 'pos) (slot-value saved 'pos)))
(delete-marker saved)))))
(defmacro with-atomic-motion (marker &body body)
`(call-with-atomic-motion ,marker (lambda () ,@body)))
(defmethod (setf pos) (new-val (m marker))
(setq new-val (copy-pos new-val))
(setf (slot-value m 'pos) (pos-to-advance-p new-val (advance-p m)))
new-val)
(defmethod print-object ((marker marker) stream)
(print-unreadable-object (marker stream :type t :identity t)
(format stream "~a" (slot-value marker 'pos))))
(defun delete-marker (marker)
(alex:deletef (markers (host marker)) marker))
(defmacro with-marker ((marker marker-or-pos &optional (advance-p t))
&body body)
"Make a temporary marker at MARKER-OR-POS and bind to MARKER during BODY."
`(let ((,marker (make-instance 'marker
:pos (resolve-marker ,marker-or-pos))))
(unwind-protect
(progn
(setf (advance-p ,marker) ,advance-p)
,@body)
(delete-marker ,marker))))
(defmacro with-advance-p ((marker advance-p) &body body)
"Temporarily set MARKER's ADVANCE-P during BODY."
`(let ((saved-advance-p (advance-p ,marker)))
(unwind-protect
(progn (setf (advance-p ,marker) ,advance-p)
,@body)
(setf (advance-p ,marker) saved-advance-p))))
(defvar *current-buffer* nil)
(defmacro with-current-buffer (buffer &body body)
"Run BODY with BUFFER as the current buffer."
`(call-with-current-buffer ,buffer (lambda () ,@body)))