Add syntax highlighting to code blocks using Pygments

Implementation was pretty straight forward after getting the regex right to treat ‘=html’ blocks as raw.

Changes located in the run_jotdown function and the new highlightcode_ function, as well as fiddling with css.

def highlight_code(code: str, language: str = None) -> str:
    """
    Highlight code using Pygments with specified or guessed language.
    """
    try:
        if language:
            lexer = get_lexer_by_name(language.lower())
        else:
            lexer = guess_lexer(code)

        formatter = HtmlFormatter(
            style=config["syntax_highlighting"]["style"],
            linenos="table"
            if config["syntax_highlighting"].get("line_numbers", False)
            else False,
            cssclass="highlight",
        )
        return highlight(code, lexer, formatter)
    except ClassNotFound:
        # If language isn't found, return code wrapped in pre tags
        return f"<pre><code>{code}</code></pre>"


def run_jotdown(plaintext: str) -> str:
    """
    Modified to handle code blocks with syntax highlighting.
    Fixed to properly handle both raw HTML and HTML code blocks.
    """
    CODE_BLOCK_RE = re.compile(r"\`\`\`\`*(=html|\s*(?:(\w+)\n))?(.*?)\`\`\`\`*", re.DOTALL)
    code_blocks = []
    marker_template = "§CODE_BLOCK_{}§"

    def save_code_block(match):
        raw_html_marker = match.group(1)
        language = match.group(2)
        code = match.group(3).strip()

        # Check if this is a raw HTML block
        if raw_html_marker == "=html":
            return f"\`\`\`=html\n{code}\n\`\`\`"

        # For all other cases, including 'html' language, highlight the code
        highlighted = highlight_code(code, language)
        marker = marker_template.format(len(code_blocks))
        code_blocks.append(highlighted)
        return f"\`\`\`=html\n{marker}\n\`\`\`"

    # First, replace all code blocks with markers
    processed_text = CODE_BLOCK_RE.sub(save_code_block, plaintext)

    # Run through jotdown
    html = run("jotdown", input=processed_text, text=True, capture_output=True).stdout

    # Replace markers with actual highlighted code
    for i, code in enumerate(code_blocks):
        marker = marker_template.format(i)
        html = html.replace(marker, code)

    return html

Note that in the above snippet, the backticks (`) have been escaped to prevent the markup on this page from breaking, remove the backslashes if you use this code yourself.

Click here to see the CSS I settled on
:root {
  --highlight-font-size: 0.9em;
  --highlight-border-radius: 6px;
  --highlight-padding: 1rem;
  --highlight-line-number-color: #6e7681;
}

.highlight {
  font-size: var(--highlight-font-size, 0.9em);
  border-radius: var(--highlight-border-radius, 6px);
  padding: var(--highlight-padding, 1rem);
}

.highlight pre {
  margin: 0;
  overflow-x: auto;
}

.highlight table {
  border-spacing: 0;
  border: none;
  margin: 0;
}

.highlight table td {
  padding: 0;
  border: none;
}

.highlight .linenos {
  user-select: none;
  padding-right: 1rem;
  color: var(--highlight-line-number-color);
  text-align: right;
  width: 1%;
  white-space: nowrap;
}

@media (prefers-color-scheme: dark) {
  .highlight {
    background: var(--code-background, #262220) !important;
  }
  pre {
    line-height: 125%;
  }
  td.linenos .normal {
    color: #4e4e4e;
    background-color: transparent;
    padding-left: 5px;
    padding-right: 5px;
  }
  span.linenos {
    color: #4e4e4e;
    background-color: transparent;
    padding-left: 5px;
    padding-right: 5px;
  }
  td.linenos .special {
    color: #8f9494;
    background-color: #ffffc0;
    padding-left: 5px;
    padding-right: 5px;
  }
  span.linenos.special {
    color: #8f9494;
    background-color: #ffffc0;
    padding-left: 5px;
    padding-right: 5px;
  }
  .highlight .hll {
    background-color: #ddd0c0;
  }
  .highlight {
    background: #262220;
    color: #ddd0c0;
  }
  .highlight .c {
    color: #70757a;
  } /* Comment */
  .highlight .err {
    color: #af5f5f;
  } /* Error */
  .highlight .esc {
    color: #ddd0c0;
  } /* Escape */
  .highlight .g {
    color: #ddd0c0;
  } /* Generic */
  .highlight .k {
    color: #919191;
  } /* Keyword */
  .highlight .l {
    color: #af875f;
  } /* Literal */
  .highlight .n {
    color: #ddd0c0;
  } /* Name */
  .highlight .o {
    color: #878787;
  } /* Operator */
  .highlight .x {
    color: #ddd0c0;
  } /* Other */
  .highlight .p {
    color: #ddd0c0;
  } /* Punctuation */
  .highlight .ch {
    color: #8f9f9f;
  } /* Comment.Hashbang */
  .highlight .cm {
    color: #70757a;
  } /* Comment.Multiline */
  .highlight .cp {
    color: #fdd0c0;
  } /* Comment.Preproc */
  .highlight .cpf {
    color: #c9b98f;
  } /* Comment.PreprocFile */
  .highlight .c1 {
    color: #70757a;
  } /* Comment.Single */
  .highlight .cs {
    color: #af5f5f;
  } /* Comment.Special */
  .highlight .gd {
    color: #bb6868;
  } /* Generic.Deleted */
  .highlight .ge {
    color: #ddd0c0;
    font-style: italic;
  } /* Generic.Emph */
  .highlight .ges {
    color: #ddd0c0;
  } /* Generic.EmphStrong */
  .highlight .gr {
    color: #af5f5f;
  } /* Generic.Error */
  .highlight .gh {
    color: #ddd0c0;
  } /* Generic.Heading */
  .highlight .gi {
    color: #849155;
  } /* Generic.Inserted */
  .highlight .go {
    color: #ddd0c0;
  } /* Generic.Output */
  .highlight .gp {
    color: #ddd0c0;
  } /* Generic.Prompt */
  .highlight .gs {
    color: #ddd0c0;
    font-weight: bold;
  } /* Generic.Strong */
  .highlight .gu {
    color: #ddd0c0;
  } /* Generic.Subheading */
  .highlight .gt {
    color: #af5f5f;
  } /* Generic.Traceback */
  .highlight .kc {
    color: #875f5f;
  } /* Keyword.Constant */
  .highlight .kd {
    color: #875f5f;
  } /* Keyword.Declaration */
  .highlight .kn {
    color: #875f5f;
  } /* Keyword.Namespace */
  .highlight .kp {
    color: #919191;
  } /* Keyword.Pseudo */
  .highlight .kr {
    color: #b46276;
  } /* Keyword.Reserved */
  .highlight .kt {
    color: #af875f;
  } /* Keyword.Type */
  .highlight .ld {
    color: #af875f;
  } /* Literal.Date */
  .highlight .m {
    color: #87afaf;
  } /* Literal.Number */
  .highlight .s {
    color: #c9b98f;
  } /* Literal.String */
  .highlight .na {
    color: #ddd0c0;
  } /* Name.Attribute */
  .highlight .nb {
    color: #ddd0c0;
  } /* Name.Builtin */
  .highlight .nc {
    color: #875f5f;
  } /* Name.Class */
  .highlight .no {
    color: #af8787;
  } /* Name.Constant */
  .highlight .nd {
    color: #fdd0c0;
  } /* Name.Decorator */
  .highlight .ni {
    color: #ddd0c0;
  } /* Name.Entity */
  .highlight .ne {
    color: #877575;
  } /* Name.Exception */
  .highlight .nf {
    color: #fdd0c0;
  } /* Name.Function */
  .highlight .nl {
    color: #ddd0c0;
  } /* Name.Label */
  .highlight .nn {
    color: #ddd0c0;
  } /* Name.Namespace */
  .highlight .nx {
    color: #ddd0c0;
  } /* Name.Other */
  .highlight .py {
    color: #dfaf87;
  } /* Name.Property */
  .highlight .nt {
    color: #87afaf;
  } /* Name.Tag */
  .highlight .nv {
    color: #ddd0c0;
  } /* Name.Variable */
  .highlight .ow {
    color: #878787;
  } /* Operator.Word */
  .highlight .pm {
    color: #ddd0c0;
  } /* Punctuation.Marker */
  .highlight .w {
    color: #ddd0c0;
  } /* Text.Whitespace */
  .highlight .mb {
    color: #87afaf;
  } /* Literal.Number.Bin */
  .highlight .mf {
    color: #87afaf;
  } /* Literal.Number.Float */
  .highlight .mh {
    color: #87afaf;
  } /* Literal.Number.Hex */
  .highlight .mi {
    color: #87afaf;
  } /* Literal.Number.Integer */
  .highlight .mo {
    color: #87afaf;
  } /* Literal.Number.Oct */
  .highlight .sa {
    color: #dfaf87;
  } /* Literal.String.Affix */
  .highlight .sb {
    color: #c9b98f;
  } /* Literal.String.Backtick */
  .highlight .sc {
    color: #c9b98f;
  } /* Literal.String.Char */
  .highlight .dl {
    color: #c9b98f;
  } /* Literal.String.Delimiter */
  .highlight .sd {
    color: #878787;
  } /* Literal.String.Doc */
  .highlight .s2 {
    color: #c9b98f;
  } /* Literal.String.Double */
  .highlight .se {
    color: #af5f5f;
  } /* Literal.String.Escape */
  .highlight .sh {
    color: #c9b98f;
  } /* Literal.String.Heredoc */
  .highlight .si {
    color: #af5f5f;
  } /* Literal.String.Interpol */
  .highlight .sx {
    color: #fdd0c0;
  } /* Literal.String.Other */
  .highlight .sr {
    color: #af5f5f;
  } /* Literal.String.Regex */
  .highlight .s1 {
    color: #c9b98f;
  } /* Literal.String.Single */
  .highlight .ss {
    color: #af5f5f;
  } /* Literal.String.Symbol */
  .highlight .bp {
    color: #87afaf;
  } /* Name.Builtin.Pseudo */
  .highlight .fm {
    color: #fdd0c0;
  } /* Name.Function.Magic */
  .highlight .vc {
    color: #ddd0c0;
  } /* Name.Variable.Class */
  .highlight .vg {
    color: #ddd0c0;
  } /* Name.Variable.Global */
  .highlight .vi {
    color: #ddd0c0;
  } /* Name.Variable.Instance */
  .highlight .vm {
    color: #ddd0c0;
  } /* Name.Variable.Magic */
  .highlight .il {
    color: #87afaf;
  } /* Literal.Number.Integer.Long */
}

@media (prefers-color-scheme: light) {
  .highlight {
    background: var(--code-background, #e7daca) !important;
  }
  pre {
    line-height: 125%;
  }
  td.linenos .normal {
    color: inherit;
    background-color: transparent;
    padding-left: 5px;
    padding-right: 5px;
  }
  span.linenos {
    color: inherit;
    background-color: transparent;
    padding-left: 5px;
    padding-right: 5px;
  }
  td.linenos .special {
    color: #000000;
    background-color: #ffffc0;
    padding-left: 5px;
    padding-right: 5px;
  }
  span.linenos.special {
    color: #000000;
    background-color: #ffffc0;
    padding-left: 5px;
    padding-right: 5px;
  }
  .highlight .hll {
    background-color: #ffffcc;
  }
  .highlight {
    background: #ffffff;
  }
  .highlight .c {
    color: #999988;
    font-style: italic;
  } /* Comment */
  .highlight .err {
    color: #a61717;
    background-color: #e3d2d2;
  } /* Error */
  .highlight .k {
    font-weight: bold;
  } /* Keyword */
  .highlight .o {
    font-weight: bold;
  } /* Operator */
  .highlight .ch {
    color: #999988;
    font-style: italic;
  } /* Comment.Hashbang */
  .highlight .cm {
    color: #999988;
    font-style: italic;
  } /* Comment.Multiline */
  .highlight .cp {
    color: #999999;
    font-weight: bold;
  } /* Comment.Preproc */
  .highlight .cpf {
    color: #999988;
    font-style: italic;
  } /* Comment.PreprocFile */
  .highlight .c1 {
    color: #999988;
    font-style: italic;
  } /* Comment.Single */
  .highlight .cs {
    color: #999999;
    font-weight: bold;
    font-style: italic;
  } /* Comment.Special */
  .highlight .gd {
    color: #000000;
    background-color: #ffdddd;
  } /* Generic.Deleted */
  .highlight .ge {
    font-style: italic;
  } /* Generic.Emph */
  .highlight .ges {
    font-weight: bold;
    font-style: italic;
  } /* Generic.EmphStrong */
  .highlight .gr {
    color: #aa0000;
  } /* Generic.Error */
  .highlight .gh {
    color: #999999;
  } /* Generic.Heading */
  .highlight .gi {
    color: #000000;
    background-color: #ddffdd;
  } /* Generic.Inserted */
  .highlight .go {
    color: #888888;
  } /* Generic.Output */
  .highlight .gp {
    color: #555555;
  } /* Generic.Prompt */
  .highlight .gs {
    font-weight: bold;
  } /* Generic.Strong */
  .highlight .gu {
    color: #aaaaaa;
  } /* Generic.Subheading */
  .highlight .gt {
    color: #aa0000;
  } /* Generic.Traceback */
  .highlight .kc {
    font-weight: bold;
  } /* Keyword.Constant */
  .highlight .kd {
    font-weight: bold;
  } /* Keyword.Declaration */
  .highlight .kn {
    font-weight: bold;
  } /* Keyword.Namespace */
  .highlight .kp {
    font-weight: bold;
  } /* Keyword.Pseudo */
  .highlight .kr {
    font-weight: bold;
  } /* Keyword.Reserved */
  .highlight .kt {
    color: #445588;
    font-weight: bold;
  } /* Keyword.Type */
  .highlight .m {
    color: #009999;
  } /* Literal.Number */
  .highlight .s {
    color: #bb8844;
  } /* Literal.String */
  .highlight .na {
    color: #008080;
  } /* Name.Attribute */
  .highlight .nb {
    color: #999999;
  } /* Name.Builtin */
  .highlight .nc {
    color: #445588;
    font-weight: bold;
  } /* Name.Class */
  .highlight .no {
    color: #008080;
  } /* Name.Constant */
  .highlight .ni {
    color: #800080;
  } /* Name.Entity */
  .highlight .ne {
    color: #990000;
    font-weight: bold;
  } /* Name.Exception */
  .highlight .nf {
    color: #990000;
    font-weight: bold;
  } /* Name.Function */
  .highlight .nn {
    color: #555555;
  } /* Name.Namespace */
  .highlight .nt {
    color: #000080;
  } /* Name.Tag */
  .highlight .nv {
    color: #008080;
  } /* Name.Variable */
  .highlight .ow {
    font-weight: bold;
  } /* Operator.Word */
  .highlight .w {
    color: #bbbbbb;
  } /* Text.Whitespace */
  .highlight .mb {
    color: #009999;
  } /* Literal.Number.Bin */
  .highlight .mf {
    color: #009999;
  } /* Literal.Number.Float */
  .highlight .mh {
    color: #009999;
  } /* Literal.Number.Hex */
  .highlight .mi {
    color: #009999;
  } /* Literal.Number.Integer */
  .highlight .mo {
    color: #009999;
  } /* Literal.Number.Oct */
  .highlight .sa {
    color: #bb8844;
  } /* Literal.String.Affix */
  .highlight .sb {
    color: #bb8844;
  } /* Literal.String.Backtick */
  .highlight .sc {
    color: #bb8844;
  } /* Literal.String.Char */
  .highlight .dl {
    color: #bb8844;
  } /* Literal.String.Delimiter */
  .highlight .sd {
    color: #bb8844;
  } /* Literal.String.Doc */
  .highlight .s2 {
    color: #bb8844;
  } /* Literal.String.Double */
  .highlight .se {
    color: #bb8844;
  } /* Literal.String.Escape */
  .highlight .sh {
    color: #bb8844;
  } /* Literal.String.Heredoc */
  .highlight .si {
    color: #bb8844;
  } /* Literal.String.Interpol */
  .highlight .sx {
    color: #bb8844;
  } /* Literal.String.Other */
  .highlight .sr {
    color: #808000;
  } /* Literal.String.Regex */
  .highlight .s1 {
    color: #bb8844;
  } /* Literal.String.Single */
  .highlight .ss {
    color: #bb8844;
  } /* Literal.String.Symbol */
  .highlight .bp {
    color: #999999;
  } /* Name.Builtin.Pseudo */
  .highlight .fm {
    color: #990000;
    font-weight: bold;
  } /* Name.Function.Magic */
  .highlight .vc {
    color: #008080;
  } /* Name.Variable.Class */
  .highlight .vg {
    color: #008080;
  } /* Name.Variable.Global */
  .highlight .vi {
    color: #008080;
  } /* Name.Variable.Instance */
  .highlight .vm {
    color: #008080;
  } /* Name.Variable.Magic */
  .highlight .il {
    color: #009999;
  } /* Literal.Number.Integer.Long */
}