当div可见时,jQuery事件触发动作

我在我的网站上使用jQuery,我想在某个div可见时触发某些操作。

是否有可能附加某种“isvisible”事件处理程序到任意div,并有一定的代码运行时,他们的div是可见的?

我想要如下伪代码:

$(function() {
$('#contentDiv').isvisible(function() {
alert("do something");
});
});

警报(“do something”)代码在contentDiv实际可见之前不应该触发。

谢谢。

567497 次浏览

没有本地事件你可以挂钩到这个,但是你可以在你的脚本中触发一个事件,使用.触发函数使div可见

//declare event to run when div is visible
function isVisible(){
//do something


}


//hookup the event
$('#someDivId').bind('isVisible', isVisible);


//show div and trigger custom event in callback when div is visible
$('#someDivId').show('slow', function(){
$(this).trigger('isVisible');
});

redsquare的解决方案是正确答案。

但是作为理论上的解决方案,你可以编写一个函数来选择由.visibilityCheck (并非所有可见元素)分类的元素,并检查它们的visibility属性值;如果true,那么做一些事情。

之后,应该使用setInterval()函数定期执行该函数。你可以在成功调用时使用clearInterval()停止计时器。

这里有一个例子:

function foo() {
$('.visibilityCheck').each(function() {
if ($(this).is(':visible')){
// do something
}
});
}


window.setInterval(foo, 100);

您还可以对其执行一些性能改进,但是,该解决方案基本上不适合实际使用。所以…

你总是可以添加到原来的,告诉()方法中,这样你就不必在每次显示某个东西时触发事件,或者当你需要它与遗留代码一起工作时:

Jquery扩展:

jQuery(function($) {


var _oldShow = $.fn.show;


$.fn.show = function(speed, oldCallback) {
return $(this).each(function() {
var obj         = $(this),
newCallback = function() {
if ($.isFunction(oldCallback)) {
oldCallback.apply(obj);
}
obj.trigger('afterShow');
};


// you can trigger a before show if you want
obj.trigger('beforeShow');


// now use the old function to show the element passing the new callback
_oldShow.apply(obj, [speed, newCallback]);
});
}
});

使用的例子:

jQuery(function($) {
$('#test')
.bind('beforeShow', function() {
alert('beforeShow');
})
.bind('afterShow', function() {
alert('afterShow');
})
.show(1000, function() {
alert('in show callback');
})
.show();
});

这可以有效地让您在执行原始.show()方法的正常行为的同时执行beforeShow和afterShow操作。

您还可以创建另一个方法,这样就不必重写原来的.show()方法。

如果您想在所有实际可见的元素(和子元素)上触发事件,则使用$。show, toggle, toggleClass, addClass,或removeClass:

$.each(["show", "toggle", "toggleClass", "addClass", "removeClass"], function(){
var _oldFn = $.fn[this];
$.fn[this] = function(){
var hidden = this.find(":hidden").add(this.filter(":hidden"));
var result = _oldFn.apply(this, arguments);
hidden.filter(":visible").each(function(){
$(this).triggerHandler("show"); //No bubbling
});
return result;
}
});

现在是元素:

$("#myLazyUl").bind("show", function(){
alert(this);
});

你可以通过将额外的jQuery函数添加到顶部的数组(如“attr”)来添加覆盖。

一个隐藏/显示事件触发器基于Glenns的想法: 移除切换,因为它会触发显示/隐藏,我们不希望一个事件发生2次火灾

$(function(){
$.each(["show","hide", "toggleClass", "addClass", "removeClass"], function(){
var _oldFn = $.fn[this];
$.fn[this] = function(){
var hidden = this.find(":hidden").add(this.filter(":hidden"));
var visible = this.find(":visible").add(this.filter(":visible"));
var result = _oldFn.apply(this, arguments);
hidden.filter(":visible").each(function(){
$(this).triggerHandler("show");
});
visible.filter(":hidden").each(function(){
$(this).triggerHandler("hide");
});
return result;
}
});
});
你可以使用jQuery的实时查询插件。 代码如下:

$('#contentDiv:visible').livequery(function() {
alert("do something");
});

然后每当contentDiv可见时,“do something”就会被提醒!

这个问题正在被DOM突变观察者解决。它们允许你将观察者(一个函数)绑定到改变dom元素的内容、文本或属性的事件上。

随着IE11的发布,所有主要的浏览器都支持这个功能,检查http://caniuse.com/mutationobserver

示例代码如下:

$(function() {
$('#show').click(function() {
$('#testdiv').show();
});


var observer = new MutationObserver(function(mutations) {
alert('Attributes changed!');
});
var target = document.querySelector('#testdiv');
observer.observe(target, {
attributes: true
});


});
<div id="testdiv" style="display:none;">hidden</div>
<button id="show">Show hidden div</button>


<script type="text/javascript" src="https://code.jquery.com/jquery-1.9.1.min.js"></script>

<div id="welcometo">Özhan</div>
<input type="button" name="ooo"
onclick="JavaScript:
if(document.all.welcometo.style.display=='none') {
document.all.welcometo.style.display='';
} else {
document.all.welcometo.style.display='none';
}">

此代码自动控制不需要查询可见或不可见的控制

我根据Glenns的想法改变了Catalint的隐藏/显示事件触发器。 我的问题是我有一个模块化的应用程序。我改变模块之间显示和隐藏div父母。然后,当我隐藏一个模块并显示另一个模块时,用他的方法,当我在模块之间更改时,我有一个可见的延迟。我只需要偶尔听一听这件事,而且是在一些特殊的孩子身上。所以我决定只通知类“displayObserver”

的子
$.each(["show", "hide", "toggleClass", "addClass", "removeClass"], function () {
var _oldFn = $.fn[this];
$.fn[this] = function () {
var hidden = this.find(".displayObserver:hidden").add(this.filter(":hidden"));
var visible = this.find(".displayObserver:visible").add(this.filter(":visible"));
var result = _oldFn.apply(this, arguments);
hidden.filter(":visible").each(function () {
$(this).triggerHandler("show");
});
visible.filter(":hidden").each(function () {
$(this).triggerHandler("hide");
});
return result;
}
});

然后,当一个孩子想要监听“show”或“hide”事件时,我必须给他添加类“displayObserver”,当它不想继续监听它时,我将他删除类

bindDisplayEvent: function () {
$("#child1").addClass("displayObserver");
$("#child1").off("show", this.onParentShow);
$("#child1").on("show", this.onParentShow);
},


bindDisplayEvent: function () {
$("#child1").removeClass("displayObserver");
$("#child1").off("show", this.onParentShow);
},

我希望得到帮助

使用jQuery Waypoints:

$('#contentDiv').waypoint(function() {
alert('do something');
});

其他关于jQuery Waypoints网站的例子。

下面的代码(从http://maximeparmentier.com/2012/11/06/bind-show-hide-events-with-jquery/中提取)将使您能够使用$('#someDiv').on('show', someFunc);

(function ($) {
$.each(['show', 'hide'], function (i, ev) {
var el = $.fn[ev];
$.fn[ev] = function () {
this.trigger(ev);
return el.apply(this, arguments);
};
});
})(jQuery);

有一个jQuery插件可以观察DOM属性的变化,

https://github.com/darcyclarke/jQuery-Watch-Plugin

你需要做的就是绑定MutationObserver

然后你可以使用它来观看div使用:

$("#selector").watch('css', function() {
console.log("Visibility: " + this.style.display == 'none'?'hidden':'shown'));
//or any random events
});

有一种方法 只适用于由css类更改引起的可见性更改,但也可以扩展到观察属性更改

var observer = new MutationObserver(function(mutations) {
var clone = $(mutations[0].target).clone();
clone.removeClass();
for(var i = 0; i < mutations.length; i++){
clone.addClass(mutations[i].oldValue);
}
$(document.body).append(clone);
var cloneVisibility = $(clone).is(":visible");
$(clone).remove();
if (cloneVisibility != $(mutations[0].target).is(":visible")){
var visibilityChangedEvent = document.createEvent('Event');
visibilityChangedEvent.initEvent('visibilityChanged', true, true);
mutations[0].target.dispatchEvent(visibilityChangedEvent);
}
});


var targets = $('.ui-collapsible-content');
$.each(targets, function(i,target){
target.addEventListener('visibilityChanged',VisbilityChanedEventHandler});
target.addEventListener('DOMNodeRemovedFromDocument',VisbilityChanedEventHandler });
observer.observe(target, { attributes: true, attributeFilter : ['class'], childList: false, attributeOldValue: true });
});


function VisbilityChanedEventHandler(e){console.log('Kaboom babe'); console.log(e.target); }

我的解决方案:

; (function ($) {
$.each([ "toggle", "show", "hide" ], function( i, name ) {
var cssFn = $.fn[ name ];
$.fn[ name ] = function( speed, easing, callback ) {
if(speed == null || typeof speed === "boolean"){
var ret=cssFn.apply( this, arguments )
$.fn.triggerVisibleEvent.apply(this,arguments)
return ret
}else{
var that=this
var new_callback=function(){
callback.call(this)
$.fn.triggerVisibleEvent.apply(that,arguments)
}
var ret=this.animate( genFx( name, true ), speed, easing, new_callback )
return ret
}
};
});


$.fn.triggerVisibleEvent=function(){
this.each(function(){
if($(this).is(':visible')){
$(this).trigger('visible')
$(this).find('[data-trigger-visible-event]').triggerVisibleEvent()
}
})
}
})(jQuery);

使用示例:

if(!$info_center.is(':visible')){
$info_center.attr('data-trigger-visible-event','true').one('visible',processMoreLessButton)
}else{
processMoreLessButton()
}


function processMoreLessButton(){
//some logic
}

我有同样的问题,并创建了一个jQuery插件来解决我们的网站。

https://github.com/shaunbowe/jquery.visibilityChanged

下面是基于你的例子你应该如何使用它:

$('#contentDiv').visibilityChanged(function(element, visible) {
alert("do something");
});

这支持动画完成后的缓和和触发事件![jQuery 2.2.4测试]

(function ($) {
$.each(['show', 'hide', 'fadeOut', 'fadeIn'], function (i, ev) {
var el = $.fn[ev];
$.fn[ev] = function () {
var result = el.apply(this, arguments);
var _self=this;
result.promise().done(function () {
_self.triggerHandler(ev, [result]);
//console.log(_self);
});
return result;
};
});
})(jQuery);

灵感来自http://viralpatel.net/blogs/jquery-trigger-custom-event-show-hide-element/

$( window ).scroll(function(e,i) {
win_top = $( window ).scrollTop();
win_bottom = $( window ).height() + win_top;
//console.log( win_top,win_bottom );
$('.onvisible').each(function()
{
t = $(this).offset().top;
b = t + $(this).height();
if( t > win_top && b < win_bottom )
alert("do something");
});
});

希望这将以最简单的方式完成工作:

$("#myID").on('show').trigger('displayShow');


$('#myID').off('displayShow').on('displayShow', function(e) {
console.log('This event will be triggered when myID will be visible');
});

在这里帮助我的是最近的ResizeObserver规格填充:

const divEl = $('#section60');


const ro = new ResizeObserver(() => {
if (divEl.is(':visible')) {
console.log("it's visible now!");
}
});
ro.observe(divEl[0]);

注意,它是跨浏览器和性能的(没有轮询)。

我做了一个简单的setinterval函数来实现这一点。如果类为div1的元素可见,则将div2设置为可见。我不知道有什么好办法,只有一个简单的补救办法。

setInterval(function(){
if($('.div1').is(':visible')){
$('.div2').show();
}
else {
$('.div2').hide();
}
}, 100);

只需将触发器与选择器绑定,并将代码放入触发器事件中:

jQuery(function() {
jQuery("#contentDiv:hidden").show().trigger('show');


jQuery('#contentDiv').on('show', function() {
console.log('#contentDiv is now visible');
// your code here
});
});

$(function() {
$(document).click(function (){
if ($('#contentDiv').is(':visible')) {
alert("Visible");
} else {
alert("Hidden");
}
});
});
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
<div id="contentDiv">Test I'm here</div>


<button onclick="$('#contentDiv').toggle();">Toggle the div</button>