<< 2010/07/06 | Home | 2010/07/08 >>
PR: 転職    お墓    エコ    通販    結婚相談所    シルバー    質屋    葬式    漫画    エステサロン   

JSF2で、バリデーションエラーが起きると、inputHiddenの値が失なわれる。

こちらは回避策を見つけたけど、やはり結構困る問題。

import javax.enterprise.context.RequestScoped;
import javax.inject.Named;

@RequestScoped
@Named
public class Bug2 {
long seed;
String value;
String value2;

public String getValue() {return value;}
public void setValue(String value) {this.value = value;}

public String getValue2() {return value2;}
public void setValue2(String value2) {this.value2 = value2;}

public long getSeed() {return seed;}
public void setSeed(long seed) {this.seed = seed;}

public void perform() {
seed = System.currentTimeMillis();
}
}
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<h:form>
<h:inputHidden value="#{bug2.seed}"/>

<h:panelGrid columns="3">
<h:outputLabel value="seed"/>
<h:outputText id="seed" value="#{bug2.seed}"/>
<h:message for="seed"/>

<h:outputLabel value="Value"/>
<h:inputText id="value" value="#{bug2.value}" required="true"/>
<h:message for="value"/>

<h:outputLabel value="Value2"/>
<h:inputText id="value2" value="#{bug2.value2}" required="true"/>
<h:message for="value2"/>
</h:panelGrid>

<h:commandButton value="Submit" action="#{bug2.perform}"/>
</h:form>
</html>

こんな風に操作する。

  • Value, Value2に適当な値'1', '2'とかを入れて、Submitをclick。
  • seedのところに、値が生成される。
  • Valueのところを空にして、Submitをclick。
  • required="true"なので、バリデーションエラーになるが、Value2の内容は残っているのに、なぜかseedの方(これはinputHiddenで保持)が消えてしまって0になる。

JSF2では、bookmarkable urlが簡単に作れるので、この手のread-onlyパラメータは、リクエストパラメータとして持たせておけば、回避できるんだけど、ちょっと面倒なんで、バグ報告してみた

P.S. これ、もうちょっと調べてみたら、hiddenが消えているのではなくて、出力コンポーネントの値が失なわれているということが分かった。エラーがあってページを再表示する時に、出力コンポーネントから、モデルのオブジェクトを見に行くと、エラーになっているから存在していなくて、何も表示できないってことみたいだ。入力コンポーネント系なら、submitted valueを書き戻せるけど、出力コンポーネントはダメってことらしい。入力コンポーネントをreadonlyにしても、やっぱり消えちゃうようだ。とりあえずJavaScriptで、hiddenの値をコピーしてやるくらいしか方法は無さそう。

JSF2で、renderedを指定していると、イベントが配信されない。

JSF2でページ制御のコンポジットコンポーネントを作ってみたら、やっかいな現象に見舞われた。

import javax.enterprise.context.RequestScoped;
import javax.inject.Named;

@RequestScoped
@Named
public class Bug1 {
int value;

public int getValue() {return value;}
public void setValue(int value) {this.value = value;}

public void decrease() {--value;}
public void increase() {++value;}
}
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<h:form>
<h:inputHidden value="#{bug1.value}"/>
<h:commandButton value="-" action="#{bug1.decrease}" rendered="#{bug1.value gt 0}"/>
<h:outputText value="#{bug1.value}"/>
<h:commandButton value="+" action="#{bug1.increase}"/>
</h:form>
</html>

これ、実行すると、整数の両側に、'-'、'+'ボタンが表示され、'-'は、値が1以上の時にだけ表示されるんだけど、'-'ボタンが全く機能しない。

どうやら、以下のようなことが起きているっぽい。

  • '-'ボタンクリック。
  • JSFがBug1インスタンスを生成。
  • JSFがイベントを配信しようとするが、その時rendered属性をチェック。でも、Bug1はnewされたばかりなのでvalueフィールドは0なので、rendered=false。このためイベントの配信を中止。
  • リクエストの値を、Bug1インスタンスに反映。ここでようやくvalueフィールドが設定されるのだけど、時、既に遅し。

inputHiddenのところに、immediate='true'したら大丈夫かと思ったけど、なぜか効果無し(なんで?)。

仕様なのかもしれないけど、これって困るよなぁ。とりあえずバグ報告してみた。xhtmlのところ思いっきり文字化けしてしまった。なんで、ここのBTSには、preview機能が無いんだろう...

CDIのsession scopeビーン

ログイン状態を保持するために、以下のようなビーンを作る。

@Named @SessionScoped
public class LoginSession implements Serializable {
private static final long serialVersionUID = -5522770756920818033L;
Long loginId;

public boolean isLogined() {return logined != null;}
public void logoff() {loginId = null;}
public long getLoginId() {return loginId;}
}

でも、これを直接別のビーンにインジェクトしたり、Viewから参照すると、

@Inject LoginSession loginSession;

問答無用で、セッションが開始してしまうので困る。ログインしてからセッションは開始して欲しい。CDIでは、Instanceというインターフェイスがあるのだけど、

@Inject Instance<loginsession> loginSessionFactory;

ここで、Instanceには自分が欲しいメソッドが無いことが分かる。get()を呼ぶと、やはり問答無用でインスタンス作っちゃうし。欲しいのはget(false)みたいなやつ。セッション上にビーンが無ければ、nullを返し、存在したら、そのインスタンスを作ってくれるようなやつ。

ちょっと悩んでみたが、解決策が見つからないんで、WeldのUser Forumに質問してみた。

このサイトの掲載内容は私自身の見解であり、必ずしもIBMの立場、戦略、意見を代表するものではありません。
日本アイ・ビー・エム 花井 志生 Since 1997.6.8