module Control.Monad.ListT (ListT(..)) where
import Data.List.Class (List(..), ListItem(..), foldrL)
import Control.Applicative (Alternative(..), Applicative(..))
import Control.Monad (MonadPlus(..), ap, liftM)
import Control.Monad.IO.Class (MonadIO(..))
import Control.Monad.Trans.Class (MonadTrans(..))
#if MIN_VERSION_base(4,9,0)
import Data.Semigroup (Semigroup(..))
#endif
import Data.Monoid (Monoid(..))
newtype ListT m a =
ListT { runListT :: m (ListItem (ListT m) a) }
deriving instance (Eq (m (ListItem (ListT m) a))) => Eq (ListT m a)
deriving instance (Ord (m (ListItem (ListT m) a))) => Ord (ListT m a)
deriving instance (Read (m (ListItem (ListT m) a))) => Read (ListT m a)
deriving instance (Show (m (ListItem (ListT m) a))) => Show (ListT m a)
foldrL' :: List l => (a -> l b -> l b) -> l b -> l a -> l b
foldrL' consFunc nilFunc =
joinL . foldrL step (return nilFunc)
where
step x = return . consFunc x . joinL
#if MIN_VERSION_base(4,9,0)
instance Monad m => Semigroup (ListT m a) where
(<>) = flip (foldrL' cons)
#endif
instance Monad m => Monoid (ListT m a) where
mempty = ListT $ return Nil
#if !(MIN_VERSION_base(4,11,0))
mappend = flip (foldrL' cons)
#endif
instance Functor m => Functor (ListT m) where
fmap func (ListT action) =
ListT (fmap f action)
where
f Nil = Nil
f (Cons x xs) = Cons (func x) (fmap func xs)
instance Monad m => Monad (ListT m) where
return = ListT . return . (`Cons` mempty)
a >>= b = foldrL' mappend mempty (fmap b a)
instance Monad m => Applicative (ListT m) where
pure = return
(<*>) = ap
instance Monad m => Alternative (ListT m) where
empty = mempty
(<|>) = mappend
instance Monad m => MonadPlus (ListT m) where
mzero = mempty
mplus = mappend
instance MonadTrans ListT where
lift = ListT . liftM (`Cons` mempty)
instance Monad m => List (ListT m) where
type ItemM (ListT m) = m
runList = runListT
joinL = ListT . (>>= runList)
cons x = ListT . return . Cons x
instance MonadIO m => MonadIO (ListT m) where
liftIO = lift . liftIO