`

无纸化办公中手写签名的实现

    博客分类:
  • java
 
阅读更多

 

  最近要实现一个使用手写板对PDF文档进行手写签名的功能,下面记录一下关键过程。

 

  实现的思路如下:

1、在画板中显示PDF文档的内容

2、用户使用签名版对PDF文档进行签名

3、保存签名后的画板内容到新的PDF中

 

 其中使用pdfbox把PDF文档转成图片,签名后又把签名后的图片转回PDF保存。

  pdfbox(http://pdfbox.apache.org/)是一个处理PDF文档的JAVA库。

 

Java代码  收藏代码
  1.     import org.apache.pdfbox.pdmodel.PDDocument;  
  2. import org.apache.pdfbox.pdmodel.PDPage;  
  3. import org.apache.pdfbox.pdmodel.edit.PDPageContentStream;  
  4. import org.apache.pdfbox.pdmodel.graphics.xobject.PDJpeg;  
  5. import org.apache.pdfbox.pdmodel.graphics.xobject.PDXObjectImage;  
  6.   
  7. import javax.imageio.ImageIO;  
  8. import javax.swing.*;  
  9. import java.awt.*;  
  10. import java.awt.event.MouseEvent;  
  11. import java.awt.event.MouseListener;  
  12. import java.awt.event.MouseMotionListener;  
  13. import java.awt.geom.AffineTransform;  
  14. import java.awt.image.AffineTransformOp;  
  15. import java.awt.image.BufferedImage;  
  16. import java.awt.image.ColorModel;  
  17. import java.awt.image.PixelGrabber;  
  18. import java.io.File;  
  19. import java.io.IOException;  
  20. import java.util.logging.Level;  
  21. import java.util.logging.Logger;  
  22.   
  23. class TestDraw extends Canvas {  
  24.     Image image;  
  25.     boolean init = true;  
  26.   
  27.     public static final int WIDTH = 1024;  
  28.   
  29.     public static final int HEIGHT = 1024;  
  30.   
  31.     public Image screen = createImage(WIDTH, HEIGHT, true);// 双缓冲  
  32.   
  33.     private Graphics graphics = screen.getGraphics();  
  34.   
  35.     public Image resultImage;  
  36.   
  37.     int x1 = -1;  
  38.     int y1 = -1;  
  39.     int x2 = -1;  
  40.     int y2 = -1;  
  41.   
  42.     public TestDraw() {  
  43.         this.setBackground(Color.white);  
  44.         this.addMouseListener(new MyMouseListener(this));  
  45.         this.addMouseMotionListener(new MyMouseMotionListener(this));  
  46.   
  47.         image = pdfToImg("pdf/tests.pdf");  
  48.   
  49.         // 设定初始构造时面板大小  
  50.         setPreferredSize(new Dimension(WIDTH, HEIGHT));  
  51.         // 初始导入一张图片  
  52.         ImageIcon icon = new ImageIcon(image);  
  53.         resultImage = icon.getImage();  
  54.   
  55.     }  
  56.   
  57.     public void paint(Graphics g) {  
  58.         update(g);  
  59.     }  
  60.   
  61.     @Override  
  62.     public void update(Graphics g) {  
  63.         BufferedImage bufferedImage= imageToBufferedImage(resultImage);  
  64.   
  65.         // Copy image to buffered image  
  66.         Graphics g2 = bufferedImage.getGraphics();  
  67.         g2.setColor(Color.black);  
  68.         if (x1 == -1 || y1 == -1) {  
  69.             x1 = x2;  
  70.             y1 = y2;  
  71.         }  
  72.         g2.drawLine(x1, y1, x2, y2);  
  73.         x1 = x2;  
  74.         y1 = y2;  
  75.         g2.dispose();  
  76.   
  77.         ImageIcon icon = new ImageIcon(bufferedImage);  
  78.         resultImage = icon.getImage();  
  79.   
  80.         graphics.drawImage(resultImage, 00this);  
  81.         g.drawImage(screen, 00null);// 最后个参数一定要用null,这样可以防止drawImage调用update方法  
  82.         g.dispose();  
  83.     }  
  84.   
  85.   
  86.     public static BufferedImage imageToBufferedImage(Image im) {  
  87.        BufferedImage bi = new BufferedImage  
  88.           (im.getWidth(null),im.getHeight(null),BufferedImage.TYPE_INT_RGB);  
  89.        Graphics bg = bi.getGraphics();  
  90.        bg.drawImage(im, 00null);  
  91.        bg.dispose();  
  92.        return bi;  
  93.     }  
  94.   
  95.     /** 
  96.      * 生成一个BufferImage 
  97.      * 生成一个BufferImage BufferImage是Image的子类,左上角坐标都为 (0, 0) 
  98.      * 第三个参数是代码Image图形类型,分为14种,以位数又分为1,2或4位 
  99.      * 
  100.      * @param width 
  101.      * @param height 
  102.      * @param flag 
  103.      * @return 
  104.      */  
  105.     final static public BufferedImage createImage(int width, int height,  
  106.                                                   boolean flag) {  
  107.         if (flag) {  
  108.             return new BufferedImage(width, height, 2);  
  109.         } else {  
  110.             return new BufferedImage(width, height, 1);  
  111.         }  
  112.     }  
  113.   
  114.     /** 
  115.      * 把PDF转成图片 
  116.      * @param pdfFileName PDF文件路径 
  117.      * @return 
  118.      */  
  119.     public static Image pdfToImg(String pdfFileName) {  
  120.         ImageIcon imageIcon = null;  
  121.         try {  
  122.             PDDocument doc = PDDocument.load(pdfFileName);  
  123.             int pageCount = doc.getNumberOfPages();  
  124.             PDPage page = (PDPage) doc.getDocumentCatalog().getAllPages().get(0);  
  125.             BufferedImage bufferedImage = page.convertToImage();  
  126.             if (bufferedImage != null) {  
  127.                 imageIcon = new ImageIcon(Toolkit.getDefaultToolkit().createImage(bufferedImage.getSource()));  
  128.             }  
  129.             doc.close();  
  130.         } catch (IOException ex) {  
  131.             Logger.getLogger(ShowJFrame.class.getName()).log(Level.SEVERE, null, ex);  
  132.         }  
  133.         return imageIcon.getImage();  
  134.     }  
  135. }  
  136.   
  137. class MyMouseListener implements MouseListener {  
  138.     TestDraw draw;  
  139.   
  140.     public MyMouseListener(TestDraw draw) {  
  141.         this.draw = draw;  
  142.     }  
  143.   
  144.     /** 
  145.      * 点击鼠标时保存新的PDF文件 
  146.      * @param e 
  147.      */  
  148.     public void mouseClicked(MouseEvent e) {  
  149.         System.out.println("mouseClicked");  
  150.   
  151.         PDDocument doc = null;  
  152.         PDPage page = null;  
  153.   
  154.         try {  
  155.             doc = new PDDocument();  
  156.             page = new PDPage();  
  157.   
  158.             BufferedImage bufferedImage = toBufferedImage(draw.resultImage);  
  159.             File outputfile = new File("saved.png");  
  160.                ImageIO.write(bufferedImage, "png", outputfile);  
  161.   
  162.             int w = bufferedImage.getWidth();  
  163.             int h = bufferedImage.getHeight();  
  164.             BufferedImage after = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);  
  165.             AffineTransform at = new AffineTransform();  
  166.             at.scale(0.50.5);  
  167.             AffineTransformOp scaleOp =  
  168.                new AffineTransformOp(at, AffineTransformOp.TYPE_BILINEAR);  
  169.             after = scaleOp.filter(bufferedImage, after);  
  170.   
  171.             doc.addPage(page);  
  172.             PDXObjectImage ximage = new PDJpeg(doc, after);  
  173.             PDPageContentStream content = new PDPageContentStream(doc, page, truetrue);  
  174. //            content.drawImage(ximage, 0, 0);  
  175.             content.drawXObject(ximage, 00800800);  
  176.             content.close();  
  177.             doc.save("save.pdf");  
  178.             doc.close();  
  179.         } catch (Exception ex) {  
  180.             System.out.println(ex);  
  181.         }  
  182.     }  
  183.   
  184.     // This method returns a buffered image with the contents of an image  
  185.     public static BufferedImage toBufferedImage(Image image) {  
  186.         if (image instanceof BufferedImage) {  
  187.             return (BufferedImage) image;  
  188.         }  
  189.   
  190.         // This code ensures that all the pixels in the image are loaded  
  191.         image = new ImageIcon(image).getImage();  
  192.   
  193.         // Determine if the image has transparent pixels; for this method's  
  194.         // implementation, see e661 Determining If an Image Has Transparent Pixels  
  195.         boolean hasAlpha = hasAlpha(image);  
  196.   
  197.         // Create a buffered image with a format that's compatible with the screen  
  198.         BufferedImage bimage = null;  
  199.         GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();  
  200.         try {  
  201.             // Determine the type of transparency of the new buffered image  
  202.             int transparency = Transparency.OPAQUE;  
  203.             if (hasAlpha) {  
  204.                 transparency = Transparency.BITMASK;  
  205.             }  
  206.   
  207.             // Create the buffered image  
  208.             GraphicsDevice gs = ge.getDefaultScreenDevice();  
  209.             GraphicsConfiguration gc = gs.getDefaultConfiguration();  
  210.             bimage = gc.createCompatibleImage(  
  211.                 image.getWidth(null), image.getHeight(null), transparency);  
  212.         } catch (HeadlessException e) {  
  213.             // The system does not have a screen  
  214.         }  
  215.   
  216.         if (bimage == null) {  
  217.             // Create a buffered image using the default color model  
  218.             int type = BufferedImage.TYPE_INT_RGB;  
  219.             if (hasAlpha) {  
  220.                 type = BufferedImage.TYPE_INT_ARGB;  
  221.             }  
  222.             bimage = new BufferedImage(image.getWidth(null), image.getHeight(null), type);  
  223.         }  
  224.   
  225.         // Copy image to buffered image  
  226.         Graphics g = bimage.createGraphics();  
  227.   
  228.         // Paint the image onto the buffered image  
  229.         g.drawImage(image, 00null);  
  230.         g.dispose();  
  231.   
  232.         return bimage;  
  233.     }  
  234.   
  235.     //Determining If an Image Has Transparent Pixels  
  236.   
  237.     // This method returns true if the specified image has transparent pixels  
  238.     public static boolean hasAlpha(Image image) {  
  239.         // If buffered image, the color model is readily available  
  240.         if (image instanceof BufferedImage) {  
  241.             BufferedImage bimage = (BufferedImage) image;  
  242.             return bimage.getColorModel().hasAlpha();  
  243.         }  
  244.   
  245.         // Use a pixel grabber to retrieve the image's color model;  
  246.         // grabbing a single pixel is usually sufficient  
  247.         PixelGrabber pg = new PixelGrabber(image, 0011false);  
  248.         try {  
  249.             pg.grabPixels();  
  250.         } catch (InterruptedException e) {  
  251.         }  
  252.   
  253.         // Get the image's color model  
  254.         ColorModel cm = pg.getColorModel();  
  255.         return cm.hasAlpha();  
  256.     }  
  257.   
  258.     public void mouseEntered(MouseEvent e) {  
  259.         //System.out.println("mouseEntered");  
  260.     }  
  261.   
  262.     public void mouseExited(MouseEvent e) {  
  263.         //System.out.println("mouseExited");  
  264.     }  
  265.   
  266.     public void mousePressed(MouseEvent e) {  
  267.         //System.out.println("mousePressed");  
  268.         draw.x1 = -1;  
  269.         draw.y1 = -1;  
  270.     }  
  271.   
  272.     public void mouseReleased(MouseEvent e) {  
  273.         //System.out.println("mouseReleased");  
  274.         draw.x1 = draw.x2;  
  275.         draw.y1 = draw.y2;  
  276.     }  
  277. }  
  278.   
  279. class MyMouseMotionListener implements MouseMotionListener {  
  280.     TestDraw draw;  
  281.   
  282.     public MyMouseMotionListener(TestDraw draw) {  
  283.         this.draw = draw;  
  284.     }  
  285.   
  286.     public void mouseDragged(MouseEvent e) {  
  287.         //System.out.println("mouseDragged");  
  288.         draw.x2 = e.getX();  
  289.         draw.y2 = e.getY();  
  290.         draw.repaint();  
  291.     }  
  292.   
  293.     public void mouseMoved(MouseEvent e) {  
  294.         //System.out.println("mouseMoved");  
  295.     }  
  296. }  
分享到:
评论

相关推荐

    js实现的带笔锋手写签名

    在IT行业中,手写签名是一种常见的交互方式,特别是在电子文档签署、在线交易验证等领域。...通过这样的技术,用户可以在数字环境中获得类似纸笔签名的体验,这对于现代无纸化办公和电子商务环境具有重要意义。

    手写签名,可保存为图片或到数据库

    在IT行业中,手写签名是一种重要的身份验证和文档认可方式,尤其在电子文档和无纸化办公环境中。本文将深入探讨“手写签名,可保存为图片或到数据库”这一技术概念及其应用。 首先,手写签名是个人身份的直观表示,...

    前端实现canvas手机端手写签名

    "前端实现canvas手机端手写签名"这一技术就是为了解决这个问题,它允许用户在手机等移动设备上通过触摸屏幕实现手写签名,为无纸化办公提供了便利。本文将深入探讨如何使用HTML5的Canvas API来构建一个手机端的手写...

    HTML5 提交手写签名到服务器端保存

    在现代Web开发中,HTML5提供了许多增强的功能,其中之一就是支持手写签名并将其提交到服务器端进行存储。这个功能通常用于需要...这一技术在电子签名、合同签署等领域有着广泛的应用,极大地提高了无纸化办公的效率。

    WPF手写签名

    执法终端通过蓝牙或Wi-Fi连接打印机,可即时打印签名确认单,或者将电子签名通过网络传输至后台系统,实现无纸化办公。 总结,WPF手写签名功能的实现依赖于Windows Ink技术,通过InkCanvas控件和事件处理,我们可以...

    基于mui的和html5电子签名

    例如,电商平台的订单确认、银行的远程开户、无纸化办公系统的文件审批等,都离不开电子签名的功能。 总的来说,基于HTML5和Mui的电子签名方案,不仅降低了开发成本,还提升了用户体验,是现代Web应用中不可或缺的...

    OA无纸化办公系统电子签章插件

    OA无纸化办公系统电子签章插件是现代企业信息化管理中的重要组成部分,它旨在提升工作效率,确保文档安全,减少纸质文件的使用,实现环保的办公环境。在数字化办公时代,电子签章作为合法有效的签名形式,对于合同、...

    web 手写面板签名插件

    这种技术为无纸化办公、在线合同签署、电子商务等领域提供了便利,减少了实体签名的必要性,同时也提升了用户体验,因为用户无需打印、签字、扫描或传真文件。 【标签】: 1. HTML:HTML(超文本标记语言)是构建...

    签名板demo_delphi_汉王签名板_汉王电子签名_DEMO_clubfai_

    汉王科技是一家知名的电子签名设备和解决方案提供商,其签名板产品广泛应用于银行、保险、电信等行业,提供无纸化办公的解决方案。 描述中提到的“汉王签名板,调用汉王签批控件接口dll”,揭示了这个DEMO程序的...

    行业分类-设备装置-加密签名手写装置及方法.zip

    政府机构也可以用它来进行无纸化办公,提高效率的同时保证信息安全。 总之,加密签名手写装置及方法是现代信息安全体系中不可或缺的一部分,它提供了安全、便捷的电子签名解决方案,促进了数字化时代的交易和沟通。...

    Sanasa:无纸化办公

    Sanasa是一款致力于实现无纸化办公的解决方案,它旨在通过数字化技术提高工作效率,减少对纸质文档的依赖,同时促进环保。无纸化办公是当前信息化社会中一个重要的发展趋势,尤其在企业、政府机构以及教育等领域得到...

    电子签名(仿毛笔字)

    在IT行业中,电子签名是一种重要的技术,特别是在数字化和无纸化办公的趋势下,它扮演着不可或缺的角色。本文将深入探讨“电子签名(仿毛笔字)”这一主题,结合使用ActionScript 3(AS3)实现的系统,来阐述相关的...

    表单手写签章控件正式版

    在IT领域,表单手写签章控件是一种常见的软件组件,主要应用于电子文档和在线表单中,允许用户以手写方式添加签名,从而实现无纸化办公和远程签署。这种控件通常具备高度的交互性和灵活性,可以集成到各种应用程序中...

    微信小程序实现手写签字

    无纸化办公,这是老板对我的要求,然而有人现场执法文件全部电子化,只有签字部分让一个搞web的人有点儿头疼,不能为了这个找个人来开发app吧于是想到了小程序,对于一个新接触小程序的人来说还是有挑战性的,因为我...

    H5手写签字

    "H5手写签字"是指利用HTML5技术实现的在线手写签名功能,允许用户通过触控设备或者鼠标模拟手写,进行电子签名,这种功能在电子商务、法律文档签署、远程办公等场景中有着广泛的应用。 一、HTML5 Canvas API 手写...

    行业-电子政务-电子装置和手写文档显示方法.zip

    随着科技的进步,无纸化办公和数字化签名已经成为趋势,电子装置上的手写文档处理技术也随之发展。这篇文档将深入探讨这一领域的关键知识点。 一、电子政务与无纸化办公 电子政务,简称e-Government,是指政府利用...

    医院信息化建设使用电子签名的必要性.rar

    纸质文件的存储、运输和管理都需要投入大量的人力物力,而电子签名的应用可以实现无纸化办公,减少纸质文件的使用,节约资源,同时也减少了与纸质文件相关的管理费用。 此外,电子签名还能促进跨机构协作。在医联体...

    通达手写签章控件{websign 4.0.4.2 正式版}

    这个控件适用于各种基于Web的应用场景,如企业办公自动化系统、电子合同平台等,大大提高了无纸化办公的效率和安全性。 在Websign 4.0.4.2版本中,开发者可能已经优化了签名的稳定性和用户体验。手写签章功能让用户...

    H5手机签字 canvas_sign 手机电子签字 电子签名

    在现代信息化社会,电子签名已经成为了商业活动中不可或缺的一部分,特别是在远程协作和无纸化办公的背景下。本技术分享将深入探讨如何使用HTML5中的Canvas API实现手机电子签字功能,即canvas_sign,它允许用户在...

    WebSign 电子手写签章系统 注册版

    该系统提供了手写签章功能,让用户可以如同在纸上一样自由地在电子文档上进行签名,大大提高了工作效率,同时也符合了现代信息化社会对无纸化办公的需求。 "完美注册版"通常指的是该软件已经成功激活,用户可以...

Global site tag (gtag.js) - Google Analytics