如何保存占据PDF文件整个页面的图

时间:2019-04-16 11:18:31

标签: python pdf matplotlib figure

我想使用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的比例的任何图像,我想控制如果可能的话,请超出尺寸。

1 个答案:

答案 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不会,因此我看到的填充是在GridSpecfigure的边界之间,而不是纸张的边界。当GridSpecfigure之间的填充设置为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')