JavaScript(第二十六天)【表单处理】
为了分担服务器处理表单的压力,JavaScript提供了一些解决方案,从而大大打破了处处依赖服务器的局面。
一.表单介绍
在HTML中,表单是由<form>元素来表示的,而在JavaScript中,表单对应的则是HTMLFormElement类型。HTMLFormElement继承了HTMLElement,因此它拥有HTML元素具有的默认属性,并且还独有自己的属性和方法:
HTMLFormElement属性和方法
|
属性或方法 |
说明 |
|
acceptCharset |
服务器能够处理的字符集 |
|
action |
接受请求的URL |
|
elements |
表单中所有控件的集合 |
|
enctype |
请求的编码类型 |
|
length |
表单中控件的数量 |
|
name |
表单的名称 |
|
target |
用于发送请求和接受响应的窗口名称 |
|
reset() |
将所有表单重置 |
|
submit() |
提交表单 |
获取表单<form>对象的方法有很多种,如下:
document.getElementById('myForm'); //使用ID获取<form>元素
document.getElementsByTagName('form')[0]; //使用获取第一个元素方式获取
document.forms[0]; //使用forms的数字下标获取元素
document.forms['yourForm']; //使用forms的名称下标获取元素
document.yourForm; //使用name名称直接获取元素
PS:最后一种方法使用name名称直接获取元素,已经不推荐使用,这是向下兼容的早期用法。问题颇多,比如有两个相同名称的,变成数组;而且这种方式以后有可能会不兼容。
提交表单
通过事件对象,可以阻止submit的默认行为,submit事件的默认行为就是携带数据跳转到指定页面。
addEvent(fm, 'submit', function (evt) {
preDef(evt);
});
我们可以可以使用submit()方法来自定义触发submit事件,也就是说,并不一定非要点击submit按钮才能提交。
if (e.ctrlKey && e.keyCode == 13) fm.submit(); //判断按住了ctrl和enter键触发
PS:在表单中尽量避免使用name="submit"或id="submit"等命名,这会和submit()方法发生冲突导致无法提交。
提交数据最大的问题就是重复提交表单。因为各种原因,当一条数据提交到服务器的时候会出现延迟等长时间没反映,导致用户不停的点击提交,从而使得重复提交了很多相同的请求,或造成错误、或写入数据库多条相同信息。
addEvent(fm, 'submit', function (evt) { //模拟延迟
preDef(evt);
setTimeout(function () {
fm.submit();
}, 3000);
});
有两种方法可以解决这种问题:第一种就是提交之后,立刻禁用点击按钮;第二种就是提交之后取消后续的表单提交操作。
document.getElementById('sub').disabled = true; //将按钮禁用
var flag = false //设置一个监听变量
if (flag == true) return //如果存在返回退出事件
flag = true; //否则确定是第一次,设置为true
PS:在某些浏览器,F5只能起到缓存刷新的效果,有可能获取不到真正的源头更新的数据。那么使用ctrl+F5就可以把源头给刷出来。
重置表单
用户点击重置按钮时,表单会被初始化。虽然这个按钮还得以保留,但目前的Web已经很少去使用了。因为用户已经填写好各种数据,不小心点了重置就会全部清空,用户体验极差。
有两种方法调用reset事件,第一个就是直接type="reset"即可;第二个就是使用fm.reset()方法调用即可。
<input type="reset" value="重置" /> //不需要JS代码即可实现
addEvent(document,'click', function () {
fm.reset(); //使用JS方法实现重置
});
addEvent(fm,'reset', function () { //获取重置按钮
//
});
表单字段
如果想访问表单元素,可以使用之前章节讲到的DOM方法访问。但使用原生的DOM访问虽然比较通用,但不是很便利。表单处理中,我们建议使用HTML DOM,它有自己的elements属性,该属性是表单中所有元素的集合。
fm.elements[0]; //获取第一个表单字段元素
fm.elements['user']; //获取name是user的表单字段元素
fm.elements.length; //获取所有表单字段的数量
如果多个表单字段都使用同一个name,那么就会返回该name的NodeList表单列表。
fm.elements['sex']; //获取相同name表单字段列表
PS:我们是通过fm.elements[0]来获取第一个表单字段的,但也可以使用fm[0]直接访问第一个字段。因为fm[0]访问方式是为了向下兼容的,所以,我们建议大家使用elements属性来获取。
共有的表单字段属性
除了<fieldset>元素之外,所有表单字段都拥有相同的一组属性。由于<input>类型可以表示多种表单字段,因此有些属性只适用于某些字段。以下罗列出共有的属性:
|
属性或方法 |
说明 |
|
disabled |
布尔值,表示当前字段是否被禁用 |
|
form |
指向当前字段所属表单的指针,只读 |
|
name |
当前字段的名称 |
|
readOnly |
布尔值,表示当前字段是否只读 |
|
tabIndex |
表示当前字段的切换 |
|
type |
当前字段的类型 |
|
value |
当前字段的值 |
这些属性其实就是HTML表单里的属性,在XHTML课程中已经详细讲解过,这里不一个个赘述,重点看几个最常用的。
fm.elements[0].value; //获取和设置value
fm.elements[0].form == fm; //查看当前字段所属表单
fm.elements[0].disabled = true; //禁用当前字段
fm.elements[0].type = 'checkbox'; //修改字段类型,极不推荐
除了<fieldset>字段之外,所有表单字段都有type属性。对于<input>元素,这个值等于HTML属性的type值。对于非<input>元素,这个type的属性值如下:
|
元素说明 |
HTML标签 |
type属性的值 |
|
单选列表 |
<select>...</select> |
select-one |
|
多选列表 |
<select multiple>...</select> |
select-multiple |
|
自定义按钮 |
<button>...</button> |
button |
|
自定义非提交按钮 |
<button type="button">...</button> |
button |
|
自定义重置按钮 |
<button type="reset">...</button> |
reset |
|
自定义提交按钮 |
<button type="submit">...</button> |
submit |
PS:<input>和<button>元素的type属性是可以动态修改的,而<select>元素的type属性则是只读的。(在不必要的情况下,建议不修改type)。
共有的表单字段方法
每个表单字段都有两个方法:foucs()和blur()。
|
方法 |
说明 |
|
focus() |
将焦点定位到表单字段里 |
|
blur() |
从元素中将焦点移走 |
fm.elements[0].focus(); //将焦点移入
fm.elements[0].blur(); //将焦点移出
共有的表单字段事件
表单共有的字段事件有以下三种:
|
事件名 |
说明 |
|
blur |
当字段失去焦点时触发 |
|
change |
对于<input>和<textarea>元素,在改变value并失去焦点时触发;对于<select>元素,在改变选项时触发 |
|
focus |
当前字段获取焦点时触发 |
addEvent(textField, 'focus', function () { //缓存blur和change再测试一下
alert('Dkf');
});
PS:关于blur和change事件的关系,并没有严格的规定。在某些浏览器中,blur事件会先于change事件发生;而在其他浏览器中,则恰好相反。
二.文本框脚本
在HTML中,有两种方式来表现文本框:一种是单行文本框<input type="text">,一种是多行文本框<textarea>。虽然<input>在字面上有value值,而<textarea>却没有,但通过都可以通过value获取他们的值。
var textField = fm.elements[0];
var areaField = fm.elements[1];
alert(textField.value + ',' + areaField.value); //得到value值
PS:使用表单的value是最推荐使用的,它是HTML DOM中的属性,不建议使用标准DOM的方法。也就是说不要使用getAttribute()获取value值。原因很简单,对value属性的修改,不一定会反映在DOM中。
除了value值,还有一个属性对应的是defaultValue,可以得到原本的value值,不会因为值的改变而变化。
alert(textField.defaultValue); //得到最初的value值
选择文本
使用select()方法,可以将文本框里的文本选中,并且将焦点设置到文本框中。
textField.select(); //选中文本框中的文本
选择部分文本
在使用文本框内容的时候,我们有时要直接选定部分文本,这个行为还没有标准。Firefox的解决方案是:setSelectionRange()方法。这个方法接受两个参数:索引和长度。
textField.setSelectionRange(0,1); //选择第一个字符
textField.focus(); //焦点移入
textField.setSelectionRange(0, textField.value.length); //选择全部
textField.focus(); //焦点移入
除了IE,其他浏览器都支持这种写法(IE9+支持),那么IE想要选择部分文本,可以使用IE的范围操作。
var range = textField.createTextRange(); //创建一个文本范围对象
range.collapse(true); //将指针移到起点
range.moveStart('character', 0); //移动起点,character表示逐字移动
range.moveEnd('character', 1); //移动终点,同上
range.select(); //焦点选定
PS:关于IE范围的详细讲解,我们将在今后的课程中继续讨论。并且W3C也有自己的范围。
//选择部分文本实现跨浏览器兼容
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', stop - start); //IE用终点减去起点得到字符数
range.select();
}
}
使用select事件,可以选中文本框文本后触发。
addEvent(textField, 'select', function () {
alert(this.value); //IE事件需要传递this才可以这么写
});
取得选择的文本
如果我们想要取得选择的那个文本,就必须使用一些手段。目前位置,没有任何规范解决这个问题。Firefox为文本框提供了两个属性:selectionStart和selectionEnd。
addEvent(textField, 'select', function () {
alert(this.value.substring(this.selectionStart, this.selectionEnd));
});
除了IE,其他浏览器均支持这两个属性(IE9+已支持)。IE不支持,而提供了另一个方案:selection对象,属于document。这个对象保存着用户在整个文档范围内选择的文本信息。导致我们需要做浏览器兼容。
function getSelectText(text) {
if (typeof text.selectionStart == 'number') { //非IE
return text.value.substring(text.selectionStart, text.selectionEnd);
} else if (document.selection) { //IE
return document.selection.createRange().text; //获取IE选择的文本
}
}
PS:有一个最大的问题,就是IE在触发select事件的时候,在选择一个字符后立即触发,而其他浏览器是选择想要的字符释放鼠标键后才触发。所以,如果使用alert()的话,导致跨浏览器的不兼容。我们没有办法让浏览器行为保持统一,但可以通过不去使用alert()来解决。
addEvent(textField, 'select', function () {
//alert(getSelectText(this)); //导致用户行为结果不一致
document.getElementById('box').innerHTML = getSelectText(this);
});
过滤输入
为了使文本框输入指定的字符,我们必须对输入进的字符进行验证。有一种做法是判断字符是否合法,这是提交后操作的。那么我们还可以在提交前限制某些字符,还过滤输入。
addEvent(areaField, 'keypress', function (evt) {
var e = evt || window.event;
var charCode = getCharCode(evt); //得到字符编码
if (!/\d/.test(String.fromCharCode(charCode)) && charCode > 8) { //条件阻止默认
preDef(evt);
}
});
PS:前半段条件判断只有数字才可以输入,导致常规按键,比如光标键、退格键、删除键等无法使用。部分浏览器比如Firfox,需要解放这些键,而非字符触发的编码均为0;在Safari3之前的浏览器,也会被阻止,而它对应的字符编码全部为8,所以最后就加上charCode > 8的判断即可。
PS:当然,这种过滤还是比较脆落的,我们还希望能够阻止裁剪、复制、粘贴和中文字符输入操作才能真正屏蔽掉这些。
如果要阻止裁剪、复制和粘贴,那么我们可以在剪贴板相关的事件上进行处理,JavaScript提供了六组剪贴板相关的事件:
|
事件名 |
说明 |
|
copy |
在发生复制操作时触发 |
|
cut |
在发生裁剪操作时触发 |
|
paste |
在发生粘贴操作时触发 |
|
beforecopy |
在发生复制操作前触发 |
|
beforecut |
在发生裁剪操作前触发 |
|
beforepaste |
在发生粘贴操作前触发 |
由于剪贴板没有标准,导致不同的浏览器有不同的解释。Safari、Chrome和Firefox中,凡是before前缀的事件,都需要在特定条件下触发。而IE则会在操作时之前触发带before前缀的事件。
如果我们想要禁用裁剪、复制、粘贴,那么只要阻止默认行为即可。
addEvent(areaField, 'cut', function (evt) { //阻止裁剪
preDef(evt);
});
addEvent(areaField, 'copy', function (evt) { //阻止复制
preDef(evt);
});
addEvent(areaField, 'paste', function (evt) { //阻止粘贴
preDef(evt);
});
当我们裁剪和复制的时候,我们可以访问剪贴板里的内容,但问题是FireFox,Opera浏览器不支持访问剪贴板。并且,不同的浏览器也有自己不同的理解。所以,这里我们就不在赘述。
最后一个问题影响到可能会影响输入的因素就是:输入法。我们知道,中文输入法,它的原理是在输入法面板上先存储文本,按下回车就写入英文文本,按下空格就写入中文文本。
有一种解决方案是通过CSS来禁止调出输入法:
style="ime-mode:disabled" //CSS直接编写
areaField.style.imeMode = 'disabled'; //或在JS里设置也可以
PS:但我们也发先,Chrome浏览器却无法禁止输入法调出。所以,为了解决谷歌浏览器的问题,最好还要使用正则验证已输入的文本。
addEvent(areaField, 'keyup', function (evt) { //keyup弹起的时候
this.value = this.value.replace(/[^\d]/g, ''); //把非数字都替换成空
});
自动切换焦点
为了增加表单字段的易用性,很多字段在满足一定条件时(比如长度),就会自动切换到下一个字段上继续填写。
<input type="text" name="user1" maxlength="1" /> //只能写1个
<input type="text" name="user2" maxlength="2" /> //只能写2个
<input type="text" name="user3" maxlength="3" /> //只能写3个
function tabForward (evt) {
var e = evt || window.event;
var target = getTarget(evt);
//判断当前长度是否和指定长度一致
if (target.value.length == target.maxLength) {
//遍历所有字段
for (var i =0; i < fm.elements.length; i ++) {
//找到当前字段
if (fm.elements[i] == target) {
//就把焦点移入下一个
fm.elements[i + 1].focus();
//中途返回
return;
}
}
}
}
三.选择框脚本
选择框是通过<select>和<option>元素创建的,除了通用的一些属性和方法外,HTMLSelectElement类型还提供了如下的属性和方法:
HTMLSelectElement对象
|
属性/方法 |
说明 |
|
add(new,rel) |
插入新元素,并指定位置 |
|
multiple |
布尔值,是否允许多项选择 |
|
options |
<option>元素的HTMLColletion集合 |
|
remove(index) |
移除给定位置的选项 |
|
selectedIndex |
基于0的选中项的索引,如果没有选中项,则值为-1 |
|
size |
选择框中可见的行数 |
在DOM中,每个<option>元素都有一个HTMLOptionElement对象,以便访问数据,这个对象有如下一些属性:
HTMLOptionElement对象
|
属性 |
说明 |
|
index |
当前选项在options集合中的索引 |
|
label |
当前选项的标签 |
|
selected |
布尔值,表示当前选项是否被选中 |
|
text |
选项的文本 |
|
value |
选项的值 |
var city = fm.elements['city']; //HTMLSelectElement
alert(city.options); //HTMLOptionsCollection
alert(city.options[0]); //HTMLOptionElement
alert(city.type); //select-one
PS:选择框里的type属性有可能是:select-one,也有可能是:select-multiple,这取决于HTML代码中有没有multiple属性。
alert(city.options[0].firstChild.nodeValue); //上海t,获取text值,不推荐的做法
alert(city.options[0].getAttribute('value')); //上海v,获取value值,不推荐的做法
alert(city.options[0].text); //上海t,获取text值,推荐
alert(city.options[0].value); //上海v,获取value值,推荐
PS:操作select时,最好使用HTML DOM,因为所有浏览器兼容的很好。而如果使用标准DOM,会因为不同的浏览器导致不同的结果。
PS:当选项没有value值的时候,IE会返回空字符串,其他浏览器会返回text值。
选择选项
对于只能选择一项的选择框,使用selectedIndex属性最为简单。
addEvent(city, 'change', function () {
alert(this.selectedIndex); //得到当前选项的索引,从0开始
alert(this.options[this.selectedIndex].text); //得到当前选项的text值
alert(this.options[this.selectedIndex].value); //得到当前选项的value值
});
PS:如果是多项选择,他始终返回的是第一个项。
city.selectedIndex = 1; //设置selectedIndex可以定位某个索引
通过option的属性(布尔值),也可以设置某个索引,设置为true即可。
city.options[0].selected = true; //设置第一个索引
而selected和selectedIndex在用途上最大的区别是,selected是返回的布尔值,所以一般用于判断上;而selectedIndex是数值,一般用于设置和获取。
addEvent(city, 'change', function () {
if (this.options[2].selected == true) { //判断第三个选项是否被选定
alert('选择正确!');
}
});
添加选项
如需动态的添加选项我们有两种方案:DOM和Option构造函数。
var option = document.createElement('option');
option.appendChild(document.createTextNode('北京t'));
option.setAttribute('value', '北京v')
city.appendChild(option);
使用Option构造函数创建:
var option = new Option('北京t', '北京v');
city.appendChild(option); //IE出现bug
使用add()方法来添加选项:
var option = new Option('北京t', '北京v');
city.add(option, 0); //0,表示添加到第一位
PS:在DOM规定,add()中两个参数是必须的,如果不确定索引,那么第二个参数设置null即可,即默认移入最后一个选项。但这是IE中规定第二个参数是可选的,所以设置null表示放入不存在的位置,导致失踪,为了兼容性,我们传递undefined即可兼容。
city.add(option, null); //IE不显示了
city.add(option, undefined); //兼容了
移除选项
有三种方式可以移除某一个选项:DOM移除、remove()方法移除和null移除。
city.removeChild(city.options[0]); //DOM移除
city.remove(0); //remove()移除,推荐
city.options[0] = null; //null移除
PS:当第一项移除后,下面的项,往上顶,所以不停的移除第一项,即可全部移除。
移动选项
如果有两个选择框,把第一个选择框里的第一项移到第二个选择框里,并且第一个选择框里的第一项被移除。
var city = fm.elements['city']; //第一个选择框
var info = fm.elements['info']; //第二个选择框
info.appendChild(city.options[0]); //移动,被自我删除
排列选项
选择框提供了一个index属性,可以得到当前选项的索引值,和selectedIndex的区别是,一个是选择框对象的调用,一个是选项对象的调用。
var option1 = city.options[1];
city.insertBefore(option1, city.options[option1.index - 1]); //往下移动移位
单选按钮
通过checked属性来获取单选按钮的值。
for (var i = 0; i < fm.sex.length; i ++) { //循环单选按钮
if (fm.sex[i].checked == true) { //遍历每一个找出选中的那个
alert(fm.sex[i].value); //得到值
}
}
PS:除了checked属性之外,单选按钮还有一个defaultChecked按钮,它获取的是原本的checked按钮对象,而不会因为checked的改变而改变。
if (fm.sex[i].defaultChecked == true) {
alert(fm.sex[i].value);
}
复选按钮
通过checked属性来获取复选按钮的值。复选按钮也具有defaultChecked属性。
var love = '';
for (var i = 0; i < fm.love.length; i ++) {
if (fm.love[i].checked == true) {
love += fm.love[i].value;
}
}
alert(love);
JavaScript(第二十六天)【表单处理】的更多相关文章
- 第一百六十节,封装库--JavaScript,ajax注册表单到数据库
封装库--JavaScript,ajax注册表单到数据库 效果图 前台js var biaodan = $().xu_lie_biao_dan($('form').sh_jd()); //序列化获取表 ...
- javaSE第二十六天
第二十六天 414 1:网络编程(理解) 414 (1)网络编程:用Java语言实现计算机间数据的信息传递和资源共享 414 (2)网络编程模型 414 (3)网络编程的三要素 ...
- javascript中的常用表单事件用法
下面介绍几种javascript中常用的表单事件: 一,onsubmit:表单中的确认按钮被点击时发生的事件,如下案例. 案例解析:弹出表单中提交的内容 <form name="tes ...
- Javascript中的Form表单知识点总结
Javascript中的Form表单知识点总结 在HTML中,表单是由form元素来表示的,但是在javascript中,表单则由HTMLFormElement类型,此元素继承了HTMLElement ...
- ajax+FormData+javascript实现无刷新表单信息提交
ajax+FormData+javascript实现无刷新表单信息提交 原理: dom收集表单信息,利用FormData快速收集表单信息 ,实例化表单数据对象 同时收集fm的表单域信息. var f ...
- ajax+FormData+javascript 实现无刷新表单注册
<!doctype html> <html> <head> <meta charset="utf-8"> <title> ...
- IT第二十六天 - Swing、上周总结
IT第二十六天 上午 Swing 1.对于方法的参数如果是int数值类型,应该直接调用该类中的常量属性,而不应该直接填入数字 2.Toolkit类中定义的方法是可以直接访问本地计算机(操作系统)信息的 ...
- JavaScript 创建一个 form 表单并提交
<!DOCTYPE HTML> <html lang="en-US"> <head> <meta charset="UTF-8& ...
- 每天一个JavaScript实例-防止反复表单提交
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content ...
随机推荐
- 芝麻HTTP:设置Selenium+Chrome代理
微博登录限制了错误次数···加上Cookie大批账号被封需要从Cookie池中 剔除被封的账号··· 需要使用代理··· 无赖百度了大半天都是特么的啥玩意儿???结果换成了 Google手到擒来 分分 ...
- hdu5937 Equation
题解其实网上有 突然有点感想 为什么可以用搜索或状压,因为方案数很有限,它要求每种方案不同就意味着搜索的次数也一定,所以现在就应该坚定往这方面想,找部分方案的贪心.这和上一题一样,都是先暴力,后面处理 ...
- Jupyter notebook Tensorflow GPU Memory 释放
Jupyter notebook 每次运行完tensorflow的程序,占着显存不释放.而又因为tensorflow是默认申请可使用的全部显存,就会使得后续程序难以运行.暂时还没有找到在jupyter ...
- 简单使用Mysql-Cluster-7.5搭建数据库集群
阅读目录 前言 mysql cluster中的几个概念解释 架构图及说明 下载mysql cluster 安装mysql cluster之前 安装配置管理节点 安装配置数据和mysql节点 测试 启动 ...
- [QNAP crontab 定時執行程式
注意要自動執行的 sh 檔不要放在 /root 裡, 不然韌體更新後檔案會不見, 要放在個人帳號的資料夾,例如 /share/homes/帳號/ QNAP 的 crontab 放在 /etc/conf ...
- 认识大前端html+css+js
认识大前端:前端就是将效果图生成网页,利用html+css+js等技术. 如果把前端比作一台汽车,那么html就是车的骨架,css就是完整的车的模型,而js就充当着车的发动机... 建议: 刚刚开 ...
- 【BZOJ3998】弦论(后缀自动机)
[BZOJ3998]弦论(后缀自动机) 题面 BZOJ 题解 这题应该很简单 构建出\(SAM\)后 求出每个点往后还能构建出几个串 按照拓扑序\(dp\)一些就好了 然后就是第\(k\)大,随便搞一 ...
- Vue-生命周期图示 注解
根据腾讯课堂视频讲解,将官网生命周期图示进行注解,以加深印象和理解 贴一个源码示例: 注意位置和写法
- canvas练手项目(一)——选取图片
今天无事可做,在春意盎然的下午突发奇想想做一个关于图片处理的在线网页应用.不要问我为什么做这个,因为我想做!关于这个项目,我想基于canvas来实现,canvas是个好东西,我一直很喜欢,就是我没有做 ...
- mysql大小写敏感问题
问题: 在创建mysql表的时候发现不论表明是大写或小写,建完之后统一被变成了小写. 原因: MySQL在windows下是不区分大小写的,将script文件导入MySQL后表名也会自动转化为小写. ...