java基于POI批量插入图片到word

前言

我们的需求:可以批量的向word中添加图片和文本。在网上找了很久,找到的都是只能向word添加一张图片的,而且都是使用的freemark,一个是找到的需求不满足,另外一个是我也不想使用freemark,因此,我决定用java结合着poi自己手动写一个。
生成效果图
java基于POI批量插入图片到word_第1张图片

该报告为代码生成,达到的效果是影像和处理建议都可以动态插入多张图片和文字
在这里插入图片描述
第一步 添加依赖

该依赖为对poi封装后的依赖,还需自己添加相关poi依赖

 <dependency>
            <groupId>com.deepoove</groupId>
            <artifactId>poi-tl</artifactId>
            <version>1.6.0</version>
        </dependency>

如果你还没有poi依赖,那么可以使用下面的依赖(如果有请忽略下面这些依赖)

<dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>4.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>4.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-excelant</artifactId>
            <version>4.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml-schemas</artifactId>
            <version>4.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-scratchpad</artifactId>
            <version>4.0.0</version>
        </dependency>

第二步 创建要封装的表单数据

CommonReportVo将这个类换成自己的实体类数据,这个方法主要目的是给表格里面的数据赋值并返回(此步骤根据自身实际情况赋值),这里表格是两列,因此我使用二维数组,如果你的word表格数据是多列,那么你可以使用一个实体类进行封装后返回。注:此部分代码仅做表格数据赋值参考

private String[][] setSatelliteReportTemplate(CommonReportVo commonReportVo) {
        String[] names = {"测试001","测试002,"测试0003","测试0004","测试0005","测试0006"};
        String lon = GCSUtils.changeToDFM(Double.parseDouble(commonReportVo.getLon()));
        String lat = GCSUtils.changeToDFM(Double.parseDouble(commonReportVo.getLat()));
        String status = commonReportVo.getDisaterSituatuin();
        String taskNo = commonReportVo.getTaskNo();
        String taskRecord = commonReportVo.getTaskRecord();
        //多张图片地址 对应测试0004
        String analysisResult = commonReportVo.getDataAnalysisResult();
        String suggestion = commonReportVo.getEmergencySuggestion();
        String otherMaterial = "详见附件文件";
        String[] values = {status,taskNo,taskRecord,analysisResult,suggestion,otherMaterial};
        String[][] arrs = new String[names.length][2];
        //设置行数据
        for (int i = 0;i < names.length;i ++){
            arrs[i][0] = names[i];
        }
        //设置列数据
        for (int i = 0;i < values.length;i ++){
            arrs[i][1] = values[i];
        }
        return arrs;
    }

第三步 将数据设置到word

//参数1:响应体
//参数2:上面的二维数据组,即表格中的数据
//参数3:设置标题、时间等字段的封装类
private File setInfo2CommonWord(HttpServletResponse response, String[][] reports,CommonReportVo commonReportVo)  {
		//创建一个word
        XWPFDocument doc = new XWPFDocument();
        XWPFParagraph title = doc.createParagraph();
        XWPFRun xwpfRun = title.createRun();
        //设置标题  测试0001报告
        xwpfRun.setText(commonReportVo.getReportName() + "报告");
        xwpfRun.setFontSize(16);
        //设置标题加粗
        xwpfRun.setBold(true);
        //设置标题居中
        title.setAlignment(ParagraphAlignment.CENTER);
        XWPFParagraph content1 = doc.createParagraph();
        XWPFRun xwpfRun1 = content1.createRun();
        xwpfRun1.setText("报告单位:测试一下");
        //设置体字体大小
        xwpfRun1.setFontSize(12);
        XWPFParagraph content2 = doc.createParagraph();
        XWPFRun xwpfRun2 = content2.createRun();
        xwpfRun2.setText("灾害报告时间:" + DateUtil.formatDate1(commonReportVo.getReportTime()));
        xwpfRun2.setFontSize(12);
        String imgResult = commonReportVo.getSatelliteImgResult();
        try {
        	//注意 这是创建表格 创建的是几行几列  这儿需要根据自己实际的表格行列数自己设置
        	//参数1:表示行  参数2:表示列
            XWPFTable table = doc.createTable(reports.length, 2);
            //下面的两个for循环就是设置表格数据  这里仅提供样例的设置思路
            for (int i = 0;i < reports.length; i ++){ //行
            	//得到列数据对象
                XWPFTableCell cell = table.getRow(i).getCell(0);
                cell.setWidth("1500");
                for (int j = 0; j < reports[0].length; j ++){ //列
                	//设置表格每个单元格的样式
                    XWPFParagraph paragraph = table.getRow(i).getCell(j).getParagraphs().get(0);
                    //对齐方式 左边
                    paragraph.setAlignment(ParagraphAlignment.LEFT);
                    XWPFRun run = paragraph.createRun();
                    //字体9号
                    run.setFontSize(9);
                    //以下setPics方法为给表格设置多张图片和文字 看明白后根据自己的业务进行设置
                   //注意 修改此处,然后替换为自己的多张图片
                    if(i == 3 && j == 1){
                    	//给对应的单元格设置多张图片
                        setPics(reports[i][j], table.getRow(i), j,imgResult);
                    }else {
                    	//给单元格设置文本
                        String[] runTexts=reports[i][j].split("\r\n");
                        for (String s:runTexts){
                            XWPFRun tempRun=paragraph.createRun();
                            tempRun.setFontSize(9);
                            tempRun.setText(s);
                            tempRun.addBreak();
                        }
                    }
                }
            }
            //设置表格为A4纸张 
            //注意  参数2:表格列数 需要改为自己的列数
            TableTools.widthTable(table, MiniTableRenderData.WIDTH_A4_FULL, 2);
            //以下方式自行选择
            //直接下载 方式一
            String fileName =commonReportVo.getReportName() + "测试报告";
//            OutputStream out = response.getOutputStream();
//            response.setHeader("Content-Type", "application/ms-winword");
//            response.addHeader("Content-Disposition", "attachment;filename=\"" + URLEncoder.encode(fileName, "UTF-8") + ".docx\"");
//            response.setContentType("application/octet-stream");
//            response.setCharacterEncoding("UTF-8");
//            doc.write(out);
            //返回文件的方式下载 方式二
            File file = new File(filePath+File.separator+fileName + ".docx");
            FileOutputStream fos = new FileOutputStream(file);
            doc.write(fos);
            doc.close();
            return file;
        } catch (IOException | InvalidFormatException e) {
            e.printStackTrace();
        }
        return null;
    }

注意:文字换行赋值如下

文字描述如果需要换行时要在设置赋值时的文本中添加\r\n,如:imgResult这个字段中的值是这样的,
imgResult:这是一段测试文字\r\n这是一段测试文字\r\n这是一段测试文字
那么取值后给对应的单元格赋值时这样的


```go
 String[] runTexts=imgResult.split("\r\n");
            for (String s:runTexts){
                run=p1.createRun();
                run.setFontSize(9);
                run.setText(s);
                run.addBreak();
            }


批量设置多张图、文字及描述

```java
//参数1:对应样例中测试0004的多张图片的路径 
//参数2:行对象
//参数3:图片上方的文字描述 
 private void setPics(String analysisResult, XWPFTableRow row, int j,String imgResult) throws InvalidFormatException, IOException {
        XWPFParagraph p1 = row.getCell(j).getParagraphs().get(0);
        //通过分隔符获取多张图片路径
        String[] filePaths = analysisResult.split(",");
        if(filePaths.length > 0){
            XWPFRun run = null;
            //设置图片上的文字描述
            String[] runTexts=imgResult.split("\r\n");
            for (String s:runTexts){
                run=p1.createRun();
                run.setFontSize(9);
                run.setText(s);
                run.addBreak();
            }
            //run.addBreak(); //换行
            //一个单元格中每行设置几张图片 我这里设置的是每行2张图片 然后换行 
            /注意: 每行几张图片根据自己业务修改
            for (int k = 0;k < filePaths.length; k +=2){
if(filePaths[k].toUpperCase().contains(".JPG")||filePaths[k].toUpperCase().contains(".PNG")||filePaths[k].toUpperCase().contains(".SVG")){
                    run.addTab();   //添加一个tab
                    int index = filePaths[k].lastIndexOf("\\");
                    String newStr = filePaths[k].substring(index+1);
                    int indexOf = newStr.lastIndexOf(".");
                    String pic = newStr.substring(indexOf);
                    int i = newStr.indexOf("_");
                    //获取文件名
                    String name = newStr.substring(0,i) + pic;
                    //获取文件流
                    InputStream stream = new FileInputStream(filePaths[k]);
                    //设置第一张图片
                    //参数1:图片流数据 参数2:图片类型  参数3 图片名称  参数4:图片宽度  参数5:图片高度
                    run.addPicture(stream, XWPFDocument.PICTURE_TYPE_PNG, "Generated"+k, Units.toEMU(128), Units.toEMU(128));
                    run.addTab();
                    run.addTab();
                    int lastK = k + 1;
                    String name2 = "";
                    if( (lastK) <= filePaths.length-1){
                        int index1 = filePaths[lastK].lastIndexOf("\\");
                        String newStr2 = filePaths[lastK].substring(index1+1);
                        int index2 = newStr2.lastIndexOf(".");
                        String pic2 = newStr2.substring(index2);
                        int i2 = newStr2.indexOf("_");
                        name2 = newStr2.substring(0,i2) + pic2;
                        InputStream stream2 = new FileInputStream(filePaths[lastK]);
                        // 设置第二张图片
                        run.addPicture(stream2, XWPFDocument.PICTURE_TYPE_PNG, "Generated"+lastK, Units.toEMU(128), Units.toEMU(128));
                        //换行
                        run.addBreak();
                    }
                    if(k == filePaths.length-1 && filePaths.length % 2 == 1){
                        run.addBreak();
                    }
                    //设置第一张图片的名称
                    run.setText(name);
                    //这里添加了两个tab 具体添加几个 根据自己的实际格式来调
                    run.addTab();
                    run.addTab();
                     //设置第二张图片的名称
                    run.setText(name2);

                    if(lastK == filePaths.length-1){
                        break;
                    }
                }
            }
        }
        row.getCell(j).setParagraph(p1);
    }

你可能感兴趣的:(spring,boot,spring,maven,java)