i18n с файлами свойств в кодировке UTF-8 в приложении JSF 2.0
Я использую jsf-ri 2.0.3, где требуется поддержка иврита и русского языка. Проблема в том, что я вижу кракозябры на экране вместо правильного текста.
прежде всего, я определил расслоения (*_locale.свойства) для каждого языка. Файлов в кодировке UTF-8. Во-вторых, я определил стандартные и поддерживаемые локали в faces-config.в XML
<locale-config>
<default-locale>iw</default-locale>
<supported-locale>en</supported-locale>
<supported-locale>ru</supported-locale>
</locale-config>
чем я добавил пользовательский фильтр, который установит кодировку charcter ответа UTF-8.
<filter>
<filter-name>encodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
<init-param>
<param-name>forceEncoding</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
и, наконец, когда я создаю простой xhtml для отладки вывода, я вижу очень странные результаты
<f:loadBundle basename="i18n.frontend.homepage" var="msg"/>
<strong>i18n: </strong><h:outputText value="#{msg.language}"/>
<br/>
<strong>Locale: </strong>
<h:outputText value="#{facesContext.externalContext.response.locale}"/>
<br/>
<strong>Encoding: </strong>
<h:outputText value="#{facesContext.externalContext.response.characterEncoding}"/>
результат:
i18n: ×¢×ר×ת
Locale: en_US
Encoding: UTF-8
что не так с моей конфигурации?
3 ответов
правильно, вы можете создать пользовательский ResourceBundle
или используйте конвертер native2ascii (при необходимости с плагином Maven 2, чтобы сделать преобразование более прозрачным). Поскольку другой ответ идет только с последним подходом подробно, вот еще один ответ, как вы могли бы создать пользовательский ResourceBundle
для загрузки файлов свойств как UTF-8 в JSF 2.X приложение на базе Java SE 1.6 окружающая среда.
faces-config.xml
<application>
<resource-bundle>
<base-name>com.example.i18n.Text</base-name>
<var>text</var>
</resource-bundle>
</application>
com.example.i18n.Text
package com.example.i18n;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;
import java.net.URLConnection;
import java.util.Enumeration;
import java.util.Locale;
import java.util.PropertyResourceBundle;
import java.util.ResourceBundle;
import javax.faces.context.FacesContext;
public class Text extends ResourceBundle {
protected static final String BUNDLE_NAME = "com.example.i18n.text";
protected static final String BUNDLE_EXTENSION = "properties";
protected static final String CHARSET = "UTF-8";
protected static final Control UTF8_CONTROL = new UTF8Control();
public Text() {
setParent(ResourceBundle.getBundle(BUNDLE_NAME,
FacesContext.getCurrentInstance().getViewRoot().getLocale(), UTF8_CONTROL));
}
@Override
protected Object handleGetObject(String key) {
return parent.getObject(key);
}
@Override
public Enumeration<String> getKeys() {
return parent.getKeys();
}
protected static class UTF8Control extends Control {
public ResourceBundle newBundle
(String baseName, Locale locale, String format, ClassLoader loader, boolean reload)
throws IllegalAccessException, InstantiationException, IOException
{
// The below code is copied from default Control#newBundle() implementation.
// Only the PropertyResourceBundle line is changed to read the file as UTF-8.
String bundleName = toBundleName(baseName, locale);
String resourceName = toResourceName(bundleName, BUNDLE_EXTENSION);
ResourceBundle bundle = null;
InputStream stream = null;
if (reload) {
URL url = loader.getResource(resourceName);
if (url != null) {
URLConnection connection = url.openConnection();
if (connection != null) {
connection.setUseCaches(false);
stream = connection.getInputStream();
}
}
} else {
stream = loader.getResourceAsStream(resourceName);
}
if (stream != null) {
try {
bundle = new PropertyResourceBundle(new InputStreamReader(stream, CHARSET));
} finally {
stream.close();
}
}
return bundle;
}
}
}
это ожидает UTF-8 закодированных файлов свойств, таких как text.properties
, text_en.properties
, etc в . Нет необходимости в native2ascii.
кстати, с новым стилем JSF 2.0 <resource-bundle>
декларация faces-config.xml
, вам не нужно <f:loadBundle>
в представлениях больше. Весь текст будет доступен непосредственно #{text}
во всех представлениях.
Ну, после глубокого исследования я нашел решение.
ранее на java 1.6 PropertyResourceBundle
только один конструктор, который имеет следующие документы The property file read with this constructor must be encoded in ISO-8859-1.
Это означает, что в пакетах ресурсов можно использовать только английский текст.
есть два решения этой проблемы:
первый из них пишет пользовательский компонент loadBundle, который будет использовать правильный ResourceBundle
метод создания экземпляра.
второй (мой выбор) использует родной-в-ASCII конвертер, который можно использовать с maven с помощью Native2Ascii мавен плагин.
вот пример конфигурации:
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>native2ascii-maven-plugin</artifactId>
<executions>
<execution>
<goals>
<goal>native2ascii</goal>
</goals>
<configuration>
<src>${basedir}/src/main/resources</src>
<dest>${project.build.directory}/native2ascii</dest>
<encoding>UTF8</encoding>
<includes>**/*.properties</includes>
</configuration>
</execution>
</executions>
</plugin>
У меня такая же проблема с автономным приложением SWT. Это измененный загрузчик ресурсов, созданный WindowBuilder. Основная идея-класс сообщений содержит только ресурсы в строковых полях. Поэтому я конвертирую их в UTF8 (если возможно) после загрузки raw ISO-8859-1.
import java.lang.reflect.Field;
import org.eclipse.osgi.util.NLS;
public class Messages extends NLS {
private static final String BUNDLE_NAME = "org.digimead.tabuddy.desktop.res.messages"; //$NON-NLS-1$
public static String MainWindow_newShell_text;
public static String MainWindow_actionOpenFile_text;
public static String MainWindow_actionCloseFile_text;
// //////////////////////////////////////////////////////////////////////////
//
// Constructor
//
// //////////////////////////////////////////////////////////////////////////
private Messages() {
// do not instantiate
}
// //////////////////////////////////////////////////////////////////////////
//
// Class initialization
//
// //////////////////////////////////////////////////////////////////////////
static {
// load message values from bundle file
NLS.initializeMessages(BUNDLE_NAME, Messages.class);
final Field[] fieldArray = Messages.class.getDeclaredFields();
final int len = fieldArray.length;
for (int i = 0; i < len; i++) {
final Field field = (Field) fieldArray[i];
if (field.getType() == java.lang.String.class) {
if (!field.isAccessible())
field.setAccessible(true);
try {
final String rawValue = (String) field.get(null);
field.set(null, new String(rawValue.getBytes("ISO-8859-1"),
"UTF-8"));
} catch (Exception e) {
// skip field modification
}
}
}
}
}