首页 > 代码库 > JFinal中使用JSP的自定义Tag解决I18N

JFinal中使用JSP的自定义Tag解决I18N

1. 设计原由

由于JFinal的国际化(I18N)支持在JSP中支持不好,因此,萌生了解决这一短板的念头。

实现时也考虑了几种方式,最终决定采用JSP中最原始的标签。因为自定义标签在JSP中容易实现,内容灵活且功能比较强大,可扩展性好。

 

2. I18N标签

自定义的I18N标签需要针对I18N的各个接口做最好的支持,使用<jf:i18n />作为标签名,下面是JFinalI18N类的几个接口:

public static String getText(String key)
public static String getText(String key, String defaultValue)
public static String getText(String key, Locale locale)
public static String getText(String key, String defaultValue, Locale locale)


完整的标签被设计成:

<jf:i18n key="" defaultValue=http://www.mamicode.com/"" locale="" paras="" />

        参数说明:

参数名

作用

说明

key

对应接口中的key

 

defaultValue

对应接口中的defaultValue

 

locale

对应接口中的locale

数据类型有差别,这里只能使用字串,如:zh_CN, en_US

paras

全新的属性,为了支持参替换而设计

1.假如key对应字串为“早上好!{0}们,现在正实验{1}。”;paras取值为“小白,I18N标签”(参数间用逗号隔开);那么得到的最终结果是“早上好!小白们,现在正实验I18N标签。”

2.另外,参数可以从Controller的attr中取值,例如:Controller中setAttr(p1小白).setAttr(p2I18N标签);标签中的paras取值为“p1,p2”,寻么最终结果还是“早上好!小白们,现在正实验I18N标签。”

3. 代码实现

3.1 I18nTag.java

package com.jfinal.tag;

import java.io.IOException;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspWriter;
import javax.servlet.jsp.tagext.Tag;
import javax.servlet.jsp.tagext.TagSupport;

import com.jfinal.i18n.I18N;
import com.jfinal.kit.LocaleKit;
import com.jfinal.kit.StringKit;

/**
 * JSP页面中用于支持I18N的标签。
 */
public class I18nTag extends TagSupport {

	private static final long serialVersionUID = -8073376431317433802L;

	
	/**
	 * I18N中的key值
	 */
	private String key;

	/**
	 * 当key值不存在时使用的默认值
	 */
	private String defaultValue;

	/**
	 * 地区属性,如:zh_CN, en_US
	 */
	private String locale;

	/**
	 * 作为value格式化值使用的参数,多个用逗号隔开,如果参数值在Request中有对应的attribute,则取attribute值
	 */
	private String paras;

	@SuppressWarnings("deprecation")
	@Override
	public int doStartTag() throws JspException {
//		if (StringKit.isBlank(this.getKey())) {
//			throw new JspException("The tag attribute of key is not exists.");
//		}

		// 定义输出给页面的text
		String text = null;

		try {
			if (StringKit.isBlank(locale)) {
				// 通过I18N接口拿到值
				text = I18N.getText(key, this.defaultValue);
			} else {
				// locale定义了值,说明指定了前端要显示的语言类型,语言类型交由LocaleKit处理
				text = I18N.getText(key, this.defaultValue, I18N.localeFromString(locale));
			}
		} catch (Exception e) {
			text = defaultValue;
		}

		if (StringKit.notBlank(paras)) {
			// 如果tag中指定了paras,则将paras解析为array
			String[] attrs = paras.split(",");
			Object[] values = new Object[attrs.length];

			// 循环将参数到Request中取值,如果有值,则替换
			for (int i = 0; i < attrs.length; i++) {
				String a = attrs[i];
				values[i] = pageContext.getRequest().getAttribute(a) == null ? attrs[i] : pageContext
						.getRequest().getAttribute(a);
			}

			pageContext.getAttribute("");

			text = String.format(text, values);
		}

		try {
			// 将结果输出到页面
			pageContext.getOut().write(text);
		} catch (IOException e) {
			return Tag.SKIP_BODY;
		}

		return Tag.EVAL_BODY_INCLUDE;
	}

	public String getKey() {
		return this.key;
	}

	public void setKey(String key) {
		this.key = key;
	}

	public String getDefaultValue() {
		return this.defaultValue;
	}

	public void setDefaultValue(String defaultValue) {
		this.defaultValue = defaultValue;
	}

	public String getLocale() {
		return this.locale;
	}

	public void setLocale(String locale) {
		this.locale = locale;
	}

	public String getParas() {
		return this.paras;
	}

	public void setParas(String paras) {
		this.paras = paras;
	}


3.2 Jfinal.tld

    文件与I18nTag.java放在同一目录下:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN" "http://java.sun.com/j2ee/dtds/web-jsptaglibrary_1_1.dtd">
<taglib>
    <tlibversion>1.0</tlibversion>
    <jspversion>1.1</jspversion>
    <shortname>bean</shortname>
    <uri>http://www.jfinal.com/tag</uri>
    <tag>
        <name>i18n</name>
        <tagclass>com.jfinal.tag.I18nTag</tagclass>
        <bodycontent>JSP</bodycontent>
        <attribute>
            <name>key</name>
            <required>true</required>
            <rtexprvalue></rtexprvalue>
        </attribute>
        <attribute>
            <name>defaultValue</name>
            <required>false</required>
            <rtexprvalue></rtexprvalue>
        </attribute>
        <attribute>
            <name>paras</name>
            <required>false</required>
            <rtexprvalue></rtexprvalue>
        </attribute>
        <attribute>
            <name>locale</name>
            <required>false</required>
            <rtexprvalue></rtexprvalue>
        </attribute>
    </tag>
</taglib>


4. Tag的使用

4.1 定义多国语言属性文件

  •         myi18n_en_US.properties

greeting=Today is %2$s/%3$s/%1$s


  •         myi18n_zh_CN.properties

greeting=今天是%1$s年%2$s月%3$s日


4.2 启动时加载I18

在JFinalConfig.configConstant(Constants me)中加入如下代码:

// 载入I18N文件
me.setI18n("myi18n", Locale.SIMPLIFIED_CHINESE, Integer.MAX_VALUE);

4.3 在Controller中设置属性

Calendar c = Calendar.getInstance();
c.setTime(new Date());

this.setAttr("yyyy", c.get(Calendar.YEAR));
this.setAttr("MM", c.get(Calendar.MONTH) + 1);
this.setAttr("dd", c.get(Calendar.DAY_OF_MONTH));

4.4 JSP页面中定义标签

<%@ taglib prefix="jf" uri="/WEB-INF/classes/com/jfinal/tag/jfinal.tld"%>

<h5><jf:i18n key="non_key" defaultValue=http://www.mamicode.com/"没有key值显示我" paras="2014,7,8" />> 

    首先引入标签,引入的是标签定义文件tld的位置,编译后,tld的位置为"/WEB-INF/classes/com/jfinal/tag/jfinal.tld",它也是目标位置。


4.5 执行结果


5. 问题

细心的读者可能已经发现了,paras参数是为替换{0},{1}这类型参数,结果却替换了%1$s年%2$s这样的参数。这是因为代码使用String.format()格式化,却不知为什么不能支持{0},{1}。作者没时间去探究,哪位读者有时间帮我找出答案吧。