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

org.dromara.hutool.poi.excel.writer.TemplateContext Maven / Gradle / Ivy

/*
 * Copyright (c) 2013-2024 Hutool Team and hutool.cn
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.dromara.hutool.poi.excel.writer;

import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellType;
import org.apache.poi.ss.usermodel.Sheet;
import org.dromara.hutool.core.bean.BeanUtil;
import org.dromara.hutool.core.collection.CollUtil;
import org.dromara.hutool.core.lang.Assert;
import org.dromara.hutool.core.map.MapUtil;
import org.dromara.hutool.core.regex.ReUtil;
import org.dromara.hutool.core.text.StrPool;
import org.dromara.hutool.core.text.StrUtil;
import org.dromara.hutool.core.util.ObjUtil;
import org.dromara.hutool.poi.excel.SheetUtil;
import org.dromara.hutool.poi.excel.cell.CellUtil;
import org.dromara.hutool.poi.excel.cell.VirtualCell;

import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.regex.Pattern;

/**
 * 模板上下文,记录了模板中变量所在的Cell
 *
 * @author Looly
 * @since 6.0.0
 */
public class TemplateContext {

	private static final String VAR_PREFIX = StrPool.DELIM_START;
	private static final String VAR_SUFFIX = StrPool.DELIM_END;

	/**
	 * 变量正则
	 * 
    *
  1. 变量名只能包含字母、数字、下划线、$符号、.符号,不能以数字开头
  2. *
  3. 变量以 { 开始,以 } 结束
  4. *
  5. \{表示转义,非变量符号
  6. *
  7. .开头的变量表示列表,.出现在中间,表示表达式子对象
  8. *
*/ private static final Pattern VAR_PATTERN = Pattern.compile("(? varMap = new LinkedHashMap<>(); /** * 构造 * * @param templateSheet 模板sheet */ public TemplateContext(final Sheet templateSheet) { init(templateSheet); } /** * 获取变量对应的当前单元格,列表变量以开头 * * @param varName 变量名 * @return 单元格 */ public Cell getCell(final String varName) { return varMap.get(varName); } /** * 获取当前替换的数据行对应变量的底部索引
* 此方法用户获取填充行,以便下移填充行后的行
*
    *
  • 如果为实体单元格,直接填充,无需下移,返回0
  • *
  • 如果为{@link VirtualCell},返回最底部虚拟单元格各行号
  • *
* * @param rowDataBean 填充数据 * @return 最大行索引,-1表示无数据填充,0表示无需下移 */ public int getBottomRowIndex(final Object rowDataBean) { final AtomicInteger bottomRowIndex = new AtomicInteger(-1); this.varMap.forEach((name, cell) -> { if(null != BeanUtil.getProperty(rowDataBean, name)){ if (cell instanceof VirtualCell) { bottomRowIndex.set(Math.max(bottomRowIndex.get(), cell.getRowIndex())); } else if(bottomRowIndex.get() < 0){ // 实体单元格,直接填充,无需下移 bottomRowIndex.set(0); } } }); return bottomRowIndex.get(); } /** * 填充变量名name指向的单元格 * * @param rowDataBean 一行数据的键值对 * @param isListVar 是否为列表填充,列表填充会自动指向下一列,否则填充结束后删除变量 * @since 6.0.0 */ public void fill(final Object rowDataBean, final boolean isListVar) { final Map varMap = this.varMap; varMap.forEach((name, cell) -> { if (null == cell) { return; } final String templateStr = cell.getStringCellValue(); // 填充单元格 if (fill(cell, name, rowDataBean)) { // 指向下一个单元格 putNext(name, cell, templateStr, isListVar); } }); if (!isListVar) { // 清理已经匹配完毕的变量 MapUtil.removeNullValue(varMap); } } /** * 将变量指向下一行的单元格
* 如果为列表,则指向下一行的虚拟单元格(不创建单元格) * 如果非列表,则清空此变量 * * @param name 变量名 * @param currentCell 当前单元格 * @param templateStr 模板字符串 * @param isListVar 是否为列表填充 */ private void putNext(final String name, final Cell currentCell, final String templateStr, final boolean isListVar) { if (isListVar) { // 指向下一列的单元格 final Cell next = new VirtualCell(currentCell, currentCell.getColumnIndex(), currentCell.getRowIndex() + 1, templateStr); varMap.put(name, next); } else { // 非列表,一次性填充,即变量填充后,和此单元格去掉关联 varMap.put(name, null); } } /** * 填充数据 * * @param cell 单元格,非模板中变量所在单元格则为{@link VirtualCell} * @param name 变量名 * @param rowDataBean 填充的数据,可以为Map或Bean * @return 是否填充成功,{@code false}表示无数据 */ private boolean fill(Cell cell, final String name, final Object rowDataBean) { final String templateStr = cell.getStringCellValue(); if (cell instanceof VirtualCell) { // 虚拟单元格,转换为实际单元格 final Cell newCell; newCell = CellUtil.getCell(cell.getSheet(), cell.getColumnIndex(), cell.getRowIndex(), true); Assert.notNull(newCell, "Can not get or create cell at {},{}", cell.getColumnIndex(), cell.getRowIndex()); newCell.setCellStyle(cell.getCellStyle()); cell = newCell; } final Object cellValue; // 模板替换 if (StrUtil.equals(name, StrUtil.unWrap(templateStr, VAR_PREFIX, VAR_SUFFIX))) { // 一个单元格只有一个变量,支持多级表达式 cellValue = BeanUtil.getProperty(rowDataBean, name); if (null == cellValue) { // 对应表达式无提供的值,跳过填充 return false; } } else { // 模板中存在多个变量或模板填充,直接赋值为String // 没有找到值的变量保留原样 cellValue = StrUtil.formatByBean(templateStr, rowDataBean, true); if (ObjUtil.equals(cellValue, templateStr)) { // 模板无修改,说明没有变量替换,跳过填充 return false; } } CellUtil.setCellValue(cell, cellValue); return true; } /** * 初始化,提取变量及位置,并将转义的变量回填 * * @param templateSheet 模板sheet */ private void init(final Sheet templateSheet) { SheetUtil.walk(templateSheet, (cell, ctx) -> { if (CellType.STRING == cell.getCellType()) { // 只读取字符串类型的单元格 final String cellValue = cell.getStringCellValue(); // 字符串中可能有多个变量 final List vars = ReUtil.findAllGroup1(VAR_PATTERN, cellValue); if (CollUtil.isNotEmpty(vars)) { // 模板变量 for (final String var : vars) { varMap.put(var, cell); } } // 替换转义的变量 final String str = ReUtil.replaceAll(cellValue, ESCAPE_VAR_PATTERN, (matcher) -> VAR_PREFIX + matcher.group(1) + VAR_SUFFIX); if (!StrUtil.equals(cellValue, str)) { cell.setCellValue(str); } } }); } @Override public String toString() { return "TemplateContext{" + "varMap=" + varMap + '}'; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy