import { Operator } from './Operator';
|
import { Observable } from './Observable';
|
import { Subscriber } from './Subscriber';
|
import { Subscription } from './Subscription';
|
import { Observer, SubscriptionLike, TeardownLogic } from './types';
|
import { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';
|
import { SubjectSubscription } from './SubjectSubscription';
|
import { rxSubscriber as rxSubscriberSymbol } from '../internal/symbol/rxSubscriber';
|
|
/**
|
* @class SubjectSubscriber<T>
|
*/
|
export class SubjectSubscriber<T> extends Subscriber<T> {
|
constructor(protected destination: Subject<T>) {
|
super(destination);
|
}
|
}
|
|
/**
|
* A Subject is a special type of Observable that allows values to be
|
* multicasted to many Observers. Subjects are like EventEmitters.
|
*
|
* Every Subject is an Observable and an Observer. You can subscribe to a
|
* Subject, and you can call next to feed values as well as error and complete.
|
*
|
* @class Subject<T>
|
*/
|
export class Subject<T> extends Observable<T> implements SubscriptionLike {
|
|
[rxSubscriberSymbol]() {
|
return new SubjectSubscriber(this);
|
}
|
|
observers: Observer<T>[] = [];
|
|
closed = false;
|
|
isStopped = false;
|
|
hasError = false;
|
|
thrownError: any = null;
|
|
constructor() {
|
super();
|
}
|
|
/**@nocollapse
|
* @deprecated use new Subject() instead
|
*/
|
static create: Function = <T>(destination: Observer<T>, source: Observable<T>): AnonymousSubject<T> => {
|
return new AnonymousSubject<T>(destination, source);
|
}
|
|
lift<R>(operator: Operator<T, R>): Observable<R> {
|
const subject = new AnonymousSubject(this, this);
|
subject.operator = <any>operator;
|
return <any>subject;
|
}
|
|
next(value?: T) {
|
if (this.closed) {
|
throw new ObjectUnsubscribedError();
|
}
|
if (!this.isStopped) {
|
const { observers } = this;
|
const len = observers.length;
|
const copy = observers.slice();
|
for (let i = 0; i < len; i++) {
|
copy[i].next(value);
|
}
|
}
|
}
|
|
error(err: any) {
|
if (this.closed) {
|
throw new ObjectUnsubscribedError();
|
}
|
this.hasError = true;
|
this.thrownError = err;
|
this.isStopped = true;
|
const { observers } = this;
|
const len = observers.length;
|
const copy = observers.slice();
|
for (let i = 0; i < len; i++) {
|
copy[i].error(err);
|
}
|
this.observers.length = 0;
|
}
|
|
complete() {
|
if (this.closed) {
|
throw new ObjectUnsubscribedError();
|
}
|
this.isStopped = true;
|
const { observers } = this;
|
const len = observers.length;
|
const copy = observers.slice();
|
for (let i = 0; i < len; i++) {
|
copy[i].complete();
|
}
|
this.observers.length = 0;
|
}
|
|
unsubscribe() {
|
this.isStopped = true;
|
this.closed = true;
|
this.observers = null;
|
}
|
|
/** @deprecated This is an internal implementation detail, do not use. */
|
_trySubscribe(subscriber: Subscriber<T>): TeardownLogic {
|
if (this.closed) {
|
throw new ObjectUnsubscribedError();
|
} else {
|
return super._trySubscribe(subscriber);
|
}
|
}
|
|
/** @deprecated This is an internal implementation detail, do not use. */
|
_subscribe(subscriber: Subscriber<T>): Subscription {
|
if (this.closed) {
|
throw new ObjectUnsubscribedError();
|
} else if (this.hasError) {
|
subscriber.error(this.thrownError);
|
return Subscription.EMPTY;
|
} else if (this.isStopped) {
|
subscriber.complete();
|
return Subscription.EMPTY;
|
} else {
|
this.observers.push(subscriber);
|
return new SubjectSubscription(this, subscriber);
|
}
|
}
|
|
/**
|
* Creates a new Observable with this Subject as the source. You can do this
|
* to create customize Observer-side logic of the Subject and conceal it from
|
* code that uses the Observable.
|
* @return {Observable} Observable that the Subject casts to
|
*/
|
asObservable(): Observable<T> {
|
const observable = new Observable<T>();
|
(<any>observable).source = this;
|
return observable;
|
}
|
}
|
|
/**
|
* @class AnonymousSubject<T>
|
*/
|
export class AnonymousSubject<T> extends Subject<T> {
|
constructor(protected destination?: Observer<T>, source?: Observable<T>) {
|
super();
|
this.source = source;
|
}
|
|
next(value: T) {
|
const { destination } = this;
|
if (destination && destination.next) {
|
destination.next(value);
|
}
|
}
|
|
error(err: any) {
|
const { destination } = this;
|
if (destination && destination.error) {
|
this.destination.error(err);
|
}
|
}
|
|
complete() {
|
const { destination } = this;
|
if (destination && destination.complete) {
|
this.destination.complete();
|
}
|
}
|
|
/** @deprecated This is an internal implementation detail, do not use. */
|
_subscribe(subscriber: Subscriber<T>): Subscription {
|
const { source } = this;
|
if (source) {
|
return this.source.subscribe(subscriber);
|
} else {
|
return Subscription.EMPTY;
|
}
|
}
|
}
|