Pure Python Markdown renderer that outputs PNG images.
- No browser dependency
- No WeasyPrint dependency
- Cross-platform
pip install - Configurable themes
- Markdown + syntax highlight + basic math support
- It parses Markdown with
markdown-it-py+dollarmath, maps tokens into a small document model, then lays out headings, paragraphs, tables, quotes, images, and code blocks directly on a Pillow canvas. No browser and no HTML screenshot step. - Font selection is based on glyph coverage instead of one fixed font. It auto-discovers system/custom fonts, scores coverage with
matplotlib.ft2font, preselects body/heading/code fonts, then falls back per text run for CJK and missing glyphs. - Emoji are detected as Unicode sequences and rendered through cached Twemoji PNG assets, so skin-tone modifiers and common ZWJ sequences do not depend on local color-font availability.
- There are now two separate font routes: the default package stays on local system fonts, while the optional
NotoSansextra syncs a curated Noto pack into a user cache directory and prefers that pack at render time. - Math is rendered with
matplotlib.mathtextinto transparent bitmaps, with a light LaTeX sanitizing pass and a readable plain-text fallback for unsupported syntax. - Syntax highlight comes from
Pygmentstoken streams, while wrapping and painting are handled by the renderer itself, so code blocks do not depend on browser CSS/JS.
pip install md2png-lite
# local development
uv syncSystem-font route:
pip install md2png-liteCurated Noto route:
pip install 'md2png-lite[NotoSans]'The NotoSans extra does not embed fonts into the wheel. It installs the sync helpers and downloads the curated Noto pack into the local cache on first use.
uv run md2png-lite input.md -o output.png --theme paperUse the mobile summary preset with theme-driven width / padding / scale defaults:
uv run md2png-lite input.md -o output.png --theme mobile-summaryIf you want serialized image content on stdout instead of a file:
uv run md2png-lite input.md --stdout-format base64
uv run md2png-lite input.md --stdout-format jsonbase64: prints only the PNG base64 payloadjson: prints the full payload withok,renderer,mime_type, andbase64
Choose the font route explicitly at call time:
uv run md2png-lite input.md -o output.png --font-pack system
uv run md2png-lite input.md -o output.png --font-pack notoLoad custom fonts explicitly:
uv run md2png-lite input.md -o output.png \
--font-path ./fonts/NotoSansCJKsc-Regular.otf \
--font-dir ./fontspython3 scripts/benchmark_examples.py --repeat 3 --keepThis renders the bundled stress samples and prints timing / output size.
Render the smaller inspection-focused samples:
python3 scripts/render_examples.pyUse a custom glob when needed:
python3 scripts/render_examples.py --pattern 'sample_*.md'from md2png_lite import render_markdown_image
payload = render_markdown_image("# Hello\n\n```python\nprint('hi')\n```")Choose the route per call:
payload = render_markdown_image(markdown, font_pack="system")
payload = render_markdown_image(markdown, font_pack="noto")
payload = render_markdown_image(markdown, theme="mobile-summary")Returned payload shape:
{
"ok": True,
"renderer": "md2png-lite",
"mime_type": "image/png",
"base64": "...",
}- Headings
- Paragraphs
- Bullet / ordered lists
- Block quotes
- Horizontal rules
- Fenced code blocks
- Custom
summaryfenced blocks rendered as editorial summary cards - Inline code
- Tables
- Links / emphasis / strong / strikethrough
- Inline and block math via
matplotlib.mathtext - Local /
data:/ remote images
paper: current warm reading style; legacy alias阅读器remains availablegithub-light: GitHub-like daytime themegithub-dark: GitHub-like dark themesolarized-light: classic Solarized light themegraphite: existing dark editorial thememobile-summary: phone-friendly editorial summary theme with large type and neutral#f2f2f2surfaces
- Auto-discovers system fonts on macOS / Windows / Linux
- Supports two separate routes:
systemand syncednoto - Chooses fonts per text run instead of forcing one global font
- Prioritizes CJK-capable fonts for Chinese / Japanese / Korean text
font_pack="system"stays on platform fonts onlyfont_pack="noto"syncs the curated Noto pack into the local cache and prefers it- Supports custom fonts through CLI:
--font-path,--font-dir - Supports route selection through CLI:
--font-pack auto|system|noto - Supports provider config:
md2png_lite.font_paths,md2png_lite.font_dirs,md2png_lite.font_pack - Supports environment variables:
MD2PNG_LITE_FONT_PATHS,MD2PNG_LITE_FONT_DIRS,MD2PNG_LITE_FONT_PACK - Supports emoji cache control through environment variables:
MD2PNG_LITE_EMOJI_CACHE_DIR,MD2PNG_LITE_EMOJI_SOURCE
Synced Noto font license note:
licenses/NOTO_CJK_LICENSE.txt
- Math support follows
matplotlib.mathtext, not full LaTeX - HTML blocks are rendered as plain text
- Deeply nested lists / tables are supported, but layout remains image-oriented rather than browser-perfect