i18n с файлами свойств в кодировке UTF-8 в приложении JSF 2.0 - PullRequest
9 голосов
/ 05 сентября 2010

Я использую jsf-ri 2.0.3, где требуется поддержка иврита и русского языка. Проблема в том, что вместо правильного текста я вижу тарабарщину на экране.

Прежде всего я определил пакеты (* _locale.properties) для каждого языка. Файлы в кодировке UTF-8. Во-вторых, я определил язык по умолчанию и поддерживаемые локали в face-config.xml

<locale-config>
    <default-locale>iw</default-locale>
    <supported-locale>en</supported-locale>
    <supported-locale>ru</supported-locale>
</locale-config>

Чем я добавил пользовательский фильтр, который установит кодировку ответа 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 ]

22 голосов
/ 05 сентября 2010

Правильно, вы можете создать пользовательский 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 и т. Д. В пакете com.example.i18n,Нет необходимости в native2ascii.

Кстати, с новым объявлением стиля <resource-bundle> в JSF 2.0 в faces-config.xml вам больше не нужно <f:loadBundle> в представлениях.Весь текст будет доступен сразу по #{text} во всех видах.

4 голосов
/ 05 сентября 2010

Ну, после глубокого изучения я нашел решение.

Раньше в Java 1.6 PropertyResourceBundle был только один конструктор со следующей документацией The property file read with this constructor must be encoded in ISO-8859-1. Это означает, что можно использовать только английскийтекст в комплектах ресурсов.

Существует два решения этой проблемы:

Первое - это запись пользовательского компонента loadBundle, который будет использовать правильный ResourceBundle метод создания экземпляров.

Второй (Мой выбор) использует конвертер Native-ASCII , который можно использовать с maven с помощью плагина Native2Ascii maven .

Здесьпример конфигурации:

<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>
0 голосов
/ 07 января 2013

У меня та же проблема с отдельным приложением SWT.Это модифицированный загрузчик ресурсов, сгенерированный WindowBuilder.Основная идея - класс Messages содержит только ресурсы в строковых полях.Поэтому я конвертирую их в UTF8 (если это возможно) после сырой загрузки 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
            }
        }
    }
}

}

...