{-# LANGUAGE DataKinds, FlexibleContexts, FlexibleInstances, GADTs, OverloadedStrings, ScopedTypeVariables, TypeApplications, TypeFamilies, UndecidableInstances #-} -- | This module exports the instances of the 'Pretty' type class necessary for printing of a Modula-2 abstract syntax -- tree. module Language.Modula2.Pretty () where import Control.Applicative (ZipList(ZipList, getZipList)) import Data.Functor.Identity (Identity(..)) import Data.List (intersperse) import Data.List.NonEmpty (NonEmpty((:|)), fromList, toList) import qualified Data.Text as Text import Data.Text.Prettyprint.Doc import Data.Text.Prettyprint.Doc.Render.Text (renderStrict) import Numeric (showHex, showOct) import qualified Language.Oberon.Abstract import qualified Language.Oberon.AST import qualified Language.Modula2.Abstract as Abstract import Language.Modula2.AST import Language.Oberon.Pretty (Precedence(Precedence)) import qualified Language.Oberon.AST as Oberon instance (Pretty (Abstract.Priority l l Identity Identity), Pretty (Abstract.Export l), Pretty (Abstract.Import l), Pretty (Abstract.Declaration l l Identity Identity), Pretty (Abstract.Definition l l Identity Identity), Pretty (Abstract.Block l l Identity Identity)) => Pretty (Module λ l Identity Identity) where pretty (DefinitionModule name imports export declarations) = vsep $ intersperse mempty $ ["DEFINITION" <+> "MODULE" <+> pretty name <> semi, vsep (pretty <$> imports), foldMap pretty export] <> (pretty <$> getZipList declarations) <> ["END" <+> pretty name <> "." <> line] pretty (ImplementationModule name priority imports body) = "IMPLEMENTATION" <+> pretty (ProgramModule name priority imports body) pretty (ProgramModule name priority imports body) = vsep $ intersperse mempty $ ["MODULE" <+> pretty name <> maybe mempty (brackets . pretty) priority <> semi, vsep (pretty <$> imports)] <> [vsep [pretty body, "END" <+> pretty name <> "." <> line]] instance Pretty (Abstract.IdentDef l) => Pretty (Import l) where pretty (Import origin names) = maybe id ((<+>) . ("FROM" <+>) . pretty) origin ("IMPORT" <+> hsep (punctuate comma $ toList $ pretty <$> names)) <> semi instance Pretty (Abstract.IdentDef l) => Pretty (Export l) where pretty (Export qualified names) = "EXPORT" <+> (if qualified then ("QUALIFIED" <+>) else id) (hsep $ punctuate comma $ toList $ pretty <$> names) <> semi instance Pretty (IdentDef l) where pretty (IdentDef name) = pretty name instance (Abstract.Nameable l, Pretty (Abstract.IdentDef l), Pretty (Abstract.Export l), Pretty (Abstract.Import l), Pretty (Abstract.Type l l Identity Identity), Pretty (Abstract.Declaration l l Identity Identity), Pretty (Abstract.Expression l l Identity Identity), Pretty (Abstract.FormalParameters l l Identity Identity), Pretty (Abstract.ProcedureHeading l l Identity Identity), Pretty (Abstract.Block l l Identity Identity)) => Pretty (Declaration False Language l Identity Identity) where pretty (ProcedureDefinition heading) = pretty heading <> semi pretty (ConstantDeclaration ident (Identity expr)) = "CONST" <+> pretty ident <+> "=" <+> pretty expr <> semi pretty (TypeDeclaration ident typeDef) = "TYPE" <+> pretty ident <+> "=" <+> pretty typeDef <> semi pretty (OpaqueTypeDeclaration ident) = "TYPE" <+> pretty ident <> semi pretty (VariableDeclaration idents varType) = "VAR" <+> hsep (punctuate comma $ pretty <$> toList idents) <+> colon <+> pretty varType <> semi instance (Abstract.Nameable l, Pretty (Abstract.IdentDef l), Pretty (Abstract.Export l), Pretty (Abstract.Import l), Pretty (Abstract.Type l l Identity Identity), Pretty (Abstract.Declaration l l Identity Identity), Pretty (Abstract.Expression l l Identity Identity), Pretty (Abstract.FormalParameters l l Identity Identity), Pretty (Abstract.ProcedureHeading l l Identity Identity), Pretty (Abstract.Block l l Identity Identity)) => Pretty (Declaration True Language l Identity Identity) where pretty (ProcedureDeclaration heading body) = vsep [pretty heading <> semi, pretty body, "END" <+> pretty (Abstract.getProcedureName $ runIdentity heading) <> semi] pretty (ModuleDeclaration name priority imports export body) = vsep $ intersperse mempty $ ["MODULE" <+> pretty name <> maybe mempty (brackets . pretty) priority <> semi, vsep (pretty <$> imports), foldMap pretty export, pretty body, "END" <+> pretty name <> semi] pretty declaration = Abstract.maybe3 mempty pretty (Abstract.coDeclaration @Language @(Abstract.WirthySubsetOf Oberon.Language) declaration) instance (Pretty (Precedence (Abstract.Expression l l Identity Identity)), Pretty (Abstract.Value l l Identity Identity), Pretty (Abstract.Expression l l Identity Identity), Pretty (Abstract.Element l l Identity Identity), Pretty (Abstract.Designator l l Identity Identity), Pretty (Abstract.QualIdent l)) => Pretty (Expression Language l Identity Identity) where pretty e = pretty (Precedence 0 e) instance (Pretty (Precedence (Abstract.Expression l l Identity Identity)), Pretty (Abstract.Value l l Identity Identity), Pretty (Abstract.Expression l l Identity Identity), Pretty (Abstract.Element l l Identity Identity), Pretty (Abstract.Designator l l Identity Identity), Pretty (Abstract.QualIdent l)) => Pretty (Precedence (Expression Language l Identity Identity)) where pretty (Precedence _ (Set ty elements)) = foldMap pretty ty <+> braces (hsep $ punctuate comma $ pretty <$> getZipList elements) pretty (Precedence p e) = Abstract.maybe3 mempty (pretty . Precedence p) (Abstract.coExpression @Language @(Abstract.WirthySubsetOf Oberon.Language) e) instance Pretty (Abstract.Value l l Identity Identity) => Pretty (Value Language l Identity Identity) where pretty (CharCode c) = pretty (showOct c "") <> "C" pretty v = Abstract.maybe3 mempty pretty (Abstract.coValue @Language @(Abstract.WirthySubsetOf Oberon.Language) v) instance (Pretty (Abstract.QualIdent l), Pretty (Abstract.Designator l l Identity Identity), Pretty (Abstract.Expression l l Identity Identity)) => Pretty (Designator Language l Identity Identity) where pretty d = Abstract.maybe3 mempty pretty (Abstract.coDesignator @Language @(Abstract.WirthySubsetOf Oberon.Language) d) instance (Pretty (Abstract.IdentDef l), Pretty (Abstract.FormalParameters l l Identity Identity), Pretty (Abstract.FieldList l l Identity Identity), Pretty (Abstract.ConstExpression l l Identity Identity), Pretty (Abstract.Type l l Identity Identity), Pretty (Abstract.BaseType l)) => Pretty (Type Language l Identity Identity) where pretty (ArrayType dimensions itemType) = "ARRAY" <+> hsep (punctuate comma $ pretty . runIdentity <$> getZipList dimensions) <+> "OF" <+> pretty itemType pretty (EnumerationType values) = "(" <> hsep (punctuate comma $ toList $ pretty <$> values) <> ")" pretty (SubrangeType enumType min max) = foldMap pretty enumType <> "[" <> pretty min <+> ".." <+> pretty max <> "]" pretty (SetType memberType) = "SET" <+> "OF" <+> pretty memberType pretty (RecordType fields) = vsep ["RECORD", indent 3 (vsep $ punctuate semi $ pretty <$> getZipList fields), "END"] pretty (ProcedureType parameters) = "PROCEDURE" <+> adjust (pretty parameters) where adjust = pretty . Text.replace " : " "" . Text.replace ";" "," . renderStrict . layoutCompact pretty ty = Abstract.maybe3 mempty pretty (Abstract.coType @Language @(Abstract.WirthySubsetOf Oberon.Language) ty) instance Pretty (QualIdent l) where pretty (QualIdent modulePath name) = mconcat (punctuate dot $ pretty <$> (modulePath <> [name])) instance (Pretty (Abstract.IdentDef l), Pretty (Abstract.QualIdent l), Pretty (Abstract.Type l l Identity Identity), Pretty (Abstract.Value l l Identity Identity), Pretty (Abstract.FieldList l l Identity Identity), Pretty (Abstract.Variant l l Identity Identity)) => Pretty (FieldList Language l Identity Identity) where pretty (CaseFieldList localName name variant variants fallback) = vsep (["CASE" <+> maybe id ((<+>) . (<+> ":") . pretty) localName (pretty name) <+> "OF"] <> punctuate' "| " (pretty <$> (variant : getZipList variants)) <> (if null fallback then [] else ["ELSE" <#> indent 3 (vsep $ punctuate semi $ pretty . runIdentity <$> getZipList fallback)]) <> ["END"]) pretty (FieldList names t) = hsep (punctuate comma $ pretty <$> toList names) <+> ":" <+> pretty t instance (Pretty (Abstract.CaseLabels l l Identity Identity), Pretty (Abstract.FieldList l l Identity Identity)) => Pretty (Variant λ l Identity Identity) where pretty (Variant label labels fields) = vsep [hsep (punctuate comma $ pretty <$> (label : getZipList labels)) <> ":", indent 3 (vsep $ punctuate semi $ pretty <$> getZipList fields)] instance (Pretty (Abstract.IdentDef l), Pretty (Abstract.FormalParameters l l Identity Identity), Pretty (Abstract.Type l l Identity Identity)) => Pretty (ProcedureHeading l l Identity Identity) where pretty (ProcedureHeading name parameters) = "PROCEDURE" <+> pretty name <> pretty parameters instance (Pretty (Abstract.ConstExpression l l Identity Identity), Pretty (Abstract.Designator l l Identity Identity), Pretty (Abstract.Case l l Identity Identity), Pretty (Abstract.ConditionalBranch l l Identity Identity), Pretty (Language.Oberon.Abstract.WithAlternative l l Identity Identity), Pretty (Abstract.StatementSequence l l Identity Identity)) => Pretty (Statement Language l Identity Identity) where prettyList l = vsep (dropEmptyTail $ punctuate semi $ pretty <$> l) where dropEmptyTail | not (null l), EmptyStatement <- last l = init | otherwise = id pretty (For index from to by body) = vsep ["FOR" <+> pretty index <+> ":=" <+> pretty from <+> "TO" <+> pretty to <+> foldMap ("BY" <+>) (pretty <$> by) <+> "DO", prettyBody body, "END"] pretty (With designator body) = vsep ["WITH" <+> pretty designator <+> "DO", prettyBody body, "END"] pretty stat = Abstract.maybe3 mempty pretty (Abstract.coStatement @Language @(Abstract.WirthySubsetOf Oberon.Language) stat) instance Language.Oberon.Abstract.Oberon Language where type WithAlternative Language = Language.Oberon.AST.WithAlternative Language --instance Pretty (Language.Oberon.AST.WithAlternative Language Language Identity Identity) where -- pretty _ = error "There's no WithAlternative in Modula-2." prettyBody :: Pretty (Abstract.StatementSequence l l Identity Identity) => Identity (Abstract.StatementSequence l l Identity Identity) -> Doc ann prettyBody (Identity statements) = indent 3 (pretty statements) punctuate' :: Doc ann -> [Doc ann] -> [Doc ann] punctuate' p [] = [] punctuate' p (x:xs) = x : ((p <>) <$> xs) a <#> b = vsep [a, b]