Нажмите внешнее меню, чтобы закрыть в 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();
});