我正在编写一份包含表格和图像混合的报告。图像 [ 实际上是图形 ] 以 .png 格式保存到文件系统中。
实际呈现 PDF 的方法是:
def _render_report(report_data):
file_name = get_file_name() # generate random filename for report
rpt = Report(settings.MEDIA_ROOT + os.sep + file_name)
Story = []
for (an, sam, event), props in report_data.iteritems():
Story.append(Paragraph("%s - sample %s results for %s" % (an.name, sam.name, event.name), styles["Heading2"]))
data_list = [['Lab', 'Method', 'Instrument', 'Unit', 'Raw Result', 'Converted Result', 'Outlier']]
for (index, series) in props['frame'].iterrows():
data_list.append(_format([
Paragraph(Lab.objects.get(pk=series['labs']).name, styles['BodyText']),
Paragraph(Method.objects.get(pk=series['methods']).name, styles['BodyText']),
Paragraph(Instrument.objects.get(pk=series['instruments']).name, styles['BodyText']),
Paragraph(Unit.objects.get(pk=series['units']).name, styles['BodyText']),
series['raw_results'],
series['results'],
series['outlier']
]))
table = Table(data_list, colWidths=[45 * mm, 35 * mm, 35 * mm, 25 * mm, 35 * mm, 35 * mm, 35 * mm], repeatRows=1)
Story.append(table)
Story.append(PageBreak())
if props['graph'] is not None:
Story.append(Image("/tmp/%s" % props['graph'], width=10 * inch, height=6 * inch))
Story.append(PageBreak())
rpt.draw(Story, onFirstPage=setup_header_and_footer, onLaterPages=setup_header_and_footer)
return file_name
背景资料
- 页面设置为 A4,横向
- 我的开发环境是virtualenv;PIL 1.1.7 和 reportlab 2.6 已安装并正常运行
上面使用的“报告”类只是一个简单的包装器,
SimpleDocTemplate
它设置了一些默认值,但委托给 SimpleDocTemplate 的build
实现。它的代码是:class Report(object): def __init__(self, filename, doctitle="Report", docauthor="<default>", docsubject="<default>", doccreator="<default>", orientation="landscape", size=A4): DEFAULTS = { 'leftMargin' : 10 * mm, 'rightMargin' : 10 * mm, 'bottomMargin' : 15 * mm, 'topMargin' : 36 * mm, 'pagesize' : landscape(size) if orientation == "landscape" else portrait(size), 'title' : doctitle, 'author' : docauthor, 'subject' : docsubject, 'creator' : doccreator } self.doc = SimpleDocTemplate(filename, **DEFAULTS) def draw(self, flowables, onFirstPage=setup_header_and_footer, onLaterPages=setup_header_and_footer): self.doc.build(flowables, onFirstPage=setup_header_and_footer, onLaterPages=setup_header_and_footer, canvasmaker=NumberedCanvas)
我已经看过的
- 我已经确认图像存在于磁盘上。这些路径是完全限定的路径。
- PIL 已安装,并且能够正确读取图像
- 分配给图像的空间足够;我已经通过计算证实了这一点。此外,如果我增加图像大小,ReportLab 会抱怨 Image Flowable 太大。当前尺寸应该适合。
- 我已经测试过分页符和不分页符;他们似乎没有任何区别
问题
表格、标题和页面模板呈现正常,但图像为空白。今天早些时候,我在设置此报告使用的模板时遇到了这个问题。解决方法是使用canvas.drawInlineImage(...
. canvas.DrawImage(...
因此,我的设置似乎存在问题;我可以使用一些关于如何调试它的指针。
更新
我能够应用此链接问题中使用的相同解决方法的变体(使用canvas.drawInlineImage
代替canvas.drawImage
。我将“图像”子类化如下:
class CustomImage(Image):
"""
Override - to use inline image instead; inexplicable bug with non inline images
"""
def draw(self):
lazy = self._lazy
if lazy>=2: self._lazy = 1
self.canv.drawInlineImage(self.filename,
getattr(self,'_offs_x',0),
getattr(self,'_offs_y',0),
self.drawWidth,
self.drawHeight
)
if lazy>=2:
self._img = None
self._lazy = lazy
“股票”图像类的唯一变化是一行 - 使用self.canv.drawInlineImage
以前的位置self.canvas.drawImage
。从图像最终在我的 PDF 中可见的意义上说,这“有效”。不工作的原因drawImage
仍然是个谜。
我已经尝试过@PedroRomano 的建议(以确保图像 RGBA ),甚至尝试使用 JPEG 图像而不是 PNG。这些并没有什么不同。