1

我目前正在编写一个应用程序,使用 iText 2.1.7 根据我们的要求“格式化”PDF。

我们基本上采用纵向 PDF,并缩小页面,因此我们可以将原始 PDF 的 2 页放在新 PDF 的一个横向页面上。我们还在页面底部留出一些空间用于后期处理。

这个过程在 90% 的时间里都能正常工作。

但是,我们收到了一个已被内容部门裁剪/修剪的 PDF,当我们在 Acrobat 中查看此 PDF 时,它看起来与预期的一样。但是,当我们处理它时,新的 PDF 包括整个原始 MediaBox 和裁剪线。

这是我们使用的代码,以及问题输出的外观。

File tempFile = new File(tempFilename);
PdfReader reader = new PdfReader(originalPdfFile);
Document doc = new Document(new RectangleReadOnly(842f, 595f), 0, 0, 0, 0);
PdfWriter writer = PdfWriter.getInstance(doc, new FileOutputStream(tempFile));
doc.open();
for (int i = 1; i < reader.getNumberOfPages(); i = i + 2) {
    doc.newPage();
    PdfContentByte cb = writer.getDirectContent();
    PdfImportedPage page = writer.getImportedPage(reader, i); // page #1

    float documentWidth = doc.getPageSize().getWidth() / 2;
    float documentHeight = doc.getPageSize().getHeight() - 65f;

    float pageWidth = page.getWidth();
    float pageHeight = page.getHeight();

    float widthScale = documentWidth / pageWidth;
    float heightScale = documentHeight / pageHeight;
    float scale = Math.min(widthScale, heightScale);

    float offsetX = (documentWidth - (pageWidth * scale)) / 2;
    float offsetY = 65f; //100f

    cb.addTemplate(page, scale, 0, 0, scale, offsetX, offsetY);

    PdfImportedPage page2 = writer.getImportedPage(reader, i+1); // page #2

    pageWidth = page.getWidth();
    pageHeight = page.getHeight();

    widthScale = documentWidth / pageWidth;
    heightScale = documentHeight / pageHeight;
    scale = Math.min(widthScale, heightScale);

    offsetX = ((documentWidth - (pageWidth * scale)) / 2) + documentWidth;
    offsetY = 65f; //100f

    cb.addTemplate(page2, scale, 0, 0, scale, offsetX, offsetY);//430f
    }

doc.close();

acrobat 中的原文:

在此处输入图像描述

在 acrobat 中修改,显示不需要的 pretrim 内容:

在此处输入图像描述

4

3 回答 3

3

尽管如果不查看 PDF 本身就很难确定,但我怀疑您的问题是该 PDFCropBox至少在其某些页面上指定了 a 。如果是这种情况,那么我认为您会想要page.setBoundingBox(reader.getCropBox(i));在获得页面参考后立即执行类似操作。

请注意,页面的默认值为CropBoxit's MediaBox,因此添加上述行不应对未指定 a 的 PDF 页面的布局产生负面影响CropBox

(我不是 iText 用户,所以这是我的一些猜测......)

祝你好运!

于 2013-10-25T05:14:15.053 回答
1

在经历了很多挫折之后,我终于在进行缩放和布局处理之前通过“硬裁剪”PDF来实现这一点。

硬裁剪采用 Acrobat 裁剪的 PDF(裁剪 = 隐藏),并用于PdfStamper创建仅包含裁剪框内内容的新 PDF。

public String cropPdf(String pdfFilePath) throws DocumentException, IOException {
    String filename = FilenameUtils.getBaseName(pdfFilePath) + "_cropped." + FilenameUtils.getExtension(pdfFilePath);
    filename = FilenameUtils.concat(System.getProperty("java.io.tmpdir"), filename);
    PdfReader reader = new PdfReader(pdfFilePath);
    try {
        PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(filename));
        try {
            for (int i = 1; i <= reader.getNumberOfPages(); i++) {
                PdfDictionary pdfDictionary = reader.getPageN(i);
                PdfArray cropArray = new PdfArray();
                Rectangle cropbox = reader.getCropBox(i);                   
                cropArray.add(new PdfNumber(cropbox.getLeft()));
                cropArray.add(new PdfNumber(cropbox.getBottom()));
                cropArray.add(new PdfNumber(cropbox.getLeft() + cropbox.getWidth()));
                cropArray.add(new PdfNumber(cropbox.getBottom() + cropbox.getHeight()));
                pdfDictionary.put(PdfName.CROPBOX, cropArray);
                pdfDictionary.put(PdfName.MEDIABOX, cropArray);
                pdfDictionary.put(PdfName.TRIMBOX, cropArray);
                pdfDictionary.put(PdfName.BLEEDBOX, cropArray);
            }
            return filename;
        } finally {
            stamper.close();
        }
    } finally {
        reader.close();
    }
}
于 2013-12-12T23:18:19.710 回答
1

对 Kabals 答案的一个小而重要的修复:盒子期望宽度/高度而不是坐标:

            ...
            cropArray.add(new PdfNumber(cropbox.getLeft()));
            cropArray.add(new PdfNumber(cropbox.getBottom()));
            cropArray.add(new PdfNumber(cropbox.getWidth()));
            cropArray.add(new PdfNumber(cropbox.getHeight()));
            ...
于 2014-10-29T10:41:01.940 回答