Skip to content

Pdfs

hypha.apply.utils.pdfs

STYLES module-attribute

STYLES = {'Question': ParagraphStyle(fontName='MontserratBold', fontSize=14, name='Question', spaceAfter=0, spaceBefore=18, leading=21), 'QuestionSmall': ParagraphStyle(fontName='MontserratBold', fontSize=12, name='QuestionSmall', spaceAfter=0, spaceBefore=16, leading=18), 'Normal': ParagraphStyle(fontName='NotoSans', name='Normal'), 'Heading1': ParagraphStyle(fontName='NotoSansBold', fontSize=12, name='Heading1', spaceAfter=4, spaceBefore=12, leading=18), 'Heading2': ParagraphStyle(fontName='NotoSansBold', fontSize=10, name='Heading2', spaceAfter=4, spaceBefore=10, leading=15), 'Heading3': ParagraphStyle(fontName='NotoSansBold', fontSize=10, name='Heading3', spaceAfter=4, spaceBefore=10, leading=15), 'Heading4': ParagraphStyle(fontName='NotoSansBold', fontSize=10, name='Heading4', spaceAfter=4, spaceBefore=10, leading=15), 'Heading5': ParagraphStyle(fontName='NotoSansBold', fontSize=10, name='Heading5', spaceAfter=4, spaceBefore=10, leading=15)}

font_location module-attribute

font_location = join(dirname(abspath(__file__)), 'media', 'fonts')

PREPARED_FONTS module-attribute

PREPARED_FONTS = False

DARK_GREY module-attribute

DARK_GREY = Color(0.0154, 0.0154, 0, 0.7451)

FRAME_PADDING module-attribute

FRAME_PADDING = 6

ReportDocTemplate

Bases: BaseDocTemplate

build

build(flowables, onFirstPage=do_nothing, onLaterPages=do_nothing)
Source code in hypha/apply/utils/pdfs.py
def build(self, flowables, onFirstPage=do_nothing, onLaterPages=do_nothing):
    frame = Frame(
        self.leftMargin, self.bottomMargin, self.width, self.height, id="normal"
    )
    self.addPageTemplates(
        [
            PageTemplate(
                id="Header",
                autoNextPageTemplate="Main",
                frames=frame,
                onPage=onFirstPage,
                pagesize=self.pagesize,
            ),
            PageTemplate(
                id="Main", frames=frame, onPage=onLaterPages, pagesize=self.pagesize
            ),
        ]
    )
    super().build(flowables)

font

font(font_name)
Source code in hypha/apply/utils/pdfs.py
def font(font_name):
    return os.path.join(font_location, font_name)

prepare_fonts

prepare_fonts()
Source code in hypha/apply/utils/pdfs.py
def prepare_fonts():
    global PREPARED_FONTS
    if PREPARED_FONTS:
        return
    pdfmetrics.registerFont(TTFont("Montserrat", font("Montserrat-Regular.ttf")))
    pdfmetrics.registerFont(TTFont("MontserratBold", font("Montserrat-Bold.ttf")))
    pdfmetrics.registerFont(TTFont("MontserratItalic", font("Montserrat-Italic.ttf")))
    pdfmetrics.registerFont(
        TTFont("MontserratBoldItalic", font("Montserrat-BoldItalic.ttf"))
    )
    pdfmetrics.registerFontFamily(
        "Montserrat",
        normal="Montserrat",
        bold="MontserratBold",
        italic="MontserratItalic",
        boldItalic="MontserratBoldItalic",
    )

    pdfmetrics.registerFont(TTFont("NotoSans", font("NotoSans-Regular.ttf")))
    pdfmetrics.registerFont(TTFont("NotoSansBold", font("NotoSans-Bold.ttf")))
    pdfmetrics.registerFont(TTFont("NotoSansItalic", font("NotoSans-Italic.ttf")))
    pdfmetrics.registerFont(
        TTFont("NotoSansBoldItalic", font("NotoSans-BoldItalic.ttf"))
    )
    pdfmetrics.registerFontFamily(
        "NotoSans",
        normal="NotoSans",
        bold="NotoSansBold",
        italic="NotoSansItalic",
        boldItalic="NotoSansBoldItalic",
    )
    PREPARED_FONTS = True

do_nothing

do_nothing(doc, canvas)
Source code in hypha/apply/utils/pdfs.py
def do_nothing(doc, canvas):
    pass

make_pdf

make_pdf(title, sections, pagesize)
Source code in hypha/apply/utils/pdfs.py
def make_pdf(title, sections, pagesize):
    prepare_fonts()
    buffer = BytesIO()
    page_width, page_height = getattr(pagesizes, pagesize)

    doc = ReportDocTemplate(
        buffer,
        pagesize=getattr(pagesizes, pagesize),
        title=title,
    )

    story = []
    for section in sections:
        story.extend(section["content"])
        story.append(NextPageTemplate("Header"))
        story.append(PageBreak())

    current_section = None
    sections = cycle(sections)

    def header_page(canvas, doc):
        nonlocal current_section
        current_section = next(sections)
        canvas.saveState()
        title_spacer = draw_title_block(
            canvas,
            doc,
            current_section["title"],
            title,
            current_section["meta"],
            page_width,
            page_height,
        )
        canvas.restoreState()
        story.insert(0, title_spacer)

    def main_page(canvas, doc):
        nonlocal current_section
        canvas.saveState()
        spacer = draw_header(
            canvas,
            doc,
            current_section["title"],
            title,
            page_width,
            page_height,
        )
        story.insert(0, spacer)
        canvas.restoreState()

    doc.build(story, onFirstPage=header_page, onLaterPages=main_page)

    buffer.seek(0)
    return buffer

split_text

split_text(canvas, text, width)
Source code in hypha/apply/utils/pdfs.py
def split_text(canvas, text, width):
    return simpleSplit(text, canvas._fontname, canvas._fontsize, width)

draw_header

draw_header(canvas, doc, page_title, title, page_width, page_height)
Source code in hypha/apply/utils/pdfs.py
def draw_header(canvas, doc, page_title, title, page_width, page_height):
    title_size = 10

    # Set canvas font to correctly calculate the splitting
    canvas.setFont("MontserratBold", title_size)

    text_width = page_width - doc.leftMargin - doc.rightMargin - 2 * FRAME_PADDING
    split_title = split_text(canvas, title, text_width)

    # only count title - assume 1 line of title in header
    total_height = (
        doc.topMargin
        + 1.5 * (len(split_title) - 1) * title_size
        + title_size / 2  # bottom padding
    )

    canvas.setFillColor(DARK_GREY)
    canvas.rect(
        0,
        page_height - total_height,
        page_width,
        total_height,
        stroke=False,
        fill=True,
    )

    pos = (
        (page_height - doc.topMargin)
        + title_size / 2  # bottom of top margin
        + 1.5 * 1 * title_size  # spacing below page title  # text
    )

    canvas.setFillColor(white)

    canvas.drawString(
        doc.leftMargin + FRAME_PADDING,
        pos,
        page_title,
    )

    pos -= title_size / 2

    for line in split_title:
        pos -= title_size
        canvas.drawString(
            doc.leftMargin + FRAME_PADDING,
            pos,
            line,
        )
        pos -= title_size / 2

    return Spacer(1, total_height - doc.topMargin)

draw_title_block

draw_title_block(canvas, doc, page_title, title, meta, page_width, page_height)
Source code in hypha/apply/utils/pdfs.py
def draw_title_block(canvas, doc, page_title, title, meta, page_width, page_height):
    page_title_size = 20
    title_size = 30
    meta_size = 10

    text_width = page_width - doc.leftMargin - doc.rightMargin - 2 * FRAME_PADDING

    # Set canvas font to correctly calculate the splitting
    canvas.setFont("MontserratBold", title_size)
    canvas.setFillColor(white)
    split_title = split_text(canvas, title, text_width)

    canvas.setFont("MontserratBold", meta_size)
    canvas.setFillColor(white)
    meta_text = "  |  ".join(str(text) for text in meta)
    split_meta = split_text(canvas, meta_text, text_width)

    total_height = (
        doc.topMargin
        + page_title_size
        + page_title_size * 3 / 4
        + len(split_title) * (title_size + title_size / 2)  # page title + spaceing
        + (1.5 * len(split_meta) + 3)  # title + spacing
        * meta_size  # 1.5 per text line + 3 for spacing
    )

    canvas.setFillColor(DARK_GREY)
    canvas.rect(
        0,
        page_height - total_height,
        page_width,
        total_height,
        stroke=False,
        fill=True,
    )

    canvas.setFont("MontserratBold", page_title_size)
    canvas.setFillColor(white)
    pos = page_height - doc.topMargin
    pos -= page_title_size
    canvas.drawString(
        doc.leftMargin + FRAME_PADDING,
        pos,
        page_title,
    )

    pos -= page_title_size * 3 / 4

    canvas.setFont("MontserratBold", title_size)
    canvas.setFillColor(white)
    for line in split_title:
        pos -= title_size
        canvas.drawString(
            doc.leftMargin + FRAME_PADDING,
            pos,
            line,
        )
        pos -= title_size / 2

    canvas.setFont("MontserratBold", meta_size)
    canvas.setFillColor(white)

    pos -= meta_size * 2

    for line in split_meta:
        canvas.drawString(
            doc.leftMargin + FRAME_PADDING,
            pos,
            line,
        )
        pos -= meta_size / 2

    return Spacer(1, total_height - doc.topMargin)

handle_block

handle_block(block, custom_style=None)
Source code in hypha/apply/utils/pdfs.py
def handle_block(block, custom_style=None):
    paragraphs = []
    if not custom_style:
        custom_style = {}

    styles = {**STYLES}
    for style, overwrite in custom_style.items():
        styles[style] = STYLES[overwrite]

    for tag in block:
        if isinstance(tag, NavigableString):
            text = tag.strip()
            if text:
                paragraphs.append(Paragraph(text, styles["Normal"]))
        elif tag.name in {"ul", "ol"}:
            style = styles["Normal"]
            if tag.name == "ul":
                bullet = "bullet"
            elif tag.name == "ol":
                bullet = "1"

            paragraphs.append(
                ListFlowable(
                    [
                        ListItem(Paragraph(bullet_item.get_text(), style))
                        for bullet_item in tag.find_all("li")
                    ],
                    bulletType=bullet,
                )
            )
        elif tag.name in {"table"}:
            paragraphs.append(
                Table(
                    [
                        [
                            Paragraph(cell.get_text(), styles["Normal"])
                            for cell in row.find_all({"td", "th"})
                        ]
                        for row in tag.find_all("tr")
                    ],
                    colWidths="*",
                    style=TableStyle(
                        [
                            ("VALIGN", (0, 0), (-1, -1), "TOP"),
                            ("LINEABOVE", (0, 0), (-1, -1), 1, DARK_GREY),
                        ]
                    ),
                )
            )
        else:
            style = None
            if tag.name in {"p"}:
                style = styles["Normal"]
            elif tag.name == "h2":
                style = styles["Heading2"]
            elif tag.name == "h3":
                style = styles["Heading3"]
            elif tag.name == "h4":
                style = styles["Heading4"]
            elif tag.name == "h5":
                style = styles["Heading5"]

            if style:
                text = tag.get_text()
                if text:
                    paragraphs.append(Paragraph(text, style))
            else:
                paragraphs.extend(handle_block(tag))
    return paragraphs

draw_submission_content

draw_submission_content(content)
Source code in hypha/apply/utils/pdfs.py
def draw_submission_content(content):
    prepare_fonts()
    paragraphs = []

    for section in BeautifulSoup(content, "html5lib").find_all("section"):
        question_text = section.select_one(".question").get_text()
        question = Paragraph(question_text, STYLES["Question"])

        # Keep the question and the first block of the answer together
        # this keeps 1 line answers tidy and ensures that bigger responses break
        # sooner instead of waiting to fill an entire page. There may still be issues
        first_answer, *rest = handle_block(section.select_one(".answer"))
        paragraphs.extend(
            [
                KeepTogether(
                    [
                        question,
                        first_answer,
                    ]
                ),
                *rest,
            ]
        )
    return paragraphs

html_to_pdf

html_to_pdf(html_body)

Convert HTML to PDF.

Parameters:

  • html_body (str) –

    The body of the html as string

Returns:

  • BytesIO ( BytesIO ) –

    PDF file

Source code in hypha/apply/utils/pdfs.py
def html_to_pdf(html_body: str) -> BytesIO:
    """Convert HTML to PDF.

    Args:
        html_body: The body of the html as string

    Returns:
        BytesIO: PDF file
    """
    packet = BytesIO()
    pisa.CreatePDF(html_body, dest=packet, raise_exception=True, encoding="utf-8")
    packet.seek(0)
    return packet

render_as_pdf

render_as_pdf(template_name, filename, context, request=None)

Convert HTML template to PDF file and return as a downloadable file.

Parameters:

  • template_name (str) –

    Django template name to render

  • filename (str) –

    Name of the output PDF file

  • context (dict) –

    Context dictionary for rendering template

  • request –

    Request object, defaults to None

Returns:

  • HttpResponse ( HttpResponse ) –

    PDF file as downloadable response

Example

response = render_as_pdf( template_name='my_template.html', filename='my_pdf.pdf', context={'title': 'My PDF'}, request=request )

Source code in hypha/apply/utils/pdfs.py
def render_as_pdf(
    template_name: str, filename: str, context: dict, request=None
) -> HttpResponse:
    """Convert HTML template to PDF file and return as a downloadable file.

    Args:
        template_name: Django template name to render
        filename: Name of the output PDF file
        context: Context dictionary for rendering template
        request: Request object, defaults to None

    Returns:
        HttpResponse: PDF file as downloadable response

    Example:
        response = render_as_pdf(
            template_name='my_template.html',
            filename='my_pdf.pdf',
            context={'title': 'My PDF'},
            request=request
        )
    """
    if "pagesize" not in context:
        pdf_page_settings = PDFPageSettings.load(request_or_site=request)
        context["pagesize"] = pdf_page_settings.download_page_size

    context.setdefault("export_date", timezone.now())
    context.setdefault("export_user", request.user if request else None)

    html = render_to_string(
        template_name=template_name, context=context, request=request
    )
    pdf = html_to_pdf(html)

    response = HttpResponse(content_type="application/pdf")
    response["Content-Disposition"] = f"attachment; filename={filename}"
    response.write(pdf.read())
    return response

merge_pdf

merge_pdf(origin_pdf, input_pdf)

Given two PDFs, merge them together.

Parameters:

  • origin_pdf (BytesIO) –

    a file-like object containing a PDF

  • input_pdf (BytesIO) –

    a file-like object containing a PDF

Returns:

  • File –

    Return a File object containing the merged PDF and with the same name as the

  • File –

    original PDF.

Source code in hypha/apply/utils/pdfs.py
def merge_pdf(origin_pdf: BytesIO, input_pdf: BytesIO) -> File:
    """Given two PDFs, merge them together.

    Args:
        origin_pdf: a file-like object containing a PDF
        input_pdf: a file-like object containing a PDF

    Returns:
        Return a File object containing the merged PDF and with the same name as the
        original PDF.
    """
    merger = PdfWriter(clone_from=BytesIO(origin_pdf.read()))
    merger.append(PdfReader(input_pdf))

    output_pdf = BytesIO()
    merger.write(output_pdf)
    output_pdf.seek(0)
    return File(output_pdf, name=origin_pdf.name)