Java pdfbox-签名横向文件错误

Java pdfbox-签名横向文件错误,java,pdf,pdfbox,Java,Pdf,Pdfbox,我正在使用pdfbox-1.8.8对PDF文件执行签名功能 它在纵向模式下与PDF文件配合良好。但对于横向文件,我有一个问题 看起来横向文件的坐标错误 有人知道这个文件有什么问题吗 这里是 这是我用来签名的代码 public void signDetached(String inputFilePath, String outputFilePath, String signatureImagePath, Sign signProperties) { OutputStream outpu

我正在使用pdfbox-1.8.8对PDF文件执行签名功能

它在纵向模式下与PDF文件配合良好。但对于横向文件,我有一个问题

看起来横向文件的坐标错误

有人知道这个文件有什么问题吗

这里是

这是我用来签名的代码

public void signDetached(String inputFilePath, String outputFilePath, String signatureImagePath, Sign signProperties) {
    OutputStream outputStream = null;
    InputStream inputStream = null;
    PDDocument document = null;
    InputStream signImageStream = null;

    try {
        setTsaClient(null);
        document = PDDocument.load(inputFilePath);
        // create signature dictionary
        PDSignature signature = new PDSignature();
        signature.setFilter(PDSignature.FILTER_ADOBE_PPKLITE);
        signature.setSubFilter(PDSignature.SUBFILTER_ADBE_PKCS7_DETACHED);
        signature.setName("VANDUC1102");
        signature.setLocation(null);
        String displayName = "Hello World, Document signed by VANDUC1102";
        String reason = reasonText+ " " + displayName;
        signature.setReason(reason);

        // the signing date, needed for valid signature
        signature.setSignDate(Calendar.getInstance());            
        int signatureInPage = signProperties.getPageNumber() + 1;
        signImageStream = new FileInputStream(new File(signatureImagePath));
        PDVisibleSignDesigner visibleSig = new PDVisibleSignDesigner(inputFilePath, signImageStream, signatureInPage);

        float xAxis = convertPixel2Point(signProperties.getX()) ;
        float yAxis = convertPixel2Point(signProperties.getY());               
        float signImageHeight = convertPixel2Point(signImageHeight);    
        float signImageWidth = convertPixel2Point(signImageWidth);

        visibleSig.xAxis(xAxis)
                .yAxis(yAxis)
                .zoom(0)
                .signatureFieldName("Signature")
                .height(signImageHeight)
                .width(signImageWidth);
        PDVisibleSigProperties signatureProperties = new PDVisibleSigProperties();

        signatureProperties.signerName(eiUser.getName())
                 .signerLocation(null)
                 .signatureReason(reason)
                 .preferredSize(0)
                 .page(signProperties.getPageNumber())
                 .visualSignEnabled(true)
                 .setPdVisibleSignature(visibleSig)
                 .buildSignature();
         // register signature dictionary and sign interface
        SignatureOptions signatureOptions = new SignatureOptions();
        signatureOptions.setVisualSignature(signatureProperties);
        signatureOptions.setPage(signatureInPage);
        document.addSignature(signature, this, signatureOptions);

        File outputFile = new File(outputFilePath);
        outputStream = new FileOutputStream(outputFile);
        inputStream = new FileInputStream(inputFilePath);
        IOUtils.copyStream(inputStream, outputStream);
        document.saveIncremental(inputStream, outputStream);
        outputStream.flush();
    } catch (COSVisitorException | SignatureException | IOException ex) {
        log.error("signDetached ", ex);
    } finally {
        IOUtils.closeStream(outputStream);
        IOUtils.closeStream(inputStream);
        IOUtils.closeStream(signImageStream);
        IOUtils.closeStream(document);
    }
}
private float convertPixel2Point(float pixel){
    return pixel * (float) 72/96;
}
正如我所说的,这段代码可以很好地与人像PDF配合使用


谢谢。

相关页面的旋转值不为零。PDFBox视觉签名类完全忽略此值,因此必须为其提供坐标和尺寸,就像页面未旋转一样

这可以通过添加以下
开关
语句来实现:

float xAxis = convertPixel2Point(/*signProperties.getX()*/x) ;
float yAxis = convertPixel2Point(/*signProperties.getY()*/y);               
float signImageHeight = convertPixel2Point(/*signImageHeight*/324);    
float signImageWidth = convertPixel2Point(/*signImageWidth*/309);

int rotation = getPageRotation(inputFilePath, page) % 360;
switch (rotation)
{
case 0:
    // all ok;
    break;
case 90:
    visibleSig.affineTransformParams(new byte[] {0, 1, -2, 0, 100, 0})
              .formaterRectangleParams(new byte[]{0, 0, 100, 100});

    float temp = yAxis;
    yAxis = visibleSig.getPageHeight() - xAxis - signImageWidth;
    xAxis = temp;

    temp = signImageHeight;
    signImageHeight = signImageWidth;
    signImageWidth = temp;

    break;
case 180:
    // Implement in a similar fashion
case 270:
    // Implement in a similar fashion
}

visibleSig.xAxis(xAxis)
          .yAxis(yAxis)
          .zoom(0)
          .signatureFieldName("Signature")
          .height(signImageHeight)
          .width(signImageWidth);
以及以下方法:

private int getPageRotation(String documentPath, int page) throws IOException
{
    try (PDDocument document = PDDocument.load(documentPath))
    {
        List<?> pages = document.getDocumentCatalog().getAllPages();
        PDPage pageObject =(PDPage) pages.get(page);
        return pageObject.getRotation();
    }
}
private int getPageRotation(字符串documentPath,int-page)引发IOException
{
try(PDDocument=PDDocument.load(documentPath))
{
列表页面=document.getDocumentCatalog().getAllPages();
PDPage pageObject=(PDPage)pages.get(page);
返回pageObject.getRotation();
}
}
对于180和270的旋转值,必须进行类似的修正


(测试方法
testLandscapeOriginal
testLandscapeFixed
in)

从2.0.5版开始,可以通过调用
adjustForRotation()
来调整旋转页面的签名字段。最后,这一行

visibleSignDesigner.xAxis(x).yAxis(y).zoom(zoomPercent);
必须更改为:

visibleSignDesigner.xAxis(x).yAxis(y).zoom(zoomPercent).adjustForRotation();
更改()部分基于@mkl的答案。1.8*版将不可用

2018年10月24日更新:

2.0.12之前的所有2.0.*版本都存在以下问题:
adjustForRotation()
硬编码为200 x 100图像大小。这已在2.0.13中修复。或者派生您自己的
PDVisibleSignDesigner
并使用正确的实现扩展
adjustForRotation()

public PDVisibleSignDesigner adjustForRotation()
{
    switch (rotation)
    {
        case 90:
            float temp = yAxis;
            yAxis = pageHeight - xAxis - imageWidth;
            xAxis = temp;

            affineTransform = new AffineTransform(
                    0, imageHeight / imageWidth, -imageWidth / imageHeight, 0, imageWidth, 0);

            temp = imageHeight;
            imageHeight = imageWidth;
            imageWidth = temp;
            break;

        case 180:
            float newX = pageWidth - xAxis - imageWidth;
            float newY = pageHeight - yAxis - imageHeight;
            xAxis = newX;
            yAxis = newY;

            affineTransform = new AffineTransform(-1, 0, 0, -1, imageWidth, imageHeight);
            break;

        case 270:
            temp = xAxis;
            xAxis = pageWidth - yAxis - imageHeight;
            yAxis = temp;

            affineTransform = new AffineTransform(
                    0, -imageHeight / imageWidth, imageWidth / imageHeight, 0, 0, imageHeight);

            temp = imageHeight;
            imageHeight = imageWidth;
            imageWidth = temp;
            break;

        case 0:
        default:
            break;
    }
    return this;
}

最有可能的是,页面有一个面向纵向的坐标系,并且是横向的,因为它还有一个旋转值,告诉PDF查看者旋转它以便显示。实际上,每个页面字典都包含一个
/Rotate 90
条目(如果以文本形式打开,则第484-519行),因此,OP在应用签名时必须考虑旋转。由于他还没有展示任何代码,甚至没有解释“签名”是一个简单的图像还是一个填充的首字母签名字段,所以说更多的话是高度推测性的。啊,根据OP同时添加到他的问题中的代码,他确实尝试在那里添加一个集成的PDF签名。不幸的是,用于创建可见签名的PDFBox类非常可怕。据我所知,他们忽略了页面旋转。因此,要么扩展这些类以明确支持旋转页面,要么自己阅读页面旋转,重新计算签名的位置和大小,旋转图像,并使用这些修改后的数据使用PDFBox类。非常感谢,我认为代码应该放在PDFBox源代码中,事件pdfbox2.0.0RC不支持此功能。我认为代码应该放在pdfbox源代码中-嗯。。。不我认为PDFBox代码在这方面应该是固定的。上面的代码只是一个解决方案……;)抱歉@mkl,我太笨了,我找不到visibleSig.affineTransformParams(新字节[]{0,1,-2,0,100,0})参数的文档。我想完成180和270个案例。我不能用visibleSig.affineTransformParams(新字节[]{0,1,-2,0,100,0})旋转180个。formaterRectangleParams(新字节[]{0,0,100,100})更改此参数后。。。。180美元也是如此。我将图像旋转了180度,然后将其放入visibleSign。270箱。我还旋转了180次,执行了90次旋转@mkl@mkl我创造了。90°情况需要稍微不同的转换,PDFBox本身不支持这种转换,因为它只接受字节(!)转换矩阵元素。这已在快照2.*和trunk中修复,并且byte方法已被弃用。