module Control.Monad.Representable.Reader
(
Reader
, runReader
, ReaderT(..), readerT, runReaderT
, MonadReader(..)
, module Data.Functor.Rep
) where
import Control.Applicative
import Control.Comonad
import Control.Monad.Reader.Class
import Control.Monad.Writer.Class as Writer
import Control.Monad.Trans.Class
import Control.Monad.IO.Class
import Data.Distributive
import Data.Functor.Bind
import Data.Functor.Extend
import Data.Functor.Identity
import Data.Functor.Rep
import Data.Foldable
import Data.Traversable
import Data.Semigroup
import Data.Semigroup.Foldable
import Data.Semigroup.Traversable
import Prelude hiding (lookup,zipWith)
type Reader f = ReaderT f Identity
runReader :: Representable f => Reader f b -> Rep f -> b
runReader = fmap runIdentity . runReaderT
newtype ReaderT f m b = ReaderT { getReaderT :: f (m b) }
readerT :: Representable f => (Rep f -> m b) -> ReaderT f m b
readerT = ReaderT . tabulate
runReaderT :: Representable f => ReaderT f m b -> Rep f -> m b
runReaderT = index . getReaderT
instance (Functor f, Functor m) => Functor (ReaderT f m) where
fmap f = ReaderT . fmap (fmap f) . getReaderT
instance (Representable f, Representable m) => Representable (ReaderT f m) where
type Rep (ReaderT f m) = (Rep f, Rep m)
tabulate = ReaderT . tabulate . fmap tabulate . curry
index = uncurry . fmap index . index . getReaderT
instance (Representable f, Apply m) => Apply (ReaderT f m) where
ReaderT ff <.> ReaderT fa = ReaderT (unCo ((<.>) <$> Co ff <.> Co fa))
instance (Representable f, Applicative m) => Applicative (ReaderT f m) where
pure = ReaderT . pureRep . pure
ReaderT ff <*> ReaderT fa = ReaderT (unCo ((<*>) <$> Co ff <*> Co fa))
instance (Representable f, Bind m) => Bind (ReaderT f m) where
ReaderT fm >>- f = ReaderT $ tabulate (\a -> index fm a >>- flip index a . getReaderT . f)
instance (Representable f, Monad m) => Monad (ReaderT f m) where
#if __GLASGOW_HASKELL__ < 710
return = ReaderT . pureRep . return
#endif
ReaderT fm >>= f = ReaderT $ tabulate (\a -> index fm a >>= flip index a . getReaderT . f)
#if __GLASGOW_HASKELL >= 704
instance (Representable f, Monad m, Rep f ~ e) => MonadReader e (ReaderT f m) where
ask = ReaderT (tabulate return)
local f m = readerT $ \r -> runReaderT m (f r)
#if MIN_VERSION_transformers(0,3,0)
reader = readerT . fmap return
#endif
#endif
instance Representable f => MonadTrans (ReaderT f) where
lift = ReaderT . pureRep
instance (Representable f, Distributive m) => Distributive (ReaderT f m) where
distribute = ReaderT . fmapRep distribute . unCo . collect (Co . getReaderT)
instance (Representable f, Representable m, Semigroup (Rep f), Semigroup (Rep m)) => Extend (ReaderT f m) where
extended = extendedRep
duplicated = duplicatedRep
instance (Representable f, Representable m, Monoid (Rep f), Monoid (Rep m)) => Comonad (ReaderT f m) where
extend = extendRep
duplicate = duplicateRep
extract = extractRep
instance (Representable f, MonadIO m) => MonadIO (ReaderT f m) where
liftIO = lift . liftIO
instance (Representable f, MonadWriter w m) => MonadWriter w (ReaderT f m) where
tell = lift . tell
listen (ReaderT m) = ReaderT $ tabulate $ Writer.listen . index m
pass (ReaderT m) = ReaderT $ tabulate $ Writer.pass . index m
instance (Foldable f, Foldable m) => Foldable (ReaderT f m) where
foldMap f = foldMap (foldMap f) . getReaderT
instance (Foldable1 f, Foldable1 m) => Foldable1 (ReaderT f m) where
foldMap1 f = foldMap1 (foldMap1 f) . getReaderT
instance (Traversable f, Traversable m) => Traversable (ReaderT f m) where
traverse f = fmap ReaderT . traverse (traverse f) . getReaderT
instance (Traversable1 f, Traversable1 m) => Traversable1 (ReaderT f m) where
traverse1 f = fmap ReaderT . traverse1 (traverse1 f) . getReaderT