{----------------------------------------------------------------------------- reactive-banana ------------------------------------------------------------------------------} -- * Synopsis -- | Merge model and implementation into a single type. Not pretty. module Reactive.Banana.Test.Plumbing where import Control.Applicative import Control.Monad (liftM) import Control.Monad.Fix import qualified Reactive.Banana.Model as X import qualified Reactive.Banana.Internal.EventBehavior1 as Y import qualified Reactive.Banana.Internal.InputOutput as Y {----------------------------------------------------------------------------- Types as pairs ------------------------------------------------------------------------------} data Event a = E (X.Event a) (Y.Event a) data Behavior a = B (X.Behavior a) (Y.Behavior a) data Moment a = M (X.Moment a) (Y.Moment a) -- pair extractions fstE (E x _) = x; sndE (E _ y) = y fstB (B x _) = x; sndB (B _ y) = y fstM (M x _) = x; sndM (M _ y) = y -- partial embedding functions ex x = E x undefined; ey y = E undefined y bx x = B x undefined; by y = B undefined y mx x = M x undefined; my y = M undefined y -- interpretation interpretModel :: (Event a -> Moment (Event b)) -> [Maybe a] -> [Maybe b] interpretModel f = X.interpret (fmap fstE . fstM . f . ex) interpretGraph :: (Event a -> Moment (Event b)) -> [Maybe a] -> IO [Maybe b] interpretGraph f = Y.interpret (fmap sndE . sndM . f . ey) {----------------------------------------------------------------------------- Primitive combinators ------------------------------------------------------------------------------} never = E X.never Y.never filterJust (E x y) = E (X.filterJust x) (Y.filterJust y) unionWith f (E x1 y1) (E x2 y2) = E (X.unionWith f x1 x2) (Y.unionWith f y1 y2) mapE f (E x y) = E (X.mapE f x) (Y.mapE f y) applyE ~(B x1 y1) (E x2 y2) = E (X.applyE x1 x2) (Y.applyE y1 y2) accumE a (E x y) = E (X.accumE a x) (Y.accumE a y) instance Functor Event where fmap = mapE stepper = stepperB stepperB a (E x y) = B (X.stepperB a x) (Y.stepperB a y) pureB a = B (X.pureB a) (Y.pureB a) applyB (B x1 y1) (B x2 y2) = B (X.applyB x1 x2) (Y.applyB y1 y2) mapB f (B x y) = B (X.mapB f x) (Y.mapB f y) instance Functor Behavior where fmap = mapB instance Applicative Behavior where pure = pureB; (<*>) = applyB instance Functor Moment where fmap = liftM instance Monad Moment where return a = M (return a) (return a) (M x y) >>= g = M (x >>= fstM . g) (y >>= sndM . g) instance MonadFix Moment where mfix f = M (mfix fx) (mfix fy) where fx a = let M x _ = f a in x fy a = let M _ y = f a in y trimE :: Event a -> Moment (Moment (Event a)) trimE (E x y) = M (fmap (fmap ex . mx) $ X.trimE x) (fmap (fmap ey . my) $ Y.trimE y) trimB :: Behavior a -> Moment (Moment (Behavior a)) trimB (B x y) = M (fmap (fmap bx . mx) $ X.trimB x) (fmap (fmap by . my) $ Y.trimB y) initialB ~(B x y) = M (X.initialB x) (Y.initialB y) observeE :: Event (Moment a) -> Event a observeE (E x y) = E (X.observeE $ X.mapE fstM x) (Y.observeE $ Y.mapE sndM y) switchE :: Event (Moment (Event a)) -> Event a switchE (E x y) = E (X.switchE $ X.mapE (fstM . fmap fstE) x) (Y.switchE $ Y.mapE (sndM . fmap sndE) y) switchB :: Behavior a -> Event (Moment (Behavior a)) -> Behavior a switchB (B x y) (E xe ye) = B (X.switchB x $ X.mapE (fstM . fmap fstB) xe) (Y.switchB y $ Y.mapE (sndM . fmap sndB) ye) {----------------------------------------------------------------------------- Derived combinators ------------------------------------------------------------------------------} accumB acc = stepperB acc . accumE acc whenE b = filterJust . applyE ((\b e -> if b then Just e else Nothing) <$> b) b <@ e = applyE (const <$> b) e