quox/lib/Quox/Pretty.idr

202 lines
4.6 KiB
Idris
Raw Normal View History

2021-07-20 16:05:19 -04:00
module Quox.Pretty
import Quox.Name
import public Text.PrettyPrint.Prettyprinter.Doc
import Text.PrettyPrint.Prettyprinter.Render.String
import Text.PrettyPrint.Prettyprinter.Render.Terminal
import public Data.String
import Data.DPair
import public Control.Monad.Identity
import public Control.Monad.Reader
2022-05-13 01:05:55 -04:00
import Generics.Derive
2021-07-20 16:05:19 -04:00
%default total
2022-05-13 01:05:55 -04:00
%language ElabReflection
%hide TT.Name
2021-07-20 16:05:19 -04:00
2022-04-11 15:58:33 -04:00
public export
record PrettyOpts where
constructor MakePrettyOpts
unicode, color : Bool
public export
defPrettyOpts : PrettyOpts
defPrettyOpts = MakePrettyOpts {unicode = True, color = True}
2021-07-20 16:05:19 -04:00
public export
data HL
2022-05-13 01:05:55 -04:00
= Delim
| Free | TVar | TVarErr
| Dim | DVar | DVarErr
| Qty
| Syntax
%runElab derive "HL" [Generic, Meta, Eq, Ord, DecEq, Show]
2021-07-20 16:05:19 -04:00
public export
data PPrec
2022-05-13 01:05:55 -04:00
= Outer
| Ann -- right of "::"
| AnnL -- left of "::"
2021-07-20 16:05:19 -04:00
-- ...
2022-05-13 01:05:55 -04:00
| App -- term/dimension application
| SApp -- substitution application
| Arg -- argument to nonfix function
%runElab derive "PPrec" [Generic, Meta, Eq, Ord, DecEq, Show]
2021-07-20 16:05:19 -04:00
export %inline
hl : HL -> Doc HL -> Doc HL
hl = annotate
export %inline
hl' : HL -> Doc HL -> Doc HL
hl' h = hl h . unAnnotate
export %inline
hlF : Functor f => HL -> f (Doc HL) -> f (Doc HL)
hlF = map . hl
export %inline
hlF' : Functor f => HL -> f (Doc HL) -> f (Doc HL)
hlF' = map . hl'
2021-09-09 17:51:29 -04:00
export %inline
2021-07-20 16:05:19 -04:00
parens : Doc HL -> Doc HL
parens doc = hl Delim "(" <+> doc <+> hl Delim ")"
2022-05-13 01:05:55 -04:00
%hide Symbols.parens
2021-07-20 16:05:19 -04:00
2021-09-09 17:51:29 -04:00
export %inline
2021-07-20 16:05:19 -04:00
parensIf : Bool -> Doc HL -> Doc HL
parensIf True = parens
parensIf False = id
2021-09-09 17:56:10 -04:00
export
separate' : Doc a -> List (Doc a) -> List (Doc a)
separate' s [] = []
separate' s [x] = [x]
separate' s (x :: xs) = x <+> s :: separate' s xs
export %inline
separate : Doc a -> List (Doc a) -> Doc a
separate s = sep . separate' s
export %inline
hseparate : Doc a -> List (Doc a) -> Doc a
hseparate s = hsep . separate' s
export %inline
vseparate : Doc a -> List (Doc a) -> Doc a
vseparate s = vsep . separate' s
2021-07-20 16:05:19 -04:00
public export
record PrettyEnv where
constructor MakePrettyEnv
||| names of bound dimension variables
dnames : List Name
||| names of bound term variables
tnames : List Name
||| use non-ascii characters for syntax
unicode : Bool
||| surrounding precedence level
prec : PPrec
2022-05-13 01:05:55 -04:00
public export
HasEnv : (Type -> Type) -> Type
HasEnv = MonadReader PrettyEnv
2021-07-20 16:05:19 -04:00
2021-09-09 17:51:29 -04:00
export %inline
ifUnicode : HasEnv m => (uni, asc : Lazy a) -> m a
2021-09-03 10:31:53 -04:00
ifUnicode uni asc = if (!ask).unicode then [|uni|] else [|asc|]
2021-07-20 16:05:19 -04:00
2021-09-09 17:51:29 -04:00
export %inline
parensIfM : HasEnv m => PPrec -> Doc HL -> m (Doc HL)
2021-09-03 10:31:53 -04:00
parensIfM d doc = pure $ parensIf ((!ask).prec > d) doc
2021-07-20 16:05:19 -04:00
2021-09-09 17:51:29 -04:00
export %inline
withPrec : HasEnv m => PPrec -> m a -> m a
2021-07-20 16:05:19 -04:00
withPrec d = local {prec := d}
public export data BinderSort = T | D
2021-09-09 17:51:29 -04:00
export %inline
under : HasEnv m => BinderSort -> Name -> m a -> m a
2022-04-09 12:55:08 -04:00
under T x = local {prec := Outer, tnames $= (x ::)}
under D x = local {prec := Outer, dnames $= (x ::)}
2021-07-20 16:05:19 -04:00
public export
interface PrettyHL a where
prettyM : HasEnv m => a -> m (Doc HL)
2021-07-20 16:05:19 -04:00
export %inline
pretty0M : (PrettyHL a, HasEnv m) => a -> m (Doc HL)
2021-07-20 16:05:19 -04:00
pretty0M = local {prec := Outer} . prettyM
export %inline
2022-04-11 15:58:33 -04:00
pretty0 : PrettyHL a => (unicode : Bool) -> a -> Doc HL
pretty0 unicode x =
2021-07-20 16:05:19 -04:00
let env = MakePrettyEnv {dnames = [], tnames = [], unicode, prec = Outer} in
runReader env $ prettyM x
export
(forall a. PrettyHL (f a)) => PrettyHL (Exists f) where
prettyM x = prettyM x.snd
export
PrettyHL a => PrettyHL (Subset a b) where
prettyM x = prettyM x.fst
export PrettyHL BaseName where prettyM = pure . pretty . baseStr
export PrettyHL Name where prettyM = pure . pretty . toDots
2021-09-09 17:51:29 -04:00
export %inline
2022-04-11 15:58:33 -04:00
prettyStr : PrettyHL a => (unicode : Bool) -> a -> String
prettyStr unicode =
2021-09-03 10:31:53 -04:00
let layout = layoutSmart (MkLayoutOptions (AvailablePerLine 80 0.8)) in
2022-04-11 15:58:33 -04:00
renderString . layout . pretty0 unicode
2021-07-20 16:05:19 -04:00
export
termHL : HL -> AnsiStyle
termHL Delim = color BrightBlack
termHL TVar = color BrightYellow
termHL TVarErr = color BrightYellow <+> underline
termHL Dim = color BrightGreen <+> bold
termHL DVar = color BrightGreen
termHL DVarErr = color BrightGreen <+> underline
termHL Qty = color BrightMagenta <+> bold
termHL Free = color BrightWhite
2022-04-11 08:09:48 -04:00
termHL Syntax = color BrightCyan
2021-07-20 16:05:19 -04:00
2021-09-09 17:51:29 -04:00
export %inline
2022-04-11 15:58:33 -04:00
prettyTerm : PrettyOpts -> PrettyHL a => a -> IO Unit
prettyTerm opts x =
let reann = if opts.color then map termHL else unAnnotate in
Terminal.putDoc $ reann $ pretty0 opts.unicode x
export %inline
prettyTermDef : PrettyHL a => a -> IO Unit
prettyTermDef = prettyTerm defPrettyOpts
2021-07-20 16:05:19 -04:00
infixr 6 <//>
export %inline
(<//>) : Doc a -> Doc a -> Doc a
a <//> b = sep [a, b]
infixr 6 </>
export %inline
(</>) : Doc a -> Doc a -> Doc a
a </> b = cat [a, b]