2018-03-18 18:46:28 +01:00
|
|
|
|
{-# LANGUAGE NoImplicitPrelude #-}
|
2018-11-05 22:04:06 +01:00
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2017-03-10 10:16:27 +01:00
|
|
|
|
module Tests.Writers.Muse (tests) where
|
|
|
|
|
|
2018-03-18 18:46:28 +01:00
|
|
|
|
import Prelude
|
2017-06-10 23:39:49 +02:00
|
|
|
|
import Data.Text (unpack)
|
2017-03-14 17:05:36 +01:00
|
|
|
|
import Test.Tasty
|
2017-03-10 10:16:27 +01:00
|
|
|
|
import Tests.Helpers
|
|
|
|
|
import Text.Pandoc
|
2017-10-28 05:28:29 +02:00
|
|
|
|
import Text.Pandoc.Arbitrary ()
|
2017-03-10 10:16:27 +01:00
|
|
|
|
import Text.Pandoc.Builder
|
|
|
|
|
|
2018-11-05 05:27:55 +01:00
|
|
|
|
defopts :: WriterOptions
|
|
|
|
|
defopts = def{ writerWrapText = WrapPreserve,
|
|
|
|
|
writerExtensions = extensionsFromList [Ext_amuse,
|
|
|
|
|
Ext_auto_identifiers] }
|
|
|
|
|
|
2017-03-10 10:16:27 +01:00
|
|
|
|
muse :: (ToPandoc a) => a -> String
|
2018-11-05 05:27:55 +01:00
|
|
|
|
muse = museWithOpts defopts
|
2017-03-10 10:16:27 +01:00
|
|
|
|
|
|
|
|
|
museWithOpts :: (ToPandoc a) => WriterOptions -> a -> String
|
2017-06-10 23:39:49 +02:00
|
|
|
|
museWithOpts opts = unpack . purely (writeMuse opts) . toPandoc
|
2017-03-10 10:16:27 +01:00
|
|
|
|
|
|
|
|
|
infix 4 =:
|
|
|
|
|
(=:) :: (ToString a, ToPandoc a)
|
2017-03-14 17:05:36 +01:00
|
|
|
|
=> String -> (a, String) -> TestTree
|
2017-03-10 10:16:27 +01:00
|
|
|
|
(=:) = test muse
|
|
|
|
|
|
2018-11-05 05:27:55 +01:00
|
|
|
|
noteLocationTestDoc :: Blocks
|
|
|
|
|
noteLocationTestDoc =
|
2018-11-05 22:04:06 +01:00
|
|
|
|
header 1 "First Header" <>
|
|
|
|
|
para ("This is a footnote." <>
|
|
|
|
|
note (para "First note.")) <>
|
|
|
|
|
blockQuote (para ("A note inside a block quote." <>
|
|
|
|
|
note (para "The second note.")) <>
|
|
|
|
|
para "A second paragraph.") <>
|
|
|
|
|
header 1 "Second Header" <>
|
|
|
|
|
para "Some more text."
|
2018-11-05 05:27:55 +01:00
|
|
|
|
|
|
|
|
|
noteLocationTests :: TestTree
|
|
|
|
|
noteLocationTests = testGroup "note location"
|
|
|
|
|
[ test (museWithOpts defopts {writerReferenceLocation=EndOfDocument})
|
|
|
|
|
"footnotes at the end of document" $
|
|
|
|
|
noteLocationTestDoc =?>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
unlines [ "* First Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "This is a footnote.[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "<quote>"
|
|
|
|
|
, "A note inside a block quote.[2]"
|
|
|
|
|
, ""
|
|
|
|
|
, "A second paragraph."
|
|
|
|
|
, "</quote>"
|
|
|
|
|
, ""
|
|
|
|
|
, "* Second Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "Some more text."
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] First note."
|
|
|
|
|
, ""
|
|
|
|
|
, "[2] The second note."
|
|
|
|
|
]
|
2018-11-05 05:27:55 +01:00
|
|
|
|
, test (museWithOpts defopts {writerReferenceLocation=EndOfBlock})
|
|
|
|
|
"footnotes at the end of block" $
|
|
|
|
|
noteLocationTestDoc =?>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
unlines [ "* First Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "This is a footnote.[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] First note."
|
|
|
|
|
, ""
|
|
|
|
|
, "<quote>"
|
|
|
|
|
, "A note inside a block quote.[2]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[2] The second note."
|
|
|
|
|
, ""
|
|
|
|
|
, "A second paragraph."
|
|
|
|
|
, "</quote>"
|
|
|
|
|
, ""
|
|
|
|
|
, "* Second Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "Some more text."
|
|
|
|
|
]
|
2018-11-05 05:27:55 +01:00
|
|
|
|
, test (museWithOpts defopts {writerReferenceLocation=EndOfSection})
|
|
|
|
|
"footnotes at the end of section" $
|
|
|
|
|
noteLocationTestDoc =?>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
unlines [ "* First Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "This is a footnote.[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "<quote>"
|
|
|
|
|
, "A note inside a block quote.[2]"
|
|
|
|
|
, ""
|
|
|
|
|
, "A second paragraph."
|
|
|
|
|
, "</quote>"
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] First note."
|
|
|
|
|
, ""
|
|
|
|
|
, "[2] The second note."
|
|
|
|
|
, ""
|
|
|
|
|
, "* Second Header"
|
|
|
|
|
, ""
|
|
|
|
|
, "Some more text."
|
|
|
|
|
]
|
2018-11-05 05:27:55 +01:00
|
|
|
|
]
|
|
|
|
|
|
2017-03-14 17:05:36 +01:00
|
|
|
|
tests :: [TestTree]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
tests = [ testGroup "block elements"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ "plain" =: plain "Foo bar." =?> "Foo bar."
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, testGroup "paragraphs"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ "single paragraph" =: para "Sample paragraph."
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?> "Sample paragraph."
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "two paragraphs" =: para "First paragraph." <>
|
|
|
|
|
para "Second paragraph."
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?> unlines [ "First paragraph."
|
|
|
|
|
, ""
|
|
|
|
|
, "Second paragraph."
|
|
|
|
|
]
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "line block" =: lineBlock ["Foo", "bar", "baz"]
|
2018-02-28 12:42:43 +01:00
|
|
|
|
=?> unlines [ "> Foo"
|
|
|
|
|
, "> bar"
|
|
|
|
|
, "> baz"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2017-06-26 15:07:45 +02:00
|
|
|
|
, "code block" =: codeBlock "int main(void) {\n\treturn 0;\n}"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?> unlines [ "<example>"
|
|
|
|
|
, "int main(void) {"
|
|
|
|
|
, "\treturn 0;"
|
|
|
|
|
, "}"
|
|
|
|
|
, "</example>"
|
|
|
|
|
]
|
|
|
|
|
, "html raw block" =: rawBlock "html" "<hr>"
|
|
|
|
|
=?> unlines [ "<literal style=\"html\">"
|
|
|
|
|
, "<hr>"
|
|
|
|
|
, "</literal>"
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "block quote" =: blockQuote (para "Foo")
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?> unlines [ "<quote>"
|
|
|
|
|
, "Foo"
|
|
|
|
|
, "</quote>"
|
|
|
|
|
]
|
|
|
|
|
, testGroup "lists"
|
|
|
|
|
[ testGroup "simple lists"
|
|
|
|
|
[
|
2018-11-05 22:04:06 +01:00
|
|
|
|
"ordered list" =: orderedList [ plain "first"
|
|
|
|
|
, plain "second"
|
|
|
|
|
, plain "third"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " 1. first"
|
|
|
|
|
, " 2. second"
|
|
|
|
|
, " 3. third"
|
|
|
|
|
]
|
|
|
|
|
, "ordered list with Roman numerals"
|
|
|
|
|
=: orderedListWith (1, UpperRoman, DefaultDelim)
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ plain "first"
|
|
|
|
|
, plain "second"
|
|
|
|
|
, plain "third"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-03-21 15:32:12 +01:00
|
|
|
|
=?> unlines [ " I. first"
|
|
|
|
|
, " II. second"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, " III. third"
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "bullet list" =: bulletList [ plain "first"
|
|
|
|
|
, plain "second"
|
|
|
|
|
, plain "third"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " - first"
|
|
|
|
|
, " - second"
|
|
|
|
|
, " - third"
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "definition list" =: definitionList [ ("first definition", [plain "first description"])
|
|
|
|
|
, ("second definition", [plain "second description"])
|
|
|
|
|
, ("third definition", [plain "third description"])
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " second definition :: second description"
|
|
|
|
|
, " third definition :: third description"
|
|
|
|
|
]
|
2018-01-19 12:58:15 +01:00
|
|
|
|
, "definition list with multiple descriptions" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
definitionList [ ("first definition", [ plain "first description"
|
|
|
|
|
, plain "second description"
|
|
|
|
|
])
|
|
|
|
|
, ("second definition", [plain "third description"])
|
2018-01-19 12:58:15 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " :: second description"
|
|
|
|
|
, " second definition :: third description"
|
|
|
|
|
]
|
2018-03-07 14:27:56 +01:00
|
|
|
|
, "definition list with empty term" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
definitionList [ ("first definition", [plain "first description"])
|
|
|
|
|
, (mempty, [plain "second description"])
|
|
|
|
|
, (str "", [plain "third description"])
|
2018-03-07 14:27:56 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " <verbatim></verbatim> :: second description"
|
|
|
|
|
, " <verbatim></verbatim> :: third description"
|
|
|
|
|
]
|
2018-04-15 15:07:43 +02:00
|
|
|
|
, "definition list terms starting with space" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
definitionList [ ("first definition", [plain "first description"])
|
|
|
|
|
, (space <> str "foo", [plain "second description"])
|
|
|
|
|
, (str " > bar", [plain "third description"])
|
2018-04-15 15:07:43 +02:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " <verbatim></verbatim> foo :: second description"
|
|
|
|
|
, " <verbatim></verbatim> > bar :: third description"
|
|
|
|
|
]
|
2018-04-16 10:40:18 +02:00
|
|
|
|
, "definition list terms starting with list markers" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
definitionList [ ("first definition", [plain "first description"])
|
|
|
|
|
, (str "-", [plain "second description"])
|
|
|
|
|
, (str "1.", [plain "third description"])
|
2018-04-16 10:40:18 +02:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " <verbatim></verbatim>- :: second description"
|
|
|
|
|
, " <verbatim></verbatim>1. :: third description"
|
|
|
|
|
]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2017-08-08 23:05:49 +02:00
|
|
|
|
-- Test that lists of the same type and style are separated with two blanklines
|
|
|
|
|
, testGroup "sequential lists"
|
|
|
|
|
[ "bullet lists" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
bulletList [ para "First"
|
|
|
|
|
, para "Second"
|
|
|
|
|
, para "Third"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] <>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
bulletList [ para "Fourth"
|
|
|
|
|
, para "Fifth"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] =?>
|
|
|
|
|
unlines [ " - First"
|
|
|
|
|
, " - Second"
|
|
|
|
|
, " - Third"
|
|
|
|
|
, ""
|
|
|
|
|
, ""
|
|
|
|
|
, " - Fourth"
|
|
|
|
|
, " - Fifth"
|
|
|
|
|
]
|
|
|
|
|
, "ordered lists of the same style" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, UpperRoman, DefaultDelim) [ para "First"
|
|
|
|
|
, para "Second"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] <>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, UpperRoman, DefaultDelim) [ para "Third"
|
|
|
|
|
, para "Fourth"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] =?>
|
2018-03-21 15:32:12 +01:00
|
|
|
|
unlines [ " I. First"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
, " II. Second"
|
|
|
|
|
, ""
|
|
|
|
|
, ""
|
2018-03-21 15:32:12 +01:00
|
|
|
|
, " I. Third"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
, " II. Fourth"
|
|
|
|
|
]
|
|
|
|
|
, "ordered lists with equal styles" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedList [ para "First"
|
|
|
|
|
, para "Second"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] <>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, Decimal, DefaultDelim) [ para "Third"
|
|
|
|
|
, para "Fourth"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] =?>
|
|
|
|
|
unlines [ " 1. First"
|
|
|
|
|
, " 2. Second"
|
|
|
|
|
, ""
|
|
|
|
|
, ""
|
|
|
|
|
, " 1. Third"
|
|
|
|
|
, " 2. Fourth"
|
|
|
|
|
]
|
|
|
|
|
, "bullet and ordered lists" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
bulletList [ para "First"
|
|
|
|
|
, para "Second"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] <>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, UpperRoman, DefaultDelim) [ para "Third"
|
|
|
|
|
, para "Fourth"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] =?>
|
|
|
|
|
unlines [ " - First"
|
|
|
|
|
, " - Second"
|
|
|
|
|
, ""
|
2018-03-21 15:32:12 +01:00
|
|
|
|
, " I. Third"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
, " II. Fourth"
|
|
|
|
|
]
|
|
|
|
|
, "different style ordered lists" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, UpperRoman, DefaultDelim) [ para "First"
|
|
|
|
|
, para "Second"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] <>
|
2018-11-05 22:04:06 +01:00
|
|
|
|
orderedListWith (1, Decimal, DefaultDelim) [ para "Third"
|
|
|
|
|
, para "Fourth"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
] =?>
|
2018-03-21 15:32:12 +01:00
|
|
|
|
unlines [ " I. First"
|
2017-08-08 23:05:49 +02:00
|
|
|
|
, " II. Second"
|
|
|
|
|
, ""
|
|
|
|
|
, " 1. Third"
|
|
|
|
|
, " 2. Fourth"
|
|
|
|
|
]
|
|
|
|
|
]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, testGroup "nested lists"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ "nested ordered list" =: orderedList [ plain "First outer"
|
|
|
|
|
, plain "Second outer:" <>
|
|
|
|
|
orderedList [ plain "first"
|
|
|
|
|
, plain "second"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, plain "Third outer"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " 1. First outer"
|
|
|
|
|
, " 2. Second outer:"
|
|
|
|
|
, " 1. first"
|
|
|
|
|
, " 2. second"
|
|
|
|
|
, " 3. Third outer"
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "nested bullet lists" =: bulletList [ plain "First outer"
|
|
|
|
|
, plain "Second outer:" <>
|
|
|
|
|
bulletList [ plain "first"
|
|
|
|
|
, plain "second"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, plain "Third outer"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " - First outer"
|
|
|
|
|
, " - Second outer:"
|
|
|
|
|
, " - first"
|
|
|
|
|
, " - second"
|
|
|
|
|
, " - Third outer"
|
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "nested definition lists" =: definitionList [ ("first definition", [plain "first description"])
|
|
|
|
|
, ("second definition",
|
|
|
|
|
[ plain "second description" <>
|
|
|
|
|
definitionList [ ("first inner definition"
|
|
|
|
|
, [plain "first inner description"])
|
|
|
|
|
, ( "second inner definition"
|
|
|
|
|
, [plain "second inner description"])
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
]
|
|
|
|
|
=?> unlines [ " first definition :: first description"
|
|
|
|
|
, " second definition :: second description"
|
2018-02-23 20:07:30 +01:00
|
|
|
|
, " first inner definition :: first inner description"
|
|
|
|
|
, " second inner definition :: second inner description"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "list item starting with list" =: bulletList [ bulletList [ plain "foo"] ] =?> " - - foo"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2017-07-12 17:16:02 +02:00
|
|
|
|
-- Check that list is intended with one space even inside a quote
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "List inside block quote" =: blockQuote (orderedList [ plain "first"
|
|
|
|
|
, plain "second"
|
|
|
|
|
, plain "third"
|
2017-07-12 17:16:02 +02:00
|
|
|
|
])
|
|
|
|
|
=?> unlines [ "<quote>"
|
|
|
|
|
, " 1. first"
|
|
|
|
|
, " 2. second"
|
|
|
|
|
, " 3. third"
|
|
|
|
|
, "</quote>"
|
|
|
|
|
]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, testGroup "headings"
|
|
|
|
|
[ "normal heading" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
header 1 "foo" =?> "* foo"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, "heading levels" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
header 1 "First level" <>
|
|
|
|
|
header 3 "Third level" =?>
|
2017-03-10 10:16:27 +01:00
|
|
|
|
unlines [ "* First level"
|
|
|
|
|
, ""
|
|
|
|
|
, "*** Third level"
|
|
|
|
|
]
|
2017-12-21 13:33:54 +01:00
|
|
|
|
, "heading with ID" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
headerWith ("bar", [], []) 2 "Foo" =?>
|
2018-04-02 14:55:04 +02:00
|
|
|
|
unlines [ "#bar"
|
|
|
|
|
, "** Foo"
|
2017-12-21 13:33:54 +01:00
|
|
|
|
]
|
2018-09-04 10:00:44 +02:00
|
|
|
|
, "empty heading" =: header 4 mempty =?> "**** <verbatim></verbatim>"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, "horizontal rule" =: horizontalRule =?> "----"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "escape horizontal rule" =: para "----" =?> "<verbatim></verbatim>----"
|
|
|
|
|
, "escape long horizontal rule" =: para "----------" =?> "<verbatim></verbatim>----------"
|
|
|
|
|
, "don't escape horizontal inside paragraph" =: para "foo ---- bar" =?> "foo ---- bar"
|
|
|
|
|
, "escape nonbreaking space" =: para "~~" =?> "<verbatim>~~</verbatim>"
|
|
|
|
|
, "escape > in the beginning of line" =: para "> foo bar" =?> "<verbatim></verbatim>> foo bar"
|
2018-09-11 10:35:25 +02:00
|
|
|
|
, "escape string with > and space in the beginning of line" =: para (str "> foo bar") =?> "<verbatim></verbatim>> foo bar"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, testGroup "tables"
|
|
|
|
|
[ "table without header" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
let rows = [[para "Para 1.1", para "Para 1.2"]
|
|
|
|
|
,[para "Para 2.1", para "Para 2.2"]]
|
2018-04-05 19:13:15 +02:00
|
|
|
|
in table mempty [(AlignDefault,0.0),(AlignDefault,0.0)]
|
|
|
|
|
[mempty, mempty] rows
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?>
|
2017-05-07 21:41:38 +02:00
|
|
|
|
unlines [ " Para 1.1 | Para 1.2"
|
|
|
|
|
, " Para 2.1 | Para 2.2"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, "table with header" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
let headers = [plain "header 1", plain "header 2"]
|
|
|
|
|
rows = [[para "Para 1.1", para "Para 1.2"]
|
|
|
|
|
,[para "Para 2.1", para "Para 2.2"]]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
in simpleTable headers rows
|
|
|
|
|
=?>
|
2017-05-07 21:41:38 +02:00
|
|
|
|
unlines [ " header 1 || header 2"
|
|
|
|
|
, " Para 1.1 | Para 1.2"
|
|
|
|
|
, " Para 2.1 | Para 2.2"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, "table with header and caption" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
let caption = "Table 1"
|
|
|
|
|
headers = [plain "header 1", plain "header 2"]
|
|
|
|
|
rows = [[para "Para 1.1", para "Para 1.2"]
|
|
|
|
|
,[para "Para 2.1", para "Para 2.2"]]
|
2018-04-05 19:13:15 +02:00
|
|
|
|
in table caption [(AlignDefault,0.0),(AlignDefault,0.0)]
|
|
|
|
|
headers rows
|
2017-05-07 21:41:38 +02:00
|
|
|
|
=?> unlines [ " header 1 || header 2"
|
|
|
|
|
, " Para 1.1 | Para 1.2"
|
|
|
|
|
, " Para 2.1 | Para 2.2"
|
|
|
|
|
, " |+ Table 1 +|"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-11-14 12:35:58 +01:00
|
|
|
|
, "table inside bullet list" =:
|
|
|
|
|
bulletList [simpleTable [] [[para "foo", para "bar"]
|
|
|
|
|
,[para "bat", para "baz"]]]
|
|
|
|
|
=?> unlines [ " - foo | bar"
|
|
|
|
|
, " bat | baz"
|
|
|
|
|
]
|
2018-11-13 16:43:46 +01:00
|
|
|
|
, "table with one column" =:
|
|
|
|
|
let headers = []
|
|
|
|
|
rows = [[para "Para 1"]
|
|
|
|
|
,[para "Para 2"]]
|
|
|
|
|
in simpleTable headers rows
|
|
|
|
|
=?>
|
|
|
|
|
unlines [ "+--------+"
|
|
|
|
|
, "| Para 1 |"
|
|
|
|
|
, "+--------+"
|
|
|
|
|
, "| Para 2 |"
|
|
|
|
|
, "+--------+"
|
|
|
|
|
]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-01-22 14:24:44 +01:00
|
|
|
|
, "div with bullet list" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
divWith nullAttr (bulletList [para "foo"]) =?>
|
2018-01-22 14:24:44 +01:00
|
|
|
|
unlines [ " - foo" ] -- Making sure bullets are indented
|
2017-03-10 10:16:27 +01:00
|
|
|
|
-- Null is trivial
|
|
|
|
|
]
|
|
|
|
|
, testGroup "inline elements"
|
|
|
|
|
[ testGroup "string"
|
|
|
|
|
[ "string" =: str "foo" =?> "foo"
|
|
|
|
|
, "escape footnote" =: str "[1]" =?> "<verbatim>[1]</verbatim>"
|
2018-03-14 10:23:51 +01:00
|
|
|
|
, "do not escape brackets" =: str "[12ab]" =?> "[12ab]"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, "escape verbatim close tag" =: str "foo</verbatim>bar"
|
|
|
|
|
=?> "<verbatim>foo<</verbatim><verbatim>/verbatim>bar</verbatim>"
|
2018-03-14 10:23:51 +01:00
|
|
|
|
, "escape link-like text" =: str "[[https://www.example.org]]"
|
|
|
|
|
=?> "<verbatim>[[https://www.example.org]]</verbatim>"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, "escape pipe to avoid accidental tables" =: str "foo | bar"
|
|
|
|
|
=?> "<verbatim>foo | bar</verbatim>"
|
2017-11-22 14:01:57 +01:00
|
|
|
|
, "escape hash to avoid accidental anchors" =: text "#foo bar"
|
|
|
|
|
=?> "<verbatim>#foo</verbatim> bar"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, "escape definition list markers" =: str "::" =?> "<verbatim>::</verbatim>"
|
2018-03-01 23:39:16 +01:00
|
|
|
|
, "normalize strings before escaping" =: fromList [Str ":", Str ":"] =?> "<verbatim>::</verbatim>"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
-- We don't want colons to be escaped if they can't be confused
|
|
|
|
|
-- with definition list item markers.
|
|
|
|
|
, "do not escape colon" =: str ":" =?> ":"
|
2018-03-25 17:11:38 +02:00
|
|
|
|
, "escape - to avoid accidental unordered lists" =: text " - foo" =?> "<verbatim></verbatim> - foo"
|
2018-03-07 13:19:36 +01:00
|
|
|
|
, "escape - inside a list to avoid accidental nested unordered lists" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
bulletList [ para "foo" <>
|
|
|
|
|
para "- bar"
|
2018-03-07 13:19:36 +01:00
|
|
|
|
] =?>
|
|
|
|
|
unlines [ " - foo"
|
|
|
|
|
, ""
|
2018-03-25 17:11:38 +02:00
|
|
|
|
, " <verbatim></verbatim>- bar"
|
2018-03-07 13:19:36 +01:00
|
|
|
|
]
|
2018-09-11 10:35:25 +02:00
|
|
|
|
, "escape strings starting with - inside a list" =:
|
|
|
|
|
bulletList [ para (str "foo") <>
|
|
|
|
|
para (str "- bar")
|
|
|
|
|
] =?>
|
|
|
|
|
unlines [ " - foo"
|
|
|
|
|
, ""
|
|
|
|
|
, " <verbatim></verbatim>- bar"
|
|
|
|
|
]
|
2018-09-11 02:50:59 +02:00
|
|
|
|
, "escape - inside a note" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
note (para "- foo") =?>
|
2018-09-11 02:50:59 +02:00
|
|
|
|
unlines [ "[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] <verbatim></verbatim>- foo"
|
|
|
|
|
]
|
2018-09-11 13:10:20 +02:00
|
|
|
|
, "escape - after softbreak in note" =:
|
|
|
|
|
note (para (str "foo" <> softbreak <> str "- bar")) =?>
|
|
|
|
|
unlines [ "[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] foo"
|
|
|
|
|
, " <verbatim></verbatim>- bar"
|
|
|
|
|
]
|
2018-03-25 17:11:38 +02:00
|
|
|
|
, "escape ; to avoid accidental comments" =: text "; foo" =?> "<verbatim></verbatim>; foo"
|
2018-09-11 10:35:25 +02:00
|
|
|
|
, "escape strings starting with ; and space" =: str "; foo" =?> "<verbatim></verbatim>; foo"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "escape ; after softbreak" =: "foo" <> softbreak <> "; bar" =?> "foo\n<verbatim></verbatim>; bar"
|
|
|
|
|
, "escape ; after linebreak" =: "foo" <> linebreak <> "; bar" =?> "foo<br>\n<verbatim></verbatim>; bar"
|
2018-03-31 21:44:24 +02:00
|
|
|
|
, "do not escape ; inside paragraph" =: text "foo ; bar" =?> "foo ; bar"
|
2018-09-16 19:59:59 +02:00
|
|
|
|
, "escape newlines" =: str "foo\nbar" =?> "foo bar"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, testGroup "emphasis"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ "emphasis" =: emph "foo" =?> "*foo*"
|
|
|
|
|
, "emphasis inside word" =: "foo" <> emph "bar" <> "baz" =?> "foo<em>bar</em>baz"
|
|
|
|
|
, "emphasis before comma" =: emph "foo" <> ", bar" =?> "*foo*, bar"
|
|
|
|
|
, "emphasis before period" =: emph "foobar" <> "." =?> "*foobar*."
|
2018-08-30 16:10:46 +02:00
|
|
|
|
, "empty emphasis" =: emph mempty =?> "<em></em>"
|
|
|
|
|
, "empty strong" =: strong mempty =?> "<strong></strong>"
|
|
|
|
|
, "empty strong emphasis" =: strong (emph mempty) =?> "**<em></em>**"
|
|
|
|
|
, "empty emphasized strong" =: emph (strong mempty) =?> "*<strong></strong>*"
|
2018-09-10 23:38:55 +02:00
|
|
|
|
, "emphasized empty string" =: emph (str "") =?> "<em></em>"
|
|
|
|
|
, "strong empty string" =: strong (str "") =?> "<strong></strong>"
|
|
|
|
|
, "strong emphasized empty string" =: strong (emph (str "")) =?> "**<em></em>**"
|
|
|
|
|
, "emphasized strong empty string" =: emph (strong (str "")) =?> "*<strong></strong>*"
|
2018-09-11 10:49:11 +02:00
|
|
|
|
, "emphasized string with space" =: emph (str " ") =?> "<em> </em>"
|
|
|
|
|
, "emphasized string ending with space" =: emph (str "foo ") =?> "<em>foo </em>"
|
|
|
|
|
, "emphasized string with tab" =: emph (str "\t") =?> "<em>\t</em>"
|
2018-09-11 00:36:11 +02:00
|
|
|
|
, "emphasized space between empty strings" =: emph (str "" <> space <> str "") =?> "<em> </em>"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "strong" =: strong "foo" =?> "**foo**"
|
|
|
|
|
, "strong inside word" =: "foo" <> strong "bar" <> "baz" =?> "foo<strong>bar</strong>baz"
|
|
|
|
|
, "strong emphasis" =: strong (emph "foo") =?> "***foo***"
|
|
|
|
|
, "strong after emphasis" =: emph "foo" <> strong "bar" =?> "*foo*<strong>bar</strong>"
|
|
|
|
|
, "strong emphasis after emphasis" =: emph "foo" <> strong (emph "bar") =?> "*foo*<strong>*bar*</strong>"
|
|
|
|
|
, "strong in the end of emphasis" =: emph ("foo" <> strong "bar") =?> "*foo<strong>bar</strong>*"
|
2018-10-21 19:19:29 +02:00
|
|
|
|
, "switch to lightweight markup after <em> tag" =:
|
|
|
|
|
strong (str "foo") <> emph (str "bar") <> strong (str "baz") =?>
|
|
|
|
|
"**foo**<em>bar</em>**baz**"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "strikeout" =: strikeout "foo" =?> "<del>foo</del>"
|
|
|
|
|
, "space at the beginning of emphasis" =: emph " foo" =?> "<em> foo</em>"
|
|
|
|
|
, "space at the end of emphasis" =: emph "foo " =?> "<em>foo </em>"
|
|
|
|
|
, "space at the beginning of strong" =: strong " foo" =?> "<strong> foo</strong>"
|
|
|
|
|
, "space at the end of strong" =: strong "foo " =?> "<strong>foo </strong>"
|
|
|
|
|
, "space at the beginning of strong emphasis" =: strong (emph " foo") =?> "**<em> foo</em>**"
|
|
|
|
|
, "space at the end of strong emphasis" =: strong (emph "foo ") =?> "**<em>foo </em>**"
|
|
|
|
|
, "space at the beginning of emphasiszed strong" =: emph (strong " foo") =?> "*<strong> foo</strong>*"
|
|
|
|
|
, "space at the end of emphasized strong" =: emph (strong "foo ") =?> "*<strong>foo </strong>*"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "superscript" =: superscript "foo" =?> "<sup>foo</sup>"
|
|
|
|
|
, "subscript" =: subscript "foo" =?> "<sub>foo</sub>"
|
|
|
|
|
, "smallcaps" =: smallcaps "foo" =?> "*foo*"
|
2018-08-30 16:10:46 +02:00
|
|
|
|
, "smallcaps near emphasis" =: emph (str "foo") <> smallcaps (str "bar") =?> "*foobar*"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "single quoted" =: singleQuoted "foo" =?> "‘foo’"
|
|
|
|
|
, "double quoted" =: doubleQuoted "foo" =?> "“foo”"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
-- Cite is trivial
|
|
|
|
|
, testGroup "code"
|
2018-08-30 16:10:46 +02:00
|
|
|
|
[ "simple" =: code "foo" =?> "=foo="
|
|
|
|
|
, "empty" =: code "" =?> "<code></code>"
|
|
|
|
|
, "space" =: code " " =?> "<code> </code>"
|
|
|
|
|
, "space at the beginning" =: code " foo" =?> "<code> foo</code>"
|
|
|
|
|
, "space at the end" =: code "foo " =?> "<code>foo </code>"
|
|
|
|
|
, "use tags for =" =: code "foo = bar" =?> "<code>foo = bar</code>"
|
2017-11-21 15:41:29 +01:00
|
|
|
|
, "escape tag" =: code "<code>foo = bar</code> baz" =?> "<code><code>foo = bar<</code><code>/code> baz</code>"
|
2018-08-30 16:10:46 +02:00
|
|
|
|
, "normalization with attributes" =: codeWith ("",["haskell"],[]) "foo" <> code "bar" =?> "=foobar="
|
|
|
|
|
, "code tag" =: code "<code>foo</code>" =?> "=<code>foo</code>="
|
|
|
|
|
, "normalization" =: code "</co" <> code "de>" <> code "=" =?> "<code><</code><code>/code>=</code>"
|
|
|
|
|
, "normalization with empty string" =: code "</co" <> str "" <> code "de>" <> code "=" =?> "<code><</code><code>/code>=</code>"
|
|
|
|
|
, "emphasized code" =: emph (code "foo") =?> "*=foo=*"
|
|
|
|
|
, "strong code" =: strong (code "foo") =?> "**=foo=**"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, testGroup "spaces"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
[ "space" =: "a" <> space <> "b" =?> "a b"
|
|
|
|
|
, "soft break" =: "a" <> softbreak <> "b" =?> "a\nb"
|
2018-03-25 17:30:32 +02:00
|
|
|
|
, test (museWithOpts def{ writerWrapText = WrapNone })
|
2018-11-05 22:04:06 +01:00
|
|
|
|
"remove soft break" $ "a" <> softbreak <> "b"
|
|
|
|
|
=?> ("a b" :: String)
|
|
|
|
|
, "line break" =: "a" <> linebreak <> "b" =?> "a<br>\nb"
|
|
|
|
|
, "no newline after line break in header" =: header 1 ("a" <> linebreak <> "b") =?> "* a<br>b"
|
|
|
|
|
, "no softbreak in header" =: header 1 ("a" <> softbreak <> "b") =?> "* a b"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, testGroup "math"
|
|
|
|
|
[ "inline math" =: math "2^3" =?> "2<sup>3</sup>"
|
2018-01-29 21:27:40 +01:00
|
|
|
|
, "display math" =: displayMath "2^3" =?> "2<sup>3</sup>"
|
2018-08-30 16:10:46 +02:00
|
|
|
|
, "multiple letters in inline math" =: math "abc" =?> "*abc*"
|
2018-03-05 17:38:11 +01:00
|
|
|
|
, "expand math before normalization" =: math "[" <> str "2]" =?> "<verbatim>[2]</verbatim>"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "multiple math expressions inside one inline list" =: math "5_4" <> ", " <> displayMath "3^2" =?> "5<sub>4</sub>, 3<sup>2</sup>"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, "raw inline"
|
|
|
|
|
=: rawInline "html" "<mark>marked text</mark>"
|
|
|
|
|
=?> "<literal style=\"html\"><mark>marked text</mark></literal>"
|
|
|
|
|
, testGroup "links"
|
|
|
|
|
[ "link with description" =: link "https://example.com" "" (str "Link 1")
|
|
|
|
|
=?> "[[https://example.com][Link 1]]"
|
|
|
|
|
, "link without description" =: link "https://example.com" "" (str "https://example.com")
|
|
|
|
|
=?> "[[https://example.com]]"
|
|
|
|
|
-- Internal links in Muse include '#'
|
|
|
|
|
, "link to anchor" =: link "#intro" "" (str "Introduction")
|
|
|
|
|
=?> "[[#intro][Introduction]]"
|
|
|
|
|
-- According to Emacs Muse manual, links to images should be prefixed with "URL:"
|
|
|
|
|
, "link to image with description" =: link "1.png" "" (str "Link to image")
|
|
|
|
|
=?> "[[URL:1.png][Link to image]]"
|
|
|
|
|
, "link to image without description" =: link "1.png" "" (str "1.png")
|
|
|
|
|
=?> "[[URL:1.png]]"
|
2018-03-20 09:15:43 +01:00
|
|
|
|
|
|
|
|
|
, testGroup "escape brackets in links"
|
|
|
|
|
[ "link with description"
|
|
|
|
|
=: link "https://example.com/foo].txt" "" (str "Description")
|
|
|
|
|
=?> "[[https://example.com/foo%5D.txt][Description]]"
|
|
|
|
|
, "link without description"
|
|
|
|
|
=: link "https://example.com/foo].txt" "" (str "https://example.com/foo].txt")
|
|
|
|
|
=?> "[[https://example.com/foo%5D.txt][<verbatim>https://example.com/foo].txt</verbatim>]]"
|
|
|
|
|
, "image link with description"
|
|
|
|
|
=: link "foo]bar.png" "" (str "Image link")
|
|
|
|
|
=?> "[[URL:foo%5Dbar.png][Image link]]"
|
|
|
|
|
, "image link without description"
|
|
|
|
|
=: link "foo]bar.png" "" (str "foo]bar.png")
|
|
|
|
|
=?> "[[URL:foo%5Dbar.png][<verbatim>foo]bar.png</verbatim>]]"
|
|
|
|
|
]
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
, "image" =: image "image.png" "Image 1" (str "") =?> "[[image.png][Image 1]]"
|
2018-02-03 16:36:38 +01:00
|
|
|
|
, "image with width" =:
|
|
|
|
|
imageWith ("", [], [("width", "60%")]) "image.png" "Image" (str "") =?>
|
|
|
|
|
"[[image.png 60][Image]]"
|
2018-05-07 13:54:20 +02:00
|
|
|
|
, "left-aligned image with width" =:
|
|
|
|
|
imageWith ("", ["align-left"], [("width", "60%")]) "image.png" "Image" (str "") =?>
|
|
|
|
|
"[[image.png 60 l][Image]]"
|
|
|
|
|
, "right-aligned image with width" =:
|
|
|
|
|
imageWith ("", ["align-right"], [("width", "60%")]) "image.png" "Image" (str "") =?>
|
|
|
|
|
"[[image.png 60 r][Image]]"
|
2018-03-20 13:01:53 +01:00
|
|
|
|
, "escape brackets in image title" =: image "image.png" "Foo]bar" (str "") =?> "[[image.png][<verbatim>Foo]bar</verbatim>]]"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "note" =: note (plain "Foo")
|
2017-03-10 10:16:27 +01:00
|
|
|
|
=?> unlines [ "[1]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[1] Foo"
|
|
|
|
|
]
|
2018-11-05 05:27:55 +01:00
|
|
|
|
, noteLocationTests
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "span with class" =: spanWith ("",["foobar"],[]) "Some text"
|
2018-03-24 23:18:33 +01:00
|
|
|
|
=?> "<class name=\"foobar\">Some text</class>"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "span without class" =: spanWith ("",[],[]) "Some text"
|
2018-03-24 23:18:33 +01:00
|
|
|
|
=?> "<class>Some text</class>"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "span with anchor" =: spanWith ("anchor", [], []) mempty <> "Foo bar"
|
2018-03-11 23:40:23 +01:00
|
|
|
|
=?> "#anchor Foo bar"
|
2018-09-04 10:00:44 +02:00
|
|
|
|
, "empty span with anchor" =: spanWith ("anchor", [], []) mempty
|
2018-03-24 23:18:33 +01:00
|
|
|
|
=?> "#anchor"
|
2018-09-04 10:00:44 +02:00
|
|
|
|
, "empty span without class and anchor" =: spanWith ("", [], []) mempty
|
2018-03-25 17:11:38 +02:00
|
|
|
|
=?> "<class></class>"
|
2018-11-05 22:04:06 +01:00
|
|
|
|
, "span with class and anchor" =: spanWith ("anchor", ["foo"], []) "bar"
|
2018-03-11 23:40:23 +01:00
|
|
|
|
=?> "#anchor <class name=\"foo\">bar</class>"
|
2018-03-24 21:51:38 +01:00
|
|
|
|
, "adjacent spans" =: spanWith ("", ["syllable"], []) (str "wa") <>
|
|
|
|
|
spanWith ("", ["syllable"], []) (str "ter")
|
|
|
|
|
=?> "<class name=\"syllable\">wa</class><class name=\"syllable\">ter</class>"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, testGroup "combined"
|
|
|
|
|
[ "emph word before" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para ("foo" <> emph "bar") =?>
|
2017-03-10 10:16:27 +01:00
|
|
|
|
"foo<em>bar</em>"
|
|
|
|
|
, "emph word after" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para (emph "foo" <> "bar") =?>
|
2017-03-10 10:16:27 +01:00
|
|
|
|
"<em>foo</em>bar"
|
|
|
|
|
, "emph quoted" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para (doubleQuoted (emph "foo")) =?>
|
2018-08-30 16:10:46 +02:00
|
|
|
|
"“*foo*”"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
, "strong word before" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para ("foo" <> strong "bar") =?>
|
2017-03-10 10:16:27 +01:00
|
|
|
|
"foo<strong>bar</strong>"
|
|
|
|
|
, "strong word after" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para (strong "foo" <> "bar") =?>
|
2017-03-10 10:16:27 +01:00
|
|
|
|
"<strong>foo</strong>bar"
|
|
|
|
|
, "strong quoted" =:
|
2018-11-05 22:04:06 +01:00
|
|
|
|
para (singleQuoted (strong "foo")) =?>
|
2018-08-30 16:10:46 +02:00
|
|
|
|
"‘**foo**’"
|
2017-03-10 10:16:27 +01:00
|
|
|
|
]
|
|
|
|
|
]
|
|
|
|
|
]
|