diff options
Diffstat (limited to 'src/Erebos/TextFormat')
| -rw-r--r-- | src/Erebos/TextFormat/Ansi.hs | 59 | ||||
| -rw-r--r-- | src/Erebos/TextFormat/Types.hs | 57 |
2 files changed, 116 insertions, 0 deletions
diff --git a/src/Erebos/TextFormat/Ansi.hs b/src/Erebos/TextFormat/Ansi.hs new file mode 100644 index 0000000..504e098 --- /dev/null +++ b/src/Erebos/TextFormat/Ansi.hs @@ -0,0 +1,59 @@ +{-# LANGUAGE OverloadedStrings #-} + +module Erebos.TextFormat.Ansi ( + FormattedText, + + AnsiText(..), + renderAnsiText, +) where + +import Control.Applicative + +import Data.Text (Text) +import Data.Text qualified as T + +import Erebos.TextFormat.Types + + +newtype AnsiText = AnsiText { fromAnsiText :: Text } + +renderAnsiText :: FormattedText -> AnsiText +renderAnsiText = AnsiText . go ( Nothing, Nothing ) + where + go cur@( cfg, cbg ) = \case + PlainText text -> text + ConcatenatedText ftexts -> mconcat $ map (go cur) ftexts + FormattedText (CustomTextColor fg bg) ftext -> mconcat + [ ansiColor fg bg + , go ( fg <|> cfg, bg <|> cbg ) ftext + , ansiColor + (if fg /= cfg then cfg <|> Just DefaultColor else Nothing) + (if bg /= cbg then cbg <|> Just DefaultColor else Nothing) + ] + + +ansiColor :: Maybe Color -> Maybe Color -> Text +ansiColor Nothing Nothing = "" +ansiColor (Just fg) Nothing = "\ESC[" <> T.pack (show (colorNum fg)) <> "m" +ansiColor Nothing (Just bg) = "\ESC[" <> T.pack (show (colorNum bg + 10)) <> "m" +ansiColor (Just fg) (Just bg) = "\ESC[" <> T.pack (show (colorNum fg)) <> ";" <> T.pack (show (colorNum bg + 10)) <> "m" + +colorNum :: Color -> Int +colorNum = \case + DefaultColor -> 39 + Black -> 30 + Red -> 31 + Green -> 32 + Yellow -> 33 + Blue -> 34 + Magenta -> 35 + Cyan -> 36 + White -> 37 + BrightBlack -> 90 + BrightRed -> 91 + BrightGreen -> 92 + BrightYellow -> 93 + BrightBlue -> 94 + BrightMagenta -> 95 + BrightCyan -> 96 + BrightWhite -> 97 diff --git a/src/Erebos/TextFormat/Types.hs b/src/Erebos/TextFormat/Types.hs new file mode 100644 index 0000000..a03bc71 --- /dev/null +++ b/src/Erebos/TextFormat/Types.hs @@ -0,0 +1,57 @@ +module Erebos.TextFormat.Types ( + FormattedText(..), + TextStyle(..), + Color(..), +) where + +import Data.String +import Data.Text (Text) + + +data FormattedText + = PlainText Text + | ConcatenatedText [ FormattedText ] + | FormattedText TextStyle FormattedText + +instance IsString FormattedText where + fromString = PlainText . fromString + +instance Semigroup FormattedText where + ConcatenatedText xs <> ConcatenatedText ys = ConcatenatedText (xs ++ ys) + x <> ConcatenatedText ys = ConcatenatedText (x : ys) + ConcatenatedText xs <> y = ConcatenatedText (xs ++ [ y ]) + x <> y = ConcatenatedText [ x, y ] + +instance Monoid FormattedText where + mempty = ConcatenatedText [] + mconcat [] = ConcatenatedText [] + mconcat [ x ] = x + mconcat xs = ConcatenatedText $ concatMap flatten xs + where + flatten (ConcatenatedText ys) = ys + flatten y = [ y ] + + +data TextStyle + = CustomTextColor (Maybe Color) (Maybe Color) + + +data Color + = DefaultColor + | Black + | Red + | Green + | Yellow + | Blue + | Magenta + | Cyan + | White + | BrightBlack + | BrightRed + | BrightGreen + | BrightYellow + | BrightBlue + | BrightMagenta + | BrightCyan + | BrightWhite + deriving (Eq) |