module Data.Singletons.Deriving.Enum ( mkEnumInstance ) where
import Language.Haskell.TH.Syntax
import Language.Haskell.TH.Ppr
import Language.Haskell.TH.Desugar
import Data.Singletons.Syntax
import Data.Singletons.Util
import Data.Singletons.Names
import Control.Monad
import Data.Maybe
mkEnumInstance :: Quasi q => Maybe DCxt -> DType -> [DCon] -> q UInstDecl
mkEnumInstance mb_ctxt ty cons = do
when (null cons ||
any (\(DCon tvbs cxt _ f rty) -> or [ not $ null $ tysOfConFields f
, not $ null tvbs
, not $ null cxt
, isJust rty ]) cons) $
fail ("Can't derive Enum instance for " ++ pprint (typeToTH ty) ++ ".")
n <- qNewName "n"
let to_enum = UFunction [DClause [DVarPa n] (to_enum_rhs cons [0..])]
to_enum_rhs [] _ = DVarE errorName `DAppE` DLitE (StringL "toEnum: bad argument")
to_enum_rhs (DCon _ _ name _ _ : rest) (num:nums) =
DCaseE (DVarE equalsName `DAppE` DVarE n `DAppE` DLitE (IntegerL num))
[ DMatch (DConPa trueName []) (DConE name)
, DMatch (DConPa falseName []) (to_enum_rhs rest nums) ]
to_enum_rhs _ _ = error "Internal error: exhausted infinite list in to_enum_rhs"
from_enum = UFunction (zipWith (\i con -> DClause [DConPa (extractName con) []]
(DLitE (IntegerL i)))
[0..] cons)
return (InstDecl { id_cxt = fromMaybe [] mb_ctxt
, id_name = singletonsEnumName
, id_arg_tys = [ty]
, id_meths = [ (singletonsToEnumName, to_enum)
, (singletonsFromEnumName, from_enum) ] })