jQuery Deferred和Promise创建响应式应用程序详细介绍

2020-05-22 15:18:24易采站长站整理

同步多个动画
  动画是另一个常见的异步任务范例。 然而在几个不相关的动画完成后执行代码仍然有点挑战性。尽管在jQuery1.6中才提供了在动画元素上取得promise对象的功能,但它是很容易的手动实现:

$.fn.animatePromise = function( prop, speed, easing, callback ) {
var elements = this;
return $.Deferred(function( defer ) {
elements.animate( prop, speed, easing, function() {
defer.resolve();
if ( callback ) {
callback.apply( this, arguments );
}
});
}).promise();
};

然后,我们可以使用$.when()同步化不同的动画:

var fadeDiv1Out = $( “#div1” ).animatePromise({ opacity: 0 }),
fadeDiv2In = $( “#div1” ).animatePromise({ opacity: 1 }, “fast” );
$.when( fadeDiv1Out, fadeDiv2In ).done(function() {
/* both animations ended */
});

我们也可以使用同样的技巧,建立了一些辅助方法:

$.each([ “slideDown”, “slideUp”, “slideToggle”, “fadeIn”, “fadeOut”, “fadeToggle” ],
function( _, name ) {
$.fn[ name + “Promise” ] = function( speed, easing, callback ) {
var elements = this;
return $.Deferred(function( defer ) {
elements[ name ]( speed, easing, function() {
defer.resolve();
if ( callback ) {
callback.apply( this, arguments );
}
});
}).promise();
};
});

然后想下面这样使用新的助手代码来同步动画:

$.when(
$( “#div1” ).fadeOutPromise(),
$( “#div2” ).fadeInPromise( “fast” )
).done(function() {
/* both animations are done */
});

一次性事件
  虽然jQuery提供你可能需要的所有的时间绑定方法,但当事件仅需要处理一次时,情况可能会变得有点棘手。( 与$.one() 不同 )
  例如,您可能希望有一个按钮,当它第一次被点击时打开一个面板,面板打开之后,执行特定的初始化逻辑。 在处理这种情况时,人们通常会这样写代码:

var buttonClicked = false;
$( “#myButton” ).click(function() {
if ( !buttonClicked ) {
buttonClicked = true;
initializeData();
showPanel();
}
});

不久后,你可能会在面板打开之后点击按钮时添加一些操作,如下:

if ( buttonClicked ) { /* perform specific action */ }

这是一个非常耦合的解决办法。 如果你想添加一些其他的操作,你必须编辑绑定代码或拷贝一份。 如果你不这样做,你唯一的选择是测试buttonClicked。由于buttonClicked可能是false,新的代码可能永远不会被执行,因此你 可能会失去这个新的动作。
  使用deferreds我们可以做的更好 (为简化起见,下面的代码将只适用于一个单一的元素和一个单一的事件类型,但它可以很容易地扩展为多个事件类型的集合):