module Control.Monad.Zombie (Zombie(..)
, liftZ
, embalm
, disembalm
, hoistZombie
) where
import Control.Applicative
import Control.Arrow
import Control.Monad
import Control.Monad.Skeleton
import Control.Monad.Skeleton.Internal
import Prelude hiding (id, (.))
data Zombie t a where
Sunlight :: Zombie t a
ReturnZ :: a -> Zombie t a -> Zombie t a
BindZ :: t x -> Cat (Kleisli (Zombie t)) x a -> Zombie t a -> Zombie t a
instance Functor (Zombie t) where
fmap = liftM
instance Applicative (Zombie t) where
pure = return
(<*>) = ap
(*>) = (>>)
instance Alternative (Zombie t) where
empty = Sunlight
Sunlight <|> ys = ys
ReturnZ x xs <|> ys = ReturnZ x (xs <|> ys)
BindZ x c xs <|> ys = BindZ x c (xs <|> ys)
instance Monad (Zombie t) where
return a = ReturnZ a Sunlight
Sunlight >>= _ = Sunlight
ReturnZ a xs >>= k = k a <|> (xs >>= k)
BindZ x c xs >>= k = BindZ x (c |> Kleisli k) (xs >>= k)
instance MonadPlus (Zombie t) where
mzero = empty
mplus = (<|>)
liftZ :: t a -> Zombie t a
liftZ t = embalm (t :>>= return)
embalm :: MonadView t (Zombie t) a -> Zombie t a
embalm (Return x) = ReturnZ x Sunlight
embalm (x :>>= k) = BindZ x (Leaf $ Kleisli k) Sunlight
disembalm :: Zombie t a -> [MonadView t (Zombie t) a]
disembalm Sunlight = []
disembalm (ReturnZ x xs) = Return x : disembalm xs
disembalm (BindZ x d xs) = (x :>>= disembalm_go d) : disembalm xs
disembalm_go :: Cat (Kleisli (Zombie t)) a b -> a -> Zombie t b
disembalm_go c a = viewL c (\(Kleisli k) -> k a) $ \(Kleisli k) c' -> disembalm_go2 (k a) c'
disembalm_go2 :: Zombie t a -> Cat (Kleisli (Zombie t)) a b -> Zombie t b
disembalm_go2 x c = case x of
Sunlight -> Sunlight
ReturnZ a xs -> disembalm_go c a <|> disembalm_go2 xs c
BindZ t c' xs -> BindZ t (Tree c' c) $ disembalm_go2 xs c
hoistZombie :: forall s t a. (forall x. s x -> t x) -> Zombie s a -> Zombie t a
hoistZombie f = go where
go :: forall x. Zombie s x -> Zombie t x
go Sunlight = Sunlight
go (ReturnZ x xs) = ReturnZ x (go xs)
go (BindZ x c xs) = BindZ (f x) (transCat (transKleisli go) c) (go xs)