| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | ! Copyright (C) 2008 Daniel Ehrenberg. | 
					
						
							|  |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							|  |  |  | USING: accessors arrays assocs binary-search grouping kernel | 
					
						
							|  |  |  | locals make math math.order sequences sequences.private sorting ;
 | 
					
						
							|  |  |  | IN: interval-maps | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-01 15:45:54 -04:00
										 |  |  | ! Intervals are triples of { start end value } | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | TUPLE: interval-map { array array read-only } ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : find-interval ( key interval-map -- interval-node )
 | 
					
						
							| 
									
										
										
										
											2017-06-01 15:45:54 -04:00
										 |  |  |     array>> [ first-unsafe <=> ] with search nip ; inline
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : interval-contains? ( key interval-node -- ? )
 | 
					
						
							|  |  |  |     first2-unsafe between? ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : all-intervals ( sequence -- intervals )
 | 
					
						
							|  |  |  |     [ [ dup number? [ dup 2array ] when ] dip ] { } assoc-map-as ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : disjoint? ( node1 node2 -- ? )
 | 
					
						
							| 
									
										
										
										
											2017-06-01 15:45:54 -04:00
										 |  |  |     [ second-unsafe ] [ first-unsafe ] bi* < ;
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : ensure-disjoint ( intervals -- intervals )
 | 
					
						
							|  |  |  |     dup [ disjoint? ] monotonic? | 
					
						
							|  |  |  |     [ "Intervals are not disjoint" throw ] unless ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : >intervals ( specification -- intervals )
 | 
					
						
							|  |  |  |     [ suffix ] { } assoc>map concat 3 group ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: not-an-interval-map obj ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : check-interval-map ( map -- map )
 | 
					
						
							| 
									
										
										
										
											2015-08-13 19:13:05 -04:00
										 |  |  |     dup interval-map? [ not-an-interval-map ] unless ; inline
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : interval-at* ( key map -- value ? )
 | 
					
						
							|  |  |  |     check-interval-map | 
					
						
							|  |  |  |     [ drop ] [ find-interval ] 2bi
 | 
					
						
							|  |  |  |     [ nip ] [ interval-contains? ] 2bi
 | 
					
						
							| 
									
										
										
										
											2017-06-01 15:45:54 -04:00
										 |  |  |     [ third-unsafe t ] [ drop f f ] if ; inline
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : interval-at ( key map -- value ) interval-at* drop ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : interval-key? ( key map -- ? ) interval-at* nip ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : interval-values ( map -- values )
 | 
					
						
							| 
									
										
										
										
											2017-06-01 15:45:54 -04:00
										 |  |  |     check-interval-map array>> [ third-unsafe ] map ;
 | 
					
						
							| 
									
										
										
										
											2015-06-29 19:43:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : <interval-map> ( specification -- map )
 | 
					
						
							|  |  |  |     all-intervals [ first-unsafe second-unsafe ] sort-with | 
					
						
							|  |  |  |     >intervals ensure-disjoint interval-map boa ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : <interval-set> ( specification -- map )
 | 
					
						
							|  |  |  |     dup zip <interval-map> ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | :: coalesce ( alist -- specification )
 | 
					
						
							|  |  |  |     ! Only works with integer keys, because they're discrete | 
					
						
							|  |  |  |     ! Makes 2array keys | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         alist sort-keys unclip swap [ first2 dupd ] dip
 | 
					
						
							|  |  |  |         [| oldkey oldval key val | ! Underneath is start | 
					
						
							|  |  |  |             oldkey 1 + key =
 | 
					
						
							|  |  |  |             oldval val = and
 | 
					
						
							|  |  |  |             [ oldkey 2array oldval 2array , key ] unless
 | 
					
						
							|  |  |  |             key val | 
					
						
							|  |  |  |         ] assoc-each [ 2array ] bi@ , | 
					
						
							|  |  |  |     ] { } make ;
 |