gpt4 book ai didi

unit-testing - 如何在 Angular2 中对 FormControl 进行单元测试

转载 作者:太空狗 更新时间:2023-10-29 16:56:09 36 4
gpt4 key购买 nike

我的测试方法如下:

/**
* Update properties when the applicant changes the payment term value.
* @return {Mixed} - Either an Array where the first index is a boolean indicating
* that selectedPaymentTerm was set, and the second index indicates whether
* displayProductValues was called. Or a plain boolean indicating that there was an
* error.
*/
onPaymentTermChange() {
this.paymentTerm.valueChanges.subscribe(
(value) => {
this.selectedPaymentTerm = value;
let returnValue = [];
returnValue.push(true);
if (this.paymentFrequencyAndRebate) {
returnValue.push(true);
this.displayProductValues();
} else {
returnValue.push(false);
}
return returnValue;
},
(error) => {
console.warn(error);
return false;
}
)
}

如您所见,paymentTerm 是一个表单控件,它返回一个 Observable,然后订阅它并检查返回值。

我似乎找不到任何关于对 FormControl 进行单元测试的文档。我最接近的是这篇关于 Mocking Http requests 的文章,这是一个类似的概念,因为它们返回 Observables 但我认为它并不完全适用。

作为引用,我使用的是 Angular RC5,使用 Karma 运行测试,框架是 Jasmine。

最佳答案

更新

至于这个关于异步行为的答案的第一部分,我发现你可以使用 fixture.whenStable()这将等待异步任务。所以不需要只使用内联模板

it('', async(() => {
fixture.whenStable().then(() => {
// your expectations.
})
})

首先让我们解决一些在组件中测试异步任务的一般问题。当我们测试不受测试控制的异步代码时,应该使用 fakeAsync ,因为它可以让我们调用 tick() ,这使得测试时 Action 看起来是同步的。例如

class ExampleComponent implements OnInit {
value;

ngOnInit() {
this._service.subscribe(value => {
this.value = value;
});
}
}

it('..', () => {
const fixture = TestBed.createComponent(ExampleComponent);
fixture.detectChanges();
expect(fixture.componentInstance.value).toEqual('some value');
});

此测试将失败,因为 ngOnInit被调用,但 Observable 是异步的,因此该值不会为测试中的同步调用及时设置(即 expect )。

为了解决这个问题,我们可以使用 fakeAsynctick强制测试等待所有当前异步任务完成,使其在测试中看起来好像是同步的。

import { fakeAsync, tick } from '@angular/core/testing';

it('..', fakeAsync(() => {
const fixture = TestBed.createComponent(ExampleComponent);
fixture.detectChanges();
tick();
expect(fixture.componentInstance.value).toEqual('some value');
}));

现在测试应该通过了,因为 Observable 订阅没有意外的延迟,在这种情况下,我们甚至可以在滴答调用中传递毫秒延迟 tick(1000) .

这个( fakeAsync )是一个有用的特性,但问题是当我们使用 templateUrl 时在我们的 @Component s,它进行 XHR 调用,然后 XHR calls can't be made in a fakeAsync .在某些情况下,您可以模拟服务以使其同步,如 this post 中所述。 ,但在某些情况下它只是不可行或太难了。在表格的情况下,这是不可行的。

因此,在处理表单时,我倾向于将模板放在 template 中。而不是外部 templateUrl并将表单分解为较小的组件(如果它们真的很大)(只是为了在组件文件中没有巨大的字符串)。我能想到的唯一其他选择是使用 setTimeout在测试里面,让异步操作通过。这是一个偏好问题。我只是决定在处理表单时使用内联模板。它打破了我的应用程序结构的一致性,但我不喜欢 setTimeout解决方案。

现在就表单的实际测试而言,我发现的最佳来源只是查看 source code integration tests .您需要将标签更改为您使用的 Angular 版本,因为默认主分支可能与您使用的版本不同。

下面是几个例子。

测试输入时,您想要更改 nativeElement 上的输入值,然后发送 input事件使用 dispatchEvent .例如

@Component({
template: `
<input type="text" [formControl]="control"/>
`
})
class FormControlComponent {
control: FormControl;
}

it('should update the control with new input', () => {
const fixture = TestBed.createComponent(FormControlComponent);
const control = new FormControl('old value');
fixture.componentInstance.control = control;
fixture.detectChanges();

const input = fixture.debugElement.query(By.css('input'));
expect(input.nativeElement.value).toEqual('old value');

input.nativeElement.value = 'updated value';
dispatchEvent(input.nativeElement, 'input');

expect(control.value).toEqual('updated value');
});

这是从源集成测试中提取的简单测试。下面有更多的测试示例,一个来自源代码,还有一些不是,只是为了展示测试中没有的其他方法。

对于您的特定情况,您似乎正在使用 (ngModelChange) ,您将调用分配给 onPaymentTermChange() .如果是这种情况,则您的实现没有多大意义。 (ngModelChange)当值改变时,它已经会吐出一些东西,但是每次模型改变时你都在订阅。你应该做的是接受 $event参数更改事件发出的内容

(ngModelChange)="onPaymentTermChange($event)"

每次更改时,您都会收到新值。因此,只需在您的方法中使用该值,而不是订阅。 $event将是新值。

如果您确实想使用 valueChangeFormControl ,你应该从 ngOnInit 开始听它,因此您只需订阅一次。您将在下面看到一个示例。我个人不会走这条路。我会按照你的方式去做,但不要订阅更改,只需接受更改中的事件值(如前所述)。

这是一些完整的测试

import {
Component, Directive, EventEmitter,
Input, Output, forwardRef, OnInit, OnDestroy
} from '@angular/core';
import { Subscription } from 'rxjs/Subscription';
import { TestBed, fakeAsync, tick } from '@angular/core/testing';
import { By } from '@angular/platform-browser/src/dom/debug/by';
import { getDOM } from '@angular/platform-browser/src/dom/dom_adapter';
import { dispatchEvent } from '@angular/platform-browser/testing/browser_util';
import { FormControl, ReactiveFormsModule } from '@angular/forms';

class ConsoleSpy {
log = jasmine.createSpy('log');
}

describe('reactive forms: FormControl', () => {
let consoleSpy;
let originalConsole;

beforeEach(() => {
consoleSpy = new ConsoleSpy();
originalConsole = window.console;
(<any>window).console = consoleSpy;

TestBed.configureTestingModule({
imports: [ ReactiveFormsModule ],
declarations: [
FormControlComponent,
FormControlNgModelTwoWay,
FormControlNgModelOnChange,
FormControlValueChanges
]
});
});

afterEach(() => {
(<any>window).console = originalConsole;
});

it('should update the control with new input', () => {
const fixture = TestBed.createComponent(FormControlComponent);
const control = new FormControl('old value');
fixture.componentInstance.control = control;
fixture.detectChanges();

const input = fixture.debugElement.query(By.css('input'));
expect(input.nativeElement.value).toEqual('old value');

input.nativeElement.value = 'updated value';
dispatchEvent(input.nativeElement, 'input');

expect(control.value).toEqual('updated value');
});

it('it should update with ngModel two-way', fakeAsync(() => {
const fixture = TestBed.createComponent(FormControlNgModelTwoWay);
const control = new FormControl('');
fixture.componentInstance.control = control;
fixture.componentInstance.login = 'old value';
fixture.detectChanges();
tick();

const input = fixture.debugElement.query(By.css('input')).nativeElement;
expect(input.value).toEqual('old value');

input.value = 'updated value';
dispatchEvent(input, 'input');
tick();

expect(fixture.componentInstance.login).toEqual('updated value');
}));

it('it should update with ngModel on-change', fakeAsync(() => {
const fixture = TestBed.createComponent(FormControlNgModelOnChange);
const control = new FormControl('');
fixture.componentInstance.control = control;
fixture.componentInstance.login = 'old value';
fixture.detectChanges();
tick();

const input = fixture.debugElement.query(By.css('input')).nativeElement;
expect(input.value).toEqual('old value');

input.value = 'updated value';
dispatchEvent(input, 'input');
tick();

expect(fixture.componentInstance.login).toEqual('updated value');
expect(consoleSpy.log).toHaveBeenCalledWith('updated value');
}));

it('it should update with valueChanges', fakeAsync(() => {
const fixture = TestBed.createComponent(FormControlValueChanges);
fixture.detectChanges();
tick();

const input = fixture.debugElement.query(By.css('input')).nativeElement;

input.value = 'updated value';
dispatchEvent(input, 'input');
tick();

expect(fixture.componentInstance.control.value).toEqual('updated value');
expect(consoleSpy.log).toHaveBeenCalledWith('updated value');
}));
});

@Component({
template: `
<input type="text" [formControl]="control"/>
`
})
class FormControlComponent {
control: FormControl;
}

@Component({
selector: 'form-control-ng-model',
template: `
<input type="text" [formControl]="control" [(ngModel)]="login">
`
})
class FormControlNgModelTwoWay {
control: FormControl;
login: string;
}

@Component({
template: `
<input type="text"
[formControl]="control"
[ngModel]="login"
(ngModelChange)="onModelChange($event)">
`
})
class FormControlNgModelOnChange {
control: FormControl;
login: string;

onModelChange(event) {
this.login = event;
this._doOtherStuff(event);
}

private _doOtherStuff(value) {
console.log(value);
}
}

@Component({
template: `
<input type="text" [formControl]="control">
`
})
class FormControlValueChanges implements OnDestroy {
control: FormControl;
sub: Subscription;

constructor() {
this.control = new FormControl('');
this.sub = this.control.valueChanges.subscribe(value => {
this._doOtherStuff(value);
});
}

ngOnDestroy() {
this.sub.unsubscribe();
}

private _doOtherStuff(value) {
console.log(value);
}
}

关于unit-testing - 如何在 Angular2 中对 FormControl 进行单元测试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39481324/

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com