@lcsga/ng-operators
is available for download at npm.
npm install @lcsga/ng-operators
To work as expected, this library needs the NgZone
configuration to enable the event coalescing:
export const appConfig: ApplicationConfig = {
providers: [
// ...
provideZoneChangeDetection({ eventCoalescing: true, runCoalescing: true }),
],
};
// You can also activate the runCoalescing option alongside with the OnPush detection strategy to provide better performances of your apps
This operator is usefull whenever you want to listen to some @ViewChild
or @ContentChild
events.
fromChildEvent<T extends Event>(
childSelector: () => ElementRef | undefined,
type: keyof HTMLElementEventMap,
options?: NgEventListenerOptions
): Observable<T>
argument | type | description |
---|---|---|
childSelector |
() => ElementRef | undefined |
A callback function used to get the child element to listen an event to. |
type |
keyof HTMLElementEventMap |
The type of event to listen. |
options |
NgEventListenerOptions |
Optional. Default is {} .Options to pass through to the underlying addEventListener function or if you need to manualy provide the injector |
Why do we need it?
Currently, when you want to avoid using the Angular's @HostListner
or the (click)="doSomething()"
, you can create a Subject
and use it like this:
@Component({
selector: "app",
template: "<button (click)="buttonClick$$.next()">Click me!</button>"
})
export class AppComponent {
protected readonly buttonClick$$ = new Subject<void>();
private readonly onButtonClick$ = this.buttonClick$$.pipe(
tap(() => console.log("hello world!"))
);
constructor() {
this.onButtonClick$.subscribe();
}
}
It actually works pretty well but since we need to specifically call the next()
method of the buttonClick$$
subject, it's not fully declarative.
To make it declarative, we would instead need to use the fromEvent
operator from RxJS but we can't do that nicely because it takes an element from the dom.
Indeed to get such an element in Angular, depending on what you've built, you can either use @ViewChild
or @ContentChild
@Component({
selector: 'app',
template: '<button #button>Click me!</button>',
})
export class AppComponent {
@ViewChild('button')
private readonly button?: ElementRef<HTMLButtonElement>;
private readonly onButtonClick$ = fromEvent(this.button?.nativeElement, 'click').pipe(
// throws an error!
tap(() => console.log('hello world!'))
);
constructor() {
this.onButtonClick$.subscribe();
}
}
The issue with the code above is that the button
element is undefined
until the dom is rendered. Thus the Cannot read properties of undefined (reading 'addEventListener')
error is thrown.
A solution to make it work would be to assign the stream of onButtonClick$
within the afterViewInit()
method but the best part of declarative is to write the assigning right at the declaration, so it wouldn't be prefect.
Here comes the fromChildEvent
custom operator, to the rescue!
It works by listening the event of your choice directly on the document
and check if the event's target
is the same as a viewChild or a contentChild you'd pass to it
@Component({
selector: 'app',
template: '<button #button>Click me!</button>',
})
export class AppComponent {
@ViewChild('button')
private readonly button?: ElementRef<HTMLButtonElement>;
private readonly onButtonClick$ = fromChildEvent(() => this.button, 'click').pipe(tap(() => console.log('hello world!')));
constructor() {
this.onButtonClick$.subscribe();
}
}
As you can see, fromChildEvent
takes a selector callback to get the viewChild or contentChild target.
Since the document's event can only be fired after the dom is rendered, we know that the element passed within the selector callback is always available.
It works exactly like fromChildEvent
but with @ViewChildren
or @ContentChildren
instead!
fromChildrenEvent<T extends Event>(
childrenSelector: () => ElementRef[] | undefined,
type: keyof HTMLElementEventMap,
options?: NgEventListenerOptions
): Observable<readonly [event: T, index: number]>
argument | type | description |
---|---|---|
childrenSelector |
() => ElementRef[] | undefined |
A callback function used to get the children elements to listen an event to. |
type |
keyof HTMLElementEventMap |
The type of event to listen. |
options |
NgEventListenerOptions |
Optional. Default is {} .Options to pass through to the underlying addEventListener function or if you need to manualy provide the injector |
<br/>
@Component({
selector: 'app',
template: `
<button #button>Click me!</button>
<button #button>Click me!</button>
<button #button>Click me!</button>
`,
})
export class AppComponent {
@ViewChildren('button')
private readonly buttons?: QueryList<ElementRef<HTMLButtonElement>>;
private readonly onButtonsClick$ = fromChildrenEvent(() => this.buttons?.toArray(), 'click').pipe(tap(() => console.log('hello world!')));
constructor() {
this.onButtonsClick$.subscribe();
}
}
This operator is usefull as an Rx replacement for @HostListner
.
fromHostEvent<T extends Event>(type: keyof HTMLElementEventMap, options?: NgEventListenerOptions): Observable<T>
argument | type | description |
---|---|---|
type |
keyof HTMLElementEventMap |
The type of event to listen. |
options |
NgEventListenerOptions |
Optional. Default is {} .Options to pass through to the underlying addEventListener function or if you need to manualy provide the injector |
@Component({
selector: 'app-root',
template: '',
})
export class AppComponent {
constructor() {
fromHostEvent('click').subscribe(() => console.log('hello world!'));
// on app-root click => Output: hello world!
}
}
This operator is usefull whenever you want to listen to some @ViewChild
or @ContentChild
outputs or observables for a Component
or a Directive
instead of an ElementRef
.
fromChildOutput<TChild extends object, TOutput extends PickOutput<TChild>, TOutputName extends keyof TOutput>(
childSelector: () => TChild | undefined,
outputName: TOutputName,
options?: InjectorOption
): Observable<TOutput[TOutputName]>
See PickOutput utility type
argument | type | description |
---|---|---|
childSelector |
() => TChild | undefined |
A callback function used to get the child element to listen an event to. |
outputName |
TOutputName |
The name of the public observable to listen. |
options |
InjectorOption |
Optional. Default is {} .Options to pass through if you need to manualy provide the injector |
<br/>
@Component({
selector: 'app-child',
template: `<button (click)="sayHello.emit('hello!')">Say hello</button>`,
})
class AppChildComponent {
// @Output is not necessary here, if you don't use the angular (eventName)="doSomething()" syntax
// 'sayHello' must be public to be accessed
sayHello = new EventEmitter<string>();
}
@Component({
selector: 'app-root',
imports: [AppChildComponent],
template: '<app-child #child />',
})
export class AppComponent {
@ViewChild('child') child!: AppChildComponent;
constructor() {
// the second argument is infered from the child as the childSelector
fromChildOutput(() => this.child, 'sayHello').subscribe(console.log);
// on child button click => Output: hello!
}
}
It works exactly like fromChildOutput
but for an array of components instead!
fromChildrenOutput<
TChildren extends object[],
TUnionOutput extends PickUnionOutput<TChildren>,
TUnionOutputName extends keyof TUnionOutput
>(
childrenSelector: () => TChildren | undefined,
outputName: TUnionOutputName,
options?: InjectorOption
): Observable<readonly [output: TUnionOutput[TUnionOutputName], index: number]>
See PickOutput utility type
argument | type | description |
---|---|---|
childrenSelector |
() => TChildren | undefined |
A callback function used to get the child element to listen an event to. |
outputName |
TOutputName |
The name of the public observable to listen. |
options |
InjectorOption |
Optional. Default is {} .Options to pass through if you need to manualy provide the injector |
@Component({
selector: 'app-child',
template: `<button (click)="say.emit('hello!')">Say hello</button>`,
})
class AppChildComponent {
// @Output is not necessary here, if you don't use the angular (eventName)="doSomething()" syntax
// 'sayHello' must be public to be accessed
say = new EventEmitter<string>();
}
@Component({
selector: 'app-child2',
template: `<button (click)="say.emit('goodbye!')">Say goodbye</button>`,
})
class AppChild2Component {
say = new EventEmitter<string>();
say$ = this.say.asObservablee(); // This won't be available as the second arguement `outputName` of `fromChildrenOutput` since it does not exist on `AppChildComponent`
}
@Component({
selector: 'app-root',
imports: [AppChildComponent, AppChild2Component],
template: `
<app-child #child />
<app-child2 #child2 />
`,
})
export class AppComponent {
@ViewChild('child') child!: AppChildComponent;
// Since it takes an array of components, you can rebuild it in your own way
@ViewChild('child2') child2!: AppChild2Component;
constructor() {
// The second argument is infered from the child as the childSelector
fromChildrenOutput(() => [this.child, this.child2], 'say').subscribe(console.log);
// On child button click => Output: ["hello!", 0]
}
}
It uses the new afterNextRender
to send an RxJS notification that the callback function has been called once.
rxAfterNextRender(injector?: Injector): Observable<void>
argument | type | description |
---|---|---|
injector |
Injector |
Optional. Default is undefined . |
@Component({
selector: 'app',
template: '<button #button>Click me!</button>',
})
export class AppComponent {
@ViewChild('button')
private readonly button?: ElementRef<HTMLButtonElement>;
constructor() {
rxAfterNextRender().subscribe(() => console.log(this.button?.clientHeight)); // the button won't be undefined here
}
}
It uses the new afterRender
to send RxJS notifications each time the callback function is called.
rxAfterRender(injector?: Injector): Observable<void>
argument | type | description |
---|---|---|
injector |
Injector |
Optional. Default is undefined . |
@Component({
selector: 'app',
template: '<button #button>Click me!</button>',
})
export class AppComponent {
@ViewChild('button')
private readonly button?: ElementRef<HTMLButtonElement>;
constructor() {
rxAfterRender().subscribe(() => console.log(this.button?.clientHeight)); // Will log the button's clientHeight each time the view is checked by angular
}
}
With the upcomming Signal-based Components, we shouldn't need to first declare @ViewChild
, @ViewChildren
, etc. anymore.-ml-1
This would greatly improve the DX of these operators and it could lead to the following improvements:
@Component({
selector: 'app',
template: '<button #button>Click me!</button>',
})
export class AppComponent {
readonly onButtonClick$ = fromViewChildEvent('button', 'click').pipe(...)
}
@Component({
selector: 'app',
template: '<button mat-button>Click me!</button>',
})
export class AppComponent {
readonly onButtonClick$ = fromViewChildEvent(MatButton, 'click').pipe(...)
}
@Component({
selector: 'app',
template: `
<button #button1>Click me!</button>
<button #button2>Click me too!</button>
`,
})
export class AppComponent {
readonly onButtonsClick$ = fromViewChildrenEvent(['button1', 'button2'], 'click').pipe(...)
// of course, the 2 buttons could simply be named #button but it is for the example only
}
@Component({
selector: 'app',
template: `
<button mat-button>Click me!</button>
<button mat-button>Click me too!</button>
`,
})
export class AppComponent {
readonly onButtonClick$ = fromViewChildrenEvent(MatButton, 'click').pipe(...)
// or could be fromViewChildren([MatButton, SomeOtherComponentOrDirective]).pipe(...);
}
With SBCs, I could either decouple the
viewChild<ren>
andcontentChild<ren>
or try to merge them together.
=> Another thing I might improve is merging fromChild<ren>Event
and fromChild<ren>Output
into one operator.