From ea0f76a89b0fb28c9700cdebb10569256afd1742 Mon Sep 17 00:00:00 2001 From: Mateusz Podlasin <Podlas29@gmail.com> Date: Sun, 26 Feb 2017 19:05:10 +0100 Subject: [PATCH] docs(generate): write documentation for the operator Describe what is generate, its input and output, by comparing it to regular for loop. Describe two modes it can be used in and give examples for each of these modes. Emphasize which arguments are optional in each mode. Describe what a scheduler does when passed to the operator. --- src/observable/GenerateObservable.ts | 194 ++++++++++++++++----------- 1 file changed, 115 insertions(+), 79 deletions(-) diff --git a/src/observable/GenerateObservable.ts b/src/observable/GenerateObservable.ts index 3db21cb8a8..b50cd1a974 100644 --- a/src/observable/GenerateObservable.ts +++ b/src/observable/GenerateObservable.ts @@ -63,112 +63,148 @@ export class GenerateObservable<T, S> extends Observable<T> { super(); } - /** - * Generates an observable sequence by running a state-driven loop - * producing the sequence's elements, using the specified scheduler - * to send out observer messages. - * - * <img src="./img/generate.png" width="100%"> - * - * @example <caption>Produces sequence of 0, 1, 2, ... 9, then completes.</caption> - * var res = Rx.Observable.generate(0, x => x < 10, x => x + 1, x => x); - * - * @example <caption>Using asap scheduler, produces sequence of 2, 3, 5, then completes.</caption> - * var res = Rx.Observable.generate(1, x => x < 5, x => x * 2, x => x + 1, Rx.Scheduler.asap); - * - * @see {@link from} - * @see {@link create} - * - * @param {S} initialState Initial state. - * @param {function (state: S): boolean} condition Condition to terminate generation (upon returning false). - * @param {function (state: S): S} iterate Iteration step function. - * @param {function (state: S): T} resultSelector Selector function for results produced in the sequence. - * @param {Scheduler} [scheduler] A {@link IScheduler} on which to run the generator loop. If not provided, defaults to emit immediately. - * @returns {Observable<T>} The generated sequence. - */ static create<T, S>(initialState: S, condition: ConditionFunc<S>, iterate: IterateFunc<S>, resultSelector: ResultFunc<S, T>, scheduler?: IScheduler): Observable<T> + static create<S>(initialState: S, + condition: ConditionFunc<S>, + iterate: IterateFunc<S>, + scheduler?: IScheduler): Observable<S> + static create<S>(options: GenerateBaseOptions<S>): Observable<S> + static create<T, S>(options: GenerateOptions<T, S>): Observable<T> /** - * Generates an observable sequence by running a state-driven loop - * producing the sequence's elements, using the specified scheduler - * to send out observer messages. - * The overload uses state as an emitted value. + * Generates an Observable by running a state-driven loop + * that emits an element on each iteration. + * + * <span class="informal">Use it instead of nexting values in a for loop.</span> * * <img src="./img/generate.png" width="100%"> * - * @example <caption>Produces sequence of 0, 1, 2, ... 9, then completes.</caption> - * var res = Rx.Observable.generate(0, x => x < 10, x => x + 1); + * `generate` allows you to create stream of values generated with a loop very similar to + * traditional for loop. First argument of `generate` is a beginning value. Second argument + * is a function that accepts this value and tests if some condition still holds. If it does, + * loop continues, if not, it stops. Third value is a function which takes previously defined + * value and modifies it in some way on each iteration. Note how these three parameters + * are direct equivalents of three expressions in regular for loop: first expression + * initializes some state (for example numeric index), second tests if loop can make next + * iteration (for example if index is lower than 10) and third states how defined value + * will be modified on every step (index will be incremented by one). * - * @example <caption>Using asap scheduler, produces sequence of 1, 2, 4, then completes.</caption> - * var res = Rx.Observable.generate(1, x => x < 5, x => x * 2, Rx.Scheduler.asap); + * Return value of a `generate` operator is an Observable that on each loop iteration + * emits a value. First, condition function is ran. If it returned true, Observable + * emits currently stored value (initial value at the first iteration) and then updates + * that value with iterate function. If at some point condition returned false, Observable + * completes at that moment. * - * @see {@link from} - * @see {@link create} + * Optionally you can pass fourth parameter to `generate` - a result selector function which allows you + * to immediately map value that would normally be emitted by an Observable. * - * @param {S} initialState Initial state. - * @param {function (state: S): boolean} condition Condition to terminate generation (upon returning false). - * @param {function (state: S): S} iterate Iteration step function. - * @param {Scheduler} [scheduler] A {@link IScheduler} on which to run the generator loop. If not provided, defaults to emit immediately. - * @returns {Observable<S>} The generated sequence. - */ - static create<S>(initialState: S, - condition: ConditionFunc<S>, - iterate: IterateFunc<S>, - scheduler?: IScheduler): Observable<S> - - /** - * Generates an observable sequence by running a state-driven loop - * producing the sequence's elements, using the specified scheduler - * to send out observer messages. - * The overload accepts options object that might contain initial state, iterate, - * condition and scheduler. + * If you find three anonymous functions in `generate` call hard to read, you can provide + * single object to the operator instead. That object has properties: `initialState`, + * `condition`, `iterate` and `resultSelector`, which should have respective values that you + * would normally pass to `generate`. `resultSelector` is still optional, but that form + * of calling `generate` allows you to omit `condition` as well. If you omit it, that means + * condition always holds, so output Observable will never complete. * - * <img src="./img/generate.png" width="100%"> + * Both forms of `generate` can optionally accept a scheduler. In case of multi-parameter call, + * scheduler simply comes as a last argument (no matter if there is resultSelector + * function or not). In case of single-parameter call, you can provide it as a + * `scheduler` property on object passed to the operator. In both cases scheduler decides when + * next iteration of the loop will happen and therefore when next value will be emitted + * by the Observable. For example to ensure that each value is pushed to the observer + * on separate task in event loop, you could use `async` scheduler. Note that + * by default (when no scheduler is passed) values are simply emitted synchronously. + * + * + * @example <caption>Use with condition and iterate functions.</caption> + * const generated = Rx.Observable.generate(0, x => x < 3, x => x + 1); + * + * generated.subscribe( + * value => console.log(value), + * err => {}, + * () => console.log('Yo!') + * ); + * + * // Logs: + * // 0 + * // 1 + * // 2 + * // "Yo!" * - * @example <caption>Produces sequence of 0, 1, 2, ... 9, then completes.</caption> - * var res = Rx.Observable.generate({ + * + * @example <caption>Use with condition, iterate and resultSelector functions.</caption> + * const generated = Rx.Observable.generate(0, x => x < 3, x => x + 1, x => x * 1000); + * + * generated.subscribe( + * value => console.log(value), + * err => {}, + * () => console.log('Yo!') + * ); + * + * // Logs: + * // 0 + * // 1000 + * // 2000 + * // "Yo!" + * + * + * @example <caption>Use with options object.</caption> + * const generated = Rx.Observable.generate({ * initialState: 0, - * condition: x => x < 10, - * iterate: x => x + 1 + * condition(value) { return value < 3; }, + * iterate(value) { return value + 1; }, + * resultSelector(value) { return value * 1000; } * }); * - * @see {@link from} - * @see {@link create} + * generated.subscribe( + * value => console.log(value), + * err => {}, + * () => console.log('Yo!') + * ); * - * @param {GenerateBaseOptions<S>} options Object that must contain initialState, iterate and might contain condition and scheduler. - * @returns {Observable<S>} The generated sequence. - */ - static create<S>(options: GenerateBaseOptions<S>): Observable<S> - - /** - * Generates an observable sequence by running a state-driven loop - * producing the sequence's elements, using the specified scheduler - * to send out observer messages. - * The overload accepts options object that might contain initial state, iterate, - * condition, result selector and scheduler. + * // Logs: + * // 0 + * // 1000 + * // 2000 + * // "Yo!" * - * <img src="./img/generate.png" width="100%"> - * - * @example <caption>Produces sequence of 0, 1, 2, ... 9, then completes.</caption> - * var res = Rx.Observable.generate({ + * @example <caption>Use options object without condition function.</caption> + * const generated = Rx.Observable.generate({ * initialState: 0, - * condition: x => x < 10, - * iterate: x => x + 1, - * resultSelector: x => x + * iterate(value) { return value + 1; }, + * resultSelector(value) { return value * 1000; } * }); * + * generated.subscribe( + * value => console.log(value), + * err => {}, + * () => console.log('Yo!') // This will never run. + * ); + * + * // Logs: + * // 0 + * // 1000 + * // 2000 + * // 3000 + * // ...and never stops. + * + * * @see {@link from} * @see {@link create} * - * @param {GenerateOptions<T, S>} options Object that must contain initialState, iterate, resultSelector and might contain condition and scheduler. - * @returns {Observable<T>} The generated sequence. + * @param {S} initialState Initial state. + * @param {function (state: S): boolean} condition Condition to terminate generation (upon returning false). + * @param {function (state: S): S} iterate Iteration step function. + * @param {function (state: S): T} [resultSelector] Selector function for results produced in the sequence. + * @param {Scheduler} [scheduler] A {@link Scheduler} on which to run the generator loop. If not provided, defaults to emitting immediately. + * @return {Observable<T>} The generated sequence. + * @static true + * @name generate + * @owner Observable */ - static create<T, S>(options: GenerateOptions<T, S>): Observable<T> - static create<T, S>(initialStateOrOptions: S | GenerateOptions<T, S>, condition?: ConditionFunc<S>, iterate?: IterateFunc<S>,