6.word占位符替換

1.技術(shù)poi-tl

項目開發(fā)中經(jīng)常會根據(jù)數(shù)據(jù)生成對應(yīng)的word模板
代碼案例:https://gitee.com/J-summit/note-sty-blogs/tree/master/src/main/java/tech/cn/note/word

image.png

image.png

2.代碼實現(xiàn)

引入依賴

    implementation group: 'com.deepoove', name: 'poi-tl', version: '1.10.0'
package tech.cn.note.word;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.util.Map;

import javax.annotation.PostConstruct;

import cn.hutool.json.JSONUtil;
import com.deepoove.poi.XWPFTemplate;
import com.deepoove.poi.config.Configure;
import com.deepoove.poi.config.ConfigureBuilder;
import com.deepoove.poi.plugin.table.LoopRowTableRenderPolicy;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import tech.cn.note.utils.CheckUtil;
import tech.cn.note.word.fun.RenderFunction;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

@Service
@Slf4j
@RequiredArgsConstructor
public class PoiTemplateDataFillServiceImpl {

    private static Configure configure;

    private static final ConfigureBuilder builder = Configure.builder();

    private final Map<String, RenderFunction> strategyMap;

    @Value("${custom.nullError:false}")
    private Boolean nullError;

    public static final ThreadLocal threadLocal = new ThreadLocal();

    @PostConstruct
    private void init() {
        builder.buildGramer("${", "}");
        builder.useSpringEL(false);
        builder.addPlugin('~', new LoopRowTableRenderPolicy());
        builder.addPlugin('%', new StrongRenderPolicy(strategyMap));
        builder.addPlugin('\u0000', new CustomTextRenderPolicy());
        configure = builder.build();
    }

    public byte[] writeToByte(Map<String, Object> valueMap, byte[] templateFileBytes) throws Exception {
        try {
            CheckUtil.checkNotNull(templateFileBytes, "模板文件不能為空");
            CheckUtil.checkNotNull(valueMap, "數(shù)據(jù)源為空");
            XWPFTemplate template = prepare(new ByteArrayInputStream(templateFileBytes), valueMap);
            //如果需要輸出PDF則進(jìn)行轉(zhuǎn)換
            ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
            template.write(byteArrayOutputStream);
            return byteArrayOutputStream.toByteArray();
        } finally {
            threadLocal.remove();
        }
    }


    /**
     * poi-tl框架填充數(shù)據(jù)
     *
     * @param inputStream 模板字節(jié)文件流
     * @param map         數(shù)據(jù)源
     * @return 填充結(jié)束的XWPFTemplate
     */
    private XWPFTemplate prepare(ByteArrayInputStream inputStream, Map<String, Object> map) {
        if (nullError) {
            builder.setValidErrorHandler(new Configure.AbortHandler());
        } else {
            builder.setValidErrorHandler(new Configure.ClearHandler());
        }
        configure = builder.build();
        log.debug("data is {}", JSONUtil.parse(map));
        threadLocal.set(map);
        return XWPFTemplate.compile(inputStream, configure).render(map);
    }

}

2.復(fù)雜運算符

官方文檔 https://deepoove.com/poi-tl/

image.png

3.表格循環(huán)

image.png

image.png

4.自定義函數(shù)

image.png

案例實現(xiàn)數(shù)字中文大寫
重寫渲染實現(xiàn)邏輯
比如我們用正則匹配到 %methodName

String functionStr;
        // 現(xiàn)在創(chuàng)建 matcher 對象
        Matcher matcher = PATTERN.matcher(placeHolder);
        if (matcher.find()) {
            functionStr = matcher.group(1);
        } else {
            throw new RuntimeException("[" + placeHolder + "]表達(dá)式不符合規(guī)則,請按照${%function(var,var2,var3)}格式");
        }
        RenderFunction renderFunction = functionMap.get(functionStr);
        if (renderFunction == null) {
            throw new RuntimeException(String.format("不支持的函數(shù)%s", functionStr));
        }

        String vars = matcher.group(2);
        CheckUtil.checkExpression(StringUtils.isNotEmpty(vars), "表達(dá)式不符合規(guī)則,請按照${%function(var,var2,var3)}格式");
        String result = renderFunction.doCalculate(vars.split(","), renderDataCompute);
        if (result == null || StrUtil.NULL.equals(result)) {
            run.setText("", 0);
            return;
        }


package tech.cn.note.word.fun;

import cn.hutool.core.convert.NumberChineseFormatter;
import com.deepoove.poi.render.compute.RenderDataCompute;

import org.springframework.stereotype.Service;

import static org.springframework.util.ObjectUtils.isEmpty;

@Service
public class ChineseMoney implements RenderFunction {
    /**
     * @param fields            參數(shù)1 數(shù)字
     * @param renderDataCompute
     * @return
     */
    @Override
    public String doCalculate(String[] fields, RenderDataCompute renderDataCompute) {
        if (isEmpty(fields)) {
            return "";
        }
        String placeHolder = fields[0];
        Object data = renderDataCompute.compute(placeHolder);
        if (data == null) {
            return "";
        }
        return NumberChineseFormatter.format(Double.parseDouble(data.toString()), true, true);
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子捻悯,更是在濱河造成了極大的恐慌微饥,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,695評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件你雌,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機超凳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來耀态,“玉大人轮傍,你說我怎么就攤上這事∶B剑” “怎么了金麸?”我有些...
    開封第一講書人閱讀 168,130評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長簿盅。 經(jīng)常有香客問我挥下,道長揍魂,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,648評論 1 297
  • 正文 為了忘掉前任棚瘟,我火速辦了婚禮现斋,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘偎蘸。我一直安慰自己庄蹋,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 68,655評論 6 397
  • 文/花漫 我一把揭開白布迷雪。 她就那樣靜靜地躺著限书,像睡著了一般。 火紅的嫁衣襯著肌膚如雪章咧。 梳的紋絲不亂的頭發(fā)上倦西,一...
    開封第一講書人閱讀 52,268評論 1 309
  • 那天,我揣著相機與錄音赁严,去河邊找鬼扰柠。 笑死,一個胖子當(dāng)著我的面吹牛疼约,可吹牛的內(nèi)容都是我干的卤档。 我是一名探鬼主播,決...
    沈念sama閱讀 40,835評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼程剥,長吁一口氣:“原來是場噩夢啊……” “哼劝枣!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起倡缠,我...
    開封第一講書人閱讀 39,740評論 0 276
  • 序言:老撾萬榮一對情侶失蹤哨免,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后昙沦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體琢唾,經(jīng)...
    沈念sama閱讀 46,286評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,375評論 3 340
  • 正文 我和宋清朗相戀三年盾饮,在試婚紗的時候發(fā)現(xiàn)自己被綠了采桃。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,505評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡丘损,死狀恐怖普办,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情徘钥,我是刑警寧澤衔蹲,帶...
    沈念sama閱讀 36,185評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響舆驶,放射性物質(zhì)發(fā)生泄漏橱健。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,873評論 3 333
  • 文/蒙蒙 一沙廉、第九天 我趴在偏房一處隱蔽的房頂上張望拘荡。 院中可真熱鬧,春花似錦撬陵、人聲如沸珊皿。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,357評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蟋定。三九已至,卻和暖如春草添,著一層夾襖步出監(jiān)牢的瞬間溢吻,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,466評論 1 272
  • 我被黑心中介騙來泰國打工果元, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人犀盟。 一個月前我還...
    沈念sama閱讀 48,921評論 3 376
  • 正文 我出身青樓而晒,卻偏偏與公主長得像,于是被迫代替她去往敵國和親阅畴。 傳聞我的和親對象是個殘疾皇子倡怎,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,515評論 2 359

推薦閱讀更多精彩內(nèi)容