Man reader: handle tables.
This still needs a bit of work. In particular, we should try to produce Plain cells when possible. See #4982.
This commit is contained in:
parent
20032c440b
commit
8f9ab3db25
1 changed files with 63 additions and 17 deletions
|
@ -34,11 +34,13 @@ Conversion of man to 'Pandoc' document.
|
|||
module Text.Pandoc.Readers.Man (readMan) where
|
||||
|
||||
import Prelude
|
||||
import Data.Char (toLower)
|
||||
import Data.Default (Default)
|
||||
import Control.Monad (liftM, mzero, guard)
|
||||
import Control.Monad.Trans (lift)
|
||||
import Control.Monad.Except (throwError)
|
||||
import Text.Pandoc.Class (PandocMonad(..), report)
|
||||
import Data.Maybe (catMaybes)
|
||||
import Data.Maybe (catMaybes, isJust)
|
||||
import Data.List (intersperse, intercalate)
|
||||
import qualified Data.Text as T
|
||||
import Text.Pandoc.Builder as B
|
||||
|
@ -73,20 +75,18 @@ readMan opts txt = do
|
|||
(Foldable.toList . unGroffTokens $ tokenz)
|
||||
either throwError return eitherdoc
|
||||
|
||||
where
|
||||
|
||||
readWithMTokens :: PandocMonad m
|
||||
readWithMTokens :: PandocMonad m
|
||||
=> ParserT [GroffToken] ManState m a -- ^ parser
|
||||
-> ManState -- ^ initial state
|
||||
-> [GroffToken] -- ^ input
|
||||
-> m (Either PandocError a)
|
||||
readWithMTokens parser state input =
|
||||
readWithMTokens parser state input =
|
||||
let leftF = PandocParsecError . intercalate "\n" $ show <$> input
|
||||
in mapLeft leftF `liftM` runParserT parser state "source" input
|
||||
|
||||
mapLeft :: (a -> c) -> Either a b -> Either c b
|
||||
mapLeft f (Left x) = Left $ f x
|
||||
mapLeft _ (Right r) = Right r
|
||||
mapLeft :: (a -> c) -> Either a b -> Either c b
|
||||
mapLeft f (Left x) = Left $ f x
|
||||
mapLeft _ (Right r) = Right r
|
||||
|
||||
|
||||
parseMan :: PandocMonad m => ManParser m Pandoc
|
||||
|
@ -113,10 +113,56 @@ parseTable :: PandocMonad m => ManParser m Blocks
|
|||
parseTable = do
|
||||
let isMTable (MTable{}) = True
|
||||
isMTable _ = False
|
||||
MTable _opts _aligns _rows pos <- msatisfy isMTable
|
||||
MTable _opts aligns rows pos <- msatisfy isMTable
|
||||
case aligns of
|
||||
[as] -> do
|
||||
let as' = map (columnTypeToAlignment . columnType) as
|
||||
if all isJust as'
|
||||
then do
|
||||
let alignments = catMaybes as'
|
||||
let (headerRow', bodyRows') =
|
||||
case rows of
|
||||
(h:[x]:bs)
|
||||
| isHrule x -> (h, bs)
|
||||
_ -> ([], rows)
|
||||
headerRow <- mapM parseTableCell headerRow'
|
||||
bodyRows <- mapM (mapM parseTableCell) bodyRows'
|
||||
return $ B.table mempty (zip alignments (repeat 0.0))
|
||||
headerRow bodyRows
|
||||
else fallback pos
|
||||
_ -> fallback pos
|
||||
|
||||
where
|
||||
|
||||
parseTableCell ts = do
|
||||
st <- getState
|
||||
let ts' = Foldable.toList $ unGroffTokens ts
|
||||
res <- lift $ readWithMTokens (mconcat <$> many parseBlock <* eof) st ts'
|
||||
case res of
|
||||
Left e -> throwError e
|
||||
Right x -> return x
|
||||
|
||||
isHrule :: GroffTokens -> Bool
|
||||
isHrule (GroffTokens ss) =
|
||||
case Foldable.toList ss of
|
||||
[MLine [RoffStr [c]]] -> c `elem` ['_','-','=']
|
||||
_ -> False
|
||||
|
||||
fallback pos = do
|
||||
report $ SkippedContent "TABLE" pos
|
||||
return $ B.para (B.text "TABLE")
|
||||
|
||||
columnTypeToAlignment :: Char -> Maybe Alignment
|
||||
columnTypeToAlignment c =
|
||||
case toLower c of
|
||||
'a' -> Just AlignLeft
|
||||
'c' -> Just AlignCenter
|
||||
'l' -> Just AlignLeft
|
||||
'n' -> Just AlignRight
|
||||
'r' -> Just AlignRight
|
||||
_ -> Nothing
|
||||
|
||||
|
||||
parseNewParagraph :: PandocMonad m => ManParser m Blocks
|
||||
parseNewParagraph = do
|
||||
mmacro "P" <|> mmacro "PP" <|> mmacro "LP" <|> memptyLine
|
||||
|
|
Loading…
Reference in a new issue