664

有人知道如何获取组件模板中定义的元素吗?$Polymer 使用和使其变得非常容易$$

我只是想知道如何在 Angular 中进行操作。

以教程中的示例为例:

import {Component} from '@angular/core';

@Component({
    selector:'display',
    template:`
     <input #myname (input)="updateName(myname.value)"/>
     <p>My name : {{myName}}</p>
     `   
})
export class DisplayComponent {
    myName: string = "Aman";
    updateName(input: String) {
        this.myName = input;
    }
}

如何从类定义中捕获或获取por元素的引用?input

4

14 回答 14

1162

与其从那里进行注入ElementRef和使用querySelector或类似操作,不如使用声明性方式直接访问视图中的元素:

<input #myname>
@ViewChild('myname') input; 

元素

ngAfterViewInit() {
  console.log(this.input.nativeElement.value);
}

StackBlitz 示例

  • @ViewChild()支持指令或组件类型作为参数,或模板变量的名称(字符串)。
  • @ViewChildren()还支持以逗号分隔的名称列表(目前不允许使用空格@ViewChildren('var1,var2,var3'))。
  • @ContentChild()@ContentChildren()做同样的事情,但在光 DOM(<ng-content>投影元素)中。

后人

@ContentChildren()是唯一允许查询后代的

@ContentChildren(SomeTypeOrVarName, {descendants: true}) someField; 

{descendants: true}应该是默认值,但不在 2.0.0 最终版本中,它被认为是一个错误
这已在 2.0.1 中修复

如果存在组件和指令,则该read参数允许指定应返回哪个实例。

例如ViewContainerRef,动态创建的组件而不是默认组件所需要的ElementRef

@ViewChild('myname', { read: ViewContainerRef }) target;

订阅更改

即使视图子级仅在ngAfterViewInit()调用时设置,内容子级仅在ngAfterContentInit()调用时设置,但如果要订阅查询结果的更改,则应在ngOnInit()

https://github.com/angular/angular/issues/9689#issuecomment-229247134

@ViewChildren(SomeType) viewChildren;
@ContentChildren(SomeType) contentChildren;

ngOnInit() {
  this.viewChildren.changes.subscribe(changes => console.log(changes));
  this.contentChildren.changes.subscribe(changes => console.log(changes));
}

直接 DOM 访问

只能查询 DOM 元素,不能查询组件或指令实例:

export class MyComponent {
  constructor(private elRef:ElementRef) {}
  ngAfterViewInit() {
    var div = this.elRef.nativeElement.querySelector('div');
    console.log(div);
  }

  // for transcluded content
  ngAfterContentInit() {
    var div = this.elRef.nativeElement.querySelector('div');
    console.log(div);
  }
}

获取任意投影内容

请参阅访问嵌入的内容

于 2016-02-04T19:17:58.300 回答
218

ElementRef您可以通过将其注入组件的构造函数来获取 DOM 元素的句柄:

constructor(private myElement: ElementRef) { ... }

文档:https ://angular.io/docs/ts/latest/api/core/index/ElementRef-class.html

于 2015-09-21T11:19:06.607 回答
56
import { Component, ElementRef, OnInit } from '@angular/core';

@Component({
  selector:'display',
  template:`
   <input (input)="updateName($event.target.value)">
   <p> My name : {{ myName }}</p>
  `
})
class DisplayComponent implements OnInit {
  constructor(public element: ElementRef) {
    this.element.nativeElement // <- your direct element reference 
  }
  ngOnInit() {
    var el = this.element.nativeElement;
    console.log(el);
  }
  updateName(value) {
    // ...
  }
}

更新示例以使用最新版本

有关本机元素的更多详细信息,请点击此处

于 2015-09-22T06:17:55.120 回答
21

Angular 4+renderer.selectRootElement与 CSS 选择器一起使用来访问元素。

我有一个最初显示电子邮件输入的表单。输入电子邮件后,表单将被展开,以允许他们继续添加与其项目相关的信息。但是,如果他们不是现有客户,表格将在项目信息部分上方包含地址部分。

到目前为止,数据输入部分还没有被分解成组件,所以这些部分是用 *ngIf 指令管理的。如果他们是现有客户,我需要将焦点放在项目注释字段上,如果他们是新客户,我需要将重点放在名字字段上。

我尝试了没有成功的解决方案。但是,答案中的更新 3 给了我最终解决方案的一半。另一半来自 MatteoNY 在这个帖子中的回应。结果是这样的:

import { NgZone, Renderer } from '@angular/core';

constructor(private ngZone: NgZone, private renderer: Renderer) {}

setFocus(selector: string): void {
    this.ngZone.runOutsideAngular(() => {
        setTimeout(() => {
            this.renderer.selectRootElement(selector).focus();
        }, 0);
    });
}

submitEmail(email: string): void {
    // Verify existence of customer
    ...
    if (this.newCustomer) {
        this.setFocus('#firstname');
    } else {
        this.setFocus('#description');
    }
}

因为我唯一要做的就是将焦点放在一个元素上,所以我不需要关心更改检测,所以我实际上可以在renderer.selectRootElementAngular 之外运行调用。因为我需要给新的部分时间来渲染,所以元素部分被包裹在一个超时中,以便在尝试选择元素之前让渲染线程有时间赶上。完成所有设置后,我可以使用基本的 CSS 选择器简单地调用元素。

我知道这个例子主要处理焦点事件,但对我来说很难在其他情况下使用它。

更新:Angular 在 Angular 4 中放弃了对RendererAngular 4 的支持,并在 Angular 9 中完全删除了它。此解决方案不应受到迁移到Renderer2. 请参阅此链接以获取更多信息: Renderer migration to Renderer2

于 2017-05-19T18:21:08.893 回答
19

对于试图在*ngIfor中获取组件实例的人*ngSwitchCase,您可以遵循此技巧。

创建init指令。

import {
    Directive,
    EventEmitter,
    Output,
    OnInit,
    ElementRef
} from '@angular/core';

@Directive({
    selector: '[init]'
})
export class InitDirective implements OnInit {
    constructor(private ref: ElementRef) {}

    @Output() init: EventEmitter<ElementRef> = new EventEmitter<ElementRef>();

    ngOnInit() {
        this.init.emit(this.ref);
    }
}

使用名称导出组件,例如myComponent

@Component({
    selector: 'wm-my-component',
    templateUrl: 'my-component.component.html',
    styleUrls: ['my-component.component.css'],
    exportAs: 'myComponent'
})
export class MyComponent { ... }

使用此模板获取ElementRefANDMyComponent实例

<div [ngSwitch]="type">
    <wm-my-component
           #myComponent="myComponent"
           *ngSwitchCase="Type.MyType"
           (init)="init($event, myComponent)">
    </wm-my-component>
</div>

在 TypeScript 中使用此代码

init(myComponentRef: ElementRef, myComponent: MyComponent) {
}
于 2016-12-29T00:21:22.123 回答
13
 */
import {Component,ViewChild} from '@angular/core' /*Import View Child*/

@Component({
    selector:'display'
    template:`

     <input #myname (input) = "updateName(myname.value)"/>
     <p> My name : {{myName}}</p>

    `
})
export class DisplayComponent{
  @ViewChild('myname')inputTxt:ElementRef; /*create a view child*/

   myName: string;

    updateName: Function;
    constructor(){

        this.myName = "Aman";
        this.updateName = function(input: String){

            this.inputTxt.nativeElement.value=this.myName; 

            /*assign to it the value*/
        };
    }
}
于 2017-01-16T14:18:49.387 回答
13

从导入ViewChild装饰器@angular/core,如下所示:

HTML 代码:

<form #f="ngForm"> 
  ... 
  ... 
</form>

TS 代码:

import { ViewChild } from '@angular/core';

class TemplateFormComponent {

  @ViewChild('f') myForm: any;
    .
    .
    .
}

现在您可以使用“myForm”对象访问类中的任何元素。

Source

于 2017-11-02T11:01:39.027 回答
5

注意:这不适用于 Angular 6 及更高版本ElementRef,因为.ElementRef<T>TnativeElement

我想补充一点,如果您ElementRef按照所有答案的建议使用 ,那么您将立即遇到ElementRef具有可怕类型声明的问题,看起来像

export declare class ElementRef {
  nativeElement: any;
}

在 nativeElement 是HTMLElement.

要解决此问题,您可以使用以下技术

import {Inject, ElementRef as ErrorProneElementRef} from '@angular/core';

interface ElementRef {
  nativeElement: HTMLElement;
}

@Component({...}) export class MyComponent {
  constructor(@Inject(ErrorProneElementRef) readonly elementRef: ElementRef) { }
}
于 2017-02-11T17:10:46.167 回答
4

我有两种方式:

第一种方式:

您可以通过 ElementRef 将其注入到组件的构造函数中来获取 DOM 元素的句柄:

constructor(private myElement: ElementRef) {
this.myElement.nativeElement // <- your direct element reference
}

第二种方式:

@Component({
  selector: 'my-app',
  template:
  `
  <input #input value="enterThere">
  `,
  styleUrls: [ './app.component.css' ]
})
export class AppComponent  {
  @ViewChild('input') input:ElementRef; 

  ngAfterViewInit() {
    console.log(this.input);
  }
于 2021-03-23T18:58:33.343 回答
3

快速使用的最小示例:

import { Component, ElementRef, ViewChild} from '@angular/core';

@Component({
  selector: 'my-app',
  template:
  `
  <input #inputEl value="hithere">
  `,
  styleUrls: [ './app.component.css' ]
})
export class AppComponent  {
  @ViewChild('inputEl') inputEl:ElementRef; 

  ngAfterViewInit() {
    console.log(this.inputEl);
  }
}
  1. 在感兴趣的 DOM 元素上放置一个模板引用变量。在我们的示例中,这是标签#inputEl上的。<input>
  2. 在我们的组件类中,通过 @ViewChild 装饰器注入 DOM 元素
  3. ngAfterViewInit访问生命周期挂钩中的元素。

笔记:

如果您想操作 DOM 元素,请使用 Renderer2 API 而不是直接访问元素。允许直接访问 DOM 会使您的应用程序更容易受到 XSS 攻击

于 2020-02-29T10:32:08.717 回答
2

要获得下一个兄弟姐妹,请使用此

event.source._elementRef.nativeElement.nextElementSibling
于 2017-10-12T12:49:37.633 回答
1

从列表中选择目标元素。从相同元素的列表中选择特定元素很容易。

组件代码:

export class AppComponent {
  title = 'app';

  listEvents = [
    {'name':'item1', 'class': ''}, {'name':'item2', 'class': ''},
    {'name':'item3', 'class': ''}, {'name':'item4', 'class': ''}
  ];

  selectElement(item: string, value: number) {
    console.log("item="+item+" value="+value);
    if(this.listEvents[value].class == "") {
      this.listEvents[value].class='selected';
    } else {
      this.listEvents[value].class= '';
    }
  }
}

html代码:

<ul *ngFor="let event of listEvents; let i = index">
   <li  (click)="selectElement(event.name, i)" [class]="event.class">
  {{ event.name }}
</li>

CSS代码:

.selected {
  color: red;
  background:blue;
}
于 2018-03-02T19:18:51.477 回答
1

对于里面的组件*ngIf,另一种方法:

我想选择的组件在 div 的 *ngIf 语句中,@jsgoupil 上面的答案可能有效(感谢@jsgoupil!),但我最终找到了一种避免使用 *ngIf 的方法,即使用 CSS 隐藏元素。

当 [className] 中的条件为真时,会显示 div,并且使用 # 命名组件有效,并且可以从打字稿代码中选择它。条件为假的时候不显示,反正我也不需要选择。

零件:

@Component({
    selector: 'bla',
    templateUrl: 'bla.component.html',
    styleUrls: ['bla.component.scss']
})
export class BlaComponent implements OnInit, OnDestroy {
    @ViewChild('myComponentWidget', {static: true}) public myComponentWidget: any;
    @Input('action') action: ActionType; // an enum defined in our code. (action could also be declared locally)

constructor() {
   etc;
}

// this lets you use an enum in the HMTL (ActionType.SomeType)
public get actionTypeEnum(): typeOf ActionType {
    return ActionType;
}

public someMethodXYZ: void {
    this.myComponentWidget.someMethod(); // use it like that, assuming the method exists
}

然后在 bla.component.html 文件中:

<div [className]="action === actionTypeEnum.SomeType ? 'show-it' : 'do-not-show'">

    <my-component #myComponentWidget etc></my-component>
</div>
<div>
    <button type="reset" class="bunch-of-classes" (click)="someMethodXYZ()">
        <span>XYZ</span>
    </button>
</div>   

和 CSS 文件:

 ::ng-deep {
    .show-it {
         display: block;   // example, actually a lot more css in our code
    }
    .do-not-show {
        display: none'; 
    }
}
于 2020-12-17T23:45:15.477 回答
0

如果您使用 Angular Material,您可以利用 cdkFocusInitial 指令。

例子:<input matInput cdkFocusInitial>

在此处阅读更多信息: https ://material.angular.io/cdk/a11y/overview#regions

于 2020-08-08T10:36:14.593 回答