How should I use the new static option for @ViewChild in Angular 8?
In most cases you will want to use {static: false}
. Setting it like this will ensure query matches that are dependent on binding resolution (like structural directives *ngIf, etc...
) will be found.
Example of when to use static: false
:
@Component({
template: `
<div *ngIf="showMe" #viewMe>Am I here?</div>
<button (click)="showMe = !showMe"></button>
`
})
export class ExampleComponent {
@ViewChild('viewMe', { static: false })
viewMe?: ElementRef<HTMLElement>;
showMe = false;
}
The static: false
is going to be the default fallback behaviour in Angular 9. Read more here and here
The { static: true }
option was introduced to support creating embedded views on the fly. When you are creating a view dynamically and want to acces the TemplateRef
, you won't be able to do so in ngAfterViewInit
as it will cause a ExpressionHasChangedAfterChecked
error. Setting the static flag to true will create your view in ngOnInit.
Nevertheless:
In most other cases, the best practice is to use
{static: false}
.
Be aware though that the { static: false }
option will be made default in Angular 9. Which means that setting the static flag is no longer necessary, unless you want to use the static: true
option.
You can use the angular cli ng update
command to automatically upgrade your current code base.
For a migration guide and even more information about this, you can check here and here
#What is the difference between static and dynamic queries? The static option for @ViewChild() and @ContentChild() queries determines when the query results become available.
With static queries (static: true), the query resolves once the view has been created, but before change detection runs. The result, though, will never be updated to reflect changes to your view, such as changes to ngIf and ngFor blocks.
With dynamic queries (static: false), the query resolves after either ngAfterViewInit() or ngAfterContentInit() for @ViewChild() and @ContentChild() respectively. The result will be updated for changes to your view, such as changes to ngIf and ngFor blocks.
A nice use-case for using static: true
, is if you are using fromEvent
to bind to an element defined in the template. Consider the following template:
<div [ngStyle]="thumbStyle$ | async" #thumb></div>
You can then handle events on this element without the need of using subscriptions or init hooks (if you don't want to or cannot use angular event binding):
@Component({})
export class ThumbComponent {
@ViewChild('thumb', { static: true })
thumb?: ElementRef<HTMLElement>;
readonly thumbStyle$ = defer(() => fromEvent(this.thumb, 'pointerdown').pipe(
switchMap((startEvent) => fromEvent(document, 'pointermove', { passive: true })
// transform to proper positioning
));
}
It is important to use defer
. This will make sure the observable is only resolved when it's subscribed to. This will happen before the ngAfterViewInit
gets triggered, when the async
pipe subscribes to it. Because we are using static: true
, the this.thumb
is already populated.
static property informs angular about the availability of our child
For example: if static is set to true, we are informing angular our child is available on the page from the beginning ( meaning it is not dependent of *ngIf,page bindings,API calls etc) so angular looks for it at the earliest lifecycle hook ( ngOnInit) and never looks for it again
In case we set static to false we are informing angular that our child is dependent on some conditional directives, so angular tries to look for our child after every change detection cycle and if it is available we can access it in ngAfterViewInit() life cycle hook
So as a rule of thumb you can go for the following:
{ static: true }
needs to be set when you want to access theViewChild
inngOnInit
.{ static: false }
can only be accessed inngAfterViewInit
. This is also what you want to go for when you have a structural directive (i.e.*ngIf
) on your element in your template.
From the angular docs
static - whether or not to resolve query results before change detection runs (i.e. return static results only). If this option is not provided, the compiler will fall back to its default behavior, which is to use query results to determine the timing of query resolution. If any query results are inside a nested view (e.g. *ngIf), the query will be resolved after change detection runs. Otherwise, it will be resolved before change detection runs.
It may be a better idea to use static:true
if the child does not depend on any conditions. If the visibility of element changes, then static:false
may give better results.
PS: Since its a new feature, we may need to run benchmarks for performance.
Edit
As mentioned by @Massimiliano Sartoretto , github commit may give you more insights.