Marks now carry values, and multiple marks with the same key are permitted so long as the positions differ
This commit is contained in:
parent
617dbcd326
commit
4c677ea730
205
ansi/rope.rkt
205
ansi/rope.rkt
|
@ -21,17 +21,19 @@
|
||||||
subrope
|
subrope
|
||||||
|
|
||||||
has-mark?
|
has-mark?
|
||||||
lookup-mark
|
find-next-mark
|
||||||
|
find-prev-mark
|
||||||
|
find-all-marks
|
||||||
set-mark
|
set-mark
|
||||||
clear-mark
|
clear-mark)
|
||||||
update-mark)
|
|
||||||
|
|
||||||
(require racket/set)
|
(require racket/set)
|
||||||
(require racket/match)
|
(require racket/match)
|
||||||
|
|
||||||
(module+ test (require rackunit racket/pretty))
|
(module+ test (require rackunit racket/pretty))
|
||||||
|
|
||||||
;; A Mark is a Symbol.
|
;; A Mark is a Symbol. Marks can be associated with a set of Any
|
||||||
|
;; values at each position in the rope.
|
||||||
|
|
||||||
;; A Strand is a (strand String Number Number), representing a
|
;; A Strand is a (strand String Number Number), representing a
|
||||||
;; substring of a string.
|
;; substring of a string.
|
||||||
|
@ -45,7 +47,7 @@
|
||||||
right ;; Rope or #f
|
right ;; Rope or #f
|
||||||
size* ;; Number, total length of this rope
|
size* ;; Number, total length of this rope
|
||||||
marks* ;; Set of Mark
|
marks* ;; Set of Mark
|
||||||
mark-index ;; Hashtable from Mark to Number, marks in this span
|
mark-index ;; (Hasheq Mark (Hash Number (Setof Any))), marks in this span
|
||||||
) #:transparent)
|
) #:transparent)
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
@ -106,7 +108,7 @@
|
||||||
(define (empty-rope) #f)
|
(define (empty-rope) #f)
|
||||||
|
|
||||||
(define (strand->rope t)
|
(define (strand->rope t)
|
||||||
(rope t #f #f (strand-count t) (set) (hash)))
|
(rope t #f #f (strand-count t) (seteq) (hasheq)))
|
||||||
|
|
||||||
(define (string->rope s)
|
(define (string->rope s)
|
||||||
(strand->rope (string->strand s)))
|
(strand->rope (string->strand s)))
|
||||||
|
@ -186,58 +188,106 @@
|
||||||
[(< pos hi) (values 'here #t)]
|
[(< pos hi) (values 'here #t)]
|
||||||
[else (values 'right (- pos hi))]))))
|
[else (values 'right (- pos hi))]))))
|
||||||
|
|
||||||
(define (find-mark mark)
|
;; Searches from pos (inclusive) in the direction indicated.
|
||||||
(define (walk offset r)
|
;; Pos points to a mark-position, not a character-position.
|
||||||
(cond
|
(define (find-mark r forward? mark start-pos)
|
||||||
[(not r)
|
(define (search-here r start-pos)
|
||||||
(values 'here #f)]
|
(define marks (hash-ref (rope-mark-index r) mark #f))
|
||||||
[(hash-ref (rope-mark-index r) mark #f) =>
|
(define lo (rope-lo r))
|
||||||
(lambda (p) (values 'here (+ offset (rope-lo r) p)))]
|
(if (not marks)
|
||||||
[(set-member? (rope-marks (rope-left r)) mark)
|
#f
|
||||||
(values 'left offset)]
|
(let ((pos-comparer (if forward? < >))
|
||||||
[(set-member? (rope-marks (rope-right r)) mark)
|
(boundary-comparer (if forward? >= <=)))
|
||||||
(values 'right (+ offset (rope-lo r) (strand-count (rope-strand r))))]
|
(for/fold [(candidate #f)] [((pos value) (in-hash marks))]
|
||||||
[else
|
(if (and (or (not candidate)
|
||||||
(values 'here #f)]))
|
(pos-comparer pos (car candidate)))
|
||||||
;; (trace walk)
|
(boundary-comparer pos start-pos))
|
||||||
walk)
|
(cons (+ pos lo) value)
|
||||||
|
candidate)))))
|
||||||
|
(define (search r start-pos)
|
||||||
|
(and r
|
||||||
|
(set-member? (rope-marks r) mark)
|
||||||
|
(let-values (((lo hi) (rope-lo+hi r)))
|
||||||
|
(cond
|
||||||
|
[(< start-pos lo)
|
||||||
|
(or (search (rope-left r) start-pos)
|
||||||
|
(and forward? (search-here r (- start-pos lo))))]
|
||||||
|
[(> start-pos hi)
|
||||||
|
(or (search (rope-right r) (- start-pos hi))
|
||||||
|
(and (not forward?) (search-here r (- start-pos lo))))]
|
||||||
|
[else
|
||||||
|
(search-here r (- start-pos lo))]))))
|
||||||
|
(search r start-pos))
|
||||||
|
|
||||||
(define (has-mark? r mark)
|
(define (has-mark? r mark)
|
||||||
(and r (set-member? (rope-marks r) mark)))
|
(and r (set-member? (rope-marks r) mark)))
|
||||||
|
|
||||||
(define (lookup-mark r mark)
|
(define (find-mark* r forward? mark start-pos)
|
||||||
(splay-to r (find-mark mark) 0))
|
(define maybe-pos+val (find-mark r forward? mark start-pos))
|
||||||
|
(if maybe-pos+val
|
||||||
|
(values (car maybe-pos+val) (cdr maybe-pos+val))
|
||||||
|
(values #f #f)))
|
||||||
|
|
||||||
(define (set-mark r0 mark position)
|
(define (find-next-mark r mark [pos 0]) (find-mark* r #t mark pos))
|
||||||
(when (set-member? (rope-marks r0) mark) (error 'set-mark "Duplicate mark: ~a" mark))
|
(define (find-prev-mark r mark [pos (rope-size r)]) (find-mark* r #f mark pos))
|
||||||
(define-values (found? r1) (splay-to r0 find-position position))
|
|
||||||
(when (not found?) (error 'set-mark "Invalid position ~a setting mark ~a" position mark))
|
(define (mark-union h1 h2 offset)
|
||||||
|
(for/fold [(h h1)] [((pos val) (in-hash h2))] (hash-set h (+ offset pos) val)))
|
||||||
|
|
||||||
|
(define (find-all-marks r mark)
|
||||||
|
(define (walk r)
|
||||||
|
(if (set-member? (rope-marks r) mark)
|
||||||
|
(let-values (((lo hi) (rope-lo+hi r)))
|
||||||
|
(mark-union (walk (rope-left r))
|
||||||
|
(mark-union (hash-ref (rope-mark-index r) mark (lambda () (hash)))
|
||||||
|
(walk (rope-right r))
|
||||||
|
hi)
|
||||||
|
lo))
|
||||||
|
(hash)))
|
||||||
|
(walk r))
|
||||||
|
|
||||||
|
(define (splay-to-pos what r0 pos [extra (lambda () "")])
|
||||||
|
(define-values (found? r1) (splay-to r0 find-position pos))
|
||||||
|
(when (not found?) (error what "Invalid position ~a~a" pos (extra)))
|
||||||
|
r1)
|
||||||
|
|
||||||
|
(define (add-mark-to-table old-marks mark pos value)
|
||||||
|
(define old-mark (hash-ref old-marks mark (lambda () (hash))))
|
||||||
|
(hash-set old-marks mark (hash-set old-mark pos value)))
|
||||||
|
|
||||||
|
(define (set-mark r0 mark position value)
|
||||||
|
(define r (splay-to-pos 'set-mark r0 position (lambda () (format " setting mark ~a" mark))))
|
||||||
(reindex
|
(reindex
|
||||||
(if (not r1)
|
(if (not r)
|
||||||
(rope (empty-strand) #f #f 'will-be-recomputed 'will-be-recomputed (hash mark position))
|
(rope (empty-strand)
|
||||||
(struct-copy rope r1
|
#f
|
||||||
[mark-index (hash-set (rope-mark-index r1) mark (- position (rope-lo r1)))]))))
|
#f
|
||||||
|
'will-be-recomputed
|
||||||
|
'will-be-recomputed
|
||||||
|
(hasheq mark (hash position value)))
|
||||||
|
(struct-copy rope r [mark-index (add-mark-to-table (rope-mark-index r)
|
||||||
|
mark
|
||||||
|
(- position (rope-lo r))
|
||||||
|
value)]))))
|
||||||
|
|
||||||
(define (clear-mark r0 mark)
|
(define (clear-mark r0 mark position)
|
||||||
(define-values (old-pos r) (lookup-mark r0 mark))
|
(define r (splay-to-pos 'clear-mark r0 position (lambda () (format " clearing mark ~a" mark))))
|
||||||
(if old-pos
|
(and r
|
||||||
(struct-copy rope r
|
(reindex
|
||||||
[marks* (set-remove (rope-marks* r) mark)]
|
(struct-copy rope r
|
||||||
[mark-index (hash-remove (rope-mark-index r) mark)])
|
[mark-index
|
||||||
r))
|
(let* ((old-marks (rope-mark-index r))
|
||||||
|
(old-mark (hash-ref old-marks mark (lambda () (hash)))))
|
||||||
(define (update-mark r0 mark position)
|
(define new-mark (hash-remove old-mark position))
|
||||||
(set-mark (if (has-mark? r0 mark)
|
(if (hash-empty? new-mark)
|
||||||
(clear-mark r0 mark)
|
(hash-remove old-marks mark)
|
||||||
r0)
|
(hash-set old-marks mark new-mark)))]))))
|
||||||
mark
|
|
||||||
position))
|
|
||||||
|
|
||||||
(define (rope-size r)
|
(define (rope-size r)
|
||||||
(if r (rope-size* r) 0))
|
(if r (rope-size* r) 0))
|
||||||
|
|
||||||
(define (rope-marks r)
|
(define (rope-marks r)
|
||||||
(if r (rope-marks* r) (set)))
|
(if r (rope-marks* r) (seteq)))
|
||||||
|
|
||||||
(define (reindex r)
|
(define (reindex r)
|
||||||
(struct-copy rope r
|
(struct-copy rope r
|
||||||
|
@ -246,11 +296,10 @@
|
||||||
(strand-count (rope-strand r)))]
|
(strand-count (rope-strand r)))]
|
||||||
[marks* (set-union (rope-marks (rope-left r))
|
[marks* (set-union (rope-marks (rope-left r))
|
||||||
(rope-marks (rope-right r))
|
(rope-marks (rope-right r))
|
||||||
(list->set (hash-keys (rope-mark-index r))))]))
|
(list->seteq (hash-keys (rope-mark-index r))))]))
|
||||||
|
|
||||||
(define (rope-split r0 position)
|
(define (rope-split r0 position)
|
||||||
(define-values (found? r) (splay-to r0 find-position position))
|
(define r (splay-to-pos 'rope-split r0 position))
|
||||||
(when (not found?) (error 'rope-split "Invalid position ~a" position))
|
|
||||||
;; We know the position is in the root of r.
|
;; We know the position is in the root of r.
|
||||||
(match-define (rope t rl rr size marks mark-index) r)
|
(match-define (rope t rl rr size marks mark-index) r)
|
||||||
(define-values (lo hi) (rope-lo+hi r))
|
(define-values (lo hi) (rope-lo+hi r))
|
||||||
|
@ -261,18 +310,19 @@
|
||||||
(define right-strand (substrand t offset))
|
(define right-strand (substrand t offset))
|
||||||
(values (if (strand-empty? left-strand)
|
(values (if (strand-empty? left-strand)
|
||||||
rl
|
rl
|
||||||
(reindex (rope left-strand rl #f 'will-be-recomputed (set) left-index)))
|
(reindex (rope left-strand rl #f 'will-be-recomputed (seteq) left-index)))
|
||||||
marks-at-split
|
marks-at-split
|
||||||
(if (strand-empty? right-strand)
|
(if (strand-empty? right-strand)
|
||||||
rr
|
rr
|
||||||
(reindex (rope right-strand #f rr 'will-be-recomputed (set) right-index)))))
|
(reindex (rope right-strand #f rr 'will-be-recomputed (seteq) right-index)))))
|
||||||
|
|
||||||
(define (partition-mark-index index offset)
|
(define (partition-mark-index index offset)
|
||||||
(for/fold [(l (hash)) (m (set)) (r (hash))]
|
(for*/fold [(l (hasheq)) (m (hasheq)) (r (hasheq))]
|
||||||
[((mark pos) (in-hash index))]
|
[((mark posvals) (in-hash index))
|
||||||
(values (if (< pos offset) (hash-set l mark pos) l)
|
((pos val) (in-hash posvals))]
|
||||||
(if (= pos offset) (set-add m mark) m)
|
(values (if (< pos offset) (add-mark-to-table l mark pos val) l)
|
||||||
(if (> pos offset) (hash-set r mark (- pos offset)) r))))
|
(if (= pos offset) (hash-set m mark (set-add (hash-ref m mark (set)) val)) m)
|
||||||
|
(if (> pos offset) (add-mark-to-table r mark (- pos offset) val) r))))
|
||||||
|
|
||||||
(define (rope-append rl0 rr0)
|
(define (rope-append rl0 rr0)
|
||||||
(cond
|
(cond
|
||||||
|
@ -287,17 +337,17 @@
|
||||||
(let ((merged-index (merge-mark-indexes (rope-mark-index rl)
|
(let ((merged-index (merge-mark-indexes (rope-mark-index rl)
|
||||||
(rope-mark-index rr)
|
(rope-mark-index rr)
|
||||||
(strand-count (rope-strand rl)))))
|
(strand-count (rope-strand rl)))))
|
||||||
(reindex (rope t (rope-left rl) (rope-right rr) 'will-be-recomputed (set) merged-index)))
|
(reindex (rope t (rope-left rl) (rope-right rr) 'will-be-recomputed (seteq) merged-index)))
|
||||||
(replace-right rl rr))]))
|
(replace-right rl rr))]))
|
||||||
|
|
||||||
(define (rope-concat rs)
|
(define (rope-concat rs)
|
||||||
(foldr rope-append (empty-rope) rs))
|
(foldr rope-append (empty-rope) rs))
|
||||||
|
|
||||||
(define (merge-mark-indexes li ri offset)
|
(define (merge-mark-indexes li ri offset)
|
||||||
(for/fold [(i li)] [((k v) (in-hash ri))]
|
(for*/fold [(i li)]
|
||||||
(if (hash-has-key? i k)
|
[((mark posvals) (in-hash ri))
|
||||||
(error 'merge-mark-indexes "Duplicate mark: ~a" k)
|
((pos val) (in-hash posvals))]
|
||||||
(hash-set i k (+ offset v)))))
|
(add-mark-to-table i mark (+ pos offset) val)))
|
||||||
|
|
||||||
(define (subrope* r0 [lo0 #f] [hi0 #f])
|
(define (subrope* r0 [lo0 #f] [hi0 #f])
|
||||||
(define lo (if (not lo0)
|
(define lo (if (not lo0)
|
||||||
|
@ -341,25 +391,38 @@
|
||||||
(check-equal? (rope->string r) text)
|
(check-equal? (rope->string r) text)
|
||||||
(loop (- n 1) r)))
|
(loop (- n 1) r)))
|
||||||
|
|
||||||
(let*-values (((r) (set-mark (rope-concat rope-pieces) 'mark 9))
|
(let*-values (((r) (set-mark (rope-concat rope-pieces) 'mark 9 "original"))
|
||||||
((_) (check-equal? (rope->string r) text))
|
((_) (check-equal? (rope->string r) text))
|
||||||
((pos r) (lookup-mark r 'mark))
|
((pos val) (find-next-mark r 'mark))
|
||||||
((_) (check-equal? pos 9))
|
((_) (check-equal? pos 9))
|
||||||
((r) (clear-mark r 'mark))
|
((_) (check-equal? val "original"))
|
||||||
((pos r) (lookup-mark r 'mark))
|
((r) (clear-mark r 'mark pos))
|
||||||
|
((_) (check-equal? (find-all-marks r 'mark) (hash)))
|
||||||
|
((pos val) (find-next-mark r 'mark))
|
||||||
((_) (check-false pos))
|
((_) (check-false pos))
|
||||||
((r) (update-mark r 'mark 9))
|
((_) (check-false val))
|
||||||
((pos r) (lookup-mark r 'mark))
|
((r) (set-mark r 'mark 9 "second"))
|
||||||
|
((pos val) (find-next-mark r 'mark))
|
||||||
((_) (check-equal? pos 9))
|
((_) (check-equal? pos 9))
|
||||||
((r) (update-mark r 'mark 6))
|
((_) (check-equal? val "second"))
|
||||||
((pos r) (lookup-mark r 'mark))
|
((r) (set-mark r 'mark 6 "first"))
|
||||||
|
((_) (check-equal? (find-all-marks r 'mark) (hash 6 "first" 9 "second")))
|
||||||
|
((pos val) (find-prev-mark r 'mark))
|
||||||
|
((_) (check-equal? pos 9))
|
||||||
|
((_) (check-equal? val "second"))
|
||||||
|
((pos val) (find-next-mark r 'mark))
|
||||||
((_) (check-equal? pos 6))
|
((_) (check-equal? pos 6))
|
||||||
|
((_) (check-equal? val "first"))
|
||||||
((l ms r) (rope-split r pos))
|
((l ms r) (rope-split r pos))
|
||||||
((_) (check-equal? ms (set 'mark)))
|
((_) (check-equal? (find-all-marks r 'mark) (hash 3 "second")))
|
||||||
|
((_) (check-equal? ms (hasheq 'mark (set "first"))))
|
||||||
((_) (check-equal? (rope->string l) (substring text 0 6)))
|
((_) (check-equal? (rope->string l) (substring text 0 6)))
|
||||||
((_) (check-equal? (rope->string r) (substring text 6 (string-length text))))
|
((_) (check-equal? (rope->string r) (substring text 6 (string-length text))))
|
||||||
|
((_) (check-equal? (rope-marks l) (seteq)))
|
||||||
|
((_) (check-equal? (rope-marks r) (seteq 'mark)))
|
||||||
((l ms r) (rope-split r 3))
|
((l ms r) (rope-split r 3))
|
||||||
((_) (check-equal? ms (set)))
|
((_) (check-equal? (find-all-marks r 'mark) (hash)))
|
||||||
|
((_) (check-equal? ms (hasheq 'mark (set "second"))))
|
||||||
((_) (check-equal? (rope->string l) (substring text 6 9)))
|
((_) (check-equal? (rope->string l) (substring text 6 9)))
|
||||||
((_) (check-equal? (rope->string r) (substring text 9 (string-length text)))))
|
((_) (check-equal? (rope->string r) (substring text 9 (string-length text)))))
|
||||||
(void)))
|
(void)))
|
||||||
|
|
Loading…
Reference in New Issue