Freemark word | 根据模板生成 word 文档及双层表格

生成word模板。

    编辑word使用占位符。双层列表循环表格, 群组里面在嵌套一个表格,无边框。其余正常占位即可。

   以xml 格式保存,注意  ${xxxxx} 必须何在一起,不可以分开。

    我这里用的是wps 保存的,(如果用的 word2003,以 2003xml格式保存,保存以 数据格式,保存xml形式。)

        

  

保存之后,后缀名更改为 .ftl 格式。

  

开始代码,获取模板,保存模板。

 String filePath = null;
        Integer len;
        Map<String, Object> map = DocUtil.getContractMap(tabList,contract);
        map.put("access",null == access ? "" : access);
        Writer out = null;
        File file = null;
        try {
            //创建
            Configuration configuration = new Configuration();
            //设置编码
            configuration.setDefaultEncoding("UTF-8");
            // 获得当前类路径
            file = new File(new PrintContract().getClass().getResource("").getPath());
            // ftl模板文件统一放至 test.ftl包下面
            configuration.setDirectoryForTemplateLoading(new File(file + "\\wordTemplate"));
            // 获取模板
            Template template = configuration.getTemplate("contract.ftl", "UTF-8");
//            filePath = file + File.separator + "商标申请合同.doc";
            filePath = file + "\\wordTemplate"+ File.separator + "商标申请合同.doc";
            // 打印word到制定目录
            File  outFile = new File( filePath);
            if (!outFile.getParentFile().exists()) {
                outFile.getParentFile().mkdirs();
            }
            // 将模板和数据模型合并生成文件
            out = new BufferedWriter(new OutputStreamWriter(
                    new FileOutputStream(outFile), "UTF-8"));
            // 生成文件
            template.process(map, out);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (out != null){
                out.flush();
                out.close();
            }

作用使 word字段居中。

/**
* 获取字符串中文个数
* @param str
* @return
*/
public static int getChineseNum(String str){
return (str.getBytes().length - str.length()) / 2;
}
/**
* 补全空格
* @param str
* @return
*/
public static String getDocString(String str,int in){
if (str.length() < in){
in = in - getChineseNum(str) - str.length();
for (int i=1; i < in; i++){
str += " ";
}
}
return str;
}
/**
* 补全空格,居中word格式,
*/
public static Map<String,Object> getContractMap(List<ContractFrom> tabList, Contract contract){

// 合同编号
map.put("contractNumber", null == contract.getContractNumber() ? "" : contract.getContractNumber());
// 甲方 联 系 人
map.put("firstPartyLinkman", null == contract.getFirstPartyLinkman() ? "" : contract.getFirstPartyLinkman());
// 甲方联系电话
map.put("firstPartyPhone", null == contract.getFirstPartyPhone() ? "" : contract.getFirstPartyPhone());
// 乙方经办人
map.put("partyBAgent", null == contract.getPartyBAgent() ? "" : contract.getPartyBAgent());
// 乙方联系电话
map.put("partyBPhone", null == contract.getPartyBPhone() ? "" : contract.getPartyBPhone());

// 甲方公司名称 (14位,小于补空格满足)
map.put("firstParty", null == contract.getFirstParty() ? "\t\t\t\t\t\t\t\t\t " : DocUtil.getDocString(contract.getFirstParty(),36));
// 甲方 联 系 人 (三位小于补空格)
map.put("firstPartyContactName", null == contract.getFirstPartyContactName() ? "\t\t\t\t\t\t\t\t " : DocUtil.getDocString(contract.getFirstPartyContactName(),34));
// 甲方 地址
map.put("firstPartyAddress", null == contract.getFirstPartyAddress() ? "\t\t\t\t\t\t\t\t\t " : DocUtil.getDocString(contract.getFirstPartyAddress(),36));
// 甲方 签约日期
map.put("firstPartyDate", null == contract.getFirstPartyDate() ? "\t\t\t\t\t\t\t\t " : DocUtil.getDocString(sf.format(contract.getFirstPartyDate()),32));
// 甲方 邮箱账号
map.put("firstPartyEmail", null == contract.getFirstPartyEmail() ? "\t\t\t\t\t\t\t\t " : DocUtil.getDocString(contract.getFirstPartyEmail(),32));

//乙方 商标顾问
map.put("partyBBrandConsultant", null == contract.getPartyBBrandConsultant() ? "" : contract.getPartyBBrandConsultant());
// 乙方 地址
map.put("partyBAddress", null == contract.getPartyBAddress() ? "" : contract.getPartyBAddress());
// 乙方 签约日期
map.put("partyBDate", null == contract.getPartyBDate() ? "" : sf.format(contract.getPartyBDate()));
// 乙方 邮箱账号
map.put("partyBEmail", null == contract.getPartyBEmail() ? "" : contract.getPartyBEmail());

// 支付金额
map.put("money", 0.0d == contract.getMoney() ? 0.0 : contract.getMoney());
if ( null != tabList){
map.put("tabList",tabList);
}
return map;
}

 

freemarker pom

<!-- freemarker jar -->
<!--<dependency>-->
<!--<groupId>org.freemarker</groupId>-->
<!--<artifactId>freemarker</artifactId>-->
<!--<version>2.3.20</version>-->
<!--</dependency>-->
<dependency>
    <groupId>org.freemarker</groupId>
    <artifactId>freemarker-gae</artifactId>
    <version>2.3.25-incubating</version>
</dependency>

 

大功告成。

SpringBoot_Freemarker生成Word_多个表格+两层嵌套循环; 步骤说明: 1.用Microsoft Office Word打开word原件;将文档中需要动态生成的内容,替换为属性名 ${name} 2.另存为,选择保存类型Word 2003 XML 文档(*.xml) 3.用Firstobject free XML editor打开文件,选择Tools下的Indent【或者按快捷键F8】格式化文件内容。左边是文档结构,右边是文档内容; 4. 文档生成后有时需要手动修改,查找第一步中设置的属性名,可能会产生类似${n.....ame}类似的样子,我们将将名字中间的标签删掉,恢复为${name} 5. word模板中有表格,需要循环的位置, 用 标签将第二对 标签(即除表头的w:tr标签后的一对)包围起来 同时表格内的属性例如${name},在这里需要修改为${user.name} (userList是集合在dataMap中的key, user是集合中的每个元素, 类似), 如图: PLUS:若表格之外还有嵌套的循环,也需要用,注意这里的标签不要和某对其他标签交叉,不可以出现这种 6. 标识替换完之后,另存为.ftl后缀文件即可。 代码里是相对有一丢丢复杂的,两层嵌套循环; 总(dataMap) deptName 部门名 list(Table)表的集合 table1(map) table-名字 ${map.table} tableName-中文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段中文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table2 table-名字 ${map.table} tableName-中文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段中文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table3 ……
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值