• 解决js中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事件时再进行上述判断分别处理比较合适。

      

    ////////////////另一种好办法

     <meta http-equiv="content-type" content="text/html;charset=gb2312">

    <style>
    #odiv{200px;height:200px;border:5px solid gray}
    </style>
    <script language=javascript>
    window.attachEvent("onload",function(){
      var rect=odiv.getBoundingClientRect();

      var l=rect.left,r=rect.right,t=rect.top,b=rect.bottom;
      var indiv=false;
      document.attachEvent("onmousemove",function(){
        var x=event.clientX+document.body.scrollLeft;
        var y=event.clientY+document.body.scrollTop;
        var temp=(x>=l&&x<=r&&y>=t&&y<=b);
        if(indiv&&!temp)
          doonmouseout();
        indiv=temp;
        });
      });
    function doonmouseout(){
    var alerttext=document.getElementById("alert");
      alerttext.innerHTML="鼠标移出DIV区域";
      }
    </script>
    <div id=odiv>

      <p><a href="order_down.php" name="passname" >http://111.com</a></p>
      <p><img src="未标题-1.png" width="80" height="80" /></p>
      <p><span>zxfdsfsdfdsfdsfdsf</span></p>
    </div>
    <div id="alert">

    </div>
  • 相关阅读:
    LeeCode-Invert Binary Tree
    LeeCode-Contains Duplicate
    LeeCode-Linked List Cycle
    LeeCode-Number of 1 Bits
    LeeCode-Delete Node in a Linked List
    LeeCode-Same Tree
    LeeCode-String to Integer (atoi)
    单链表复习
    POJ1258 (最小生成树prim)
    HDU1248 (完全背包简单变形)
  • 原文地址:https://www.cnblogs.com/zzh/p/1607011.html
Copyright © 2020-2023  润新知