Freemarker自定义指令和方法


之前在项目中使用了Freemarker的自定义指令和方法,感觉挺方便的,记录下,防止遗忘。Talk is cheap, show the code!

Freemarker模板

添加Freemarker的maven依赖

	  <dependency>
          <groupId>org.freemarker</groupId>
          <artifactId>freemarker</artifactId>
          <version>2.3.28</version>
      </dependency>

hello.ftl

模板包含一个自定义手机号码隐位的方法和一个菜单展示的自定义指令

<html>
<head>
    <title>你好,${username}</title>
</head>
<body>
手机号码:${phoneMask(phone, 2, 4)}
<@menus  method="menus"; showAll>
    <#if showAll>
        <#if menus?? && menus?size gt 0>
            <#list menus as item>
            <h1>${item}</h1>
            </#list>
        </#if>
    <#else>
        没有可用菜单
    </#if>
</@menus>
</body>
</html>

自定义指令

自定义一个菜单展示指令:MenusTagDirective.java

import java.io.IOException;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import freemarker.core.Environment;
import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapperBuilder;
import freemarker.template.TemplateBooleanModel;
import freemarker.template.TemplateDirectiveBody;
import freemarker.template.TemplateDirectiveModel;
import freemarker.template.TemplateException;
import freemarker.template.TemplateModel;

public class MenusTagDirective implements TemplateDirectiveModel {

    /**
     * @param environment           环境变量(实现复杂功能时可能会用)
     * @param map                   在.ftl模板中使用自定义指令传的参数(key-value形式)
     * @param templateModels        返回值,数组形式
     * @param templateDirectiveBody 指令内容
     * @throws TemplateException
     * @throws IOException
     */
    @Override
    public void execute(Environment environment, Map map, TemplateModel[] templateModels, TemplateDirectiveBody templateDirectiveBody) throws TemplateException, IOException {
        DefaultObjectWrapperBuilder builder = new DefaultObjectWrapperBuilder(Configuration.VERSION_2_3_28);
        if (map.containsKey("method") && "menus".equalsIgnoreCase(map.get("method").toString())) {
            // 这里直接new一个菜单列表,真实环境可以自己获取
            List<String> menusList = new LinkedList<>();
            menusList.add("用户管理");
            menusList.add("角色管理");
            menusList.add("权限管理");
            menusList.add("系统设置");
            environment.setVariable("menus", builder.build().wrap(menusList));

            templateModels[0] = TemplateBooleanModel.TRUE;
        }else{
            templateModels[0] = TemplateBooleanModel.FALSE;
        }
        templateDirectiveBody.render(environment.getOut());
    }
}

自定义方法

自定义手机号码隐位的method,继承自TemplateMethodModelEx,以前TemplateMethodModel被Deprecated。

import java.util.List;

import freemarker.template.TemplateMethodModelEx;
import freemarker.template.TemplateModelException;

/**
 * 手机隐位
 * ${phoneMask(phone, beginIdx, suffixIdx)}
 */
public class PhoneMask implements TemplateMethodModelEx {

    @Override
    public Object exec(List args) throws TemplateModelException {
        if (args.size() != 3) {
            throw new TemplateModelException(
                    "the field number of phoneMask is wrong");
        }

        try {
            String phoneNum  = args.get(0).toString();
            int beginIdx = Integer.valueOf(args.get(1).toString().trim());
            int suffixIdx = Integer.valueOf(args.get(1).toString().trim());

            // 这里没有对begin和suffix做严格校验
            StringBuilder sb = new StringBuilder();
            int size = phoneNum.length();
            if(size > beginIdx + suffixIdx) {
                String prefix = phoneNum.substring(0, beginIdx);
                String suffix = phoneNum.substring(size - suffixIdx);
                String middle = phoneNum.substring(beginIdx, size - suffixIdx).replaceAll(".", "*");
                sb.append(prefix).append(middle).append(suffix);
            } else {
                sb.append(phoneNum);
            }

            return sb.toString();
        } catch (Exception e) {
            throw new TemplateModelException(
                    "phoneMask 处理异常", e);
        }
    }
}

测试方法

public static void main( String[] args )
    {
        Configuration cfg = new Configuration(Configuration.getVersion());
        try {
            // 设置模板位置
            cfg.setDirectoryForTemplateLoading(new File("D:\\workspaces\\freemarkerdemo\\src\\main\\resources"));
            cfg.setDefaultEncoding("UTF-8");
            cfg.setTemplateExceptionHandler(TemplateExceptionHandler.RETHROW_HANDLER);
            // 自定义指令\方法
            cfg.setSharedVariable("menus", new MenusTagDirective());
            cfg.setSharedVariable("phoneMask", new PhoneMask());

            Template temmplate = cfg.getTemplate("hello.ftl");
            Map<String, String> map = new HashMap<>();
            map.put("username", "admin");
            map.put("phone", "13849323456");

            StringWriter sw = new StringWriter();
            try {
                temmplate.process(map,sw);
            } catch (TemplateException e) {
                e.printStackTrace();
            }
            System.out.println(sw.toString());

        } catch (IOException e) {
            e.printStackTrace();
        }
    }

测试结果

<html>
<head>
    <title>你好,admin</title>
</head>
<body>
手机号码:13*******56
            <h1>用户管理</h1>
            <h1>角色管理</h1>
            <h1>权限管理</h1>
            <h1>系统设置</h1>
</body>
</html>
  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
自定义一个名为 ListDirective 的 FreeMarker 指令,你需要遵循以下步骤: 1. 创建一个 Java 类来实现指令。这个类应该继承 `freemarker.template.TemplateDirectiveModel` 接口,并且需要实现 `execute()` 方法,该方法会被 FreeMarker 引擎调用来处理指令。 ```java import java.util.Map; import javax.servlet.jsp.JspException; import javax.servlet.jsp.PageContext; import freemarker.core.Environment; import freemarker.template.*; public class ListDirective implements TemplateDirectiveModel { public void execute(Environment env, Map params, TemplateModel[] loopVars, TemplateDirectiveBody body) throws TemplateException, java.io.IOException { // 在这里编写逻辑来处理指令 // ... } } ``` 2. 在 `execute()` 方法中编写处理逻辑。在这里,你需要解析和验证指令参数,并使用 `TemplateModel` 对象来获取指令所在的模板环境和输出流。你也需要创建一个 `SimpleScalar` 对象来包装列表数据,并将其放入模板环境中。 ```java public void execute(Environment env, Map params, TemplateModel[] loopVars, TemplateDirectiveBody body) throws TemplateException, java.io.IOException { // 解析和验证指令参数 String listName = DirectiveUtils.getRequiredParam(params, "name"); Object listData = DirectiveUtils.getRequiredParam(params, "data"); // 获取模板环境和输出流 Writer out = env.getOut(); PageContext pageContext = (PageContext) env.getCustomAttribute("PageContext"); if (pageContext == null) { throw new JspException("PageContext not found"); } // 创建 SimpleScalar 对象来包装列表数据 SimpleScalar listModel = new SimpleScalar(listData.toString()); // 将 SimpleScalar 对象放入模板环境中 env.setVariable(listName, listModel); // 处理指令内容 if (body != null) { body.render(out); } } ``` 3. 将自定义指令注册到 FreeMarker 引擎中。你可以使用 `Configuration` 对象的 `setSharedVariable()` 方法将指令实例添加到引擎中。 ```java import freemarker.template.Configuration; Configuration cfg = new Configuration(Configuration.VERSION_2_3_30); cfg.setSharedVariable("list", new ListDirective()); ``` 现在,你就可以在模板中使用自定义指令了: ``` <@list name="myList" data=[1, 2, 3]> <#list myList as item> ${item} </#list> </@list> ``` 这将输出: ``` 1 2 3 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值