JavaScript DOM 學(xué)習(xí)第七章 表單的擴(kuò)展
更新時間:2010年02月19日 13:49:33 作者:
這一章我會處理一個簡單的W3C DOM腳本。他會幫助我們從一個新的角度來看待交互設(shè)計(jì)。
想法
假設(shè)你有一個在線的CD評級工具。你希望用戶查看他們喜歡的所有CD。但是你怎么知道用戶平均會查看多少張呢?你在這個頁面上需要添加多少字段呢?
在W3CDOM出現(xiàn)之前這確實(shí)是一個問題。假設(shè)你放置了7張CD。但是用戶很可能只想查看一張,太多可能會嚇著用戶,而有些用戶想查看自己的所有CD,這樣就不得不反復(fù)提交很多次。這確實(shí)很煩人。
只有使用W3C DOM才能讓用戶決定生成多少個字段。這個效果和之前的大不相同。
例子
當(dāng)你點(diǎn)擊發(fā)送的時候,表單就會把得到的所有參數(shù)以數(shù)組的形式發(fā)送。這用來檢查是否真正的發(fā)送到了服務(wù)器端。不幸的是,mac版的IE和Safari不能發(fā)送。
IE的問題
windows下的IE有兩個嚴(yán)重的問題:
第一個問題是所有生成的單選框都同屬于一個數(shù)組,即使他們的names不同。這樣用戶就只能在所有的單選框里面選擇一個。也就是說你不能在生成的表單里面實(shí)用單選框。
有讀者說通過innerHTML生成的單選框沒有問題。如果你非要實(shí)用單選框的話,你可以試試。
第二個問題是生成的表單通過傳統(tǒng)的document.forms無法訪問:IE沒有在數(shù)組里面包含他們。這個可以通過給他們設(shè)置ID來解決。
解釋
表單的HTML代碼:
<div id="readroot" style="display: none">
<input type="button" value="Remove review"
onclick="this.parentNode.parentNode.removeChild(this.parentNode);" /><br /><br />
<input name="cd" value="title" />
<select name="rankingsel">
<option>Rating</option>
<option value="excellent">Excellent</option>
<option value="good">Good</option>
<option value="ok">OK</option>
<option value="poor">Poor</option>
<option value="bad">Bad</option>
</select><br /><br />
<textarea rows="5" cols="20" name="review">Short review</textarea>
<br />Radio buttons included to test them in Explorer:<br />
<input type="radio" name="something" value="test1" />Test 1<br />
<input type="radio" name="something" value="test2" />Test 2
</div>
<form method="post" action="/cgi-bin/show_params.cgi">
<span id="writeroot"></span>
<input type="button" id="moreFields" value="Give me more fields!" />
<input type="submit" value="Send form" />
</form>
真正的表單項(xiàng)目在ID為readroot的DIV并且display值為none。這個DIV是一個模板,用戶不能修改。當(dāng)用戶需要更多的表單的時候我們就復(fù)制這個模板然后添加在表單之后。我們在一開始的就加載,這樣當(dāng)用戶打開頁面的時候就能看到。
這個DIV在表單之外,所以當(dāng)用戶提交表單的時候,這個模板的內(nèi)容不會被提交。
ID為writeroot的SPAN是一個標(biāo)記。新的生成的表單就插入在他的前面。
添加表單項(xiàng)目
下面的代碼可以再需要的時候用來添加表單項(xiàng)目:
var counter = 0;
function moreFields() {
counter++;
var newFields = document.getElementById('readroot').cloneNode(true);
newFields.id = '';
newFields.style.display = 'block';
var newField = newFields.childNodes;
for (var i=0;i<newField.length;i++) {
var theName = newField[i].name
if (theName)
newField[i].name = theName + counter;
}
var insertHere = document.getElementById('writeroot');
insertHere.parentNode.insertBefore(newFields,insertHere);
}
window.onload = moreFields;
首先我們需要一個計(jì)數(shù)器counter,因?yàn)樗械谋韱雾?xiàng)都應(yīng)該有唯一的一個名字。我們把計(jì)數(shù)器的值添加在生成name后面。初始化計(jì)數(shù)器:
然后是實(shí)際的函數(shù)。我們給計(jì)數(shù)器加1:
function moreFields() {
counter++;
然后復(fù)制我們的模板,移除ID,并且把display設(shè)置為block。readroot應(yīng)該是整個文檔里面唯一的ID,復(fù)制模板后應(yīng)該顯示出來讓用戶看到。
var newFields = document.getElementById('readroot').cloneNode(true);
newFields.id = '';
newFields.style.display = 'block';
我們遍歷這個拷貝的子元素:
var newField = newFields.childNodes;
for (var i=0;i<newField.length;i++) {
如果子元素有name屬性,那么我們就在name值上加上計(jì)數(shù)器的值,以保證他的唯一性:
var theName = newField[i].name
if (theName)
newField[i].name = theName + counter;
}
現(xiàn)在這個拷貝已經(jīng)準(zhǔn)備好插入了。我們把他插入到writeroot之前:
然后我們在頁面加載的時候就執(zhí)行一次,這樣用戶進(jìn)入的時候就能看到:
移除表單項(xiàng)
每一個模板的拷貝都有一個移除按鈕:
點(diǎn)擊這個按鈕就會從移除他的父元素(DIV)。整個生成的表單就都會消失,并且不會再出現(xiàn)。
翻譯地址:http://www.quirksmode.org/dom/domform.html
轉(zhuǎn)載請保留以下信息
作者:北玉(tw:@rehawk)
假設(shè)你有一個在線的CD評級工具。你希望用戶查看他們喜歡的所有CD。但是你怎么知道用戶平均會查看多少張呢?你在這個頁面上需要添加多少字段呢?
在W3CDOM出現(xiàn)之前這確實(shí)是一個問題。假設(shè)你放置了7張CD。但是用戶很可能只想查看一張,太多可能會嚇著用戶,而有些用戶想查看自己的所有CD,這樣就不得不反復(fù)提交很多次。這確實(shí)很煩人。
只有使用W3C DOM才能讓用戶決定生成多少個字段。這個效果和之前的大不相同。
例子
當(dāng)你點(diǎn)擊發(fā)送的時候,表單就會把得到的所有參數(shù)以數(shù)組的形式發(fā)送。這用來檢查是否真正的發(fā)送到了服務(wù)器端。不幸的是,mac版的IE和Safari不能發(fā)送。
IE的問題
windows下的IE有兩個嚴(yán)重的問題:
第一個問題是所有生成的單選框都同屬于一個數(shù)組,即使他們的names不同。這樣用戶就只能在所有的單選框里面選擇一個。也就是說你不能在生成的表單里面實(shí)用單選框。
有讀者說通過innerHTML生成的單選框沒有問題。如果你非要實(shí)用單選框的話,你可以試試。
第二個問題是生成的表單通過傳統(tǒng)的document.forms無法訪問:IE沒有在數(shù)組里面包含他們。這個可以通過給他們設(shè)置ID來解決。
解釋
表單的HTML代碼:
復(fù)制代碼 代碼如下:
<div id="readroot" style="display: none">
<input type="button" value="Remove review"
onclick="this.parentNode.parentNode.removeChild(this.parentNode);" /><br /><br />
<input name="cd" value="title" />
<select name="rankingsel">
<option>Rating</option>
<option value="excellent">Excellent</option>
<option value="good">Good</option>
<option value="ok">OK</option>
<option value="poor">Poor</option>
<option value="bad">Bad</option>
</select><br /><br />
<textarea rows="5" cols="20" name="review">Short review</textarea>
<br />Radio buttons included to test them in Explorer:<br />
<input type="radio" name="something" value="test1" />Test 1<br />
<input type="radio" name="something" value="test2" />Test 2
</div>
<form method="post" action="/cgi-bin/show_params.cgi">
<span id="writeroot"></span>
<input type="button" id="moreFields" value="Give me more fields!" />
<input type="submit" value="Send form" />
</form>
真正的表單項(xiàng)目在ID為readroot的DIV并且display值為none。這個DIV是一個模板,用戶不能修改。當(dāng)用戶需要更多的表單的時候我們就復(fù)制這個模板然后添加在表單之后。我們在一開始的就加載,這樣當(dāng)用戶打開頁面的時候就能看到。
這個DIV在表單之外,所以當(dāng)用戶提交表單的時候,這個模板的內(nèi)容不會被提交。
ID為writeroot的SPAN是一個標(biāo)記。新的生成的表單就插入在他的前面。
添加表單項(xiàng)目
下面的代碼可以再需要的時候用來添加表單項(xiàng)目:
復(fù)制代碼 代碼如下:
var counter = 0;
function moreFields() {
counter++;
var newFields = document.getElementById('readroot').cloneNode(true);
newFields.id = '';
newFields.style.display = 'block';
var newField = newFields.childNodes;
for (var i=0;i<newField.length;i++) {
var theName = newField[i].name
if (theName)
newField[i].name = theName + counter;
}
var insertHere = document.getElementById('writeroot');
insertHere.parentNode.insertBefore(newFields,insertHere);
}
window.onload = moreFields;
首先我們需要一個計(jì)數(shù)器counter,因?yàn)樗械谋韱雾?xiàng)都應(yīng)該有唯一的一個名字。我們把計(jì)數(shù)器的值添加在生成name后面。初始化計(jì)數(shù)器:
復(fù)制代碼 代碼如下:
var counter = 0;
然后是實(shí)際的函數(shù)。我們給計(jì)數(shù)器加1:
復(fù)制代碼 代碼如下:
function moreFields() {
counter++;
然后復(fù)制我們的模板,移除ID,并且把display設(shè)置為block。readroot應(yīng)該是整個文檔里面唯一的ID,復(fù)制模板后應(yīng)該顯示出來讓用戶看到。
復(fù)制代碼 代碼如下:
var newFields = document.getElementById('readroot').cloneNode(true);
newFields.id = '';
newFields.style.display = 'block';
我們遍歷這個拷貝的子元素:
復(fù)制代碼 代碼如下:
var newField = newFields.childNodes;
for (var i=0;i<newField.length;i++) {
如果子元素有name屬性,那么我們就在name值上加上計(jì)數(shù)器的值,以保證他的唯一性:
復(fù)制代碼 代碼如下:
var theName = newField[i].name
if (theName)
newField[i].name = theName + counter;
}
現(xiàn)在這個拷貝已經(jīng)準(zhǔn)備好插入了。我們把他插入到writeroot之前:
復(fù)制代碼 代碼如下:
var insertHere = document.getElementById('writeroot');
insertHere.parentNode.insertBefore(newFields,insertHere);
}
insertHere.parentNode.insertBefore(newFields,insertHere);
}
然后我們在頁面加載的時候就執(zhí)行一次,這樣用戶進(jìn)入的時候就能看到:
復(fù)制代碼 代碼如下:
window.onload = moreFields;
移除表單項(xiàng)
每一個模板的拷貝都有一個移除按鈕:
復(fù)制代碼 代碼如下:
<input type="button" value="Remove review"
onclick="this.parentNode.parentNode.removeChild(this.parentNode);" />
onclick="this.parentNode.parentNode.removeChild(this.parentNode);" />
點(diǎn)擊這個按鈕就會從移除他的父元素(DIV)。整個生成的表單就都會消失,并且不會再出現(xiàn)。
翻譯地址:http://www.quirksmode.org/dom/domform.html
轉(zhuǎn)載請保留以下信息
作者:北玉(tw:@rehawk)
相關(guān)文章
深入理解JavaScript系列(41):設(shè)計(jì)模式之模板方法詳解
這篇文章主要介紹了深入理解JavaScript系列(41):設(shè)計(jì)模式之模板方法詳解,模板方法(TemplateMethod)定義了一個操作中的算法的骨架,而將一些步驟延遲到子類中,模板方法使得子類可以不改變一個算法的結(jié)構(gòu)即可重定義該算法的某些特定步驟,需要的朋友可以參考下2015-03-03在JavaScript中操作時間之getYear()方法的使用教程
這篇文章主要介紹了在JavaScript中操作時間之getYear()方法的使用教程,是JS入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-06-06JavaScript多線程運(yùn)行庫Nexus.js詳解
這篇文章主要介紹了JavaScript多線程運(yùn)行庫Nexus.js的學(xué)習(xí)心得以及代碼分享,有需要的朋友一起參考學(xué)習(xí)下吧。2017-12-12JavaScript中的prototype.bind()方法介紹
在JavaScript中,我們經(jīng)常用到函數(shù)綁定,而當(dāng)你需要在另一個函數(shù)中保持this上下文時,使用Function.prototype.bind()會很方便2014-04-04微信公眾號開發(fā) 自定義菜單跳轉(zhuǎn)頁面并獲取用戶信息實(shí)例詳解
這篇文章主要介紹了微信公眾號開發(fā) 自定義菜單跳轉(zhuǎn)頁面并獲取用戶信息實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2016-12-12