{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE CPP #-}
module Control.Monad.Class.Choice
  ( MonadChoice (..)
  , chooseM
  )
  where
import Control.Monad
  ( join
  )
import Control.Monad.Trans
  ( lift
  )
import Control.Monad.Trans.Accum
  ( AccumT
  )
import Control.Monad.Trans.Cont
  ( ContT
  )
import Control.Monad.Trans.Except
  ( ExceptT
  )
import Control.Monad.Trans.Identity
  ( IdentityT
  )
import Control.Monad.Trans.Maybe
  ( MaybeT
  )
import Control.Monad.Trans.Reader
  ( ReaderT
  )
#if MIN_VERSION_transformers(0,5,6)
import Control.Monad.Trans.RWS.CPS as RWS.CPS
  ( RWST
  )
#endif
import Control.Monad.Trans.RWS.Lazy as RWS.Lazy
  ( RWST
  )
import Control.Monad.Trans.RWS.Strict as RWS.Strict
  ( RWST
  )
import Control.Monad.Trans.Select
  ( SelectT
  )
import Control.Monad.Trans.State.Lazy as State.Lazy
  ( StateT
  )
import Control.Monad.Trans.State.Strict as State.Strict
  ( StateT
  )
#if MIN_VERSION_transformers(0,5,6)
import Control.Monad.Trans.Writer.CPS as Writer.CPS
  ( WriterT
  )
#endif
import Control.Monad.Trans.Writer.Lazy as Writer.Lazy
  ( WriterT
  )
import Control.Monad.Trans.Writer.Strict as Writer.Strict
  ( WriterT
  )
class Monad m => MonadChoice f m where
  
  choose :: f a -> m a
chooseM ::
  ( MonadChoice f m
  )
    => f (m a) -> m a
chooseM = join . choose
instance MonadChoice f m => MonadChoice f (MaybeT m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (ExceptT e m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (IdentityT m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (State.Lazy.StateT s m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (State.Strict.StateT s m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (ReaderT r m) where
  choose = lift . choose
#if MIN_VERSION_transformers(0,5,6)
instance MonadChoice f m => MonadChoice f (Writer.CPS.WriterT w m) where
  choose = lift . choose
#endif
instance (Monoid w, MonadChoice f m) => MonadChoice f (Writer.Lazy.WriterT w m) where
  choose = lift . choose
instance (Monoid w, MonadChoice f m) => MonadChoice f (Writer.Strict.WriterT w m) where
  choose = lift . choose
instance (Monoid w, MonadChoice f m) => MonadChoice f (AccumT w m) where
  choose = lift . choose
instance MonadChoice f m => MonadChoice f (ContT r m) where
  choose = lift . choose
instance (Monoid w, MonadChoice f m) => MonadChoice f (RWS.Lazy.RWST r w s m) where
  choose = lift . choose
instance (Monoid w, MonadChoice f m) => MonadChoice f (RWS.Strict.RWST r w s m) where
  choose = lift . choose
#if MIN_VERSION_transformers(0,5,6)
instance MonadChoice f m => MonadChoice f (RWS.CPS.RWST r w s m) where
  choose = lift . choose
#endif
instance MonadChoice f m => MonadChoice f (SelectT r m) where
  choose = lift . choose