也就是说,移到对象的子对象上,也算onmouseout了。但这往往会达不到我们想要的预期效果。这是由于javascript自身的冒泡特性导致的(即在子元素上触发了事件,并冒泡到了父元素-堆栈后进先出算法)。今天在网上搜了一下,找了以下的解决办法(兼容ie和firefox)。
在ie下解决问题很简单,用onmouseenter、onmouseleave来代替onmouseover、onmouseout就行了,他们的作用基本相同,但前者不会发生冒泡(如果用 jquery的event事件,只要绑定mouseleave、mouseenter即可)。但firefox下并没有onmouseenter、onmouseleave这两个事件。那么只能使用纯js来解决ie及firefox的兼容性问题了:
原理:通过判断触发onmouseout事件后,鼠标到达的元素是不是包含在父元素信息栏(div)内,如果是就表示鼠标还在信息栏(div)上,则不隐藏。如果否就表示鼠标真的移出了信息栏(div),那么信息栏就要隐藏。
复制代码 代码如下:
// 首先来获取触发onmouseout事件的元素,ie下通过event的toelement属性来获得,firefox下通过event的relatedtarget属性来获得。
ie:event.toelement ,firefox:event.relatedtarget(注意:firefox下的event须通过在函数调用时传入,而ie下的event则可以直接通过window.event系统对象来获得)
// ① 接下来就是判断获取的元素是否是主体div的子元素(ie下可以通过元素的obj.contains(element)方法来判断,但firefox下没有这个方法,所以需要给firefox定义元素的obj.contains()方法)。
代码如下:
if(typeof(htmlelement)!=undefined) // 给firefox定义contains()方法,ie已经系统自带有这个方法了
{
htmlelement.prototype.contains=function(obj) {
while(obj!=null&&typeof(obj.tagname)!=undefind) { // 通过循环对比来判断是不是obj的父元素
if(obj==this) { return true; }
obj=obj.parentnode;
}
return false;
};
}
// ② 获取和判断搞定后,我们就可以通过判断ie和firefox来针对处理了,通过navigator.useragent来判断浏览器类型:
if(navigator.useragent.indexof(msie)>0) {
return msie;
}
if(navigator.useragent.indexof(firefox)>0){
return firefox;
}
// ③ 到此为止所有要解决的问题都得到了解决,当触发onmouseout事件时,我们将针对不同的浏览器先获取鼠标到达的元素,然后通过判断该元素是否在信息栏(div)内,如果元素是子元素,那么不执行onmouseout事件,反之则执行事件,隐藏信息栏,完成后的代码如下:
function hidemsgbox(theevent){ //theevent用来传入事件,firefox的方式
if (theevent){
var browser=navigator.useragent; //取得浏览器属性
if (browser.indexof(firefox)>0){ //如果是firefox
if (document.getelementbyid('msgbox').contains(theevent.relatedtarget)) {//如果是子元素
return; //结束函数
}
}
if (browser.indexof(msie)>0){ //如果是ie
if (document.getelementbyid('msgbox').contains(event.toelement)) { //如果是子元素
return; //结束函数
}
}
}
/*要执行的操作(如:隐藏)*/
document.getelementbyid('msgbox').style.display='none' ;
……
}
// ④ 在信息栏(div)上设置onmouseout=hidemsgbox(event)来调用。
另外,通过设置window.event.cancelbubble = true (ie) ,event.stoppropagation() event.preventdefault() (firefox) 也可以解决问题,但是需要遍历所有子元素,影响效率,所以还是在触发onmouseout事件时再进行上述判断分别处理比较合适。