首先,使用您发布的代码,您在重定向之前不会看到 FacesMessage,您会在重定向之后看到它。而且,为了实现这一点,您需要添加一个过滤器,因为重定向时消息会丢失。这是您需要的过滤器的代码(不要忘记在 web.xml 中声明它):
public class MultiPageMessagesSupport implements PhaseListener {
private static final long serialVersionUID = 1250469273857785274L;
private static final String sessionToken = "MULTI_PAGE_MESSAGES_SUPPORT";
@Override
public PhaseId getPhaseId() {
return PhaseId.ANY_PHASE;
}
/*
* Check to see if we are "naturally" in the RENDER_RESPONSE phase. If we
* have arrived here and the response is already complete, then the page is
* not going to show up: don't display messages yet.
*/
@Override
public void beforePhase(final PhaseEvent event) {
FacesContext facesContext = event.getFacesContext();
int msg = this.saveMessages(facesContext);
if (PhaseId.RENDER_RESPONSE.equals(event.getPhaseId())) {
if (!facesContext.getResponseComplete()) {
this.restoreMessages(facesContext);
}
}
}
/*
* Save messages into the session after every phase.
*/
@Override
public void afterPhase(final PhaseEvent event) {
if (event.getPhaseId() == PhaseId.APPLY_REQUEST_VALUES ||
event.getPhaseId() == PhaseId.PROCESS_VALIDATIONS ||
event.getPhaseId() == PhaseId.INVOKE_APPLICATION) {
FacesContext facesContext = event.getFacesContext();
int msg = this.saveMessages(facesContext);
}
}
@SuppressWarnings("unchecked")
private int saveMessages(final FacesContext facesContext) {
List<FacesMessage> messages = new ArrayList<FacesMessage>();
for (Iterator<FacesMessage> iter = facesContext.getMessages(null); iter.hasNext();) {
messages.add(iter.next());
iter.remove();
}
if (messages.isEmpty()) {
return 0;
}
Map<String, Object> sessionMap = facesContext.getExternalContext().getSessionMap();
List<FacesMessage> existingMessages = (List<FacesMessage>) sessionMap.get(sessionToken);
if (existingMessages != null) {
existingMessages.addAll(messages);
} else {
sessionMap.put(sessionToken, messages);
}
return messages.size();
}
@SuppressWarnings("unchecked")
private int restoreMessages(final FacesContext facesContext) {
Map<String, Object> sessionMap = facesContext.getExternalContext().getSessionMap();
List<FacesMessage> messages = (List<FacesMessage>) sessionMap.remove(sessionToken);
if (messages == null) {
return 0;
}
int restoredCount = messages.size();
for (Object element : messages) {
facesContext.addMessage(null, (FacesMessage) element);
}
return restoredCount;
}
}
如果这对您不起作用,并且您需要在之前显示消息,那么您必须执行以下操作:使方法返回 void,通过 ajax 调用它,并在添加成功消息后调用一些 javascript 方法这将等待几秒钟,然后进行重定向(可能通过以编程方式单击重定向到下一页的隐藏按钮)。我认为这不值得麻烦,您只会延迟登录过程。无论如何,用户都会知道登录成功,因为他将被重定向到主页(或您发送给他的任何页面)
编辑:
当方法完成时,消息会显示在页面中,因此在托管 bean 方法中等待将不起作用。添加 FacesMessage 后,使用
RequestContext.getCurrentInstance().execute("waitAndRedirect()");
在您的 xhtml 中,您需要有一个类似于此的 javascript 函数:
function waitAndRedirect() {
setTimeout(function() {
hiddenButtonId.click();
}, 2000);
}
其中 hiddenButtonId 是 ap:button 的 ID,它重定向到主页并被隐藏 (display:none)
但同样,这是一种讨厌的方法,在我看来没有必要这样做,你只会延迟登录过程。