module Data.Sequence.FastQueue(module Data.SequenceClass, FastQueue) where
import Control.Applicative (pure, (<$>), (<*>))
import Control.Applicative.Backwards
import Data.SequenceClass
import Data.Foldable
import Data.Traversable
import Prelude hiding (foldr,foldl)
revAppend l r = rotate l r []
rotate :: [a] -> [a]-> [a] -> [a]
rotate []  [y] r = y : r
rotate (x : f) (y : r) a = x : rotate f r (y : a)
rotate f        a     r  = error "Invariant |a| = |f| - (|r| - 1) broken"
data FastQueue a where
  RQ :: ![a] -> ![a] -> ![a] -> FastQueue a
queue :: [a] -> [a] -> [a] -> FastQueue a
queue f r [] = let f' = revAppend f r 
                 in RQ f' [] f'
queue f r (h : t) = RQ f r t
instance Functor FastQueue where
  fmap phi q = case viewl q of
     EmptyL -> empty
     h :< t -> phi h <| fmap phi t
instance Foldable FastQueue where
  foldl f = loop where
    loop i s = case viewl s of
          EmptyL -> i
          h :< t -> loop (f i h) t
  foldr f i s = foldr f i (reverse $ toRevList s)
    where toRevList s = case viewl s of
           EmptyL -> []
           h :< t -> h : toRevList t
instance Sequence FastQueue where
 empty = RQ [] [] []
 singleton x = let c = [x] in queue c [] c
 (RQ f r a) |> x = queue f (x : r) a
 viewl (RQ [] [] []) = EmptyL
 viewl (RQ (h : t) f a) = h :< queue t f a
instance Traversable FastQueue where
  sequenceA q = case viewl q of
     EmptyL -> pure empty
     h :< t  -> (<|) <$> h <*> sequenceA t