使用PDFBox从PDF获取可见签名?

时间:2013-06-05 21:06:24

标签: pdf digital-signature capture signature pdfbox

是否可以使用OSS libary PDFBox提取已签名PDF的可见签名(捕获图像)?

工作流:   - 列出文件的所有签名   - 带签名的节目包括可见的签名   - 显示哪些是有效的   - 提取签名图像(需要为每个签名提取正确的图像)

像下面这样的oop风格的东西会很棒: PDFSignatures [] sigs = document.getPDFSignatures() SIG [0] .getCN() ... (缓冲)图像visibleSig = sig [0] .getVisibleSignature()

找到类PDSignature以及如何签名PDF,但不是将可见签名提取为图像的解决方案。

感谢您的帮助! 迎接

1 个答案:

答案 0 :(得分:9)

由于没有人回答,我自己在评论中尝试了我的提议。第一个结果:

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.pdfbox.cos.COSName;
import org.apache.pdfbox.pdfviewer.PageDrawer;
import org.apache.pdfbox.pdmodel.PDPage;
import org.apache.pdfbox.pdmodel.common.PDRectangle;
import org.apache.pdfbox.pdmodel.graphics.PDGraphicsState;
import org.apache.pdfbox.pdmodel.interactive.annotation.PDAnnotation;
import org.apache.pdfbox.pdmodel.interactive.annotation.PDAppearanceDictionary;
import org.apache.pdfbox.pdmodel.interactive.annotation.PDAppearanceStream;

public class AnnotationDrawer extends PageDrawer
{
    public AnnotationDrawer(int imageType, int resolution) throws IOException
    {
        super();
        this.imageType = imageType;
        this.resolution = resolution;
    }

    public Map<String, BufferedImage> convertToImages(PDPage p) throws IOException
    {
        page = p;
        final Map<String, BufferedImage> result = new HashMap<String, BufferedImage>();

        List<PDAnnotation> annotations = page.getAnnotations();
        for (PDAnnotation annotation: annotations)
        {
            String appearanceName = annotation.getAppearanceStream();
            PDAppearanceDictionary appearDictionary = annotation.getAppearance();
            if( appearDictionary != null )
            {
                if( appearanceName == null )
                {
                    appearanceName = "default";
                }
                Map<String, PDAppearanceStream> appearanceMap = appearDictionary.getNormalAppearance();
                if (appearanceMap != null) 
                {
                    PDAppearanceStream appearance = 
                        (PDAppearanceStream)appearanceMap.get( appearanceName ); 
                    if( appearance != null ) 
                    {
                        BufferedImage image = initializeGraphics(annotation);
                        setTextMatrix(null);
                        setTextLineMatrix(null);
                        getGraphicsStack().clear();
                        processSubStream( page, appearance.getResources(), appearance.getStream() );

                        String name = annotation.getAnnotationName();
                        if (name == null || name.length() == 0)
                        {
                            name = annotation.getDictionary().getString(COSName.T);
                            if (name == null || name.length() == 0)
                            {
                                name = Long.toHexString(annotation.hashCode());
                            }
                        }

                        result.put(name, image);
                    }
                }
            }
        }

        return result;
    }

    BufferedImage initializeGraphics(PDAnnotation annotation)
    {
        PDRectangle rect = annotation.getRectangle();
        float widthPt = rect.getWidth();
        float heightPt = rect.getHeight();
        float scaling = resolution / (float)DEFAULT_USER_SPACE_UNIT_DPI;
        int widthPx = Math.round(widthPt * scaling);
        int heightPx = Math.round(heightPt * scaling);
        //TODO The following reduces accuracy. It should really be a Dimension2D.Float.
        Dimension pageDimension = new Dimension( (int)widthPt, (int)heightPt );
        BufferedImage retval = new BufferedImage( widthPx, heightPx, imageType );
        Graphics2D graphics = (Graphics2D)retval.getGraphics();
        graphics.setBackground( TRANSPARENT_WHITE );
        graphics.clearRect( 0, 0, retval.getWidth(), retval.getHeight() );
        graphics.scale( scaling, scaling );
        setGraphics(graphics);
        pageSize = pageDimension;
        graphics.setRenderingHint( RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON );
        graphics.setRenderingHint( RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON );
        setGraphicsState(new PDGraphicsState(new PDRectangle(widthPt, heightPt)));

        return retval;
    }

    void setGraphics(Graphics2D graphics)
    {
        try {
            Field field = PageDrawer.class.getDeclaredField("graphics");
            field.setAccessible(true);
            field.set(this, graphics);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    private static final int DEFAULT_USER_SPACE_UNIT_DPI = 72;
    private static final Color TRANSPARENT_WHITE = new Color( 255, 255, 255, 0 );

    private int imageType;
    private int resolution;
}

如果您想要渲染给定PDPage page,的注释,您只需执行以下操作:

AnnotationDrawer drawer = new AnnotationDrawer(8, 288);
Map<String, BufferedImage> images = drawer.convertToImages(page);

构造函数参数对应于PDPage.convertToImage(int imageType, int resolution).

的参数

小心,这有

一个。基于PDFBox 1.8.2被黑客攻击;它可能包含特定于版本的代码; 湾仅检查了我在这里看到的一些可见的签名注释;它可能是不完整的,对于其他注释类型可能尤其失败。