在元素边界出触发不断触发hover事件如何解决?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在元素边界出触发不断触发hover事件如何解决?相关的知识,希望对你有一定的参考价值。

鼠标移入元素 升高 离开降低,但是鼠标在元素边界的时候就不断触发,就开始疯狂升高降低,录屏可能开不太出效果,怎么解决

这种情况一般不用特别处理。
因为鼠标的精度本来就不可能很高,在边界附近可能会出现有时在内部,有时在外部的情况,所以会不断触发。
如果需要解决只能自己用js写一个hover方法,在移入移出的时候进行延迟,比如延迟0.3秒再更改元素的样式。
参考技术A 河南新华专注互联网教育32年 到校参观路费报销 免费试听课程!! 参考技术B 这个好像本来就是会出现这样的一个情况的,就算在百度知道这么大的一个网站hover效果也是会在边界疯狂跳动;这个也不知道算不算html的一个bug,也没看到有人去刻意改动过(可能他们就算尝试去改也没改掉)

解决内部元素onMouseOver/onMouseOut事件冒泡触发父元素的相应事件

前阵子为BS项目模板做了一个左侧滑动信息栏,效果类似于windows状态栏的自动隐藏效果,鼠标移进滑出,鼠标移出隐藏,浮动时不占用空间,也可以固定住占一块位置。做的过程中遇到一个问题,鼠标在信息栏内部移动时会触发onMouseOut事件,信息栏放在div中,内部有table、img、a等元素,看来是由于事件冒泡,子元素上触发了事件冒泡到了父元素,导致滑动时不断闪动,头晕眼花,于是着手解决问题。

 

技术分享
方法一:

在IE下解决问题很简单,用onMouseEnter、onMouseLeave来代替onMouseOver、onMouseOut就行了,他们的作用基本相同,前者不会发生冒泡。但是firefox下没有这两个事件,公司BS项目模板要同时兼容IE和firefox,没办法,只能再找别的方法(多浏览器分天下的年代,解决问题总是没那么简单......(/_\)!)


方法二:
有一个在IE和firefox下都行得通的解决思路,通过判断触发onMouseOut事件后鼠标到达的元素是不是包含在父元素(信息栏Div)内,如果是就表示鼠标还在信息栏上,则不隐藏,如果否就表示鼠标真的移出了信息栏,那么信息栏隐藏,思路有了,那么就一步步来解决问题

首先来获取触发onMouseOut事件的相关元素,IE下event的属性toElement来获得,在firefox下变成了relatedTarget(相关元素)
IE:event.toElement    Firefox:event.relatedTarget(注意Firefox下event须要调用函数时传入)

接下来就是判断获取的元素是否是子元素,IE下通过元素的contains(Element)方法可以判断,同样的firefox下没有这个方法!!,不过可以给firefox下的元素定义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;  //结束函式
                          }
                   }
             }
            /*要执行的操作*/
 }

在信息栏(Div)上设置onMouseOut=hideMsgBox(event)来调用,圆满解决冒泡问题。

方法三:
事实上通过设置
window.event.cancelBubble = true (IE)   event.stopPropagation()  event.preventDefault() (Firefox)
也可以解决问题,但是需要遍历所有子元素,影响效率,所以还是在触发onMouseOut事件时再进行上述判断分别处理比较合适。

以上是关于在元素边界出触发不断触发hover事件如何解决?的主要内容,如果未能解决你的问题,请参考以下文章

Hover/mouseenter 事件反复触发 jQuery UI 的“动画”

编程中hover啥意思

javascript 如何防止多次触发hover事件

浅析jQuery里面的.hover事件

解决内部元素onMouseOver/onMouseOut事件冒泡触发父元素的相应事件

JQuery事件与效果