import cn.afterturn.easypoi.word.WordExportUtil;
import org.apache.poi.xwpf.usermodel.*;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.CTStyles;

import java.io.File;
import java.io.FileOutputStream;
import java.util.Map;

/**
 * @Author xiaoxin
 * @Date 2022/10/22 22:21
 * @Version 1.0
 */

public class ExportUtil {

    public static void export(Map<String, Object> map, String url, File tempFile) {

        try {
            XWPFDocument doc = WordExportUtil.exportWord07(url, map);
            FileOutputStream fos = new FileOutputStream(tempFile);
            doc.write(fos);
            fos.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}
import org.jfree.chart.*;
import org.jfree.chart.axis.CategoryAxis;
import org.jfree.chart.axis.ValueAxis;
import org.jfree.chart.block.BlockBorder;
import org.jfree.chart.labels.StandardPieSectionLabelGenerator;
import org.jfree.chart.plot.*;
import org.jfree.chart.title.LegendTitle;
import org.jfree.chart.title.TextTitle;
import org.jfree.data.category.CategoryDataset;
import org.jfree.data.general.DefaultPieDataset;

import java.awt.*;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;

/**
 * @Author xiaoxin
 * @Date 2022/10/22 22:22
 * @Version 1.0
 */


public class JFreeChartToFileUtil {


    /***
     * 生成饼图
     * @param pds
     * @param file
     * @param title
     */
    public static void createPieChart(DefaultPieDataset pds, File file,String title) {
        try {
            // 分别是:显示图表的标题、需要提供对应图表的DateSet对象、是否显示图例、是否生成贴士以及是否生成URL链接
            JFreeChart chart = ChartFactory.createPieChart(title, pds, false, false, true);
            // 如果不使用Font,中文将显示不出来
            Font font = new Font("宋体", Font.BOLD, 16);
            // 设置图片标题的字体
            chart.getTitle().setFont(font);
            // 得到图块,准备设置标签的字体
            PiePlot plot = (PiePlot) chart.getPlot();
            // 设置标签字体
            plot.setLabelFont(font);
            plot.setStartAngle(3.14f / 2f);

            // 设置plot的前景色透明度
            plot.setForegroundAlpha(0.7f);
            // 设置plot的背景色透明度
            plot.setBackgroundAlpha(0.0f);
            // 设置标签生成器(默认{0})
            // {0}:key {1}:value {2}:百分比 {3}:sum
            plot.setLabelGenerator(new StandardPieSectionLabelGenerator("{0}{2}"));
            // 将内存中的图片写到本地硬盘
            ChartUtils.saveChartAsJPEG(file, chart, 600, 300);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /***
     * 线状图
     * @param pds
     * @param file
     * @param title
     */
    public static void createBarChart(CategoryDataset pds, File file,String title) {
        try {
            // 分别是:显示图表的标题、需要提供对应图表的DateSet对象、是否显示图例、是否生成贴士以及是否生成URL链接
            JFreeChart chart = ChartFactory.createBarChart(title, null,
                    null, pds, PlotOrientation.VERTICAL,
                    true, true, true);
            // 如果不使用Font,中文将显示不出来
            Font font = new Font("宋体", Font.BOLD, 12);
            // 设置图片标题的字体
            chart.getTitle().setFont(font);
            chart.getLegend().setItemFont(font);
            // 得到图块,准备设置标签的字体
            CategoryPlot plot = (CategoryPlot) chart.getPlot();
            // 设置plot的前景色透明度
            plot.setForegroundAlpha(0.7f);
            // 设置plot的背景色透明度
            plot.setBackgroundAlpha(0.0f);
            // 设置标签生成器(默认{0})

            ValueAxis rangeAxis = plot.getRangeAxis();
            CategoryAxis domainAxis = plot.getDomainAxis();

            rangeAxis.setLabelFont(font);
            rangeAxis.setTickLabelFont(font);
            domainAxis.setLabelFont(font);
            domainAxis.setTickLabelFont(font);
            domainAxis.setMaximumCategoryLabelLines(10);
            domainAxis.setMaximumCategoryLabelWidthRatio(0.5f);
            // 将内存中的图片写到本地硬盘
            ChartUtils.saveChartAsJPEG(file, chart, 600, 300);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    /***
     * 生成空心圆
     * @param pds
     * @param file
     * @param title
     */
    public static void createCircularChart(DefaultPieDataset pds, File file,String title) {

        StandardChartTheme mChartTheme = new StandardChartTheme("CN");
        // Y柱标签字体(浓度(单位:ppm))
        mChartTheme.setLargeFont(new Font("黑体", Font.BOLD, 14));
        //标题 (二氧化碳浓度趋势分析图)
        mChartTheme.setExtraLargeFont(new Font("宋体", Font.PLAIN, 20));
        //应用主题样式
        ChartFactory.setChartTheme(mChartTheme);
        //定义图标对象
        //
        JFreeChart chart = ChartFactory.createRingChart ("",// 报表题目,字符串类型
                pds, // 获得数据集
                false, // 显示图例
                false, // 不用生成工具
                false // 不用生成URL地址
        );

        Font font = new Font("宋体", Font.BOLD, 12);

        //图表
        RingPlot ringplot=(RingPlot) chart.getPlot();
        ringplot.setLabelFont(font);

        // 设置饼状图和环形图的显示数字。0代表显示文字说明,1代表显示数字,2代表显示数字以百分比的方式如果多个结合{0}:{1}
        ringplot.setLabelGenerator(new StandardPieSectionLabelGenerator("{0}:{1}"));
        ringplot.setSimpleLabels(true);

        ringplot.setBackgroundPaint(Color.WHITE);//设置背景色
        //设置简单标签
        ringplot.setSimpleLabels(true);
        //标题
        TextTitle texttitle=chart.getTitle();
        texttitle.setFont(font);
        //图示,就是点击后会弹出一个图片,显示的文字样式
//        LegendTitle legendtitle =chart.getLegend();
        //图示的标题
//        legendtitle.setItemFont(new Font("宋体", Font.BOLD, 14));
        ChartFrame mChartFrame = new ChartFrame("环形图", chart);
        mChartFrame.pack();
        //设置为true,点击请求后会弹出一个图片,设置为flase不弹出
        mChartFrame.setVisible(false);


        extracted(ringplot);

        try {
            // 分别是:显示图表的标题、需要提供对应图表的DateSet对象、是否显示图例、是否生成贴士以及是否生成URL链接
            ChartUtils.saveChartAsJPEG(file, chart, 550, 350);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    /***
     * 绘制环形图案颜色
     * @param ringplot
     */
    private static void extracted(RingPlot ringplot) {
        ringplot.setDrawingSupplier(new DefaultDrawingSupplier(
                new Paint[] {
                        new Color(46, 199, 201),
                        new Color(182, 162, 222),
                        new Color(90, 177, 239),
                        new Color(255, 185, 128),
                        new Color(226, 117, 123)
                },
                DefaultDrawingSupplier.DEFAULT_OUTLINE_PAINT_SEQUENCE,
                DefaultDrawingSupplier.DEFAULT_STROKE_SEQUENCE,
                DefaultDrawingSupplier.DEFAULT_OUTLINE_STROKE_SEQUENCE,
                DefaultDrawingSupplier.DEFAULT_SHAPE_SEQUENCE));
    }
import cn.afterturn.easypoi.entity.ImageEntity;
import io.ctc.commons.tools.utils.Result;
import io.ctc.commons.tools.utils.StringUtils;
import lombok.extern.slf4j.Slf4j;
import org.jfree.data.category.DefaultCategoryDataset;
import org.jfree.data.general.DefaultPieDataset;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import sun.misc.BASE64Decoder;

import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.text.SimpleDateFormat;
import java.util.*;

/**
 * 统计报表Controller
 *
 * @author sunmingwei
 * @version 2021-05-25
 */
@Slf4j
@RestController
@RequestMapping("test")
public class TestController {






    @RequestMapping("export")
    public void export(HttpServletResponse response) throws IOException {
        response.setContentType("application/msword");
        response.setHeader("Content-disposition", "attachment;filename=" + URLEncoder.encode("测试.docx", StandardCharsets.UTF_8.name()));
        OutputStream outputStream = response.getOutputStream();

        Map<String, Object> map = new HashMap<>();
        putBaseInfo(map);
        putImg(map);
        putList(map);
        putBar(map);


        String url =  Objects.requireNonNull(getClass().getClassLoader().getResource("export.docx")).getPath();
        File tempFile = File.createTempFile("tempDoc", ".docx");
        ExportUtil.export(map, url, tempFile);

        InputStream in = new FileInputStream(tempFile);

        //创建存放文件内容的数组
        byte[] buff = new byte[1024];
        //所读取的内容使用n来接收
        int n;
        //当没有读取完时,继续读取,循环
        while ((n = in.read(buff)) != -1) {
            //将字节数组的数据全部写入到输出流中
            outputStream.write(buff, 0, n);
        }
        //强制将缓存区的数据进行输出
        outputStream.flush();
        //关流
        outputStream.close();
        in.close();
        tempFile.deleteOnExit();

    }


    private void putBaseInfo(Map<String, Object> map) {
        map.put("theme", "集中化测试平台");
        map.put("nowDate",new SimpleDateFormat("yyyyMMdd").format(new Date()));
        map.put("person", "1、\t缺陷概览");
        map.put("title1", "2、\t缺陷等级分布");
        map.put("title2", "1)\t缺陷等级分布图");
        map.put("title3", "2)\t缺陷等级分布列表");
    }

    /***
     * 部分图像数据负责的,由前端传base64过来直接转图片
     * @param map
     */
    private void putImg(Map<String, Object> map) {
        ImageEntity image = getImage(str);
        map.put("img1", image);
    }

    /***
     * 环形图数据
     * @param map
     * @throws IOException
     */
    private void putBar(Map<String, Object> map) throws IOException {
        File file2 = File.createTempFile("temp", "jpg");
        DefaultPieDataset pds = new DefaultPieDataset();
        pds.setValue("建议", 100);
        pds.setValue("轻微", 100);
        pds.setValue("一般", 300);
        pds.setValue("严重", 400);
        pds.setValue("致命", 500);
        JFreeChartToFileUtil.createCircularChart(pds, file2, "1)缺陷等级分布图");
        ImageEntity image = new ImageEntity();
        image.setHeight(350);
        image.setWidth(550);
        image.setUrl(file2.getAbsolutePath());
        image.setType(ImageEntity.URL);
        map.put("img2", image);
    }




    private void putList(Map<String, Object> map) {
        List<Map<String, String>> list = new ArrayList<>();
        for (int i = 0; i < 10; i++) {
            Map<String, String> map1 = new HashMap<>();
            map1.put("name", "xiao");
            map1.put("age", "12");

            list.add(map1);
        }

        map.put("list", list);
    }

    private ImageEntity getImage(String base64CodeStr){
        if (StringUtils.isBlank(base64CodeStr)){
            return null;
        }
        ImageEntity image = new ImageEntity();
        image.setHeight(150);
        image.setWidth(550);
        base64CodeStr = base64CodeStr.replaceAll("data:image/png;base64,","");
        try {
            BASE64Decoder decode = new BASE64Decoder();
            byte[] b = decode.decodeBuffer(base64CodeStr.trim());
            image.setData(b);
        } catch (IOException e) {
            e.printStackTrace();
        }
        image.setType(ImageEntity.Data);
        return image;
    }



    public static String str = "";

}
<!--word导出需要的依赖-->
        <dependency>
            <groupId>org.jfree</groupId>
            <artifactId>jfreechart</artifactId>
            <version>1.5.3</version>
        </dependency>
        <dependency>
            <groupId>cn.afterturn</groupId>
            <artifactId>easypoi-base</artifactId>
            <version>4.4.0</version>
        </dependency>
        <dependency>
            <groupId>cn.afterturn</groupId>
            <artifactId>easypoi-web</artifactId>
            <version>4.4.0</version>
        </dependency>
        <dependency>
            <groupId>cn.afterturn</groupId>
            <artifactId>easypoi-annotation</artifactId>
            <version>4.4.0</version>
        </dependency>
        <!--word导出需要的依赖-->

还有一个模板!这里上传不了啊,我只是自己记录,需要的人再找我吧

Java 导出word 附带多张图片 java导出word带图表_Java 导出word 附带多张图片

这个工具有弊端,我后面遇到了很多问题,下面看图:

1.windows正常导出,但linux报错,不知道什么原因。按照网上说的,windows需要把java.awt.headless设置为false,在linux的话设置为true,设置来设置去咱也没解决,下面是报错截图

Java 导出word 附带多张图片 java导出word带图表_Java 导出word 附带多张图片_02

Java 导出word 附带多张图片 java导出word带图表_java_03

 

Java 导出word 附带多张图片 java导出word带图表_spring_04

后来搞了几天都没解决,后来想了一个方法,从前端传base64值过来直接添加进去,但这里也有坑,你会在日志中看到各种问题,例如找不到路径、找不到文件、找不到模板的等问题,因为我是Springboot,线上发布只有一个jar包,并不像那些tomcat一样,所以当你用Classloader.getResourcesAsStream("classpath:/template/export.docx")去获取模板路径的时候绝对会报错。所以下面的代码我将InputStream转成MyXWPFDocument对象了,然后再进行操作就可以了。下面看最新代码:

@PostMapping("/word")
    public void export(@RequestBody WordExportRequestDto wordExportDto, HttpServletResponse response) throws Exception {

        //根据项目id获取数据,也就是页面上获取的数据,转成entity
        Map<String, Object> hashMap = new HashMap<>();
        hashMap.put("projectId",wordExportDto.getProjectId());
        Map<String, Object> stringObjectMap = projectOverviewService.defectTopData(hashMap);
        String json = GsonUtils.GsonString(stringObjectMap);
        WordDataDto wordDataDto = GsonUtils.GsonToBean(json, WordDataDto.class);

        //查询出项目名称数据
        ProjectBaseinfoDto projectInfo = projectInfoDao.selectByIdProjectInfo(wordExportDto.getProjectId());

        String fileName = projectInfo.getName() + "项目概览-缺陷详情";
        response.setCharacterEncoding("UTF-8");
        response.setHeader("content-Type", "application/msword");
        response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName+".docx", StandardCharsets.UTF_8.name()));

        List<DefectStatusSpreadDto> defectStatusSpread = defectInfoService.getDefectStatusSpread(hashMap);

        //根据项目id获取缺陷列表
        List<String> list = new ArrayList<>();
        list.add(wordExportDto.getProjectId().toString());
        List<DefectInfoDto> defectByProjectIdList = defectInfoService.getDefectByProjectIdListCopy(list);


        Map<String, Object> map = new HashMap<>();
        map.put("defectSurveyPic",wordExportDto.getDefectSurveyPic());

        //添加大标题、小标题等信息
        putBaseInfo(map,projectInfo);
        //添加图片信息,此图片信息是前端传过来的base64
        putImg(map,wordExportDto);
        //添加缺陷等级分布列表数据
        putList(map,wordDataDto);

        //缺陷状态分布列表数据
        putDefectStatusList(map,defectStatusSpread);

        //缺陷列表数据,最下面的一张图
        putGetDefectList(map,defectByProjectIdList);

        //将流转换成word对象
        InputStream inputStream = ResourceUtil.getStream("classpath:template/export.docx");
        MyXWPFDocument doc = new MyXWPFDocument(inputStream);

        //导出word并指定word导出模板
        WordExportUtil.exportWord07(doc, map);
        //设置编码格式
        response.setCharacterEncoding(StandardCharsets.UTF_8.name());
        //设置内容类型
        response.setContentType("application/octet-stream");

        //设置头及文件命名。
        response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("测试.docx", StandardCharsets.UTF_8.name()));

        //写入
        doc.write(response.getOutputStream());
    }



    private void putBaseInfo(Map<String, Object> map,ProjectBaseinfoDto projectBaseinfoDto) {
        map.put("theme", projectBaseinfoDto.getName());
        map.put("nowDate",new SimpleDateFormat("yyyyMMdd").format(new Date()));
        map.put("person", "1、\t缺陷概览");
        map.put("title1", "2、\t缺陷等级分布");
        map.put("title2", "1)\t缺陷等级分布图");
        map.put("title3", "2)\t缺陷等级分布列表");
        map.put("title4", "3、\t缺陷状态分布");
        map.put("title5", "1)\t缺陷状态分布图");
        map.put("title6", "2)\t缺陷状态分布列表");
        map.put("title7", "4、\t缺陷列表");
    }

    /***
     * 部分图像数据负责的,由前端传base64过来直接转图片
     * @param map
     */
    private void putImg(Map<String, Object> map,WordExportRequestDto wordExportDto) {

        //顶部图
        if (Objects.nonNull(wordExportDto.getDefectSurveyPic())){
            ImageEntity image = getImage(map.get("defectSurveyPic").toString());
            map.put("img1", image);
        }

        //环形图数据 缺陷等级分布图
        if (Objects.nonNull(wordExportDto.getDefectLevelPic())){
            ImageEntity image = getImage(map.get("defectLevelPic").toString());
            map.put("img2", image);
        }

        //环形图数据 缺陷状态分布图
        if (Objects.nonNull(wordExportDto.getDefectStatusPic())){
            ImageEntity image = getImage(map.get("defectStatusPic").toString());
            map.put("img3", image);
        }
    }


    /***
     *  添加缺陷等级分布列表数据
     * @param map
     * @param wordDataDto
     */
    private void putList(Map<String, Object> map,WordDataDto wordDataDto) {
        List<Map<String, String>> list = new ArrayList<>();

        WordDefectLevelDto defectLevel = wordDataDto.getDefectLevel();
        if (Objects.nonNull(defectLevel)){
            List<WordTableDataDto> tableData = defectLevel.getTableData();
            if (!tableData.isEmpty()){
                for (WordTableDataDto dto : tableData){
                    Map<String, String> map1 = new HashMap<>();
                    map1.put("level", dto.getDefectLevel());
                    map1.put("defectCount", dto.getDefectCount());
                    map1.put("suspend", dto.getSuspend());
                    map1.put("moreThreeDaysRepair", dto.getThreeDaysRepair());
                    map1.put("repair", dto.getRepair());
                    map1.put("newTodays", dto.getNewTodays());
                    map1.put("rate", dto.getRate());
                    map1.put("twoDaysRepair", dto.getTwoDaysRepair());
                    map1.put("threeDaysRepair", dto.getThreeDaysRepair());
                    map1.put("closed", dto.getClosed());
                    map1.put("toVerified", dto.getToVerified());
                    list.add(map1);
                }
            }
        }

        map.put("list", list);
    }


    /***
     * 	缺陷状态分布列表数据
     * @param map
     * @param defectStatusSpread
     */
    private void putDefectStatusList(Map<String, Object> map,List<DefectStatusSpreadDto> defectStatusSpread) {

        //计算出总和
        IntSummaryStatistics repairTotal = defectStatusSpread.stream().mapToInt((x) -> Math.toIntExact(x.getCount())).summaryStatistics();

        List<Map<String, String>> list = new ArrayList<>();
            if (!defectStatusSpread.isEmpty()){
                for (DefectStatusSpreadDto dto : defectStatusSpread){
                    Map<String, String> map1 = new HashMap<>();
                    map1.put("statusName", dto.getStatusName());
                    map1.put("count", dto.getCount().toString());

                    //转换百分比
                    BigDecimal a = new BigDecimal(dto.getCount());
                    BigDecimal b = new BigDecimal(repairTotal.getSum());
                    int rate = NumberUtil.div(a, b,2).multiply(BigDecimal.valueOf(100)).intValue();
                    map1.put("rate", rate+"%");
                    list.add(map1);
                }
            }
        map.put("list2", list);
        }


    /***
     * 获取缺陷列表数据,最下面的表格
     * @param map
     * @param defectByProjectIdList
     */
    private void putGetDefectList(Map<String, Object> map,List<DefectInfoDto> defectByProjectIdList) {
        List<Map<String, String>> list = new ArrayList<>();

            if (!defectByProjectIdList.isEmpty()){
                int num = 0;
                for (int i = 0;i < defectByProjectIdList.size();i++){
                    num++;
                    Map<String, String> map1 = new HashMap<>();
                    map1.put("id", num+"");
                    map1.put("number", defectByProjectIdList.get(i).getDefectCode());
                    map1.put("defectName", defectByProjectIdList.get(i).getName());
                    if (Objects.nonNull(defectByProjectIdList.get(i).getDefectEnv())){
                        map1.put("dev", initializationEnv(defectByProjectIdList.get(i).getDefectEnv()));
                    }
                    if (Objects.nonNull(defectByProjectIdList.get(i).getDefectLevel())){
                        map1.put("level", initializationDefectLevel(defectByProjectIdList.get(i).getDefectLevel()));
                    }
                    if (Objects.nonNull(defectByProjectIdList.get(i).getDefectType())){
                        map1.put("type", initializationDefectType(defectByProjectIdList.get(i).getDefectType()));
                    }
                    map1.put("createator",defectByProjectIdList.get(i).getCreatorName());
                    map1.put("date", new SimpleDateFormat("yyyy-MM-dd").format(defectByProjectIdList.get(i).getCreateDate()));
                    list.add(map1);
                }

            }

        map.put("list3", list);
    }





    /***
     * 将base64封装成ImageEntity对象
     * @param base64CodeStr
     * @return
     */
    private ImageEntity getImage(String base64CodeStr){
        if (StringUtils.isBlank(base64CodeStr)){
            return null;
        }
        ImageEntity image = new ImageEntity();
        image.setHeight(150);
        image.setWidth(550);
        base64CodeStr = base64CodeStr.replaceAll("data:image/png;base64,","");
        try {
            BASE64Decoder decode = new BASE64Decoder();
            byte[] b = decode.decodeBuffer(base64CodeStr.trim());
            image.setData(b);
        } catch (IOException e) {
            e.printStackTrace();
        }
        image.setType(ImageEntity.Data);
        return image;
    }

,但问题又来了,前端传base64到后端,前端生成base64很慢,很不理想。仅作为自己参考