130 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			130 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| "use strict";
 | |
| module.exports = function(Promise, PromiseArray, apiRejection, debug) {
 | |
| var util = require("./util");
 | |
| var tryCatch = util.tryCatch;
 | |
| var errorObj = util.errorObj;
 | |
| var async = Promise._async;
 | |
| 
 | |
| Promise.prototype["break"] = Promise.prototype.cancel = function() {
 | |
|     if (!debug.cancellation()) return this._warn("cancellation is disabled");
 | |
| 
 | |
|     var promise = this;
 | |
|     var child = promise;
 | |
|     while (promise._isCancellable()) {
 | |
|         if (!promise._cancelBy(child)) {
 | |
|             if (child._isFollowing()) {
 | |
|                 child._followee().cancel();
 | |
|             } else {
 | |
|                 child._cancelBranched();
 | |
|             }
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         var parent = promise._cancellationParent;
 | |
|         if (parent == null || !parent._isCancellable()) {
 | |
|             if (promise._isFollowing()) {
 | |
|                 promise._followee().cancel();
 | |
|             } else {
 | |
|                 promise._cancelBranched();
 | |
|             }
 | |
|             break;
 | |
|         } else {
 | |
|             if (promise._isFollowing()) promise._followee().cancel();
 | |
|             promise._setWillBeCancelled();
 | |
|             child = promise;
 | |
|             promise = parent;
 | |
|         }
 | |
|     }
 | |
| };
 | |
| 
 | |
| Promise.prototype._branchHasCancelled = function() {
 | |
|     this._branchesRemainingToCancel--;
 | |
| };
 | |
| 
 | |
| Promise.prototype._enoughBranchesHaveCancelled = function() {
 | |
|     return this._branchesRemainingToCancel === undefined ||
 | |
|            this._branchesRemainingToCancel <= 0;
 | |
| };
 | |
| 
 | |
| Promise.prototype._cancelBy = function(canceller) {
 | |
|     if (canceller === this) {
 | |
|         this._branchesRemainingToCancel = 0;
 | |
|         this._invokeOnCancel();
 | |
|         return true;
 | |
|     } else {
 | |
|         this._branchHasCancelled();
 | |
|         if (this._enoughBranchesHaveCancelled()) {
 | |
|             this._invokeOnCancel();
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
|     return false;
 | |
| };
 | |
| 
 | |
| Promise.prototype._cancelBranched = function() {
 | |
|     if (this._enoughBranchesHaveCancelled()) {
 | |
|         this._cancel();
 | |
|     }
 | |
| };
 | |
| 
 | |
| Promise.prototype._cancel = function() {
 | |
|     if (!this._isCancellable()) return;
 | |
|     this._setCancelled();
 | |
|     async.invoke(this._cancelPromises, this, undefined);
 | |
| };
 | |
| 
 | |
| Promise.prototype._cancelPromises = function() {
 | |
|     if (this._length() > 0) this._settlePromises();
 | |
| };
 | |
| 
 | |
| Promise.prototype._unsetOnCancel = function() {
 | |
|     this._onCancelField = undefined;
 | |
| };
 | |
| 
 | |
| Promise.prototype._isCancellable = function() {
 | |
|     return this.isPending() && !this._isCancelled();
 | |
| };
 | |
| 
 | |
| Promise.prototype.isCancellable = function() {
 | |
|     return this.isPending() && !this.isCancelled();
 | |
| };
 | |
| 
 | |
| Promise.prototype._doInvokeOnCancel = function(onCancelCallback, internalOnly) {
 | |
|     if (util.isArray(onCancelCallback)) {
 | |
|         for (var i = 0; i < onCancelCallback.length; ++i) {
 | |
|             this._doInvokeOnCancel(onCancelCallback[i], internalOnly);
 | |
|         }
 | |
|     } else if (onCancelCallback !== undefined) {
 | |
|         if (typeof onCancelCallback === "function") {
 | |
|             if (!internalOnly) {
 | |
|                 var e = tryCatch(onCancelCallback).call(this._boundValue());
 | |
|                 if (e === errorObj) {
 | |
|                     this._attachExtraTrace(e.e);
 | |
|                     async.throwLater(e.e);
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             onCancelCallback._resultCancelled(this);
 | |
|         }
 | |
|     }
 | |
| };
 | |
| 
 | |
| Promise.prototype._invokeOnCancel = function() {
 | |
|     var onCancelCallback = this._onCancel();
 | |
|     this._unsetOnCancel();
 | |
|     async.invoke(this._doInvokeOnCancel, this, onCancelCallback);
 | |
| };
 | |
| 
 | |
| Promise.prototype._invokeInternalOnCancel = function() {
 | |
|     if (this._isCancellable()) {
 | |
|         this._doInvokeOnCancel(this._onCancel(), true);
 | |
|         this._unsetOnCancel();
 | |
|     }
 | |
| };
 | |
| 
 | |
| Promise.prototype._resultCancelled = function() {
 | |
|     this.cancel();
 | |
| };
 | |
| 
 | |
| };
 |