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に質問してみた。





