赞
踩
- 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导出需要的依赖-->
还有一个模板!这里上传不了啊,我只是自己记录,需要的人再找我吧
这个工具有弊端,我后面遇到了很多问题,下面看图:
1.windows正常导出,但linux报错,不知道什么原因。按照网上说的,windows需要把java.awt.headless设置为false,在linux的话设置为true,设置来设置去咱也没解决,下面是报错截图
后来搞了几天都没解决,后来想了一个方法,从前端传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很慢,很不理想。仅作为自己参考
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。