JavaScript DOM 学习第七章 表单的扩展


Posted in Javascript onFebruary 19, 2010

想法
假设你有一个在线的CD评级工具。你希望用户查看他们喜欢的所有CD。但是你怎么知道用户平均会查看多少张呢?你在这个页面上需要添加多少字段呢?
在W3CDOM出现之前这确实是一个问题。假设你放置了7张CD。但是用户很可能只想查看一张,太多可能会吓着用户,而有些用户想查看自己的所有CD,这样就不得不反复提交很多次。这确实很烦人。
只有使用W3C DOM才能让用户决定生成多少个字段。这个效果和之前的大不相同。
例子
当你点击发送的时候,表单就会把得到的所有参数以数组的形式发送。这用来检查是否真正的发送到了服务器端。不幸的是,mac版的IE和Safari不能发送。
IE的问题
windows下的IE有两个严重的问题:
第一个问题是所有生成的单选框都同属于一个数组,即使他们的names不同。这样用户就只能在所有的单选框里面选择一个。也就是说你不能在生成的表单里面实用单选框。
有读者说通过innerHTML生成的单选框没有问题。如果你非要实用单选框的话,你可以试试。
第二个问题是生成的表单通过传统的document.forms无法访问:IE没有在数组里面包含他们。这个可以通过给他们设置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>

真正的表单项目在ID为readroot的DIV并且display值为none。这个DIV是一个模板,用户不能修改。当用户需要更多的表单的时候我们就复制这个模板然后添加在表单之后。我们在一开始的就加载,这样当用户打开页面的时候就能看到。
这个DIV在表单之外,所以当用户提交表单的时候,这个模板的内容不会被提交。
ID为writeroot的SPAN是一个标记。新的生成的表单就插入在他的前面。
添加表单项目
下面的代码可以再需要的时候用来添加表单项目:
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;

首先我们需要一个计数器counter,因为所有的表单项都应该有唯一的一个名字。我们把计数器的值添加在生成name后面。初始化计数器:
var counter = 0;

然后是实际的函数。我们给计数器加1:
function moreFields() { 
counter++;

然后复制我们的模板,移除ID,并且把display设置为block。readroot应该是整个文档里面唯一的ID,复制模板后应该显示出来让用户看到。
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值上加上计数器的值,以保证他的唯一性:
var theName = newField[i].name 
if (theName) 
newField[i].name = theName + counter; 
}

现在这个拷贝已经准备好插入了。我们把他插入到writeroot之前:
var insertHere = document.getElementById('writeroot'); 
insertHere.parentNode.insertBefore(newFields,insertHere); 
}

然后我们在页面加载的时候就执行一次,这样用户进入的时候就能看到:
window.onload = moreFields;

移除表单项
每一个模板的拷贝都有一个移除按钮:
<input type="button" value="Remove review" 
onclick="this.parentNode.parentNode.removeChild(this.parentNode);" />

点击这个按钮就会从移除他的父元素(DIV)。整个生成的表单就都会消失,并且不会再出现。
翻译地址:http://www.quirksmode.org/dom/domform.html
转载请保留以下信息
作者:北玉(tw:@rehawk)
Javascript 相关文章推荐
JS的递增/递减运算符和带操作的赋值运算符的等价式
Dec 08 Javascript
GWT中复制到剪贴板 js+flash实现复制 兼容性比较好
Mar 07 Javascript
Javascript异步编程模型Promise模式详细介绍
May 08 Javascript
容易造成JavaScript内存泄露几个方面
Sep 04 Javascript
jquery.ajax之beforeSend方法使用介绍
Dec 08 Javascript
基于Javascript实现文件实时加载进度的方法
Oct 12 Javascript
在 Node.js 中使用 async 函数的方法
Nov 17 Javascript
PHP自动加载autoload和命名空间的应用小结
Dec 01 Javascript
node.js读取Excel数据(下载图片)的方法示例
Aug 02 Javascript
angularJS实现不同视图同步刷新详解
Oct 09 Javascript
vue webpack build资源相对路径的问题及解决方法
Jun 04 Javascript
Nest.js环境变量配置与序列化详解
Feb 21 Javascript
JavaScript DOM学习第六章 表单实例
Feb 19 #Javascript
JavaScript DOM 学习第五章 表单简介
Feb 19 #Javascript
JavaScript DOM学习第四章 getElementByTagNames
Feb 19 #Javascript
JavaScript DOM 学习第三章 内容表格
Feb 19 #Javascript
JavaScript DOM 学习第二章 编辑文本
Feb 19 #Javascript
JavaScript DOM学习第一章 W3C DOM简介
Feb 19 #Javascript
JavaScript 题型问答有答案参考
Feb 17 #Javascript
You might like
农民和部队如何穿矿
2020/03/04 星际争霸
XAMPP安装与使用方法详细解析
2013/11/27 PHP
php获取字段名示例分享
2014/03/03 PHP
PHP使用in_array函数检查数组中是否存在某个值
2015/03/25 PHP
WordPress中对访客评论功能的一些优化方法
2015/11/24 PHP
PHP接收json 并将接收数据插入数据库的实现代码
2015/12/01 PHP
javascript静态的url如何传递
2007/05/03 Javascript
Javascript Select操作大集合
2009/05/26 Javascript
JavaScript this 深入理解
2009/07/30 Javascript
{}与function(){}选用空对象{}来存放keyValue
2012/05/23 Javascript
js判断一个字符串是否包含一个子串的方法
2015/01/26 Javascript
javascript中eval函数用法分析
2015/04/25 Javascript
每天一篇javascript学习小结(Date对象)
2015/11/13 Javascript
jQuery Mobile 和 Kendo UI 的比较
2016/05/05 Javascript
详解vue-router 2.0 常用基础知识点之router.push()
2017/05/10 Javascript
理解javascript async的用法
2017/08/22 Javascript
vue中的event bus非父子组件通信解析
2017/10/27 Javascript
使用Nuxt.js改造已有项目的方法
2018/08/07 Javascript
vue实现分环境打包步骤(给不同的环境配置相对应的打包命令)
2019/06/04 Javascript
JS删除String里某个字符的方法
2021/01/06 Javascript
移动端吸顶fixbar的解决方案详解
2019/07/17 Javascript
基于js实现判断浏览器类型代码实例
2020/07/17 Javascript
[34:27]DOTA2上海特级锦标赛B组败者赛 VG VS Spirit第一局
2016/02/26 DOTA
python练习程序批量修改文件名
2014/01/16 Python
分享几道你可能遇到的python面试题
2017/07/24 Python
在pycharm中使用pipenv创建虚拟环境和安装django的详细教程
2020/11/30 Python
TUMI马来西亚官方网站:国际领先的高品质商旅箱包品牌
2018/04/26 全球购物
如何写出好的Java代码
2014/04/25 面试题
客户接待方案
2014/02/26 职场文书
物业管理工作方案
2014/05/10 职场文书
对党的十八届四中全会的期盼
2014/10/17 职场文书
2014年爱国卫生工作总结
2014/11/22 职场文书
2015年“七七卢沟桥事变”纪念活动总结
2015/03/24 职场文书
小学生暑假安全保证书
2015/07/13 职场文书
导游词之苏州寒山寺
2019/12/05 职场文书
Vue h函数的使用详解
2022/02/18 Vue.js