Нажмите внешнее меню, чтобы закрыть в jquery

Итак, у меня есть раскрывающееся меню, которое отображается щелчком по бизнес-требованиям. Меню скрывается снова после того, как вы от него отмахнетесь.

Но теперь меня просят, чтобы он оставался на месте, пока пользователь не щелкнет где-нибудь в документе. Как это можно сделать?

Это упрощенная версия того, что у меня есть сейчас:

$(document).ready(function() {
  $("ul.opMenu li").click(function(){
   $('#MainOptSubMenu',this).css('visibility', 'visible');
  });

  $("ul.opMenu li").mouseleave(function(){
      $('#MainOptSubMenu',this).css('visibility', 'hidden');
  });
});



<ul  class="opMenu">
  <li id="footwo" class="">
    <span id="optImg" style="display: inline-block;"> <img src="http://localhost.vmsinfo.com:8002/insight/images/options-hover2.gif"/> </span>
      <ul id="MainOptSubMenu" style="visibility: hidden; top: 25px; border-top: 0px solid rgb(217, 228, 250); background-color: rgb(217, 228, 250); padding-bottom: 15px;">
        <li>some</li>
       <li>nav</li>
       <li>links</li>
       </ul>
    </li>
</ul> 

Я пробовал что-то вроде этого $('document[id!=MainOptSubMenu]').click(function(), думая, что он выйдет на что-либо, что не было в меню, но оно не работает.

Ответ 1

Взгляните на подход, который использовал этот вопрос:

Как обнаружить клик вне элемента?

Прикрепите событие клика к телу документа, который закрывает окно. Прикрепите отдельное событие клика к окну, которое прекратит распространение на тело документа.
$('html').click(function() {
  //Hide the menus if visible
});

$('#menucontainer').click(function(event){
    event.stopPropagation();
});

Ответ 2

Ответ правильный, но он добавит слушателя, который будет запускаться каждый раз, когда на вашей странице будет клик. Чтобы этого избежать, вы можете добавить слушателя только один раз:

$('a#menu-link').on('click', function(e) {
    e.preventDefault();
    e.stopPropagation();

    $('#menu').toggleClass('open');

    $(document).one('click', function closeMenu (e){
        if($('#menu').has(e.target).length === 0){
            $('#menu').removeClass('open');
        } else {
            $(document).one('click', closeMenu);
        }
    });
});

Изменить: если вы хотите избежать stopPropagation() на начальной кнопке, вы можете использовать этот

var $menu = $('#menu');

$('a#menu-link').on('click', function(e) {
    e.preventDefault();

    if (!$menu.hasClass('active')) {
        $menu.addClass('active');

        $(document).one('click', function closeTooltip(e) {
            if ($menu.has(e.target).length === 0 && $('a#menu-link').has(e.target).length === 0) {
                $menu.removeClass('active');
            } else if ($menu.hasClass('active')) {
                $(document).one('click', closeTooltip);
            }
        });
    } else {
        $menu.removeClass('active');
    }
});

Ответ 3

Если вы используете плагин в порядке, то я предлагаю плагин click clickside от Ben Alman, расположенный здесь:

его использование так же просто, как это:

$('#menu').bind('clickoutside', function (event) {
    $(this).hide();
});

надеюсь, что это поможет.

Ответ 4

Параметры stopPropagation плохие, потому что они могут мешать другим обработчикам событий, включая другие меню, которые могли бы прикрепить обработчики закрытия к элементу HTML.

Вот простое решение, основанное на ответе пользователя user2989143:

$('html').click(function(event) {
    if ($(event.target).closest('#menu-container, #menu-activator').length === 0) {
        $('#menu-container').hide();
    }
});

Ответ 5

2 варианта, которые вы можете исследовать:

  • При отображении меню поместите большой пустой DIV за ним, закрывая остальную часть страницы и давая возможность щелкнуть событие, чтобы закрыть меню (и его самого). Это сродни методам, используемым в лайтбоксах, когда щелчок по фону закрывает лайтбокс.
  • При отображении меню присоедините обработчик события однократного клика на теле, который закрывает меню. Для этого используется jQuery '.one()'.

Ответ 6

Недавно я столкнулся с той же проблемой. Я написал следующий код:

    $('html').click(function(e) {
      var a = e.target;
      if ($(a).parents('.menu_container').length === 0) {
        $('.ofSubLevelLinks').removeClass('active'); //hide menu item
        $('.menu_container li > img').hide(); //hide dropdown image, if any
     }
    });

Он отлично работал у меня.

Ответ 7

как насчет этого?

    $(this).mouseleave(function(){  
        var thisUI = $(this);
        $('html').click(function(){
                thisUI.hide();
            $('html').unbind('click');
         });
     });

Ответ 8

Я нашел вариант решения Grsmto и решение Dennis исправил мою проблему.

$(".MainNavContainer").click(function (event) {
    //event.preventDefault();  // Might cause problems depending on implementation
    event.stopPropagation();

    $(document).one('click', function (e) {
        if(!$(e.target).is('.MainNavContainer')) {
            // code to hide menus
        }
    });
});

Ответ 9

Я использую это решение с несколькими элементами с одинаковым поведением на одной странице:

$("html").click(function(event){
    var otarget = $(event.target);
    if (!otarget.parents('#id_of element').length && otarget.attr('id')!="id_of element" && !otarget.parents('#id_of_activator').length) {
        $('#id_of element').hide();
    }
});

stopPropagation() - плохая идея, это нарушает стандартное поведение многих вещей, включая кнопки и ссылки.

Ответ 10

даже я столкнулся с той же ситуацией, и один из моих наставников передал эту идею себе.

шаг: 1 при нажатии на кнопку, на которой мы должны отобразить раскрывающееся меню. затем добавьте нижележащее имя класса "more_wrap_background" к текущей активной странице, как показано ниже.

$('.ui-page-active').append("<div class='more_wrap_background' id='more-wrap-bg'> </div>");

шаг-2, затем добавьте клики для тега div, например

$(document).on('click', '#more-wrap-bg', hideDropDown);

где hideDropDown - это функция, вызываемая для скрытия выпадающего меню

Шаг-3, а важный шаг при скрытии выпадающего меню заключается в том, что удалить тот класс, который вы добавили ранее, например

$('#more-wrap-bg').remove();

Я удаляю, используя свой идентификатор в приведенном выше коде

.more_wrap_background {
  top: 0;
  padding: 0;
  margin: 0;
  background: rgba(0, 0, 0, 0.1);
  position: fixed;
  display: block;
  width: 100% !important;
  z-index: 999;//should be one less than the drop down menu z-index
  height: 100% !important;
}

Ответ 11

Я считаю более полезным использовать mousedown-событие вместо click-event. Событие click-event не работает, если пользователь нажимает на другие элементы на странице с помощью событий click-events. В сочетании с методом jQuery one() он выглядит следующим образом:

$("ul.opMenu li").click(function(event){

   //event.stopPropagation(); not required any more
   $('#MainOptSubMenu').show();

   // add one mousedown event to html
   $('html').one('mousedown', function(){
       $('#MainOptSubMenu').hide();
   });
});

// mousedown must not be triggered inside menu
$("ul.opMenu li").bind('mousedown', function(evt){
    evt.stopPropagation();
});

Ответ 12

$("html").click( onOutsideClick );
onOutsideClick = function( e )
{
    var t = $( e.target );
    if ( !(
        t.is("#mymenu" ) ||     //Where #mymenu - is a div container of your menu
        t.parents( "#mymenu" ).length > 0
        )   )
    {
        //TODO: hide your menu
    }
};

И лучше установить слушателя только тогда, когда ваше меню будет видимым и всегда удалит слушателя после того, как меню станет скрытым.

Ответ 13

Я думаю, вам нужно что-то вроде этого: http://jsfiddle.net/BeenYoung/BXaqW/3/

$(document).ready(function() {
  $("ul.opMenu li").each(function(){
      $(this).click(function(){
            if($(this).hasClass('opened')==false){          
                $('.opMenu').find('.opened').removeClass('opened').find('ul').slideUp();
                $(this).addClass('opened'); 
                $(this).find("ul").slideDown();
            }else{
                $(this).removeClass('opened'); 
                $(this).find("ul").slideUp();               
            }
      });
  });    
});

Я надеюсь, что это будет полезно для вас!

Ответ 14

Используйте селектор ": visible". Где .menuitem - это скрытый элемент (ы)...

$('body').click(function(){
  $('.menuitem:visible').hide('fast');
});

Или, если у вас уже есть элемент .menuitem в var...

var menitems = $('.menuitem');
$('body').click(function(){
  menuitems.filter(':visible').hide('fast');
});

Ответ 15

Это не должно быть сложным.

$(document).on('click', function() {
    $("#menu:not(:hover)").hide();
});