module Mida.Language.Environment
( MidaEnv
, HasEnv (..)
, runMidaEnv
, addDef
, remDef
, clearDefs
, getPrin
, getSrc
, fullSrc
, getRefs
, purgeEnv
, checkRecur )
where
import Control.Applicative (empty)
import Control.Arrow ((***), (>>>))
import Control.Monad.Catch (MonadThrow, MonadCatch, MonadMask)
import Control.Monad.Reader
import Control.Monad.State.Class
import Control.Monad.State.Strict
import Data.Map (Map)
import Data.Maybe (fromMaybe)
import Data.Monoid ((<>))
import Data.Text.Lazy (Text)
import Mida.Language.SyntaxTree
import Mida.Representation.Base (noteAlias, modifiers)
import Mida.Representation.Show (showDefinition)
import Numeric.Natural
import System.Console.Haskeline.MonadException
import System.Random (split)
import System.Random.TF (TFGen, mkTFGen)
import qualified Data.Map.Strict as M
data MidaEnvSt = MidaEnvSt
{ meDefs :: Defs
, meRandGen :: TFGen
} deriving Show
type Defs = Map String SyntaxTree
newtype MidaEnv m a = MidaEnv
{ unMidaEnv :: StateT MidaEnvSt m a }
deriving ( Functor
, Applicative
, Monad
, MonadIO
, MonadState MidaEnvSt
, MonadException
, MonadThrow
, MonadCatch
, MonadMask )
class Monad m => HasEnv m where
getDefs :: m Defs
setDefs :: Defs -> m ()
setRandGen :: Natural -> m ()
newRandGen :: m TFGen
instance Monad m => HasEnv (MidaEnv m) where
getDefs = gets meDefs
setDefs defs = modify $ \env -> env { meDefs = defs }
setRandGen gen = modify $ \e -> e { meRandGen = mkTFGen (fromIntegral gen) }
newRandGen = do
(g, g') <- split <$> gets meRandGen
modify $ \e -> e { meRandGen = g' }
return g
instance HasEnv m => HasEnv (StateT e m) where
getDefs = lift getDefs
setDefs = lift . setDefs
setRandGen = lift . setRandGen
newRandGen = lift newRandGen
instance HasEnv m => HasEnv (ReaderT e m) where
getDefs = lift getDefs
setDefs = lift . setDefs
setRandGen = lift . setRandGen
newRandGen = lift newRandGen
runMidaEnv :: Monad m => MidaEnv m a -> m a
runMidaEnv m = evalStateT (unMidaEnv m) MidaEnvSt
{ meDefs = defaultDefs
, meRandGen = mkTFGen 0 }
defaultDefs :: Defs
defaultDefs = M.fromList $
zip noteAlias (f <$> [0..]) <>
zip modifiers (f <$> [128,256..])
where f = pure . Value
addDef :: HasEnv m
=> String
-> SyntaxTree
-> m ()
addDef name tree = getDefs >>= setDefs . M.insert name tree
remDef :: HasEnv m
=> String
-> m ()
remDef name = getDefs >>= setDefs . M.delete name
clearDefs :: HasEnv m => m ()
clearDefs = setDefs defaultDefs
getPrin :: HasEnv m
=> String
-> m SyntaxTree
getPrin name = (fromMaybe [] . M.lookup name) <$> getDefs
getSrc :: HasEnv m
=> String
-> m Text
getSrc name = showDefinition name <$> getPrin name
fullSrc :: HasEnv m => m Text
fullSrc = (M.foldMapWithKey showDefinition . (M.\\ defaultDefs)) <$> getDefs
getRefs :: HasEnv m => m [String]
getRefs = M.keys <$> getDefs
tDefs
:: String
-> Defs
-> [String]
tDefs name defs = maybe empty cm (name `M.lookup` defs)
where cm = (>>= f)
f (Value _) = mempty
f (Section x) = cm x
f (Multi x) = cm x
f (CMulti x) = x >>= (cm *** cm >>> uncurry (<>))
f (Reference x) = return x <> tDefs x defs
f (Range _ _) = mempty
f (Product x y) = f x <> f y
f (Division x y) = f x <> f y
f (Sum x y) = f x <> f y
f (Diff x y) = f x <> f y
f (Loop x y) = f x <> f y
f (Rotation x y) = f x <> f y
f (Reverse x) = f x
purgeEnv :: HasEnv m
=> [String]
-> m ()
purgeEnv tops = getDefs >>= setDefs . f
where f defs = M.intersection defs (M.unions [ts, ms defs, defaultDefs])
ms = M.unions . fmap toDefs . zipWith tDefs tops . repeat
ts = toDefs tops
checkRecur :: HasEnv m
=> String
-> SyntaxTree
-> m Bool
checkRecur name tree = check <$> getDefs
where check = elem name . tDefs name . M.insert name tree
toDefs :: [String] -> Defs
toDefs = M.fromList . fmap (, [])