需要生成一个菜单树,使用了freemarker,由于采用了tiles2,在和freemarker整合的时候出现了问题,所以直接使用freemarker 2.3.15将模板通过template.process方法产生html code显示到前端,主要代码如下:
[code="java"]
//产生freemarker config
package project.util.freemarker;
import freemarker.cache.ClassTemplateLoader;
import freemarker.template.Configuration;
import freemarker.template.DefaultObjectWrapper;
import freemarker.template.TemplateExceptionHandler;
public class FreeMarkerService {
private static Configuration cfg = new Configuration();
static {
//设置模版加载的路径,相对于FreemarkerService类路径下的templates路径
cfg.setTemplateLoader(new ClassTemplateLoader(FreeMarkerService.class,
"ftl"));
//设置对象包装器
cfg.setObjectWrapper(new DefaultObjectWrapper());
//设置默认编码
cfg.setDefaultEncoding("UTF-8");
//设置异常处理器
cfg.setTemplateExceptionHandler(TemplateExceptionHandler.IGNORE_HANDLER);
}
public static Configuration getConfiguration() {
return cfg;
}
}
[/code]
[code="java"]
//freemarker工具类,将数据和模板合成html
package project.util.freemarker
import java.io.IOException;
import java.io.StringWriter;
import java.io.Writer;
import java.util.Map;
import freemarker.template.Configuration;
import freemarker.template.Template;
import freemarker.template.TemplateException;
public class FreeMarkerUtil {
@SuppressWarnings("rawtypes")
public static String dataToString(Map data,String templatePath) throws IOException,TemplateException{
if(data == null){
return null;
}
Writer out = null;
Configuration cfg = FreeMarkerService.getConfiguration();
Template template = cfg.getTemplate(templatePath);
out = new StringWriter();
template.process(data,out);
return out.toString();
}
}
[/code]
freemarker 模板
[code="html"]
<#-- 定义宏buildNode -->
<#macro buildNode child parent>
<ul>
<@buildNode child=rootMenu.children parent=rootMenu/>
</ul>
[/code]
[code="java"]
//实体类
package project.bean;
import java.util.ArrayList;
import java.util.List;
import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import org.hibernate.annotations.GenericGenerator;
import org.hibernate.annotations.Parameter;
@Entity
@Table(name="RBAC_MENUS")
public class Menu {
private int menuId;
private String menuName;
private String description;
private Menu parent;
private List
public Menu(){}
@Id
@GeneratedValue(generator = "paymentableGenerator")
@GenericGenerator(name = "paymentableGenerator", strategy = "sequence", parameters= {@Parameter(name="sequence",value="RBAC_MENU_SEQ")})
@Column(name="MENU_ID")
public int getMenuId() {
return menuId;
}
public void setMenuId(int menuId) {
this.menuId = menuId;
}
@Column(name="MENU_NAME")
public String getMenuName() {
return menuName;
}
public void setMenuName(String menuName) {
this.menuName = menuName;
}
@ManyToOne(cascade = {CascadeType.MERGE,CascadeType.REFRESH }, optional = true)
@JoinColumn(name="PARENT_MENU_ID")
public Menu getParent() {
return parent;
}
public void setParent(Menu parent) {
this.parent = parent;
}
@OneToMany(cascade = { CascadeType.REFRESH, CascadeType.PERSIST,
CascadeType.MERGE, CascadeType.REMOVE },mappedBy ="parent")
public List<Menu> getChildren() {
return children;
}
public void setChildren(List<Menu> children) {
this.children = children;
}
public void addChildren(Menu menu){
if(this.children == null){
this.children = new ArrayList<Menu>();
}
this.children.add(menu);
}
@OneToMany(cascade = { CascadeType.REFRESH, CascadeType.PERSIST,CascadeType.MERGE, CascadeType.REMOVE },mappedBy ="menu")
public List<Function> getFunctions() {
return functions;
}
public void setFunctions(List<Function> functions) {
this.functions = functions;
}
@Column(name="IS_DEL")
public String getIsDel() {
return isDel;
}
public void setIsDel(String isDel) {
this.isDel = isDel;
}
@Column(name="description")
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
}
[/code]
[code="java"]
//Struts action
package project.web;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.annotation.Resource;
import project.bean.Menu;
import project.service.MenuService;
import project.util.freemarker.FreeMarkerUtil;
import com.opensymphony.xwork2.ActionSupport;
import org.apache.struts2.ServletActionContext;
public class MenuAction extends ActionSupport {
...
@Resource private MenuService menuService;
private String menuTree;
public String getMenuTree(){
return menuTree;
}
...
@SuppressWarnings({ "rawtypes", "unchecked" })
public String initMenu(){
Menu rootMenu = menuService.getRootMenu();
try{
Map freemarkerData = new HashMap();
freemarkerData.put("rootMenu", rootMenu);
freemarkerData.put("webRoot", ServletActionContext.getServletContext().getContextPath());
menuTree = FreeMarkerUtil.dataToString(freemarkerData, "menumgmt.ftl");
}
catch(Exception e){
e.printStackTrace();
}
return ActionSupport.SUCCESS;
}
...
[/code]
[code="java"]
//Menu Service实现类
public class MenuServiceImpl implements MenuService {
@Resource private SessionFactory sessionFactory;
...
//得到根菜单
@SuppressWarnings("unchecked")
@Override
public Menu getRootMenu() {
List<Menu> menus = sessionFactory.getCurrentSession().createQuery("from Menu m where m.parent is null").list();
if(menus != null && menus.size()>0)
return menus.get(0);
return null;
}
//插入菜单
@Override
public void insertMenu(Menu menu) {
sessionFactory.getCurrentSession().persist(menu);
}
...
}
[/code]
jsp代码
[code="html"]
...
异常信息
...
Hibernate:
select
menu0_.MENU_ID as MENU1_2_,
menu0_.description as descript2_2_,
menu0_.IS_DEL as IS3_2_,
menu0_.MENU_NAME as MENU4_2_,
menu0_.PARENT_MENU_ID as PARENT5_2_
from
RBAC_MENUS menu0_
where
menu0_.PARENT_MENU_ID is null
Hibernate:
select
children0_.PARENT_MENU_ID as PARENT5_1_,
children0_.MENU_ID as MENU1_1_,
children0_.MENU_ID as MENU1_2_0_,
children0_.description as descript2_2_0_,
children0_.IS_DEL as IS3_2_0_,
children0_.MENU_NAME as MENU4_2_0_,
children0_.PARENT_MENU_ID as PARENT5_2_0_
from
RBAC_MENUS children0_
where
children0_.PARENT_MENU_ID=?
[ERROR] [freemarker.log.Log4JLoggerFactory$Log4JLogger.error(Log4JLoggerFactory.java:96)] er.runtime -
Expression parent.parent is undefined on line 3, column 43 in menumgmt.ftl.
==> ${parent.parent.menuId} [on line 3, column 41 in menumgmt.ftl]
freemarker.core.InvalidReferenceException: Expression parent.parent is undefined on line 3, column 43 in menumgmt.ftl.
at freemarker.core.TemplateObject.assertNonNull(TemplateObject.java:124)
at freemarker.core.TemplateObject.invalidTypeException(TemplateObject.java:134)
at freemarker.core.Dot._getAsTemplateModel(Dot.java:78)
at freemarker.core.Expression.getAsTemplateModel(Expression.java:89)
at freemarker.core.Expression.getStringValue(Expression.java:93)
at freemarker.core.DollarVariable.accept(DollarVariable.java:76)
at freemarker.core.Environment.visit(Environment.java:209)
at freemarker.core.MixedContent.accept(MixedContent.java:92)
at freemarker.core.Environment.visit(Environment.java:209)
at freemarker.core.Macro$Context.runMacro(Macro.java:168)
at freemarker.core.Environment.visit(Environment.java:602)
at freemarker.core.UnifiedCall.accept(UnifiedCall.java:106)
at freemarker.core.Environment.visit(Environment.java:209)
at freemarker.core.MixedContent.accept(MixedContent.java:92)
at freemarker.core.Environment.visit(Environment.java:209)
at freemarker.core.Environment.process(Environment.java:189)
at freemarker.template.Template.process(Template.java:237)
at project.rbac.util.freemarker.FreeMarkerUtil.dataToString(FreeMarkerUtil.java:24)
at project.rbac.web.MenuAction.initMenu(MenuAction.java:91)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at com.opensymphony.xwork2.DefaultActionInvocation.invokeAction(DefaultActionInvocation.java:404)
at com.opensymphony.xwork2.DefaultActionInvocation.invokeActionOnly(DefaultActionInvocation.java:267)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:229)
at com.opensymphony.xwork2.interceptor.DefaultWorkflowInterceptor.doIntercept(DefaultWorkflowInterceptor.java:221)
at com.opensymphony.xwork2.interceptor.MethodFilterInterceptor.intercept(MethodFilterInterceptor.java:86)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.validator.ValidationInterceptor.doIntercept(ValidationInterceptor.java:150)
at org.apache.struts2.interceptor.validation.AnnotationValidationInterceptor.doIntercept(AnnotationValidationInterceptor.java:48)
at com.opensymphony.xwork2.interceptor.MethodFilterInterceptor.intercept(MethodFilterInterceptor.java:86)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ConversionErrorInterceptor.intercept(ConversionErrorInterceptor.java:123)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ParametersInterceptor.doIntercept(ParametersInterceptor.java:184)
at com.opensymphony.xwork2.interceptor.MethodFilterInterceptor.intercept(MethodFilterInterceptor.java:86)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.StaticParametersInterceptor.intercept(StaticParametersInterceptor.java:105)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.interceptor.CheckboxInterceptor.intercept(CheckboxInterceptor.java:83)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.interceptor.FileUploadInterceptor.intercept(FileUploadInterceptor.java:207)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ModelDrivenInterceptor.intercept(ModelDrivenInterceptor.java:74)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ScopedModelDrivenInterceptor.intercept(ScopedModelDrivenInterceptor.java:127)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.interceptor.ProfilingActivationInterceptor.intercept(ProfilingActivationInterceptor.java:107)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.interceptor.debugging.DebuggingInterceptor.intercept(DebuggingInterceptor.java:206)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ChainingInterceptor.intercept(ChainingInterceptor.java:115)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.I18nInterceptor.intercept(I18nInterceptor.java:143)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.PrepareInterceptor.doIntercept(PrepareInterceptor.java:121)
at com.opensymphony.xwork2.interceptor.MethodFilterInterceptor.intercept(MethodFilterInterceptor.java:86)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.interceptor.ServletConfigInterceptor.intercept(ServletConfigInterceptor.java:170)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.AliasInterceptor.intercept(AliasInterceptor.java:123)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at com.opensymphony.xwork2.interceptor.ExceptionMappingInterceptor.intercept(ExceptionMappingInterceptor.java:176)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:224)
at com.opensymphony.xwork2.DefaultActionInvocation$2.doProfiling(DefaultActionInvocation.java:223)
at com.opensymphony.xwork2.util.profiling.UtilTimerStack.profile(UtilTimerStack.java:455)
at com.opensymphony.xwork2.DefaultActionInvocation.invoke(DefaultActionInvocation.java:221)
at org.apache.struts2.impl.StrutsActionProxy.execute(StrutsActionProxy.java:50)
at org.apache.struts2.dispatcher.Dispatcher.serviceAction(Dispatcher.java:504)
at org.apache.struts2.dispatcher.FilterDispatcher.doFilter(FilterDispatcher.java:422)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.springframework.orm.hibernate3.support.OpenSessionInViewFilter.doFilterInternal(OpenSessionInViewFilter.java:198)
at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:96)
at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233)
at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191)
at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127)
at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:102)
at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109)
at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:298)
at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:857)
at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:588)
at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489)
at java.lang.Thread.run(Thread.java:662)
Hibernate:
select
children0_.PARENT_MENU_ID as PARENT5_1_,
children0_.MENU_ID as MENU1_1_,
children0_.MENU_ID as MENU1_2_0_,
children0_.description as descript2_2_0_,
children0_.IS_DEL as IS3_2_0_,
children0_.MENU_NAME as MENU4_2_0_,
children0_.PARENT_MENU_ID as PARENT5_2_0_
from
RBAC_MENUS children0_
where
children0_.PARENT_MENU_ID=?
Hibernate:
select
children0_.PARENT_MENU_ID as PARENT5_1_,
children0_.MENU_ID as MENU1_1_,
children0_.MENU_ID as MENU1_2_0_,
children0_.description as descript2_2_0_,
children0_.IS_DEL as IS3_2_0_,
children0_.MENU_NAME as MENU4_2_0_,
children0_.PARENT_MENU_ID as PARENT5_2_0_
from
RBAC_MENUS children0_
where
children0_.PARENT_MENU_ID=?
...
菜单能够正常展现,但是这个异常为什么发生呢,我还碰到一个问题就是当我选择一个菜单项,并且插入一个子菜单的时候,发生一个异常,并且页面也无法展现最新插入的那个菜单,需要刷新一下才能看到新插入的菜单
程序界面如下:
[img]http://hiphotos.baidu.com/lucky_sonic/pic/item/c81382cf5b0ce869f9dc612a.jpg[/img]
[quote]当我客户端插入一个新的菜单的时候,由Action返回,菜单树上面新的菜单没有被加载,html代码中也看不到,我在Action中调试,发现查询出来的数据是对的,包含了新插入的菜单,我就怀疑是否是freemarker的问题,不知是否能够给出解答. [/quote]
因为我没有你的开发环境,也不知道你菜单生成的机制,也不知道调用结构。但你可以通过debug看看数据在哪里出错:
1.action
2.freemarker渲染数据
3.界面某些js出错
可以用一些js调试技巧发现问题,很多问题会是界面的问题,比如一些空格、标点、参数不对、id重复等都有可能照成界面出不来。
freemarker里面层级对象引用xx.xx.xx,每一级都需要判断是否为空,否则抛异常。
[code="java"]Expression parent.parent is undefined on line 3, column 43 in menumgmt.ftl. [/code]
看到没有,说parent.parent未定义。结贴吧。哈哈。
给出我以前的代码,以供参考:
[code="java"]<#--
Description:栏目模板(图片Tab方式,上面横向展示商品,有图片,一行五列;下面只展示商品内容,无图片)
Author: Peter Wei
Date: 2010-1-26
-->
<#--
参数配置,如果ctx,images有值传进来,则用传进来的值;没有,则赋值.
-->
<#if !(ctx?exists && ctx?has_content)>
<#assign ctx = "/web" />
</#if>
<#if !(images?exists && images?has_content)>
<#assign images = "/web/images" />
</#if>
<div class="tab_content">
<#if root.subList?exists>
<#list root.subList as subList>
<div class="tab_item">
<ul> <#if subList.productList?exists>
<#list subList.productList as product>
<li>
<div class="sku-img"><a href="${ctx}/product/${product.sid}.html"><img
src="${images}/${product.proPicture?if_exists.proPictDir?if_exists}/${product.proPicture?if_exists.proPictName?if_exists}"
alt="${product.proSku?if_exists}"/>
<div class="zhekou">
${product.proPrice?if_exists.offValue?default("0")}折
</div>
</a>
</div>
<div><a href="${ctx}/product/${product.sid}.html">
<p>${product.productName?if_exists}</p>
</a>
<h3>
<em>${product.proPrice?if_exists.originalPrice?if_exists}</em>¥${product.proPrice?if_exists.promotionPrice?if_exists}
</h3>
<span class="color666666">
<#if (product.productCates?exists)&&(product.productCates?size>=1)>
<a href="${ctx}/list--${product.productCates[0].sid}.html">更多同类商品>></a>
<#else>
<a href="${ctx}/list-${product.brandSid}.html">更多同类商品>></a>
</#if>
</span>
</div>
</li>
<#if (product_index>=4)> <#break> </#if>
</#list>
</#if>
</ul>
<ul class="tab-list">
<#assign i=0 />
<#if subList.productList?exists>
<#list subList.productList as product>
<#if (product_index>4)>
<#if (i%3==0)&&(i>2)>
</ul>
<ul class="tab-list">
</#if>
<li><a href="${ctx}/product/${product.sid}.html"><span
class="ff6600">[立省${product.proPrice?if_exists.saveMoney?if_exists}
元]</span>${product.productName?if_exists}
</a>
</li>
<#if (product_index>=10)> <#break> </#if>
<#assign i=i+1 />
</#if>
</#list>
</#if>
</ul>
</div>
</#list>
</#if>
</div>