HTML 事件可以是浏览器行为,也可以是用户行为。
通常,在事件触发时 JavaScript 可以执行一些代码。
当一个DOM事件被触发时,它不仅仅只是单纯地在本身对象上触发一次,而是会经历三个不同的阶段:
冒泡阶段:再从目标事件位置往文档的根节点方向回溯,从内向外冒泡事件对象。
冒泡型事件流
捕获型事件流
.W3C标准模型
"DOM2事件"规范要求的事件流包括三个阶段:事件捕获阶段、处于目标阶段、事件冒泡阶段
W3C标准则采用捕获+冒泡
W3C事件模型中发生的任何事件,先从其祖先元素window开始一路向下捕获, 直到达到目标元素,其后再次从目标元素开始冒泡。
而作为开发者, 可以决定事件处理器是注册在捕获或者是冒泡阶段。如果addEventListener的最后一个参数是true, 那么处理函数将在捕获阶段被触发; 否则(false), 会在冒泡阶段被触发!默认false!
总结:
当使用on或者addEventListener绑定事件的时候,如果绑定同一个事件同一个方法,只会执行一次,但是attachEvent会多次绑定。在jquery中,无论是on添加事件,还是直接加事件函数,同一个事件都可以多次绑定
type="button" value="showClick" onclick="showClick()" />
如果handle是同一个方法,只执行一次。
//事件绑定
document.getElementById("box").onclick = handle;/
document.getElementById("box")["onmousemover"] = handle;
//事件解绑
document.getElementById("box")["onmousemover"] = null;
//阻止默认事件(默认事件行为:href=""链接,submit表单提交等)
document.getElementById("box").onclick = function() {
…… //你的代码
return false; //通过返回false值阻止默认事件行为
};
注释:
return false 的含义不是阻止事件继续向顶层元素传播,而是阻止浏览器对事件的默认处理。
return false 只在当前函数有效,不会影响其他外部函数的执行。
总结:
retrun true; 返回正确的处理结果。
return false;返回错误的处理结果,终止处理;阻止提交表单;阻止执行默认的行为。
return;把控制权返回给页面。
事件类型没有on,第三个参数false,表示在事件第三阶段(冒泡)触发,true表示在事件第一阶段(捕获)触发。
如果handle是同一个方法,只执行一次。
var element=document.getElementById("box");
var handler=function(){
event.preventDefault( ); //阻止默认事件
}
//绑定事件
element.addEventListener('click', handler, false);
//解绑事件
element.removeEventListener('click', handle, false);
如果handle是同一个方法,绑定几次执行几次,这点和addEventListener不同。事件类型要加on,例如onclick而不是click
特别注意:
在 IE 中使用 attachEvent() 与DOM0和DOM2addEventListener有一主要区别:事件处理程序的作用域。在这些方法中,事件处理程序会在其所属元素的作用域内运行;在使用 attachEvent() 方法的情况下,事件处理程序会在全局作用域中运行,因此 this 等于 window。
var element=document.getElementById("box");
var handler=function(){
event.returnValue = false; //阻止默认事件
}
/绑定事件
element.attachEvent('onclick', handler);
//解绑事件,参数和绑定一样
element.detachEvent("onclick", handler);
// 事件绑定
function addEvent(element, eType, handler, bol) {
var boll = bol | false;
if(element.addEventListener){ //如果支持addEventListener
element.addEventListener(eType, handler, boll );
}else if(element.attachEvent){ //如果支持attachEvent
element.attachEvent("on"+eType, handler);
}else{ //否则使用兼容的onclick绑定
element["on"+eType] = handle;
}
}
// 事件解绑
function removeEvent(element, eType, handler, bol) {
var boll = bol | false;
if(element.addEventListener){
element.removeEventListener(eType, handler, boll );
}else if(element.attachEvent){
element.detachEvent("on"+eType, handler);
}else{
element["on"+eType] = null;
}
}
我们可以将两个函数写成函数的方法模式
//用事件冒泡方式,如果想兼容事件捕获只需要添加个bool参数
var EventUtil = {
addEvent: function(element,type,handler) {
if (element.addEventListener) {
element.addEventListener(type,handler,false);
}
else if (element.attachEvent) {
element.attachEvent('on'+type,handler);
}
else {
element['on'+type] = handler;
}
},
removeEvent: function(element,type,handler) {
if (element.removeEventListener)
{
element.removeEventListener(type,handler,false);
}
else if(element.detachEvent) {
element.detachEvent('on' +type,handler);
}
else {
element['on'+type] = null;
}
}
}
//使用形式
var box= document.getElementById("box");
var handler = function(){
console.log(111);
};
EventUtil.addEvent(box, "click", handler);
EventUtil.removeEvent(box, "click", handler);
"box">
"spa">子子元素
document.getElementById('spa').addEventListener('click',function(e){
console.log('spa')
var e = event||window.event;
e.stopPropagation()
//在子级及以下的一个元素上进行阻止冒泡,在这块区域不会触发父级及以上的相同事件
})
document.getElementById('box').addEventListener('click',function(){
console.log('box')
})
阻止事件冒泡的方法:
function stopPropagation(event){
event=window.event||event;
if(event.stopPropagation){
event.stopPropagation();
}else{
event.cancelBubble=true;
}
}
一般有两种情况会用到事件委托
使用事件委托的原理:
<ul id="J_List">
<li><div class="id">1div>li>
<li><div class="id">2div>li>
<li><div class="user-delete">3div>li>
ul>
var lis = document.querySelector('#J_List');
lis.addEventListener('click', function(e){
stopPropagation(e)
var target = e.target || e.srcElement;
//在这里处理子元素的事件
if (target && target.className.toLowerCase()==='user-delete') {
target.parentNode.parentNode.removeChild(target.parentNode);
}
})
在jquery中经常用到的,即事件委托
$('body').on('click','.box1',function(){})
var EventUtil={
getEvent:function(event){
return event||window.event;
},
getTarget:function(event){
return event.target||event.srcElement;
},
preventDefault:function(){
if(event.preventDefault){
event.preventDefault();
}else{
event.returnValue=false;
}
},
stopPropagation:function(){
if(event.stopPropagation){
event.stopPropagation();
}else{
event.cancelBubble=true;
}
},
addEvent:function(element,type,handler){
if(element.addEventListener){
element.addEventListener(type,handler,false);
}else if(element.attachEvent){
element["e"+type]=function(){
handler.call(element)
}
element.attachEvent("on"+type,element["e"+type]);
}else{
element["on"+type]=handler;
}
},
removeEvent:function(element,type,handler){
if(element.removeEventListener){
element.removeEventListener(type,handler,false);
}else if(element.detachEvent){
element.detachEvent("on"+type,element["e"+type]);
element["e"+type]=null;
}else{
element["on"+type]=null;
}
}
};