Draw images with canvas and use SimpleDocTemplate
我正在django视图中用reportlab编写pdf文件,它们非常简单,包括页眉,内容和页脚。
我正在使用非常适合的SimpleDocTemplate,在内容中绘制表格,页脚和页眉使用以下方法:
1 | build([data], onFirstPage=drawPageFrame, onLaterPages=drawPageFrame). |
我的问题是,如何像使用Canvas.drawImage(...)一样绘制图像? 我需要一个"浮动"图像...放置在我想要的文本上方,并且使用SimpleDocTemplate我没有Canvas对象来执行此操作。
搜索我发现了这一点:
The platypus layout stuff uses flowables. Packers normally set the
attribute canv onto each flowable when it is being wrapped, split or
drawn ie around the wrap, split and draw methods. Inside those methods
you have access to the canvas using the canv attribute of self.
如何使用?
嗯,还有更多要测试的东西:
1 2 | flowables.Macro flowables.CallerMacro |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 | # -*- coding: utf-8 -*- from reportlab.lib.pagesizes import A4, landscape, portrait from reportlab.lib.styles import getSampleStyleSheet from reportlab.platypus import Table, Flowable, SimpleDocTemplate, Paragraph, Spacer, Image from reportlab.lib import randomtext from reportlab import platypus import os,random styles = getSampleStyleSheet() path = os.path.realpath(os.path.dirname(__file__)) def drawPageFrame(canvas, doc): canvas.saveState() canvas.drawImage(path+"/ujiPDF.jpg",50,50,57,57) canvas.restoreState() doc = SimpleDocTemplate("salida.pdf",pagesize=A4) elementos = [] com = 'canvas.drawImage("'+path+'/ujiPDF.jpg",100,100,57,57)' print com elementos.append(platypus.flowables.Macro('canvas.saveState()')) print platypus.flowables.Macro(com) elementos.append(platypus.flowables.Macro(com)) elementos.append(platypus.flowables.Macro('canvas.restoreState()')) para = Paragraph(randomtext.randomText(randomtext.PYTHON,20), styles["Normal"]) elementos.append(para) doc.build(elementos,onFirstPage=drawPageFrame, onLaterPages=drawPageFrame) |
这是宏方法...干净的出口,但没有第二个图像。
如果我理解正确的话,您不需要将整个文档模板作为子类,您只需要一个可将其放入SimpleDocTemplate中的可流动对象即可。您可以使用Flowable本身的非常简单的子类来实现这一点。
->特别是由于人们经常询问如何将matplotlib对象放入reportlab,因此我将展示如何使用matplotlib生成图,然后使用修改后的flowable将该图放入SimpleDocTemplate(而不将文件保存到磁盘)。该概念适用于任何文件或可以馈入cStringIO的任何内容
->在下面的DOES中,您可以将图形放在文本上方(将高度从负值更改为正值,以将其推到绘图所在部分顶部的假想线上方)
关键概念是每个Flowable本身都包含我们可以在其上绘制的画布。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 | import matplotlib.pyplot as plt import cStringIO from reportlab.lib.units import inch, cm from reportlab.lib.pagesizes import letter from reportlab.platypus import SimpleDocTemplate, Paragraph, Spacer, Image, Table from reportlab.lib.styles import getSampleStyleSheet, ParagraphStyle import reportlab.lib, reportlab.platypus class flowable_fig(reportlab.platypus.Flowable): def __init__(self, imgdata): reportlab.platypus.Flowable.__init__(self) self.img = reportlab.lib.utils.ImageReader(imgdata) def draw(self): self.canv.drawImage(self.img, 0, 0, height = -2*inch, width=4*inch) # http://www.reportlab.com/apis/reportlab/2.4/pdfgen.html doc = SimpleDocTemplate(("report.pdf"),pagesize=letter, rightMargin=72,leftMargin=72, topMargin=72,bottomMargin=18) Story=[] styles=getSampleStyleSheet() ptext = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Morbi massa dolor, vulputate quis elit sed, sagittis consectetur erat. Sed lobortis nisi eros, eu maximus enim iaculis ac. Vestibulum sagittis urna nec interdum aliquam. Pellentesque ornare velit ut ante ullamcorper, vulputate accumsan nisi vulputate. Fusce consectetur dolor quam. Phasellus hendrerit, ligula vel consectetur pretium, lorem neque dapibus eros, ornare suscipit ipsum dolor id nisl. Sed vel orci id leo efficitur lobortis sit amet id risus. Nullam euismod, ipsum a posuere scelerisque, ante lorem ultrices nibh, ut feugiat metus ex congue enim. Nam lobortis, metus id pellentesque feugiat, arcu orci rutrum felis, quis luctus urna nisl at nulla. Donec eu eros pharetra dolor congue facilisis at ac magna. Nullam eu ultricies metus. Sed sodales, libero viverra pellentesque tempus, magna purus convallis nibh, eu condimentum tortor erat tincidunt turpis. Vestibulum scelerisque tincidunt egestas. Nullam commodo diam nisl, sed consequat ex sagittis eu.' Story.append(Paragraph(ptext, styles["Normal"])) fig = plt.figure(figsize=(10, 3)) plt.plot([1,2,3,4]) plt.ylabel('This is a boring plot') imgdata = cStringIO.StringIO() fig.savefig(imgdata, format='png') imgdata.seek(0) # rewind the data pic = flowable_fig(imgdata) Story.append(pic) doc.build(Story) |
您可以扩展此最小化的代码片段,以完成所需的操作,还可以向Story变量中添加尽可能多的片段(实际上这只是可流动对象列表)。神奇之处在于,我们正在为文档构建器提供一个Flowable,该Flowable使用Flowable自己的画布绘制图形。
编辑:我差点忘了,因为这给了我们对drawImage的完全访问权限,我们还可以使图片或绘图具有透明度,以便可以通过它看到其他事物。 drawImage上的文档具有详细信息。
鸭嘴兽中有Image类。只需使用
最好的选择是创建SimpleDocTemplate或BaseDocTemplate的子类。在构建方法中,您将有权访问画布。如果您想利用SimpleDocTemplate所做的一切,则可以尝试直接从