SpringBoot使用Pio-tl动态填写合同(文档)
poi-tl(poi template language)是Word模板引擎,使用Word模板和数据创建很棒的Word文档。 poi-tl官方网址
项目中有需求需要动态填充交易合同,因此想到了使用poi-tl技术来实现
一、引入依赖
<!-- 生成word并且导出 -->
<dependency>
<groupId>com.deepoove</groupId>
<artifactId>poi-tl</artifactId>
<version>1.5.0</version>
</dependency>
<!-- 上面需要的依赖-->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>4.1.2</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml-schemas</artifactId>
<version>4.1.2</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>4.1.2</version>
</dependency>
<dependency>
<groupId>com.deepoove</groupId>
<artifactId>poi-tl</artifactId>
<version>1.9.1</version>
</dependency>
二、填充合同
1、金额转换工具类
public class MoneyUtils {
private static final String UNIT = "万千佰拾亿千佰拾万千佰拾元角分";
private static final String DIGIT = "零壹贰叁肆伍陆柒捌玖";
private static final double MAX_VALUE = 9999999999999.99D;
public static String change(double v) {
if (v < 0 || v > MAX_VALUE){
return "参数非法!";
}
long l = Math.round(v * 100);
if (l == 0){
return "零元整";
}
String strValue = l + "";
// i用来控制数
int i = 0;
// j用来控制单位
int j = UNIT.length() - strValue.length();
String rs = "";
boolean isZero = false;
for (; i < strValue.length(); i++, j++) {
char ch = strValue.charAt(i);
if (ch == '0') {
isZero = true;
if (UNIT.charAt(j) == '亿' || UNIT.charAt(j) == '万' || UNIT.charAt(j) == '元') {
rs = rs + UNIT.charAt(j);
isZero = false;
}
} else {
if (isZero) {
rs = rs + "零";
isZero = false;
}
rs = rs + DIGIT.charAt(ch - '0') + UNIT.charAt(j);
}
}
if (!rs.endsWith("分")) {
rs = rs + "整";
}
rs = rs.replaceAll("亿万", "亿");
return rs;
}
public static void main(String[] args){
System.out.println(MoneyUtils.change(12356789.9845));
}
}
2、电子合同生成相关配置及其配置类
# 电子合同生成相关配置
contract:
#生产环境合同模板存放路径
prodTemplateUrl: https://xxxxxx-cn-beijing.aliyuncs.com/templets/64042201/20221116/cd9aeb196edd46aeacc607c18d1c481d/xxxxxx活牛交易市场肉牛买卖合同模板.docx
#生产环境合同临时存放路径(应用服务器路径)
prodContractPath: /ekode/offlineFile/
#测试环境合同模板存放路径
devTemplateUrl: https://xxxxxx-cn-beijing.aliyuncs.com/templets/64042201/20221116/cd9aeb196edd46aeacc607c18d1c481d/xxxxxx活牛交易市场肉牛买卖合同模板.docx
#测试环境合同临时存放路径(应用服务器路径)
devContractPath: C://Users//xxxxxx//Desktop//临时文件//
#合同名后缀
fileNameSuffix: 肉牛买卖合同
@Component
@ConfigurationProperties(prefix = "contract")
public class ContractConfig {
private String prodTemplateUrl;
private String prodContractPath;
private String devTemplateUrl;
private String devContractPath;
private String fileNameSuffix;
public String getProdTemplateUrl() {
return prodTemplateUrl;
}
public void setProdTemplateUrl(String prodTemplateUrl) {
this.prodTemplateUrl = prodTemplateUrl;
}
public String getProdContractPath() {
return prodContractPath;
}
public void setProdContractPath(String prodContractPath) {
this.prodContractPath = prodContractPath;
}
public String getDevTemplateUrl() {
return devTemplateUrl;
}
public void setDevTemplateUrl(String devTemplateUrl) {
this.devTemplateUrl = devTemplateUrl;
}
public String getDevContractPath() {
return devContractPath;
}
public void setDevContractPath(String devContractPath) {
this.devContractPath = devContractPath;
}
public String getFileNameSuffix() {
return fileNameSuffix;
}
public void setFileNameSuffix(String fileNameSuffix) {
this.fileNameSuffix = fileNameSuffix;
}
}
3、word生成工具类
public class WordUtils {
private static Logger logger = LoggerFactory.getLogger(WordUtils.class);
/**
* @param templatePath 文档模板存放路径
* @param createFileDir 生成的文档存档地址
* @param createFileName 生成的文件名(不带格式后缀)
* @param paramsMap 需要填充的内容
* @return
*/
public static String createWord(String templatePath, String createFileDir, String createFileName, Map<String, Object> paramsMap) throws IOException {
logger.info("【动态填充文档公共方法】被操作文档:{},填充后新文档存放地址:{},填充后新文档名称:{}," +
"填充字段内容:{}", JSON.toJSONString(templatePath),JSON.toJSONString(createFileDir),
JSON.toJSONString(createFileName),JSON.toJSONString(paramsMap));
Assert.notNull(templatePath,"文档模板存放路径不能为空");
Assert.notNull(createFileDir,"生成的文档存档地址不能为空");
Assert.notNull(createFileName,"生成的文件名不能为空");
Assert.notNull(paramsMap,"需要填充的内容不能为空");
//生成的文件格式
String formatSuffix = ".docx";
//拼接后的文件名
String fileName = createFileName + formatSuffix;
logger.info("生成合同的文件名(带后缀)为:{}",JSON.toJSONString(fileName));
//生成文件存放路径
if(!createFileDir.endsWith("/")){
createFileDir = createFileDir + File.separator;
}
File dir = new File(createFileDir);
if(!dir.exists()){
dir.mkdirs();
logger.info("生成合同时存储文件的路径{}不存在,已自动创建文件夹",createFileDir);
}
//生成的文件全路径
String filePath = createFileDir + fileName;
logger.info("生成合同的存放路径(绝对路径):{}",JSON.toJSONString(filePath));
XWPFTemplate template = null;
//poi-tl使用网络文件作为合同模板,需要转换为文件流动态填写
if(templatePath.contains("https://xxxxxx-cn-beijing.aliyuncs.com")){
URL url = new URL(templatePath);
InputStream in = url.openStream();
//渲染表格
HackLoopTableRenderPolicy policy = new HackLoopTableRenderPolicy();
//此list是步骤四中动态传入的数据列表
Configure config = Configure.newBuilder().bind("list", policy).build();
template = XWPFTemplate.compile(in,config).render(paramsMap);
}else {
//hetong muban wei bendi wenjian
template = XWPFTemplate.compile(templatePath).render(paramsMap);
}
//将填写后的内容写入生成的合同中
template.writeToFile(filePath);
template.close();
logger.info("【生成的电子合同本地存放路径为】:{}",JSON.toJSONString(filePath));
return filePath;
}
}
4、组装合同填充的内容
Map<String,Object> paramsMap = new HashMap<>();
paramsMap.put("bianHao",bianHao);
paramsMap.put("jfName",contractDTO.getJfName());
paramsMap.put("yfName",contractDTO.getYfName());
paramsMap.put("zongJia",contractDTO.getPrice());
paramsMap.put("dingJin",contractDTO.getDingJin());
paramsMap.put("yuKuan",contractDTO.getWeiKuan());
paramsMap.put("yuKuanJieQingDate",contractDTO.getJiaoFuDateEnd()+"日");
paramsMap.put("jiaoFuDateStart",contractDTO.getJiaoFuDateStart()+"日");
paramsMap.put("jiaoFuDateEnd",contractDTO.getJiaoFuDateEnd()+"日");
paramsMap.put("gaoZhiDay",contractDTO.getGaoZhiDay());
paramsMap.put("jiaoFuDiDian",contractDTO.getJiaoFuDiDian());
paramsMap.put("yunShuFangShi",contractDTO.getYunShuFangShi());
paramsMap.put("jfwyJin",contractDTO.getJfwyJin());
paramsMap.put("jfwyDay",contractDTO.getJfwyDay());
//牛只列表
List<Map<String,Object>> cattleList=new ArrayList<Map<String,Object>>();
for (int i = 0; i < contractDTO.getCattleLists().size(); i++) {
Map<String,Object> cattleMap = new HashMap<>();
cattleMap.put("index",i+1);
cattleMap.put("eartagNo",contractDTO.getCattleLists().get(i).getEartagNo());
cattleMap.put("pz",contractDTO.getCattleLists().get(i).getVarieties());
cattleMap.put("sex",contractDTO.getCattleLists().get(i).getGender());
cattleMap.put("ms",contractDTO.getCattleLists().get(i).getFulColor());
cattleMap.put("tz",contractDTO.getCattleLists().get(i).getSecondWeight());
cattleMap.put("jiaGe",contractDTO.getCattleLists().get(i).getFinalPrice());
cattleList.add(cattleMap);
}
//金额中文大写
String zongJiaChinese = MoneyUtils.change(Double.valueOf(contractDTO.getPrice()));
paramsMap.put("rmb",zongJiaChinese);
paramsMap.put("list",cattleList);
//根据日期创建文件夹
String nowDate = DateUtils.dateTimeNow("yyyyMMdd");
//获取当前运行环境是生产还是测试
String active = SpringUtils.getActiveProfile();
String templatePath = null; //模板存放路径 OSS地址
String createFileDir = null; //生成的合同临时存放路径
//生产环境
if("prod".equals(active)){
templatePath = contractConfig.getDevTemplateUrl();
createFileDir = contractConfig.getProdContractPath() + nowDate + "/";
}
//测试环境
if("dev".equals(active)){
templatePath = contractConfig.getDevTemplateUrl();
createFileDir = contractConfig.getDevContractPath() + nowDate + "//";
}
//生成的合同名
String createFileName = contractDTO.getJfName() + "_" +
contractDTO.getYfName() + "_" + nowDate + "_" + contractConfig.getFileNameSuffix();
logger.info("--------------【开始动态填充电子合同】--------------");
InputStream is = null;
String ossPath = null;
String wordPath = null;
try {
wordPath = WordUtils.createWord(templatePath,createFileDir,createFileName,paramsMap);
} catch (IOException e) {
logger.error("生成合同异常,异常信息:{}",JSON.toJSONString(e.getMessage()));
e.printStackTrace();
return null;
}
三、效果预览