Thymeleaf自定义方言实现页面过滤功能

       目前使用的所有th:x属性都只是一个标准的、开箱即用的功能集,如果想用想要的名称定义你自己的一组属性(或标签),并在thymeleaf中使用它们来处理你的模板。你可以定义自己的方言。现在我们使用自定义的方言来实现页面权限过滤效果。

方言

       Thymeleaf本身提供了StandardDialect,以及结合了Spring之后提供的SpringStandardDialect。Thymeleaf默认的语法 th:if等,就是定义在了StandardDialect中,th为方言的前缀,if为方言的处理器名称。

StandardDialect的源代码中定义了如下的内容

Java代码  
  1. public class StandardDialect  
  2.             extends AbstractProcessorDialect  
  3.             implements IExecutionAttributeDialect, IExpressionObjectDialect {  
  4.   
  5.     public static final String NAME = "Standard";  
  6.     public static final String PREFIX = "th";  
  7.     public static final int PROCESSOR_PRECEDENCE = 1000;  
  8.   
  9.     ...  
  10. }  

其中的 PREFIX = "th" 定义了在模板中使用时,需要以 th:XX 的形式调用。

详细的接口介绍,可以查看:官方文档 ,本篇文章中暂不进行介绍。先按照源代码的实例,进行我们自己的方言与表达式的编写。

自定义方言

       Dialect是接口,因此需要创建自定义的方言 SecurityDialect 类,然后指定具体的处理器。不直接实现接口,而是继承了 AbstractProcessorDialect 抽象类,同时需要指定名称,以及前缀 prefix。

Java代码  
  1. package edu.uestc.avatar.dialect;  
  2.   
  3. //*************** import ******************//   
  4.   
  5. /** 
  6.  * 自定义Thymeleaf方言:用于处理自定义方言:过滤权限操作 
  7.  */  
  8. @Component  
  9. public class SecurityDialect extends AbstractProcessorDialect {  
  10.     //方言名称  
  11.     public static final String DIALECT_NAME = "wise_authority";  
  12.     //方言前缀  
  13.     public static final String PREFIX = "wise";  
  14.     //方言处理优先级,和标准方言平级  
  15.     public static final int PROCESSOR_PRECEDENCE = 1000;  
  16.   
  17.     public SecurityDialect() {  
  18.         super(DIALECT_NAME, PREFIX, PROCESSOR_PRECEDENCE);  
  19.     }  
  20.   
  21.     //添加方言处理器  
  22.     @Override  
  23.     public Set<IProcessor> getProcessors(String dialectPrefix) {  
  24.         final Set<IProcessor> processors = new HashSet<>();  
  25.         processors.add(new SecurityElementTagProcessor(dialectPrefix));  
  26.         return processors;  
  27.     }  
  28. }  

 @Component表示向Spring IoC容器中注册该自定义方言,在自定义方言中需要添加方言处理器。

自定义方言处理器  

       方言处理器有多种,都以接口的形式定义,使用元素处理器(IElementProcessor)接口,此接口为元素Element处理的基础接口。thymeleaf提供了两种基本的IElementTagProcessor实现,处理器可以方便地实现这些实现:

  • org.thymeleaf.processor.element.AbstractElementTagProcessor,用于按元素名称匹配元素事件的处理器(即不查看属性)。
  • org.thymeleaf.processor.element.AbstractAttributeTagProcessor,用于按元素事件的或者属性(也可以是元素名称)匹配元素事件的处理器。

        官方建议一般不要直接实现此接口实现我们自己的处理器,而是继承类 AbstractAttributeTagProcessor/AbstractElementTagProcessor。

Java代码  
  1. package edu.uestc.avatar.dialect;  
  2. //*************** import ******************/   
  3.   
  4. /** 
  5.  * 定义方言处理器 
  6.  * 
  7.  * <avatar:authority module="department" permission="save"> 
  8.  *      <button>添加部门</button> 
  9.  * </avatar:authority> 
  10.  * 
  11.  * 判定当前登录员工所拥有的权限是否包含module及permission所定义的权限值 
  12.  * 如果包含,不处理,如果不包含,隐藏该标签标记的内容 
  13.  */  
  14. public class SecurityElementTagProcessor extends AbstractElementTagProcessor{  
  15.     //标签名称  
  16.     private static final String PRO_NAME = "authority";  
  17.     //优先级  
  18.     private static final int PRECEDENCE = 1000;  
  19.     public SecurityElementTagProcessor(String dialectPrefix) {  
  20.         super(TemplateMode.HTML,   //此处理器将仅应用于HTML模式  
  21.                 dialectPrefix,  //方言前缀wise,相当于th:if中的th  
  22.                 PRO_NAME,//处理器名称,相当于th:if中的if  
  23.                 true,//应用方言前缀作为标签名  
  24.                 null,//没有属性名:将按标记名匹配  
  25.                 false,//属性名不要前缀  
  26.                 PRECEDENCE);//方言优先级,标准方言默认为1000  
  27.     }  
  28.   
  29.     @Override  
  30.     protected void doProcess(ITemplateContext context,  
  31.                              IProcessableElementTag tag,  
  32.                              IElementTagStructureHandler structureHandler) {  
  33.         //获取tag的module属性值  
  34.         String module = tag.getAttributeValue("module");  
  35.         //获取tag的permission属性值  
  36.         String permission = tag.getAttributeValue("permission");  
  37.         //获取到当前线程绑定的请求对象  
  38.         HttpServletRequest request = ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest();  
  39.         //已经拿到session,就可以拿到session中保存的用户信息了。  
  40.         Employee emp = (Employee) request.getSession().getAttribute("employee");  
  41.         //构建标签标记的权限  
  42.         Privilege privilege = new Privilege(module,permission);  
  43.   
  44.         if (!isPermitted(emp,privilege)){  
  45.             structureHandler.setAttribute("style","display:none");  
  46.         }  
  47.     }  
  48.   
  49.     /** 
  50.      * 判断登录员工是否具有操作权限 
  51.      * @param emp 登录员工 
  52.      * @param privilege 权限值 
  53.      * @return 
  54.      */  
  55.     private boolean isPermitted(Employee emp, Privilege privilege){  
  56.         for(Role role : emp.getRoles()){  
  57.             if(role.getPrivileges().contains(privilege)){  
  58.                 return true;  
  59.             }  
  60.         }  
  61.         return false;  
  62.     }  
  63. }  

使用自定义方言  

Java代码  
  1. <html xmlns:th="http://www.thymeleaf.org" xmlns:avatar="http://www.thymeleaf.org">  
  2. 。。。。。。。。  
  3. <avatar:authority module="department" permission="save">  
  4.        <button>添加部门</button>  
  5.  </avatar:authority>  
posted @ 2022-06-15 08:56  Tiger-Adan  阅读(413)  评论(0编辑  收藏  举报