Markdownosaur uses Apple's excellent and relatively new Swift Markdown library to analyze a Markdown source, and then takes that analyzed result and turns it into an NSAttributedString
for displaying on screen on iOS, iPadOS, or Mac. It's just a few hundred lines long, and uses Apple's library's visitor pattern, so it's hopefully pretty easy to understand. :)
iOS 15 already added a new initializer to NSAttributedString
that takes Markdown, you should probably check if that does what you want first! Beyond that, Markdownosaur offers:
- Increased flexibility in styling the resulting
NSAttributedString
- Support for more Markdown elements (like tables, though this requires work from the user to integrate as it's beyond the scope of
NSAttributedString
) - Support for iOS versions earlier than 15
Using Swift Package Manager, add Apple's Swift Markdown library to your project. Then simply take the Markdownosaur.swift
file and add it to your project.
(Note that this library exists more to be helpful and instructive and was quickly generalized from how I use it, you'll likely want to modify it yourself for your own specific styling for instance!)
Here's the resulting NSAttributedString
set on a UILabel
(with the source Markdown visible).
Simply pass in the source Markdown and then use the attributed string for whatever you'd like.
let source = "Here is some **very** cool and simple Markdown"
let document = Document.parsingRedditMarkdown(source: source)
var markdownosaur = Markdownosaur()
let attributedString = markdownosaur.attributedString(from: document)
// For instance…
label.attributedText = attributedString
It's fast! Tiny bit faster than the NSAttributedString
implementation in iOS 15 (Apple's implementation is probably more powerful though). For reference on an iPhone 6S Plus (oldest device that still gets iOS 15), I took test.md
(which is included, and is just the 'Example Result' source above repeated 8x over) and it took about 0.04 seconds on average.
Thanks to Apple for the awesome Swift Markdown library, and thanks to the Down library for existing which helped me figure out how to tackle some aspects.