SiteMesh3文档 http://wiki.sitemesh.org/wiki/pages/viewpage.action?pageId=1081348

重新搭建项目偶然发现SiteMesh有了新版本SiteMesh3,本着用新不用旧原则果断升级,多少遇了点坑,顺便记录下

SiteMesh3配置

  1. 添加maven依赖

    <dependency>
    <groupId>org.sitemesh</groupId>
    <artifactId>sitemesh</artifactId>
    <version>3.0.1</version>
    </dependency>
  2. 添加filter

    在web.xml中添加filter

    <filter>
    <filter-name>sitemesh</filter-name>
    <filter-class>org.sitemesh.config.ConfigurableSiteMeshFilter</filter-class>
    </filter>
    <filter-mapping>
    <filter-name>sitemesh</filter-name>
    <url-pattern>/*</url-pattern>
    </filter-mapping>
  3. 配置servlet

    <servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.ftl</url-pattern>
    </servlet-mapping>
  4. 添加sitemesh配置文件

    1. 添加配置文件 sitemesh3.xml
      默认配置文件路径为:/WEB-INF/sitemesh3.xml

      sitemesh3.xml
      <sitemesh>
      <!--
      By default, SiteMesh will only intercept responses that set the Content-Type HTTP header to text/html
      This can be altered to allow SiteMesh to intercept responses for other types.
      默认 SiteMesh 只对HTTP响应头中Content-Type为 text/html 的类型进行拦截和装饰,若需要处理其它mime类型需要自行添加配置
      -->
      <mime-type>text/html</mime-type>
      <!--
      Map default decorator. This shall be applied to all paths if no other paths match.
      配置装饰器,仅设置decorator参数时表示为默认的装饰器,当没有任何路径被匹配时会使用默认装饰器装配
      -->
      <mapping decorator="/WEB-INF/decorators/decorator.ftl"/>
      <!--对不同的路径指定特定的装饰器-->
      <!--<mapping path="/admin/*" decorator="/WEB-INF/decorators/admin-decorator.ftl"/>-->
      <!--
      Alternative convention. This is more verbose but allows multiple decorators to be applied to a single path.
      对同一路径可以同时使用多个装饰器
      -->
      <mapping>
      <path>/category/*</path>
      <decorator>/WEB-INF/decorators/common-decorator.ftl</decorator>
      <decorator>/WEB-INF/decorators/menu-decorator.ftl</decorator>
      <decorator>/WEB-INF/decorators/category-decorator.ftl</decorator>
      </mapping>
      <!--
      Exclude path from decoration.
      排除路径,只需将exclue设置为true即可
      -->
      <mapping path="/static/*" exclue="true"/>
      <!--
      An advanced feature of SiteMesh is the ability to define custom rules that manipulate tags on a page.
      These are classes that implement org.sitemesh.content.tagrules.TagRuleBundle.
      默认SiteMesh仅支持title、head、meta、body等tag,可以自定义tag,实现TagRuleBundle接口即可
      -->
      <content-processor>
      <tag-rule-bundle class="com.sankuai.shangchao.util.HtmlTagRuleBundle"/>
      </content-processor>
      </sitemesh>
    2. 修改配置文件路径
      默认配置文件路径为:/WEB-INF/sitemesh3.xml 若需要修改配置文件路径需要在filter里配置configFile参数

      <filter>
      <filter-name>sitemesh</filter-name>
      <filter-class>org.sitemesh.config.ConfigurableSiteMeshFilter</filter-class>
      <init-param>
      <param-name>configFile</param-name>
      <param-value>/WEB-INF/sitemesh3.xml</param-value>
      </init-param>
      </filter>
    3. 自定义tag

      HtmlTagRuleBundle.java
      import org.sitemesh.SiteMeshContext;
      import org.sitemesh.content.ContentProperty;
      import org.sitemesh.content.tagrules.TagRuleBundle;
      import org.sitemesh.content.tagrules.html.ExportTagToContentRule;
      import org.sitemesh.tagprocessor.State; /**
      * Description: FootTagRuleBundle
      * Author: liuzhao
      * Create: 2015-08-22 09:21
      */
      public class HtmlTagRuleBundle implements TagRuleBundle {
      @Override
      public void install(State defaultState, ContentProperty contentProperty, SiteMeshContext siteMeshContext) {
      defaultState.addRule("foot", new ExportTagToContentRule(siteMeshContext, contentProperty.getChild("foot"), false));
      } @Override
      public void cleanUp(State defaultState, ContentProperty contentProperty, SiteMeshContext siteMeshContext) { }
      }
  5. decorator示例

    decorator配置页面布局layout,对应的tag会被进行装饰替换

    decorator.ftl
    <!DOCTYPE html>
    <html>
    <head>
        <title>
            <sitemesh:write property="title"/>
        </title>
        <sitemesh:write property='head'/>
    </head>
    <body>
    <h1>啦拉拉,我是卖报的小行家</h1>
    <sitemesh:write property='body'/>
    <sitemesh:write property="foot"/>
    </body>
    </html>
  6. SpringMVC、FreeMarker配置(404问题处理)

    sitemesh3是完全独立的,不和任何框架进行偶合,因此SpringMVC、FreeMarker配置完全不需要考虑sitemesh3的兼容问题
    在加载装饰器的时候,会出现404问题,可能的原因是找不到ftl文件的解析器,所以需要配置下ftl使用默认的Servlet解析,在web.xml中添加如下配置

    <servlet-mapping>
        <servlet-name>default</servlet-name>
        <url-pattern>*.ftl</url-pattern>
    </servlet-mapping>

  1. decorate源码

    @Override
    protected boolean postProcess(String contentType, CharBuffer buffer,
                                  HttpServletRequest request, HttpServletResponse response,
                                  ResponseMetaData metaData)
            throws IOException, ServletException {
        WebAppContext context = createContext(contentType, request, response, metaData);
        Content content = contentProcessor.build(buffer, context);
        if (content == null) {
            return false;
        }
     
        String[] decoratorPaths = decoratorSelector.selectDecoratorPaths(content, context);
        //遍历装饰器进行装饰
        for (String decoratorPath : decoratorPaths) {
            content = context.decorate(decoratorPath, content);
        }
     
        if (content == null) {
            return false;
        }
        try {
            content.getData().writeValueTo(response.getWriter());
        catch (IllegalStateException ise) {  // If getOutputStream() has already been called
            content.getData().writeValueTo(new PrintStream(response.getOutputStream()));
        }
        return true;
    }
     
    public Content decorate(String decoratorName, Content content) throws IOException {
        if (decoratorName == null) {
            return null;
        }
     
        class CharBufferWriter extends CharArrayWriter {
            public CharBuffer toCharBuffer() {
                return CharBuffer.wrap(this.buf, 0this.count);
            }
        }
        CharBufferWriter out = new CharBufferWriter();
        decorate(decoratorName, content, out);
     
        CharBuffer decorated = out.toCharBuffer();
     
        Content lastContent = currentContent;
        currentContent = content;
        try {
            return contentProcessor.build(decorated, this);
        finally {
            currentContent = lastContent;
        }
    }
     
    @Override
    protected void decorate(String decoratorPath, Content content, Writer out) throws IOException {
        HttpServletRequest filterableRequest = new HttpServletRequestFilterable(request);
        // Wrap response so output gets buffered.
        HttpServletResponseBuffer responseBuffer = new HttpServletResponseBuffer(response, metaData, new BasicSelector(new PathMapper<Boolean>(), includeErrorPages) {
            @Override
            public boolean shouldBufferForContentType(String contentType, String mimeType, String encoding) {
                return true// We know we should buffer.
            }
        });
        responseBuffer.setContentType(response.getContentType()); // Trigger buffering.
     
        // It's possible that this is reentrant, so we need to take a copy
        // of additional request attributes so we can restore them afterwards.
        Object oldContent = request.getAttribute(CONTENT_KEY);
        Object oldContext = request.getAttribute(CONTEXT_KEY);
     
        request.setAttribute(CONTENT_KEY, content);
        request.setAttribute(CONTEXT_KEY, this);
     
        try {
            // Main dispatch.
            dispatch(filterableRequest, responseBuffer, decoratorPath);
     
            // Write out the buffered output.
            CharBuffer buffer = responseBuffer.getBuffer();
            out.append(buffer);
        catch (ServletException e) {
            //noinspection ThrowableInstanceNeverThrown
            throw (IOException) new IOException("Could not dispatch to decorator").initCause(e);
        finally {
            // Restore previous state.
            request.setAttribute(CONTENT_KEY, oldContent);
            request.setAttribute(CONTEXT_KEY, oldContext);
        }
    }
     
    protected void dispatch(HttpServletRequest request, HttpServletResponse response, String path)
            throws ServletException, IOException {
        //这里调用加载文件会出现404错误 /WEB-INF/decorators/decorator.ftl
        RequestDispatcher dispatcher = servletContext.getRequestDispatcher(path);
        if (dispatcher == null) {
            throw new ServletException("Not found: " + path);
        }
        dispatcher.forward(request, response);
    }

SiteMesh3整合SpringMVC+FreeMarker的更多相关文章

  1. sonne_game网站开发03 spring-mvc+freemarker整合

    今天的任务就是在spring+mybatis+springmvc的基础上,将freemarker整合进来. freemarker是什么? freemarker是一种模板引擎.它的目的是基于模板和数据, ...

  2. spring-mvc+freemarker整合(sonne_game网站开发03)

    今天的任务就是在spring+mybatis+springmvc的基础上,将freemarker整合进来. freemarker是什么? freemarker是一种模板引擎.它的目的是基于模板和数据, ...

  3. SpringBoot整合系列-整合SpringMVC

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9984607.html SpringBoot整合Spring MVC 步骤 第一步:添加必 ...

  4. freemarker学习 (servlet + freemarker -> Struts2+freemarker -> springMVC+freemarker)

    什么是freemarker? freemarker类似于jsp,但不是jsp!怎么说呢?freemarker文件后缀是.ftl,它不像jsp本质是servlet,它将构建模板.解析模板.使用模板分离开 ...

  5. CXF WebService整合SpringMVC的maven项目

    首先推荐博客:http://www.cnblogs.com/xdp-gacl/p/4259481.html   http://blog.csdn.net/hu_shengyang/article/de ...

  6. SpringMVC+Freemarker+JSTL支持

    前提: 网页编程中,我的思路是,通用的模块不仅仅只有后台代码,前端页面也可以独立为模块. 这个和asp.net中的UserController很像 比如有个人员基本信息的展示界面,需要在多个界面中嵌入 ...

  7. springMVC+freemarker中Could not resolve view with name... 问题解决

    用到springMVC+freemarker,自己在做demo过程中报: 严重: Servlet.service() for servlet springmvc threw exception jav ...

  8. spring整合springmvc和hibernate

    上篇文章使用maven搭建了web环境,这篇来记录下如何使用spring整合springmvc和hibernate,亦即spring+springmvc+hibernate框架整合. 第一步:首先配置 ...

  9. springmvc 项目完整示例07 设置配置整合springmvc springmvc所需jar包springmvc web.xml文件配置

    前面主要是后台代码,spring以及mybatis的整合 下面主要是springmvc用来处理请求转发,展现层的处理 之前所有做到的,完成了后台,业务层和持久层的开发完成了 接下来就是展现层了 有很多 ...

随机推荐

  1. 接收新信息,在会话中看不到(thread表数据插入/更新失败)

    分析原因:收到短信,sms表插入信息,触发器会自动更新thread表,更新失败导致一直有一条未读信息数量显示,但在会话列表中却看不到. (偶现,低概率. 解决方法:接收新信息插入后,立即查询threa ...

  2. ATL 获取flash信息

    // This goes past the ATL includes #import "C:/WINDOWS/system32/Macromed/Flash/Flash9e.ocx" ...

  3. ZOJ3790_Consecutive Blocks

    给出一个数组,最多可以删除k个数,问能够获得的最长的一个数字连续段为多少? 把所有相同的数字都提取出来,保存取得每个数字需要删除的数字,然后二分枚举就可以了. 召唤代码君: #include < ...

  4. JS-随机函数

    // alert( Math.round(3.4) );// 0~1 : Math.round(Math.random());// 0~10// alert( Math.round(Math.rand ...

  5. 复星昆仲杨光:VR行业四大痛点

    2016年,可是说是VR产业的爆发之年,从公司层面到资本层面都对VR产业给予了厚望,期望其能够在移动互联网之后带来革命性的突破,掀起新一轮技术主导的商业革命.而创业者们已经跃跃欲试,期望在资本的支持下 ...

  6. 算法入门笔记------------Day1

    1.C语言使用%d显示float值,不会把float值转换为近似的int值,而是显示垃圾值,使用%f显示int值,也不会把该int值转换为浮点值 2.三位数反转:输入一个三位数,分离它的百位,十位和个 ...

  7. maven 简介

    本书代码下载 大家可以从我的网站下载本书的代码:http://www.juvenxu.com/mvn-in-action/,也可以通过我的网站与我取得联系,欢迎大家与我交流任何关于本书的问题和关于Ma ...

  8. 你真的已经搞懂JavaScript了吗?

    题目一: if (!("a" in window)) { var a = 1; } alert(a); 题目二: var a = 1, b = function a(x) { x ...

  9. tomcat 8在win8.1中的配置

    配置jdk: 三个步骤: 1.环境变量=>新建 JAVA_HOME C:\Program Files\Java\jdk1.8.0_40 2.环境变量=>新建 CLASS_PATH .;%J ...

  10. 《理解 ES6》阅读整理:函数(Functions)(五)Name Property

    名字属性(The name Property) 在JavaScript中识别函数是有挑战性的,因为你可以使用各种方式来定义一个函数.匿名函数表达式的流行使用导致函数调试困难,在栈信息中难以找出函数名. ...