{-# LANGUAGE DeriveFoldable #-}
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE DeriveTraversable #-}
module Zinza.Expr (
Expr (..),
LExpr,
abstract1,
instantiate1ret,
) where
import Control.Monad (ap)
import Data.Maybe (fromMaybe)
import Zinza.Var
import Zinza.Pos
data Expr a
= EVar (Located a)
| EField (LExpr a) (Located Var)
| ENot (LExpr a)
deriving (Show, Functor, Foldable, Traversable)
type LExpr a = Located (Expr a)
instance TraversableWithLoc Expr where
traverseWithLoc f (EVar (L l x)) = EVar . L l <$> f l x
traverseWithLoc f (EField (L l e) v) = (\e' -> EField (L l e') v)
<$> traverseWithLoc f e
traverseWithLoc f (ENot (L l e)) = ENot . L l <$> traverseWithLoc f e
instance Monad Expr where
return = EVar . L zeroLoc
EVar (L _ x) >>= k = k x
EField (L l expr) var >>= k = EField (L l (expr >>= k)) var
ENot (L l expr) >>= k = ENot (L l (expr >>= k))
instance Applicative Expr where
pure = return
(<*>) = ap
abstract1 :: (Functor f, Eq a) => a -> f a -> f (Maybe a)
abstract1 x = fmap $ \y ->
if x == y
then Nothing
else Just y
instantiate1ret :: Functor f => a -> f (Maybe a) -> f a
instantiate1ret = fmap . fromMaybe