我尝试将事件(单击)设置为按钮元素,但我仍然有错误
ERROR Error: Uncaught (in promise): TypeError: Cannot read property 'nativeElement' of undefined
TypeError: Cannot read property 'nativeElement' of undefined
当我试图调用<a (click)=EditButton(data)>Edytuj</a>
时没有动作,所以我尝试在少数版本中使用ChildView,Renderer,nativeElement,但没有成功。
按钮嵌套在组件文件
...
columns: [
{ data: "Imie" },
{ data: "Nazwisko" },
{ data: "Brygada" },
{ data: "Stawka" },
{ data: "Aktywny" },
{
data: null, defaultContent :`
<a #EditButton >Edytuj</a>
<a #DeleteButton >Usuń</a>
`
},
],
buttons: [
...
Bellow是组件代码:
import { Component, OnInit, ViewChild, TemplateRef, Renderer2, AfterViewInit } from '@angular/core';
import { FadeInTop } from "../shared/animations/fade-in-top.decorator";
import { Http, Response } from '@angular/http';
import { BsModalService, ModalDirective } from 'ngx-bootstrap/modal';
import { BsModalRef } from 'ngx-bootstrap/modal/bs-modal-ref.service';
import { Observable } from "rxjs/Rx";
//import 'rxjs/add/operator/map';
//import 'rxjs/add/operator/catch';
import { bootstrapValidationRoutes } from '../+forms/+bootstrap-validation/bootstrap-validation.routing';
@FadeInTop()
@Component({
selector: 'sa-czas-pracy',
templateUrl: './czas-pracy.component.html',
styleUrls: ['./czas-pracy.component.css']
})
export class CzasPracyComponent implements OnInit {
public REST_ROOT = 'http://localhost:3000/pracownicy/pracownicy';
//private static lgModal;
options = {
dom: "Bfrtip",
ajax: (data, callback, settings) => {
this.http.get(this.REST_ROOT)
.map(this.extractData)
.catch(this.handleError)
.subscribe((data) => {
console.log('data from rest endpoint', data);
callback({
aaData: data.slice(0, 100)
})
})
},
columns: [
{ data: "Imie" },
{ data: "Nazwisko" },
{ data: "Brygada" },
{ data: "Stawka" },
{ data: "Aktywny" },
{
data: null, defaultContent :`
<button #EditButton >Edytuj</button>
<button #DeleteButton >Usuń</button>
`
},
],
buttons: [
{ text: 'Dodaj',
action: () => {this.showAddModal()}
}
],
};
@ViewChild('AddEditModal') AddEditModal :ModalDirective;
@ViewChild('EditButton') myButton;
modalRef: BsModalRef;
title: string;
constructor(
private http:Http,
private modalService: BsModalService,
private renderer: Renderer2
) {}
ngOnInit() {
// let simple1 = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
// console.log('Clicking the button', evt);
// });
}
ngAfterViewInit() {
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
}
editPracownik(){
console.log('edit');
}
...
我搜索并尝试了很多方法,但没有任何效果。
任何解决方案都会有所帮助谢谢。
谢谢你,Benedikt Schmidt,
是的,我查了一下,你说得对。在DataTable.option之前和之后调用ngAfterContentChecked()。所以我认为没关系。但myButton仍未定义。
加载组件时的控制台:
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:34 data from rest endpoint (3) [{…}, {…}, {…}]
3czas-pracy.component.ts:48 Create button
czas-pracy.component.ts:48 Create button
3czas-pracy.component.ts:48 Create button
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
我尝试通过方法引用引用,女巫链接到html模板中定义的按钮 - 而myButton仍未定义
<a class="btn btn-danger btn-xs" (click)="findMyButton()">FindMyButton</a>
...
findMyButton(){
console.log('find my button clicked')
console.log((this.myButton) ? 'is myButton' : "myButton undefined");
if(this.myButton && !this.hasAttachedListener){
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
this.hasAttachedListener=true;
}
结果:
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:90 find my button clicked
czas-pracy.component.ts:91 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
czas-pracy.component.ts:79 AfterContentChecked
czas-pracy.component.ts:80 myButton undefined
更改了代码
export class CzasPracyComponent implements OnInit, AfterContentChecked {
public REST_ROOT = 'http://localhost:3000/pracownicy/pracownicy';
//private static lgModal;
options = {
dom: "Bfrtip",
ajax: (data, callback, settings) => {
this.http.get(this.REST_ROOT)
.map(this.extractData)
.catch(this.handleError)
.subscribe((data) => {
console.log('data from rest endpoint', data);
callback({
aaData: data.slice(0, 100)
})
})
},
columns: [
{ data: "Imie" },
{ data: "Nazwisko" },
{ data: "Brygada" },
{ data: "Stawka" },
{ data: "Aktywny" },
{
data: null, render : function (data, type, row) {
console.log('Create button');
return `
<button #EditButton >Edytuj</button>
<button #DeleteButton >Usuń</button>
`
}
},
],
buttons: [
{ text: 'Dodaj',
action: () => {this.showAddModal()}
}
],
};
@ViewChild('AddEditModal') AddEditModal :ModalDirective;
@ViewChild('EditButton') myButton;
@ViewChild('button') myButton1;
modalRef: BsModalRef;
title: string;
hasAttachedListener:boolean=false;
constructor(
private http:Http,
private modalService: BsModalService,
private renderer: Renderer2
) {}
ngOnInit() {
}
ngAfterContentChecked() {
console.log('AfterContentChecked');
console.log((this.myButton) ? 'is myButton' : "myButton undefined");
if(this.myButton && !this.hasAttachedListener){
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
this.hasAttachedListener=true;
}
}
findMyButton(){
console.log('find my button clicked')
console.log((this.myButton) ? 'is myButton' : "myButton undefined");
if(this.myButton && !this.hasAttachedListener){
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
this.hasAttachedListener=true;
}
}
您是否知道如何实现角度
可见的按钮代码答案 0 :(得分:2)
我对DataTable.net并不熟悉,但我认为这是正在发生的事情。通过DataTable.net,您将从后端加载信息以初始化您的视图。这是通过调用REST_ROOT
链接完成的,但您必须考虑这需要时间,看起来结果与组件初始化时的结果不同。
现在问题就在于此。您正在尝试访问尚不存在的按钮。
ngAfterViewInit() {
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
}
初始化视图后,ngAfterViewInit()生命周期钩子被称为一次。这是在创建组件后立即完成的,并且它不依赖于后续阶段中动态创建的内容。所以此时,您的http调用可能仍未完成,因此视图中不存在该按钮。所以你正在使用的生命周期钩子只是试图过早找到你的按钮。以下是文档说的内容:
在Angular初始化组件的视图和子视图后做出响应 观点。在第一次ngAfterContentChecked()之后调用一次。一个 仅限组件钩。
您可能需要在另一个生命周期钩子中实现它,例如ngAfterViewChecked()。这样的钩子被更频繁地调用,所以请确保你总是检查你的按钮是否存在,特别是只附加一次点击事件,所以跟踪你是否已经连接它,否则你将有很多听众按钮。
hasAttachedListener:boolean=false;
ngAfterViewChecked() {
if(this.myButton && !this.hasAttachedListener){
let simple = this.renderer.listen(this.myButton.nativeElement, 'click', (evt) => {
console.log('Clicking the button', evt);
});
this.hasAttachedListener=true;
}
}
最后一点,我不确定@ViewChild是否可以正常工作,如果html刚刚创建为例如innerHTML,而不是通过渲染器或某些东西。我认为在这种情况下,angular不知道它应该观看#EditButton,因为它在创建时没有意识到它。因此,即使上面的这个修复可能有帮助(我不确定),仍然可能是按钮引用始终未定义。
我试着让它更容易理解。使用ViewChild()
和#
运算符angular为用户提供了获取真实DOM元素的引用的机会,这与您通常使用$
选择器的方式类似。但这仅适用于HTML 通过角度本身生成的情况。例如,如果您有组件的模板代码,angular将获取该HTML代码并将其写入DOM本身,这样就可以了解内部的所有内容。如果我打开浏览器的开发人员工具并动态编辑一些HTML代码,那么angular就不会知道任何关于它的信息。因此,由角色生成的HTML代码与由人生成的HTML代码或与角度无关的外部脚本之间存在差异。
那么这里有什么问题?问题是您的数据表HTML是在没有角度帮助的情况下生成的,它是由一个完全不了解角度的外部脚本生成的。这就像你在浏览器中打开开发人员工具并复制一些代码一样.Angular不知道它刚刚发生。现在,如果你有一个像#MyButton
这样的代码中的按钮,你的角度组件只能知道如果angular已经自己创建了这个引用。但是在使用外部脚本的情况下,angular几乎不知道此按钮是否存在,因此您对@ViewChild()
的引用始终未定义。
这是你可以尝试的。虽然您不能使用@ViewChild
来引用动态创建的DOM元素,但您可以在模板中引用顶级DOM元素,然后在其中创建数据表代码。
<div #table>
<!-- this is where your table will be placed -->
</div>
@ViewChild('table') tableRef;
现在有了这个引用,你可以等到你的代码准备好了(你已经附加了日志文件,你可以在那里看到数据表完成加载的确切时间)并创建了HTML。您的引用现在可以完全访问其所有子div,包括您创建的表。您可以使用此引用来迭代其子项。
this.tableRef.nativeElement.childNodes
或者可能是更好的解决方案,您可以通过查询选择找到您可以在此顶级节点上调用的按钮。假设您已经更新了按钮,现在它将具有ID而不是角度参考标记。
<button id="EditButton">Edytuj</button>
然后,当创建表时,我们在顶级节点上调用querySelection
并找到具有该ID的对象。
var editButton = this.tableRef.nativeElement.querySelector("#EditButton");
this.renderer.listen(editButton, 'click', (evt) => {
console.log('Clicking the button', evt);
});
这应该可以解决问题。
答案 1 :(得分:2)
我终于解决了这个问题。当然非常感谢Benedikt Schmidt。 解决方案下方:
...
dataPracownicy : any;
options = {
dom: "Bfrtip",
ajax: (data, callback, settings) => {
this.http.get(this.REST_ROOT)
.map(this.extractData)
.catch(this.handleError)
.subscribe((data) => {
this.dataPracownicy = data;
console.log('data from rest endpoint', data);
callback({
aaData: data.slice(0, 100)
})
})
},
columns: [
{ data: "Imie" },
{ data: "Nazwisko" },
{ data: "Brygada" },
{ data: "Stawka" },
{ data: "Aktywny" },
{
data: null, render : function (data, type, row) {
console.log('Create button');
return `
<button id=EditButton >Edytuj</button>
<button #DeleteButton >Usuń</button>
`
}
},
],
buttons: [
{ text: 'Dodaj',
action: () => {this.showAddModal()}
}
],
};
@ViewChild('AddEditModal') AddEditModal :ModalDirective;
@ViewChild('TablePracownicy') tableRef;
modalRef: BsModalRef;
title: string;
hasAttachedListener:boolean=false;
constructor(
private http:Http,
private modalService: BsModalService,
private renderer: Renderer
) {}
ngOnInit() {}
ngAfterViewChecked() {
if (!this.hasAttachedListener){
var buttons = document.querySelectorAll("#EditButton");
console.log(buttons.length );
if(buttons.length>0){
console.log(buttons);
for(let i=0;i<buttons.length;i++){
this.renderer.listen(buttons[i], 'click', (evt) => {
console.log('Clicking the button'+[i], evt);
});
}
this.hasAttachedListener=true;
}
}
}
...
&#13;
我希望我的解决方案有用 问候