查看原文
其他

SpringBoot实现电子文件签字+合同系统!

关注Java就该这么学带你全面认识Java
Java就该这么学后台回复 1024 有特别礼包

责编:Java就该这么学 | 来源:blog.csdn.net/weixin_44385486/article/details/126481493

上一篇精彩:SpringBoot 配置 HTTPS 安全证书的两种方案
大家好,我是Java就该这么学。

一、前言

今天公司领导提出一个功能,说实现一个文件的签字+盖章功能,然后自己进行了简单的学习,对文档进行数字签名与签署纸质文档的原因大致相同,数字签名通过使用计算机加密来验证 (身份验证:验证人员和产品所声明的身份是否属实的过程。例如,通过验证用于签名代码的数字签名来确认软件发行商的代码来源和完整性。)数字信息,如文档、电子邮件和宏。数字签名有助于确保:真实性,完整性,不可否认性。目前市面上的电子签章产品也是多样化,但是不管是哪个厂家的产品,在线签章简单易用,同时也能保证签章的有效性,防篡改,防伪造,稳定,可靠就是好产品。

此次开源的系统模拟演示了文件在OA系统中的流转,主要为办公系统跨平台在线处理Office文档提供了完美的解决方案。Word文档在线处理的核心环节,包括:起草文档、领导审批、核稿、领导盖章、正式发文。PageOffice产品支持PC端Word文档在线处理的所有环节;MobOffice产品支持了移动端领导审批和领导盖章的功能。支持PC端和移动端对文档审批和盖章的互认。然后此次博客中使用的卓正软件的电子签章采用自主知识产权的核心智能识别验证技术,确保文档安全可靠。采用 COM、ActiveX嵌入式技术开发,确保软件能够支持多种应用。遵循《中华人民共和国电子签名法》关于电子签名的规范,同时支持国际通用的 RSA算法,符合国家安全标准。

PageOffice和MobOffice产品结合使用为跨平台处理Office文件提供了完美的解决方案,主要功能有word在线编辑保存和留痕,word和pdf文件在线盖章(电子印章)。

二、项目源码及部署

1、项目结构及使用框架

该签字+盖章流程系统使用了SpringBoot+thymeleaf实现的,然后jar包依赖使用了maven

  • 控制层
@Controller@RequestMapping("/mobile")public class MobileOfficeController {    @Value("${docpath}")    private  String docPath;    @Value("${moblicpath}")    private  String moblicpath;    @Autowired    DocService m_docService;    /**     * 添加MobOffice的服务器端授权程序Servlet(必须)     *     */    @RequestMapping("/opendoc")    public void opendoc(HttpServletRequest request, HttpServletResponse response, HttpSession session,String type,String userName)throws  Exception {        String fileName = "";        userName= URLDecoder.decode(userName,"utf-8");        Doc doc=m_docService.getDocById(1);        if(type.equals("word")){            fileName = doc.getDocName();        }else{            fileName = doc.getPdfName();        }        OpenModeType openModeType = OpenModeType.docNormalEdit;        if (fileName.endsWith(".doc")) {            openModeType = OpenModeType.docNormalEdit;        } else if (fileName.endsWith(".pdf")) {            String mode = request.getParameter("mode");            if (mode.equals("normal")) {                openModeType = OpenModeType.pdfNormal;            } else {                openModeType = OpenModeType.pdfReadOnly;            }        }        MobOfficeCtrl mobCtrl = new MobOfficeCtrl(request,response);        mobCtrl.setSysPath(moblicpath);        mobCtrl.setServerPage("/mobserver.zz");        //mobCtrl.setZoomSealServer("http://xxx.xxx.xxx.xxx:8080/ZoomSealEnt/enserver.zz");        mobCtrl.setSaveFilePage("/mobile/savedoc?testid="+Math.random());        mobCtrl.webOpen("file://"+docPath+fileName,  openModeType , userName);    }    @RequestMapping("/savedoc")    public  void  savedoc(HttpServletRequest request,  HttpServletResponse response){        FileSaver fs = new FileSaver(request, response);        fs.saveToFile(docPath+fs.getFileName());        fs.close();    }}
  • 项目业务层源码
@Servicepublic class DocServiceImpl implements DocService {    @Autowired    DocMapper docMapper;    @Override    public Doc getDocById(int id) throws Exception {        Doc  doc=docMapper.getDocById(id);        //如果doc为null的话,页面所有doc.属性都报错        if(doc==null) {            doc=new Doc();        }        return doc;    }    @Override    public Integer addDoc(Doc doc) throws Exception {       int id=docMapper.addDoc(doc);        return id;    }    @Override    public Integer updateStatusForDocById(Doc doc) throws Exception {        int id=docMapper.updateStatusForDocById(doc);        return id;    }    @Override    public Integer updateDocNameForDocById(Doc doc) throws Exception {        int id=docMapper.updateDocNameForDocById(doc);        return id;    }    @Override    public Integer updatePdfNameForDocById(Doc doc) throws Exception {        int id=docMapper.updatePdfNameForDocById(doc);        return id;    }}
  • 拷贝文件
public class CopyFileUtil {  //拷贝文件  public static boolean copyFile(String oldPath, String newPath) throws Exception {      boolean copyStatus=false;      int bytesum = 0;      int byteread = 0;      File oldfile = new File(oldPath);      if (oldfile.exists()) { //文件存在时          InputStream inStream = new FileInputStream(oldPath); //读入原文件          FileOutputStream fs = new FileOutputStream(newPath);          byte[] buffer = new byte[1444];          int length;          while ((byteread = inStream.read(buffer)) != -1) {              bytesum += byteread; //字节数 文件大小              //System.out.println(bytesum);              fs.write(buffer, 0, byteread);          }          fs.close();          inStream.close();          copyStatus=true;      }else{          copyStatus=false;      }      return copyStatus;  }}
  • 二维码源码
public class QRCodeUtil {    private String codeText;//二维码内容    private BarcodeFormat barcodeFormat;//二维码类型    private int width;//图片宽度    private int height;//图片高度    private String imageformat;//图片格式    private int backColorRGB;//背景色,颜色RGB的数值既可以用十进制表示,也可以用十六进制表示    private int codeColorRGB;//二维码颜色    private ErrorCorrectionLevel errorCorrectionLevel;//二维码纠错能力    private String encodeType;    public QRCodeUtil() {        codeText = "www.zhuozhengsoft.com";        barcodeFormat = BarcodeFormat.PDF_417;        width = 400;        height = 400;        imageformat = "png";        backColorRGB = 0xFFFFFFFF;        codeColorRGB = 0xFF000000;        errorCorrectionLevel = ErrorCorrectionLevel.H;        encodeType = "UTF-8";    }    public QRCodeUtil(String text) {        codeText = text;        barcodeFormat = BarcodeFormat.PDF_417;        width = 400;        height = 400;        imageformat = "png";        backColorRGB = 0xFFFFFFFF;        codeColorRGB = 0xFF000000;        errorCorrectionLevel = ErrorCorrectionLevel.H;        encodeType = "UTF-8";    }    public String getCodeText() {        return codeText;    }    public void setCodeText(String codeText) {        this.codeText = codeText;    }    public BarcodeFormat getBarcodeFormat() {        return barcodeFormat;    }    public void setBarcodeFormat(BarcodeFormat barcodeFormat) {        this.barcodeFormat = barcodeFormat;    }    public int getWidth() {        return width;    }    public void setWidth(int width) {        this.width = width;    }    public int getHeight() {        return height;    }    public void setHeight(int height) {        this.height = height;    }    public String getImageformat() {        return imageformat;    }    public void setImageformat(String imageformat) {        this.imageformat = imageformat;    }    public int getBackColorRGB() {        return backColorRGB;    }    public void setBackColorRGB(int backColorRGB) {        this.backColorRGB = backColorRGB;    }    public int getCodeColorRGB() {        return codeColorRGB;    }    public void setCodeColorRGB(int codeColorRGB) {        this.codeColorRGB = codeColorRGB;    }    public ErrorCorrectionLevel getErrorCorrectionLevel() {        return errorCorrectionLevel;    }    public void setErrorCorrectionLevel(ErrorCorrectionLevel errorCorrectionLevel) {        this.errorCorrectionLevel = errorCorrectionLevel;    }    private BufferedImage toBufferedImage(BitMatrix bitMatrix) {        int width = bitMatrix.getWidth();        int height = bitMatrix.getHeight();        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);        for (int x = 0; x < width; x++) {            for (int y = 0; y < height; y++) {                image.setRGB(x, y, bitMatrix.get(x, y) ? this.codeColorRGB: this.backColorRGB);            }        }        return image;    }    private byte[] writeToBytes(BitMatrix bitMatrix)            throws IOException {        try {            BufferedImage bufferedimage = toBufferedImage(bitMatrix);            //将图片保存到临时路径中            File file = java.io.File.createTempFile("~pic","."this.imageformat);            //System.out.println("临时图片路径:"+file.getPath());            ImageIO.write(bufferedimage,this.imageformat,file);            //获取图片转换成的二进制数组            FileInputStream fis = new FileInputStream(file);            int fileSize = fis.available();            byte[] imageBytes = new byte[fileSize];            fis.read(imageBytes);            fis.close();            //删除临时文件            if (file.exists()) {                file.delete();            }            return imageBytes;        } catch (Exception e) {            System.out.println(" Image err :" + e.getMessage());            return null;        }    }    //获取二维码图片的字节数组    public byte[] getQRCodeBytes()            throws IOException {        try {            MultiFormatWriter multiFormatWriter = new MultiFormatWriter();            //设置二维码参数            Map hints = new HashMap();            if (this.errorCorrectionLevel != null) {                //设置二维码的纠错级别                hints.put(EncodeHintType.ERROR_CORRECTION, this.errorCorrectionLevel);            }            if (this.encodeType!=null && this.encodeType.trim().length() > 0) {                //设置编码方式                hints.put(EncodeHintType.CHARACTER_SET, this.encodeType);            }            BitMatrix bitMatrix = multiFormatWriter.encode(this.codeText, BarcodeFormat.QR_CODE, this.width, this.height, hints);            byte[] bytes = writeToBytes(bitMatrix);            return bytes;        } catch (Exception e) {            e.printStackTrace();            return null;        }    }}

2、项目下载及部署

  • 下载项目源码后,使用idea导入slndemo项目并运行
  • 将项目slndemo下的slndemodata.zip压缩包拷贝到本地D盘根目录下并解压
  • 点击启动项目

三、功能展示

1、项目启动后登录首页

  • 项目地址:http://localhost:8888/pc/login
  • 账户:张三 密码:123456

2、系统首页功能简介

这是一个简单的Demo项目,模拟Word文件在办公系统中的主要流转环节,并不意味着PageOffice产品只能支持这样的文档处理流程。PageOffice产品只提供文档在线处理的功能,包括:打开、编辑、保存、动态填充、文档合并、套红、留痕、盖章等上百项功能(详细请参考PageOffice产品开发包中的示例),不提供流程控制功能,所以不管开发什么样的Web系统,只要是需要在线处理Office文档,都可以根据自己的项目需要,调用PageOffice产品相应的功能即可。「注意:为了简化代码逻辑,此演示程序只能创建一个文档进行流转。」

3、点击起草文档

  • 点击起草文档,点击提交
  • 我们使用了PageOffice企业版,必须要注册序列化

  • 版 本:PageOffice企业版5(试用)

  • 序列号:35N8V-2YUC-LY77-W14XL

  • 当我们注册成功以后,就可以编辑发布的文件或者公告了
  • 编辑好以后点击保存
  • 点击审批

4、审批

  • 登录李总审批
  • 退出系统,然后输入李总
  • 然后点击批阅,下一步
  • 登录赵六进行审核稿子

5、审稿

  • 审稿
  • 审核然后到盖章环节
  • 使用王总登录进行盖章

6、盖章和签字的实现

  • 王总登录
  • 点击盖章
  • 点击加盖印章
  • 我们盖章前需要输入姓名+密码,需要输入错误报错
  • 正确的账户密码是:

  • 账户:王五

  • 密码:123456

  • 登录成功后有选择王五的个人章进行签字
  • 另外,搜索公众号顶级科技后台回复“API接口”,获取一份惊喜礼包。
  • 签字成功
  • 公司盖章,重复以上步骤
  • 签字盖章成功

7、完整签字盖章文件

  • 保存之后发布文件
  • 公司文件展示
  • 盖章签字后的文件
欢迎有需要的同学试试,如果本文对您有帮助,也请帮忙点个 赞 + 在看 啦!❤️

在 GitHub猿 还有更多优质项目系统学习资源,欢迎分享给其他同学吧!



公众号后台回复 Java 或者 面试 有惊喜礼包!Java就该这么学交流群

 「Java就该这么学」建立了读者Java交流群,大家可以添加小编微信进行加群。欢迎有想法、乐于分享的朋友们一起交流学习。

扫描添加好友邀你进Java群,加我时注明姓名+公司+职位】


版权申明:内容来源网络,版权归原作者所有。如有侵权烦请告知,我们会立即删除并表示歉意。谢谢。

往日文章:

想了解Java后端学习路线?看完这篇就够了!
2天3夜开发了一个商城系统,用它挣了10.7万!
别再用 BeanUtils 了,这款 PO VO DTO 转换神器不香么?
使用Spring AOP实现异步文件上传
CTO 说了:谁再用 Redis 过期监听实现定时任务,立马滚蛋!
SpringBoot蓝天幼儿园管理系统
淦,为什么 "𠮷𠮷𠮷" .length !== 3 ??
一个网站部署的完整流程(包教包会)
别再写 main 方法测试了,太 Low!这才是专业 Java 测试方法!
Spring中毒太深,离开Spring我连最基本的CRUD都不会写了...
Api 接口优化的几个技巧
RedisJson 横空出世,比 ES 快7 倍,惊爆了!

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存