测试来自角度属性指令的点击

时间:2019-03-21 14:23:37

标签: angular angular-directive angular-test

我无法为我编写的Angular Attribute Directive创建单元测试。该指令称为 TrackClickDirective ,我正在尝试测试以下内容。

  • 单击附加了该指令的元素时,应在该指令上调用特定方法。

我怀疑我的单元测试配置有问题。

请查看我在StackBlitz上的实现,您可以在其中查看测试的运行情况:

StackBltiz: https://stackblitz.com/edit/angular-test-click-on-attribute-directive-with-hostlistener

这是我的单元测试代码- track-click.directive.spec.ts

import { Component, ElementRef, DebugElement } from '@angular/core';
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { By } from "@angular/platform-browser";
import { TrackClickDirective } from './track-click.directive';
import { Analytics} from './analytics.service';

class MockAnalytics {
  eventTrack() {}
};
class MockElementRef {
}

@Component({
  template: '<button appTrackClick>Test</button>'
})
export class TestButtonWithoutNameComponent { }

describe('TrackClickDirective', () => {

  let component: TestButtonWithoutNameComponent;
  let fixture: ComponentFixture<TestButtonWithoutNameComponent>;
  let directive: TrackClickDirective;
  let inputEl: DebugElement;
  let spy: any;

  beforeEach(() => {
    TestBed.configureTestingModule({
      declarations: [
        TestButtonWithoutNameComponent
      ],
      providers: [
        TrackClickDirective,
        { provide: Analytics, useClass: MockAnalytics },
        { provide: ElementRef, useClass: MockElementRef }
      ]
    });
    fixture = TestBed.createComponent(TestButtonWithoutNameComponent);
    component = fixture.componentInstance;
    directive = TestBed.get(TrackClickDirective);
    inputEl = fixture.debugElement.query(By.css('button'));
  });

  it('should call the trackClick methoe when the button is clicked', () => {
    spy = spyOn(directive, 'trackClick');
    inputEl.triggerEventHandler('click', null);
    fixture.detectChanges();
    expect(directive.trackClick).toHaveBeenCalled();
  });
});

我在这里做什么错了?运行单元测试时,我得到以下信息:

FAILED TESTS:
  TrackClickDirective
    ✖ should call the trackClick method when the button is clicked
      HeadlessChrome 72.0.3626 (Mac OS X 10.14.0)
    Expected spy trackClick to have been called.

1 个答案:

答案 0 :(得分:1)

遵循Angular's docs上的示例。

以下应该可以解决问题:

import { Component, ElementRef, DebugElement } from '@angular/core';
import { ComponentFixture, TestBed } from '@angular/core/testing';
import { By } from "@angular/platform-browser";
import { TrackClickDirective } from './track-click.directive';
import { Analytics} from './analytics.service';

class MockAnalytics {
  eventTrack() {}
};
class MockElementRef {
}

@Component({
  template: '<button appTrackClick>Test</button>'
})
export class TestButtonWithoutNameComponent { }

describe('TrackClickDirective', () => {

  let component: TestButtonWithoutNameComponent;
  let fixture: ComponentFixture<TestButtonWithoutNameComponent>;
  let directive: TrackClickDirective;
  let inputEl: DebugElement;
  let spy: any;

  beforeEach(() => {
    TestBed.configureTestingModule({
      declarations: [
        TestButtonWithoutNameComponent,
        TrackClickDirective
      ],
      providers: [
        TrackClickDirective,
        { provide: Analytics, useClass: MockAnalytics },
        { provide: ElementRef, useClass: MockElementRef }
      ]
    });
    fixture = TestBed.createComponent(TestButtonWithoutNameComponent);
    component = fixture.componentInstance;
    directive = TestBed.get(TrackClickDirective);
    inputEl = fixture.debugElement.query(By.css('button'));
  });

  it('should call the trackClick method when the button is clicked', () => {
    directive = fixture.debugElement.query(By.directive(TrackClickDirective)).injector.get(TrackClickDirective) as TrackClickDirective;
    spy = spyOn(directive, 'trackClick');
    inputEl.triggerEventHandler('click', null);
    fixture.detectChanges();
    expect(directive.trackClick).toHaveBeenCalled();
  });
});

关键在这一行:

fixture.debugElement.query(By.directive(TrackClickDirective)).injector.get(TrackClickDirective) as TrackClickDirective;

虽然非常冗长,但是它允许您获取已创建并在组件中使用的指令的实际实例,从而可以窥探其方法。

更新后的StackBlitz示例: https://stackblitz.com/edit/angular-test-click-on-attribute-directive-with-hostlistener