{-# LANGUAGE Arrows #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE TypeFamilies #-}
module FRP.Rhine.SyncSF.Except
( module FRP.Rhine.SyncSF.Except
, module X
, safe, safely, Empty, exceptS, runMSFExcept
)
where
import Control.Monad.Trans.Class (lift)
import Control.Monad.Trans.Except as X
import Control.Monad.Trans.Reader
import Control.Monad.Trans.MSF.Except hiding (try, once, once_, throwOn, throwOn', throwS)
import qualified Control.Monad.Trans.MSF.Except as MSFE
import FRP.Rhine
import FRP.Rhine.SyncSF.Except.Util
type SyncExcept m cl a b e = MSFExcept (ReaderT (TimeInfo cl) m) a b e
type BehaviourFExcept m td a b e
= forall cl. td ~ TimeDomainOf cl => SyncExcept m cl a b e
type BehaviorFExcept m td a b e = BehaviourFExcept m td a b e
commuteExceptReader :: ExceptT e (ReaderT r m) a -> ReaderT r (ExceptT e m) a
commuteExceptReader a = ReaderT $ \r -> ExceptT $ runReaderT (runExceptT a) r
runSyncExcept :: Monad m => SyncExcept m cl a b e -> SyncSF (ExceptT e m) cl a b
runSyncExcept = liftMSFPurer commuteExceptReader . runMSFExcept
try :: Monad m => SyncSF (ExceptT e m) cl a b -> SyncExcept m cl a b e
try = MSFE.try . liftMSFPurer commuteReaderExcept
once :: Monad m => (a -> m e) -> SyncExcept m cl a b e
once f = MSFE.once $ lift . f
once_ :: Monad m => m e -> SyncExcept m cl a b e
once_ = once . const
throwS :: Monad m => SyncSF (ExceptT e m) cl e a
throwS = arrMSync throwE
throwOn :: Monad m => e -> SyncSF (ExceptT e m) cl Bool ()
throwOn e = proc b -> throwOn' -< (b, e)
throwOn' :: Monad m => SyncSF (ExceptT e m) cl (Bool, e) ()
throwOn' = proc (b, e) -> if b
then throwS -< e
else returnA -< ()
step :: Monad m => (a -> m (b, e)) -> SyncExcept m cl a b e
step f = MSFE.step $ lift . f
keepFirst :: Monad m => SyncSF m cl a a
keepFirst = safely $ do
a <- try throwS
safe $ arr $ const a
timer
:: ( Monad m
, TimeDomain td
, Ord (Diff td)
)
=> Diff td
-> BehaviorF (ExceptT () m) td a (Diff td)
timer diff = proc _ -> do
time <- timeInfoOf absolute -< ()
startTime <- keepFirst -< time
let remainingTime = time `diffTime` startTime
_ <- throwOn () -< remainingTime > diff
returnA -< remainingTime
scaledTimer
:: ( Monad m
, TimeDomain td
, Fractional (Diff td)
, Ord (Diff td)
)
=> Diff td
-> BehaviorF (ExceptT () m) td a (Diff td)
scaledTimer diff = timer diff >>> arr (/ diff)