9

成功メッセージを表示し、5 秒などのタイムアウト後にユーザーを別のページにリダイレクトするにはどうすればよいですか?

ログインが成功した後のログインページにこれが必要です。次のことを試してみましたが、ログイン失敗時に警告メッセージが表示されますが、ログイン成功時の成功メッセージは表示されません。目的のページがすぐに表示されます。

public String check(){
      if (username.equals("test") && password.equals("test")) {
          FacesContext.getCurrentInstance().addMessage(null, new FacesMessage(FacesMessage.SEVERITY_INFO,"Sample info message", "PrimeFaces rocks!")); 
            return "Success";
        }else{
          FacesContext.getCurrentInstance().addMessage(null, new FacesMessage(FacesMessage.SEVERITY_WARN,"Sample warn message", "Watch out for PrimeFaces!"));  
            return "Failure";
        }
    }

ナビゲーションには Seam の PageFlow を使用しています。

私は

<p:messages id="messages" showDetail="true" autoUpdate="true" closable="true" />

ログインページで。

4

3 に答える 3

17

Flash のユーティリティの 1 つです。それ以外の

FacesContext.getCurrentInstance().addMessage(null, new FacesMessage(FacesMessage.SEVERITY_INFO,"Sample info message", "PrimeFaces rocks!"));:

このコードを使用するだけです

FacesContext facesContext = FacesContext.getCurrentInstance();
Flash flash = facesContext.getExternalContext().getFlash();
flash.setKeepMessages(true);
flash.setRedirect(true);
facesContext.addMessage(null, new FacesMessage(FacesMessage.SEVERITY_INFO,"Sample info message", "PrimeFaces rocks!"));
于 2013-04-04T12:32:59.030 に答える
7

まず、投稿したコードでは、リダイレクト前に 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)

しかし、繰り返しになりますが、これは厄介なアプローチです。私の意見では、これを行う必要はありません。ログインプロセスを遅らせるだけです。

于 2013-04-04T11:57:41.043 に答える