module Control.Comonad.Trans.Store.Lazy
(
Store, store, runStore
, StoreT(..), runStoreT
, get
, put
, modify
, experiment
) where
import Control.Comonad
import Control.Comonad.Hoist.Class
import Control.Comonad.Trans.Class
import Data.Functor.Identity
type Store s = StoreT s Identity
store :: (s -> a) -> s -> Store s a
store f s = StoreT (Identity f) s
runStore :: Store s a -> (s -> a, s)
runStore ~(StoreT (Identity f) s) = (f, s)
data StoreT s w a = StoreT (w (s -> a)) s
runStoreT :: StoreT s w a -> (w (s -> a), s)
runStoreT ~(StoreT wf s) = (wf, s)
instance Functor w => Functor (StoreT s w) where
fmap f ~(StoreT wf s) = StoreT (fmap (f .) wf) s
instance Comonad w => Comonad (StoreT s w) where
extract ~(StoreT wf s) = extract wf s
duplicate ~(StoreT wf s) = StoreT (extend StoreT wf) s
extend f ~(StoreT wf s) = StoreT (extend (\wf' s' -> f (StoreT wf' s')) wf) s
instance ComonadTrans (StoreT s) where
lower ~(StoreT f s) = fmap ($s) f
instance ComonadHoist (StoreT s) where
cohoist ~(StoreT f s) = StoreT (Identity (extract f)) s
get :: StoreT s w a -> s
get ~(StoreT _ s) = s
put :: Comonad w => s -> StoreT s w a -> a
put s ~(StoreT f _) = extract f s
modify :: Comonad w => (s -> s) -> StoreT s w a -> a
modify f ~(StoreT g s) = extract g (f s)
experiment :: (Comonad w, Functor f) => f (s -> s) -> StoreT s w a -> f a
experiment fs ~(StoreT g s) = fmap (\f -> extract g (f s)) fs