module Data.ReactiveValue where
import Control.Monad
import Control.GFunctor
import Data.Functor.Contravariant
class ReactiveValueRead a b m | a -> b, a -> m where
reactiveValueOnCanRead :: a -> m () -> m ()
reactiveValueRead :: a -> m b
class ReactiveValueWrite a b m | a -> b, a -> m where
reactiveValueWrite :: a -> b -> m ()
class (ReactiveValueRead a b m, ReactiveValueWrite a b m) => ReactiveValueReadWrite a b m
infix 9 =:=
infix 9 =:>
infix 9 <:=
(=:>) :: Monad m => (ReactiveValueRead a b m, ReactiveValueWrite c b m) => a -> c -> m ()
(=:>) v1 v2 = reactiveValueOnCanRead v1 sync1
where sync1 = reactiveValueRead v1 >>= reactiveValueWrite v2
(<:=) :: Monad m => (ReactiveValueRead a b m, ReactiveValueWrite c b m) => c -> a -> m ()
(<:=) v2 v1 = reactiveValueOnCanRead v1 sync1
where sync1 = reactiveValueRead v1 >>= reactiveValueWrite v2
(=:=) :: Monad m => (ReactiveValueReadWrite a b m, ReactiveValueReadWrite c b m) => a -> c -> m ()
(=:=) v1 v2 = do
v1 =:> v2
v1 <:= v2
type FieldGetter m a = m a
type FieldSetter m a = a -> m ()
type FieldNotifier m a = m () -> m ()
data ReactiveFieldRead m a = ReactiveFieldRead (FieldGetter m a) (FieldNotifier m a)
data ReactiveFieldWrite m a = ReactiveFieldWrite (FieldSetter m a)
data ReactiveFieldReadWrite m a = ReactiveFieldReadWrite (FieldSetter m a) (FieldGetter m a) (FieldNotifier m a)
instance ReactiveValueRead (ReactiveFieldRead m a) a m where
reactiveValueOnCanRead (ReactiveFieldRead _ notifier) = notifier
reactiveValueRead (ReactiveFieldRead getter _) = getter
instance ReactiveValueWrite (ReactiveFieldWrite m a) a m where
reactiveValueWrite (ReactiveFieldWrite setter) = setter
instance ReactiveValueRead (ReactiveFieldReadWrite m a) a m where
reactiveValueOnCanRead (ReactiveFieldReadWrite _ _ notifier) = notifier
reactiveValueRead (ReactiveFieldReadWrite _ getter _) = getter
instance ReactiveValueWrite (ReactiveFieldReadWrite m a) a m where
reactiveValueWrite (ReactiveFieldReadWrite setter _ _) = setter
instance ReactiveValueReadWrite (ReactiveFieldReadWrite m a) a m
type ReactiveFieldActivatable m = ReactiveFieldRead m ()
mkActivatable :: Monad m => (m () -> m ()) -> ReactiveFieldActivatable m
mkActivatable f = ReactiveFieldRead getter notifier
where getter = return ()
notifier = f
class ReactiveValueActivatable m a where
defaultActivation :: a -> ReactiveFieldActivatable m
constR :: Monad m => a -> ReactiveFieldRead m a
constR e = ReactiveFieldRead getter notifier
where notifier _ = return ()
getter = return e
initRW :: Monad m => a -> ReactiveFieldRead m a
initRW e = ReactiveFieldRead getter notifier
where notifier _ = return ()
getter = return e
liftR :: (Monad m, ReactiveValueRead a b m) => (b -> c) -> a -> ReactiveFieldRead m c
liftR f e = ReactiveFieldRead getter notifier
where notifier = reactiveValueOnCanRead e
getter = liftM f (reactiveValueRead e)
(<^>) :: (Monad m, ReactiveValueRead a b m) => (b -> c) -> a -> ReactiveFieldRead m c
(<^>) = liftR
liftR2 :: (Monad m, ReactiveValueRead a b m, ReactiveValueRead c d m)
=> (b -> d -> e) -> a -> c -> ReactiveFieldRead m e
liftR2 f e1 e2 = ReactiveFieldRead getter notifier
where getter = do v1 <- reactiveValueRead e1
v2 <- reactiveValueRead e2
return (f v1 v2)
notifier p = do reactiveValueOnCanRead e1 p
reactiveValueOnCanRead e2 p
liftR3 :: ( Monad m, ReactiveValueRead a b m, ReactiveValueRead c d m
, ReactiveValueRead e f m)
=> (b -> d -> f -> g) -> a -> c -> e -> ReactiveFieldRead m g
liftR3 f e1 e2 e3 = ReactiveFieldRead getter notifier
where getter = do v1 <- reactiveValueRead e1
v2 <- reactiveValueRead e2
v3 <- reactiveValueRead e3
return (f v1 v2 v3)
notifier p = do reactiveValueOnCanRead e1 p
reactiveValueOnCanRead e2 p
reactiveValueOnCanRead e3 p
liftMR :: (Monad m, ReactiveValueRead a b m) => (b -> m c) -> a -> ReactiveFieldRead m c
liftMR f e = ReactiveFieldRead getter notifier
where notifier = reactiveValueOnCanRead e
getter = f =<< reactiveValueRead e
constW :: (Monad m, ReactiveValueWrite v a m) => a -> v -> ReactiveFieldWrite m b
constW c v = ReactiveFieldWrite $ \_ -> reactiveValueWrite v c
liftW :: (Monad m, ReactiveValueWrite a b m)
=> (c -> b) -> a -> ReactiveFieldWrite m c
liftW f e = ReactiveFieldWrite setter
where setter = reactiveValueWrite e . f
liftW2 :: (Monad m, ReactiveValueWrite a b m, ReactiveValueWrite d e m)
=> (c -> (b,e)) -> a -> d -> ReactiveFieldWrite m c
liftW2 f e1 e2 = ReactiveFieldWrite setter
where setter x = do let (v1,v2) = f x
reactiveValueWrite e1 v1
reactiveValueWrite e2 v2
(&.&) :: (Monad m, ReactiveValueWrite a b m, ReactiveValueWrite c b m)
=> a -> c -> ReactiveFieldWrite m b
(&.&) v1 v2 = ReactiveFieldWrite $ \x -> do
reactiveValueWrite v1 x
reactiveValueWrite v2 x
liftMW :: (Monad m, ReactiveValueWrite a b m)
=> (c -> m b) -> a -> ReactiveFieldWrite m c
liftMW f e = ReactiveFieldWrite setter
where setter x = reactiveValueWrite e =<< f x
readOnly :: ReactiveValueRead r a m => r -> ReactiveFieldRead m a
readOnly r = ReactiveFieldRead (reactiveValueRead r) (reactiveValueOnCanRead r)
writeOnly :: ReactiveValueWrite r a m => r -> ReactiveFieldWrite m a
writeOnly r = ReactiveFieldWrite (reactiveValueWrite r)
wrapMW :: (a -> m ()) -> ReactiveFieldWrite m a
wrapMW f = ReactiveFieldWrite f
wrapDo :: m () -> ReactiveFieldWrite m a
wrapDo f = wrapMW (const f)
wrapDo_ :: m () -> ReactiveFieldWrite m ()
wrapDo_ f = wrapMW (\() -> f)
wrapMR :: m a -> (m () -> m ()) -> ReactiveFieldRead m a
wrapMR f p = ReactiveFieldRead f p
wrapMRPassive :: Monad m => m a -> ReactiveFieldRead m a
wrapMRPassive f = ReactiveFieldRead f (const (return ()))
eventR :: Monad m => (m () -> m ()) -> ReactiveFieldRead m ()
eventR notifInstaller = ReactiveFieldRead (return ()) notifInstaller
newtype BijectiveFunc a b = BijectiveFunc
{ unBijectiveFunc :: (a -> b, b -> a) }
bijection :: (a -> b, b -> a) -> BijectiveFunc a b
bijection = BijectiveFunc
direct :: BijectiveFunc a b -> (a -> b)
direct = fst . unBijectiveFunc
inverse :: BijectiveFunc a b -> (b -> a)
inverse = snd . unBijectiveFunc
type Involution a = BijectiveFunc a a
involution :: (a -> a) -> Involution a
involution f = BijectiveFunc (f, f)
liftRW :: (Monad m, ReactiveValueReadWrite a b m)
=> BijectiveFunc b c -> a -> ReactiveFieldReadWrite m c
liftRW (BijectiveFunc (f1, f2)) e = ReactiveFieldReadWrite setter getter notifier
where ReactiveFieldRead getter notifier = liftR f1 e
ReactiveFieldWrite setter = liftW f2 e
liftRW2 :: (Monad m, ReactiveValueReadWrite a b m, ReactiveValueReadWrite c d m)
=> BijectiveFunc e (b,d) -> a -> c -> ReactiveFieldReadWrite m e
liftRW2 (BijectiveFunc (f1, f2)) e1 e2 = ReactiveFieldReadWrite setter getter notifier
where ReactiveFieldRead getter notifier = liftR2 (curry f2) e1 e2
ReactiveFieldWrite setter = liftW2 f1 e1 e2
pairRW :: (Monad m,
ReactiveValueReadWrite a b m,
ReactiveValueReadWrite c d m)
=> a -> c -> ReactiveFieldReadWrite m (b, d)
pairRW a b = liftRW2 (bijection (id, id)) a b
eqCheck :: (Eq v, Monad m) => ReactiveFieldReadWrite m v -> ReactiveFieldReadWrite m v
eqCheck (ReactiveFieldReadWrite setter getter notifier) = ReactiveFieldReadWrite setter' getter notifier
where setter' v = do o <- getter
when (o /= v) $ setter v
modRW :: (Monad m, ReactiveValueReadWrite a b m)
=> (b -> c -> b) -> a -> ReactiveFieldWrite m c
modRW f rv = ReactiveFieldWrite setter
where setter c = do b <- reactiveValueRead rv
let b' = f b c
reactiveValueWrite rv b'
reactiveValueModify :: (Monad m, ReactiveValueReadWrite a b m) => a -> (b -> b) -> m ()
reactiveValueModify r f = reactiveValueWrite r . f =<< reactiveValueRead r
lMerge :: (Monad m, ReactiveValueRead a v m, ReactiveValueRead b v m)
=> a -> b -> ReactiveFieldRead m v
lMerge = liftR2 (\a _ -> a)
rMerge :: (Monad m, ReactiveValueRead a v m, ReactiveValueRead b v m)
=> a -> b -> ReactiveFieldRead m v
rMerge = liftR2 (\_ b -> b)
passivelyR :: (Monad m, ReactiveValueRead a b m)
=> a -> ReactiveFieldRead m b
passivelyR rv =
ReactiveFieldRead (reactiveValueRead rv) (\_ -> return ())
passivelyRW :: (Monad m, ReactiveValueReadWrite a b m)
=> a -> ReactiveFieldReadWrite m b
passivelyRW rv =
ReactiveFieldReadWrite (reactiveValueWrite rv) (reactiveValueRead rv) (\_ -> return ())
governingR :: (ReactiveValueRead a b m, ReactiveValueRead c d m)
=> a -> c -> ReactiveFieldRead m d
governingR r c = ReactiveFieldRead getter notifier
where getter = reactiveValueRead c
notifier = reactiveValueOnCanRead r
ifRW_ :: (Monad m, ReactiveValueRead c Bool m, ReactiveValueReadWrite v a m)
=> c -> v
-> ReactiveFieldReadWrite m a
ifRW_ c r = ReactiveFieldReadWrite setter getter notifier
where setter x = reactiveValueWrite r x
getter = reactiveValueRead r
notifier p = do reactiveValueOnCanRead c (when' p)
reactiveValueOnCanRead r (when' p)
where when' m = do x <- reactiveValueRead c
when x m
ifRW :: (Monad m, ReactiveValueRead c Bool m, ReactiveValueReadWrite v a m)
=> c -> v
-> ReactiveFieldReadWrite m a
ifRW c r = ReactiveFieldReadWrite setter getter notifier
where setter x = do b <- reactiveValueRead c
when b $
reactiveValueWrite r x
getter = reactiveValueRead r
notifier p = do reactiveValueOnCanRead c (when' p)
reactiveValueOnCanRead r (when' p)
where when' m = do b <- reactiveValueRead c
when b m
guardRO :: (Monad m, ReactiveValueRead c Bool m)
=> c
-> ReactiveFieldRead m Bool
guardRO c = ReactiveFieldRead getter notifier
where getter = reactiveValueRead c
notifier p = reactiveValueOnCanRead c (when' p)
where when' m = do x <- reactiveValueRead c
when x m
guardRO' :: (Monad m, ReactiveValueRead c a m)
=> c
-> (a -> Bool)
-> ReactiveFieldRead m a
guardRO' c p = ReactiveFieldRead getter notifier
where getter = reactiveValueRead c
notifier = reactiveValueOnCanRead c . when'
where when' m = do x <- reactiveValueRead c
when (p x) m
instance (Functor m, Monad m) => Functor (ReactiveFieldRead m) where
fmap = liftR
instance (Monad m) => Contravariant (ReactiveFieldWrite m) where
contramap = liftW
instance Monad m => GFunctor (ReactiveFieldReadWrite m) BijectiveFunc where
gmap = liftRW