我必须创建PDF文件,其中需要在页面左下方添加像页脚一样的行。
以下代码正在运行:
import StringIO
from reportlab.pdfgen import canvas
import uuid
def test(pdf_file_name="abc.pdf", pdf_size=(432, 648), font_details=("Times-Roman", 9)):
# create a new PDF with Reportla
text_to_add = "I am writing here.."
new_pdf = "test_%s.pdf"%(str(uuid.uuid4()))
packet = StringIO.StringIO()
packet.seek(0)
c = canvas.Canvas(pdf_file_name, pagesize = pdf_size)
#- Get the length of text in a PDF.
text_len = c.stringWidth(text_to_add, font_details[0], font_details[1])
#- take margin 20 and 20 in both axis
#- Adjust starting point on x axis according to text_len
x = pdf_size[0]-20 - text_len
y = 20
#- set font.
c.setFont(font_details[0], font_details[1])
#- write text,
c.drawString(x, y, text_to_add)
c.showPage()
c.save()
return pdf_file_name
现在,如果文本有多行,那么这不起作用,因为文本的长度大于页面大小的宽度。理解。
我尝试使用Frame
和paragraph
,但仍然无法在PDF中的正确位置写入文字
以下是代码:
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. " * 10, styleN)
w, h = P.wrap(doc.width, doc.bottomMargin)
print "w, h:", w, h
print "doc.leftMargin:", doc.leftMargin
P.drawOn(canvas, 10, 30)
canvas.restoreState()
def test():
doc = BaseDocTemplate('test.pdf', pagesize=(432, 648))
print "doc.leftMargin:", doc.leftMargin
print "doc.bottomMargin:", doc.bottomMargin
print "doc.width:", doc.width
print "doc.height:", doc.height
frame = Frame(10, 50, 432, 648, id='normal')
template = PageTemplate(id='test', frames=frame, onPage=footer)
doc.addPageTemplates([template])
text = []
for i in range(1):
text.append(Paragraph("", styleN))
doc.build(text)
不明白为什么页面大小会发生变化,因为我设置(432, 648)
但显示(288.0, 504.0)
doc.leftMargin: 72.0
doc.bottomMargin: 72.0
doc.width: 288.0
doc.height: 504.0
帧大小:
w, h: 288.0 96
doc.leftMargin: 72.0
不知道如何解决这个问题。 我引用this链接
答案 0 :(得分:6)
首先是关于doc.width
的谜,doc.width
不是文档的实际宽度。它是边距之间区域的宽度,因此在这种情况下doc.width + doc.leftMargin + doc.rightMargin
等于实际页面的宽度。
现在回到为什么页脚没有按照你想要的方式跨越页面的整个宽度。这是因为上述问题相同,即doc.width
不是实际的纸张宽度。
假设您希望页脚跨越整个页面
def footer(canvas, doc):
canvas.saveState()
P = Paragraph("This is a multi-line footer. It goes on every page. " * 10, styleN)
# Notice the letter[0] which is the width of the letter page size
w, h = P.wrap(letter[0] - 20, doc.bottomMargin)
P.drawOn(canvas, 10, 10)
canvas.restoreState()
假设您希望页脚跨越可写区域的宽度
注意:默认设置的边距非常大,这就是为什么边上有这么多空白空间的原因。
def footer(canvas, doc):
canvas.saveState()
P = Paragraph("This is a multi-line footer. It goes on every page. " * 10, styleN)
w, h = P.wrap(doc.width, doc.bottomMargin)
print "w, h:", w, h
print "doc.leftMargin:", doc.leftMargin
P.drawOn(canvas, doc.leftMargin, 10)
canvas.restoreState()
修改强>
因为知道正常文本应该从哪里开始可能是有用的。我们需要弄清楚页脚的高度。在正常情况下,我们无法使用P.height
,因为它取决于文本的宽度,调用它会引发AttributeError
。
在我们的案例中,我们实际上可以直接从P.wrap
(h
)或在我们调用P.height
后调用P.wrap
来获取页脚的高度。
通过在页脚高度处开始我们的Frame
,我们永远不会有重叠的文本。但重要的是要记住将Frame
的高度设置为doc.height - footer.height
,以确保文本不会被放置在页面之外。