reportlab 中的多行(段落)页脚和页眉

Posted

技术标签:

【中文标题】reportlab 中的多行(段落)页脚和页眉【英文标题】:A multiline(paragraph) footer and header in reportlab 【发布时间】:2012-02-08 07:35:50 【问题描述】:

在reportlab中拥有页脚和页眉的最佳方式是什么,而不仅仅是一条线,可以在onPage函数中使用canvas.drawString绘制。没有找到将类似段落的内容放入 onPage 函数的页眉/页脚的方法。处理此问题的最佳方法是什么?有没有办法将段落放入页脚?

【问题讨论】:

【参考方案1】:

我会看看这个答案:

Properly add page numbers and total number of pages to PDF using ReportLab

有一个关于在 ReportLab 中使用 Story 和 SimpleDocTemplate 添加页眉和页脚的惊人示例

【讨论】:

【参考方案2】:

在所有页面上添加页眉或页脚的其他方法:构建方法有参数可以做到这一点。

不要在jochen的答案中使用框架和模板。在最后一行,使用

doc.build(text, onFirstPage=footer, onLaterPages=footer)

其余方法与 jochen 相同。

【讨论】:

【参考方案3】:

我知道这有点旧,但我遇到了这个问题并且能够解决它。当您的 PDF 中有多个页面并希望在每一页上都有页脚/页眉时,您必须使用 NextPageTemplate('template_id')。我只是在编写相关代码,其余代码如上面的@jochen 示例所示。

在我的例子中,我使用的是PageBreak(),我花了一段时间才明白为什么我只在首页获得页脚。

from reportlab.platypus import Paragraph, PageBreak, PageTemplate, Frame, NextPageTemplate

frame = Frame(doc.leftMargin, doc.bottomMargin, doc.width, doc.height, id='normal')
template = PageTemplate(id='footer', onPage=footer, frames=[frame])


# add a NextPageTemplate before a PageBreak to have the footer in the next page

text.append(Paragraph('some text', style)),
text.append(NextPageTemplate('footer')), # this will make the footer to be on the next page if exists
text.append(PageBreak())
doc.build(text)

【讨论】:

【参考方案4】:

Jochen 的回答很棒,但我发现它不完整。它适用于页脚,但不适用于页眉,因为 Reportlab 将在页眉顶部绘制所有可流动的内容。您需要确保您创建的 Frame 的大小不包括标题占用的空间,因此 flowabls 不会打印在标题顶部。

使用jochen的代码,这是一个完整的标题示例:

from reportlab.lib.pagesizes import letter, cm
from reportlab.lib.styles import getSampleStyleSheet
from reportlab.platypus import BaseDocTemplate, Frame, PageTemplate, Paragraph
from functools import partial

styles = getSampleStyleSheet()
styleN = styles['Normal']
styleH = styles['Heading1']

def header(canvas, doc, content):
    canvas.saveState()
    w, h = content.wrap(doc.width, doc.topMargin)
    content.drawOn(canvas, doc.leftMargin, doc.height + doc.topMargin - h)
    canvas.restoreState()

doc = BaseDocTemplate('test.pdf', pagesize=letter)
frame = Frame(doc.leftMargin, doc.bottomMargin, doc.width, doc.height-2*cm, id='normal')
header_content = Paragraph("This is a multi-line header.  It goes on every page.  " * 8, styleN)
template = PageTemplate(id='test', frames=frame, onPage=partial(header, content=header_content))
doc.addPageTemplates([template])

text = []
for i in range(111):
    text.append(Paragraph("This is line %d." % i, styleN))
doc.build(text)

注意框架的减重,它从框架的高度减去 2 厘米,以便为标题留出空间。 flowables 将打印在框架内,因此您可以更改框架的大小以允许不同大小的标题。

我也发现我通常需要将变量传递到header中,所以我使用了一个分配给onPage的偏函数,以便可以传入header的内容。这样就可以根据内容有一个可变的header页面。

【讨论】:

【参考方案5】:

您可以在 onPage 函数中使用任意绘图命令,因此您可以从您的函数中仅绘制一个段落(参见reportlab user guide 中的第 5.3 节)。

这是一个完整的例子:

from reportlab.lib.pagesizes import letter
from reportlab.lib.styles import getSampleStyleSheet
from reportlab.platypus import BaseDocTemplate, Frame, PageTemplate, Paragraph

styles = getSampleStyleSheet()
styleN = styles['Normal']
styleH = styles['Heading1']

def footer(canvas, doc):
    canvas.saveState()
    P = Paragraph("This is a multi-line footer.  It goes on every page.  " * 5,
                  styleN)
    w, h = P.wrap(doc.width, doc.bottomMargin)
    P.drawOn(canvas, doc.leftMargin, h)
    canvas.restoreState()

doc = BaseDocTemplate('test.pdf', pagesize=letter)
frame = Frame(doc.leftMargin, doc.bottomMargin, doc.width, doc.height,
               id='normal')
template = PageTemplate(id='test', frames=frame, onPage=footer)
doc.addPageTemplates([template])

text = []
for i in range(111):
    text.append(Paragraph("This is line %d." % i,
                          styleN))
doc.build(text)

【讨论】:

以上是关于reportlab 中的多行(段落)页脚和页眉的主要内容,如果未能解决你的问题,请参考以下文章

边框页脚和页眉长度调整为 s-s-rS

在单个单元格中显示页眉页脚和单元格?

带有 % extends 的每页上的 Django 页脚和页眉

markdown 页眉页脚和溢出内容

滚动 tableView 部分的页脚和页眉

实现固定页眉和滚动,页脚和移动问题