{-# LANGUAGE CPP #-}
module Foreign.Hoppy.Generator.Std.UnorderedSet (
Options (..),
defaultOptions,
Contents (..),
instantiate,
instantiate',
toExports,
) where
import Control.Monad (when)
#if !MIN_VERSION_base(4,8,0)
import Data.Monoid (mconcat)
#endif
import Foreign.Hoppy.Generator.Language.Haskell (
HsTypeSide (HsHsSide),
addImports,
cppTypeToHsTypeAndUse,
indent,
ln,
prettyPrint,
sayLn,
saysLn,
)
import Foreign.Hoppy.Generator.Spec
import Foreign.Hoppy.Generator.Spec.Class (
Class,
MethodApplicability (MNormal),
makeClass,
makeFnMethod,
mkConstMethod,
mkConstMethod',
mkCtor,
mkMethod,
mkMethod',
toHsCastMethodName,
toHsDataTypeName,
toHsClassEntityName,
)
import Foreign.Hoppy.Generator.Std (ValueConversion (ConvertPtr, ConvertValue))
import Foreign.Hoppy.Generator.Std.Internal (includeHelper)
import Foreign.Hoppy.Generator.Std.Iterator
import Foreign.Hoppy.Generator.Types
data Options = Options
{ optUnorderedSetClassFeatures :: [ClassFeature]
, optValueConversion :: Maybe ValueConversion
}
defaultOptions :: Options
defaultOptions = Options [] Nothing
data Contents = Contents
{ c_set :: Class
, c_iterator :: Class
, c_constIterator :: Class
}
instantiate :: String -> Type -> Reqs -> Contents
instantiate setName t tReqs = instantiate' setName t tReqs defaultOptions
instantiate' :: String -> Type -> Reqs -> Options -> Contents
instantiate' setName t tReqs opts =
let reqs = mconcat
[ tReqs
, reqInclude $ includeHelper "unordered_set.hpp"
, reqInclude $ includeStd "unordered_set"
]
iteratorName = setName ++ "Iterator"
set =
(case optValueConversion opts of
Nothing -> id
Just conversion -> addAddendumHaskell $ makeAddendum conversion) $
addReqs reqs $
classAddFeatures (Assignable : Copyable : optUnorderedSetClassFeatures opts) $
makeClass (ident1T "std" "unordered_set" [t]) (Just $ toExtName setName) []
[ mkCtor "new" np
, mkMethod' "begin" "begin" np $ toGcT $ objT iterator
, mkConstMethod' "begin" "beginConst" np $ toGcT $ objT constIterator
, mkMethod "clear" np voidT
, mkConstMethod "count" [t] sizeT
, mkConstMethod "empty" np boolT
, mkMethod' "end" "end" np $ toGcT $ objT iterator
, mkConstMethod' "end" "endConst" np $ toGcT $ objT constIterator
, mkMethod' "erase" "erase" [objT iterator] voidT
, mkMethod' "erase" "eraseRange" [objT iterator, objT iterator] voidT
, mkMethod "find" [t] $ toGcT $ objT iterator
, makeFnMethod (ident2 "hoppy" "unordered_set" "insert") "insert"
MNormal Nonpure [refT $ objT set, t] boolT
, makeFnMethod (ident2 "hoppy" "unordered_set" "insertAndGetIterator")
"insertAndGetIterator" MNormal Nonpure [refT $ objT set, t] $ toGcT $ objT iterator
, mkConstMethod' "max_size" "maxSize" np sizeT
, mkConstMethod "size" np sizeT
, mkMethod "swap" [refT $ objT set] voidT
]
iterator =
addReqs reqs $
makeForwardIterator Constant (Just t) $
makeClass (identT' [("std", Nothing), ("unordered_set", Just [t]), ("iterator", Nothing)])
(Just $ toExtName iteratorName) [] []
constIterator =
addReqs reqs $
makeForwardIterator Constant (Just t) $
makeClass (identT' [("std", Nothing),
("unordered_set", Just [t]),
("const_iterator", Nothing)])
(Just $ toExtName iteratorName) [] []
makeAddendum conversion = do
addImports $ mconcat [hsImport1 "Prelude" "($)",
hsImportForPrelude,
hsImportForRuntime]
when (conversion == ConvertValue) $
addImports $ mconcat [hsImport1 "Prelude" "(=<<)"]
hsDataNameConst <- toHsDataTypeName Const set
hsDataName <- toHsDataTypeName Nonconst set
let computeValueType cst =
cppTypeToHsTypeAndUse HsHsSide $
(case conversion of
ConvertPtr -> ptrT
ConvertValue -> id) $
case cst of
Const -> constT t
Nonconst -> t
hsValueTypeConst <- computeValueType Const
hsValueType <- computeValueType Nonconst
setConstCast <- toHsCastMethodName Const set
setEmpty <- toHsClassEntityName set "empty"
setBeginConst <- toHsClassEntityName set "beginConst"
setEndConst <- toHsClassEntityName set "endConst"
iterEq <- toHsClassEntityName iterator OpEq
iterGetConst <- toHsClassEntityName iterator "getConst"
iterNext <- toHsClassEntityName iterator "next"
ln
saysLn ["instance HoppyFHR.HasContents ", hsDataNameConst,
" (", prettyPrint hsValueTypeConst, ") where"]
indent $ do
sayLn "toContents this' = do"
indent $ do
saysLn ["empty' <- ", setEmpty, " this'"]
sayLn "if empty' then HoppyP.return [] else do"
indent $ do
saysLn ["end' <- ", setEndConst, " this'"]
saysLn ["iter' <- ", setBeginConst, " this'"]
sayLn "go' iter' end' []"
sayLn "where"
indent $ do
sayLn "go' iter' end' acc' = do"
indent $ do
saysLn ["stop' <- ", iterEq, " iter' end'"]
sayLn "if stop' then HoppyP.return (HoppyP.reverse acc') else do"
indent $ do
saysLn ["_ <- ", iterNext, " iter'"]
saysLn ["value' <- ",
case conversion of
ConvertPtr -> ""
ConvertValue -> "HoppyFHR.decode =<< ",
iterGetConst, " iter'"]
sayLn "go' iter' end' $ value':acc'"
ln
saysLn ["instance HoppyFHR.HasContents ", hsDataName,
" (", prettyPrint hsValueTypeConst, ") where"]
indent $
saysLn ["toContents = HoppyFHR.toContents . ", setConstCast]
ln
saysLn ["instance HoppyFHR.FromContents ", hsDataName,
" (", prettyPrint hsValueType, ") where"]
indent $ do
sayLn "fromContents values' = do"
indent $ do
setNew <- toHsClassEntityName set "new"
setInsert <- toHsClassEntityName set "insert"
saysLn ["set' <- ", setNew]
saysLn ["HoppyP.mapM_ (", setInsert, " set') values'"]
sayLn "HoppyP.return set'"
in Contents
{ c_set = set
, c_iterator = iterator
, c_constIterator = constIterator
}
toExports :: Contents -> [Export]
toExports m = map (Export . ($ m)) [c_set, c_iterator, c_constIterator]