All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jeecgframework.poi.util.PoiMergeCellUtil Maven / Gradle / Ivy

There is a newer version: 1.4.11
Show newest version
package org.jeecgframework.poi.util;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.commons.lang3.StringUtils;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.util.CellRangeAddress;
import org.jeecgframework.poi.excel.entity.params.MergeEntity;
import org.jeecgframework.poi.exception.excel.ExcelExportException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 纵向合并单元格工具类
 * 
 * @author JEECG
 * @date 2015年6月21日 上午11:21:40
 */
public final class PoiMergeCellUtil {
	private final static Logger LOGGER = LoggerFactory.getLogger(PoiMergeCellUtil.class);
	private PoiMergeCellUtil() {
	}

	/**
	 * 纵向合并相同内容的单元格
	 * 
	 * @param sheet
	 * @param startRow
	 *            开始行
	 * @param columns
	 *            需要处理的列
	 */
	public static void mergeCells(Sheet sheet, int startRow, Integer... columns) {
		if (columns == null) {
			throw new ExcelExportException("至少需要处理1列");
		}
		Map mergeMap = new HashMap();
		for (int i = 0; i < columns.length; i++) {
			mergeMap.put(columns[i], null);
		}
		mergeCells(sheet, mergeMap, startRow, sheet.getLastRowNum());
	}

	/**
	 * 纵向合并相同内容的单元格
	 * 
	 * @param sheet
	 * @param mergeMap
	 *            key--列,value--依赖的列,没有传空
	 * @param startRow
	 *            开始行
	 */
	public static void mergeCells(Sheet sheet, Map mergeMap, int startRow) {
		mergeCells(sheet, mergeMap, startRow, sheet.getLastRowNum());
	}

	/**
	 * 纵向合并相同内容的单元格
	 * 
	 * @param sheet
	 * @param mergeMap
	 *            key--列,value--依赖的列,没有传空
	 * @param startRow
	 *            开始行
	 * @param endRow
	 *            结束行
	 */
	public static void mergeCells(Sheet sheet, Map mergeMap, int startRow, int endRow) {
		Map mergeDataMap = new HashMap();
		if (mergeMap.size() == 0) {
			return;
		}
		Row row;
		Set sets = mergeMap.keySet();
		String text;
		for (int i = startRow; i <= endRow; i++) {
			row = sheet.getRow(i);
			for (Integer index : sets) {
				if (row == null || row.getCell(index) == null) {
					if (mergeDataMap.get(index) == null) {
						continue;
					}
					if (mergeDataMap.get(index).getEndRow() == 0) {
						mergeDataMap.get(index).setEndRow(i - 1);
					}
				} else {
					text = row.getCell(index).getStringCellValue();
					if (StringUtils.isNotEmpty(text)) {
						hanlderMergeCells(index, i, text, mergeDataMap, sheet, row.getCell(index), mergeMap.get(index));
					} else {
						mergeCellOrContinue(index, mergeDataMap, sheet);
					}
				}
			}
		}
		if (mergeDataMap.size() > 0) {
			for (Integer index : mergeDataMap.keySet()) {
				//update-begin-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
				try{
				sheet.addMergedRegion(new CellRangeAddress(mergeDataMap.get(index).getStartRow(), mergeDataMap.get(index).getEndRow(), index, index));
				}catch (IllegalArgumentException e){
					LOGGER.error("合并单元格错误日志:"+e.getMessage());
					e.fillInStackTrace();
				}
				//update-end-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
			}
		}

	}

	/**
	 * 处理合并单元格
	 * 
	 * @param index
	 * @param rowNum
	 * @param text
	 * @param mergeDataMap
	 * @param sheet
	 * @param cell
	 * @param delys
	 */
	private static void hanlderMergeCells(Integer index, int rowNum, String text, Map mergeDataMap, Sheet sheet, Cell cell, int[] delys) {
		if (mergeDataMap.containsKey(index)) {
			if (checkIsEqualByCellContents(mergeDataMap.get(index), text, cell, delys, rowNum)) {
				mergeDataMap.get(index).setEndRow(rowNum);
			} else {
				//update-begin-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
				try{
				sheet.addMergedRegion(new CellRangeAddress(mergeDataMap.get(index).getStartRow(), mergeDataMap.get(index).getEndRow(), index, index));
				}catch (IllegalArgumentException e){
					LOGGER.error("合并单元格错误日志:"+e.getMessage());
					e.fillInStackTrace();
				}
				//update-end-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
				mergeDataMap.put(index, createMergeEntity(text, rowNum, cell, delys));
			}
		} else {
			mergeDataMap.put(index, createMergeEntity(text, rowNum, cell, delys));
		}
	}

	/**
	 * 字符为空的情况下判断
	 * 
	 * @param index
	 * @param mergeDataMap
	 * @param sheet
	 */
	private static void mergeCellOrContinue(Integer index, Map mergeDataMap, Sheet sheet) {
		if (mergeDataMap.containsKey(index) && mergeDataMap.get(index).getEndRow() != mergeDataMap.get(index).getStartRow()) {
			//update-begin-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
			try{
			sheet.addMergedRegion(new CellRangeAddress(mergeDataMap.get(index).getStartRow(), mergeDataMap.get(index).getEndRow(), index, index));
			}catch (IllegalArgumentException e){
				LOGGER.error("合并单元格错误日志:"+e.getMessage());
				e.fillInStackTrace();
			}
			//update-end-author:wangshuai date:20201118 for:一对多导出needMerge 子表数据对应数量小于2时报错 github#1840、gitee I1YH6B
			mergeDataMap.remove(index);
		}
	}

	private static MergeEntity createMergeEntity(String text, int rowNum, Cell cell, int[] delys) {
		MergeEntity mergeEntity = new MergeEntity(text, rowNum, rowNum);
		// 存在依赖关系
		if (delys != null && delys.length != 0) {
			List list = new ArrayList(delys.length);
			mergeEntity.setRelyList(list);
			for (int i = 0; i < delys.length; i++) {
				list.add(getCellNotNullText(cell, delys[i], rowNum));
			}
		}
		return mergeEntity;
	}

	private static boolean checkIsEqualByCellContents(MergeEntity mergeEntity, String text, Cell cell, int[] delys, int rowNum) {
		// 没有依赖关系
		if (delys == null || delys.length == 0) {
			return mergeEntity.getText().equals(text);
		}
		// 存在依赖关系
		if (mergeEntity.getText().equals(text)) {
			for (int i = 0; i < delys.length; i++) {
				if (!getCellNotNullText(cell, delys[i], rowNum).equals(mergeEntity.getRelyList().get(i))) {
					return false;
				}
			}
			return true;
		}
		return false;
	}

	/**
	 * 获取一个单元格的值,确保这个单元格必须有值,不然向上查询
	 * 
	 * @param cell
	 * @param index
	 * @param rowNum
	 * @return
	 */
	private static String getCellNotNullText(Cell cell, int index, int rowNum) {
		String temp = cell.getRow().getCell(index).getStringCellValue();
		while (StringUtils.isEmpty(temp)) {
			temp = cell.getRow().getSheet().getRow(--rowNum).getCell(index).getStringCellValue();
		}
		return temp;
	}
	//update-begin---author:liusq  Date:20220104  for:[LOWCOD-2521]【autopoi】大数据导出方法【全局】----
    /**
     *处理合并的单元格区域
     * @param sheet
     * @param firstRow 开始行
     * @param lastRow 结束行
     * @param firstCol 开始列
     * @param lastCol 结束列
	 * @date    2022年1月4号
     */
	public static void addMergedRegion(Sheet sheet, int firstRow, int lastRow, int firstCol, int lastCol) {
		try {
			//添加合并的单元格区域
			sheet.addMergedRegion(new CellRangeAddress(firstRow, lastRow, firstCol, lastCol));
		} catch (Exception e) {
			LOGGER.debug("发生了一次合并单元格错误,{},{},{},{}", new Integer[]{
					firstRow, lastRow, firstCol, lastCol
			});
			// 忽略掉合并的错误,不打印异常
			LOGGER.debug(e.getMessage(), e);
		}
	}
	//update-end---author:liusq  Date:20220104  for:[LOWCOD-2521]【autopoi】大数据导出方法【全局】----

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy