{-# LANGUAGE CPP #-}
module Control.Monad.Trans.MultiState.Strict
(
MultiStateT(..)
, MultiStateTNull
, MultiState
, MonadMultiGet(..)
, MonadMultiState(..)
, runMultiStateT
, runMultiStateTAS
, runMultiStateTSA
, runMultiStateTA
, runMultiStateTS
, runMultiStateT_
, runMultiStateTNil
, runMultiStateTNil_
, withMultiState
, withMultiStateAS
, withMultiStateSA
, withMultiStateA
, withMultiStateS
, withMultiState_
, withMultiStates
, withMultiStatesAS
, withMultiStatesSA
, withMultiStatesA
, withMultiStatesS
, withMultiStates_
, withoutMultiState
, inflateState
, inflateReader
, inflateWriter
, mapMultiStateT
, mGetRaw
, mPutRaw
) where
import Data.HList.HList
import Data.HList.ContainsType
import Control.Monad.State.Strict ( StateT(..)
, MonadState(..)
, evalStateT
, execStateT
, mapStateT )
import Control.Monad.Reader ( ReaderT(..) )
import Control.Monad.Writer.Strict ( WriterT(..) )
import Control.Monad.Trans.Class ( MonadTrans
, lift )
import Control.Monad.Writer.Class ( MonadWriter
, listen
, tell
, writer
, pass )
import Control.Monad.Trans.MultiState.Class
import Data.Functor.Identity ( Identity )
import Control.Applicative ( Applicative(..)
, Alternative(..)
)
import Control.Monad ( MonadPlus(..)
, liftM
, ap
, void )
import Control.Monad.Base ( MonadBase(..)
, liftBaseDefault
)
import Control.Monad.Trans.Control ( MonadTransControl(..)
, MonadBaseControl(..)
, ComposeSt
, defaultLiftBaseWith
, defaultRestoreM
)
import Data.Monoid ( Monoid )
import Control.Monad.Fix ( MonadFix(..) )
import Control.Monad.IO.Class ( MonadIO(..) )
newtype MultiStateT x m a = MultiStateT {
runMultiStateTRaw :: StateT (HList x) m a
}
type MultiStateTNull = MultiStateT '[]
type MultiState x = MultiStateT x Identity
instance (Functor f) => Functor (MultiStateT x f) where
fmap f = MultiStateT . fmap f . runMultiStateTRaw
instance (Applicative m, Monad m) => Applicative (MultiStateT x m) where
pure = MultiStateT . pure
(<*>) = ap
instance Monad m => Monad (MultiStateT x m) where
return = MultiStateT . return
k >>= f = MultiStateT $ runMultiStateTRaw k >>= (runMultiStateTRaw.f)
instance MonadTrans (MultiStateT x) where
lift = MultiStateT . lift
#if MIN_VERSION_base(4,8,0)
instance {-# OVERLAPPING #-} (Monad m, ContainsType a c)
#else
instance (Monad m, ContainsType a c)
#endif
=> MonadMultiGet a (MultiStateT c m) where
mGet = MultiStateT $ liftM getHListElem get
#if MIN_VERSION_base(4,8,0)
instance {-# OVERLAPPING #-} (Monad m, ContainsType a c)
#else
instance (Monad m, ContainsType a c)
#endif
=> MonadMultiState a (MultiStateT c m) where
mSet v = MultiStateT $ get >>= put . setHListElem v
instance MonadFix m => MonadFix (MultiStateT s m) where
mfix f = MultiStateT $ mfix (runMultiStateTRaw . f)
mGetRaw :: Monad m => MultiStateT a m (HList a)
mGetRaw = MultiStateT get
mPutRaw :: Monad m => HList s -> MultiStateT s m ()
mPutRaw = MultiStateT . put
mapMultiStateT :: (m (a, HList w) -> m' (a', HList w))
-> MultiStateT w m a
-> MultiStateT w m' a'
mapMultiStateT f = MultiStateT . mapStateT f . runMultiStateTRaw
runMultiStateT :: Functor m => HList s -> MultiStateT s m a -> m (a, HList s)
runMultiStateTAS :: Functor m => HList s -> MultiStateT s m a -> m (a, HList s)
runMultiStateTSA :: Monad m => HList s -> MultiStateT s m a -> m (HList s, a)
runMultiStateTA :: Monad m => HList s -> MultiStateT s m a -> m a
runMultiStateTS :: Monad m => HList s -> MultiStateT s m a -> m (HList s)
runMultiStateT_ :: Functor m => HList s -> MultiStateT s m a -> m ()
runMultiStateT s k = runMultiStateTAS s k
runMultiStateTAS s k = runStateT (runMultiStateTRaw k) s
runMultiStateTSA s k = (\(a,b) -> (b,a)) `liftM` runStateT (runMultiStateTRaw k) s
runMultiStateTA s k = evalStateT (runMultiStateTRaw k) s
runMultiStateTS s k = execStateT (runMultiStateTRaw k) s
runMultiStateT_ s k = void $ runStateT (runMultiStateTRaw k) s
runMultiStateTNil :: Monad m => MultiStateT '[] m a -> m a
runMultiStateTNil_ :: Functor m => MultiStateT '[] m a -> m ()
runMultiStateTNil k = evalStateT (runMultiStateTRaw k) HNil
runMultiStateTNil_ k = void $ runStateT (runMultiStateTRaw k) HNil
withMultiState :: Monad m => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m (a, s)
withMultiStateAS :: Monad m => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m (a, s)
withMultiStateSA :: Monad m => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m (s, a)
withMultiStateA :: Monad m => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m a
withMultiStateS :: Monad m => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m s
withMultiState_ :: (Functor m, Monad m) => s -> MultiStateT (s ': ss) m a -> MultiStateT ss m ()
withMultiState = withMultiStateAS
withMultiStateAS x k = MultiStateT $ do
s <- get
(a, s') <- lift $ runStateT (runMultiStateTRaw k) (x :+: s)
case s' of x' :+: sr' -> do put sr'; return (a, x')
withMultiStateSA s k = (\(a,b) -> (b,a)) `liftM` withMultiStateAS s k
withMultiStateA s k = fst `liftM` withMultiStateAS s k
withMultiStateS s k = snd `liftM` withMultiStateAS s k
withMultiState_ s k = void $ withMultiStateAS s k
withMultiStates :: Monad m => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m (a, HList s1)
withMultiStatesAS :: Monad m => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m (a, HList s1)
withMultiStatesSA :: Monad m => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m (HList s1, a)
withMultiStatesA :: Monad m => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m a
withMultiStatesS :: Monad m => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m (HList s1)
withMultiStates_ :: (Functor m, Monad m) => HList s1 -> MultiStateT (Append s1 s2) m a -> MultiStateT s2 m ()
withMultiStates = withMultiStatesAS
withMultiStatesAS HNil = liftM (\r -> (r, HNil))
withMultiStatesAS (x :+: xs) = liftM (\((a, x'), xs') -> (a, x' :+: xs'))
. withMultiStatesAS xs
. withMultiStateAS x
withMultiStatesSA HNil = liftM (\r -> (HNil, r))
withMultiStatesSA (x :+: xs) = liftM (\((a, x'), xs') -> (x' :+: xs', a))
. withMultiStatesAS xs
. withMultiStateAS x
withMultiStatesA HNil = id
withMultiStatesA (x :+: xs) = withMultiStatesA xs . withMultiStateA x
withMultiStatesS HNil = liftM (const HNil)
withMultiStatesS (x :+: xs) = liftM (\(x', xs') -> x' :+: xs')
. withMultiStatesAS xs
. withMultiStateS x
withMultiStates_ HNil = liftM (const ())
withMultiStates_ (x :+: xs) = withMultiStates_ xs . withMultiState_ x
withoutMultiState :: (Functor m, Monad m) => MultiStateT ss m a -> MultiStateT (s ': ss) m a
withoutMultiState k = MultiStateT $ get >>= \case
s :+: sr -> do
(a, sr') <- lift $ runMultiStateT sr k
put (s :+: sr')
return a
inflateState :: (Monad m, ContainsType s ss)
=> StateT s m a
-> MultiStateT ss m a
inflateState k = do
s <- mGet
(x, s') <- lift $ runStateT k s
mSet s'
return x
inflateReader :: (Monad m, ContainsType r ss)
=> ReaderT r m a
-> MultiStateT ss m a
inflateReader k = mGet >>= lift . runReaderT k
inflateWriter :: (Monad m, ContainsType w ss, Monoid w)
=> WriterT w m a
-> MultiStateT ss m a
inflateWriter k = do
(x, w) <- lift $ runWriterT k
mSet w
return x
instance (MonadState s m) => MonadState s (MultiStateT c m) where
put = lift . put
get = lift $ get
state = lift . state
instance (MonadWriter w m) => MonadWriter w (MultiStateT c m) where
writer = lift . writer
tell = lift . tell
listen = MultiStateT .
mapStateT (liftM (\((a,w), w') -> ((a, w'), w)) . listen) .
runMultiStateTRaw
pass = MultiStateT .
mapStateT (pass . liftM (\((a, f), w) -> ((a, w), f))) .
runMultiStateTRaw
instance MonadIO m => MonadIO (MultiStateT c m) where
liftIO = lift . liftIO
instance (Functor m, Applicative m, MonadPlus m) => Alternative (MultiStateT s m) where
empty = lift mzero
MultiStateT m <|> MultiStateT n = MultiStateT $ m <|> n
instance MonadPlus m => MonadPlus (MultiStateT s m) where
mzero = MultiStateT $ mzero
MultiStateT m `mplus` MultiStateT n = MultiStateT $ m `mplus` n
instance MonadBase b m => MonadBase b (MultiStateT s m) where
liftBase = liftBaseDefault
instance MonadTransControl (MultiStateT s) where
type StT (MultiStateT s) a = (a, HList s)
liftWith f = MultiStateT $ liftWith $ \s -> f $ \r -> s $ runMultiStateTRaw r
restoreT = MultiStateT . restoreT
instance MonadBaseControl b m => MonadBaseControl b (MultiStateT s m) where
type StM (MultiStateT s m) a = ComposeSt (MultiStateT s) m a
liftBaseWith = defaultLiftBaseWith
restoreM = defaultRestoreM