easyexcle处理导出合并单元格图片填充

@GetMapping("getStyleSummaryExport")
@ApiOperation("款式汇总报表--导出")
@ApiImplicitParams({@ApiImplicitParam(name = Constant.PAGE, value = "当前页码,从1开始", paramType = "query", dataType = "int"),@ApiImplicitParam(name = Constant.LIMIT, value = "每页显示记录数", paramType = "query", dataType = "int"),@ApiImplicitParam(name = Constant.ORDER_FIELD, value = "排序字段", paramType = "query", dataType = "String"),@ApiImplicitParam(name = Constant.ORDER, value = "排序方式,可选值(asc、desc)", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "season", value = "季节", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "batch", value = "批次", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "series", value = "系列", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "styleNo", value = "款号", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "brand", value = "品牌", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "stage", value = "阶段", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "materialNo", value = "主身面料编号", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "supplier", value = "供应商", paramType = "query", dataType = "String"),@ApiImplicitParam(name = "shipmentDate", value = "上货日期", paramType = "query", dataType = "Date"),@ApiImplicitParam(name = "ids", value = "勾选列表id,多个逗号隔开", paramType = "query", dataType = "String")
})
public void getStyleSummaryExport(@ApiIgnore @RequestParam Map<String, Object> params, HttpServletResponse response) throws Exception {if(!ParamUtil.paramsIsNotNull(params,"limit") || !ParamUtil.paramsIsNotNull(params,"page")){params.put("limit", "1000");params.put("page", "1");}if(ParamUtil.paramsIsNotNull(params, "ids")){params.put("ids", Arrays.asList(params.get("ids").toString().split(",")));}else {params.remove("ids");}List<StyleSummaryDTO> list = styleSummaryService.getStyleSummaryReportPage(params).getList();List<List<Object>> resList = new ArrayList<>();//合并图片问题List<Integer> imgCount = new ArrayList<>();//合并的行List<Integer> merge = new ArrayList<>();AtomicInteger index = new AtomicInteger();SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");if(CollectionUtils.isNotEmpty(list)) {list.forEach(e ->{if(CollectionUtils.isNotEmpty(e.getItemList())){for (int i = 0; i < e.getItemList().size(); i++) {index.addAndGet(1);if(i > 0){merge.add(index.intValue());}List<Object> itemList = new ArrayList<>();if (i == 0 && CollectionUtils.isNotEmpty(e.getStyleUrlList())){try {itemList.add(new URL(e.getStyleUrlList().get(0).getUrl()));imgCount.add(e.getItemList().size());} catch (MalformedURLException malformedURLException) {log.info(malformedURLException.getMessage());}}else {itemList.add("");}itemList.add(e.getBrand());itemList.add(e.getSeason());itemList.add(e.getStage());itemList.add(e.getYbType());itemList.add(e.getBatch());itemList.add(e.getSeries());itemList.add(e.getStyleNo());if(e.getShipmentDate() != null){itemList.add(simpleDateFormat.format(e.getShipmentDate()));}else {itemList.add("");}itemList.add(e.getItemList().get(i).getMaterialNo());itemList.add(e.getItemList().get(i).getSupplier());resList.add(itemList);}}else {index.addAndGet(1);List<Object> itemList = new ArrayList<>();if (CollectionUtils.isNotEmpty(e.getStyleUrlList())){try {itemList.add(new URL(e.getStyleUrlList().get(0).getUrl()));imgCount.add(1);} catch (MalformedURLException malformedURLException) {log.info(malformedURLException.getMessage());}}else {itemList.add("");}itemList.add(e.getBrand());itemList.add(e.getSeason());itemList.add(e.getStage());itemList.add(e.getYbType());itemList.add(e.getBatch());itemList.add(e.getSeries());itemList.add(e.getStyleNo());if(e.getShipmentDate() != null){itemList.add(simpleDateFormat.format(e.getShipmentDate()));}else {itemList.add("");}resList.add(itemList);}});}response.setContentType("application/json;charset=utf-8");response.setCharacterEncoding("UTF-8");String fileName = URLEncoder.encode("开发款式汇总导出-"+  DateUtils.format(new Date()), "UTF-8");response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");ExcelWriter excelWriter = null;try{excelWriter = EasyExcel.write(response.getOutputStream()).build();//excelWriter = EasyExcel.write("D:\\2023\\test222111.xlsx").build();if(null != excelWriter) {//需要合并的列int[] mergeColumeIndex = {0,1,2,3,4,5,6,7,8};// 从那一列开始合并int mergeRowIndex = 1;// 内容的策略WriteCellStyle contentWriteCellStyle = new WriteCellStyle();contentWriteCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);WriteCellStyle headWriteCellStyle = new WriteCellStyle();headWriteCellStyle.setFillForegroundColor(IndexedColors.WHITE.index);HorizontalCellStyleStrategy horizontalCellStyleStrategy =new HorizontalCellStyleStrategy(headWriteCellStyle, contentWriteCellStyle);List<List<String>> userHeadList = getHeader();WriteSheet writeSheet = EasyExcel.writerSheet( "开发款式汇总").head(userHeadList).registerWriteHandler(horizontalCellStyleStrategy).registerWriteHandler(new CustomCellWeightWeightForDevStyleUtil()).registerWriteHandler(new MergeSameRowsStrategy(mergeRowIndex, mergeColumeIndex, merge)).registerWriteHandler(new StyleImageCellWriteHandler(imgCount)).build();excelWriter.write(resList, writeSheet);}}finally {//关闭流if(excelWriter != null){excelWriter.finish();}}
}

package com.jack.modules.dsr.common.utils;import com.alibaba.excel.metadata.CellData;
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import lombok.Data;
import org.apache.commons.collections.CollectionUtils;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellType;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.util.CellRangeAddress;import java.util.List;@Data
public class MergeSameRowsStrategy implements CellWriteHandler{private int[] mergeColumnIndex;private int mergeRowIndex;private List<Integer> merge;public MergeSameRowsStrategy() {}public MergeSameRowsStrategy(int mergeRowIndex, int[] mergeColumnIndex, List<Integer> merge) {this.mergeRowIndex = mergeRowIndex;this.mergeColumnIndex = mergeColumnIndex;this.merge = merge;}@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<CellData> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {if (CollectionUtils.isNotEmpty(merge)){// 当前行int curRowIndex = cell.getRowIndex();if (merge.contains(curRowIndex)){// 当前列int curColIndex = cell.getColumnIndex();if (curRowIndex > mergeRowIndex) {for (int i = 0; i < mergeColumnIndex.length; i++) {if (curColIndex == mergeColumnIndex[i]) {mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);break;}}}}}}/*** 当前单元格向上合并** @param writeSheetHolder sheet保持对象* @param cell             当前单元格* @param curRowIndex      当前行* @param curColIndex      当前列*/private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {// 获取当前行的当前列的数据和上一行的当前列列数据,通过上一行数据是否相同进行合并Object curData = cell.getCellTypeEnum() == CellType.STRING ? cell.getStringCellValue() : cell.getNumericCellValue();if (cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex) == null){return;}Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);Object preData = preCell.getCellTypeEnum() == CellType.STRING ? preCell.getStringCellValue() : preCell.getNumericCellValue();// 比较当前行的第一列的单元格与上一行是否相同,相同合并当前单元格与上一行if (curData.equals(preData) || ("0.0".equals(preData.toString()) && "".equals(curData))) {Sheet sheet = writeSheetHolder.getSheet();List<CellRangeAddress> mergedRegions = sheet.getMergedRegions();boolean isMerged = false;for (int i = 0; i < mergedRegions.size() && !isMerged; i++) {CellRangeAddress cellRangeAddr = mergedRegions.get(i);// 若上一个单元格已经被合并,则先移出原有的合并单元,再重新添加合并单元if (cellRangeAddr.isInRange(curRowIndex - 1, curColIndex)) {sheet.removeMergedRegion(i);cellRangeAddr.setLastRow(curRowIndex);sheet.addMergedRegion(cellRangeAddr);isMerged = true;}}// 若上一个单元格未被合并,则新增合并单元if (!isMerged) {CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, curColIndex, curColIndex);sheet.addMergedRegion(cellRangeAddress);}}}@Overridepublic void afterCellDataConverted(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, CellData cellData, Cell cell, Head head, Integer integer, Boolean aBoolean) {}@Overridepublic void beforeCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Head head, Integer integer, Integer integer1, Boolean aBoolean) {}@Overridepublic void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer integer, Boolean aBoolean) {}}

package com.jack.modules.dsr.common.handler;import com.alibaba.excel.enums.CellDataTypeEnum;
import com.alibaba.excel.metadata.CellData;
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import lombok.Data;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.*;import java.util.ArrayList;
import java.util.List;/*** EasyExcel图片填充拦截器** @author piyc*/
@Data
public class StyleImageCellWriteHandler implements CellWriteHandler {private List<Integer> imgCount;public StyleImageCellWriteHandler() {}public StyleImageCellWriteHandler(List<Integer> imgCount) {this.imgCount = imgCount;}int zab = 0;List<byte[]> imageList = new ArrayList<>();@Overridepublic void beforeCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Head head, Integer integer, Integer integer1, Boolean aBoolean) {}@Overridepublic void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer integer, Boolean aBoolean) {}@Overridepublic void afterCellDataConverted(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, CellData cellData, Cell cell, Head head, Integer integer, Boolean aBoolean) {// 在 数据转换成功后 ,修改第一列 当然这里也可以根据其他判断 然后不是头 就把类型设置成空 这样easyexcel 不会去处理该单元格if (head.getColumnIndex() != 0 || aBoolean) {return;}if (cellData.getImageValue() != null && cellData.getImageValue().length > 0){cellData.setType(CellDataTypeEnum.EMPTY);}}@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<CellData> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {// 在 单元格写入完毕后 ,自己填充图片if (head.getColumnIndex() != 0 || aBoolean || CollectionUtils.isEmpty(list) || list.get(0) == null) {return;}if (list.get(0).getStringValue() != null && StringUtils.isNotEmpty(list.get(0).getStringValue())){list.get(0).setType(CellDataTypeEnum.STRING);return;}Sheet sheet = cell.getSheet();// cellDataList 是list的原因是 填充的情况下 可能会多个写到一个单元格 但是如果普通写入 一定只有一个if (list.get(0).getImageValue() != null && list.get(0).getImageValue().length > 0){int index = sheet.getWorkbook().addPicture(list.get(0).getImageValue(), HSSFWorkbook.PICTURE_TYPE_PNG);Drawing drawing = sheet.getDrawingPatriarch();if (drawing == null) {drawing = sheet.createDrawingPatriarch();}CreationHelper helper = sheet.getWorkbook().getCreationHelper();ClientAnchor anchor = helper.createClientAnchor();// 设置图片坐标anchor.setDx1(0);anchor.setDx2(0);anchor.setDy1(0);anchor.setDy2(0);//设置图片位置anchor.setCol1(cell.getColumnIndex());anchor.setCol2(cell.getColumnIndex() +1);anchor.setRow1(cell.getRowIndex());if (head.getColumnIndex() == 0 && CollectionUtils.isNotEmpty(imgCount)){anchor.setRow2(cell.getRowIndex() + imgCount.get(zab));zab++;} else {anchor.setRow2(cell.getRowIndex() + 1);}// 设置图片可以随着单元格移动anchor.setAnchorType(ClientAnchor.AnchorType.MOVE_AND_RESIZE);drawing.createPicture(anchor, index);}}
}
package com.jack.modules.dsr.common.utils;
import com.alibaba.excel.metadata.CellData;
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.style.column.AbstractColumnWidthStyleStrategy;
import com.alibaba.nacos.common.utils.CollectionUtils;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Sheet;import java.util.List;public class CustomCellWeightWeightForDevStyleUtil extends AbstractColumnWidthStyleStrategy {@Overrideprotected void setColumnWidth(WriteSheetHolder writeSheetHolder, List<CellData> cellDataList, Cell cell, Head head, Integer integer, Boolean isHead) {boolean needSetWidth = isHead || !CollectionUtils.isEmpty(cellDataList);if (needSetWidth) {Sheet sheet = writeSheetHolder.getSheet();int columnIndex = cell.getColumnIndex();if (columnIndex == 0 || columnIndex == 6 || columnIndex == 7 || columnIndex == 8 || columnIndex == 9 || columnIndex == 10){sheet.setColumnWidth(columnIndex,4576);}}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/236370.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

jvm相关命令操作

查看jvm使用情况 jmap -heap PID 查看线程使用情况 jstack pid 查看当前线程数 jstack 21294 |grep -E (#[0-9]) -o -c 查看系统线程数 top -H top -Hp pid #查看具体的进程中的线程信息 使用 jps 命令查看配置了JVM的服务 查看某个进程JVM的GC使用情况 jstat -gc 进程…

STM32G4x FLASH 读写(修改HAL库驱动)

主要工作就是把HAL的超时用LL库延时替代&#xff0c;保留了中断擦写模式、轮询等待擦写&#xff0c;待验证哈。 笔者用的芯片为STM32G473CBT6 128KB Flash&#xff0c;开环环境为CUBEMXMDK5.32&#xff0c;因为G4已经没有标准库了&#xff0c;笔者还是习惯使用标准库的开发方式…

快速能访问服务器的文件

1、背景 访问ubuntu上的文件 2、方法 python3 -m http.server 8081 --directory /home/ NAS 共享访问协议 — NFS、SMB、FTP、WebDAV 各有何优势&#xff1f;http://1 Ubuntu 搭建文件服务器&#xff08;Nginx&#xff09;

FreeSWITCH回铃音补偿

回铃音介绍 回铃音就是拨打电话时&#xff0c;电话接通之前听到的嘟嘟的声音&#xff0c;如果拨打了开通了彩铃的手机&#xff0c;听到的是音乐声&#xff0c;彩铃也是一种回铃音&#xff0c;5G时代&#xff0c;又有了视频彩铃这样更新的回铃音方式了。 SIP信令之回铃音 在s…

Git初始

一)git的介绍: 1)假设现在有一个文档&#xff0c;你的老板要求你针对于这份文件进行修改&#xff0c;进行完成的修改的版本是版本1&#xff0c;接下来是文档2&#xff0c;修改完文档2以后&#xff0c;接下来老板还不同意&#xff0c;于是又有了文档三&#xff0c;文档四&#x…

Java 干净的Stream,Lambda代码写法

目录 一. 前期准备1.1 各种实体类1.2 生成List<CommonUser>的方法1.3 需求 二. 难以阅读的Lambda表达式三. 干净的Lambda表达式四. 效果 一. 前期准备 1.1 各种实体类 ⏹Address实体类&#xff0c;用来存储地址 import lombok.Builder; import lombok.Getter;Builder …

力扣刷题记录(17)LeetCode:416、1049

416. 分割等和子集 可以将该问题看成是一个背包问题。背包的容量就是nums数组和的一半。我们如果能够将背包装满就意味着可以将数组分割成两个元素和相等的子集。 1.确定dp[i]的含义 索引i表示背包的容量&#xff0c;dp[i]表示当前容量能够装载的最大值 2.确定动态转移方程 …

Uniapp + Vue3 封装请求工具挂载全局

新建request.js工具类 const http {// baseUrl 地址baseUrl: http://localhost:8080,// 请求方法request(config) {// config&#xff1a;请求配置对象&#xff0c;具体参照uniapp文档config beforeRequest(config)// 请求地址拼接config.url this.baseUrl config.url// 异…

前端性能优化四:精简html代码

1. 精简HTML代码: ①. 减少HTML嵌套,层级关系缩小.②. 减少DOM节点数.③. 减少无语义代码:a. 如:<div class"clear"></div>消除浮动,或者用css的方式处理.④. 删除http或者https:a. 如果URL的协议头恶化当前页面的协议头一致的.b. 或者此URL在多个协议头…

Linux-----14、vim

# vim Linux平台下的文本编辑器&#xff1a; emacs、nano、gedit、vi、vim vi&#xff08;visual editor&#xff09;编辑器通常被简称为vi&#xff0c;它是Linux和Unix系统上最基本的文本编辑器&#xff0c;类似于Windows 系统下的记事本。学会它后&#xff0c;我们将在Linu…

Vue数组变更方法和替换方法

一、可以引起UI界面变化 Vue 将被侦听的数组的变更方法进行了包裹&#xff0c;所以它们也将会触发视图更新。这些被包裹过的方法包括&#xff1a; push()pop()shift()unshift()splice()sort()reverse() 以上七个数组都会改变原数组&#xff0c;下面来分别讲解它们的区别&…

【Unity】【WebRTC】如何用Unity而不是浏览器接收远程画面

【背景】 之前几篇我们讨论了如何设置信令服务器&#xff0c;如何发送画面给远端以及如何用浏览器查看同步画面&#xff0c;今天来讨论如何实现Unity内部接收画面。 看本篇之前请先看过之前将web服务器设置和基本远程画面功能的几篇博文。&#xff08;同专栏下查看&#xff09…

Linux 脚本监控cpu输出到日志,每秒执行1次

vi test.sh while true; doecho date "%Y-%m-%d %H:%M:%S"", "$(top -bn1 | grep "Cpu(s)" | awk {print $1$2$3}) >> cpu_log.txtsleep 1 done top -bn1 | grep "Cpu(s)" 表示打印top命令的输出结果。 sleep 1 表示睡眠1秒…

TDengine 签约中船九院,助力航运业智能化转型升级

在大数据时代背景下&#xff0c;船舶智能化已经成为船舶制造与航运领域发展的必然趋势。智能船舶作为《中国制造 2025》中明确重点发展的领域&#xff0c;代表了船舶未来的方向&#xff0c;对于航运业的转型升级至关重要。其中&#xff0c;大数据的处理和运用成为船舶智能化转型…

每日一题(LeetCode)----栈和队列--逆波兰表达式求值

每日一题(LeetCode)----栈和队列–逆波兰表达式求值 1.题目&#xff08;150. 逆波兰表达式求值&#xff09; 给你一个字符串数组 tokens &#xff0c;表示一个根据 逆波兰表示法 表示的算术表达式。 请你计算该表达式。返回一个表示表达式值的整数。 注意&#xff1a; 有效的算…

如何通过宝塔面板搭建一个MySQL数据库服务并实现无公网ip远程访问?

文章目录 前言1.Mysql服务安装2.创建数据库3.安装cpolar3.2 创建HTTP隧道 4.远程连接5.固定TCP地址5.1 保留一个固定的公网TCP端口地址5.2 配置固定公网TCP端口地址 前言 宝塔面板的简易操作性,使得运维难度降低,简化了Linux命令行进行繁琐的配置,下面简单几步,通过宝塔面板cp…

python-dlib实现人脸提取和分割

效果 → 参考资料和资源 GitHub - Onwaier/SegfaceAndAlignByDlib: 用dlib实现脸部分割和人脸对齐 shape_predictor_68_face_landmarks.dat 下载地址_shape_predictor_68_face_landmarks.dat下载-CSDN博客 未运行的参考资料 dlib实现脸部分割与人脸对齐 - 知乎 py代码 &…

自定义通用物联网网关的MIB结构的参考

文章目录 说明参考代码开源项目地址 说明 由于时间有限&#xff0c;本人精力有限&#xff0c;所以给出的网关MIB结构&#xff0c;并非完美&#xff01;本文网关MIB结构的结构定义构思在使用MIB builder自定义物联网网关的MIB结构欢迎小伙伴&#xff0c;点赞、关注和评论&#…

【Lidar】Open3D点云DBSCAN聚类算法:基于密度的点云聚类(单木分割)附Python代码

1 DBSCAN算法介绍 DBSCAN聚类算法是一种基于密度的聚类算法&#xff0c;全称为“基于密度的带有噪声的空间聚类应用”&#xff0c;英文名称为Density-Based Spatial Clustering of Applications with Noise。 DBSCAN聚类算法能够发现任意形状的类别&#xff0c;并且对噪音数据具…

全网最全pytest大型攻略,单元测试学这就够了!

pytest 是一款以python为开发语言的第三方测试&#xff0c;主要特点如下&#xff1a; 比自带的 unittest 更简洁高效&#xff0c;兼容 unittest框架支持参数化可以更精确的控制要测试的测试用例丰富的插件&#xff0c;已有300多个各种各样的插件&#xff0c;也可自定义扩展&am…