bindCallback
function
stable
Converts a callback API to a function that returns an Observable.
bindCallback(callbackFunc: (args_0: any, args_1: (...res: any) => void) => void, resultSelector?: SchedulerLike | ((...args: any[]) => any), scheduler?: SchedulerLike): (...args: any[]) => Observable<unknown>
Parameters
callbackFunc | Type: |
resultSelector | Optional. Default is Type: |
scheduler | Optional. Default is The scheduler on which to schedule the callbacks. |
Returns
(...args: any[]) => Observable<unknown>
: A function which returns the Observable that delivers the same values the callback would deliver.
Description
Give it a function f
of type f(x, callback)
and it will return a function g
that when called as g(x)
will output an Observable.
bindCallback
is not an operator because its input and output are not Observables. The input is a function func
with some parameters. The last parameter must be a callback function that func
calls when it is done.
The output of bindCallback
is a function that takes the same parameters as func
, except the last one (the callback). When the output function is called with arguments it will return an Observable. If function func
calls its callback with one argument, the Observable will emit that value. If on the other hand the callback is called with multiple values the resulting Observable will emit an array with said values as arguments.
It is very important to remember that input function func
is not called when the output function is, but rather when the Observable returned by the output function is subscribed. This means if func
makes an AJAX request, that request will be made every time someone subscribes to the resulting Observable, but not before.
The last optional parameter - scheduler
- can be used to control when the call to func
happens after someone subscribes to Observable, as well as when results passed to callback will be emitted. By default, the subscription to an Observable calls func
synchronously, but using asyncScheduler
as the last parameter will defer the call to func
, just like wrapping the call in setTimeout
with a timeout of 0
would. If you were to use the async Scheduler and call subscribe
on the output Observable, all function calls that are currently executing will end before func
is invoked.
By default, results passed to the callback are emitted immediately after func
invokes the callback. In particular, if the callback is called synchronously, then the subscription of the resulting Observable will call the next
function synchronously as well. If you want to defer that call, you may use asyncScheduler
just as before. This means that by using Scheduler.async
you can ensure that func
always calls its callback asynchronously, thus avoiding terrifying Zalgo.
Note that the Observable created by the output function will always emit a single value and then complete immediately. If func
calls the callback multiple times, values from subsequent calls will not appear in the stream. If you need to listen for multiple calls, you probably want to use fromEvent
or fromEventPattern
instead.
If func
depends on some context (this
property) and is not already bound, the context of func
will be the context that the output function has at call time. In particular, if func
is called as a method of some objec and if func
is not already bound, in order to preserve the context it is recommended that the context of the output function is set to that object as well.
If the input function calls its callback in the "node style" (i.e. first argument to callback is optional error parameter signaling whether the call failed or not), bindNodeCallback
provides convenient error handling and probably is a better choice. bindCallback
will treat such functions the same as any other and error parameters (whether passed or not) will always be interpreted as regular callback argument.
Examples
Convert jQuery's getJSON to an Observable API
import { bindCallback } from 'rxjs'; import * as jQuery from 'jquery'; // Suppose we have jQuery.getJSON('/my/url', callback) const getJSONAsObservable = bindCallback(jQuery.getJSON); const result = getJSONAsObservable('/my/url'); result.subscribe(x => console.log(x), e => console.error(e));
Receive an array of arguments passed to a callback
import { bindCallback } from 'rxjs'; const someFunction = (cb) => { cb(5, 'some string', {someProperty: 'someValue'}) }; const boundSomeFunction = bindCallback(someFunction); boundSomeFunction(12, 10).subscribe(values => { console.log(values); // [22, 2] });
Compare behaviour with and without async Scheduler
import { bindCallback, asyncScheduler } from 'rxjs'; function iCallMyCallbackSynchronously(cb) { cb(); } const boundSyncFn = bindCallback(iCallMyCallbackSynchronously); const boundAsyncFn = bindCallback(iCallMyCallbackSynchronously, null, asyncScheduler); boundSyncFn().subscribe(() => console.log('I was sync!')); boundAsyncFn().subscribe(() => console.log('I was async!')); console.log('This happened...'); // Logs: // I was sync! // This happened... // I was async!
Use bindCallback on an object method
import { bindCallback } from 'rxjs'; const boundMethod = bindCallback(someObject.methodWithCallback); boundMethod .call(someObject) // make sure methodWithCallback has access to someObject .subscribe(subscriber);
See Also
© 2015–2021 Google, Inc., Netflix, Inc., Microsoft Corp. and contributors.
Code licensed under an Apache-2.0 License. Documentation licensed under CC BY 4.0.
https://rxjs.dev/api/index/function/bindCallback