Elm Markdown

Pure Elm markdown parsing and rendering.

Based on the latest CommonMark Spec, with some differences. Demo.

Basic Usage

import Markdown

view : Html msg
view =
    div []
        <| Markdown.toHtml Nothing "# Heading with *emphasis*"

Supported Syntax


To create a heading, add one to six # symbols before your heading text. The number of # you use will determine the size of the heading.

# Heading 1
## Heading 2
###### Heading 6

You can also use = or - after a paragraph for level 1 or 2 heading.

Heading 1

Heading 2


Lines starting with a > will be a block quote.

> Block quote


Use a sequence of single backticks (`) to output code. The text within the backticks will not be formatted. To format code or text into its own distinct block, use at least three backticks or four spaces or tab.

Example of `inline code`

    Example of block code

Example of block code with defined language

If the language in the fenced code block is defined, it will be added a class="language-optionalLang" to the code element.


You can create an inline link by wrapping link text in brackets [ ], and then wrapping the URL in parentheses ( ), with a optional title using single quotes, double quotes or parentheses.

Do you know the Elm [slack channel](https://elmlang.slack.com/ "title")?

Or create a reference link:

[slackLink]: https://elmlang.slack.com/ 'title'

Do you know the Elm [slack channel][slackLink]?

Or even:

[slack channel]: https://elmlang.slack.com/ (title)

Do you know the Elm [slack channel]?

All examples output the same html.

Autolinks and emails are supported with < >:

Autolink: <http://elm-lang.org/>
Email link: <[email protected]>


You can make a list by preceding one or more lines of text with - or *.

- Unordered list
  * Nested unordered list
5. Ordered list starting at 5
    1) Nested ordered list starting at 1

Paragraphs and line breaks

You can create a new paragraph by leaving a blank line between lines of text.

Here's a paragraph with soft break line
at the end.

Here's a paragraph with hard break line\
at the end.

By default, soft line break (\n) will be rendered as it is, unless it's preceded by two spaces or \, which will output hard break line (<br>).

You can customize to always render soft line breaks as hard line breaks setting softAsHardLineBreak = True in the options.

Thematic Break Line

You can create a thematic break line with a sequence of three or more matching -, _, or * characters.



You can create emphasis using the * or _ characters. Double emphasis is strong emphasis.

*Emphasis*, **strong emphasis**, ***both***

_Emphasis_, __strong emphasis__, ___both___


You can insert images using the following syntax:

![alt text](src-url "title")

For more information about supported syntax and parsing rules, see CommonMark Spec.

Differences from CommonMark

  • No entity references encoding/decoding support (e.g.: &nbsp;, &amp;, &copy;);
  • No decimal numeric characters decoding support (e.g.: &#35;, &#1234;, &#992;);
  • No hexadecimal numeric character decoding support (e.g.: &#X22;, &#XD06;, &#xcab;);
  • No comment tag support (<!-- -->);
  • No CDATA tag support (<![CDATA[ ]]>);
  • No processing instruction tag support (<? ?>);
  • No declaration tag support (<! >);
  • No malformed html tag support (e.g.: <div class);
  • No balanced parenthesis in inline link's url support (e.g.: [link](url() "title"), use [link](<url()> "title") instead);
  • To create a HTML block, wich is not surrounded by paragraph tag (<p>), start and finish a paragraph with the html tag you want the HTML block to be, with no blankline between the start and end tag. E.g.:
    First paragraph.
                Table element
    Next paragraph.


Use Markdown.toHtml to specify parsing options:

import Markdown
import Markdown.Config exposing (Options, defaultOptions)

customOptions : Options
customOptions =
    { defaultOptions
        | softAsHardLineBreak = True

view : Html msg
view =
    div []
        <| Markdown.toHtml (Just customOptions)
        <| "# Heading with *emphasis*"

The following options are available:

type alias Options =
    { softAsHardLineBreak : Bool
    , rawHtml : HtmlOption

type HtmlOption
    = ParseUnsafe
    | Sanitize SanitizeOptions
    | DontParse

type alias SanitizeOptions =
    { allowedHtmlElements : List String
    , allowedHtmlAttributes : List String
  • softAsHardLineBreak: Default False. If set to True, will render \n as <br>.
  • rawHtml: Default Sanitize defaultSanitizeOptions. You can choose to not parse any html tags (DontParse), parse any html tag without any sanitization (ParseUnsafe) or parse only specific html elements and attributes (Sanitize SanitizeOptions).

Default allowed elements and attributes:

defaultSanitizeOptions : SanitizeOptions
defaultSanitizeOptions =
    { allowedHtmlElements =
        [ "address", "article", "aside", "b", "blockquote", "br"
        , "caption", "center", "cite", "code", "col", "colgroup"
        , "dd", "details", "div", "dl", "dt", "figcaption", "figure"
        , "footer", "h1", "h2", "h3", "h4", "h5", "h6", "hr", "i"
        , "legend", "li", "menu", "menuitem", "nav", "ol", "optgroup"
        , "option", "p", "pre", "section", "strike", "summary"
        , "small", "table", "tbody", "td", "tfoot", "th", "thead"
        , "tr", "ul" ]
    , allowedHtmlAttributes =
        [ "name", "class" ]

Note: Only basic sanitization is provided. If you are receiving user submitted content, you should use a specific library to sanitize user input.


You can customize how each markdown element is rendered by first parsing the markdown string into blocks, then mapping the resulting blocks through a custom renderer, created with the help of Blocks.defaultHtml and/or Inline.defaultHtml, then concatenate the resulting list.

Example of rendering: - All blockquotes as a detail element; - Images using figure and figcaption; - Links not starting with http://elm-lang.org with a target="_blank" attribute.

import Html exposing (..)
import Html.Attributes exposing (..)
import Markdown.Block as Block exposing (Block(..))
import Markdown.Inline as Inline exposing (Inline(..))

view : Html msg
view =
        |> Block.parse Nothing -- using Config.defaultOptions
        |> List.map (customHtmlBlock)
        |> List.concat
        |> article []

customHtmlBlock : Block b i -> List (Html msg)
customHtmlBlock block =
    case block of
        BlockQuote blocks ->
            List.map customHtmlBlock blocks
                |> List.concat
                |> details []
                |> flip (::) []

        _ ->
                (Just customHtmlBlock)
                (Just customHtmlInline)

customHtmlInline : Inline i -> Html msg
customHtmlInline inline =
    case inline of
        Image url maybeTitle inlines ->
            figure []
                [ img
                    [ alt (Inline.extractText inlines)
                    , src url
                    , title (Maybe.withDefault "" maybeTitle)
                    ] []
                , figcaption []
                    [ text (Inline.extractText inlines) ]

        Link url maybeTitle inlines ->
            if String.startsWith "http://elm-lang.org" url then
                a [ href url
                  , title (Maybe.withDefault "" maybeTitle)
                  ] (List.map customHtmlInline inlines)

                a [ href url
                  , title (Maybe.withDefault "" maybeTitle)
                  , target "_blank"
                  , rel "noopener noreferrer"
                  ] (List.map customHtmlInline inlines)

        _ ->
            Inline.defaultHtml (Just customHtmlInline) inline


Performance improvement is being made constantly. Feedbacks are welcome.

Parsing a 1.2MB markdown text file to html in my notebook using node:

Advanced Usage

Todo: - Custom Blocks - Custom Inlines


Thank you John MacFarlane, for creating CommonMark specification and tests.

Thank you everyone who gave feedback. Special thanks to Jan Tojnar, for discussing about the API.

Thank you Evan for bringing joy to the frontend.