JS学习13(表单脚本)

表单基础知识

在HTML中,表单是使用form元素来表示的,JS中对应的是HTMLFormElement类型。它同样继承自HTMLElement,还有一些自己的方法:

  • acceptCharset:服务器能处理的字符集
  • action:接受请求的URL
  • elements:表单中所有控件的集合(HTMLCollection)
  • enctype:请求的编码类型
  • length:表单中所有控件的数量
  • method:要发送的HTTP请求类型,通常是get或post
  • name:表单的名称
  • reset():将所有表单域重置为默认值
  • submit():提交表单
  • target:用于发送请求和接收响应的窗口名称

除了通过普通的方法获得form元素,还可以通过document.forms来获得页面中的所有表单。在这个集合中可以通过数值索引或name值来取得特定的表单。

var firstForm = document.forms[0]; 
var myForm = document.forms["myForm"];
alert(firstForm.name);

提交表单

用户单击提交按钮或图像按钮时就回提交表单。

<input type="submit" value="Submit Form">
<button type="submit">Submit Form</button>
<input type="image" src="graphic.gif">

表单中存在上述任何一种按钮时,在表单中元素具有焦点的情况下,按回车就可以提交该表单。
submit同样是一种事件类型,通过设置事件处理函数可以改变其行为,比如取消掉。

EventUtil.addHandler(firstForm, "submit", function(event){
    event = EventUtil.getEvent(event);
    EventUtil.preventDefault(event);
});

可以通过代码来触发submit操作,但是不会触发submit事件。

firstForm.submit();

重置表单

就是清空所有表单数据

<input type="reset" value="Reset Form">
<button type="reset">Reset Form</button>

这同样会触发一个reset事件

EventUtil.addHandler(form, "reset", function(event){
    event = EventUtil.getEvent(event);
    EventUtil.preventDefault(event);
});

使用form.reset()方法重置表单时还是会触发reset事件的。

表单字段

每个表单都有elements属性,该属性时表单中所有表单元素的集合,是有序的。通过位置和name都可以访问。

alert(firstForm.elements[0].name);
alert(firstForm.elements["password"].name);
alert(firstForm.elements.length);

如果有相同name的元素,那么就回返回一个NodeList
共有的表单字段属性

  • disabled
  • form:指向当前字段所属的表单的的指针,只读
  • name:当前字段的名称
  • readOnly:当前字段是否只读
  • tabIndex:当前字段的切换序号
  • type:“checkbox”、“radio”
  • value:当前字段将被提交给服务器的值,对于文件字段来说这个属性是只读的,包含文件在计算机中的路径

JS可以动态修改他们。

var field = firstForm.elements[0];
field.value = "Another value";
alert(field.form === firstForm);
field.focus();
field.disabled = true;
field.type = "checkbox";

避免多次提交表单就可以在submit事件中禁用提交按钮。
共有的表单字段方法
每个表单字段都有两个方法focus()和blur(),focus不能聚焦到没有被显示出来的属性。HTML5有autofocus属性。
共有的表单字段事件
blur
change
focus

文本框脚本

文本框有两种input和textarea

<input type="text" size="25" maxlength="50" value="initial value">
<textarea rows="25" cols="5">initial value</textarea>

在JS中可以使用value属性来获取他们的内容。
特别提醒,处理文本框的值的时候不要使用标准DOM修改特性或文本子节点的方法,使用value属性,这样才能保证value属性的值(最终被提交到服务器的)和显示的值是一致的。

选择文本

上述两种文本框都支持select方法,这个方法用于选择文本框中所有文本。
这个在焦点移到输入框时自动选择其中的文本。

var field = firstForm.elements[1];
EventUtil.addHandler(field, "focus", function(event){
    event = EventUtil.getEvent(event);
    var target = EventUtil.getTarget(event);
    target.select();
});

选择事件
选择文本框内的文本或使用select()时会触发select事件。
取得选择的文本
HTML5中使用这两个属性selectionStart、selectionEnd取得选择的文本。
IE8及以下不支持,使用替代方案,不管用户在页面上选择了什么都会创建一个document.selection对象

EventUtil.addHandler(field, "select", function(event){
    console.log(getSelectedText(field));
});
function getSelectedText(textbox) {
    if (typeof textbox.selectionStart == "number") {
        return textbox.value.substring(textbox.selectionStart, textbox.selectionEnd);
    } else if (document.selection) {
        return document.selection.createRange().text;
    }
}

选择部分文本
刚才的select方法只能选择全部文本,setSelectionRange()方法接收两个参数要选择的第一个字符的索引和最后一个字符的索引。然后把焦点移过来才能看到

textbox.setSelectionRange(0, 3);
text.focus();

IE8不支持,又得用范围那一套。

function selectText(textbox, startIndex, stopIndex){
    if (textbox.setSelectionRange){
        textbox.setSelectionRange(startIndex, stopIndex);
    } else if (textbox.createTextRange){
        var range = textbox.createTextRange();
        range.collapse(true);
        range.moveStart("character", startIndex);
        range.moveEnd("character", stopIndex - startIndex);
        range.select();
    }
    textbox.focus();
}
selectText(text,0,3);

过滤输入

屏蔽字符
通过阻止keypress的默认事件来阻止用户输入。
像这样就所有的字符都输入不了啦,不过粘贴,用输入法之类的输入并不通过keypress的就阻止不了了呦~,改成keydown事件用ctrl的粘贴也会被阻止:

EventUtil.addHandler(textbox, "keypress", function(event){
    event = EventUtil.getEvent(event);
    EventUtil.preventDefault(event);
});

针对特定的字符可以做特定的操作,比如只屏蔽数字键:

EventUtil.addHandler(text, "keypress", function(event){
    event = EventUtil.getEvent(event);
    var target = EventUtil.getTarget(event);
    var charCode = EventUtil.getCharCode(event);
    if (!/\d/.test(String.fromCharCode(charCode)) && charCode > 9){
        EventUtil.preventDefault(event);
    }
});

操作剪贴板
有6个属于剪贴板的事件:

  • beforecopy
  • copy
  • beforecut
  • cut
  • beforepaste
  • paste

beforecopy、beforecut、beforepaste,在IE中一直会在真正的复制粘贴前被触发,其他浏览器中貌似不大好用。就用另外3个就足够啦。
在这个事件中clipboardData使用来访问剪贴板中的数据,IE中是window.clipboardData,其他是event.clipboardData需要做下兼容。它有3个方法:getData()、setData()、clearData(),他们接收的参数是数据类型,浏览器之间也有差异需要兼容~:

getClipboardText: function(event){         
    var clipboardData = (event.clipboardData || window.clipboardData);         
    return clipboardData.getData("text");     
},
setClipboardText: function(event, value){
    if (event.clipboardData){
        return event.clipboardData.setData("text/plain", value);
    } else if (window.clipboardData){             
        return window.clipboardData.setData("text", value);         
    }     
}

剪贴板事件时调用一下,就可以控制复制粘贴的内容啦,知乎就是这么干的。

EventUtil.addHandler(text, "paste", function(event){
    event = EventUtil.getEvent(event);
    var text = EventUtil.getClipboardText(event);
    if (!/^\d*$/.test(text)){
        EventUtil.preventDefault(event);
    }
});

自动切换焦点

挺便利的小功能

(function(){
    function tabForward(event){         
        event = EventUtil.getEvent(event);         
        var target = EventUtil.getTarget(event);
        if (target.value.length == target.maxLength){             
            var form = target.form;
            for (var i=0, len=form.elements.length; i < len; i++) {                 
                if (form.elements[i] == target) {                     
                    if (form.elements[i+1]){                         
                        form.elements[i+1].focus();                     
                    }                     
                    return;                 
                }             
            }         
        }     
    }
    var textbox1 = document.getElementById("txtTel1");     
    var textbox2 = document.getElementById("txtTel2");     
    var textbox3 = document.getElementById("txtTel3");
    EventUtil.addHandler(textbox1, "keyup", tabForward);     
    EventUtil.addHandler(textbox2, "keyup", tabForward);     
    EventUtil.addHandler(textbox3, "keyup", tabForward);
})();

HTML5约束验证API

脱离JS即便JS不能加载,用HTML5的特性也能完成部分验证。
必填字段

<input type="text" name="username" required> 

其他输入类型
有的输入字段自带验证:

<input type="email" name ="email"> 
<input type="url" name="homepage"> 

数值范围
除了email,url还有"number","range","datetime","datetime-local","date","month","week","time"不过支持并不好。
数值类型一般都支持min,max,step

<input type="number" min="0" max="100" step="5" name="count"> 
input.stepUp();     //加 1 
input.stepUp(5);    //加 5 
input.stepDown();   
input.stepDown(10);

输入模式
这个属性是一个正则表达式

<input type="text" pattern="\d+" name="count"> 
var pattern = document.forms[0].elements["count"].pattern; 

检测有效性
checkValidity(),这个方法可以用在整个表单,也可以用在某个字段。有效返回true。
validity,这个属性是个对象,会更加详细的告诉你为什么字段有效或无效。这个对象中包含一系列属性,都是布尔值:

  • customError:如果设置了setCustomValidity()则为true,否则为false
  • patternMismatch:pattern
  • rangeOverflow:max
  • rangeUnderflow:min
  • stepMisMatch
  • tooLong
  • typeMismatch:mail、url
  • valid:所有的,和checkValidity()等同 �
  • valueMissing:required
if (input.validity && !input.validity.valid){     
    if (input.validity.valueMissing){
        alert("Please specify a value.")
    } else if (input.validity.typeMismatch){
        alert("Please enter an email address.");
    } else {
        alert("Value is invalid.");
    }
}

禁用验证

<form method="post" action="signup.php" novalidate>     
    <!--表-->  
</form> 
document.forms[0].noValidate = true; // 

或者按钮

<input type="submit" formnovalidate name="btnNoValidate"         value="Non-validating Submit"> 
document.forms[0].elements["btnNoValidate"].formNoValidate = true; 

选择框脚本

选择框通过select和option元素创建。select在JS中为HTMLSelectElement。有下列属性和方法:

  • add(newOption, relOption):向控件中插入新的option元素,位置在relOption之前。 �
  • multiple:就是HTML中的multiple,表示是否允许多项选择 �
  • options控件中所有option的HTMLCollection
  • remove(index)
  • selectedIndex:没有选中的项时为-1
  • size:选择框中可见行数

option在JS中为HTMLOptionElement,为了便于访问数据,有下列属性和方法:

  • index
  • label
  • selected
  • text
  • value

表单序列化

随着Ajax的出现,表单序列化已经成为了常见需求。在JS中,可以利用表单字段的type属性,连同name和value一起实现对表单的序列化。
在浏览器将表单发送给服务器之前:

  • 对表单字段的名称和值进行URL编码,使用&分割
  • 不发送禁用的表单字段
  • 只发送勾选的复选框和单选按钮
  • 不发送type为reset和button的按钮
  • 多选选择框中的每个选中的值单独一个条目
  • select元素的值,就是选中的option元素的value特性的值如果option元素没有value,则是option元素的文本值。
function serialize(form){
    var parts = [],
        field = null,
        i,
        len,
        j,
        optLen,
        option,
        optValue;
    for (i=0, len=form.elements.length; i < len; i++){
        field = form.elements[i];
        switch(field.type){
            case "select-one":
            case "select-multiple":
                if (field.name.length){
                    for (j=0, optLen = field.options.length; j < optLen; j++){
                        option = field.options[j];
                        if (option.selected){
                            optValue = "";
                            if (option.hasAttribute){
                                optValue = (option.hasAttribute("value") ?
                                    option.value : option.text);
                            } else {
                                //兼容IE
                                optValue = (option.attributes["value"].specified ?
                                    option.value : option.text);
                            }
                            parts.push(encodeURIComponent(field.name) + "=" +
                                encodeURIComponent(optValue));
                        }
                    }
                }
                break;
            case undefined:
            case "file":
            case "submit":
            case "reset":
            case "button":
                break;
            case "radio":
            case "checkbox":
                if (!field.checked){
                    break;
                }
            default:
                if (field.name.length){
                    parts.push(encodeURIComponent(field.name) + "=" +
                        encodeURIComponent(field.value));
                }
        }
    }
    return parts.join("&");
}

富文本编辑

坑待填

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容

  • 表单基础知识 在HTML中,表单是由 元素来表示的,而在JS中,表单对应的则是HTMLFormElement类型。...
    oWSQo阅读 903评论 0 1
  • 本人做php的,最近发现JS真的是博大精深啊,比PHP难.在HTML中,表单是由form元素来表示的,但是在jav...
    linfree阅读 2,133评论 3 17
  • 本章内容 理解表单 文本框验证与交互 使用其他表单控制 14.1 表单的基础知识 通过document.forms...
    闷油瓶小张阅读 360评论 0 0
  • 14.1 表单的基础知识 表单由 元素来表示,继承自HTMLElement类型,除具有HTML元素相同的默认属性外...
    Elevens_regret阅读 361评论 0 0
  • 阅读进度:P29-P50 阅读时长:一个潮汐钟 1:通过阅读中,知道了“天真练习”的概念,以及如何避免出现这种情况...
    探花小王阅读 201评论 0 0