nativeElement.click() 和事件处理程序的点击有啥区别?

Posted

技术标签:

【中文标题】nativeElement.click() 和事件处理程序的点击有啥区别?【英文标题】:What is the difference between nativeElement.click() and event handler's click?nativeElement.click() 和事件处理程序的点击有什么区别? 【发布时间】:2018-03-07 15:05:46 【问题描述】:

我一直在尝试对单击调用函数的提交按钮的 Angular 2 应用程序进行测试。我通常使用两种方法来执行相同的操作。

element.nativeElement.click()

element.triggerEventHandler('click',null);

我认为这两种方法都是相同的,直到我遇到触发事件处理程序方法不起作用的情况。

element = fixture.debugElement.query(By.css('.dropList li'));
element.triggerEventHandler('click',null);  //Click event works here
fixture.detectChanges();
let button = fixture.debugElement.query(By.css('button'));
//button.triggerEventHandler('click',null);    //Does not call function
button.nativeElement.click();    //Calls function
fixture.detectChanges();

供您参考的模板:

<form (ngSubmit)="printSelection()">
   <div class="dropList">
     <ul>
        <li *ngFor="let element of data.elements" (click)="selectElement(element)"> </li>
    </ul>
   </div>
   <button type="submit">Submit</button>
</form>

那么,这两种方法之间有什么区别,还是您认为我的代码中的某个地方可能出错了?

【问题讨论】:

可以添加组件的模板吗? @yurzui 添加了模板... 谢谢。您的模板中的button 在哪里? 【参考方案1】:

element.nativeElement.click()

是native method 模拟鼠标点击元素。它在冒泡,其行为方式与我们单击元素时的行为方式相同。

debugElement.triggerEventHandler(eventName, eventObj)

是 Angular 内置方法,它只为当前调试元素上的给定 eventName 调用所有侦听器:

triggerEventHandler(eventName: string, eventObj: any) 
  this.listeners.forEach((listener) => 
    if (listener.name == eventName) 
      listener.callback(eventObj);
    
  );

DebugRenderer2运行listen方法时添加监听器:

listen(
    target: 'document'|'windows'|'body'|any, eventName: string,
    callback: (event: any) => boolean): () => void 
  if (typeof target !== 'string') 
    const debugEl = getDebugNode(target);
    if (debugEl) 
      debugEl.listeners.push(new EventListener(eventName, callback));
    
  

  return this.delegate.listen(target, eventName, callback);

当我们将事件绑定应用到类似元素时会发生这种情况

(click)="handler()" @HostListener('click') host: ' '(mouseenter): 'handler()' renderer.listen

假设我们有以下模板:

<div (click)="test()">
  <div class="innerDiv">
     title
  </div>
</div>

我们的测试将如下所示:

de = fixture.debugElement.query(By.css('.innerDiv'));

de.nativeElement.click(); // event will be bubbled and test handler will be called

de.triggerEventHandler('click', null); // there is not handlers for click event 
                                       // that angular added to this element 
                                       // so test method won't be called

然后让我们看看你的模板。 button 没有处理程序,所以 triggerEventHandler 不会有任何效果。另一方面,button.nativeElement.click(); 将触发提交,因为按钮具有提交类型,并且它在点击事件时按钮的标准行为。

【讨论】:

很好的解释!谢谢!

以上是关于nativeElement.click() 和事件处理程序的点击有啥区别?的主要内容,如果未能解决你的问题,请参考以下文章

事件流 事件冒泡和事件捕获

事件冒泡和事件捕获

事件流:事件冒泡和事件捕获

[JS]笔记12之事件机制--事件冒泡和捕获--事件监听--阻止事件传播

你真的理解事件绑定事件冒泡和事件委托吗?

change事件和onchange事件,click事件和onclick事件这些都有啥区别