Angular @ViewChild() error: Expected 2 arguments, but got 1

AngularTypescriptViewchild

Angular Problem Overview


When trying ViewChild I am getting the error. Error is "An argument for 'opts' was not provided."

Both @ViewChild is giving the error.

import { Component, OnInit, ElementRef, ViewChild, Output, EventEmitter } from '@angular/core';
import { Ingredient } from 'src/app/shared/ingredient.model';

@Component({
  selector: 'app-shopping-edit',
  templateUrl: './shopping-edit.component.html',
  styleUrls: ['./shopping-edit.component.css']
})
export class ShoppingEditComponent implements OnInit {

@ViewChild('nameInput') nameInputRef: ElementRef;
@ViewChild('amountInput') amountInputRef: ElementRef;
@Output() ingredientAdded = new EventEmitter<Ingredient>();
  constructor() {}

  ngOnInit() {
  }

  onAddItem() {
    const ingName = this.nameInputRef.nativeElement.value;
    const ingAmount = this.amountInputRef.nativeElement.value;
    const newIngredient = new Ingredient(ingName, ingAmount);
    this.ingredientAdded.emit(newIngredient);
  }

}

>ts(11,2): error TS2554: Expected 2 arguments, but got 1.

Angular Solutions


Solution 1 - Angular

In Angular 8 , ViewChild takes 2 parameters

 @ViewChild(ChildDirective, {static: false}) Component

Solution 2 - Angular

Angular 8

In Angular 8, ViewChild has another param

@ViewChild('nameInput', {static: false}) component : Component

You can read more about it here and here

Angular 9 & Angular 10

In Angular 9 default value is static: false, so doesn't need to provide param unless you want to use {static: true}

Solution 3 - Angular

In Angular 8 , ViewChild takes 2 parameters:

Try like this:

@ViewChild('nameInput', { static: false }) nameInputRef: ElementRef;

Explanation:

>{ static: false }

If you set static false, the child component ALWAYS gets initialized after the view initialization in time for the ngAfterViewInit/ngAfterContentInit callback functions.

>{ static: true}

If you set static true, the child component initialization will take place at the view initialization at ngOnInit

By default you can use { static: false }. If you are creating a dynamic view and want to use the template reference variable, then you should use { static: true}

For more info, you can read this article

> Working Demo

In the demo, we will scroll to a div using template reference variable.

 @ViewChild("scrollDiv", { static: true }) scrollTo: ElementRef;

With { static: true }, we can use this.scrollTo.nativeElement in ngOnInit, but with { static: false }, this.scrollTo will be undefined in ngOnInit , so we can access in only in ngAfterViewInit

Solution 4 - Angular

it is because view child require two argument try like this

> @ViewChild('nameInput', { static: false, }) nameInputRef: ElementRef; > > @ViewChild('amountInput', { static: false, }) amountInputRef: > ElementRef;

Solution 5 - Angular

In Angular 8, ViewChild always takes 2 param, and second params always has static: true or static: false

You can try like this:

@ViewChild('nameInput', {static: false}) component

Also,the static: false is going to be the default fallback behaviour in Angular 9.

What are static false/true: So as a rule of thumb you can go for the following:

  • { static: true } needs to be set when you want to access the ViewChild in ngOnInit.

    { static: false } can only be accessed in ngAfterViewInit. This is also what you want to go for when you have a structural directive (i.e. *ngIf) on your element in your template.

Solution 6 - Angular

Try this in angular 8.0:

@ViewChild('result',{static: false}) resultElement: ElementRef;

Solution 7 - Angular

Regex for replacing all via IDEA (tested with Webstorm)

Find: \@ViewChild\('(.*)'\)

Replace: \@ViewChild\('$1', \{static: true\}\)

Solution 8 - Angular

you should use second argument with ViewChild like this:

@ViewChild("eleDiv", { static: false }) someElement: ElementRef;

Solution 9 - Angular

Use this

> @ViewChild(ChildDirective, {static: false}) Component

Solution 10 - Angular

In Angular 8, ViewChild has another param

@ViewChild('nameInput', {static: false}) component

I resolved my issue like below

@ViewChild(MatSort, {static: false}) sort: MatSort;

Solution 11 - Angular

That also resolved my issue.

@ViewChild('map', {static: false}) googleMap;

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
Questionstack overflowView Question on Stackoverflow
Solution 1 - AngularJensenView Answer on Stackoverflow
Solution 2 - AngularRezaView Answer on Stackoverflow
Solution 3 - AngularAdrita SharmaView Answer on Stackoverflow
Solution 4 - Angularparas shahView Answer on Stackoverflow
Solution 5 - AngularPinki DhakadView Answer on Stackoverflow
Solution 6 - AngularSoNuView Answer on Stackoverflow
Solution 7 - AngularTorsten SimonView Answer on Stackoverflow
Solution 8 - AngularHammad AhmadView Answer on Stackoverflow
Solution 9 - AngularDhilrukshan PradeepView Answer on Stackoverflow
Solution 10 - AngularManoj TyagiView Answer on Stackoverflow
Solution 11 - AngularHelmar BachleView Answer on Stackoverflow