我想使用python创建一个多页PDF文件进行打印。我宁愿使用在任何环境下都能正常运行的库来执行此操作,因此我尝试使用 matplotlib 。
我可以生成所需的A4大小的图形,并创建多页PDF文件,但是图形艺术家会缩小并且不会占据整个页面。
我可以生成一个A4大小的PDF页面,并在其周围填充空白,或者生成一个PDF文件,其页面尺寸较小(页面在该图形周围收缩以删除填充,而图形保持相同大小)。 / p>
有人可以帮忙吗?
(我不认为这是重复的,还有关于填充的其他问题,但目的与我不同,我看不到任何对我有用的答案-我已经搜寻了好几天!)
到目前为止,在尝试一些SO问题/答案后,我尝试做的是:
bbox_inches='tight', pad_inches=0
函数中使用pyplot.savefig()
,该页面小于A4 PDF页面,没有填充。(奇怪的是,这在EPS格式下效果很好,它可以在A4纸上直接打印成实际大小,但是我不知道如何用它做多页文件。)
如果我不使用bbox_inches='tight', pad_inches=0
,则图形的大小似乎或多或少都相同,但是周围的填充填充了空间以形成A4页面。
使用fig.tight_layout(rect=[0,0,1,1])
或fig.tight_layout(pad=0)
保留填充。
此示例代码以EPS(A4无填充),PDF(A4有填充)和TIFF(无填充但尺寸较小,因此在打印时具有填充)生成图形:
import matplotlib.pyplot as plt
from matplotlib import gridspec
def plot_form_page():
# A4 canvas
fig_width_cm = 21 # A4 page
fig_height_cm = 29.7
inches_per_cm = 1 / 2.54 # Convert cm to inches
fig_width = fig_width_cm * inches_per_cm # width in inches
fig_height = fig_height_cm * inches_per_cm # height in inches
fig_size = [fig_width, fig_height]
plt.rc('text', usetex=False) # so that LaTeX is not needed when creating a PDF with PdfPages later on
fig = plt.figure()
fig.set_size_inches(fig_size)
fig.set_facecolor('#9999ff')
gs = gridspec.GridSpec(29, 21, wspace=0.1, hspace=0.1)
# external axis
ax0 = fig.add_subplot(gs[:, :])
# for side in ["top", "bottom", "left", "right"]:
# ax0.spines[side].set_visible(False)
# header
ax1 = fig.add_subplot(gs[1:4, 1:4])
ax1.set_facecolor('#bbffbb')
ax2 = fig.add_subplot(gs[1:4, 4:-6])
ax2.set_facecolor('#ffbbff')
ax3 = fig.add_subplot(gs[1:4, -6:-1])
ax3.set_facecolor('#00bbff')
# form
ax4, ax5, ax6, ax7 = [fig.add_subplot(gs[ 5+(i*6):10+(i*6), 1:10]) for i in range(4)]
[ax8, ax9, ax10, ax11] = [fig.add_subplot(gs[ 5+(i*6):10+(i*6), 11:20]) for i in range(4)]
axes = [ax4, ax8, ax5, ax9, ax6, ax10, ax7, ax11]
for ax in axes:
ax.set_facecolor('#bbbbbb')
for ax in fig.axes:
ax.set_xticks([])
ax.set_yticks([])
print(fig.properties()['figwidth'], fig.properties()['figheight'])
# from previous tests:
#fig.tight_layout(pad=0)
plot_form_page()
plt.savefig('page.tiff',
dpi=300,
orientation='portrait',
box_inches='tight', pad_inches=0)
plt.savefig('page.pdf',
dpi=300,
orientation='portrait',
bbox_inches='tight', pad_inches=0)
plt.savefig('page.eps',
dpi=300,
orientation='portrait',
papertype='a4',
bbox_inches='tight', pad_inches=0)
fig.properties()给我合适的A4尺寸(以英寸为单位):
fig.properties()['figwidth']
fig.properties()['figheight']
8.26771653543307 11.69291338582677
但是打印时图形会缩小以允许填充
您可以在以下PDF文件中看到效果:
https://www.dropbox.com/s/naqy9vj4ht6g2mv/Screenshot%202019-04-16%20at%2011.46.48.png?dl=0
使用EPS没什么问题:
https://www.dropbox.com/s/umycd8ofwod3nk5/Screenshot%202019-04-16%20at%2012.05.52.png?dl=0
(但同样,我不知道如何创建多页EPS)
使用TIFF文件,我得到的是相同的:
https://www.dropbox.com/s/eid1zstbm954121/Screenshot%202019-04-16%20at%2012.11.59.png?dl=0
这不一定是因为打印软件添加了填充物-现在的数字实际上更小了,因为它在300dpi时为1929××2655像素,这使其为6.43×8.85英寸-肯定小于A4。
编辑:当然,这可以通过使用“适合页面”选项进行打印来解决,但是它可以处理具有A4的比例的任何图像,我想控制如果可能的话,请超出尺寸。
答案 0 :(得分:0)
由于@ImportanceOfBeingErnest的评论以及一些试验和错误,我找到了一个解决方案,该解决方案创建了一个A4形图形,当另存为A4 PDF页面时,它占据了整个页面(不是100%正确,虽然利润很小,但可以达到目的。
只需添加gs.tight_layout(fig, pad=0)
。我不完全理解为什么,但是我怀疑当我尝试消除figure
对象(使用plt.savefig(..., bbox_inches='tight', pad_inches=0)
或fig.tight_layout(...)
的填充时,图形可能会到达图形的边界)。 A4页面,但是GridSpec
不会,因此我看到的填充是在GridSpec
和figure
的边界之间,而不是纸张的边界。当GridSpec
和figure
之间的填充设置为0,并且figure
的大小为A4时,事情似乎按我的意愿工作。
(我仍然不明白为什么在某些情况下PDF文件的页面会缩小,但是我将其留在这里)。
这是执行我想要的代码,如果创建PDF文件,打开并检查属性,大小为A4,矩形靠近边框。与其他选项相比,页边距较大或页面尺寸较小。
import matplotlib.pyplot as plt
from matplotlib import gridspec
def plot_form_page():
# A4 canvas
fig_width_cm = 21 # A4 page
fig_height_cm = 29.7
inches_per_cm = 1 / 2.54 # Convert cm to inches
fig_width = fig_width_cm * inches_per_cm # width in inches
fig_height = fig_height_cm * inches_per_cm # height in inches
fig_size = [fig_width, fig_height]
plt.rc('text', usetex=False) # so that LaTeX is not needed when creating a PDF with PdfPages later on
fig = plt.figure()
fig.set_size_inches(fig_size)
fig.set_facecolor('#9999ff')
gs = gridspec.GridSpec(29, 21, wspace=0.1, hspace=0.1)
# external axis
ax0 = fig.add_subplot(gs[:, :])
# for side in ["top", "bottom", "left", "right"]:
# ax0.spines[side].set_visible(False)
# header
ax1 = fig.add_subplot(gs[1:4, 1:4])
ax1.set_facecolor('#bbffbb')
ax2 = fig.add_subplot(gs[1:4, 4:-6])
ax2.set_facecolor('#ffbbff')
ax3 = fig.add_subplot(gs[1:4, -6:-1])
ax3.set_facecolor('#00bbff')
# form
ax4, ax5, ax6, ax7 = [fig.add_subplot(gs[ 5+(i*6):10+(i*6), 1:10]) for i in range(4)]
[ax8, ax9, ax10, ax11] = [fig.add_subplot(gs[ 5+(i*6):10+(i*6), 11:20]) for i in range(4)]
axes = [ax4, ax8, ax5, ax9, ax6, ax10, ax7, ax11]
for ax in axes:
ax.set_facecolor('#bbbbbb')
for ax in fig.axes:
ax.set_xticks([])
ax.set_yticks([])
#print(fig.properties()['figwidth'], fig.properties()['figheight'])
gs.tight_layout(fig, pad=0)
plot_form_page()
plt.savefig('page.pdf',
dpi=300,
orientation='portrait')