在PDF文档中使用ReportLab插入PDF图像(Python)
我把一些用matplotlib生成的图保存成了pdf格式,因为这个格式看起来质量更好。现在我想知道怎么把这个pdf图片放到一个用ReportLab制作的pdf文档里。方便的方法Image(filename)
对这个格式不适用。
4 个回答
0
你可以使用svg格式从matplotlib导出图形,然后利用svglib
这个Python库把矢量图形放到用reportlab生成的PDF文件里。svglib
会把一个svg文件转换成一个可以直接在reportlab中使用的绘图对象。
想了解更多细节,可以看看这个问题:从SVG输入生成PDF
2
根据ReportLab的常见问题解答,这只有在使用ReportLab PLUS的情况下才能实现:
我可以在我的PDF中使用矢量图形吗?
不可以,开源版本不支持这个功能。PageCatcher(见之前的回答)可以让你轻松地将任何矢量图像保存为PDF,然后像使用图片文件一样使用它。而且,报告标记语言(Report Markup Language)可以接受PDF文件以及JPG、GIF和PNG格式的文件。
更新:我有一段时间没关注这个了,但在pdfrw的页面上写着:
pdfrw可以读取和写入PDF文件,也可以用来读取PDF,这些PDF可以在reportlab中使用。
4
你可以使用很棒的pdfrw库,配合reportlab,把matplotlib图形直接传入一个可流动的对象中:
这个问题之前有人回答过,但我想在这里给一个简单的例子。你也可以看看这里: https://stackoverflow.com/a/13870512/4497962
from io import BytesIO
import matplotlib.pyplot as plt
from pdfrw import PdfReader, PdfDict
from pdfrw.buildxobj import pagexobj
from pdfrw.toreportlab import makerl
from reportlab.platypus import Flowable
from reportlab.lib.enums import TA_JUSTIFY,TA_LEFT,TA_CENTER,TA_RIGHT
class PdfImage(Flowable):
"""
PdfImage wraps the first page from a PDF file as a Flowable
which can be included into a ReportLab Platypus document.
Based on the vectorpdf extension in rst2pdf (http://code.google.com/p/rst2pdf/)
This can be used from the place where you want to return your matplotlib image
as a Flowable:
img = BytesIO()
fig, ax = plt.subplots(figsize=(canvaswidth,canvaswidth))
ax.plot([1,2,3],[6,5,4],antialiased=True,linewidth=2,color='red',label='a curve')
fig.savefig(img,format='PDF')
return(PdfImage(img))
"""
def __init__(self, filename_or_object, width=None, height=None, kind='direct'):
# If using StringIO buffer, set pointer to begining
if hasattr(filename_or_object, 'read'):
filename_or_object.seek(0)
#print("read")
self.page = PdfReader(filename_or_object, decompress=False).pages[0]
self.xobj = pagexobj(self.page)
self.imageWidth = width
self.imageHeight = height
x1, y1, x2, y2 = self.xobj.BBox
self._w, self._h = x2 - x1, y2 - y1
if not self.imageWidth:
self.imageWidth = self._w
if not self.imageHeight:
self.imageHeight = self._h
self.__ratio = float(self.imageWidth)/self.imageHeight
if kind in ['direct','absolute'] or width==None or height==None:
self.drawWidth = width or self.imageWidth
self.drawHeight = height or self.imageHeight
elif kind in ['bound','proportional']:
factor = min(float(width)/self._w,float(height)/self._h)
self.drawWidth = self._w*factor
self.drawHeight = self._h*factor
def wrap(self, availableWidth, availableHeight):
"""
returns draw- width and height
convenience function to adapt your image
to the available Space that is available
"""
return self.drawWidth, self.drawHeight
def drawOn(self, canv, x, y, _sW=0):
"""
translates Bounding Box and scales the given canvas
"""
if _sW > 0 and hasattr(self, 'hAlign'):
a = self.hAlign
if a in ('CENTER', 'CENTRE', TA_CENTER):
x += 0.5*_sW
elif a in ('RIGHT', TA_RIGHT):
x += _sW
elif a not in ('LEFT', TA_LEFT):
raise ValueError("Bad hAlign value " + str(a))
#xobj_name = makerl(canv._doc, self.xobj)
xobj_name = makerl(canv, self.xobj)
xscale = self.drawWidth/self._w
yscale = self.drawHeight/self._h
x -= self.xobj.BBox[0] * xscale
y -= self.xobj.BBox[1] * yscale
canv.saveState()
canv.translate(x, y)
canv.scale(xscale, yscale)
canv.doForm(xobj_name)
canv.restoreState()