ReportLab SimpleDoctemplate-设置表格的固定高度,行数可变



我尝试使用Python中的CC_1中生成PDF发票。

发票仅是一个页面,并且永远不会比单个页面上的空间更多的详细信息。我的代码在生成PDF之前先检查最大详细信息。

现在,我正在使用SimpleDocTemplate将所有内容添加到页面上,并构建我使用Table的详细信息。这是一个减少的代码样本:

from reportlab.lib.units import mm
from reportlab.platypus import Paragraph, Spacer, Table, TableStyle
from reportlab.platypus import SimpleDocTemplate
# add invoice header
flowable_list = [
    Spacer(1, 5*mm),
    Paragraph('Date: ...', pg_style_1),
    Spacer(1, 5*mm),
]
# add invoice details
detail_list = [
    ('Item 1', 8, 45),
    ('Item 2', 1, 14),
]
row_list = [
    [
        Paragraph(desc, pg_style_1),
        quantity,
        amount,
    ]
    for desc, quantity, amount in detail_list]
story.append(
    Table(
        data=row_list,
        colWidths=[100*mm, 40*mm, 40*mm],
        style=TableStyle([
            ('VALIGN', (0, 0), (-1, -1), 'TOP'),
            ... some other options ...
        ])))
# add invoice footer; this should be at a specific position on the page
flowable_list.append(Spacer(1, 5*mm))
flowable_list.append(Paragraph('Total: 0', pg_style_1))
# build PDF
buffer = io.BytesIO()
doc = SimpleDocTemplate(buffer)
doc.build(flowable_list)

我的问题:每次都必须在特定位置(例如x*mm),但可能会有可变数量的详细信息,这会导致详细信息表具有非固定高度。

我当前的解决方案:表之后添加Spacer;必须根据表中的行数来计算此间隔者的高度(更多的行意味着垫片会较小;较小的行产生较大的垫片)。但是,如果其中一排环绕并占用比一行更多的空间,这将失败。

我的问题:有没有办法为详细信息表设置固定高度,无论有多少行,但仍然继续使用SimpleDocTemplate

这个类似的问题显示了一种解决方案,可以手动将所有内容绘制到画布上,但是如果可能的话,我希望一种继续使用SimpleDocTemplate的方法。

尝试使用" ToppAdder"的可行示例(结果是将总数推到框架的底部,我假设您可以在其下方添加一个垫子,以控制底部的高度):

########################################################################
from reportlab.lib.units import mm
from reportlab.platypus import Paragraph, Spacer, Table, TableStyle
from reportlab.lib.styles import getSampleStyleSheet
from reportlab.platypus import SimpleDocTemplate
from reportlab.platypus.flowables import TopPadder
import io
def create_pdf():
    styles=getSampleStyleSheet()
    # add invoice header
    flowable_list = [
        Spacer(1, 5 * mm),
        Paragraph(f'Date: ...', styles['Normal']),
        Spacer(1, 5 * mm),
    ]
    # add invoice details
    detail_list = [
        ('Item 1', 8, 45),
        ('Item 2', 1, 14),
    ]
    row_list = [
        [
            Paragraph(f'desc', styles['Normal']),
            # quantity,
            # amount,
        ]
        for desc, quantity, amount in detail_list]
    story = []
    story.append(
        Table(
            data=row_list,
            colWidths=[100 * mm, 40 * mm, 40 * mm],
            style=TableStyle([
                ('VALIGN', (0, 0), (-1, -1), 'TOP'),
                # ... some other options...
            ])))
    # add invoice footer; this should be at a specific position on the page
    flowable_list.append(Spacer(1, 5 * mm))
    flowable_list.append(TopPadder(Paragraph(f'Total: 0', styles['Normal'])))
    # build PDF
    buffer = io.BytesIO()
    doc = SimpleDocTemplate("test2.pdf")
    doc.build(flowable_list)
    # ----------------------------------------------------------------------
if __name__ == "__main__":
    create_pdf()  # Printing the pdf

我还没有找到更好的方法,因此我将添加当前的解决方案;也许这将帮助一些未来的读者遇到类似问题。

...
story.append(
    Table(
        data=row_list,
        colWidths=[100*mm, 40*mm, 40*mm],
        style=TableStyle([
            ('VALIGN', (0, 0), (-1, -1), 'TOP'),
            ... some other options ...
        ])))
# calculate real height of details table, so we can add a 'Spacer' for the missing
# part to have the invoice totals at the correct height at the bottom of the page
_, real_height = story[-1].wrap(doc_width, doc_height)
height_reserved_for_details = 100*mm
remaining_height = max(0, height_reserved_for_details - real_height)
story.append(Spacer(1, remaining_height))
flowable_list.append(Paragraph('Total: 0', pg_style_1))

最新更新