9

在我的应用程序中,用户应该能够切换区域设置(用于在页面上呈现文本的语言)。大量教程使用 FacesContext.getCurrentInstance().getViewRoot().setLocale()。例如:http ://www.mkyong.com/jsf2/jsf-2-internationalization-example/ 。但是,这在 JSF 2.0 中根本不起作用(它在 1.2 中确实起作用)。语言从不切换。没有错误或任何东西。相同的代码在 JSF 1.2 中运行良好。

什么是正确和确定的方法?我拼凑了一个解决方案,但不确定这是否正确。这工作正常。用户点击英语或法语后,语言切换。这是给你一些想法的代码片段。

@ManagedBean(name = "switcher")
@SessionScoped
public class LanguageSwitcher {
    Locale locale = FacesContext.getCurrentInstance().getViewRoot().getLocale();
    public String switchLocale(String lang) {
        locale = new Locale(lang);

        return FacesContext.getCurrentInstance().getViewRoot().getViewId() +
            "?faces-redirect=true";
    }
    //getLocale() etc. omitted for brevity
}

XHTML:

<f:view locale="#{switcher.locale}">
    <h:outputText value="#{msg.greeting}" />
    <h:commandLink value="English" action="#{switcher.switchLocale('en')}" />
    <h:commandLink value="French" action="#{switcher.switchLocale('fr')}" />
</f:view>

只是为了给你更多信息,这里是配置文件。

<application>
    <locale-config>
        <supported-locale>en</supported-locale>
        <supported-locale>fr</supported-locale>
    </locale-config>
    <resource-bundle>
        <base-name>com.resources.Messages</base-name>
        <var>msg</var>
    </resource-bundle>
</application>

再一次,这行得通。但是,我没有通过以任何方式调用任何 API 来更改 JSF 本身的语言环境。这让我有些毛骨悚然的感觉。这是更改用户语言环境的正确方法吗?

4

3 回答 3

9

好的,冒着回答我自己问题的风险,我想总结一下我发现的所有不同方法。

基本方法是我已经在做的。也就是说,在会话范围内有一个托管 bean,它返回用户的 Locale。这个语言环境需要在每个使用<f:view locale="...">. 我从 BalusC 的一篇文章中学到了这项技术,所以在这里感谢。

现在,需要关注的是 f:view 元素的使用。这需要在每一页中重复,如果被错误地省略,这可能是缺陷的来源。我找到了解决这个问题的几种方法。

方法 #1:创建一个 Facelet 模板并在其中添加 f:view 元素。单个模板用户页面不必担心添加此元素。

方法#2 使用相位监听器。@meriton 已在此处发布解决方案。谢谢你。

方法 #3 使用自定义视图处理程序,它扩展 MultiViewHandler 并从 calculateLocale() 方法返回用户的区域设置。这在《Beginning JSF 2 APIs and JBoss Seam》一书中有所描述,作者:Kent Ka Iok Tong。这是本书中稍作改动的示例:

public class MyViewHandler extends MultiViewHandler {
    public Locale calculateLocale(FacesContext context) {
        HttpSession session = (HttpSession) context.getExternalContext()
            .getSession(false);
        if (session != null) {
            //Return the locale saved by the managed bean earlier
            Locale locale = (Locale) session.getAttribute("locale");
            if (locale != null) {
                return locale;
            }
        }
       return super.calculateLocale(context);
    }
}

然后在faces config中注册它。

<application>
    <view-handler>com.package.MyViewHandler</view-handler>
<!-- Other stuff ... -->
</application>

这比相位监听器要优雅一些。不幸的是,MultiViewHandler 是 com.sun.faces.application.view 包中的一个内部非 API 类。这会带来一些未来的风险。

无论是方法#2 还是#3,页面中都不需要 f:view 元素。

于 2012-08-23T14:58:45.650 回答
2

可以使用自定义视图处理程序,该处理程序javax.faces.application.ViewHandlerWrapper从 calculateLocale() 方法扩展和返回用户的语言环境。

这绝对比MultiViewHandler从专有的 SUN 包扩展更好com.sun.faces.application.view,无论您的建议中提到的“开始 JSF 2 API ”一书中描述了什么。除此之外,您的原始方法绝对可以:

public class MyViewHandler extends ViewHandlerWrapper {

  public Locale calculateLocale(FacesContext context) {
    HttpSession session = (HttpSession) context.getExternalContext()
        .getSession(false);
    if (session != null) {
      //Return the locale saved by the managed bean earlier
      Locale locale = (Locale) session.getAttribute("locale");
      if (locale != null) {
        return locale;
      }
    }
    return super.calculateLocale(context);
  }
}

然后在faces config中注册它。

    <application>
      <view-handler>com.package.MyViewHandler</view-handler>
      <!-- Other stuff ... -->
    </application>
于 2015-09-19T10:38:10.907 回答
1

我最近遇到了一个相关的问题。在我的例子中,JSF 实现UIViewRoot.setLocale()在导航到另一个视图后忘记了视图区域设置。我宁愿认为这是 JSF impl 中的一个错误,但我没有时间确定。

我并不特别喜欢这种<f:view>方法,因为该标签已被 facelets 淘汰——似乎除了保留语言环境。这让我很想将它包含在 Facelets 模板中。因此,我编写了以下 PhaseListener:

/**
 * PhaseListener that keeps the current view locale in the session while no request is being processed, to work around
 * bugs where JSF forgets the changed locale.
 */
public class SaveViewLocaleToSessionPhaseListener implements PhaseListener {

    private static final String key = "locale";

    @Override
    public PhaseId getPhaseId() {
        return PhaseId.ANY_PHASE;
    }

    @Override
    public void beforePhase(PhaseEvent event) {
        // do nothing
    }

    @Override
    public void afterPhase(PhaseEvent event) {
        PhaseId currentPhase = event.getPhaseId();
        if (currentPhase == PhaseId.RESTORE_VIEW) {
            viewRoot().setLocale((Locale) sessionMap().get(key));
        } else if (currentPhase == PhaseId.RENDER_RESPONSE) {
            sessionMap().put(key, viewRoot().getLocale());
        }
    }

    private  Map<String, Object> sessionMap() {
        return FacesContext.getCurrentInstance().getExternalContext().getSessionMap();
    }

    private UIViewRoot viewRoot() {
        return FacesContext.getCurrentInstance().getViewRoot();
    }
}

但是,我无法提供任何确凿的证据证明这确实比简单地使用<f:view>.

但是,我没有以任何方式更改 JSF 本身的语言环境。

当然你做到了:<f:view>标签从值表达式中读取语言环境,并将其传递给UIViewRoot.setLocale().

于 2012-08-22T18:47:51.813 回答