当前位置: 首页 > ds >正文

关于easyexcel动态下拉选问题处理

前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导致的问题,也通过将下拉数据存入隐藏Sheet再引用的方式解决。

直接上代码:

package com.tlit.platform.excel.annotations;import java.lang.annotation.*;/*** @author : Doug Liang* @date : 2025/1/22 14:21* @version: 1.0* @description: 支持easyExcel的下拉选的自定义注解*/@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ExcelSelected {/*** 固定下拉内容*/String[] source() default {};}

静态的下拉选支持直接注解配置

package com.tlit.platform.excel.util;import com.tlit.platform.excel.annotations.ExcelSelected;import java.util.Optional;/*** @author : Doug Liang* @date : 2025/1/22 14:24* @version: 1.0* @description: 解析下拉选注解工具类*/
public class ResoveSelectedUtil {public static String[] resove(ExcelSelected selected){if(!Optional.ofNullable(selected).isPresent()){return null;}// 获取固定下拉信息String[] source = selected.source();if(null != source && source.length > 0){return source;}return null;}
}

动态下拉选代码

package com.tlit.platform.excel.handler;import com.alibaba.excel.write.handler.SheetWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteWorkbookHolder;
import com.tlit.platform.excel.annotations.ExcelSelected;
import com.tlit.platform.excel.util.ResoveSelectedUtil;
import org.apache.poi.hssf.usermodel.HSSFDataValidation;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.ss.util.CellRangeAddressList;import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.Map;/*** @author : Doug Liang* @date : 2025/1/22 14:33* @version: 1.0* @description: 自定义拦截器处理*/
public class SelectedWriteHandler implements SheetWriteHandler {private final Map<Integer, String[]> map;// 调整为20大小 private static final Integer LIMIT = 20;private final char[] alphabet = new char[]{'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L','M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'};public SelectedWriteHandler(Class<?> clazz) {Field[] fields = clazz.getDeclaredFields();Map<Integer, String[]> map = new HashMap<>();for (int i = 0; i < fields.length; i++) {Field field = fields[i];// 解析注解信息ExcelSelected excelSelected = field.getAnnotation(ExcelSelected.class);if (null != excelSelected) {String[] sources = ResoveSelectedUtil.resove(excelSelected);if (null != sources && sources.length > 0) {map.put(i, sources);}}}this.map = map;}/*** 如果想动态传入对应的下拉选,则只需要将key和对应的下拉选列表传入即可** @param map Map<key , value>*            value -> {"","",""}*/public SelectedWriteHandler(Map<Integer, String[]> map) {this.map = map;}@Overridepublic void beforeSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {}@Overridepublic void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {// 这里可以对cell进行任何操作Sheet sheet = writeSheetHolder.getSheet();DataValidationHelper helper = sheet.getDataValidationHelper();String dictSheetName = "字典sheet" ;Workbook workbook = writeWorkbookHolder.getWorkbook();Sheet dictSheet = workbook.createSheet(dictSheetName);workbook.setSheetHidden(1, true);// k 为存在下拉数据集的单元格下表 v为下拉数据集map.forEach((k, v) -> {// 设置下拉单元格的首行 末行 首列 末列CellRangeAddressList rangeList = new CellRangeAddressList(1, 65536, k, k);// 下拉列表约束数据DataValidationConstraint constraint;if (v.length < LIMIT) {constraint = helper.createExplicitListConstraint(v);} else {// 获取动态数据的大小int rowLen = v.length;// 设置字典sheet页的值 每一列一个字典项for (int i = 0; i < rowLen; i++) {Row row = dictSheet.getRow(i);if (row == null) {row = dictSheet.createRow(i);}row.createCell(k).setCellValue(v[i]);}String excelColumn = getExcelColumn(k);// 下拉框数据来源 eg:字典sheet!$B1:$B2String refers = dictSheetName + "!$" + excelColumn + "$1:$" + excelColumn + "$" + rowLen;// 创建可被其他单元格引用的名称Name name = workbook.createName();// 设置名称的名字name.setNameName("dict" + k);// 设置公式name.setRefersToFormula(refers);// 设置引用约束constraint = helper.createFormulaListConstraint("dict" + k);}// 设置约束DataValidation validation = helper.createValidation(constraint, rangeList);if (validation instanceof HSSFDataValidation) {validation.setSuppressDropDownArrow(false);} else {validation.setSuppressDropDownArrow(true);validation.setShowErrorBox(true);}// 阻止输入非下拉选项的值validation.setErrorStyle(DataValidation.ErrorStyle.STOP);validation.createErrorBox("提示", "此值与单元格定义格式不一致");sheet.addValidationData(validation);});}/*** 将数字列转化成为字母列** @param num 数字* @return 字母列*/private String getExcelColumn(int num) {String column;int len = alphabet.length - 1;int first = num / len;int second = num % len;if (num <= len) {column = alphabet[num] + "" ;} else {column = alphabet[first - 1] + "" ;if (second == 0) {column = column + alphabet[len] ;} else {column = column + alphabet[second - 1] ;}}return column;}}

实际引用

EasyExcel.writerSheet(sheetName).head(head).registerWriteHandler(new SelectedWriteHandler(head))// 自适应宽度.registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).build();

效果图如下:

在这里插入图片描述

http://www.xdnf.cn/news/12322.html

相关文章:

  • Kerberos面试内容整理-Kerberos 的历史与发展
  • 【Linux】 Linux 进程控制
  • 【Android基础回顾】七:内存管理机制
  • 44、web实验-后台管理系统基本功能
  • MySQL——视图 用户管理 语言访问
  • 【JS进阶】ES6 实现继承的方式
  • CppCon 2015 学习:C++ Coroutines
  • LeetCode 1356.根据数字二进制下1的数目排序
  • Python异步爬虫与代理完美结合
  • Prompt Tuning:生成的模型文件有什么构成
  • 购物商城网站 Java+Vue.js+SpringBoot,包括商家管理、商品分类管理、商品管理、在线客服管理、购物订单模块
  • LUA+Reids实现库存秒杀预扣减 记录流水 以及自己的思考
  • uefi协议设计目的
  • linux——磁盘和文件系统管理
  • python打卡训练营打卡记录day45
  • 数学运算在 OpenCV 中的核心作用与视觉效果演示
  • 本地部署大模型实战:使用AIStarter一键安装Ollama+OpenWeb教程(含最新版本更新指南)
  • 【图像处理3D】:焦距的像素单位标定
  • 使用API有效率地管理Dynadot域名,查看域名市场中所售域名的详细信息
  • 宠物车载安全座椅市场报告:解读行业趋势与投资前景
  • MyBatis-Plus深度全解:从入门到企业级实战
  • 旋转字符串的解题思路与算法分享
  • Offline Transition Modeling via Contrastive Energy Learning
  • 【iSAQB软件架构】软件架构中构建块的视图:黑箱、灰箱和白箱及其交互机制
  • vue和uniapp聊天页面右侧滚动条自动到底部
  • 计算机网络领域所有CCF-A/B/C类期刊汇总!
  • 低代码逻辑引擎配置化实战:三步穿透审批记录查询
  • 鞋内测量新方案:Moticon传感器鞋垫OpenGo在运动科学中的平衡测试应用
  • BIM Revit教程(十一)如何使用机器学习实现 MEP 布局自动化?
  • NumPy数组操作完全指南:从入门到精通