1 | import { isArray } from './util/isArray';
|
---|
2 | import { isObject } from './util/isObject';
|
---|
3 | import { isFunction } from './util/isFunction';
|
---|
4 | import { UnsubscriptionError } from './util/UnsubscriptionError';
|
---|
5 | export class Subscription {
|
---|
6 | constructor(unsubscribe) {
|
---|
7 | this.closed = false;
|
---|
8 | this._parentOrParents = null;
|
---|
9 | this._subscriptions = null;
|
---|
10 | if (unsubscribe) {
|
---|
11 | this._ctorUnsubscribe = true;
|
---|
12 | this._unsubscribe = unsubscribe;
|
---|
13 | }
|
---|
14 | }
|
---|
15 | unsubscribe() {
|
---|
16 | let errors;
|
---|
17 | if (this.closed) {
|
---|
18 | return;
|
---|
19 | }
|
---|
20 | let { _parentOrParents, _ctorUnsubscribe, _unsubscribe, _subscriptions } = this;
|
---|
21 | this.closed = true;
|
---|
22 | this._parentOrParents = null;
|
---|
23 | this._subscriptions = null;
|
---|
24 | if (_parentOrParents instanceof Subscription) {
|
---|
25 | _parentOrParents.remove(this);
|
---|
26 | }
|
---|
27 | else if (_parentOrParents !== null) {
|
---|
28 | for (let index = 0; index < _parentOrParents.length; ++index) {
|
---|
29 | const parent = _parentOrParents[index];
|
---|
30 | parent.remove(this);
|
---|
31 | }
|
---|
32 | }
|
---|
33 | if (isFunction(_unsubscribe)) {
|
---|
34 | if (_ctorUnsubscribe) {
|
---|
35 | this._unsubscribe = undefined;
|
---|
36 | }
|
---|
37 | try {
|
---|
38 | _unsubscribe.call(this);
|
---|
39 | }
|
---|
40 | catch (e) {
|
---|
41 | errors = e instanceof UnsubscriptionError ? flattenUnsubscriptionErrors(e.errors) : [e];
|
---|
42 | }
|
---|
43 | }
|
---|
44 | if (isArray(_subscriptions)) {
|
---|
45 | let index = -1;
|
---|
46 | let len = _subscriptions.length;
|
---|
47 | while (++index < len) {
|
---|
48 | const sub = _subscriptions[index];
|
---|
49 | if (isObject(sub)) {
|
---|
50 | try {
|
---|
51 | sub.unsubscribe();
|
---|
52 | }
|
---|
53 | catch (e) {
|
---|
54 | errors = errors || [];
|
---|
55 | if (e instanceof UnsubscriptionError) {
|
---|
56 | errors = errors.concat(flattenUnsubscriptionErrors(e.errors));
|
---|
57 | }
|
---|
58 | else {
|
---|
59 | errors.push(e);
|
---|
60 | }
|
---|
61 | }
|
---|
62 | }
|
---|
63 | }
|
---|
64 | }
|
---|
65 | if (errors) {
|
---|
66 | throw new UnsubscriptionError(errors);
|
---|
67 | }
|
---|
68 | }
|
---|
69 | add(teardown) {
|
---|
70 | let subscription = teardown;
|
---|
71 | if (!teardown) {
|
---|
72 | return Subscription.EMPTY;
|
---|
73 | }
|
---|
74 | switch (typeof teardown) {
|
---|
75 | case 'function':
|
---|
76 | subscription = new Subscription(teardown);
|
---|
77 | case 'object':
|
---|
78 | if (subscription === this || subscription.closed || typeof subscription.unsubscribe !== 'function') {
|
---|
79 | return subscription;
|
---|
80 | }
|
---|
81 | else if (this.closed) {
|
---|
82 | subscription.unsubscribe();
|
---|
83 | return subscription;
|
---|
84 | }
|
---|
85 | else if (!(subscription instanceof Subscription)) {
|
---|
86 | const tmp = subscription;
|
---|
87 | subscription = new Subscription();
|
---|
88 | subscription._subscriptions = [tmp];
|
---|
89 | }
|
---|
90 | break;
|
---|
91 | default: {
|
---|
92 | throw new Error('unrecognized teardown ' + teardown + ' added to Subscription.');
|
---|
93 | }
|
---|
94 | }
|
---|
95 | let { _parentOrParents } = subscription;
|
---|
96 | if (_parentOrParents === null) {
|
---|
97 | subscription._parentOrParents = this;
|
---|
98 | }
|
---|
99 | else if (_parentOrParents instanceof Subscription) {
|
---|
100 | if (_parentOrParents === this) {
|
---|
101 | return subscription;
|
---|
102 | }
|
---|
103 | subscription._parentOrParents = [_parentOrParents, this];
|
---|
104 | }
|
---|
105 | else if (_parentOrParents.indexOf(this) === -1) {
|
---|
106 | _parentOrParents.push(this);
|
---|
107 | }
|
---|
108 | else {
|
---|
109 | return subscription;
|
---|
110 | }
|
---|
111 | const subscriptions = this._subscriptions;
|
---|
112 | if (subscriptions === null) {
|
---|
113 | this._subscriptions = [subscription];
|
---|
114 | }
|
---|
115 | else {
|
---|
116 | subscriptions.push(subscription);
|
---|
117 | }
|
---|
118 | return subscription;
|
---|
119 | }
|
---|
120 | remove(subscription) {
|
---|
121 | const subscriptions = this._subscriptions;
|
---|
122 | if (subscriptions) {
|
---|
123 | const subscriptionIndex = subscriptions.indexOf(subscription);
|
---|
124 | if (subscriptionIndex !== -1) {
|
---|
125 | subscriptions.splice(subscriptionIndex, 1);
|
---|
126 | }
|
---|
127 | }
|
---|
128 | }
|
---|
129 | }
|
---|
130 | Subscription.EMPTY = (function (empty) {
|
---|
131 | empty.closed = true;
|
---|
132 | return empty;
|
---|
133 | }(new Subscription()));
|
---|
134 | function flattenUnsubscriptionErrors(errors) {
|
---|
135 | return errors.reduce((errs, err) => errs.concat((err instanceof UnsubscriptionError) ? err.errors : err), []);
|
---|
136 | }
|
---|
137 | //# sourceMappingURL=Subscription.js.map |
---|