module Bio.Bam.Regions where
import Bio.Bam.Header ( Refseq(..) )
import Data.List ( foldl' )
import qualified Data.IntMap as IM
import Prelude
data Region = Region { refseq :: !Refseq, start :: !Int, end :: !Int }
deriving (Eq, Ord, Show)
newtype Regions = Regions (IM.IntMap Subsequence) deriving Show
newtype Subsequence = Subsequence (IM.IntMap Int) deriving Show
toList :: Regions -> [(Refseq, Subsequence)]
toList (Regions m) = [ (Refseq $ fromIntegral k, v) | (k,v) <- IM.toList m ]
fromList :: [Region] -> Regions
fromList = foldl' (flip add) (Regions IM.empty)
add :: Region -> Regions -> Regions
add (Region (Refseq r) b e) (Regions m) =
let single = Just . Subsequence $ IM.singleton b e
in Regions $ IM.alter (maybe single (Just . addInt b e)) (fromIntegral r) m
addInt :: Int -> Int -> Subsequence -> Subsequence
addInt b e (Subsequence m0) = Subsequence $ merge_into b e m0
where
merge_into x y m = case lookupLT y m of
Just (u,v) | x < u && y <= v -> merge_into x v $ IM.delete u m
| x < u -> merge_into x y $ IM.delete u m
| y <= v -> m
| x <= v -> merge_into u y $ IM.delete u m
_ -> IM.insert x y m
overlaps :: Int -> Int -> Subsequence -> Bool
overlaps b e (Subsequence m) = case lookupLT e m of
Just (_,v) -> b < v
Nothing -> False
lookupLT :: IM.Key -> IM.IntMap a -> Maybe (IM.Key, a)
lookupLT k m | IM.null m1 = Nothing
| otherwise = Just $ IM.findMax m1
where (m1,_) = IM.split k m