2006-12-20 20:54:23 +00:00
|
|
|
{-
|
|
|
|
Copyright (C) 2006 John MacFarlane <jgm at berkeley dot edu>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
-}
|
|
|
|
|
2006-12-20 06:50:14 +00:00
|
|
|
{- |
|
|
|
|
Module : Text.Pandoc.Readers.HTML
|
|
|
|
Copyright : Copyright (C) 2006 John MacFarlane
|
|
|
|
License : GNU GPL, version 2 or above
|
|
|
|
|
|
|
|
Maintainer : John MacFarlane <jgm at berkeley dot edu>
|
2006-12-20 20:20:10 +00:00
|
|
|
Stability : alpha
|
2006-12-20 06:50:14 +00:00
|
|
|
Portability : portable
|
|
|
|
|
|
|
|
Conversion of HTML to 'Pandoc' document.
|
|
|
|
-}
|
2006-10-17 14:22:29 +00:00
|
|
|
module Text.Pandoc.Readers.HTML (
|
|
|
|
readHtml,
|
|
|
|
rawHtmlInline,
|
|
|
|
rawHtmlBlock,
|
|
|
|
anyHtmlBlockTag,
|
2006-12-30 22:51:49 +00:00
|
|
|
anyHtmlInlineTag,
|
|
|
|
anyHtmlTag,
|
|
|
|
anyHtmlEndTag,
|
|
|
|
htmlEndTag,
|
|
|
|
extractTagType,
|
|
|
|
htmlBlockElement
|
2006-10-17 14:22:29 +00:00
|
|
|
) where
|
|
|
|
|
|
|
|
import Text.ParserCombinators.Parsec
|
|
|
|
import Text.ParserCombinators.Pandoc
|
|
|
|
import Text.Pandoc.Definition
|
|
|
|
import Text.Pandoc.Shared
|
2007-01-02 00:40:12 +00:00
|
|
|
import Text.Pandoc.Entities ( decodeEntities, entityToChar )
|
2006-10-17 14:22:29 +00:00
|
|
|
import Maybe ( fromMaybe )
|
2007-01-24 20:26:06 +00:00
|
|
|
import Data.List ( intersect, takeWhile, dropWhile )
|
2007-01-24 19:40:32 +00:00
|
|
|
import Data.Char ( toUpper, toLower, isAlphaNum )
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
-- | Convert HTML-formatted string to 'Pandoc' document.
|
|
|
|
readHtml :: ParserState -- ^ Parser state
|
|
|
|
-> String -- ^ String to parse
|
|
|
|
-> Pandoc
|
|
|
|
readHtml = readWith parseHtml
|
|
|
|
|
|
|
|
-- for testing
|
|
|
|
testString :: String -> IO ()
|
|
|
|
testString = testStringWith parseHtml
|
|
|
|
|
|
|
|
--
|
|
|
|
-- Constants
|
|
|
|
--
|
|
|
|
|
2006-12-20 06:50:14 +00:00
|
|
|
inlineHtmlTags = ["a", "abbr", "acronym", "b", "basefont", "bdo", "big",
|
|
|
|
"br", "cite", "code", "dfn", "em", "font", "i", "img",
|
|
|
|
"input", "kbd", "label", "q", "s", "samp", "select",
|
|
|
|
"small", "span", "strike", "strong", "sub", "sup",
|
|
|
|
"textarea", "tt", "u", "var"]
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
--
|
|
|
|
-- HTML utility functions
|
|
|
|
--
|
|
|
|
|
|
|
|
-- | Read blocks until end tag.
|
|
|
|
blocksTilEnd tag = try (do
|
|
|
|
blocks <- manyTill (do {b <- block; spaces; return b}) (htmlEndTag tag)
|
|
|
|
return blocks)
|
|
|
|
|
|
|
|
-- | Read inlines until end tag.
|
|
|
|
inlinesTilEnd tag = try (do
|
|
|
|
inlines <- manyTill inline (htmlEndTag tag)
|
|
|
|
return inlines)
|
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
-- | Extract type from a tag: e.g. 'br' from '<br>'
|
2007-01-24 17:43:39 +00:00
|
|
|
extractTagType :: String -> String
|
2007-01-24 20:26:06 +00:00
|
|
|
extractTagType ('<':rest) =
|
|
|
|
let isSpaceOrSlash c = c `elem` "/ \n\t" in
|
|
|
|
map toLower $ takeWhile isAlphaNum $ dropWhile isSpaceOrSlash rest
|
2007-01-24 17:43:39 +00:00
|
|
|
extractTagType _ = ""
|
2006-10-17 14:22:29 +00:00
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
-- | Parse any HTML tag (closing or opening) and return text of tag
|
2006-10-17 14:22:29 +00:00
|
|
|
anyHtmlTag = try (do
|
|
|
|
char '<'
|
|
|
|
spaces
|
|
|
|
tag <- many1 alphaNum
|
2006-12-30 22:51:49 +00:00
|
|
|
attribs <- htmlAttributes
|
2006-10-17 14:22:29 +00:00
|
|
|
spaces
|
|
|
|
ender <- option "" (string "/")
|
|
|
|
let ender' = if (null ender) then "" else " /"
|
|
|
|
spaces
|
|
|
|
char '>'
|
|
|
|
return ("<" ++ tag ++ attribs ++ ender' ++ ">"))
|
|
|
|
|
|
|
|
anyHtmlEndTag = try (do
|
|
|
|
char '<'
|
|
|
|
spaces
|
|
|
|
char '/'
|
|
|
|
spaces
|
|
|
|
tagType <- many1 alphaNum
|
|
|
|
spaces
|
|
|
|
char '>'
|
|
|
|
return ("</" ++ tagType ++ ">"))
|
|
|
|
|
|
|
|
htmlTag :: String -> GenParser Char st (String, [(String, String)])
|
|
|
|
htmlTag tag = try (do
|
|
|
|
char '<'
|
|
|
|
spaces
|
|
|
|
stringAnyCase tag
|
|
|
|
attribs <- many htmlAttribute
|
|
|
|
spaces
|
|
|
|
option "" (string "/")
|
|
|
|
spaces
|
|
|
|
char '>'
|
|
|
|
return (tag, (map (\(name, content, raw) -> (name, content)) attribs)))
|
|
|
|
|
|
|
|
-- parses a quoted html attribute value
|
|
|
|
quoted quoteChar = do
|
2006-12-20 06:50:14 +00:00
|
|
|
result <- between (char quoteChar) (char quoteChar)
|
|
|
|
(many (noneOf [quoteChar]))
|
2006-10-17 14:22:29 +00:00
|
|
|
return (result, [quoteChar])
|
|
|
|
|
|
|
|
htmlAttributes = do
|
|
|
|
attrList <- many htmlAttribute
|
|
|
|
return (concatMap (\(name, content, raw) -> raw) attrList)
|
|
|
|
|
|
|
|
htmlAttribute = htmlRegularAttribute <|> htmlMinimizedAttribute
|
|
|
|
|
|
|
|
-- minimized boolean attribute (no = and value)
|
|
|
|
htmlMinimizedAttribute = try (do
|
2007-01-24 19:07:35 +00:00
|
|
|
many1 space
|
2006-10-17 14:22:29 +00:00
|
|
|
name <- many1 (choice [letter, oneOf ".-_:"])
|
|
|
|
spaces
|
|
|
|
notFollowedBy (char '=')
|
|
|
|
let content = name
|
|
|
|
return (name, content, (" " ++ name)))
|
|
|
|
|
|
|
|
htmlRegularAttribute = try (do
|
2007-01-24 19:07:35 +00:00
|
|
|
many1 space
|
2006-10-17 14:22:29 +00:00
|
|
|
name <- many1 (choice [letter, oneOf ".-_:"])
|
|
|
|
spaces
|
|
|
|
char '='
|
|
|
|
spaces
|
|
|
|
(content, quoteStr) <- choice [ (quoted '\''),
|
|
|
|
(quoted '"'),
|
2006-12-20 06:50:14 +00:00
|
|
|
(do
|
|
|
|
a <- many (alphaNum <|> (oneOf "-._:"))
|
|
|
|
return (a,"")) ]
|
2006-12-30 22:51:49 +00:00
|
|
|
return (name, content,
|
2006-12-20 06:50:14 +00:00
|
|
|
(" " ++ name ++ "=" ++ quoteStr ++ content ++ quoteStr)))
|
2006-10-17 14:22:29 +00:00
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
-- | Parse an end tag of type 'tag'
|
2006-10-17 14:22:29 +00:00
|
|
|
htmlEndTag tag = try (do
|
|
|
|
char '<'
|
|
|
|
spaces
|
|
|
|
char '/'
|
|
|
|
spaces
|
|
|
|
stringAnyCase tag
|
|
|
|
spaces
|
|
|
|
char '>'
|
|
|
|
return ("</" ++ tag ++ ">"))
|
|
|
|
|
|
|
|
-- | Returns @True@ if the tag is an inline tag.
|
|
|
|
isInline tag = (extractTagType tag) `elem` inlineHtmlTags
|
|
|
|
|
|
|
|
anyHtmlBlockTag = try (do
|
|
|
|
tag <- choice [anyHtmlTag, anyHtmlEndTag]
|
2006-12-20 06:50:14 +00:00
|
|
|
if isInline tag then fail "inline tag" else return tag)
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
anyHtmlInlineTag = try (do
|
|
|
|
tag <- choice [ anyHtmlTag, anyHtmlEndTag ]
|
2006-12-20 06:50:14 +00:00
|
|
|
if isInline tag then return tag else fail "not an inline tag")
|
2006-10-17 14:22:29 +00:00
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
-- | Parses material between script tags.
|
|
|
|
-- Scripts must be treated differently, because they can contain '<>' etc.
|
2006-10-17 14:22:29 +00:00
|
|
|
htmlScript = try (do
|
|
|
|
open <- string "<script"
|
|
|
|
rest <- manyTill anyChar (htmlEndTag "script")
|
|
|
|
return (open ++ rest ++ "</script>"))
|
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
htmlBlockElement = choice [ htmlScript, htmlComment, xmlDec, definition ]
|
|
|
|
|
2006-11-26 07:01:37 +00:00
|
|
|
rawHtmlBlock = try (do
|
|
|
|
notFollowedBy' (choice [htmlTag "/body", htmlTag "/html"])
|
2006-12-30 22:51:49 +00:00
|
|
|
body <- htmlBlockElement <|> anyHtmlBlockTag
|
2006-10-17 14:22:29 +00:00
|
|
|
sp <- (many space)
|
|
|
|
state <- getState
|
2006-12-20 06:50:14 +00:00
|
|
|
if stateParseRaw state then return (RawHtml (body ++ sp)) else return Null)
|
2006-10-17 14:22:29 +00:00
|
|
|
|
2006-12-30 22:51:49 +00:00
|
|
|
-- | Parses an HTML comment.
|
2006-10-17 14:22:29 +00:00
|
|
|
htmlComment = try (do
|
|
|
|
string "<!--"
|
|
|
|
comment <- manyTill anyChar (try (string "-->"))
|
|
|
|
return ("<!--" ++ comment ++ "-->"))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- parsing documents
|
|
|
|
--
|
|
|
|
|
|
|
|
xmlDec = try (do
|
|
|
|
string "<?"
|
|
|
|
rest <- manyTill anyChar (char '>')
|
|
|
|
return ("<?" ++ rest ++ ">"))
|
|
|
|
|
|
|
|
definition = try (do
|
|
|
|
string "<!"
|
|
|
|
rest <- manyTill anyChar (char '>')
|
|
|
|
return ("<!" ++ rest ++ ">"))
|
|
|
|
|
|
|
|
nonTitleNonHead = try (do
|
|
|
|
notFollowedBy' (htmlTag "title")
|
|
|
|
notFollowedBy' (htmlTag "/head")
|
|
|
|
result <- choice [do {rawHtmlBlock; return ' '}, anyChar]
|
|
|
|
return result)
|
|
|
|
|
|
|
|
parseTitle = try (do
|
|
|
|
(tag, attribs) <- htmlTag "title"
|
|
|
|
contents <- inlinesTilEnd tag
|
|
|
|
spaces
|
|
|
|
return contents)
|
|
|
|
|
|
|
|
-- parse header and return meta-information (for now, just title)
|
|
|
|
parseHead = try (do
|
|
|
|
htmlTag "head"
|
|
|
|
spaces
|
|
|
|
skipMany nonTitleNonHead
|
|
|
|
contents <- option [] parseTitle
|
|
|
|
skipMany nonTitleNonHead
|
|
|
|
htmlTag "/head"
|
|
|
|
return (contents, [], ""))
|
|
|
|
|
|
|
|
skipHtmlTag tag = option ("",[]) (htmlTag tag)
|
|
|
|
|
|
|
|
-- h1 class="title" representation of title in body
|
|
|
|
bodyTitle = try (do
|
|
|
|
(tag, attribs) <- htmlTag "h1"
|
|
|
|
cl <- case (extractAttribute "class" attribs) of
|
|
|
|
Just "title" -> do {return ""}
|
|
|
|
otherwise -> fail "not title"
|
|
|
|
inlinesTilEnd "h1"
|
|
|
|
return "")
|
|
|
|
|
|
|
|
parseHtml = do
|
|
|
|
sepEndBy (choice [xmlDec, definition, htmlComment]) spaces
|
|
|
|
skipHtmlTag "html"
|
|
|
|
spaces
|
|
|
|
(title, authors, date) <- option ([], [], "") parseHead
|
|
|
|
spaces
|
|
|
|
skipHtmlTag "body"
|
|
|
|
spaces
|
|
|
|
option "" bodyTitle -- skip title in body, because it's represented in meta
|
|
|
|
blocks <- parseBlocks
|
|
|
|
spaces
|
|
|
|
option "" (htmlEndTag "body")
|
|
|
|
spaces
|
|
|
|
option "" (htmlEndTag "html")
|
|
|
|
many anyChar -- ignore anything after </html>
|
|
|
|
eof
|
|
|
|
state <- getState
|
|
|
|
let keyBlocks = stateKeyBlocks state
|
|
|
|
return (Pandoc (Meta title authors date) (blocks ++ (reverse keyBlocks)))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- parsing blocks
|
|
|
|
--
|
|
|
|
|
|
|
|
parseBlocks = do
|
|
|
|
spaces
|
|
|
|
result <- sepEndBy block spaces
|
|
|
|
return result
|
|
|
|
|
|
|
|
block = choice [ codeBlock, header, hrule, list, blockQuote, para, plain,
|
|
|
|
rawHtmlBlock ] <?> "block"
|
|
|
|
|
|
|
|
--
|
|
|
|
-- header blocks
|
|
|
|
--
|
|
|
|
|
|
|
|
header = choice (map headerLevel (enumFromTo 1 5)) <?> "header"
|
|
|
|
|
|
|
|
headerLevel n = try (do
|
|
|
|
let level = "h" ++ show n
|
|
|
|
(tag, attribs) <- htmlTag level
|
|
|
|
contents <- inlinesTilEnd level
|
|
|
|
return (Header n (normalizeSpaces contents)))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- hrule block
|
|
|
|
--
|
|
|
|
|
|
|
|
hrule = try (do
|
|
|
|
(tag, attribs) <- htmlTag "hr"
|
|
|
|
state <- getState
|
2006-12-20 06:50:14 +00:00
|
|
|
if (not (null attribs)) && (stateParseRaw state)
|
|
|
|
then -- in this case we want to parse it as raw html
|
|
|
|
unexpected "attributes in hr"
|
|
|
|
else return HorizontalRule)
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
--
|
|
|
|
-- code blocks
|
|
|
|
--
|
|
|
|
|
|
|
|
codeBlock = choice [ preCodeBlock, bareCodeBlock ] <?> "code block"
|
|
|
|
|
|
|
|
preCodeBlock = try (do
|
|
|
|
htmlTag "pre"
|
|
|
|
spaces
|
|
|
|
htmlTag "code"
|
|
|
|
result <- manyTill anyChar (htmlEndTag "code")
|
|
|
|
spaces
|
|
|
|
htmlEndTag "pre"
|
2006-11-26 07:01:37 +00:00
|
|
|
return (CodeBlock (stripTrailingNewlines (decodeEntities result))))
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
bareCodeBlock = try (do
|
|
|
|
htmlTag "code"
|
|
|
|
result <- manyTill anyChar (htmlEndTag "code")
|
2006-11-26 07:01:37 +00:00
|
|
|
return (CodeBlock (stripTrailingNewlines (decodeEntities result))))
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
--
|
|
|
|
-- block quotes
|
|
|
|
--
|
|
|
|
|
|
|
|
blockQuote = try (do
|
|
|
|
tag <- htmlTag "blockquote"
|
|
|
|
spaces
|
|
|
|
blocks <- blocksTilEnd "blockquote"
|
|
|
|
return (BlockQuote blocks))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- list blocks
|
|
|
|
--
|
|
|
|
|
|
|
|
list = choice [ bulletList, orderedList ] <?> "list"
|
|
|
|
|
|
|
|
orderedList = try (do
|
|
|
|
tag <- htmlTag "ol"
|
|
|
|
spaces
|
|
|
|
items <- sepEndBy1 listItem spaces
|
|
|
|
htmlEndTag "ol"
|
|
|
|
return (OrderedList items))
|
|
|
|
|
|
|
|
bulletList = try (do
|
|
|
|
tag <- htmlTag "ul"
|
|
|
|
spaces
|
|
|
|
items <- sepEndBy1 listItem spaces
|
|
|
|
htmlEndTag "ul"
|
|
|
|
return (BulletList items))
|
|
|
|
|
|
|
|
listItem = try (do
|
|
|
|
tag <- htmlTag "li"
|
|
|
|
spaces
|
|
|
|
blocks <- blocksTilEnd "li"
|
|
|
|
return blocks)
|
|
|
|
|
|
|
|
--
|
|
|
|
-- paragraph block
|
|
|
|
--
|
|
|
|
|
|
|
|
para = try (do
|
|
|
|
tag <- htmlTag "p"
|
|
|
|
result <- inlinesTilEnd "p"
|
|
|
|
return (Para (normalizeSpaces result)))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- plain block
|
|
|
|
--
|
|
|
|
|
|
|
|
plain = do
|
|
|
|
result <- many1 inline
|
|
|
|
return (Plain (normalizeSpaces result))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- inline
|
|
|
|
--
|
|
|
|
|
|
|
|
inline = choice [ text, special ] <?> "inline"
|
|
|
|
|
|
|
|
text = choice [ entity, strong, emph, code, str, linebreak, whitespace ] <?> "text"
|
|
|
|
|
2006-12-20 06:50:14 +00:00
|
|
|
special = choice [ link, image, rawHtmlInline ] <?>
|
|
|
|
"link, inline html, or image"
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
entity = try (do
|
|
|
|
char '&'
|
2006-12-20 06:50:14 +00:00
|
|
|
body <- choice [(many1 letter), (try (do
|
|
|
|
char '#'
|
|
|
|
num <- many1 digit
|
|
|
|
return ("#" ++ num)))]
|
2006-10-17 14:22:29 +00:00
|
|
|
char ';'
|
2007-01-02 00:40:12 +00:00
|
|
|
return (Str [fromMaybe '?' (entityToChar ("&" ++ body ++ ";"))]))
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
code = try (do
|
|
|
|
htmlTag "code"
|
|
|
|
result <- manyTill anyChar (htmlEndTag "code")
|
2006-12-20 06:50:14 +00:00
|
|
|
-- remove internal line breaks, leading and trailing space,
|
|
|
|
-- and decode entities
|
|
|
|
let result' = decodeEntities $ removeLeadingTrailingSpace $
|
|
|
|
joinWithSep " " $ lines result
|
2006-10-17 14:22:29 +00:00
|
|
|
return (Code result'))
|
|
|
|
|
|
|
|
rawHtmlInline = do
|
|
|
|
result <- choice [htmlScript, anyHtmlInlineTag]
|
|
|
|
state <- getState
|
2006-12-20 06:50:14 +00:00
|
|
|
if stateParseRaw state then return (HtmlInline result) else return (Str "")
|
2006-10-17 14:22:29 +00:00
|
|
|
|
|
|
|
betweenTags tag = try (do
|
|
|
|
htmlTag tag
|
|
|
|
result <- inlinesTilEnd tag
|
|
|
|
return (normalizeSpaces result))
|
|
|
|
|
|
|
|
emph = try (do
|
|
|
|
result <- choice [betweenTags "em", betweenTags "it"]
|
|
|
|
return (Emph result))
|
|
|
|
|
|
|
|
strong = try (do
|
|
|
|
result <- choice [betweenTags "b", betweenTags "strong"]
|
|
|
|
return (Strong result))
|
|
|
|
|
|
|
|
whitespace = do
|
|
|
|
many1 space
|
|
|
|
return Space
|
|
|
|
|
|
|
|
-- hard line break
|
|
|
|
linebreak = do
|
|
|
|
htmlTag "br"
|
2007-01-03 20:52:12 +00:00
|
|
|
option ' ' newline
|
2006-10-17 14:22:29 +00:00
|
|
|
return LineBreak
|
|
|
|
|
|
|
|
str = do
|
|
|
|
result <- many1 (noneOf "<& \t\n")
|
|
|
|
return (Str (decodeEntities result))
|
|
|
|
|
|
|
|
--
|
|
|
|
-- links and images
|
|
|
|
--
|
|
|
|
|
|
|
|
-- extract contents of attribute (attribute names are case-insensitive)
|
|
|
|
extractAttribute name [] = Nothing
|
|
|
|
extractAttribute name ((attrName, contents):rest) =
|
|
|
|
let name' = map toLower name
|
|
|
|
attrName' = map toLower attrName in
|
|
|
|
if (attrName' == name') then Just contents else extractAttribute name rest
|
|
|
|
|
|
|
|
link = try (do
|
|
|
|
(tag, attributes) <- htmlTag "a"
|
|
|
|
url <- case (extractAttribute "href" attributes) of
|
|
|
|
Just url -> do {return url}
|
|
|
|
Nothing -> fail "no href"
|
|
|
|
let title = fromMaybe "" (extractAttribute "title" attributes)
|
|
|
|
label <- inlinesTilEnd "a"
|
|
|
|
ref <- generateReference url title
|
|
|
|
return (Link (normalizeSpaces label) ref))
|
|
|
|
|
|
|
|
image = try (do
|
|
|
|
(tag, attributes) <- htmlTag "img"
|
|
|
|
url <- case (extractAttribute "src" attributes) of
|
|
|
|
Just url -> do {return url}
|
|
|
|
Nothing -> fail "no src"
|
|
|
|
let title = fromMaybe "" (extractAttribute "title" attributes)
|
|
|
|
let alt = fromMaybe "" (extractAttribute "alt" attributes)
|
|
|
|
ref <- generateReference url title
|
|
|
|
return (Image [Str alt] ref))
|