3

我正在编写一份包含表格和图像混合的报告。图像 [ 实际上是图形 ] 以 .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。这些并没有什么不同。

4

2 回答 2

1

我最终通过使用自定义Image子类结束了这件事:

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

将图形保存为矢量图形格式,例如 EPS、保存为 JPEG、保存为带有和不带有 alpha 通道的 PNG 似乎都没有什么不同。

于 2012-10-29T09:19:29.623 回答
0

最佳解决方案是以矢量格式(如reportlab支持的 postscript)生成图表。许多 UNIX 软件开箱即用,您可以在 Windows 上使用出色的PDFCreator

如果您的图形必须使用光栅图像,请尝试将图像转换为 JPEG 格式。这些可以使用DCTDecode过滤器轻松嵌入到 PDF 文件中。(这就是jpeg2pdf 所做的。)

于 2012-10-22T21:28:12.003 回答