Struts2默認theme是xhtml,它用表格來對表單中的控件進行排版。它也提供一個客戶端的js驗證功能,但是它的js腳本卻有些問題,在某些情況下,前次驗證的提示信息無法被清除,提示信息會不斷的累積顯示在屏幕上。而按照設計,每次提交表單時應只顯示每次驗證的出錯信息。
它的客戶端驗證的流程大概是這樣,用戶提交表單時,對各個控件的輸入按預先設置的規(guī)則進行驗證,如果有問題,則清除表單里原有的出錯提示信息,并寫入新的提示。其設計的功能是把出錯信息寫表格里出錯控件的上方,以便用戶看得更加清楚。問題就出在其用來清除原出錯信息的函數(shù),其代碼是這樣的(在struts.jar的template/xhtml目錄下可以找到):
1 function clearErrorMessages(form) {
2
3 var table = form.childNodes[1];
4 if( typeof table == "undefined" ) {
5 table = form.childNodes[0];
6 }
7
8 // clear out any rows with an "errorFor" attribute
9 var rows = table.rows;
10 var rowsToDelete = new Array();
11 if (rows == null){
12 return;
13 }
14
15 for(var i = 0; i < rows.length; i++) {
16 var r = rows[i];
17 if (r.getAttribute("errorFor")) {
18 rowsToDelete.push(r);
19 }
20 }
21
22 // now delete the rows
23 for (var i = 0; i < rowsToDelete.length; i++) {
24 var r = rowsToDelete[i];
25 table.deleteRow(r.rowIndex);
26 //table.removeChild(rowsToDelete[i]);
27 }
28 }
看這個函數(shù)的前三行,它試圖取得form的第1個或第2個子節(jié)點,并把它作為table來處理(看接下來的幾行)。要想清除表格里的錯誤信息,首先要取得表格本身,這沒錯,但是
如果第1個或第2個子節(jié)點不是table的話,腳本就會出錯,造成原出錯信息無法清除,這樣每次提交后的提示信息就會累積在屏幕上。
要解決這個問題有兩個辦法:
- 寫代碼時要小心,保證form的第1或2個子節(jié)點是table,不要在生成table前加其他代碼。
- 或,修改xhtml的validation.js,使它總能獲得正確的table元素,重新打包到struts.jar。
剛看了一下Struts的JIRA,已經(jīng)有人報告了這個問題(id
WW-1802),而且這個bug在2.1版本中已經(jīng)解決了。