From fb9228d47c94fe9480cff42ebbc4ed815465deb2 Mon Sep 17 00:00:00 2001 From: John MacFarlane Date: Fri, 20 Oct 2023 09:10:35 -0700 Subject: [PATCH 1/3] Typst writer: add `#box` around image to make it inline. An `#image` by itself in typst is a block-level element. To force images to be inline (as they are in pandoc), we need to add a box with an explicit width. When a width is not given in image attributes, we compute one from the image itself, when possible. Closes #9104, supersedes #9105. --- src/Text/Pandoc/Writers/Typst.hs | 28 +++++++++++++++++++++++----- test/writer.typst | 4 ++-- 2 files changed, 25 insertions(+), 7 deletions(-) diff --git a/src/Text/Pandoc/Writers/Typst.hs b/src/Text/Pandoc/Writers/Typst.hs index f8cc651de025..e3c14cf55b86 100644 --- a/src/Text/Pandoc/Writers/Typst.hs +++ b/src/Text/Pandoc/Writers/Typst.hs @@ -17,7 +17,8 @@ module Text.Pandoc.Writers.Typst ( writeTypst ) where import Text.Pandoc.Definition -import Text.Pandoc.Class.PandocMonad ( PandocMonad ) +import Text.Pandoc.Class ( PandocMonad, fetchItem ) +import Text.Pandoc.ImageSize (imageSize, sizeInPoints) import Text.Pandoc.Options ( WriterOptions(..), WrapOption(..), isEnabled ) import Data.Text (Text) import Data.List (intercalate, intersperse) @@ -30,6 +31,7 @@ import Text.Pandoc.Writers.Math (convertMath) import qualified Text.TeXMath as TM import Text.DocLayout import Text.DocTemplates (renderTemplate) +import Control.Monad.Except (catchError) import Text.Pandoc.Extensions (Extension(..)) import Text.Collate.Lang (Lang(..), parseLang) @@ -280,10 +282,26 @@ inlineToTypst inline = then mempty else nowrap $ brackets contents Image (_,_,kvs) _inlines (src,_tit) -> do - let width' = maybe mempty ((", width: " <>) . literal) $ lookup "width" kvs - let height' = maybe mempty ((", height: " <>) . literal) $ - lookup "height" kvs - return $ "#image(" <> doubleQuoted src <> width' <> height' <> ")" + opts <- gets stOptions + let height' = lookup "height" kvs + width' <- case lookup "width" kvs of + Just w -> pure $ Just w + Nothing -> catchError + (do (bs, _mt) <- fetchItem src + case imageSize opts bs of + Right x -> pure $ Just $ T.pack $ + show (fst (sizeInPoints x)) <> "pt" + Left _ -> pure Nothing) + (\_ -> pure Nothing) + let coreImage = "image" <> + parens (doubleQuoted src <> + maybe mempty (\w -> ", width: " <> literal w) width' <> + maybe mempty (\h -> ", height: " <> literal h) height') + case width' of + Nothing -> return $ "#" <> coreImage + -- see #9104; we need a box or the image is treated as block-level: + Just w -> return $ "#box" <> + parens ("width: " <> literal w <> ", " <> coreImage) Note blocks -> do contents <- blocksToTypst blocks return $ "#footnote" <> brackets (chomp contents) diff --git a/test/writer.typst b/test/writer.typst index aa5400e570ba..21303a5949d2 100644 --- a/test/writer.typst +++ b/test/writer.typst @@ -779,13 +779,13 @@ or here: From "Voyage dans la Lune" by Georges Melies \(1902): -#figure([#image("lalune.jpg")], +#figure([#box(width: 150.0pt, image("lalune.jpg", width: 150.0pt))], caption: [ lalune ] ) -Here is a movie #image("movie.jpg") icon. +Here is a movie #box(width: 20.0pt, image("movie.jpg", width: 20.0pt)) icon. #horizontalrule From b70e3e2f1cc32517bc80aec6cb94233692b56e2f Mon Sep 17 00:00:00 2001 From: John MacFarlane Date: Fri, 20 Oct 2023 09:52:19 -0700 Subject: [PATCH 2/3] Handle case where height but not width is provided. --- src/Text/Pandoc/Writers/Typst.hs | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/src/Text/Pandoc/Writers/Typst.hs b/src/Text/Pandoc/Writers/Typst.hs index e3c14cf55b86..3a2e258789f1 100644 --- a/src/Text/Pandoc/Writers/Typst.hs +++ b/src/Text/Pandoc/Writers/Typst.hs @@ -297,11 +297,13 @@ inlineToTypst inline = parens (doubleQuoted src <> maybe mempty (\w -> ", width: " <> literal w) width' <> maybe mempty (\h -> ", height: " <> literal h) height') - case width' of - Nothing -> return $ "#" <> coreImage + case (width', height') of + (Nothing, Nothing) -> return $ "#" <> coreImage -- see #9104; we need a box or the image is treated as block-level: - Just w -> return $ "#box" <> - parens ("width: " <> literal w <> ", " <> coreImage) + (Just w, _) -> return $ "#box" <> + parens ("width: " <> literal w <> ", " <> coreImage) + (_, Just h) -> return $ "#box" <> + parens ("height: " <> literal h <> ", " <> coreImage) Note blocks -> do contents <- blocksToTypst blocks return $ "#footnote" <> brackets (chomp contents) From 4a4cfb1dcdc9fb75b170e930a6e57b62879ad026 Mon Sep 17 00:00:00 2001 From: John MacFarlane Date: Fri, 20 Oct 2023 10:46:28 -0700 Subject: [PATCH 3/3] Make the code more perspicuous. --- src/Text/Pandoc/Writers/Typst.hs | 35 +++++++++++++++++--------------- test/writer.typst | 4 ++-- 2 files changed, 21 insertions(+), 18 deletions(-) diff --git a/src/Text/Pandoc/Writers/Typst.hs b/src/Text/Pandoc/Writers/Typst.hs index 3a2e258789f1..d66897fba0d8 100644 --- a/src/Text/Pandoc/Writers/Typst.hs +++ b/src/Text/Pandoc/Writers/Typst.hs @@ -282,24 +282,27 @@ inlineToTypst inline = then mempty else nowrap $ brackets contents Image (_,_,kvs) _inlines (src,_tit) -> do - opts <- gets stOptions - let height' = lookup "height" kvs - width' <- case lookup "width" kvs of - Just w -> pure $ Just w - Nothing -> catchError - (do (bs, _mt) <- fetchItem src - case imageSize opts bs of - Right x -> pure $ Just $ T.pack $ - show (fst (sizeInPoints x)) <> "pt" - Left _ -> pure Nothing) - (\_ -> pure Nothing) + opts <- gets stOptions + let mbHeight = lookup "height" kvs + let mdWidth = lookup "width" kvs let coreImage = "image" <> parens (doubleQuoted src <> - maybe mempty (\w -> ", width: " <> literal w) width' <> - maybe mempty (\h -> ", height: " <> literal h) height') - case (width', height') of - (Nothing, Nothing) -> return $ "#" <> coreImage - -- see #9104; we need a box or the image is treated as block-level: + maybe mempty (\w -> ", width: " <> literal w) mdWidth <> + maybe mempty (\h -> ", height: " <> literal h) mbHeight) + -- see #9104; we need a box or the image is treated as block-level: + case (mdWidth, mbHeight) of + (Nothing, Nothing) -> do + realWidth <- catchError + (do (bs, _mt) <- fetchItem src + case imageSize opts bs of + Right x -> pure $ Just $ T.pack $ + show (fst (sizeInPoints x)) <> "pt" + Left _ -> pure Nothing) + (\_ -> pure Nothing) + case realWidth of + Just w -> return $ "#box" <> + parens ("width: " <> literal w <> ", " <> coreImage) + Nothing -> return $ "#" <> coreImage (Just w, _) -> return $ "#box" <> parens ("width: " <> literal w <> ", " <> coreImage) (_, Just h) -> return $ "#box" <> diff --git a/test/writer.typst b/test/writer.typst index 21303a5949d2..edcc87cc8cca 100644 --- a/test/writer.typst +++ b/test/writer.typst @@ -779,13 +779,13 @@ or here: From "Voyage dans la Lune" by Georges Melies \(1902): -#figure([#box(width: 150.0pt, image("lalune.jpg", width: 150.0pt))], +#figure([#box(width: 150.0pt, image("lalune.jpg"))], caption: [ lalune ] ) -Here is a movie #box(width: 20.0pt, image("movie.jpg", width: 20.0pt)) icon. +Here is a movie #box(width: 20.0pt, image("movie.jpg")) icon. #horizontalrule