7a261b149de84c5aebbc42a745a400af4f6b50ee
[lice.git] / intervals.lisp
blob7a261b149de84c5aebbc42a745a400af4f6b50ee
1 ;;; implentation of an interval tree
3 (in-package :lice)
5 (defvar *text-property-default-nonsticky* nil
6 "Alist of properties vs the corresponding non-stickinesses.
7 Each element has the form (PROPERTY . NONSTICKINESS).
9 If a character in a buffer has PROPERTY, new text inserted adjacent to
10 the character doesn't inherit PROPERTY if NONSTICKINESS is non-nil,
11 inherits it if NONSTICKINESS is nil. The front-sticky and
12 rear-nonsticky properties of the character overrides NONSTICKINESS.")
14 (defvar *char-property-alias-alist* nil
15 "Alist of alternative properties for properties without a value.
16 Each element should look like (PROPERTY ALTERNATIVE1 ALTERNATIVE2...).
17 If a piece of text has no direct value for a particular property, then
18 this alist is consulted. If that property appears in the alist, then
19 the first non-nil value from the associated alternative properties is
20 returned.")
22 (defvar *default-text-properties* nil
23 "Property-list used as default values.
24 The value of a property in this list is seen as the value for every
25 character that does not have its own value for that property.")
27 (defmacro doplist ((sym val plist &optional ret) &body body)
28 "Loop through each symbol value pair in PLIST executing BODY."
29 (let ((csym (gensym)))
30 `(do* ((,csym ,plist (cddr ,csym)))
31 ((endp ,csym) ,ret)
32 (let ((,sym (car ,csym))
33 (,val (cadr ,csym)))
34 ,@body))))
36 ;; interval node is a list: (key left right &rest plist)
38 (defun print-interval (i s d)
39 (declare (ignore d))
40 (format s "#S(interval ~s ~s ~s | ~s ~s)"
41 (interval-pt i)
42 (interval-length i)
43 (interval-plist i)
44 (interval-left i)
45 (interval-right i)))
47 (defstruct (interval
48 (:print-function print-interval))
49 (pt nil)
50 (length nil)
51 (left nil)
52 (right nil)
53 (parent nil :type (or null pstring buffer interval))
54 (plist nil :type list))
56 ;; MOVITZ's defstruct doesn't create copy-interval
57 #+movitz
58 (defun copy-interval (interval)
59 (make-interval :pt (interval-pt interval)
60 :length (interval-length interval)
61 :left (interval-left interval)
62 :right (interval-right interval)
63 :parent (interval-parent interval)
64 :plist (interval-plist interval)))
66 (defun interval-has-object (interval)
67 (and (interval-parent interval)
68 (not (typep (interval-parent interval) 'interval))))
70 (defun interval-has-parent (interval)
71 (and (interval-parent interval)
72 (typep (interval-parent interval) 'interval)))
74 (defun check-total-length (interval)
75 (when (< (interval-length interval) 0)
76 (error "Interval length < 0 ~a" interval)))
78 (defun create-root-interval (object)
79 "Return a fresh interval for OBJECT."
80 (let ((i (make-interval :pt 0 :length 0 :parent object
81 :left nil :right nil :plist nil)))
82 (cond
83 ((typep object 'buffer)
84 (setf (intervals object) i
85 ;; XXX: are buffer-max buffer-min the right functions to use?
86 (interval-length i) (- (buffer-max object) (buffer-min object))
87 (interval-pt i) (buffer-min object)))
88 ((typep object 'pstring)
89 (setf (intervals object) i
90 (interval-length i) (pstring-length object))))
91 i))
93 (defun plists-equal (p1 p2)
94 "Return 1 if the two properties lists are equal, 0 otherwise."
95 (when (= (length p1)
96 (length p2))
97 (doplist (sym val p1 t)
98 (unless (eql val (getf p2 sym))
99 (return-from plists-equal nil)))))
101 (defun intervals-equal (i1 i2)
102 "Return T if the two intervals have the same properties, NIL otherwise."
103 (plists-equal (interval-plist i1) (interval-plist i2)))
105 (defun lookup-char-property (plist prop textprop)
106 (let* ((val (getf plist prop))
107 (cat (getf plist 'category)))
108 (when val
109 (return-from lookup-char-property val))
110 ;; This is what GNU emacs does...
111 (when (and (symbolp cat)
112 (get cat prop))
113 (return-from lookup-char-property (get cat prop)))
114 ;; Check for alternative properties
115 (let ((tail (assoc prop *char-property-alias-alist*)))
116 (when tail
117 (or (find-if (lambda (p)
118 (getf plist p))
119 (cdr tail))
120 (and textprop
121 (consp *default-text-properties*)
122 (getf *default-text-properties* prop)))))))
124 (defun textget (plist sym)
125 "Get the value of property PROP from PLIST,
126 which is the plist of an interval.
127 We check for direct properties, for categories with property PROP,
128 and for PROP appearing on the default-text-properties list."
129 (lookup-char-property plist sym t))
131 (defun split-interval-left (interval offset)
132 (let* ((new-length offset)
133 (new (make-interval :pt (interval-pt interval)
134 :length offset
135 :parent interval)))
136 (incf (interval-pt interval) offset)
137 (if (interval-left interval)
138 (progn
139 (setf (interval-left new) (interval-left interval)
140 (interval-parent (interval-left new)) new
141 (interval-left interval) new
142 (interval-length new) (+ new-length (interval-length (interval-left new))))
143 (check-total-length new)
144 (balance-an-interval new))
145 (progn
146 (setf (interval-left interval) new
147 (interval-length new) new-length)
148 (check-total-length new)))
149 (balance-possible-root-interval interval)
150 new))
152 (defun split-interval-right (interval offset)
153 (let* ((position (interval-pt interval))
154 (new-length (- (interval-text-length interval) offset))
155 (new (make-interval :pt (+ position offset)
156 :length 0
157 :parent interval)))
158 (setf (interval-parent new) interval)
159 (if (interval-right interval)
160 (progn
161 (setf (interval-right new) (interval-right interval)
162 (interval-parent (interval-right interval)) new
163 (interval-right interval) new
164 (interval-length new) (+ new-length (interval-length (interval-right new))))
165 (check-total-length new)
166 (balance-an-interval new))
167 (progn
168 (setf (interval-right interval) new
169 (interval-length new) new-length)
170 (check-total-length new)))
171 (balance-possible-root-interval interval)
172 new))
174 (defun right-child-p (root)
175 (eq root (interval-right (interval-parent root))))
177 (defun left-child-p (root)
178 (eq root (interval-left (interval-parent root))))
180 (defun interval-past-top-p (interval)
181 "Return t if INTERVAL is not an interval. Used to check when we've
182 climbed past the root interval."
183 (not (typep (interval-parent interval) 'interval)))
185 (defun rotate-right (interval)
186 "Assuming that a left child exists, perform the following operation:
189 / \ / \
190 B => A
191 / \ / \
194 (let ((old-total (interval-length interval))
195 (b (interval-left interval))
197 ;; Change interval's parent to point b.
198 (unless (root-interval-p interval)
199 (if (left-child-p interval)
200 (setf (interval-left (interval-parent interval)) b)
201 (setf (interval-right (interval-parent interval)) b)))
202 (setf (interval-parent b) (interval-parent interval))
203 ;; Make b the parent of a
204 (setf i (interval-right b)
205 (interval-right b) interval
206 (interval-parent interval) b)
207 ;; make a point to c
208 (setf (interval-left interval) i)
209 (when i
210 (setf (interval-parent i) interval))
211 ;; A's total length is decreased by the length of B and its left child.
212 (decf (interval-length interval) (- (interval-length b)
213 (left-total-length interval)))
214 (check-total-length interval)
215 ;; B must have the same total length of A.
216 (setf (interval-length b) old-total)
217 (check-total-length b)
220 (defun rotate-left (interval)
221 "Assuming that a right child exists, perform the following operation:
224 / \ / \
225 B => A
226 / \ / \
229 (let ((old-total (interval-length interval))
230 (b (interval-right interval))
232 ;; Change interval's parent to point b.
233 (unless (root-interval-p interval)
234 (if (left-child-p interval)
235 (setf (interval-left (interval-parent interval)) b)
236 (setf (interval-right (interval-parent interval)) b)))
237 (setf (interval-parent b) (interval-parent interval))
238 ;; Make b the parent of a
239 (setf i (interval-left b)
240 (interval-left b) interval
241 (interval-parent interval) b)
242 ;; make a point to c
243 (setf (interval-right interval) i)
244 (when i
245 (setf (interval-parent i) interval))
246 ;; A's total length is decreased by the length of B and its left child.
247 (decf (interval-length interval) (- (interval-length b)
248 (right-total-length interval)))
249 (check-total-length interval)
250 ;; B must have the same total length of A.
251 (setf (interval-length b) old-total)
252 (check-total-length b)
255 (defun total-length (root)
256 "TOTAL_LENGTH"
257 (if root
258 (interval-length root)
261 (defun left-total-length (root)
262 (if (interval-left root)
263 (interval-length (interval-left root))
266 (defun right-total-length (root)
267 (if (interval-right root)
268 (interval-length (interval-right root))
271 (defun interval-text-length (root)
272 "The size of text represented by this interval alone. LENGTH."
273 (if root
274 (- (total-length root)
275 (total-length (interval-right root))
276 (total-length (interval-left root)))
279 (defun balance-an-interval (i)
280 (let (old-diff
281 new-diff)
282 (loop
283 (setf old-diff (- (left-total-length i) (right-total-length i)))
284 (cond ((> old-diff 0)
285 ;; Since the left child is longer, there must be one.
286 (setf new-diff (+ (- (interval-length i)
287 (interval-length (interval-left i)))
288 (- (right-total-length (interval-left i))
289 (left-total-length (interval-left i)))))
290 (when (>= (abs new-diff) old-diff)
291 (return-from balance-an-interval i))
292 (setf i (rotate-right i))
293 (balance-an-interval (interval-right i)))
294 ((< old-diff 0)
295 (setf new-diff (+ (- (interval-length i)
296 (interval-length (interval-right i)))
297 (- (left-total-length (interval-right i))
298 (right-total-length (interval-right i)))))
299 (when (>= (abs new-diff) (- old-diff))
300 (return-from balance-an-interval i))
301 (setf i (rotate-left i))
302 (balance-an-interval (interval-left i)))
303 (t (return-from balance-an-interval i))))))
305 (defun balance-possible-root-interval (interval)
306 (let ((has-parent nil)
307 parent)
308 (when (null (interval-parent interval))
309 (return-from balance-possible-root-interval interval))
310 (when (interval-has-object interval)
311 (setf parent (interval-parent interval)
312 has-parent t))
313 (setf interval (balance-intervals interval))
314 (when has-parent
315 (setf (intervals parent) interval))
316 interval))
318 (defun balance-intervals (tree)
319 "Balance the interval tree TREE. Balancing is by weight: the amount
320 of text."
321 (labels ((balance (tree)
322 (when (interval-left tree)
323 (balance (interval-left tree)))
324 (when (interval-right tree)
325 (balance (interval-right tree)))
326 (balance-an-interval tree)))
327 (when tree
328 (balance tree))))
330 (defun find-interval (tree position)
331 (let ((relative-position position))
332 (when (null tree)
333 (return-from find-interval nil))
334 (assert (<= relative-position (total-length tree)))
335 (balance-possible-root-interval tree)
336 (loop
337 (cond ((< relative-position (left-total-length tree))
338 (setf tree (interval-left tree)))
339 ((and (interval-right tree)
340 (>= relative-position (- (total-length tree)
341 (right-total-length tree))))
342 (decf relative-position (- (total-length tree)
343 (right-total-length tree)))
344 (setf tree (interval-right tree)))
346 (setf (interval-pt tree) (+ (- position relative-position)
347 (left-total-length tree)))
348 (return-from find-interval tree))))))
350 (defun next-interval (interval)
351 (unless (null interval)
352 (let ((i interval)
353 (next-position (+ (interval-pt interval)
354 (interval-text-length interval))))
355 (when (interval-right interval)
356 (setf i (interval-right i))
357 (while (interval-left i)
358 (setf i (interval-left i)))
359 (setf (interval-pt i) next-position)
360 (return-from next-interval i))
361 (loop until (interval-past-top-p i)
362 if (left-child-p i)
363 do (progn
364 (setf i (interval-parent i)
365 (interval-pt i) next-position)
366 (return-from next-interval i))
367 do (setf i (interval-parent i))))))
369 (defun previous-interval (interval)
370 (unless (null interval)
371 (let ((i interval))
372 (when (interval-left interval)
373 (setf i (interval-left i))
374 (while (interval-right i)
375 (setf i (interval-right i)))
376 (setf (interval-pt i) (- (interval-pt interval)
377 (interval-text-length i)))
378 (return-from previous-interval i))
379 (loop until (interval-past-top-p i)
380 if (right-child-p i)
381 do (progn
382 (setf i (interval-parent i)
383 (interval-pt i) (- (interval-pt interval)
384 (interval-text-length i)))
385 (return-from previous-interval i))
386 do (setf i (interval-parent i))))))
388 (defun merge-interval-right (i)
389 (let ((absorb (interval-text-length i))
390 successor)
391 (decf (interval-length i) absorb)
392 (check-total-length i)
393 (when (interval-right i)
394 (setf successor (interval-right i))
395 (while (interval-left successor)
396 (incf (interval-length successor) absorb)
397 (check-total-length successor)
398 (setf successor (interval-left successor)))
399 (incf (interval-length successor) absorb)
400 (check-total-length successor)
401 (delete-interval i)
402 (return-from merge-interval-right successor))
403 (setf successor i)
404 (while (interval-parent successor)
405 (if (left-child-p successor)
406 (progn
407 (setf successor (interval-parent successor))
408 (delete-interval i)
409 (return-from merge-interval-right successor))
410 (progn
411 (setf successor (interval-parent successor))
412 (decf (interval-length successor) absorb)
413 (check-total-length successor))))
414 (error "merge-interval-right: gak")))
416 (defun merge-interval-left (i)
417 (let ((absorb (interval-text-length i))
418 predecessor)
419 (decf (interval-length i) absorb)
420 (check-total-length i)
421 (when (interval-left i)
422 (setf predecessor (interval-left i))
423 (while (interval-right predecessor)
424 (incf (interval-length predecessor) absorb)
425 (check-total-length predecessor)
426 (setf predecessor (interval-right predecessor)))
427 (incf (interval-length predecessor) absorb)
428 (check-total-length predecessor)
429 (delete-interval i)
430 (return-from merge-interval-left predecessor))
431 (setf predecessor i)
432 (while (interval-parent predecessor)
433 (when (interval-right predecessor)
434 (setf predecessor (interval-parent predecessor))
435 (delete-interval i)
436 (return-from merge-interval-left predecessor))
437 (setf predecessor (interval-parent predecessor))
438 (decf (interval-length predecessor) absorb)
439 (check-total-length predecessor)
441 (error "merge-interval-left: gak")))
444 ;; adjust_intervals_for_insertion (tree, position, length)
446 (defun adjust-intervals-for-insertion (tree position length)
447 "Effect an adjustment corresponding to the addition of LENGTH characters
448 of text. Do this by finding the interval containing POSITION in the
449 interval tree TREE, and then adjusting all of its ancestors by adding
450 LENGTH to them.
452 If POSITION is the first character of an interval, meaning that point
453 is actually between the two intervals, make the new text belong to
454 the interval which is \"sticky\".
456 If both intervals are \"sticky\", then make them belong to the left-most
457 interval. Another possibility would be to create a new interval for
458 this text, and make it have the merged properties of both ends."
459 (let* ((parent (interval-parent tree))
460 (offset (if (typep parent 'buffer)
461 (buffer-min parent)
463 i temp eobp)
464 ;; If inserting at point-max of a buffer, that position will be out
465 ;; of range. Remember that buffer positions are 1-based.
466 (when (>= position (+ (total-length tree) offset))
467 (setf position (+ (total-length tree) offset)
468 eobp t))
469 (setf i (find-interval tree position))
470 ;; If in middle of an interval which is not sticky either way,
471 ;; we must not just give its properties to the insertion.
472 ;; So split this interval at the insertion point.
474 ;; Originally, the if condition here was this:
475 ;; (! (position == i->position || eobp)
476 ;; && END_NONSTICKY_P (i)
477 ;; && FRONT_NONSTICKY_P (i))
478 ;; But, these macros are now unreliable because of introduction of
479 ;; Vtext_property_default_nonsticky. So, we always check properties
480 ;; one by one if POSITION is in middle of an interval.
481 (unless (or (= position (interval-pt i))
482 eobp)
483 (let* ((rear (getf (interval-plist i) 'rear-nonsticky))
484 (front (getf (interval-plist i) 'front-sticky))
485 (problem t))
486 (when (or (and (not (consp rear)) rear)
487 (and (not (consp front)) front))
488 ;; All properties are nonsticky. We split the interval.
489 (setf problem nil))
491 ;; Does any actual property pose an actual problem? We break
492 ;; the loop if we find a nonsticky property.
493 (when problem
494 (setf problem (do* ((tail (interval-plist i) (cddr tail))
495 (prop (cdr tail) (cdr tail)))
496 ((or (endp tail)
497 (and (not (and (consp front)
498 (not (find prop front))))
499 (or (and (consp rear)
500 (not (find prop rear)))
501 (let ((tmp (assoc prop *text-property-default-nonsticky*)))
502 (and (consp tmp) tmp)))))
503 tail))))
504 ;; If any property is a real problem, split the interval.
505 (when problem
506 (setf temp (split-interval-right i (- position (interval-pt i))))
507 (copy-properties i temp)
508 (setf i temp))))
509 ;; If we are positioned between intervals, check the stickiness of
510 ;; both of them. We have to do this too, if we are at BEG or Z.
511 (if (or (= position (interval-pt i))
512 eobp)
513 (let ((prev (cond ((= position +beg+) nil)
514 (eobp i)
515 (t (previous-interval i)))))
516 (when eobp
517 (setf i nil))
518 ;; Even if we are positioned between intervals, we default
519 ;; to the left one if it exists. We extend it now and split
520 ;; off a part later, if stickiness demands it.
521 (do ((tmp (or prev i) (when (interval-has-parent tmp)
522 (interval-parent tmp))))
523 ((null tmp))
524 (incf (interval-length tmp) length)
525 ;; CHECK_TOTAL_LENGTH (temp);
526 (setf tmp (balance-possible-root-interval tmp)))
527 ;; If at least one interval has sticky properties, we check
528 ;; the stickiness property by property.
530 ;; Originally, the if condition here was this:
531 ;; (END_NONSTICKY_P (prev) || FRONT_STICKY_P (i))
532 ;; But, these macros are now unreliable because of introduction
533 ;; of Vtext_property_default_nonsticky. So, we always have to
534 ;; check stickiness of properties one by one. If cache of
535 ;; stickiness is implemented in the future, we may be able to
536 ;; use those macros again.
537 (let* ((pleft (when prev
538 (interval-plist prev)))
539 (pright (when i
540 (interval-plist i)))
541 (new-props (merge-properties-sticky pleft pright)))
542 (cond ((not prev)
543 ;; /* i.e. position == BEG */
544 (unless (plists-equal (interval-plist i) new-props)
545 (setf i (split-interval-left i length)
546 (interval-plist i) new-props)))
547 ((not (plists-equal (interval-plist prev) new-props))
548 (setf prev (split-interval-right prev (- position (interval-pt prev)))
549 (interval-plist prev) new-props)
550 (when (and i
551 (plists-equal (interval-plist prev) (interval-plist i)))
552 (merge-interval-right prev)))
553 ((and (not prev)
554 (not (null (interval-plist i))))
555 ;; Just split off a new interval at the left.
556 ;; Since I wasn't front-sticky, the empty plist is ok.
557 (setf i (split-interval-left i length))))))
558 ;; Otherwise just extend the interval.
559 (do ((tmp i (when (interval-has-parent tmp)
560 (interval-parent tmp))))
561 ((null tmp))
562 (incf (interval-length tmp) length)
563 ;; CHECK_TOTAL_LENGTH (temp);
564 (setf tmp (balance-possible-root-interval tmp))))
565 tree))
567 (defun interval-deletion-adjustment (tree from amount)
568 "Find the interval in TREE corresponding to the relative position
569 FROM and delete as much as possible of AMOUNT from that interval.
570 Return the amount actually deleted, and if the interval was
571 zeroed-out, delete that interval node from the tree.
573 Note that FROM is actually origin zero, aka relative to the
574 leftmost edge of tree. This is appropriate since we call ourselves
575 recursively on subtrees.
577 Do this by recursing down TREE to the interval in question, and
578 deleting the appropriate amount of text."
579 (let ((relative-position from))
580 (cond
581 ((null tree)
583 ;; Left branch
584 ((< relative-position (left-total-length tree))
585 (let ((subtract (interval-deletion-adjustment (interval-left tree)
586 relative-position
587 amount)))
588 (decf (interval-length tree) subtract)
589 ;; CHECK_TOTAL_LENGTH
590 subtract))
591 ;; Right branch
592 ((>= relative-position (- (total-length tree)
593 (right-total-length tree)))
594 (decf relative-position (- (interval-length tree)
595 (right-total-length tree)))
596 (let ((subtract (interval-deletion-adjustment (interval-right tree)
597 relative-position
598 amount)))
599 (decf (interval-length tree) subtract)
600 ;; CHECK_TOTAL_LENGTH
601 subtract))
602 ;; This node
604 ;; How much can we delete from this interval?
605 (let ((my-amount (- (interval-length tree)
606 (right-total-length tree)
607 relative-position)))
608 (when (> amount my-amount)
609 (setf amount my-amount))
610 (decf (interval-length tree) amount)
611 ;; CHECK_TOTAL_LENGTH
612 (when (zerop (total-length tree))
613 (delete-interval tree))
614 amount)))))
616 (defun adjust-intervals-for-deletion (buffer start length)
617 "Effect the adjustments necessary to the interval tree of BUFFER to
618 correspond to the deletion of LENGTH characters from that buffer
619 text. The deletion is effected at position START (which is a
620 buffer position, i.e. origin 1)."
621 (let ((left-to-delete length)
622 (tree (intervals buffer))
623 (offset (buffer-min buffer)))
624 (cond
625 ((null tree))
626 ((or (> start (+ offset (total-length tree)))
627 (> (+ start length) (+ offset (total-length tree))))
628 (error "gak ~a ~a ~a ~a" tree offset (total-length tree) length))
629 ((= length (total-length tree))
630 (setf (intervals buffer) nil))
631 ((only-interval-p tree)
632 (decf (interval-length tree) length)) ;; CHECK_TOTAL_LENGTH
634 (when (> start (+ offset (total-length tree)))
635 (setf start (+ offset (total-length tree))))
636 (while (> left-to-delete 0)
637 (decf left-to-delete
638 (interval-deletion-adjustment tree (- start offset) left-to-delete))
639 (setf tree (intervals buffer))
640 (when (= left-to-delete (interval-length tree))
641 (setf (intervals buffer) nil)
642 (return)))))))
644 (defun interval-start-pos (source)
645 (if (or (null source)
646 (not (typep (interval-parent source) 'buffer)))
648 (buffer-min (interval-parent source))))
650 (defun graft-intervals-into-buffer (source position length buffer inherit)
651 "Insert the intervals of SOURCE into BUFFER at POSITION.
652 LENGTH is the length of the text in SOURCE.
654 The `position' field of the SOURCE intervals is assumed to be
655 consistent with its parent; therefore, SOURCE must be an
656 interval tree made with copy_interval or must be the whole
657 tree of a buffer or a string.
659 This is used in insdel.c when inserting Lisp_Strings into the
660 buffer. The text corresponding to SOURCE is already in the buffer
661 when this is called. The intervals of new tree are a copy of those
662 belonging to the string being inserted; intervals are never
663 shared.
665 If the inserted text had no intervals associated, and we don't
666 want to inherit the surrounding text's properties, this function
667 simply returns -- offset_intervals should handle placing the
668 text in the correct interval, depending on the sticky bits.
670 If the inserted text had properties (intervals), then there are two
671 cases -- either insertion happened in the middle of some interval,
672 or between two intervals.
674 If the text goes into the middle of an interval, then new
675 intervals are created in the middle with only the properties of
676 the new text, *unless* the macro MERGE_INSERTIONS is true, in
677 which case the new text has the union of its properties and those
678 of the text into which it was inserted.
680 If the text goes between two intervals, then if neither interval
681 had its appropriate sticky property set (front_sticky, rear_sticky),
682 the new text has only its properties. If one of the sticky properties
683 is set, then the new text \"sticks\" to that region and its properties
684 depend on merging as above. If both the preceding and succeeding
685 intervals to the new text are \"sticky\", then the new text retains
686 only its properties, as if neither sticky property were set. Perhaps
687 we should consider merging all three sets of properties onto the new
688 text..."
689 (let ((tree (intervals buffer))
690 over-used
691 under over this prev)
692 ;; /* If the new text has no properties, then with inheritance it
693 ;; becomes part of whatever interval it was inserted into.
694 ;; To prevent inheritance, we must clear out the properties
695 ;; of the newly inserted text. */
696 (when (null source)
697 (when (and (not inherit)
698 tree
699 (> length 0))
700 ;; XSETBUFFER (buf, buffer);
701 (set-text-properties-1 position (+ position length) nil buffer 0))
702 (when (intervals buffer)
703 (setf (intervals buffer) (balance-an-interval (intervals buffer))))
704 (return-from graft-intervals-into-buffer))
705 (cond ((null tree)
706 ;; /* The inserted text constitutes the whole buffer, so
707 ;; simply copy over the interval structure. */
708 (when (= (- (buffer-size buffer) (buffer-min buffer))
709 (total-length source))
710 (setf (intervals buffer) (reproduce-tree source buffer)
711 (interval-pt (intervals buffer)) (buffer-min buffer))
712 (return-from graft-intervals-into-buffer))
713 ;; /* Create an interval tree in which to place a copy
714 ;; of the intervals of the inserted string. */
715 (setf tree (create-root-interval buffer)))
716 ((= (total-length tree)
717 (total-length source))
718 ;; /* If the buffer contains only the new string, but
719 ;; there was already some interval tree there, then it may be
720 ;; some zero length intervals. Eventually, do something clever
721 ;; about inserting properly. For now, just waste the old intervals. */
722 (setf (intervals buffer) (reproduce-tree source (interval-parent tree))
723 (interval-pt (intervals buffer)) (buffer-min buffer))
724 (return-from graft-intervals-into-buffer))
725 ((zerop (total-length tree))
726 (error "bork")))
727 (setf under (find-interval tree position)
728 this under
729 over (find-interval source (interval-start-pos source)))
730 (if (> position (interval-pt under))
731 (let ((end-unchanged (split-interval-left this (- position (interval-pt under)))))
732 (copy-properties under end-unchanged)
733 (setf (interval-pt under) position))
734 (setf prev (previous-interval under)))
735 (setf over-used 0)
736 (while over
737 (if (< (- (interval-text-length over) over-used)
738 (interval-text-length under))
739 (progn
740 (setf this (split-interval-left under (- (interval-text-length over)
741 over-used)))
742 (copy-properties under this))
743 (setf this under))
744 ;; /* THIS is now the interval to copy or merge into.
745 ;; OVER covers all of it. */
746 (if inherit
747 (merge-properties over this)
748 (copy-properties over this))
749 ;; /* If THIS and OVER end at the same place,
750 ;; advance OVER to a new source interval. */
751 (if (= (interval-text-length this)
752 (- (interval-text-length over) over-used))
753 (progn
754 (setf over (next-interval over)
755 over-used 0))
756 ;; /* Otherwise just record that more of OVER has been used. */
757 (incf over-used (interval-text-length this)))
758 ;; /* Always advance to a new target interval. */
759 (setf under (next-interval this)))
760 (when (intervals buffer)
761 (setf (intervals buffer) (balance-an-interval (intervals buffer))))))
763 (defun root-interval-p (i)
764 "Return true if i is the root interval node."
765 (or (null (interval-parent i))
766 (not (typep (interval-parent i) 'interval))))
768 (defun root-interval (interval)
769 "Return the root of interval."
770 (do ((i interval (interval-parent i)))
771 ((root-interval-p i) i)))
773 (defun leaf-interval-p (i)
774 "Return T if this interval has no children."
775 (and (null (interval-left i))
776 (null (interval-right i))))
778 (defun only-interval-p (i)
779 "Return T if this interval is the only interval in the interval tree."
780 (and (root-interval-p i)
781 (leaf-interval-p i)))
784 (defun delete-node (i)
785 ;; Trivial cases
786 (when (null (interval-left i))
787 (return-from delete-node (interval-right i)))
788 (when (null (interval-right i))
789 (return-from delete-node (interval-left i)))
790 ;; Meat
791 (let ((migrate (interval-left i))
792 (this (interval-right i))
793 (migrate-amt (interval-length (interval-left i))))
794 (while (interval-left this)
795 (setf this (interval-left this))
796 (incf (interval-length this) migrate-amt))
797 (check-total-length this)
798 (setf (interval-left this) migrate)
799 (setf (interval-parent migrate) this)
800 (interval-right i)))
802 (defun delete-interval (i)
803 (let ((amt (interval-text-length i))
804 parent)
805 (and (> amt 0)
806 (error "only used on zero length intervals."))
807 (when (root-interval-p i)
808 (let ((owner (interval-parent i)))
809 (setf parent (delete-node i))
810 (when (interval-parent parent)
811 (setf (interval-parent parent) owner))
812 (setf (intervals owner) parent)
813 (return-from delete-interval)))
814 (setf parent (interval-parent i))
815 (if (left-child-p i)
816 (progn
817 (setf (interval-left parent) (delete-node i))
818 (when (interval-left parent)
819 (setf (interval-parent (interval-left parent)) parent)))
820 (progn
821 (setf (interval-right parent) (delete-node i))
822 (when (interval-right parent)
823 (setf (interval-parent (interval-right parent)) parent))))))
825 (defun default-interval-p (i)
826 (or (null i)
827 (null (interval-plist i))))
829 (defun reproduce-tree (source parent)
830 (let ((tree (copy-interval source)))
831 (setf (interval-plist tree) (copy-list (interval-plist source))
832 (interval-parent tree) parent)
833 (when (interval-left source)
834 (setf (interval-left tree) (reproduce-tree (interval-left source) tree)))
835 (when (interval-right source)
836 (setf (interval-right tree) (reproduce-tree (interval-right source) tree)))
837 tree))
839 (defun merge-properties (source target)
840 "/* Merge the properties of interval SOURCE into the properties of
841 interval TARGET. That is to say, each property in SOURCE is added to
842 TARGET if TARGET has no such property as yet. */"
843 (unless (and (default-interval-p source)
844 (default-interval-p target))
845 (doplist (sym val (interval-plist source))
846 (let ((found (getf (interval-plist target) sym)))
847 (unless found
848 (setf (getf (interval-plist target) sym) val))))))
850 (defun merge-properties-sticky (pleft pright)
851 "Any property might be front-sticky on the left, rear-sticky on the left,
852 front-sticky on the right, or rear-sticky on the right; the 16 combinations
853 can be arranged in a matrix with rows denoting the left conditions and
854 columns denoting the right conditions:
855 _ __ _
856 _ FR FR FR FR
857 FR__ 0 1 2 3
858 _FR 4 5 6 7
859 FR 8 9 A B
860 FR C D E F
862 left-props = '(front-sticky (p8 p9 pa pb pc pd pe pf)
863 rear-nonsticky (p4 p5 p6 p7 p8 p9 pa pb)
864 p0 L p1 L p2 L p3 L p4 L p5 L p6 L p7 L
865 p8 L p9 L pa L pb L pc L pd L pe L pf L)
866 right-props = '(front-sticky (p2 p3 p6 p7 pa pb pe pf)
867 rear-nonsticky (p1 p2 p5 p6 p9 pa pd pe)
868 p0 R p1 R p2 R p3 R p4 R p5 R p6 R p7 R
869 p8 R p9 R pa R pb R pc R pd R pe R pf R)
871 We inherit from whoever has a sticky side facing us. If both sides
872 do (cases 2, 3, E, and F), then we inherit from whichever side has a
873 non-nil value for the current property. If both sides do, then we take
874 from the left.
876 When we inherit a property, we get its stickiness as well as its value.
877 So, when we merge the above two lists, we expect to get this:
879 result = '(front-sticky (p6 p7 pa pb pc pd pe pf)
880 rear-nonsticky (p6 pa)
881 p0 L p1 L p2 L p3 L p6 R p7 R
882 pa R pb R pc L pd L pe L pf L)
884 The optimizable special cases are:
885 left rear-nonsticky = nil, right front-sticky = nil (inherit left)
886 left rear-nonsticky = t, right front-sticky = t (inherit right)
887 left rear-nonsticky = t, right front-sticky = nil (inherit none)"
888 (labels ((tmem (sym set)
889 ;; Test for membership, allowing for t (actually any
890 ;; non-cons) to mean the universal set."
891 (if (consp set)
892 (find sym set)
893 set)))
894 (let (props
895 front
896 rear
897 (lfront (getf pleft 'front-sticky))
898 (lrear (getf pleft 'rear-nonsticky))
899 (rfront (getf pright 'front-sticky))
900 (rrear (getf pright 'rear-nonsticky))
901 cat use-left use-right)
902 (doplist (sym rval pright)
903 (unless (or (eq sym 'rear-nonsticky)
904 (eq sym 'front-sticky))
905 ;; Indicate whether the property is explicitly
906 ;; defined on the left. (We know it is defined
907 ;; explicitly on the right because otherwise we don't
908 ;; get here.)
909 (let* ((lval (getf pleft sym))
910 ;; Even if lrear or rfront say nothing about the
911 ;; stickiness of SYM,
912 ;; Vtext_property_default_nonsticky may give
913 ;; default stickiness to SYM.
914 (tmp (assoc sym *text-property-default-nonsticky*)))
915 (setf use-left (and lval
916 (not (or (tmem sym lrear)
917 (and (consp tmp)
918 (cdr tmp)))))
919 use-right (or (tmem sym lrear)
920 (and (consp tmp)
921 (null (cdr tmp)))))
922 (when (and use-left
923 use-right)
924 (cond ((null lval)
925 (setf use-left nil))
926 ((null rval)
927 (setf use-right nil))))
928 (cond (use-left
929 ;; We build props as (value sym ...) rather than (sym value ...)
930 ;; because we plan to nreverse it when we're done.
931 (setf (getf props sym) lval)
932 (when (tmem sym lfront)
933 (push sym front))
934 (when (tmem sym lrear)
935 (push sym rear)))
936 (use-right
937 (setf (getf props sym) rval)
938 (when (tmem sym rfront)
939 (push sym front))
940 (when (tmem sym rrear)
941 (push sym rear)))))))
942 ;; Now go through each element of PLEFT.
943 (doplist (sym lval pleft)
944 (unless (or (eq sym 'rear-nonsticky)
945 (eq sym 'front-sticky))
946 ;; If sym is in PRIGHT, we've already considered it.
947 (let* ((present (getf pright sym))
948 ;; Even if lrear or rfront say nothing about the
949 ;; stickiness of SYM,
950 ;; Vtext_property_default_nonsticky may give
951 ;; default stickiness to SYM.
952 (tmp (assoc sym *text-property-default-nonsticky*)))
953 ;; XXX: if sym is set in pright to nil, its the same
954 ;; as sym not being in the list.
955 (unless present
956 ;; Since rval is known to be nil in this loop, the test simplifies.
957 (cond ((not (or (tmem sym lrear)
958 (and (consp tmp)
959 (cdr tmp))))
960 (setf (getf props sym) lval)
961 (when (tmem sym lfront)
962 (push sym front)))
963 ((or (tmem sym rfront)
964 (and (consp tmp)
965 (null (cdr tmp))))
966 ;; The value is nil, but we still inherit the stickiness
967 ;; from the right.
968 (setf (getf props sym) lval)
969 (when (tmem sym rrear)
970 (push sym rear))))))))
971 (when rear
972 (setf (getf props 'rear-nonsticky) (nreverse rear)))
973 (setf cat (textget props 'category))
974 ;; If we have inherited a front-stick category property that is t,
975 ;; we don't need to set up a detailed one.
976 (when (and front
977 (not (and cat
978 (symbolp cat)
979 (eq (get cat 'front-sticky) t))))
980 (setf (getf props 'front-sticky) (nreverse front)))
981 props)))
983 (defun offset-intervals (buffer start length)
984 "Make the adjustments necessary to the interval tree of BUFFER to
985 represent an addition or deletion of LENGTH characters starting
986 at position START. Addition or deletion is indicated by the sign
987 of LENGTH."
988 (unless (or (null (intervals buffer))
989 (zerop length))
990 (if (> length 0)
991 (adjust-intervals-for-insertion (intervals buffer) start length)
992 (adjust-intervals-for-deletion buffer (+ start length) (- length)))))
994 (provide :lice-0.1/intervals)