2014-12-15 06:29:48 +00:00
#lang racket/base
;; Ropes for text editing
( provide empty-strand
string->strand
strand->string
substrand
strand-equal?
strand-empty?
2014-12-22 19:03:02 +00:00
( rename-out [ rope?* rope? ] )
rope-empty?
2014-12-15 06:29:48 +00:00
empty-rope
strand->rope
string->rope
rope->string
rope-size
rope-marks
rope-split
rope-append
rope-concat
subrope
2014-12-20 03:40:10 +00:00
rope-generator
2014-12-20 04:45:39 +00:00
rope-seek
2014-12-15 06:29:48 +00:00
2014-12-20 17:52:35 +00:00
( struct-out mark-type )
2014-12-15 06:29:48 +00:00
has-mark?
2014-12-20 18:08:54 +00:00
find-mark
find-mark-pos
find-all-marks/type
2014-12-15 06:29:48 +00:00
set-mark
2014-12-20 04:45:39 +00:00
clear-mark
2014-12-20 18:08:54 +00:00
replace-mark
clear-all-marks )
2014-12-15 06:29:48 +00:00
( require racket/set )
( require racket/match )
2014-12-20 03:40:10 +00:00
( require racket/generator )
2014-12-15 06:29:48 +00:00
( module+ test ( require rackunit racket/pretty ) )
2014-12-20 17:52:35 +00:00
;; A Stickiness is one of
;; -- 'left or
;; -- 'right
;; and indicates the side after a rope-split to which a mark with this
;; Stickiness adheres. What Finseth calls a "normal mark" has 'right
;; stickiness, and what he calls a "fixed mark" has 'left stickiness.
;; A MarkType is a (mark-type String Stickiness). MarkTypes can be
;; associated with a set of Any values at each position in the rope.
2014-12-22 22:17:05 +00:00
( struct mark-type ( name stickiness ) #:prefab )
2014-12-15 06:29:48 +00:00
;; A Strand is a (strand String Number Number), representing a
;; substring of a string.
2014-12-22 22:17:05 +00:00
( struct strand ( text offset count ) #:prefab )
2014-12-15 06:29:48 +00:00
;; A Rope is a splay tree representing a long piece of text.
;; #f is the empty Rope; otherwise a (rope) struct instance.
;; INVARIANT: Adjacent ropes will be merged to maximize sharing.
( struct rope ( strand ;; Strand
left ;; Rope or #f
right ;; Rope or #f
size* ;; Number, total length of this rope
2014-12-20 18:19:35 +00:00
marks* ;; (Seteq MarkType)
mark-index ;; (Hasheq MarkType (Hash Number (Set Any))), marks in this span
2014-12-22 22:17:05 +00:00
) #:prefab )
2014-12-15 06:29:48 +00:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; Strands
( define *glom-up-to* 128 )
( define ( empty-strand ) ( strand " " 0 0 ) )
( define ( string->strand s )
( strand s 0 ( string-length s ) ) )
( define ( strand->string t )
( match-define ( strand text offset count ) t )
( if ( and ( zero? offset ) ( = count ( string-length text ) ) )
text
( substring text offset ( + offset count ) ) ) )
2014-12-23 16:47:53 +00:00
( define ( compute-range-index index default limit )
2014-12-22 19:03:02 +00:00
( cond [ ( not index ) default ]
[ ( zero? limit ) 0 ]
2014-12-23 16:44:18 +00:00
[ else ( max 0 ( min limit ( if ( negative? index ) ( + index limit ) index ) ) ) ] ) )
2014-12-22 19:03:02 +00:00
2014-12-15 06:29:48 +00:00
( define ( substrand t0 [ lo0 #f ] [ hi0 #f ] )
( define t ( if ( string? t0 ) ( string->strand t0 ) t0 ) )
2014-12-23 16:47:53 +00:00
( define lo ( compute-range-index lo0 0 ( strand-count t ) ) )
( define hi ( compute-range-index hi0 ( strand-count t ) ( strand-count t ) ) )
2014-12-15 06:29:48 +00:00
( strand ( strand-text t )
( + ( strand-offset t ) lo )
( - hi lo ) ) )
( define ( strand-maybe-append t1 t2 )
( match-define ( strand text1 offset1 count1 ) t1 )
( match-define ( strand text2 offset2 count2 ) t2 )
( or ( and ( zero? count1 ) t2 )
( and ( zero? count2 ) t1 )
( and ( eq? text1 text2 )
( = ( + offset1 count1 ) offset2 )
( strand text1 offset1 ( + count1 count2 ) ) )
;; TODO: measure to see if the following improves or worsens memory usage
( and ( < ( + count1 count2 ) *glom-up-to* )
( string->strand ( string-append ( strand->string t1 ) ( strand->string t2 ) ) ) ) ) )
( define ( strand-equal? t1 t2 )
( string=? ( strand->string t1 )
( strand->string t2 ) ) )
( define ( strand-empty? t )
( zero? ( strand-count t ) ) )
( module+ test
( check-equal? ( strand-count ( empty-strand ) ) 0 )
( check-equal? ( strand-count ( string->strand " " ) ) 0 )
( check-true ( strand-equal? ( empty-strand ) ( string->strand " " ) ) ) )
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; Ropes
( define ( empty-rope ) #f )
2014-12-22 19:03:02 +00:00
( define ( rope-empty? r )
( equal? r ( empty-rope ) ) )
( define ( rope?* r )
( or ( rope-empty? r )
( rope? r ) ) )
2014-12-15 06:29:48 +00:00
( define ( strand->rope t )
2014-12-22 19:03:02 +00:00
( rope t ( empty-rope ) ( empty-rope ) ( strand-count t ) ( seteq ) ( hasheq ) ) )
2014-12-15 06:29:48 +00:00
( define ( string->rope s )
( strand->rope ( string->strand s ) ) )
( define ( rope->string r )
( define buf ( make-string ( rope-size r ) ) )
( let fill! ( ( r r ) ( offset 0 ) )
( when r
( fill! ( rope-left r ) offset )
( define lo ( rope-lo r ) )
( define s ( rope-strand r ) )
( string-copy! buf
( + offset lo )
( strand-text s )
( strand-offset s )
( + ( strand-offset s ) ( strand-count s ) ) )
( fill! ( rope-right r ) ( + offset lo ( strand-count s ) ) ) ) )
buf )
( define ( replace-left r n ) ( if r ( reindex ( struct-copy rope r [ left n ] ) ) n ) )
( define ( replace-right r n ) ( if r ( reindex ( struct-copy rope r [ right n ] ) ) n ) )
( define ( replace-both r rl rr ) ( reindex ( struct-copy rope r [ left rl ] [ right rr ] ) ) )
( define ( splay-to r direction-finder arg0 )
;; zig: last. desired position is a direct (left/right) child of r.
;; zig-zig: desired position is within a (left-left/right-right) grandchild of r.
;; zig-zag: desired position is within a (left-right/right-left) grandchild of r.
( define-values ( where arg1 ) ( direction-finder arg0 r ) )
( match where
[ ' here ( values arg1 r ) ]
[ ' left
( define rl ( rope-left r ) )
( define-values ( where arg2 ) ( direction-finder arg1 rl ) )
( match where
[ ' here ;; zig.
( values arg2 ( replace-right rl ( replace-left r ( and rl ( rope-right rl ) ) ) ) ) ]
[ ' left ;; zig-zig
( define-values ( v rll ) ( splay-to ( rope-left rl ) direction-finder arg2 ) )
( values v ( replace-right rll ( replace-both rl
( and rll ( rope-right rll ) )
( replace-left r ( rope-right rl ) ) ) ) ) ]
[ ' right ;; zig-zag
( define-values ( v rlr ) ( splay-to ( rope-right rl ) direction-finder arg2 ) )
( values v ( replace-both rlr
( replace-right rl ( rope-left rlr ) )
( replace-left r ( rope-right rlr ) ) ) ) ] ) ]
[ ' right
( define rr ( rope-right r ) )
( define-values ( where arg2 ) ( direction-finder arg1 rr ) )
( match where
[ ' here ;; zig.
( values arg2 ( replace-left rr ( replace-right r ( and rr ( rope-left rr ) ) ) ) ) ]
[ ' left ;; zig-zag
( define-values ( v rrl ) ( splay-to ( rope-left rr ) direction-finder arg2 ) )
( values v ( replace-both rrl
( replace-right r ( rope-left rrl ) )
( replace-left rr ( rope-right rrl ) ) ) ) ]
[ ' right ;; zig-zig
( define-values ( v rrr ) ( splay-to ( rope-right rr ) direction-finder arg2 ) )
( values v ( replace-left rrr ( replace-both rr
( replace-right r ( rope-left rr ) )
( and rrr ( rope-left rrr ) ) ) ) ) ] ) ] ) )
( define ( rope-lo r )
( rope-size ( rope-left r ) ) )
( define ( rope-lo+hi r )
( define lo ( rope-lo r ) )
( values lo ( + lo ( strand-count ( rope-strand r ) ) ) ) )
( define ( find-position pos r )
2014-12-22 19:03:02 +00:00
( if ( rope-empty? r )
2014-12-15 06:29:48 +00:00
( values ' here ( zero? pos ) )
( let-values ( ( ( lo hi ) ( rope-lo+hi r ) ) )
( cond
[ ( < pos lo ) ( values ' left pos ) ]
[ ( < pos hi ) ( values ' here #t ) ]
[ else ( values ' right ( - pos hi ) ) ] ) ) ) )
2014-12-20 03:04:10 +00:00
;; Searches from pos (inclusive) in the direction indicated.
;; Pos points to a mark-position, not a character-position.
2014-12-20 18:08:54 +00:00
( define ( find-mark* r forward? mtype start-pos )
2014-12-20 03:04:10 +00:00
( define ( search-here r start-pos )
2014-12-20 17:52:35 +00:00
( define marks ( hash-ref ( rope-mark-index r ) mtype #f ) )
2014-12-20 03:04:10 +00:00
( define lo ( rope-lo r ) )
( if ( not marks )
#f
( let ( ( pos-comparer ( if forward? < > ) )
( boundary-comparer ( if forward? >= <= ) ) )
( for/fold [ ( candidate #f ) ] [ ( ( pos value ) ( in-hash marks ) ) ]
( if ( and ( or ( not candidate )
( pos-comparer pos ( car candidate ) ) )
( boundary-comparer pos start-pos ) )
( cons ( + pos lo ) value )
candidate ) ) ) ) )
( define ( search r start-pos )
( and r
2014-12-20 17:52:35 +00:00
( set-member? ( rope-marks r ) mtype )
2014-12-20 03:04:10 +00:00
( 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 ) )
2014-12-15 06:29:48 +00:00
2014-12-20 17:52:35 +00:00
( define ( has-mark? r mtype )
( and r ( set-member? ( rope-marks r ) mtype ) ) )
2014-12-15 06:29:48 +00:00
2014-12-20 18:08:54 +00:00
( define ( find-mark r mtype
#:forward? [ forward? #t ]
#:position [ start-pos ( if forward? 0 ( rope-size r ) ) ] )
( define maybe-pos+val ( find-mark* r forward? mtype start-pos ) )
2014-12-20 03:04:10 +00:00
( if maybe-pos+val
( values ( car maybe-pos+val ) ( cdr maybe-pos+val ) )
( values #f #f ) ) )
2014-12-20 18:08:54 +00:00
( define ( find-mark-pos r mtype
#:forward? [ forward? #t ]
#:position [ start-pos ( if forward? 0 ( rope-size r ) ) ] )
( cond [ ( find-mark* r forward? mtype start-pos ) => car ]
[ else #f ] ) )
2014-12-20 04:45:39 +00:00
2014-12-20 03:04:10 +00:00
( define ( mark-union h1 h2 offset )
( for/fold [ ( h h1 ) ] [ ( ( pos val ) ( in-hash h2 ) ) ] ( hash-set h ( + offset pos ) val ) ) )
2014-12-20 18:08:54 +00:00
( define ( find-all-marks/type r mtype )
2014-12-20 03:04:10 +00:00
( define ( walk r )
2014-12-20 17:52:35 +00:00
( if ( set-member? ( rope-marks r ) mtype )
2014-12-20 03:04:10 +00:00
( let-values ( ( ( lo hi ) ( rope-lo+hi r ) ) )
( mark-union ( walk ( rope-left r ) )
2014-12-20 17:52:35 +00:00
( mark-union ( hash-ref ( rope-mark-index r ) mtype ( lambda ( ) ( hash ) ) )
2014-12-20 03:04:10 +00:00
( 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 )
2014-12-20 17:52:35 +00:00
( define ( add-mark-to-table old-marks mtype pos value )
( define old-mark ( hash-ref old-marks mtype ( lambda ( ) ( hash ) ) ) )
( hash-set old-marks mtype ( hash-set old-mark pos value ) ) )
2014-12-20 03:04:10 +00:00
2014-12-20 17:52:35 +00:00
( define ( set-mark r0 mtype position value )
( define r ( splay-to-pos ' set-mark r0 position ( lambda ( ) ( format " setting mark ~a " mtype ) ) ) )
2014-12-15 06:29:48 +00:00
( reindex
2014-12-22 19:03:02 +00:00
( if ( rope-empty? r )
2014-12-20 03:04:10 +00:00
( rope ( empty-strand )
2014-12-22 19:03:02 +00:00
( empty-rope )
( empty-rope )
2014-12-20 03:04:10 +00:00
' will-be-recomputed
' will-be-recomputed
2014-12-20 18:19:35 +00:00
( hasheq mtype ( hash position value ) ) )
2014-12-20 03:04:10 +00:00
( struct-copy rope r [ mark-index ( add-mark-to-table ( rope-mark-index r )
2014-12-20 17:52:35 +00:00
mtype
2014-12-20 03:04:10 +00:00
( - position ( rope-lo r ) )
value ) ] ) ) ) )
2014-12-20 17:52:35 +00:00
( define ( clear-mark r0 mtype position )
( define r ( splay-to-pos ' clear-mark r0 position ( lambda ( ) ( format " clearing mark ~a " mtype ) ) ) )
2014-12-20 03:04:10 +00:00
( and r
( reindex
( struct-copy rope r
[ mark-index
( let* ( ( old-marks ( rope-mark-index r ) )
2014-12-20 17:52:35 +00:00
( old-mark ( hash-ref old-marks mtype ( lambda ( ) ( hash ) ) ) ) )
2014-12-20 03:04:10 +00:00
( define new-mark ( hash-remove old-mark position ) )
( if ( hash-empty? new-mark )
2014-12-20 17:52:35 +00:00
( hash-remove old-marks mtype )
( hash-set old-marks mtype new-mark ) ) ) ] ) ) ) )
2014-12-15 06:29:48 +00:00
2014-12-20 17:52:35 +00:00
( define ( replace-mark r0 mtype new-pos new-value )
2014-12-20 18:08:54 +00:00
( define pos ( find-mark-pos r0 mtype ) )
2014-12-23 06:43:01 +00:00
( set-mark ( if pos ( clear-mark r0 mtype pos ) r0 ) mtype new-pos new-value ) )
2014-12-20 04:45:39 +00:00
2014-12-20 18:08:54 +00:00
( define ( clear-all-marks r )
( and r
( struct-copy rope r
2014-12-20 18:19:35 +00:00
[ marks* ( seteq ) ]
[ mark-index ( hasheq ) ]
2014-12-20 18:08:54 +00:00
[ left ( clear-all-marks ( rope-left r ) ) ]
[ right ( clear-all-marks ( rope-right r ) ) ] ) ) )
2014-12-15 06:29:48 +00:00
( define ( rope-size r )
( if r ( rope-size* r ) 0 ) )
( define ( rope-marks r )
2014-12-20 18:19:35 +00:00
( if r ( rope-marks* r ) ( seteq ) ) )
2014-12-15 06:29:48 +00:00
( define ( reindex r )
( struct-copy rope r
[ size* ( + ( rope-size ( rope-left r ) )
( rope-size ( rope-right r ) )
( strand-count ( rope-strand r ) ) ) ]
[ marks* ( set-union ( rope-marks ( rope-left r ) )
( rope-marks ( rope-right r ) )
2014-12-20 18:19:35 +00:00
( list->seteq ( hash-keys ( rope-mark-index r ) ) ) ) ] ) )
2014-12-15 06:29:48 +00:00
( define ( rope-split r0 position )
2014-12-22 19:03:02 +00:00
( match ( splay-to-pos ' rope-split r0 position )
[ ( ? rope-empty? ) ( values ( empty-rope ) ( empty-rope ) ) ]
[ ( and r ( rope t rl rr size marks mark-index ) )
;; We know the position is in the root of r.
( define-values ( lo hi ) ( rope-lo+hi r ) )
( define offset ( - position lo ) )
( define-values ( left-index right-index ) ( partition-mark-index mark-index offset ) )
( define left-strand ( substrand t 0 offset ) )
( define right-strand ( substrand t offset ) )
( values ( if ( strand-empty? left-strand )
rl
( reindex
( rope left-strand rl ( empty-rope ) ' will-be-recomputed ( seteq ) left-index ) ) )
( if ( strand-empty? right-strand )
rr
( reindex
( rope right-strand ( empty-rope ) rr ' will-be-recomputed ( seteq ) right-index ) ) ) ) ] ) )
2014-12-15 06:29:48 +00:00
( define ( partition-mark-index index offset )
2014-12-20 18:19:35 +00:00
( for*/fold [ ( l ( hasheq ) ) ( r ( hasheq ) ) ]
2014-12-20 17:52:35 +00:00
[ ( ( mtype posvals ) ( in-hash index ) )
2014-12-20 03:04:10 +00:00
( ( pos val ) ( in-hash posvals ) ) ]
2014-12-20 17:52:35 +00:00
( values ( if ( or ( < pos offset ) ( and ( = pos offset ) ( eq? ( mark-type-stickiness mtype ) ' left ) ) )
( add-mark-to-table l mtype pos val )
l )
( if ( or ( > pos offset ) ( and ( = pos offset ) ( eq? ( mark-type-stickiness mtype ) ' right ) ) )
( add-mark-to-table r mtype ( - pos offset ) val )
r ) ) ) )
2014-12-15 06:29:48 +00:00
( define ( rope-append rl0 rr0 )
( cond
2014-12-22 19:03:02 +00:00
[ ( rope-empty? rl0 ) rr0 ]
[ ( rope-empty? rr0 ) rl0 ]
2014-12-15 06:29:48 +00:00
[ else
( define-values ( _l rl ) ( splay-to rl0 find-position ( rope-size rl0 ) ) )
( define-values ( _r rr ) ( splay-to rr0 find-position 0 ) )
2014-12-22 19:03:02 +00:00
;; Both rl's right and rr's left are (empty-rope).
2014-12-15 06:29:48 +00:00
( define t ( strand-maybe-append ( rope-strand rl ) ( rope-strand rr ) ) )
( if t
( let ( ( merged-index ( merge-mark-indexes ( rope-mark-index rl )
( rope-mark-index rr )
( strand-count ( rope-strand rl ) ) ) ) )
2014-12-20 18:19:35 +00:00
( reindex ( rope t ( rope-left rl ) ( rope-right rr ) ' will-be-recomputed ( seteq ) merged-index ) ) )
2014-12-15 06:29:48 +00:00
( replace-right rl rr ) ) ] ) )
( define ( rope-concat rs )
( foldr rope-append ( empty-rope ) rs ) )
( define ( merge-mark-indexes li ri offset )
2014-12-20 03:04:10 +00:00
( for*/fold [ ( i li ) ]
2014-12-20 17:52:35 +00:00
[ ( ( mtype posvals ) ( in-hash ri ) )
2014-12-20 03:04:10 +00:00
( ( pos val ) ( in-hash posvals ) ) ]
2014-12-20 17:52:35 +00:00
( add-mark-to-table i mtype ( + pos offset ) val ) ) )
2014-12-15 06:29:48 +00:00
( define ( subrope r0 [ lo0 #f ] [ hi0 #f ] )
2014-12-23 16:47:53 +00:00
( define lo ( compute-range-index lo0 0 ( rope-size r0 ) ) )
( define hi ( compute-range-index hi0 ( rope-size r0 ) ( rope-size r0 ) ) )
2014-12-22 19:03:02 +00:00
( define-values ( _l mr ) ( rope-split r0 lo ) )
( define-values ( m _r ) ( rope-split mr ( - hi lo ) ) )
2014-12-15 06:29:48 +00:00
m )
2014-12-20 03:40:10 +00:00
( define ( rope-generator r #:forward? [ forward? #t ] )
( if forward?
( generator ( )
( let outer ( ( r r ) )
( and r
( begin ( outer ( rope-left r ) )
( match-let ( ( ( strand text offset count ) ( rope-strand r ) ) )
( do ( ( i 0 ( + i 1 ) ) )
( ( = i count ) )
( yield ( string-ref text ( + offset i ) ) ) ) )
( outer ( rope-right r ) ) ) ) ) )
( generator ( )
( let outer ( ( r r ) )
( and r
( begin ( outer ( rope-right r ) )
( match-let ( ( ( strand text offset count ) ( rope-strand r ) ) )
( do ( ( i ( - count 1 ) ( - i 1 ) ) )
( ( negative? i ) )
( yield ( string-ref text ( + offset i ) ) ) ) )
( outer ( rope-left r ) ) ) ) ) ) ) )
2014-12-20 04:45:39 +00:00
( define ( rope-seek r0 pos )
( splay-to-pos ' rope-seek r0 pos ) )
2014-12-15 06:29:48 +00:00
;; (require racket/trace)
;; (trace splay-to find-position rope-concat rope-append rope-split rope->string)
( module+ test
( require ( only-in racket/string string-append* ) )
( check-equal? ( rope-size ( empty-rope ) ) 0 )
( define ( test-with-pieces string-pieces )
( define rope-pieces ( map string->rope string-pieces ) )
( define text ( string-append* string-pieces ) )
( check-equal? ( rope->string ( car rope-pieces ) ) ( car string-pieces ) )
( check-equal? ( rope->string ( rope-concat rope-pieces ) ) text )
( check-equal? ( rope-size ( rope-concat rope-pieces ) ) ( string-length text ) )
( check-eq? ( rope-append ( empty-rope ) ( car rope-pieces ) ) ( car rope-pieces ) )
( check-eq? ( rope-append ( car rope-pieces ) ( empty-rope ) ) ( car rope-pieces ) )
( let loop ( ( n 1000 ) ( r0 ( rope-concat rope-pieces ) ) )
( when ( positive? n )
( define pos ( random ( + ( rope-size r0 ) 1 ) ) )
;; (pretty-print (list pos r0))
( define-values ( found? r ) ( splay-to r0 find-position pos ) )
( check-true found? )
( check-equal? ( rope->string r ) text )
( loop ( - n 1 ) r ) ) )
2014-12-20 17:52:35 +00:00
( define mtype1 ( mark-type " Mark1 " ' left ) )
( define mtype2 ( mark-type " Mark2 " ' right ) )
( let*-values ( ( ( r ) ( set-mark ( rope-concat rope-pieces ) mtype1 9 " original " ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? ( rope->string r ) text ) )
2014-12-20 18:08:54 +00:00
( ( pos val ) ( find-mark r mtype1 ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? pos 9 ) )
2014-12-20 03:04:10 +00:00
( ( _ ) ( check-equal? val " original " ) )
2014-12-20 17:52:35 +00:00
( ( r ) ( clear-mark r mtype1 pos ) )
2014-12-20 18:08:54 +00:00
( ( _ ) ( check-equal? ( find-all-marks/type r mtype1 ) ( hash ) ) )
( ( pos val ) ( find-mark r mtype1 ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-false pos ) )
2014-12-20 03:04:10 +00:00
( ( _ ) ( check-false val ) )
2014-12-20 17:52:35 +00:00
( ( r ) ( set-mark r mtype1 9 " second " ) )
2014-12-20 18:08:54 +00:00
( ( pos val ) ( find-mark r mtype1 ) )
2014-12-20 03:04:10 +00:00
( ( _ ) ( check-equal? pos 9 ) )
( ( _ ) ( check-equal? val " second " ) )
2014-12-20 17:52:35 +00:00
( ( r ) ( set-mark r mtype1 6 " first " ) )
( ( r ) ( set-mark r mtype2 6 " third " ) )
2014-12-20 18:08:54 +00:00
( ( _ ) ( check-equal? ( find-all-marks/type r mtype1 ) ( hash 6 " first " 9 " second " ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type r mtype2 ) ( hash 6 " third " ) ) )
( ( pos val ) ( find-mark r mtype1 #:forward? #f ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? pos 9 ) )
2014-12-20 03:04:10 +00:00
( ( _ ) ( check-equal? val " second " ) )
2014-12-20 18:08:54 +00:00
( ( pos val ) ( find-mark r mtype1 ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? pos 6 ) )
2014-12-20 03:04:10 +00:00
( ( _ ) ( check-equal? val " first " ) )
2014-12-20 17:52:35 +00:00
( ( l r ) ( rope-split r pos ) )
2014-12-20 18:08:54 +00:00
( ( _ ) ( check-equal? ( find-all-marks/type r mtype1 ) ( hash 3 " second " ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type l mtype1 ) ( hash 6 " first " ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type r mtype2 ) ( hash 0 " third " ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type l mtype2 ) ( hash ) ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? ( rope->string l ) ( substring text 0 6 ) ) )
( ( _ ) ( check-equal? ( rope->string r ) ( substring text 6 ( string-length text ) ) ) )
2014-12-20 18:19:35 +00:00
( ( _ ) ( check-equal? ( rope-marks l ) ( seteq mtype1 ) ) )
( ( _ ) ( check-equal? ( rope-marks r ) ( seteq mtype1 mtype2 ) ) )
2014-12-20 17:52:35 +00:00
( ( l r ) ( rope-split r 3 ) )
2014-12-20 18:08:54 +00:00
( ( _ ) ( check-equal? ( find-all-marks/type r mtype1 ) ( hash ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type l mtype1 ) ( hash 3 " second " ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type r mtype2 ) ( hash ) ) )
( ( _ ) ( check-equal? ( find-all-marks/type l mtype2 ) ( hash 0 " third " ) ) )
2014-12-15 06:29:48 +00:00
( ( _ ) ( check-equal? ( rope->string l ) ( substring text 6 9 ) ) )
( ( _ ) ( check-equal? ( rope->string r ) ( substring text 9 ( string-length text ) ) ) ) )
( void ) ) )
( define prejudice-pieces
( list " It is a truth universally acknowledged, that a single man in possession of a good fortune must be in want of a wife. \n "
" \n "
" However little known the feelings or views of such a man may be on his first entering a neighbourhood, this truth is so well fixed in the minds of the surrounding families, that he is considered as the rightful property of some one or other of their daughters. \n "
" \n "
" ``My dear Mr. Bennet,'' said his lady to him one day, ``have you heard that Netherfield Park is let at last?'' \n "
" \n "
" Mr. Bennet replied that he had not. \n " ) )
( define ( atomize-pieces pieces )
( map string ( string->list ( string-append* pieces ) ) ) )
( test-with-pieces ( list " hello " " , " " world " ) )
( test-with-pieces prejudice-pieces )
( test-with-pieces ( atomize-pieces prejudice-pieces ) )
2014-12-22 19:03:02 +00:00
( check-equal? ( call-with-values ( lambda ( ) ( rope-split ( empty-rope ) 0 ) ) list )
( list ( empty-rope ) ( empty-rope ) ) )
2014-12-23 16:44:18 +00:00
( check-equal? ( map rope->string
( call-with-values ( lambda ( ) ( rope-split ( string->rope " abc " ) 0 ) ) list ) )
( list " " " abc " ) )
( check-equal? ( map rope->string
( call-with-values ( lambda ( ) ( rope-split ( string->rope " abc " ) 2 ) ) list ) )
( list " ab " " c " ) )
( check-equal? ( map rope->string
( call-with-values ( lambda ( ) ( rope-split ( string->rope " abc " ) 3 ) ) list ) )
( list " abc " " " ) )
2014-12-23 16:47:53 +00:00
( check-equal? ( map ( lambda ( i ) ( compute-range-index i ' default 10 ) )
( list 0 10 3 -1 -2 11 12 -8 -9 -10 -11 -12 ) )
( list 0 10 3 9 8 10 10 2 1 0 0 0 ) )
2014-12-15 06:29:48 +00:00
)