ExcelUtil.java 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935
  1. package com.ruoyi.common.utils.poi;
  2. import java.io.File;
  3. import java.io.FileOutputStream;
  4. import java.io.IOException;
  5. import java.io.InputStream;
  6. import java.io.OutputStream;
  7. import java.lang.reflect.Field;
  8. import java.lang.reflect.Method;
  9. import java.math.BigDecimal;
  10. import java.text.DecimalFormat;
  11. import java.util.ArrayList;
  12. import java.util.Arrays;
  13. import java.util.Comparator;
  14. import java.util.Date;
  15. import java.util.HashMap;
  16. import java.util.List;
  17. import java.util.Map;
  18. import java.util.UUID;
  19. import java.util.stream.Collectors;
  20. import org.apache.poi.hssf.usermodel.HSSFDateUtil;
  21. import org.apache.poi.ss.usermodel.BorderStyle;
  22. import org.apache.poi.ss.usermodel.Cell;
  23. import org.apache.poi.ss.usermodel.CellStyle;
  24. import org.apache.poi.ss.usermodel.CellType;
  25. import org.apache.poi.ss.usermodel.DataValidation;
  26. import org.apache.poi.ss.usermodel.DataValidationConstraint;
  27. import org.apache.poi.ss.usermodel.DataValidationHelper;
  28. import org.apache.poi.ss.usermodel.DateUtil;
  29. import org.apache.poi.ss.usermodel.FillPatternType;
  30. import org.apache.poi.ss.usermodel.Font;
  31. import org.apache.poi.ss.usermodel.HorizontalAlignment;
  32. import org.apache.poi.ss.usermodel.IndexedColors;
  33. import org.apache.poi.ss.usermodel.Row;
  34. import org.apache.poi.ss.usermodel.Sheet;
  35. import org.apache.poi.ss.usermodel.VerticalAlignment;
  36. import org.apache.poi.ss.usermodel.Workbook;
  37. import org.apache.poi.ss.usermodel.WorkbookFactory;
  38. import org.apache.poi.ss.util.CellRangeAddressList;
  39. import org.apache.poi.xssf.streaming.SXSSFWorkbook;
  40. import org.apache.poi.xssf.usermodel.XSSFDataValidation;
  41. import org.slf4j.Logger;
  42. import org.slf4j.LoggerFactory;
  43. import com.ruoyi.common.annotation.Excel;
  44. import com.ruoyi.common.annotation.Excel.ColumnType;
  45. import com.ruoyi.common.annotation.Excel.Type;
  46. import com.ruoyi.common.annotation.Excels;
  47. import com.ruoyi.common.config.RuoYiConfig;
  48. import com.ruoyi.common.core.domain.AjaxResult;
  49. import com.ruoyi.common.core.text.Convert;
  50. import com.ruoyi.common.exception.CustomException;
  51. import com.ruoyi.common.utils.DateUtils;
  52. import com.ruoyi.common.utils.DictUtils;
  53. import com.ruoyi.common.utils.StringUtils;
  54. import com.ruoyi.common.utils.reflect.ReflectUtils;
  55. /**
  56. * Excel相关处理
  57. *
  58. * @author ruoyi
  59. */
  60. public class ExcelUtil<T>
  61. {
  62. private static final Logger log = LoggerFactory.getLogger(ExcelUtil.class);
  63. /**
  64. * Excel sheet最大行数,默认65536
  65. */
  66. public static final int sheetSize = 65536;
  67. /**
  68. * 工作表名称
  69. */
  70. private String sheetName;
  71. /**
  72. * 导出类型(EXPORT:导出数据;IMPORT:导入模板)
  73. */
  74. private Type type;
  75. /**
  76. * 工作薄对象
  77. */
  78. private Workbook wb;
  79. /**
  80. * 工作表对象
  81. */
  82. private Sheet sheet;
  83. /**
  84. * 样式列表
  85. */
  86. private Map<String, CellStyle> styles;
  87. /**
  88. * 导入导出数据列表
  89. */
  90. private List<T> list;
  91. /**
  92. * 注解列表
  93. */
  94. private List<Object[]> fields;
  95. /**
  96. * 实体对象
  97. */
  98. public Class<T> clazz;
  99. public ExcelUtil(Class<T> clazz)
  100. {
  101. this.clazz = clazz;
  102. }
  103. public void init(List<T> list, String sheetName, Type type)
  104. {
  105. if (list == null)
  106. {
  107. list = new ArrayList<T>();
  108. }
  109. this.list = list;
  110. this.sheetName = sheetName;
  111. this.type = type;
  112. createExcelField();
  113. createWorkbook();
  114. }
  115. /**
  116. * 对excel表单默认第一个索引名转换成list
  117. *
  118. * @param is 输入流
  119. * @return 转换后集合
  120. */
  121. public List<T> importExcel(InputStream is) throws Exception
  122. {
  123. return importExcel(StringUtils.EMPTY, is);
  124. }
  125. /**
  126. * 对excel表单指定表格索引名转换成list
  127. *
  128. * @param sheetName 表格索引名
  129. * @param is 输入流
  130. * @return 转换后集合
  131. */
  132. public List<T> importExcel(String sheetName, InputStream is) throws Exception
  133. {
  134. this.type = Type.IMPORT;
  135. this.wb = WorkbookFactory.create(is);
  136. List<T> list = new ArrayList<T>();
  137. Sheet sheet = null;
  138. if (StringUtils.isNotEmpty(sheetName))
  139. {
  140. // 如果指定sheet名,则取指定sheet中的内容.
  141. sheet = wb.getSheet(sheetName);
  142. }
  143. else
  144. {
  145. // 如果传入的sheet名不存在则默认指向第1个sheet.
  146. sheet = wb.getSheetAt(0);
  147. }
  148. if (sheet == null)
  149. {
  150. throw new IOException("文件sheet不存在");
  151. }
  152. int rows = sheet.getPhysicalNumberOfRows();
  153. if (rows > 0)
  154. {
  155. // 定义一个map用于存放excel列的序号和field.
  156. Map<String, Integer> cellMap = new HashMap<String, Integer>();
  157. // 获取表头
  158. Row heard = sheet.getRow(0);
  159. for (int i = 0; i < heard.getPhysicalNumberOfCells(); i++)
  160. {
  161. Cell cell = heard.getCell(i);
  162. if (StringUtils.isNotNull(cell))
  163. {
  164. String value = this.getCellValue(heard, i).toString();
  165. cellMap.put(value, i);
  166. }
  167. else
  168. {
  169. cellMap.put(null, i);
  170. }
  171. }
  172. // 有数据时才处理 得到类的所有field.
  173. Field[] allFields = clazz.getDeclaredFields();
  174. // 定义一个map用于存放列的序号和field.
  175. Map<Integer, Field> fieldsMap = new HashMap<Integer, Field>();
  176. for (int col = 0; col < allFields.length; col++)
  177. {
  178. Field field = allFields[col];
  179. Excel attr = field.getAnnotation(Excel.class);
  180. if (attr != null && (attr.type() == Type.ALL || attr.type() == type))
  181. {
  182. // 设置类的私有字段属性可访问.
  183. field.setAccessible(true);
  184. Integer column = cellMap.get(attr.name());
  185. if (column != null)
  186. {
  187. fieldsMap.put(column, field);
  188. }
  189. }
  190. }
  191. for (int i = 1; i < rows; i++)
  192. {
  193. // 从第2行开始取数据,默认第一行是表头.
  194. Row row = sheet.getRow(i);
  195. T entity = null;
  196. for (Map.Entry<Integer, Field> entry : fieldsMap.entrySet())
  197. {
  198. Object val = this.getCellValue(row, entry.getKey());
  199. // 如果不存在实例则新建.
  200. entity = (entity == null ? clazz.newInstance() : entity);
  201. // 从map中得到对应列的field.
  202. Field field = fieldsMap.get(entry.getKey());
  203. // 取得类型,并根据对象类型设置值.
  204. Class<?> fieldType = field.getType();
  205. if (String.class == fieldType)
  206. {
  207. String s = Convert.toStr(val);
  208. if (StringUtils.endsWith(s, ".0"))
  209. {
  210. val = StringUtils.substringBefore(s, ".0");
  211. }
  212. else
  213. {
  214. val = Convert.toStr(val);
  215. }
  216. }
  217. else if ((Integer.TYPE == fieldType) || (Integer.class == fieldType))
  218. {
  219. val = Convert.toInt(val);
  220. }
  221. else if ((Long.TYPE == fieldType) || (Long.class == fieldType))
  222. {
  223. val = Convert.toLong(val);
  224. }
  225. else if ((Double.TYPE == fieldType) || (Double.class == fieldType))
  226. {
  227. val = Convert.toDouble(val);
  228. }
  229. else if ((Float.TYPE == fieldType) || (Float.class == fieldType))
  230. {
  231. val = Convert.toFloat(val);
  232. }
  233. else if (BigDecimal.class == fieldType)
  234. {
  235. val = Convert.toBigDecimal(val);
  236. }
  237. else if (Date.class == fieldType)
  238. {
  239. if (val instanceof String)
  240. {
  241. val = DateUtils.parseDate(val);
  242. }
  243. else if (val instanceof Double)
  244. {
  245. val = DateUtil.getJavaDate((Double) val);
  246. }
  247. }
  248. if (StringUtils.isNotNull(fieldType))
  249. {
  250. Excel attr = field.getAnnotation(Excel.class);
  251. String propertyName = field.getName();
  252. if (StringUtils.isNotEmpty(attr.targetAttr()))
  253. {
  254. propertyName = field.getName() + "." + attr.targetAttr();
  255. }
  256. else if (StringUtils.isNotEmpty(attr.readConverterExp()))
  257. {
  258. val = reverseByExp(Convert.toStr(val), attr.readConverterExp(), attr.separator());
  259. }
  260. else if (StringUtils.isNotEmpty(attr.dictType()))
  261. {
  262. val = reverseDictByExp(Convert.toStr(val), attr.dictType(), attr.separator());
  263. }
  264. ReflectUtils.invokeSetter(entity, propertyName, val);
  265. }
  266. }
  267. list.add(entity);
  268. }
  269. }
  270. return list;
  271. }
  272. /**
  273. * 对list数据源将其里面的数据导入到excel表单
  274. *
  275. * @param list 导出数据集合
  276. * @param sheetName 工作表的名称
  277. * @return 结果
  278. */
  279. public AjaxResult exportExcel(List<T> list, String sheetName)
  280. {
  281. this.init(list, sheetName, Type.EXPORT);
  282. return exportExcel();
  283. }
  284. /**
  285. * 对list数据源将其里面的数据导入到excel表单
  286. *
  287. * @param sheetName 工作表的名称
  288. * @return 结果
  289. */
  290. public AjaxResult importTemplateExcel(String sheetName)
  291. {
  292. this.init(null, sheetName, Type.IMPORT);
  293. return exportExcel();
  294. }
  295. /**
  296. * 对list数据源将其里面的数据导入到excel表单
  297. *
  298. * @return 结果
  299. */
  300. public AjaxResult exportExcel()
  301. {
  302. OutputStream out = null;
  303. try
  304. {
  305. // 取出一共有多少个sheet.
  306. double sheetNo = Math.ceil(list.size() / sheetSize);
  307. for (int index = 0; index <= sheetNo; index++)
  308. {
  309. createSheet(sheetNo, index);
  310. // 产生一行
  311. Row row = sheet.createRow(0);
  312. int column = 0;
  313. // 写入各个字段的列头名称
  314. for (Object[] os : fields)
  315. {
  316. Excel excel = (Excel) os[1];
  317. this.createCell(excel, row, column++);
  318. }
  319. if (Type.EXPORT.equals(type))
  320. {
  321. fillExcelData(index, row);
  322. }
  323. }
  324. String filename = encodingFilename(sheetName);
  325. out = new FileOutputStream(getAbsoluteFile(filename));
  326. wb.write(out);
  327. return AjaxResult.success(filename);
  328. }
  329. catch (Exception e)
  330. {
  331. log.error("导出Excel异常{}", e.getMessage());
  332. throw new CustomException("导出Excel失败,请联系网站管理员!");
  333. }
  334. finally
  335. {
  336. if (wb != null)
  337. {
  338. try
  339. {
  340. wb.close();
  341. }
  342. catch (IOException e1)
  343. {
  344. e1.printStackTrace();
  345. }
  346. }
  347. if (out != null)
  348. {
  349. try
  350. {
  351. out.close();
  352. }
  353. catch (IOException e1)
  354. {
  355. e1.printStackTrace();
  356. }
  357. }
  358. }
  359. }
  360. /**
  361. * 填充excel数据
  362. *
  363. * @param index 序号
  364. * @param row 单元格行
  365. */
  366. public void fillExcelData(int index, Row row)
  367. {
  368. int startNo = index * sheetSize;
  369. int endNo = Math.min(startNo + sheetSize, list.size());
  370. for (int i = startNo; i < endNo; i++)
  371. {
  372. row = sheet.createRow(i + 1 - startNo);
  373. // 得到导出对象.
  374. T vo = (T) list.get(i);
  375. int column = 0;
  376. for (Object[] os : fields)
  377. {
  378. Field field = (Field) os[0];
  379. Excel excel = (Excel) os[1];
  380. // 设置实体类私有属性可访问
  381. field.setAccessible(true);
  382. this.addCell(excel, row, vo, field, column++);
  383. }
  384. }
  385. }
  386. /**
  387. * 创建表格样式
  388. *
  389. * @param wb 工作薄对象
  390. * @return 样式列表
  391. */
  392. private Map<String, CellStyle> createStyles(Workbook wb)
  393. {
  394. // 写入各条记录,每条记录对应excel表中的一行
  395. Map<String, CellStyle> styles = new HashMap<String, CellStyle>();
  396. CellStyle style = wb.createCellStyle();
  397. style.setAlignment(HorizontalAlignment.CENTER);
  398. style.setVerticalAlignment(VerticalAlignment.CENTER);
  399. style.setBorderRight(BorderStyle.THIN);
  400. style.setRightBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
  401. style.setBorderLeft(BorderStyle.THIN);
  402. style.setLeftBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
  403. style.setBorderTop(BorderStyle.THIN);
  404. style.setTopBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
  405. style.setBorderBottom(BorderStyle.THIN);
  406. style.setBottomBorderColor(IndexedColors.GREY_50_PERCENT.getIndex());
  407. Font dataFont = wb.createFont();
  408. dataFont.setFontName("Arial");
  409. dataFont.setFontHeightInPoints((short) 10);
  410. style.setFont(dataFont);
  411. styles.put("data", style);
  412. style = wb.createCellStyle();
  413. style.cloneStyleFrom(styles.get("data"));
  414. style.setAlignment(HorizontalAlignment.CENTER);
  415. style.setVerticalAlignment(VerticalAlignment.CENTER);
  416. style.setFillForegroundColor(IndexedColors.GREY_50_PERCENT.getIndex());
  417. style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
  418. Font headerFont = wb.createFont();
  419. headerFont.setFontName("Arial");
  420. headerFont.setFontHeightInPoints((short) 10);
  421. headerFont.setBold(true);
  422. headerFont.setColor(IndexedColors.WHITE.getIndex());
  423. style.setFont(headerFont);
  424. styles.put("header", style);
  425. return styles;
  426. }
  427. /**
  428. * 创建单元格
  429. */
  430. public Cell createCell(Excel attr, Row row, int column)
  431. {
  432. // 创建列
  433. Cell cell = row.createCell(column);
  434. // 写入列信息
  435. cell.setCellValue(attr.name());
  436. setDataValidation(attr, row, column);
  437. cell.setCellStyle(styles.get("header"));
  438. return cell;
  439. }
  440. /**
  441. * 设置单元格信息
  442. *
  443. * @param value 单元格值
  444. * @param attr 注解相关
  445. * @param cell 单元格信息
  446. */
  447. public void setCellVo(Object value, Excel attr, Cell cell)
  448. {
  449. if (ColumnType.STRING == attr.cellType())
  450. {
  451. cell.setCellType(CellType.STRING);
  452. cell.setCellValue(StringUtils.isNull(value) ? attr.defaultValue() : value + attr.suffix());
  453. }
  454. else if (ColumnType.NUMERIC == attr.cellType())
  455. {
  456. cell.setCellType(CellType.NUMERIC);
  457. cell.setCellValue(Integer.parseInt(value + ""));
  458. }
  459. }
  460. /**
  461. * 创建表格样式
  462. */
  463. public void setDataValidation(Excel attr, Row row, int column)
  464. {
  465. if (attr.name().indexOf("注:") >= 0)
  466. {
  467. sheet.setColumnWidth(column, 6000);
  468. }
  469. else
  470. {
  471. // 设置列宽
  472. sheet.setColumnWidth(column, (int) ((attr.width() + 0.72) * 256));
  473. row.setHeight((short) (attr.height() * 20));
  474. }
  475. // 如果设置了提示信息则鼠标放上去提示.
  476. if (StringUtils.isNotEmpty(attr.prompt()))
  477. {
  478. // 这里默认设了2-101列提示.
  479. setXSSFPrompt(sheet, "", attr.prompt(), 1, 100, column, column);
  480. }
  481. // 如果设置了combo属性则本列只能选择不能输入
  482. if (attr.combo().length > 0)
  483. {
  484. // 这里默认设了2-101列只能选择不能输入.
  485. setXSSFValidation(sheet, attr.combo(), 1, 100, column, column);
  486. }
  487. }
  488. /**
  489. * 添加单元格
  490. */
  491. public Cell addCell(Excel attr, Row row, T vo, Field field, int column)
  492. {
  493. Cell cell = null;
  494. try
  495. {
  496. // 设置行高
  497. row.setHeight((short) (attr.height() * 20));
  498. // 根据Excel中设置情况决定是否导出,有些情况需要保持为空,希望用户填写这一列.
  499. if (attr.isExport())
  500. {
  501. // 创建cell
  502. cell = row.createCell(column);
  503. cell.setCellStyle(styles.get("data"));
  504. // 用于读取对象中的属性
  505. Object value = getTargetValue(vo, field, attr);
  506. String dateFormat = attr.dateFormat();
  507. String readConverterExp = attr.readConverterExp();
  508. String separator = attr.separator();
  509. String dictType = attr.dictType();
  510. if (StringUtils.isNotEmpty(dateFormat) && StringUtils.isNotNull(value))
  511. {
  512. cell.setCellValue(DateUtils.parseDateToStr(dateFormat, (Date) value));
  513. }
  514. else if (StringUtils.isNotEmpty(readConverterExp) && StringUtils.isNotNull(value))
  515. {
  516. cell.setCellValue(convertByExp(Convert.toStr(value), readConverterExp, separator));
  517. }
  518. else if (StringUtils.isNotEmpty(dictType) && StringUtils.isNotNull(value))
  519. {
  520. cell.setCellValue(convertDictByExp(Convert.toStr(value), dictType, separator));
  521. }
  522. else if (value instanceof BigDecimal && -1 != attr.scale())
  523. {
  524. cell.setCellValue((((BigDecimal) value).setScale(attr.scale(), attr.roundingMode())).toString());
  525. }
  526. else
  527. {
  528. // 设置列类型
  529. setCellVo(value, attr, cell);
  530. }
  531. }
  532. }
  533. catch (Exception e)
  534. {
  535. log.error("导出Excel失败{}", e);
  536. }
  537. return cell;
  538. }
  539. /**
  540. * 设置 POI XSSFSheet 单元格提示
  541. *
  542. * @param sheet 表单
  543. * @param promptTitle 提示标题
  544. * @param promptContent 提示内容
  545. * @param firstRow 开始行
  546. * @param endRow 结束行
  547. * @param firstCol 开始列
  548. * @param endCol 结束列
  549. */
  550. public void setXSSFPrompt(Sheet sheet, String promptTitle, String promptContent, int firstRow, int endRow,
  551. int firstCol, int endCol)
  552. {
  553. DataValidationHelper helper = sheet.getDataValidationHelper();
  554. DataValidationConstraint constraint = helper.createCustomConstraint("DD1");
  555. CellRangeAddressList regions = new CellRangeAddressList(firstRow, endRow, firstCol, endCol);
  556. DataValidation dataValidation = helper.createValidation(constraint, regions);
  557. dataValidation.createPromptBox(promptTitle, promptContent);
  558. dataValidation.setShowPromptBox(true);
  559. sheet.addValidationData(dataValidation);
  560. }
  561. /**
  562. * 设置某些列的值只能输入预制的数据,显示下拉框.
  563. *
  564. * @param sheet 要设置的sheet.
  565. * @param textlist 下拉框显示的内容
  566. * @param firstRow 开始行
  567. * @param endRow 结束行
  568. * @param firstCol 开始列
  569. * @param endCol 结束列
  570. * @return 设置好的sheet.
  571. */
  572. public void setXSSFValidation(Sheet sheet, String[] textlist, int firstRow, int endRow, int firstCol, int endCol)
  573. {
  574. DataValidationHelper helper = sheet.getDataValidationHelper();
  575. // 加载下拉列表内容
  576. DataValidationConstraint constraint = helper.createExplicitListConstraint(textlist);
  577. // 设置数据有效性加载在哪个单元格上,四个参数分别是:起始行、终止行、起始列、终止列
  578. CellRangeAddressList regions = new CellRangeAddressList(firstRow, endRow, firstCol, endCol);
  579. // 数据有效性对象
  580. DataValidation dataValidation = helper.createValidation(constraint, regions);
  581. // 处理Excel兼容性问题
  582. if (dataValidation instanceof XSSFDataValidation)
  583. {
  584. dataValidation.setSuppressDropDownArrow(true);
  585. dataValidation.setShowErrorBox(true);
  586. }
  587. else
  588. {
  589. dataValidation.setSuppressDropDownArrow(false);
  590. }
  591. sheet.addValidationData(dataValidation);
  592. }
  593. /**
  594. * 解析导出值 0=男,1=女,2=未知
  595. *
  596. * @param propertyValue 参数值
  597. * @param converterExp 翻译注解
  598. * @param separator 分隔符
  599. * @return 解析后值
  600. */
  601. public static String convertByExp(String propertyValue, String converterExp, String separator)
  602. {
  603. StringBuilder propertyString = new StringBuilder();
  604. String[] convertSource = converterExp.split(",");
  605. for (String item : convertSource)
  606. {
  607. String[] itemArray = item.split("=");
  608. if (StringUtils.containsAny(separator, propertyValue))
  609. {
  610. for (String value : propertyValue.split(separator))
  611. {
  612. if (itemArray[0].equals(value))
  613. {
  614. propertyString.append(itemArray[1] + separator);
  615. break;
  616. }
  617. }
  618. }
  619. else
  620. {
  621. if (itemArray[0].equals(propertyValue))
  622. {
  623. return itemArray[1];
  624. }
  625. }
  626. }
  627. return StringUtils.stripEnd(propertyString.toString(), separator);
  628. }
  629. /**
  630. * 反向解析值 男=0,女=1,未知=2
  631. *
  632. * @param propertyValue 参数值
  633. * @param converterExp 翻译注解
  634. * @param separator 分隔符
  635. * @return 解析后值
  636. */
  637. public static String reverseByExp(String propertyValue, String converterExp, String separator)
  638. {
  639. StringBuilder propertyString = new StringBuilder();
  640. String[] convertSource = converterExp.split(",");
  641. for (String item : convertSource)
  642. {
  643. String[] itemArray = item.split("=");
  644. if (StringUtils.containsAny(separator, propertyValue))
  645. {
  646. for (String value : propertyValue.split(separator))
  647. {
  648. if (itemArray[1].equals(value))
  649. {
  650. propertyString.append(itemArray[0] + separator);
  651. break;
  652. }
  653. }
  654. }
  655. else
  656. {
  657. if (itemArray[1].equals(propertyValue))
  658. {
  659. return itemArray[0];
  660. }
  661. }
  662. }
  663. return StringUtils.stripEnd(propertyString.toString(), separator);
  664. }
  665. /**
  666. * 解析字典值
  667. *
  668. * @param dictValue 字典值
  669. * @param dictType 字典类型
  670. * @param separator 分隔符
  671. * @return 字典标签
  672. */
  673. public static String convertDictByExp(String dictValue, String dictType, String separator)
  674. {
  675. return DictUtils.getDictLabel(dictType, dictValue, separator);
  676. }
  677. /**
  678. * 反向解析值字典值
  679. *
  680. * @param dictLabel 字典标签
  681. * @param dictType 字典类型
  682. * @param separator 分隔符
  683. * @return 字典值
  684. */
  685. public static String reverseDictByExp(String dictLabel, String dictType, String separator)
  686. {
  687. return DictUtils.getDictValue(dictType, dictLabel, separator);
  688. }
  689. /**
  690. * 编码文件名
  691. */
  692. public String encodingFilename(String filename)
  693. {
  694. filename = UUID.randomUUID().toString() + "_" + filename + ".xlsx";
  695. return filename;
  696. }
  697. /**
  698. * 获取下载路径
  699. *
  700. * @param filename 文件名称
  701. */
  702. public String getAbsoluteFile(String filename)
  703. {
  704. String downloadPath = RuoYiConfig.getDownloadPath() + filename;
  705. File desc = new File(downloadPath);
  706. if (!desc.getParentFile().exists())
  707. {
  708. desc.getParentFile().mkdirs();
  709. }
  710. return downloadPath;
  711. }
  712. /**
  713. * 获取bean中的属性值
  714. *
  715. * @param vo 实体对象
  716. * @param field 字段
  717. * @param excel 注解
  718. * @return 最终的属性值
  719. * @throws Exception
  720. */
  721. private Object getTargetValue(T vo, Field field, Excel excel) throws Exception
  722. {
  723. Object o = field.get(vo);
  724. if (StringUtils.isNotEmpty(excel.targetAttr()))
  725. {
  726. String target = excel.targetAttr();
  727. if (target.indexOf(".") > -1)
  728. {
  729. String[] targets = target.split("[.]");
  730. for (String name : targets)
  731. {
  732. o = getValue(o, name);
  733. }
  734. }
  735. else
  736. {
  737. o = getValue(o, target);
  738. }
  739. }
  740. return o;
  741. }
  742. /**
  743. * 以类的属性的get方法方法形式获取值
  744. *
  745. * @param o
  746. * @param name
  747. * @return value
  748. * @throws Exception
  749. */
  750. private Object getValue(Object o, String name) throws Exception
  751. {
  752. if (StringUtils.isNotEmpty(name))
  753. {
  754. Class<?> clazz = o.getClass();
  755. String methodName = "get" + name.substring(0, 1).toUpperCase() + name.substring(1);
  756. Method method = clazz.getMethod(methodName);
  757. o = method.invoke(o);
  758. }
  759. return o;
  760. }
  761. /**
  762. * 得到所有定义字段
  763. */
  764. private void createExcelField()
  765. {
  766. this.fields = new ArrayList<Object[]>();
  767. List<Field> tempFields = new ArrayList<>();
  768. tempFields.addAll(Arrays.asList(clazz.getSuperclass().getDeclaredFields()));
  769. tempFields.addAll(Arrays.asList(clazz.getDeclaredFields()));
  770. for (Field field : tempFields)
  771. {
  772. // 单注解
  773. if (field.isAnnotationPresent(Excel.class))
  774. {
  775. putToField(field, field.getAnnotation(Excel.class));
  776. }
  777. // 多注解
  778. if (field.isAnnotationPresent(Excels.class))
  779. {
  780. Excels attrs = field.getAnnotation(Excels.class);
  781. Excel[] excels = attrs.value();
  782. for (Excel excel : excels)
  783. {
  784. putToField(field, excel);
  785. }
  786. }
  787. }
  788. this.fields = this.fields.stream().sorted(Comparator.comparing(objects -> ((Excel) objects[1]).sort())).collect(Collectors.toList());
  789. }
  790. /**
  791. * 放到字段集合中
  792. */
  793. private void putToField(Field field, Excel attr)
  794. {
  795. if (attr != null && (attr.type() == Type.ALL || attr.type() == type))
  796. {
  797. this.fields.add(new Object[] { field, attr });
  798. }
  799. }
  800. /**
  801. * 创建一个工作簿
  802. */
  803. public void createWorkbook()
  804. {
  805. this.wb = new SXSSFWorkbook(500);
  806. }
  807. /**
  808. * 创建工作表
  809. *
  810. * @param sheetNo sheet数量
  811. * @param index 序号
  812. */
  813. public void createSheet(double sheetNo, int index)
  814. {
  815. this.sheet = wb.createSheet();
  816. this.styles = createStyles(wb);
  817. // 设置工作表的名称.
  818. if (sheetNo == 0)
  819. {
  820. wb.setSheetName(index, sheetName);
  821. }
  822. else
  823. {
  824. wb.setSheetName(index, sheetName + index);
  825. }
  826. }
  827. /**
  828. * 获取单元格值
  829. *
  830. * @param row 获取的行
  831. * @param column 获取单元格列号
  832. * @return 单元格值
  833. */
  834. public Object getCellValue(Row row, int column)
  835. {
  836. if (row == null)
  837. {
  838. return row;
  839. }
  840. Object val = "";
  841. try
  842. {
  843. Cell cell = row.getCell(column);
  844. if (StringUtils.isNotNull(cell))
  845. {
  846. if (cell.getCellTypeEnum() == CellType.NUMERIC || cell.getCellTypeEnum() == CellType.FORMULA)
  847. {
  848. val = cell.getNumericCellValue();
  849. if (HSSFDateUtil.isCellDateFormatted(cell))
  850. {
  851. val = DateUtil.getJavaDate((Double) val); // POI Excel 日期格式转换
  852. }
  853. else
  854. {
  855. if ((Double) val % 1 > 0)
  856. {
  857. val = new BigDecimal(val.toString());
  858. }
  859. else
  860. {
  861. val = new DecimalFormat("0").format(val);
  862. }
  863. }
  864. }
  865. else if (cell.getCellTypeEnum() == CellType.STRING)
  866. {
  867. val = cell.getStringCellValue();
  868. }
  869. else if (cell.getCellTypeEnum() == CellType.BOOLEAN)
  870. {
  871. val = cell.getBooleanCellValue();
  872. }
  873. else if (cell.getCellTypeEnum() == CellType.ERROR)
  874. {
  875. val = cell.getErrorCellValue();
  876. }
  877. }
  878. }
  879. catch (Exception e)
  880. {
  881. return val;
  882. }
  883. return val;
  884. }
  885. }