module Control.Monad.Generator (
GeneratorT, generate, yield, yields
) where
import Control.Applicative (Applicative(..))
import Control.Monad (liftM, ap)
import Control.Monad.Cont (Cont (..))
import Control.Monad.DList (DListT, joinDListT)
import Control.Monad.Trans (MonadTrans(..), MonadIO(..))
import Data.List.Class (cons)
import Data.Monoid (Monoid(..))
newtype GeneratorT v m a =
GeneratorT { runGeneratorT :: Cont (DListT m v) a }
instance Monad m => Functor (GeneratorT v m) where
fmap = liftM
instance Monad m => Monad (GeneratorT v m) where
return = GeneratorT . return
GeneratorT a >>= f = GeneratorT $ a >>= runGeneratorT . f
fail = lift . fail
instance Monad m => Applicative (GeneratorT v m) where
pure = return
(<*>) = ap
instance MonadTrans (GeneratorT v) where
lift = GeneratorT . Cont . (joinDListT .) . flip liftM
instance MonadIO m => MonadIO (GeneratorT v m) where
liftIO = lift . liftIO
generate :: Monad m => GeneratorT v m () -> DListT m v
generate = ($ const mempty) . runCont . runGeneratorT
mkContNil :: (r -> r) -> Cont r ()
mkContNil = Cont . (. ($ ()))
yield :: Monad m => v -> GeneratorT v m ()
yield = GeneratorT . mkContNil . cons
yields :: Monad m => DListT m v -> GeneratorT v m ()
yields = GeneratorT . mkContNil . mappend