一 表單介紹
// 在HTML中,表單是由<form>元素來表示的,而在JavaScript中,表單對應的則是HTMLFormElement類型;
1 // HTMLFormElement繼承了HTMLElement;因此它擁有HTML元素具有的默認屬性,別且還獨有自己的屬性和方法; 2 HTMLFormElement屬性和方法 3 屬性或方法 說明 4 acceptCharset 服務器能夠處理的字符集; 5 action 接受請求的URL; 6 elements 表單中所有控件的集合; 7 enctype 請求的編碼類型; 8 length 表單中控件的數量; 9 method 要發送的HTTP請求類型,通常是'get'或'post'; 10 name 表單的名稱; 11 target 用於發送請求和接受響應的窗口名稱; 12 reset() 將所有表單重置; 13 submit() 提交表單;
1 1.獲取表單<form>對象; 2 document.getElementById('myForm'); // 使用ID獲取<form>元素; 3 document.getElementsByTagName('form')[0]; // 使用獲取form元素集合里的第一個元素; 4 document.forms[0]; // 使用forms的數字下標獲取元素; 5 document.forms['myForm']; // 使用forms的名稱下標獲取元素;
1 2.提交表單 2 (1).通過事件對象,可以阻止submit的默認行為,submit事件的默認行為就是攜帶數據跳轉到指定頁面; 3 addEvent(fm,'submit',function(evt){ 4 preDef(evt); 5 }); 6 7 (2).我們可以使用submit()方法來自定義觸發submit事件;也就是說,並不一定非要點擊submit按鈕才能提交; 8 if(e.ctrlKey && e.keyCode ==13){ 9 fm.submit(); // 判斷按住了ctrl和enter鍵觸發提交; 10 } 11 // PS:在表單中盡量避免使用name="submit"或id="submit"等命名,這會和submit()方法發生沖突導致無法提交; 12 13 (3).重復提交 14 // 當一條數據提交到服務器的時候會出現延遲等長時間沒反映,導致用戶不停的點擊提交, 15 // 從而使得重復提交了很多相同的請求,或造成錯誤或寫入數據庫多條相同信息; 16 addEvent(fm,'submit',function(evt){ // 模擬服務器延遲; 17 preDef(evt); 18 setTimeout(function(){ // 3秒后才處理提交到服務器; 19 fm.submit(); 20 },3000); 21 }); 22 23 // 解決重復提交方案 24 // 第一種:提交之后,立刻禁用點擊按鈕; 25 document.getElementById('sub').disabled = true; // 將按鈕禁用; 26 // 第二種:提交之后,取消后續的表單提交操作; 27 var flag = false; // 設置一個監聽變量; 28 if(flag == true)return; // 如果存在則返回退出事件; 29 flag = true; // 否則確定是第一次,改變監聽變量的值;
1 3.重置表單 2 // 用戶點擊重置按鈕時,表單會被初始化; 3 // 雖然這個按鈕還得以保留,但目前Web已經很少去使用了;因為用戶填好信息后,不小心點擊了重置就會全部清空,用戶體驗極差; 4 // 有兩種方法調用reset事件:第一個就是直接type="reset"即可;第二個就是使用fm.reset()方法調用; 5 <input type="reset" value="重置" /> // 不需要JS代碼即可實現; 6 addEvent(document,'click',function(){ 7 fm.reset(); // 使用JS方法實現重置; 8 });
1 4.表單字段 2 // 表單處理中,建議使用HTMLDOM,它有自己的elements屬性,該屬性是表單中所有元素的集合; 3 fm.elements[0]; // 獲取第一個表單字段元素; 4 fm.elements['user']; // 獲取name值是user的表單字段元素; 5 fm.elements.length; // 獲取所有表單字段的數量; 6 7 (1).共有的表單字段屬性 8 // 除了<fieldset>元素之外,所有表單字段都擁有相同的一組屬性; 9 屬性 說明 10 disabled 布爾值,表示當前字段是否被禁用; 11 form 指向當前字段所屬表單的指針,只讀; 12 name 當前字段的名稱; 13 readOnly 布爾值,表示當前字段是否只讀; 14 tabIndex 表示當前字段的切換; 15 type 當前字段的類型; 16 value 當前字段的值; 17 fm.elements[0].value; // 獲取和設置value; 18 fm.elements[0].disabled = true; // 禁用當前字段; 19 20 (2).共有的表單字段方法 21 方法 說明 22 focus() 將焦點定位到表單字段里; 23 blur() 從元素中將焦點移走; 24 25 (3).共有的表單字段事件 26 事件名 說明 27 blur 當字段失去焦點時觸發; 28 change 對於<input>和<textarea>元素,在改變value並失去焦點時觸發;對於<select>元素,在改變選項時觸發; 29 focus 當前字段獲取焦點時觸發; 30 31 // 利用focus事件修改文本框的背景色; 32 // 利用change事件在用戶輸入非數值字符時再次修改背景色; 33 var bextbox = document.forms[0].elements[0]; 34 EventUtil.addHandler(textbox,'focus',function(evt){ 35 evt = EventUtil.getEvent(evt); 36 var target = EventUtil.getTarget(evt); 37 if(target.style.backgroundColor != 'red'){ 38 target.style.backgroundColor = 'yellow'; // 選中狀態時文本框的背景是黃色; 39 } 40 }); 41 42 EventUtil.addHandler(textbox,'blur',function(evt){ // 失去焦點事件; 43 evt = EventUtil.getEvent(evt); 44 var target = EventUtil.getTarget(evt); 45 if(/[^\d]/.test(target.value)){ // 輸入非數值字符時; 46 target.style.backgroundColor = 'red'; // 文本框背景變成紅色; 47 }else{ 48 target.style.backgroundColor = ''; 49 } 50 }); 51 52 EventUtil.addHandler(textbox,'change',function(evt){ // 改變value值且失去焦點事件; 53 evt = EventUtil.getEvent(evt); 54 var target = EventUtil.getTarget(evt); 55 if(/[^\d]/.test(target.value)){ 56 target.style.backgroundColor = 'red'; // 文本框變成紅色; 57 }else{ 58 target.style.backgroundColor = ''; 59 } 60 });
二 文本框腳本
// 在HTML中,有兩種方式來表現文本框:
// 一種是單行文本框<input type="text">;
// 一種是多行文本框<textarea>;
1 1.獲取value值 2 // 雖然<input>在字面上有value值,而<textarea>卻沒有,但可以通過value獲取它們的值; 3 var textField = fm.elements[0]; 4 var areaField = fm.elements[1]; 5 alert(textField.value+','+areaField.value); // 得到<input>和<textarea>的value值; 6 // PS:使用表單的value是最推薦的,它是HTMLDOM中的屬性,不建議使用標准的DOM的方法getAttribute(); 7 // 原因是:對value屬性的修改,不一定反映在DOM中; 8 9 // defaultValue:得到原本的value值;不會因為值的改變而變化; 10 alert(textField.defaultValue); // 得到最初設置的value值;
2.選擇文本 // 使用select()方法,可以將文本框里的內容選中,並將焦點設置到文本框中; textField.select(); // 在文本框獲得焦點時選擇其所有文本; // 選取部分文本 // 在使用文本框內容的時候,我們有時要直接選定部分文本,這個行為還沒有標准; // Firefox的解決方案是:setSelectionRange()方法;它接受兩個參數:索引和長度; textField.setSelectionRange(0,1); // 選擇第一個字符; textField.setSelectionRange(0,textField.value.length); // 選擇全部; // IE8以下不支持這種寫法,可以使用IE的范圍操作代替; var range = textField.createTextRange(); // 創建一個文本范圍對象; range.collapse(true); // 將指針移動到起點; range.moveStart('character',0); // 移動指針,character表示逐字移動; range.moveEnd('character',1); // 移動終點; range.select(); // 焦點選定; // 選擇部分文本兼容函數 function selectText(text,start,stop){ if(text.setSelectionRange){ text.setSelectionRange(start,stop); text.focus(); }else if(text.createTextRange){ var range = text.createTextRange(); range.collapse(true); range.moveStart('character',start); range.moveEnd('character',sotp-start); range.select(); } } // 與select()方法對應的,是一個select事件,可以選中文本框文本后觸發; addEvent(textField,'select',function(){ alert(this.value); // IE事件需要傳遞this才可以這么寫; });
1 // 獲得選擇的文本 2 // Firefox提供了兩個屬性:selectionStart和selectionEnd; 3 addEvent(textField,'select',function(){ 4 alert(this.value.substring(this.selectionStart,this.selectionEnd)); 5 }); 6 // IE8以下,提供了另一個方案:selection對象,屬於document; 7 // 這個對象保存着用戶在整個文檔范圍內選擇的文本信息; 8 // 兼容函數 9 function getSelecText(text){ 10 if(typeof text.selectioinStart =='number'){ // 非IE; 11 return text.value.substring(text.selectionStart,text.selectionEnd); 12 }else if(document.selection){ // IE; 13 return document.selection.createRange().text; // 獲取IE選擇的文本; 14 } 15 } 16 // PS:存在問題:IE在觸發select事件的時候,在選擇一個字符后立即觸發,而其他瀏覽器是選擇想要的字符后釋放鼠標鍵時再觸發; 17 // 所以使用alert()的話,導致跨瀏覽器的不兼容; 18 // 所以我們可以通過將得到的選擇文本賦值給別的對象; 19 addEvent(textField,'select',function(){ 20 // alert(getSelecText(this)); // 導致用戶行為結果不一致; 21 document.getElementById('box').innerHTML = getSelecText(this); 22 })
1 3.過濾輸入 2 // 為了使文本框輸入指定的字符,我們必須對輸入進文本框的字符進行驗證; 3 addEvent(areaField,'keypress',function(evt){ 4 var e = evt || window.event; 5 var charCode = getCharCode(evt); // 得到字符編碼; 6 if(!/\d/.test(String.formCharCode(charCode)) && charCode>9 && !e.ctrlKey){ 7 preDef(evt); // 條件阻止默認; 8 } 9 }); 10 // PS:前半段條件判斷只有數字才可以輸入,導致常規按鍵,比如光標鍵/退格鍵/刪除鍵等無法使用; 11 // 部分瀏覽器比如Firefox,需要解放這些鍵,而非字符觸發的編碼均為0; 12 // 在Safari3之前的瀏覽器,也會被阻止,而它對應的字符編碼全部為8,所以最后加上charCode>9的判斷; 13 // 確保用戶沒有按下ctrl鍵;
1 // 阻止文本框裁剪/復制和粘貼; 2 事件名 說明 3 copy 在發生復制操作時觸發; 4 cut 在發生裁剪操作時觸發; 5 paste 在發生粘貼操作時觸發; 6 beforecopy 在發生復制操作時觸發; 7 beforecut 在發生裁剪操作時觸發; 8 beforepaste 在發生粘貼操作時觸發; 9 // 如果我們想要禁用裁剪/復制/粘貼,那么可以阻止默認行為即可; 10 addEvent(areaField,'cut',function(evt){ 11 preDef(evt); 12 }); 13 addEvent(areaField,'copy',function(evt){ 14 preDef(evt); 15 }); 16 addEvent(areaField,'paste',function(evt){ 17 preDef(evt); 18 }); 19 20 // 最后一個影響輸入的因素,就是:輸入法; 21 // 中文輸入法,它的原理是在輸入法面板上先儲存文本,按下回車就寫入英文文本,按下空格就寫入中文文本; 22 // 解決方案:通過CSS來禁止調出輸入法; 23 style='ime-mode:disabled'; // CSS直接編寫; 24 areaField.style.imeMode='disabled'; // 在JS中設置; 25 // PS:Chrome無法禁止輸入法調出,所以,最后使用正則驗證已輸入的文本; 26 addEvent(areaField,'keyup',function(evt){ 27 this.value = this.value.replace(/[^\d]/g,''); // 把非數字都替換成空; 28 });
1 4.自動切換焦點 2 // 為了增加表單字段的易用性,很多字段在滿足一定條件時(比如長度),就會自動切換到下一個字段上繼續填寫; 3 <input type='text' name='user1' maxlength='1'/> // 只能寫1個; 4 <input type='text' name='user2' maxlength='2'/> 5 <input type='text' name='user3' maxlength='3'/> 6 function tabForward(evt){ 7 var e = evt || window.event; 8 var target = getTarget(evt); 9 // 判斷當前長度是否和指定長度一致; 10 if(target.value.length == target.maxLength){ 11 // 遍歷所有字段; 12 for(var i=0; i<fm.elements.length; i++){ 13 // 找到當前字段; 14 if(fm.elements[i]==target){ 15 // 就把焦點移入下一個字段; 16 if(fm.elements[i+1]){ 17 fm.elements[i+1].focus(); 18 } 19 // 中途返回; 20 return; 21 } 22 } 23 } 24 }
三 選擇框腳本
1 // 選擇框是通過<select>和<option>元素創建的; 2 HTMLSelectElement對象 3 屬性/方法 說明 4 add(new,rel) 插入新元素,並指定位置; 5 multiple 布爾值,是否允許多項選擇; 6 options <option>元素的HTMLCollection集合; 7 remove(index) 移除給定位置的選項; 8 selectedIndex 基於0的選中項的索引,如果沒有選中項,則值為-1; 9 size 選擇框中可見的行數;
1 // 在DOM中,每個<option>元素都有一個HTMLOptionElement對象,以便訪問數據; 2 HTMLOptionElement對象 3 屬性 說明 4 index 當前選項在options集合中的索引; 5 label 當前選項的標簽; 6 selected 布爾值,表示當前選項是否被選中; 7 text 選項的文本; 8 value 選項的值;
1 var city = fm.elements['city']; // HTMLSelectElement; 2 alert(city.options); // HTMLOptionsCollection; 3 alert(city.options[0]); // HTMLOptionElement; 4 alert(city.type); // select-one; 5 // PS:選擇框里的type屬性有可能是:select-one,也有可能是:select-multiple; 6 // 這取決於HTML代碼中有沒有multiple屬性; 7 8 alert(city.options[0].text); // 上海text,獲取text值; 9 alert(city.options[0].value); // 上海value,獲取value值; 10 // PS:操作select時,最好使用HTMLDOM,以為瀏覽器兼容性比較好; 11 // 如果使用標准DOM,會因為不同的瀏覽器導致不同的結果; 12 // PS:當選項沒有value值的時候,IE會返回空字符串,其他瀏覽器會返回text值;
2.選擇選項 // 對於只能選擇一項的選擇框,使用selectedIndex屬性最為簡單; addEvent(city,'change',function(){ alert(this.selectedIndex); // 得到當前選項的索引,從0開始; alert(this.options[this.selectedIndex].text); // 得到當前選項的text值; alert(this.options[this.selectedIndex].value); // 得到當前選項的value值; }); city.selectedIndex = 1; // 設置selectedIndex可以定位某個索引; // 通過option的屬性selected(布爾值),也可以設置定位某個索引,設置為true即可; city.options[0].selected = true; // 設置第一個索引; // selected和selectedIndex區別: // 1.selected是返回的布爾值;所以一般用於判斷上; // 2.selectedIndex返回的是數值,一般用於設置和獲取; addEvent(city,'change',function(){ if(this.options[2].selected == true){ // 判斷第三個選項是否被選定; alert('正確!'); } });
1 3.添加選項 2 // 如需要動態的添加選項,我們提供兩種方案:DOM和Option構造函數; 3 (1).DOM 4 var option = document.createElement('option'); 5 option.appendChild(document.createTextNode('北京 text')); 6 option.setAttribute('value','北京 value'); 7 city.appendChild(option); 8 9 (2).Option構造函數 10 var option = new Option('北京 text','北京 value'); 11 city.appendChild(option); // IE出現bug; 12 13 (3).使用add()方法來添加選項: 14 var option = new Option('北京 text','北京 value'); 15 city.add(option,0); // 0,表示添加到第一位; 16 // PS:在DOM規定,add()中兩個參數是必須的,如果不正確索引,那么第二個參數設置null即可,即默認移入最后一個選項; 17 // 但在IE中第二個參數是可選的,所以設置null表示放入不存在的位置,導致失蹤; 18 // 為了兼容性,可以傳遞undefined即可兼容; 19 city.add(option,null); // IE不顯示了; 20 city.add(option,undefined); // 兼容;
1 4.移除選項 2 // 有三種方式可以移除某一個選項:DOM移除/remove()方法移除和null移除; 3 city.removeChild(city.option[0]); // DOM移除; 4 city.remove(0); // remove()移除,推薦; 5 city.options[0] = null; // null移除; 6 // PS:當第一項移除后,下面的項,往上移,所以不停的移除第一項,即可全部移除;
1 5.移動選項 2 // 如果有兩個選擇框,把第一個選擇框里的第一個選項移到第二個選擇框里,並且第一個選擇框里的第一項被移除; 3 var city = fm.elements['city']; // 第一個選擇框; 4 var info = fm.elements['info']; // 第二個選擇框; 5 info.appendChild(city.options[0]); // 移動,並在第一個選擇框中刪除;
1 6.排列選項 2 // 選擇框提供了一個index屬性,可以得到當前選項的索引值,和selectedIndex的區別是,一個是選擇框對象的調用,一個是選項對象的調用; 3 var option1 = city.options[1]; 4 city.insertBefore(option1,city.options[option1.index-1]); // 往上移位;
1 7.單選按鈕 2 // 通過checked屬性來獲取單選按鈕的值; 3 for(var i=0; i<fm.sex.length; i++){ // 循環單選按鈕; 4 if(fm.sex[i].checked == true){ // 遍歷每一個,找出處於選中狀態的那一個; 5 alert(fm.sex[i].value); // 得到值; 6 } 7 } 8 // PS:除了checked屬性之外,單選按鈕還有一個defaultChecked按鈕, 9 // 它獲取的是原本的checked按鈕對象,而不會因為checked的改變而改變; 10 if(fm.sex[i].defaultChecked == true){ 11 alert(fm.sex[i].value); 12 }
1 8.復選按鈕 2 // 通過checked屬性來獲取復選按鈕的值, 3 var love = ''; 4 for(var i=0; i<fm.love.length; i++){ 5 if(fm.love[i].checked == true){ 6 love += fm.love[i].value; 7 } 8 } 9 alert(love);