springboot+easyexcel实现下载excels模板下拉选择

定义下拉注解

java 复制代码
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ExcelDropDown {
    /**
     * 固定下拉选项
     */
    String[] source() default {};

    /**
     * 动态数据源key(从上下文中获取)
     */
    String sourceMethod() default "";

    /**
     * 下拉框起始行(默认从第2行开始)
     */
    int firstRow() default 1;

    /**
     * 下拉框结束行(默认到10000行)
     */
    int lastRow() default 10000;
}

实现CellWriteHandler 接口

java 复制代码
@Slf4j
public class DynamicDropDownHandler implements CellWriteHandler {
    private final ApplicationContext applicationContext;

    public DynamicDropDownHandler(ApplicationContext applicationContext) {
        this.applicationContext = applicationContext;
    }

    @Override
    public void beforeCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Head head, Integer columnIndex, Integer relativeRowIndex, Boolean isHead) {

    }

    @Override
    public void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {


        Class clazz = writeSheetHolder.getClazz();
        try {
            String fieldName = head.getFieldName();
            Field field = clazz.getDeclaredField(fieldName);
            ExcelDropDown dropDown = field.getAnnotation(ExcelDropDown.class);
            if (dropDown != null) {
                String[] source = dropDown.source();

                if (source.length == 0) {
                    //判断是否有从库中查询数据的方法
                    String method = dropDown.sourceMethod();
                    if (null != method && !"".equals(method)) {
                        source = applicationContext.getBean(method, String[].class);
                    }
                    if (source.length == 0) {
                        return;
                    }

                }

                Sheet sheet = writeSheetHolder.getSheet();
                Workbook workbook = sheet.getWorkbook();
                // 1. 创建隐藏Sheet存储下拉选项
                String optionSheetName = fieldName+"_options";
                Sheet optionsSheet = workbook.getSheet(optionSheetName);
                if (optionsSheet == null) {
                    optionsSheet = workbook.createSheet(optionSheetName);
                    workbook.setSheetHidden(workbook.getSheetIndex(optionsSheet), true); // 隐藏Sheet
                }
                // 2. 写入选项数据到隐藏Sheet
                int optionCol = 0; // 使用第一列存储选项
                for (int i = 0; i < source.length; i++) {
                    Row row = optionsSheet.getRow(i);
                    if (row == null) {
                        row = optionsSheet.createRow(i);
                    }
                    row.createCell(optionCol).setCellValue(source[i]);
                }

                // 3. 创建名称引用
                String rangeName = "DROP_DOWN_" + fieldName.toUpperCase();
                Name namedRange = workbook.getName(rangeName);
                if (namedRange == null) {
                    namedRange = workbook.createName();
                    namedRange.setNameName(rangeName);
                }
                namedRange.setRefersToFormula(
                        String.format(optionSheetName+"!$A$1:$A$%d", source.length)
                );
                // 4. 设置数据验证(使用公式引用)
                DataValidationHelper helper = sheet.getDataValidationHelper();
                DataValidationConstraint constraint =
                        helper.createFormulaListConstraint(rangeName);

                // 设置下拉范围:从第二行开始到最大行,当前列
                int columnIndex = cell.getColumnIndex();
                CellRangeAddressList addressList = new CellRangeAddressList(
                        1, // 从第二行开始
                        2, // Excel最大行号
                        columnIndex,
                        columnIndex
                );
                DataValidation validation = helper.createValidation(constraint, addressList);
                validation.setSuppressDropDownArrow(true);
                validation.setShowErrorBox(true);
                sheet.addValidationData(validation);
            }
        } catch (Exception e) {
            log.error(e.getMessage());
        }
    }

    @Override
    public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<CellData> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {

    }
}

使用下载模板接口

java 复制代码
    public void downloadTemplate(HttpServletResponse response) {
        try {
            // 设置响应头
            response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
            response.setCharacterEncoding("utf-8");
            String fileName = URLEncoder.encode("模板", "UTF-8").replaceAll("\\+", "%20");
            response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName + ".xlsx");
            // 写入Excel
            EasyExcel.write(response.getOutputStream(), ImportVo.class)
                    .autoCloseStream(true)
                    .registerWriteHandler(new DynamicDropDownHandler(applicationContext))
                    .sheet("Sheet")
                    .doWrite(Collections.singletonList(Collections.emptyList())); 
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

实体类

java 复制代码
public class ImportVo {
	@ExcelProperty(value = "动态数据下拉")
    @ExcelDropDown(sourceMethod = "getNameList")
    private String name;
    @ExcelProperty(value = "性别")
    @ExcelDropDown(source = {"男","女"})
    private String sex;
}
相关推荐
烁34738 分钟前
每日一题(小白)模拟娱乐篇13
java·算法·娱乐·暴力
2301_7930698238 分钟前
前后端分离下,Spring Boot 请求从发起到响应的完整执行流程
java·spring boot·mvc
褚瑱琅42 分钟前
T-SQL语言的压力测试
开发语言·后端·golang
烁3471 小时前
每日一题(小白)模拟娱乐篇14
java·开发语言·算法·娱乐·暴力
xiaolingting3 小时前
Java 二叉树非递归遍历核心实现
java··二叉树非递归遍历
嘵奇3 小时前
深入解析 Java 8 Function 接口:函数式编程的核心工具
java·开发语言
东方靖岚4 小时前
R语言的数据库交互
开发语言·后端·golang
一路向北North5 小时前
IDEA加载项目时依赖无法更新
java·ide·intellij-idea
武昌库里写JAVA5 小时前
SpringCloud
vue.js·spring boot·毕业设计·layui·课程设计
uhakadotcom6 小时前
Python 量化计算入门:基础库和实用案例
后端·算法·面试