/
PCancelable.ts
93 lines (79 loc) · 2.07 KB
/
PCancelable.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
/**
* Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
/* eslint-disable no-this-before-super, constructor-super */
class CancelError extends Error {
constructor() {
super('Promise was canceled');
this.name = 'CancelError';
}
}
// @ts-ignore
class PCancelable<T> extends Promise {
private _pending: boolean;
private _canceled: boolean;
private _cancel?: Function;
// @ts-ignore
private _reject: (reason?: any) => void;
private _promise: Promise<T>;
static CancelError: CancelError;
static fn<F extends (...args: any[]) => any>(fn: F) {
return function(...args: Parameters<F>) {
return new PCancelable<ReturnType<F>>((onCancel, resolve, reject) => {
args.unshift(onCancel);
fn.apply(null, args).then(resolve, reject);
});
};
}
// @ts-ignore
constructor(executor: (onCancel, resolve, reject) => any) {
// @ts-ignore
this._pending = true;
// @ts-ignore
this._canceled = false;
// @ts-ignore
this._promise = new Promise<T>((resolve, reject) => {
this._reject = reject;
return executor(
(fn: Function) => {
this._cancel = fn;
},
(val: T) => {
this._pending = false;
resolve(val);
},
(err?: any) => {
this._pending = false;
reject(err);
},
);
});
}
then(...args: Parameters<Promise<T>['then']>) {
return this._promise.then.apply(this._promise, args);
}
catch(...args: Parameters<Promise<T>['catch']>) {
return this._promise.catch.apply(this._promise, args);
}
cancel() {
if (!this._pending || this._canceled) {
return;
}
if (typeof this._cancel === 'function') {
try {
this._cancel();
} catch (err) {
this._reject(err);
}
}
this._canceled = true;
this._reject(new CancelError());
}
get canceled() {
return this._canceled;
}
}
export = PCancelable;