我尝试使用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))