まず、投稿したコードでは、リダイレクト前に 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 メソッドを呼び出します。数秒待ってからリダイレクトします (次のページにリダイレクトする非表示のボタンをプログラムでクリックするなど)。私の意見では、これは問題に値するものではなく、ログイン プロセスを遅らせるだけです。いずれにせよ、ユーザーはホーム ページ (または送信先のページ) にリダイレクトされるため、tlogin が成功したことを知ることができます。
編集:
メソッドが終了するとメッセージがページに表示されるため、マネージド Bean メソッドで待機しても機能しません。FacesMessage を追加した後、使用します
RequestContext.getCurrentInstance().execute("waitAndRedirect()");
また、xhtml には、次のような JavaScript 関数が必要です。
function waitAndRedirect() {
setTimeout(function() {
hiddenButtonId.click();
}, 2000);
}
hiddenButtonId は ap:button の ID で、ホームページにリダイレクトされ、非表示になっています (display:none)
しかし、繰り返しになりますが、これは厄介なアプローチです。私の意見では、これを行う必要はありません。ログインプロセスを遅らせるだけです。