#local variable внутри *ngIf

На локальную переменную шаблона (#myVar) можно ссылаться только внутри блока шаблона (<template>). В случае когда необходимо сослаться извне, ссылка будет пустой. Это немного неочевидно при использовании структурных директив, таких как *ngIf.s

<div *ngIf="true">
  <p class="inside" #p> Feature </p>
  <p>Class: "{{ p?.className }}"</p> <!-- Вот тут работает -->
</div>

<p>Class: "{{ p?.className }}"</p>   <!-- А здесь не работает -->

Есть 2 варианта решения

@ViewChild

Используетмя декоратор @ViewChild с сеттером. Он будет запускаться каждый раз, когда изменяется ngIf.

export class AppComponent {

  ref: ElementRef; // В шаблоне Class: "{{ ref?.nativeElement.className }}"

  constructor(private cdr: ChangeDetectorRef) {}

  @ViewChild('p') 
  set paragraph(e: ElementRef) {
    this.ref = e;
    // ExpressionChangedAfterItHasBeenCheckedError
    this.cdr.detectChanges();
  }

}

@ViewChildren

Используется декортор @ViewChildren. Он даёт QueryList (а-ля SelectorAll), на который следует подписаться. Список будет обновляться при каждом изменении переменной template (ngIf включается или выключается).

export class AppComponent implements AfterViewInit, OnDestroy {

  ref: ElementRef; // В шаблоне Class: "{{ ref?.nativeElement.className }}"

  constructor(private cdr: ChangeDetectorRef) {}

  @ViewChildren('p') 
  pQueryList: QueryList<ElementRef>;
  /** destroy subject (pattern) */
  private _destroy$ = new Subject<void>();  

  ngAfterViewInit(): void {
    this.processChanges();
    this.pQueryList.changes.pipe(
      takeUntil(this._destroy$)
    ).subscribe(() => this.processChanges());
  }

  processChanges(): void {
    this.ref = this.pQueryList.first;
    this.cdr.detectChanges();   
  }

  ngOnDestroy(): void {
    this._destroy$.next();
    this._destroy$.complete();
  }

}

Demo


Похожие записи

@Attribute() декоратор

Аналогично @Input() позволяет получить значение атрибута с хоста компонента/директивы, но не отслеживает дальнейшее изменение атрибута.

14 сентября 2019 г. в Angular

Angular & MVVM

  • Model - just file like user.class.ts
  • View - HTML template of component
  • ViewModel - Typescript part of a component
14 апреля 2019 г. в Angular

Angular URL Matcher

Функция сопоставления маршрута с URL-адресами. Возможность динамически подбирать компонент для маршрута

04 октября 2020 г. в Angular