ExcelDownHandler.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. package com.ruoyi.common.excel;
  2. import cn.hutool.core.collection.CollUtil;
  3. import cn.hutool.core.util.ArrayUtil;
  4. import cn.hutool.core.util.EnumUtil;
  5. import cn.hutool.core.util.ObjectUtil;
  6. import cn.hutool.core.util.StrUtil;
  7. import com.alibaba.excel.annotation.ExcelProperty;
  8. import com.alibaba.excel.write.handler.SheetWriteHandler;
  9. import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
  10. import com.alibaba.excel.write.metadata.holder.WriteWorkbookHolder;
  11. import com.ruoyi.common.annotation.ExcelDictFormat;
  12. import com.ruoyi.common.annotation.ExcelEnumFormat;
  13. import com.ruoyi.common.core.service.DictService;
  14. import com.ruoyi.common.exception.ServiceException;
  15. import com.ruoyi.common.utils.StreamUtils;
  16. import com.ruoyi.common.utils.spring.SpringUtils;
  17. import lombok.extern.slf4j.Slf4j;
  18. import org.apache.poi.ss.usermodel.*;
  19. import org.apache.poi.ss.util.CellRangeAddressList;
  20. import org.apache.poi.ss.util.WorkbookUtil;
  21. import org.apache.poi.xssf.usermodel.XSSFDataValidation;
  22. import java.lang.reflect.Field;
  23. import java.util.*;
  24. /**
  25. * <h1>Excel表格下拉选操作</h1>
  26. * 考虑到下拉选过多可能导致Excel打开缓慢的问题,只校验前1000行
  27. * <p>
  28. * 即只有前1000行的数据可以用下拉框,超出的自行通过限制数据量的形式,第二次输出
  29. *
  30. * @author Emil.Zhang
  31. */
  32. @Slf4j
  33. public class ExcelDownHandler implements SheetWriteHandler {
  34. /**
  35. * Excel表格中的列名英文
  36. * 仅为了解析列英文,禁止修改
  37. */
  38. private static final String EXCEL_COLUMN_NAME = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
  39. /**
  40. * 单选数据Sheet名
  41. */
  42. private static final String OPTIONS_SHEET_NAME = "options";
  43. /**
  44. * 联动选择数据Sheet名的头
  45. */
  46. private static final String LINKED_OPTIONS_SHEET_NAME = "linkedOptions";
  47. /**
  48. * 下拉可选项
  49. */
  50. private final List<DropDownOptions> dropDownOptions;
  51. /**
  52. * 当前单选进度
  53. */
  54. private int currentOptionsColumnIndex;
  55. /**
  56. * 当前联动选择进度
  57. */
  58. private int currentLinkedOptionsSheetIndex;
  59. private final DictService dictService;
  60. public ExcelDownHandler(List<DropDownOptions> options) {
  61. this.dropDownOptions = options;
  62. this.currentOptionsColumnIndex = 0;
  63. this.currentLinkedOptionsSheetIndex = 0;
  64. this.dictService = SpringUtils.getBean(DictService.class);
  65. }
  66. /**
  67. * <h2>开始创建下拉数据</h2>
  68. * 1.通过解析传入的@ExcelProperty同级是否标注有@DropDown选项
  69. * 如果有且设置了value值,则将其直接置为下拉可选项
  70. * <p>
  71. * 2.或者在调用ExcelUtil时指定了可选项,将依据传入的可选项做下拉
  72. * <p>
  73. * 3.二者并存,注意调用方式
  74. */
  75. @Override
  76. public void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {
  77. Sheet sheet = writeSheetHolder.getSheet();
  78. // 开始设置下拉框 HSSFWorkbook
  79. DataValidationHelper helper = sheet.getDataValidationHelper();
  80. Field[] fields = writeWorkbookHolder.getClazz().getDeclaredFields();
  81. Workbook workbook = writeWorkbookHolder.getWorkbook();
  82. int length = fields.length;
  83. for (int i = 0; i < length; i++) {
  84. // 循环实体中的每个属性
  85. // 可选的下拉值
  86. List<String> options = new ArrayList<>();
  87. if (fields[i].isAnnotationPresent(ExcelDictFormat.class)) {
  88. // 如果指定了@ExcelDictFormat,则使用字典的逻辑
  89. ExcelDictFormat format = fields[i].getDeclaredAnnotation(ExcelDictFormat.class);
  90. String dictType = format.dictType();
  91. String converterExp = format.readConverterExp();
  92. if (StrUtil.isNotBlank(dictType)) {
  93. // 如果传递了字典名,则依据字典建立下拉
  94. Collection<String> values = Optional.ofNullable(dictService.getAllDictByDictType(dictType))
  95. .orElseThrow(() -> new ServiceException(String.format("字典 %s 不存在", dictType)))
  96. .values();
  97. options = new ArrayList<>(values);
  98. } else if (StrUtil.isNotBlank(converterExp)) {
  99. // 如果指定了确切的值,则直接解析确切的值
  100. options = StrUtil.split(converterExp, format.separator(), true, true);
  101. }
  102. } else if (fields[i].isAnnotationPresent(ExcelEnumFormat.class)) {
  103. // 否则如果指定了@ExcelEnumFormat,则使用枚举的逻辑
  104. ExcelEnumFormat format = fields[i].getDeclaredAnnotation(ExcelEnumFormat.class);
  105. List<Object> values = EnumUtil.getFieldValues(format.enumClass(), format.textField());
  106. options = StreamUtils.toList(values, String::valueOf);
  107. }
  108. if (ObjectUtil.isNotEmpty(options)) {
  109. // 仅当下拉可选项不为空时执行
  110. // 获取列下标,默认为当前循环次数
  111. int index = i;
  112. if (fields[i].isAnnotationPresent(ExcelProperty.class)) {
  113. // 如果指定了列下标,以指定的为主
  114. index = fields[i].getDeclaredAnnotation(ExcelProperty.class).index();
  115. }
  116. if (options.size() > 20) {
  117. // 这里限制如果可选项大于20,则使用额外表形式
  118. dropDownWithSheet(helper, workbook, sheet, index, options);
  119. } else {
  120. // 否则使用固定值形式
  121. dropDownWithSimple(helper, sheet, index, options);
  122. }
  123. }
  124. }
  125. if (CollUtil.isEmpty(dropDownOptions)) {
  126. return;
  127. }
  128. dropDownOptions.forEach(everyOptions -> {
  129. // 如果传递了下拉框选择器参数
  130. if (!everyOptions.getNextOptions().isEmpty()) {
  131. // 当二级选项不为空时,使用额外关联表的形式
  132. dropDownLinkedOptions(helper, workbook, sheet, everyOptions);
  133. } else if (everyOptions.getOptions().size() > 10) {
  134. // 当一级选项参数个数大于10,使用额外表的形式
  135. dropDownWithSheet(helper, workbook, sheet, everyOptions.getIndex(), everyOptions.getOptions());
  136. } else if (everyOptions.getOptions().size() != 0) {
  137. // 当一级选项个数不为空,使用默认形式
  138. dropDownWithSimple(helper, sheet, everyOptions.getIndex(), everyOptions.getOptions());
  139. }
  140. });
  141. }
  142. /**
  143. * <h2>简单下拉框</h2>
  144. * 直接将可选项拼接为指定列的数据校验值
  145. *
  146. * @param celIndex 列index
  147. * @param value 下拉选可选值
  148. */
  149. private void dropDownWithSimple(DataValidationHelper helper, Sheet sheet, Integer celIndex, List<String> value) {
  150. if (ObjectUtil.isEmpty(value)) {
  151. return;
  152. }
  153. this.markOptionsToSheet(helper, sheet, celIndex, helper.createExplicitListConstraint(ArrayUtil.toArray(value, String.class)));
  154. }
  155. /**
  156. * <h2>额外表格形式的级联下拉框</h2>
  157. *
  158. * @param options 额外表格形式存储的下拉可选项
  159. */
  160. private void dropDownLinkedOptions(DataValidationHelper helper, Workbook workbook, Sheet sheet, DropDownOptions options) {
  161. String linkedOptionsSheetName = String.format("%s_%d", LINKED_OPTIONS_SHEET_NAME, currentLinkedOptionsSheetIndex);
  162. // 创建联动下拉数据表
  163. Sheet linkedOptionsDataSheet = workbook.createSheet(WorkbookUtil.createSafeSheetName(linkedOptionsSheetName));
  164. // 将下拉表隐藏
  165. workbook.setSheetHidden(workbook.getSheetIndex(linkedOptionsDataSheet), true);
  166. // 完善横向的一级选项数据表
  167. List<String> firstOptions = options.getOptions();
  168. Map<String, List<String>> secoundOptionsMap = options.getNextOptions();
  169. // 创建名称管理器
  170. Name name = workbook.createName();
  171. // 设置名称管理器的别名
  172. name.setNameName(linkedOptionsSheetName);
  173. // 以横向第一行创建一级下拉拼接引用位置
  174. String firstOptionsFunction = String.format("%s!$%s$1:$%s$1",
  175. linkedOptionsSheetName,
  176. getExcelColumnName(0),
  177. getExcelColumnName(firstOptions.size())
  178. );
  179. // 设置名称管理器的引用位置
  180. name.setRefersToFormula(firstOptionsFunction);
  181. // 设置数据校验为序列模式,引用的是名称管理器中的别名
  182. this.markOptionsToSheet(helper, sheet, options.getIndex(), helper.createFormulaListConstraint(linkedOptionsSheetName));
  183. for (int columIndex = 0; columIndex < firstOptions.size(); columIndex++) {
  184. // 先提取主表中一级下拉的列名
  185. String firstOptionsColumnName = getExcelColumnName(columIndex);
  186. // 一次循环是每一个一级选项
  187. int finalI = columIndex;
  188. // 本次循环的一级选项值
  189. String thisFirstOptionsValue = firstOptions.get(columIndex);
  190. // 创建第一行的数据
  191. Optional.ofNullable(linkedOptionsDataSheet.getRow(0))
  192. // 如果不存在则创建第一行
  193. .orElseGet(() -> linkedOptionsDataSheet.createRow(finalI))
  194. // 第一行当前列
  195. .createCell(columIndex)
  196. // 设置值为当前一级选项值
  197. .setCellValue(thisFirstOptionsValue);
  198. // 第二行开始,设置第二级别选项参数
  199. List<String> secondOptions = secoundOptionsMap.get(thisFirstOptionsValue);
  200. if (CollUtil.isEmpty(secondOptions)) {
  201. // 必须保证至少有一个关联选项,否则将导致Excel解析错误
  202. secondOptions = Collections.singletonList("暂无_0");
  203. }
  204. // 以该一级选项值创建子名称管理器
  205. Name sonName = workbook.createName();
  206. // 设置名称管理器的别名
  207. sonName.setNameName(thisFirstOptionsValue);
  208. // 以第二行该列数据拼接引用位置
  209. String sonFunction = String.format("%s!$%s$2:$%s$%d",
  210. linkedOptionsSheetName,
  211. firstOptionsColumnName,
  212. firstOptionsColumnName,
  213. secondOptions.size() + 1
  214. );
  215. // 设置名称管理器的引用位置
  216. sonName.setRefersToFormula(sonFunction);
  217. // 数据验证为序列模式,引用到每一个主表中的二级选项位置
  218. // 创建子项的名称管理器,只是为了使得Excel可以识别到数据
  219. String mainSheetFirstOptionsColumnName = getExcelColumnName(options.getIndex());
  220. for (int i = 0; i < 100; i++) {
  221. // 以一级选项对应的主体所在位置创建二级下拉
  222. String secondOptionsFunction = String.format("=INDIRECT(%s%d)", mainSheetFirstOptionsColumnName, i + 1);
  223. // 二级只能主表每一行的每一列添加二级校验
  224. markLinkedOptionsToSheet(helper, sheet, i, options.getNextIndex(), helper.createFormulaListConstraint(secondOptionsFunction));
  225. }
  226. for (int rowIndex = 0; rowIndex < secondOptions.size(); rowIndex++) {
  227. // 从第二行开始填充二级选项
  228. int finalRowIndex = rowIndex + 1;
  229. int finalColumIndex = columIndex;
  230. Row row = Optional.ofNullable(linkedOptionsDataSheet.getRow(finalRowIndex))
  231. // 没有则创建
  232. .orElseGet(() -> linkedOptionsDataSheet.createRow(finalRowIndex));
  233. Optional
  234. // 在本级一级选项所在的列
  235. .ofNullable(row.getCell(finalColumIndex))
  236. // 不存在则创建
  237. .orElseGet(() -> row.createCell(finalColumIndex))
  238. // 设置二级选项值
  239. .setCellValue(secondOptions.get(rowIndex));
  240. }
  241. }
  242. currentLinkedOptionsSheetIndex++;
  243. }
  244. /**
  245. * <h2>额外表格形式的普通下拉框</h2>
  246. * 由于下拉框可选值数量过多,为提升Excel打开效率,使用额外表格形式做下拉
  247. *
  248. * @param celIndex 下拉选
  249. * @param value 下拉选可选值
  250. */
  251. private void dropDownWithSheet(DataValidationHelper helper, Workbook workbook, Sheet sheet, Integer celIndex, List<String> value) {
  252. // 创建下拉数据表
  253. Sheet simpleDataSheet = Optional.ofNullable(workbook.getSheet(WorkbookUtil.createSafeSheetName(OPTIONS_SHEET_NAME)))
  254. .orElseGet(() -> workbook.createSheet(WorkbookUtil.createSafeSheetName(OPTIONS_SHEET_NAME)));
  255. // 将下拉表隐藏
  256. workbook.setSheetHidden(workbook.getSheetIndex(simpleDataSheet), true);
  257. // 完善纵向的一级选项数据表
  258. for (int i = 0; i < value.size(); i++) {
  259. int finalI = i;
  260. // 获取每一选项行,如果没有则创建
  261. Row row = Optional.ofNullable(simpleDataSheet.getRow(i))
  262. .orElseGet(() -> simpleDataSheet.createRow(finalI));
  263. // 获取本级选项对应的选项列,如果没有则创建
  264. Cell cell = Optional.ofNullable(row.getCell(currentOptionsColumnIndex))
  265. .orElseGet(() -> row.createCell(currentOptionsColumnIndex));
  266. // 设置值
  267. cell.setCellValue(value.get(i));
  268. }
  269. // 创建名称管理器
  270. Name name = workbook.createName();
  271. // 设置名称管理器的别名
  272. String nameName = String.format("%s_%d", OPTIONS_SHEET_NAME, celIndex);
  273. name.setNameName(nameName);
  274. // 以纵向第一列创建一级下拉拼接引用位置
  275. String function = String.format("%s!$%s$1:$%s$%d",
  276. OPTIONS_SHEET_NAME,
  277. getExcelColumnName(currentOptionsColumnIndex),
  278. getExcelColumnName(currentOptionsColumnIndex),
  279. value.size());
  280. // 设置名称管理器的引用位置
  281. name.setRefersToFormula(function);
  282. // 设置数据校验为序列模式,引用的是名称管理器中的别名
  283. this.markOptionsToSheet(helper, sheet, celIndex, helper.createFormulaListConstraint(nameName));
  284. currentOptionsColumnIndex++;
  285. }
  286. /**
  287. * 挂载下拉的列,仅限一级选项
  288. */
  289. private void markOptionsToSheet(DataValidationHelper helper, Sheet sheet, Integer celIndex,
  290. DataValidationConstraint constraint) {
  291. // 设置数据有效性加载在哪个单元格上,四个参数分别是:起始行、终止行、起始列、终止列
  292. CellRangeAddressList addressList = new CellRangeAddressList(1, 1000, celIndex, celIndex);
  293. markDataValidationToSheet(helper, sheet, constraint, addressList);
  294. }
  295. /**
  296. * 挂载下拉的列,仅限二级选项
  297. */
  298. private void markLinkedOptionsToSheet(DataValidationHelper helper, Sheet sheet, Integer rowIndex,
  299. Integer celIndex, DataValidationConstraint constraint) {
  300. // 设置数据有效性加载在哪个单元格上,四个参数分别是:起始行、终止行、起始列、终止列
  301. CellRangeAddressList addressList = new CellRangeAddressList(rowIndex, rowIndex, celIndex, celIndex);
  302. markDataValidationToSheet(helper, sheet, constraint, addressList);
  303. }
  304. /**
  305. * 应用数据校验
  306. */
  307. private void markDataValidationToSheet(DataValidationHelper helper, Sheet sheet,
  308. DataValidationConstraint constraint, CellRangeAddressList addressList) {
  309. // 数据有效性对象
  310. DataValidation dataValidation = helper.createValidation(constraint, addressList);
  311. // 处理Excel兼容性问题
  312. if (dataValidation instanceof XSSFDataValidation) {
  313. //数据校验
  314. dataValidation.setSuppressDropDownArrow(true);
  315. //错误提示
  316. dataValidation.setErrorStyle(DataValidation.ErrorStyle.STOP);
  317. dataValidation.createErrorBox("提示", "此值与单元格定义数据不一致");
  318. dataValidation.setShowErrorBox(true);
  319. //选定提示
  320. dataValidation.createPromptBox("填写说明:", "填写内容只能为下拉中数据,其他数据将导致导入失败");
  321. dataValidation.setShowPromptBox(true);
  322. sheet.addValidationData(dataValidation);
  323. } else {
  324. dataValidation.setSuppressDropDownArrow(false);
  325. }
  326. sheet.addValidationData(dataValidation);
  327. }
  328. /**
  329. * <h2>依据列index获取列名英文</h2>
  330. * 依据列index转换为Excel中的列名英文
  331. * <p>例如第1列,index为0,解析出来为A列</p>
  332. * 第27列,index为26,解析为AA列
  333. * <p>第28列,index为27,解析为AB列</p>
  334. *
  335. * @param columnIndex 列index
  336. * @return 列index所在得英文名
  337. */
  338. private String getExcelColumnName(int columnIndex) {
  339. // 26一循环的次数
  340. int columnCircleCount = columnIndex / 26;
  341. // 26一循环内的位置
  342. int thisCircleColumnIndex = columnIndex % 26;
  343. // 26一循环的次数大于0,则视为栏名至少两位
  344. String columnPrefix = columnCircleCount == 0
  345. ? StrUtil.EMPTY
  346. : StrUtil.subWithLength(EXCEL_COLUMN_NAME, columnCircleCount - 1, 1);
  347. // 从26一循环内取对应的栏位名
  348. String columnNext = StrUtil.subWithLength(EXCEL_COLUMN_NAME, thisCircleColumnIndex, 1);
  349. // 将二者拼接即为最终的栏位名
  350. return columnPrefix + columnNext;
  351. }
  352. }