module Control.Joint.Effects.Reader where

import Control.Joint.Operators ((<$$>), (<**>))
import Control.Joint.Abilities.Interpreted (Interpreted (Primary, run))
import Control.Joint.Abilities.Transformer (Transformer (build, unite), Schema, (:>) (T))
import Control.Joint.Abilities.Adaptable (Adaptable (adapt))
import Control.Joint.Schemes (TU (TU))

newtype Reader e a = Reader (e -> a)

instance Functor (Reader e) where
        fmap f (Reader g) = Reader (f . g)

instance Applicative (Reader e) where
        pure = Reader . const
        Reader f <*> Reader g = Reader $ \e -> f e (g e)

instance Monad (Reader e) where
        Reader g >>= f = Reader $ \e -> run (f (g e)) e

instance Interpreted (Reader e) where
        type Primary (Reader e) a = (->) e a
        run (Reader x) = x

type instance Schema (Reader e) = TU ((->) e)

instance Transformer (Reader e) where
        build x = T. TU $ pure <$> run x
        unite = T . TU

instance Functor u => Functor (TU ((->) e) u) where
        fmap f (TU x) = TU $ f <$$> x

instance Applicative u => Applicative (TU ((->) e) u) where
        pure = TU . pure . pure
        TU f <*> TU x = TU $ f <**> x

instance (Applicative u, Monad u) => Monad (TU ((->) e) u) where
        TU x >>= f = TU $ \e -> x e >>= ($ e) . run . f

type Configured e = Adaptable (Reader e)

get :: Configured e t => t e
get = adapt $ Reader id