{-# LANGUAGE FlexibleInstances #-} module Syntax.PrettyPrinter ( ppTerm ) where import Control.Monad.State import Text.PrettyPrint import Data.Bifunctor import Data.Bifoldable import Data.Void import Syntax import qualified Syntax.ErrorDoc as E instance E.Pretty1 (Term Syntax) where pretty1 t = ppTermCtx (freeVars t) t freeVars :: Term Syntax a -> [String] freeVars = biconcatMap (\t -> case t of Name _ (Ident s) -> [s] _ -> []) (const []) ppTerm :: Term Syntax Void -> Doc ppTerm t = ppTermCtx (freeVars t) (vacuous t) ppTermCtx :: [String] -> Term Syntax Doc -> Doc ppTermCtx ctx (Var d ts) = d <+> ppList ctx ts ppTermCtx ctx (Apply s ts) = ppSyntax ctx s ts ppTermCtx ctx (Lambda t) = text "{{" <+> ppTermCtx ctx (fmap (\v -> case v of Bound -> text "(error: Bound)" Free d -> text "Free(" <> d <> text ")") t) <+> text "}}" ppSyntax :: [String] -> Syntax -> [Term Syntax Doc] -> Doc ppSyntax ctx p@(Pi e vs) (t1:t2:ts) = let (vs',r1) = ppBound (prec p) ctx vs t2 r2 = (if null vs then ppTermPrec (prec p + 1) ctx t1 else (if e == Explicit then parens else braces) $ hsep (map text vs') <+> colon <+> ppTermCtx ctx t1) <+> arrow <+> r1 in if null ts then r2 else parens r2 <+> ppList ctx ts ppSyntax ctx l@(Lam vs) (t:ts) = let (vs',r) = ppBound (prec l) ctx vs t in bparens (not $ null ts) (text "\\" <> hsep (map text vs') <+> arrow <+> r) <+> ppList ctx ts ppSyntax ctx t@PathImp [_,t2,t3] = ppTermPrec (prec t + 1) ctx t2 <+> equals <+> ppTermPrec (prec t + 1) ctx t3 ppSyntax ctx t@At (_:_:t3:t4:ts) = bparens (not $ null ts) (ppTermPrec (prec t) ctx t3 <+> text "@" <+> ppTermPrec (prec t + 1) ctx t4) <+> ppList ctx ts ppSyntax ctx t@(Name (Infix ft _) n) (t1:t2:ts) = bparens (not $ null ts) (ppTermPrec (opFixity InfixL ft $ prec t) ctx t1 <+> text (nameToInfix n) <+> ppTermPrec (opFixity InfixR ft $ prec t) ctx t2) <+> ppList ctx ts ppSyntax ctx (Name _ n) ts = text (nameToPrefix n) <+> ppList ctx ts ppSyntax ctx (Case pats) (expr:terms) = let (terms1,terms2) = splitAt (length pats) terms caseDoc = hang (text "case" <+> ppTermCtx ctx expr <+> text "of") 4 $ vcat $ map (\(pat,term) -> let (vs,r) = ppBound 0 ctx (bifoldMap (const []) return pat) term pat' = evalState (replaceVars $ bimap (Name Prefix . snd) id pat) vs in ppTermCtx ctx pat' <+> arrow <+> r) (zip pats terms1) in if null terms2 then caseDoc else parens caseDoc <+> ppList ctx terms2 where replaceVars :: Term Syntax String -> State [String] (Term Syntax Doc) replaceVars (Var a []) = do v:vs <- get put vs return $ cvar (text v) replaceVars (Var a as) = fmap (Var $ text a) (mapM replaceVars as) replaceVars (Apply a as) = fmap (Apply a) (mapM replaceVars as) replaceVars Lambda{} = error "replaceVars" ppSyntax ctx Null [t] = ppTermCtx ctx t ppSyntax _ Null _ = empty ppSyntax ctx (Conds k) (t:ts) = ppTermCtx ctx $ apps t (drop k ts) ppSyntax ctx (Constr k s) ts = ppSyntax ctx s (drop k ts) ppSyntax ctx t@(FieldAcc k (PIdent _ fn)) (t1:ts) = let b = prec t > precTerm t1 || isAtom t1 isAtom (Apply Name{} []) = True isAtom (Apply Null [t]) = isAtom t isAtom (Apply (Conds k') (t1':ts')) = isAtom t1' && null (drop k' ts') isAtom (Apply (Constr k' _) ts') = null (drop k ts') isAtom (Apply FieldAcc{} [_]) = True isAtom (Var _ []) = True isAtom _ = False in (if b then (<>) else (<+>)) (ppTermPrec (prec t) ctx t1) (text $ '.' : fn) <+> ppList ctx (drop k ts) ppSyntax _ _ _ = error "ppSyntax" opFixity :: Infix -> Infix -> Int -> Int opFixity ft ft' p = if ft == ft' then p else p + 1 ppList :: [String] -> [Term Syntax Doc] -> Doc ppList ctx ts = hsep $ map (ppTermPrec 110 ctx) ts ppBound :: Int -> [String] -> [String] -> Term Syntax Doc -> ([String],Doc) ppBound p ctx (v:vs) t = let (ctx',v') = renameName v ctx (vs',r) = ppBound p ctx' vs $ apps t [capply $ Name Prefix $ Ident v'] in (v':vs',r) ppBound p ctx [] t = ([], ppTermPrec p ctx t) ppTermPrec :: Int -> [String] -> Term Syntax Doc -> Doc ppTermPrec p ctx t = bparens (p > precTerm t) (ppTermCtx ctx t) bparens :: Bool -> Doc -> Doc bparens True d = parens d bparens False d = d arrow :: Doc arrow = text "->" renameName :: String -> [String] -> ([String], String) renameName var ctx = if var `elem` ctx then renameName (var ++ "'") ctx else (var:ctx,var) prec :: Syntax -> Int prec (Name Prefix _) = 110 prec (Name (Infix _ p) _) = p prec FieldAcc{} = 100 prec At = 80 prec PathImp{} = 70 prec Pi{} = 60 prec Lam{} = 50 prec (Constr _ s) = prec s prec _ = 0 precTerm :: Term Syntax a -> Int precTerm Var{} = 110 precTerm (Apply Name{} (_:_)) = 100 precTerm (Apply Null [t]) = precTerm t precTerm (Apply Conds{} (t:_)) = precTerm t precTerm (Apply (Constr _ s) ts) = precTerm (Apply s ts) precTerm (Apply s _) = prec s precTerm (Lambda t) = precTerm t