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;
}
相关推荐
程序员曼布2 分钟前
RabbitMQ 深度解析:从核心组件到复杂应用场景
java·开发语言·后端·rabbitmq
n33(NK)13 分钟前
【算法基础】插入排序算法 - JAVA
java·数据结构·算法·排序算法
liaokailin17 分钟前
Spring AI 实战:第九章、Spring AI MCP之万站直通
java·人工智能·spring
帅得不敢出门19 分钟前
Android Framework学习三:zygote剖析
android·java·学习·framework·安卓·zygote
记得开心一点嘛1 小时前
Rockermq的部署与使用(0-1)
java·rocketmq
大G哥1 小时前
用 Java 和 DL4J 实现验证码识别系统
java·linux·开发语言·前端·python
Allen Bright2 小时前
【Java JUnit单元测试框架-60】深入理解JUnit:Java单元测试的艺术与实践
java·junit·单元测试
武昌库里写JAVA2 小时前
iview table组件 自定义表头
vue.js·spring boot·毕业设计·layui·课程设计
武昌库里写JAVA2 小时前
iview 分页改变每页条数时请求两次问题
vue.js·spring boot·毕业设计·layui·课程设计
琢磨先生David3 小时前
深入探索 Java 区块链技术:从核心原理到企业级实践
java·区块链