module Control.Monad.Trans.List (ListT (..), toListM) where
import Control.Applicative
import Control.Arrow
import Control.Monad
import Control.Monad.Fix
import Control.Monad.Trans.Class
import Data.Functor.Classes
import Data.List.NonEmpty (NonEmpty (..), intersperse)
import Data.Maybe
import Data.Monoid hiding ((<>))
import Data.Semigroup
newtype ListT m a = ListT { runListT :: m (Maybe (a, ListT m a)) }
deriving (Functor, Foldable, Traversable)
toListM :: Monad m => ListT m a -> m [a]
toListM (ListT xm) = xm >>= \ case Nothing -> pure []
Just (x, xs) -> (x:) <$> toListM xs
instance MonadTrans ListT where lift = ListT . fmap (Just . flip (,) empty)
instance Eq1 m => Eq1 (ListT m) where
liftEq (==) (ListT x) (ListT y) = (liftEq . liftEq) (\ (x, xs) (y, ys) -> x == y && liftEq (==) xs ys) x y
instance Ord1 m => Ord1 (ListT m) where
liftCompare cmp (ListT x) (ListT y) = (liftCompare . liftCompare) (\ (x, xs) (y, ys) -> x `cmp` y <> liftCompare cmp xs ys) x y
instance Show1 m => Show1 (ListT m) where
liftShowsPrec sp sl n (ListT x) = fst (show1Methods sp sl) n x
liftShowList sp sl = snd (show1Methods sp sl) . fmap runListT
show1Methods sp sl =
(l . l) (pure f,
list id $
between '[' ']' . appEndo . intercalate (Endo (", " ++)) . fmap (Endo . f))
where l :: Show1 f => (Int -> a -> ShowS, [a] -> ShowS) -> (Int -> f a -> ShowS, [f a] -> ShowS)
l (sp, sl) = (liftShowsPrec sp sl, liftShowList sp sl)
between :: a -> a -> ([a] -> [a]) -> [a] -> [a]
between x y f = (:) x . f . (:) y
f (x, xs) = between '(' ')' $ sp 0 x . (++) ", " . liftShowsPrec sp sl 0 xs
instance (Eq a, Eq1 m) => Eq (ListT m a) where (==) = liftEq (==)
instance (Ord a, Ord1 m) => Ord (ListT m a) where compare = liftCompare compare
instance (Show a, Show1 m) => Show (ListT m a) where
showsPrec = liftShowsPrec showsPrec showList
showList = liftShowList showsPrec showList
instance Monad m => Applicative (ListT m) where
pure x = ListT . pure . Just $ (x, ListT (pure Nothing))
(<*>) = ap
instance Monad m => Alternative (ListT m) where
empty = (ListT . pure) Nothing
ListT xm <|> ys = ListT $ xm >>= \ case
Nothing -> runListT ys
Just (x, xs) -> (pure . Just) (x, xs <|> ys)
instance Monad m => Monad (ListT m) where
xm >>= f = join (f <$> xm)
where join (ListT xm) = ListT $ xm >>= \ case
Nothing -> pure Nothing
Just (ListT ym, xss) -> ym >>= \ case
Nothing -> runListT (join xss)
Just (y, ys) -> (pure . Just) (y, ys <|> join xss)
instance Monad m => MonadPlus (ListT m)
instance MonadFix m => MonadFix (ListT m) where
mfix f = ListT $ (flip fmap . mfix) (runListT . f . fst . fromJust) . fmap $
id *** (pure . mfix $
ListT <<< runListT . f >=> \ case Just (_, xs) -> runListT xs
Nothing -> error "Nothing")
intercalate :: Semigroup a => a -> NonEmpty a -> a
intercalate a = sconcat . intersperse a
list :: b -> (NonEmpty a -> b) -> [a] -> b
list x _ [] = x
list _ f (x:xs) = f (x:|xs)