Converting an Attributed String to a PDF

Table of contents

    Converting an Attributed String to a PDF

    Attributed strings are a powerful way to stylize textual content, as they contain the string alongside the visual representation of said text. As such, they are great candidates for converting into graphical outputs, like an image or a PDF.In this article, I’ll cover how an attributed string can be converted to a PDF on iOS. There are multiple ways a PDF can be generated from NSAttributedString(opens in a new tab) via system APIs, with various ways to handle the conversion. I’ll provide an overview of two of these methods, but ultimately, you should choose the option that best fits your needs.

    Let’s assume we create an NSAttributedString(opens in a new tab) to work with and that we want to convert it to a PDF, like what’s shown in this Gist(opens in a new tab).

    Using a Graphics Context

    One solution for converting an attributed string to a PDF would be to use a PDF graphics context via UIGraphicsPDFRenderer(opens in a new tab). This method is mostly suitable for short strings that should only be converted into a single-page PDF. However, this solution is advanced, since it is possible to literally draw anything on the PDF in the graphics context and it also requires more specific handling. It isn’t straightforward to create multiple pages from a long string, so you will need to manually handle splitting up the string into multiple parts for each page.

    This is how an attributed string can be converted into a single-page PDF that has the size of a DIN A4 page:

    let documentURL = URL(fileURLWithPath: NSTemporaryDirectory()).appendingPathComponent("Converted.pdf")
    let dinA4PaperRect = CGRect(origin: .zero, size: CGSize(width: 595, height: 842))
    let graphicsRenderer = UIGraphicsPDFRenderer(bounds: dinA4PaperRect)
    try? graphicsRenderer.writePDF(to: documentURL) { context in
    context.beginPage()
    let pageMargin: CGFloat = 60
    let drawRect = dinA4PaperRect.inset(by: UIEdgeInsets(top: pageMargin, left: pageMargin, bottom: pageMargin, right: pageMargin))
    attributedString.draw(in: drawRect)
    }
    // Use the generated PDF stored at `documentURL`.
    NSURL *documentURL = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:@"Converted.pdf"]];
    CGRect dinA4PaperRect = (CGRect){.size = CGSizeMake(595.f, 842.f)};
    UIGraphicsPDFRenderer *graphicsRenderer = [[UIGraphicsPDFRenderer alloc] initWithBounds:dinA4PaperRect];
    [graphicsRenderer writePDFToURL:documentURL withActions:^(UIGraphicsPDFRendererContext *context) {
    [context beginPage];
    CGFloat pageMargin = 60.f;
    CGRect drawRect = UIEdgeInsetsInsetRect(dinA4PaperRect, UIEdgeInsetsMake(pageMargin, pageMargin, pageMargin, pageMargin));
    [attributedString drawInRect:drawRect];
    } error:nil];
    // Use the generated PDF stored at `documentURL`.

    This results in a single-page PDF being created, as shown below.

    Graphics Context created PDF document

    Any text that flows off the first page is, by default, cut off and not rendered. In such cases, it is necessary to manually handle content fitting on the page and take care of adding new pages as needed.

    Using a Print Formatter

    The easiest way to handle conversion is using a print formatter. If you want something flexible and customizable that also provides high-quality output and support for automatic page wrapping of strings, this is the way to go. UIPrintPageRenderer(opens in a new tab) can be used to create a PDF from a print formatter like UISimpleTextPrintFormatter(opens in a new tab).

    UISimpleTextPrintFormatter(opens in a new tab) takes an attributed string and handles drawing it into one page or wrapping it into multiple pages automatically. You also have the ability to customize various properties — like content insets — on the print formatter.

    To utilize this, we can create a subclass of UIPrintPageRenderer(opens in a new tab) that provides paperRect and printableRect — along with a custom method to generate the PDF — from the print formatter to a URL on disk:

    class PDFDINA4PrintRenderer: UIPrintPageRenderer {
    let pageSize = CGSize(width: 595, height: 842)
    override var paperRect: CGRect {
    return CGRect(origin: .zero, size: pageSize)
    }
    override var printableRect: CGRect {
    let pageMargin: CGFloat = 60
    let margins = UIEdgeInsets(top: pageMargin, left: pageMargin, bottom: pageMargin, right: pageMargin)
    return paperRect.inset(by: margins)
    }
    func renderPDF(to url: URL) throws {
    prepare(forDrawingPages: NSMakeRange(0, numberOfPages))
    let graphicsRenderer = UIGraphicsPDFRenderer(bounds: paperRect)
    try graphicsRenderer.writePDF(to: url) { context in
    for pageIndex in 0..<numberOfPages {
    context.beginPage()
    drawPage(at: pageIndex, in: context.pdfContextBounds)
    }
    }
    }
    }
    @interface PSPDFDINA4PrintRenderer: UIPrintPageRenderer
    - (BOOL)renderPDFToURL:(NSURL *)url error:(NSError *__autoreleasing *)error;
    @end
    @implementation PSPDFDINA4PrintRenderer
    - (CGSize)pageSize {
    return CGSizeMake(595.f, 842.f);
    }
    - (CGRect)paperRect {
    return (CGRect){.size = self.pageSize};
    }
    - (CGRect)printableRect {
    CGFloat pageMargin = 60.f;
    UIEdgeInsets margins = UIEdgeInsetsMake(pageMargin, pageMargin, pageMargin, pageMargin);
    return UIEdgeInsetsInsetRect(self.paperRect, margins);
    }
    - (BOOL)renderPDFToURL:(NSURL *)url error:(NSError *__autoreleasing *)error {
    [self prepareForDrawingPages:NSMakeRange(0, self.numberOfPages)];
    UIGraphicsPDFRenderer *graphicsRenderer = [[UIGraphicsPDFRenderer alloc] initWithBounds:self.paperRect];
    return [graphicsRenderer writePDFToURL:url withActions:^(UIGraphicsPDFRendererContext *context) {
    for (NSInteger pageIndex = 0; pageIndex < self.numberOfPages; pageIndex++) {
    [context beginPage];
    [self drawPageAtIndex:pageIndex inRect:context.pdfContextBounds];
    }
    } error:error];
    }
    @end

    Now we can use our custom print renderer and provide it with a print formatter. The print formatter handles splitting up the string into multiple pages in case it doesn’t fit on one page:

    let pdfRenderer = PDFDINA4PrintRenderer()
    let printFormatter = UISimpleTextPrintFormatter(attributedText: attributedString)
    pdfRenderer.addPrintFormatter(printFormatter, startingAtPageAt: 0)
    let documentURL = URL(fileURLWithPath: NSTemporaryDirectory()).appendingPathComponent("Converted from Attributed String.pdf")
    try pdfRenderer.renderPDF(to: documentURL)
    // Use the generated PDF stored at `documentURL`.
    PSPDFDINA4PrintRenderer *pdfRenderer = [PSPDFDINA4PrintRenderer new];
    UISimpleTextPrintFormatter *printFormatter = [[UISimpleTextPrintFormatter alloc] initWithAttributedText:attributedString];
    [pdfRenderer addPrintFormatter:printFormatter startingAtPageAtIndex:0];
    NSURL *documentURL = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:@"Converted from Attributed String.pdf"]];
    NSError *error;
    if ([pdfRenderer renderPDFToURL:documentURL error:&error]) {
    // Use the generated PDF stored at `documentURL`.
    }

    Converting a long, attributed string with a few different attributes will generate a PDF with multiple pages that looks like the one below.

    Print Renderer Created PDF Document

    You can even add multiple print formatters to a single print renderer to append differently formatted strings. For example, you can render an attributed string on the first few pages and append pages containing a converted HTML string afterward.

    Work with the Generated PDF

    Now that we’ve explored different ways of how an attributed string can be converted into a PDF, you might want to continue working with the document. Depending on your use case, you might want to display, modify, or mark up the PDF. All of this can be done with PSPDFKit. Head over to our guides to get an overview of how PDFs can be utilized in your app.

    Stefan Kieleithner

    Stefan Kieleithner

    iOS Senior Software Engineer

    Stefan began his journey into iOS development in 2013 and has been passionate about it ever since. In his free time, he enjoys playing board and video games, spending time with his cats, and gardening on his balcony.

    Explore related topics

    FREE TRIAL Ready to get started?