package com.hyzs.szcg.doc.utils; import java.awt.image.BufferedImage; import java.io.ByteArrayInputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.HttpURLConnection; import java.net.URL; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.*; import java.util.List; import java.util.regex.Matcher; import java.util.regex.Pattern; import com.hyzs.gz.common.core.exception.CommonException; import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang.StringUtils; import org.apache.poi.openxml4j.exceptions.InvalidFormatException; import org.apache.poi.util.Units; import org.apache.poi.xwpf.usermodel.*; import sun.misc.BASE64Decoder; import javax.imageio.ImageIO; /** * * @Title: WordExport.java * @Description: 导出word文档 * @author zzh * @version 1.0 */ @Slf4j public class PoiExportWord { private String templatePath; private XWPFDocument doc = null; private FileInputStream is = null; private OutputStream os = null; public PoiExportWord(String templatePath) { this.templatePath = templatePath; } public void init() throws IOException { File file=new File(this.templatePath); if(file.exists()){ is = new FileInputStream(file); doc = new XWPFDocument(is); }else{ log.error("io error -> 文件模板不存在》"+this.templatePath); throw CommonException.exception("io error 模板文件不存在"); } } /** * 替换掉占位符 * @param params * @return * @throws Exception */ public boolean export(Mapparams) throws Exception{ this.replaceInPara(doc, params); return true; } /** * 替换掉表格中的占位符 * @param params * @param tableIndex * @return * @throws Exception */ public boolean export(Map params,int tableIndex) throws Exception{ this.replaceInTable(doc, params,tableIndex); return true; } /** * 生成word文档 * @param outDocPath * @return * @throws IOException */ public boolean generate(String outDocPath) throws IOException{ os = new FileOutputStream(outDocPath); doc.write(os); this.close(os); this.close(is); return true; } /** * 替换表格里面的变量 * * @param doc * 要替换的文档 * @param params * 参数 * @throws Exception */ private void replaceInTable(XWPFDocument doc, Map params,int tableIndex) throws Exception { List tableList = doc.getTables(); if(tableList.size()>tableIndex){ for (int i=0;i rows; List cells; List paras; rows = table.getRows(); for (XWPFTableRow row : rows) { cells = row.getTableCells(); for (XWPFTableCell cell : cells) { paras = cell.getParagraphs(); for (XWPFParagraph para : paras) { this.replaceInPara(para, params); } } } } } } /** * 替换段落里面的变量 * * @param doc * 要替换的文档 * @param params * 参数 * @throws Exception */ private void replaceInPara(XWPFDocument doc, Map params) throws Exception { Iterator iterator = doc.getParagraphsIterator(); XWPFParagraph para; while (iterator.hasNext()) { para = iterator.next(); this.replaceInPara(para, params); } } /** * 替换段落里面的变量 * * @param para * 要替换的段落 * @param params * 参数 * @throws Exception * @throws IOException * @throws */ private boolean replaceInPara(XWPFParagraph para, Map params) throws Exception { boolean data = false; List runs; //有符合条件的占位符 if (this.matcher(para.getParagraphText()).find()) { runs = para.getRuns(); data = true; Map tempMap = new TreeMap<>(); for (int i = 0; i < runs.size(); i++) { XWPFRun run = runs.get(i); String runText = run.toString(); //以"$"开头 boolean begin = runText.indexOf("$")>-1; boolean end = runText.indexOf("}")>-1; if(begin && end){ tempMap.put(i, runText); fillBlock(para, params, tempMap, i); continue; }else if(begin && !end){ tempMap.put(i, runText); continue; }else if(!begin && end){ tempMap.put(i, runText); fillBlock(para, params, tempMap, i); continue; }else{ if(tempMap.size()>0){ tempMap.put(i, runText); continue; } continue; } } } else if (this.matcherRow(para.getParagraphText())) { runs = para.getRuns(); data = true; } return data; } /** * 填充run内容 * @param para * @param params * @param tempMap * @param * @param * @throws * @throws IOException * @throws Exception */ @SuppressWarnings("rawtypes") private void fillBlock(XWPFParagraph para, Map params, Map tempMap, int index) throws InvalidFormatException, IOException, Exception { Matcher matcher; if(tempMap!=null&&tempMap.size()>0){ String wholeText = ""; List tempIndexList = new ArrayList (); for(Map.Entry entry :tempMap.entrySet()){ tempIndexList.add(entry.getKey()); wholeText+=entry.getValue(); } if(wholeText.equals("")){ return; } matcher = this.matcher(wholeText); if (matcher.find()) { boolean isPic = false;//图片标识 boolean isBasePic = false;//base64图片标识 boolean isUnderLine=false;//下划线标识 boolean checkbox=false;//复选框标识 ByteArrayInputStream inputStream=null; int width = 0; int height = 0; int picType = 0; String path = null; String keyText = matcher.group().substring(2,matcher.group().length()-1); String newRunText = ""; Object value = params.get(keyText); if(value instanceof String){ newRunText = matcher.replaceFirst(String.valueOf(value)); }else if(value instanceof Map){ Map pic = (Map)value; if(pic.containsKey("fileSuffix")&&pic.containsKey("filePath")){ //插入图片 isPic = true; path = pic.get("filePath").toString(); //获取图片大小 BufferedImage sourceImg = ImageIO.read(getPicStream(path)); width=sourceImg.getWidth(); height=sourceImg.getHeight(); if(width>200){ width=200; } if(height>200){ height=200; } if(keyText.endsWith("Sign")||keyText.endsWith("_sign")){//签名限制大小 width=50; height=30; } picType=sourceImg.getType(); }else if(pic.containsKey("personalSignature")){ // base64字符串的 图片形式 isBasePic=true; //将字符串转换为byte数组 String file= (String) pic.get("personalSignature"); file=file.replaceFirst("data:image/png;base64,",""); byte[] bytes = new BASE64Decoder().decodeBuffer(file.trim()); //转化为输入流 inputStream = new ByteArrayInputStream(bytes); //获取图片大小 BufferedImage sourceImg = ImageIO.read(new ByteArrayInputStream(bytes)); width = sourceImg.getWidth(); height = sourceImg.getHeight(); if(width>200){ width=200; } if(height>200){ height=200; } if(keyText.endsWith("Sign")||keyText.endsWith("_sign")){//签名限制大小 width=50; height=30; } picType=sourceImg.getType(); }else if(pic.containsKey("name")&&pic.containsKey("address")&&pic.containsKey("latitude")){ //是一个 地址 newRunText = matcher.replaceFirst(String.valueOf(pic.get("name"))); } }else if(value instanceof List){ //在表单中为下拉选择 在模板中为文本框 List list=(List)value; if(list.size()>0){ if(list.get(0) instanceof String){ String listStr=String.join(",",list); newRunText = matcher.replaceFirst(String.valueOf(listStr)); }else if(list.get(0) instanceof Map){ Map m=(Map) list.get(0); if(m.containsKey("filePath")&&m.containsKey("fileSuffix")&&m.containsKey("fileName")) { //是图片 //插入图片 isPic = true; path = m.get("filePath").toString(); //获取图片大小 BufferedImage sourceImg = ImageIO.read(getPicStream(path)); width = sourceImg.getWidth(); height = sourceImg.getHeight(); if(width>200){ width=200; } if(height>200){ height=200; } if(keyText.endsWith("Sign")||keyText.endsWith("_sign")){//签名限制大小 width=50; height=30; } picType=sourceImg.getType(); }else if(m.containsKey("personalSignature")){ // base64字符串的 图片形式 isBasePic=true; //将字符串转换为byte数组 String file= (String) m.get("personalSignature"); byte[] bytes = new BASE64Decoder().decodeBuffer(file.trim()); //转化为输入流 inputStream = new ByteArrayInputStream(bytes); //获取图片大小 BufferedImage sourceImg = ImageIO.read(new ByteArrayInputStream(bytes)); width = sourceImg.getWidth(); height = sourceImg.getHeight(); if(width>200){ width=200; } if(height>200){ height=200; } if(keyText.endsWith("Sign")||keyText.endsWith("_sign")){//签名限制大小 width=50; height=30; } picType=sourceImg.getType(); }else if(m.containsKey("q")&&m.containsKey("a")){ //问答形式 StringBuffer str=new StringBuffer(); for (int i=0;i 0){ isUnderLine=true; newRunText = matcher.replaceFirst(str.toString()); } }else{ StringBuffer listStr=new StringBuffer(); for (int i = 0; i 0){ newRunText = matcher.replaceFirst(listStr.toString().substring(0,listStr.length()-1)); } } } } }else if(value == null){ //如果value为null 则说明在from表单中没有传该值 //判断 keyText 是否是复选框 key 以 flag 作为前缀 if(keyText!=null&&!"".equals(keyText)&&keyText.startsWith("flag_")){ //判断 params 是否包含 keyText这个Key 若 包含 //截取字符串 列如 flag_sourceType_1 ---> sourceType //在表单中为下拉选择 在模板中为复选框 String str[]=keyText.split("_"); boolean flag=false; checkbox=true; if(str.length>2){ String key=str[1];//StringUtils.substringBetween(keyText,"_","_"); Object v=params.get(key); if(v instanceof List){ List list=(List)v; for (int i = 0; i -1){ String[] textArr = newRunText.split("\n"); if(textArr.length>0){ //设置字体信息 String fontFamily = tempRun.getFontFamily(); int fontSize = tempRun.getFontSize(); //logger.info("------------------"+fontSize); for(int i=0;i
//测试
/** * 立案审批 * */ public void test2(){ String str="{\"istName\":\"粤海街道组长3-一般程序-2019-11-14\",\"caseTypeLabel\":[\"环境卫生管理\"],\"punishmentBasis\":\"《广东省城乡生活垃圾处理条例》第五十二条违反本条例第十条规定,生活垃圾分类管理责任人未履行工作责任的,由市、县(区)人民政府环境卫生主管部门责令限期改正;逾期不改正的,处一千元以上一万元以下的罚款。\",\"caseAddress\":{\"name\":\"南头中共深圳市南山区委员会(桃园路北)(广东省深圳市南山区桃园路2号)\",\"address\":\"广东省深圳市南山区桃园路2号\",\"latitude\":22.53332,\"longitude\":113.93041,\"pointWkt\":\"POINT(113.93041 22.53332)\",\"pointWkt2000\":\"\",\"adcode\":\"\",\"city\":\"\",\"citycode\":\"\",\"country\":\"\",\"district\":\"\",\"province\":\"\"},\"qrCode\":\"https://cgzf-1259237188.cos.ap-guangzhou.myqcloud.com/2019/10/29/10/802ee3cac0a8437eb64fac588f6c28f0.png?q-sign-algorithm=sha1&q-ak=AKIDicJm23hNfhdOxHrCpFUcXGiwuf9PZdRz&q-sign-time=1572316614;1572323814&q-key-time=1572316614;1572323814&q-header-list=&q-url-param-list=&q-signature=e9a08d199916e53eb69d8f1adbf4d686df22b8eb\",\"areaName\":\"南山区\",\"caseNumber\":\"[2019]深南城综粤海简罚决字第741号\",\"assistantPerson\":\"117\",\"certificateAddress\":\"13\",\"mainEnforcement\":\"1001\",\"mainEnforcementLabel\":[\"张三\"],\"busId\":\"51dd1aef2c77433883c29ee0315527c3\",\"certificatePicture\":[{\"id\":557,\"originalFileName\":\"wxba6bffebe25e971d.o6zAJsy_fHtwKYhcuMDnWuettYwc.Juwhm2AEhjyN32eabfa5571cd6e21fbd0a9eba7176cc.jpg\",\"fileName\":\"e886a753a95c4eb4a2e5f152b1d19a60.jpg\",\"fileSuffix\":\"jpg\",\"filePath\":\"https://cgzf-1257892252.cos.ap-chengdu.myqcloud.com/2019/11/14/09/e886a753a95c4eb4a2e5f152b1d19a60.jpg\",\"fileSize\":37545}],\"deptId\":44,\"personTypeLabel\":[\"公民\"],\"initiatorId\":\"ZZ003\",\"assistantPersonLabel\":[\"粤海街道片长1\"],\"areaCode\":440305007,\"phone\":\"123\",\"name\":\"123\",\"crimeTime\":\"2019-11-14 09:36:43\",\"deptCode\":\"SZCG_ZFD_440305007\",\"certificateType\":\"0\",\"clientName\":\"粤海街道组长3\",\"caseReasonLabel\":[\"生活垃圾分类管理责任人未履行工作责任的\"],\"isForceLabel\":\"否\",\"caseType\":\"A0101\",\"applicableBasis\":\"【适用】《广东省城乡生活垃圾处理条例》第十条生活垃圾分类管理责任人的工作责任:(一)建立生活垃圾日常分类管理制度,记录产生的生活垃圾种类和去向,并接受环境卫生主管部门的监督检查;(二)开展生活垃圾分类知识宣传,指导、监督单位和个人开展生活垃圾分类;(三)根据生活垃圾产生量和分类方法,按照标准和分类标志设置生活垃圾分类收集点和收集容器,并保持生活垃圾分类收集容器正常使用;(四)明确生活垃圾的投放时间、地点;(五)制止混合已分类的生活垃圾;(六)督促检查垃圾分类,把垃圾交由相关单位处理。【竞合】《深圳市生活垃圾分类和减量管理办法》第四十条生活垃圾分类投放管理责任人违反本办法第十二条规定,有下列行为之一的,由主管部门责令限期改正,逾期不改正的,处以罚款:(一)未按规定对生活垃圾分类投放工作进行指导的,处3000元罚款;(二)未公示生活垃圾分类投放时间或者地点的,处2000元罚款;(三)未记录或者未如实记录责任范围内生活垃圾排放情况的,处2000元罚款;(四)未按规定设置生活垃圾分类收集容器的,处3000元罚款;(五)未将分类投放的生活垃圾交由符合规定的单位分类收集、运输的,处5000元罚款。\",\"isAuth\":\"1\",\"isForce\":\"0\",\"personType\":\"1\",\"certificateTypeLabel\":[\"身份证证件\"],\"caseOrigin\":\"213\",\"surveyInfo\":\"123\",\"initiatorName\":\"粤海街道组长3\",\"sponsorLawEnforcer\":\"\",\"scenePicture\":[{\"id\":556,\"originalFileName\":\"wxba6bffebe25e971d.o6zAJsy_fHtwKYhcuMDnWuettYwc.F5bXFN3MwU2Q32eabfa5571cd6e21fbd0a9eba7176cc.jpg\",\"fileName\":\"c3e49fd3ba344af29ce08d92b76c7313.jpg\",\"fileSuffix\":\"jpg\",\"filePath\":\"https://cgzf-1257892252.cos.ap-chengdu.myqcloud.com/2019/11/14/09/c3e49fd3ba344af29ce08d92b76c7313.jpg\",\"fileSize\":37545}],\"caseReason\":\"A0101038\",\"areaId\":3277,\"certificateNumber\":\"440101199901010000\",\"age\":\"123\",\"busStatusName\":\"待立案\",\"busStatus\":\"201\",\"mainEnforcementSign\":{\"userId\":1001,\"personalSignature\":\"\"},\"assistantPersonSign\":{\"userId\":117,\"personalSignature\":\"\"},\"problemOriginCode\":\"0\",\"registerAdvise\":\"121\",\"problemOriginCodeLabel\":[\"\\\"双随机\\\"抽查\"],\"question\":[]}"; Map map= JSON.parseObject(str); //String jsonStr=new JSONObject(map).toJSONString(); // System.out.println("jsonStr="+jsonStr); //存放段落数据 /* for (int i=1;i<1;i++) { excelMapTemp = new HashMap(); excelMapTemp.put("filingRegister.no1", "one-"+i); excelMapTemp.put("filingRegister.no2", "two-"+i); excelMapTemp.put("filingRegister.no3", "three-"+i); excelMapTemp.put("filingRegister.no4", "four-"+i); excelMapTemp.put("filingRegister.no5", "five-"+i); excelMapList.add(excelMapTemp); }*/ //模板存放位置 String templatePath="D:/word/立案登记审批表.docx";//"D:/word/poi.docx";// //生成文档存放位置 String targetPath = "D:/word/立案"+ UUID.randomUUID() +".docx"; //初始化导出 PoiExportWord export = new PoiExportWord(templatePath); try { export.init(); } catch (IOException e) { e.printStackTrace(); } boolean flag=false; try { export.export(map); //0为表格的下标,第一个表格下标为0,以此类推 export.export(map, 0); flag=export.generate(targetPath); } catch (Exception e) { e.printStackTrace(); } if(flag){ // wordToPft("D:/word/f694a031-8b28-40ea-987e-ff16d081bcf2.docx","D:/word/45.pdf"); //XWPFDocument doc = new XWPFDocument(new FileInputStream(targetPath)); // 以下几行代码是把这个word文件转化成pdf文件 // wordToPdf1(targetPath,"D:/word/立案登记审批表.pdf"); //jsonStr=null; try { new DocxToPdf().docxToPdf(targetPath,"D:/word/立案登记审批表2.pdf"); } catch (IOException e) { e.printStackTrace(); } } }