module Language.PureScript.Interactive.Printer where
import Prelude.Compat
import Data.List (intersperse)
import qualified Data.Map as M
import Data.Maybe (mapMaybe)
import Data.Monoid ((<>))
import qualified Data.Text as T
import Data.Text (Text)
import qualified Language.PureScript as P
import qualified Text.PrettyPrint.Boxes as Box
textT :: Text -> Box.Box
textT = Box.text . T.unpack
printModuleSignatures :: P.ModuleName -> P.Environment -> String
printModuleSignatures moduleName P.Environment{..} =
let moduleNamesIdent = byModuleName names
moduleTypeClasses = byModuleName typeClasses
moduleTypes = byModuleName types
byModuleName = filter ((== Just moduleName) . P.getQual) . M.keys
in
(unlines . map trimEnd . lines . Box.render . Box.vsep 1 Box.left)
[ printModule's (mapMaybe (showTypeClass . findTypeClass typeClasses)) moduleTypeClasses
, printModule's (mapMaybe (showType typeClasses dataConstructors typeSynonyms . findType types)) moduleTypes
, printModule's (map (showNameType . findNameType names)) moduleNamesIdent
]
where printModule's showF = Box.vsep 1 Box.left . showF
findNameType :: M.Map (P.Qualified P.Ident) (P.Type, P.NameKind, P.NameVisibility)
-> P.Qualified P.Ident
-> (P.Ident, Maybe (P.Type, P.NameKind, P.NameVisibility))
findNameType envNames m = (P.disqualify m, M.lookup m envNames)
showNameType :: (P.Ident, Maybe (P.Type, P.NameKind, P.NameVisibility)) -> Box.Box
showNameType (mIdent, Just (mType, _, _)) = textT (P.showIdent mIdent <> " :: ") Box.<> P.typeAsBox mType
showNameType _ = P.internalError "The impossible happened in printModuleSignatures."
findTypeClass
:: M.Map (P.Qualified (P.ProperName 'P.ClassName)) P.TypeClassData
-> P.Qualified (P.ProperName 'P.ClassName)
-> (P.Qualified (P.ProperName 'P.ClassName), Maybe P.TypeClassData)
findTypeClass envTypeClasses name = (name, M.lookup name envTypeClasses)
showTypeClass
:: (P.Qualified (P.ProperName 'P.ClassName), Maybe P.TypeClassData)
-> Maybe Box.Box
showTypeClass (_, Nothing) = Nothing
showTypeClass (P.Qualified _ name, Just P.TypeClassData{..}) =
let constraints =
if null typeClassSuperclasses
then Box.text ""
else Box.text "("
Box.<> Box.hcat Box.left (intersperse (Box.text ", ") $ map (\(P.Constraint (P.Qualified _ pn) lt _) -> textT (P.runProperName pn) Box.<+> Box.hcat Box.left (map P.typeAtomAsBox lt)) typeClassSuperclasses)
Box.<> Box.text ") <= "
className =
textT (P.runProperName name)
Box.<> textT (foldMap ((" " <>) . fst) typeClassArguments)
classBody =
Box.vcat Box.top (map (\(i, t) -> textT (P.showIdent i <> " ::") Box.<+> P.typeAsBox t) typeClassMembers)
in
Just $
(Box.text "class "
Box.<> constraints
Box.<> className
Box.<+> if null typeClassMembers then Box.text "" else Box.text "where")
Box.// Box.moveRight 2 classBody
findType
:: M.Map (P.Qualified (P.ProperName 'P.TypeName)) (P.Kind, P.TypeKind)
-> P.Qualified (P.ProperName 'P.TypeName)
-> (P.Qualified (P.ProperName 'P.TypeName), Maybe (P.Kind, P.TypeKind))
findType envTypes name = (name, M.lookup name envTypes)
showType
:: M.Map (P.Qualified (P.ProperName 'P.ClassName)) P.TypeClassData
-> M.Map (P.Qualified (P.ProperName 'P.ConstructorName)) (P.DataDeclType, P.ProperName 'P.TypeName, P.Type, [P.Ident])
-> M.Map (P.Qualified (P.ProperName 'P.TypeName)) ([(Text, Maybe P.Kind)], P.Type)
-> (P.Qualified (P.ProperName 'P.TypeName), Maybe (P.Kind, P.TypeKind))
-> Maybe Box.Box
showType typeClassesEnv dataConstructorsEnv typeSynonymsEnv (n@(P.Qualified modul name), typ) =
case (typ, M.lookup n typeSynonymsEnv) of
(Just (_, P.TypeSynonym), Just (typevars, dtType)) ->
if M.member (fmap P.coerceProperName n) typeClassesEnv
then
Nothing
else
Just $
textT ("type " <> P.runProperName name <> foldMap ((" " <>) . fst) typevars)
Box.// Box.moveRight 2 (Box.text "=" Box.<+> P.typeAsBox dtType)
(Just (_, P.DataType typevars pt), _) ->
let prefix =
case pt of
[(dtProperName,_)] ->
case M.lookup (P.Qualified modul dtProperName) dataConstructorsEnv of
Just (dataDeclType, _, _, _) -> P.showDataDeclType dataDeclType
_ -> "data"
_ -> "data"
in
Just $ textT (prefix <> " " <> P.runProperName name <> foldMap ((" " <>) . fst) typevars) Box.// printCons pt
_ ->
Nothing
where printCons pt =
Box.moveRight 2 $
Box.vcat Box.left $
mapFirstRest (Box.text "=" Box.<+>) (Box.text "|" Box.<+>) $
map (\(cons,idents) -> (textT (P.runProperName cons) Box.<> Box.hcat Box.left (map prettyPrintType idents))) pt
prettyPrintType t = Box.text " " Box.<> P.typeAtomAsBox t
mapFirstRest _ _ [] = []
mapFirstRest f g (x:xs) = f x : map g xs
trimEnd = reverse . dropWhile (== ' ') . reverse