3

HTML:

<ul class="topnav">
    <li><a href="#"><span>One</span></a></li>
    <li><a href="#"><span>Two</span></a></li>
    <li>
        <li><a href="#"><span>Three</span></a></li>
        <ul class="subnav">
            <li><a href="#">A</a></li>
            <li><a href="#">B</a></li>
            <li><a href="#">C</a></li>
        </ul>
    </li>
</ul>

jQuery:

var timeout = null;

$(document).ready(function() {

    $("ul.topnav li").mouseover(function() {

        if (timeout) clearTimeout(timeout);

        $(this).find("ul.subnav").slideDown('fast').show();

    }).mouseout(function() {
        timeout = setTimeout(closemenu, 500);
    });

    // sub menu mouseovers keep dropdown open
    $("ul.subnav li").mouseover(function() {
        if (timeout) clearTimeout(timeout);
    }
    ).mouseout(function() {
        timeout = setTimeout(closemenu, 500);
        // alert(timeout);

    });

    // any click closes
    $(document).click(closemenu);
});

// Closes all open menus 
function closemenu() {
    $('ul.subnav:visible').hide();
    if (timeout) clearTimeout(timeout);
} 

我遇到了超时问题。在使用中,如果我将鼠标悬停在“三”上,下拉菜单将永远存在。如果我将鼠标悬停在“A”上,下拉菜单将永远消失,但如果我将鼠标悬停在“B”或更低的位置,菜单将关闭我。如果您取消注释“//警报(超时);” 它为 B(和 A)到达那里,但超时将有一个值。为什么是这样?我认为 clearTimeout 会使超时变量为空?

4

2 回答 2

5

.hover()你可以通过使用和.data()这样的方式来简化你的代码:

$(function() {
  $("ul.topnav li").hover(function() {
    var timeout = $(this).data("timeout");
    if(timeout) clearTimeout(timeout);
    $(this).find("ul.subnav").slideDown('fast');
  }, function() {
      $(this).data("timeout", setTimeout($.proxy(function() {
          $(this).find("ul.subnav").slideUp();
      }, this), 500));
  });
  $(document).click(function() {
      $('ul.subnav:visible').hide();
  });
});​

你可以在这里看到一个工作演示

这不是共享一个全局timeout变量,而是为每个顶层设置一个超时<li>,每个都有一个独立的计时器,当你将鼠标悬停在那个元素上时,只有它的计时器被清除。也.hover()使用mouseenterand mouseleave,而不是mouseoverand mouseout,区别在于当你进入一个孩子或孩子之间时,mouseenter不会再次开火,也不会对我们关心 mouseleave的父母开火。<li>

您可以使用上面的演示链接对此进行测试,我也在第一个菜单中添加了子项,以证明它们是独立的。如果您碰巧对那里有疑问$.proxy,它只是this在该超时匿名函数内部引用我想要的(当前this)...在超时后需要关闭的元素。

于 2010-06-05T11:12:24.817 回答
0

你想建立一个下拉菜单?为什么不为此使用现有的 jquery 插件,或者更好的是,只有 css 的下拉菜单,如http://purecssmenu.com/

于 2010-02-23T05:07:20.063 回答