繁体   English   中英

Angular Reactive Form 单元测试:HTML 值和控制值不同步

[英]Angular Reactive Form Unit Testing: HTML value and control value out of sync

我在此处遵循 Angular 反应式单元测试指南,但始终无法获得要同步的控制值和 HTML 值。 下面是我的实现; 请注意,除了指定默认值外,我还尝试调用setValue

import { Component, OnInit } from '@angular/core';
import { FormBuilder, FormGroup, Validators } from '@angular/forms';

@Component({
    selector: 'user',
    templateUrl: './user.component.html',
    styleUrls: ['./user.component.css']
})
export class UserComponent implements OnInit {

    loginForm!: FormGroup;

    constructor(private formBuilder: FormBuilder) { }

    ngOnInit(): void {
        this.initLoginForm();

        const usernameControl = this.loginForm.get('username');
        usernameControl?.valueChanges.subscribe(username => {
            debugger; // This doesn't fire
            this.loginForm.patchValue({username: username});
        });
    }

    initLoginForm() {
        this.loginForm = this.formBuilder.group({
            username: ['jack', Validators.compose([Validators.required])],
            password: ['JacksPassword']
        });

        this.loginForm.setValue({
            username: 'jack',
            password: 'JacksPassword'
        })

        this.loginForm.updateValueAndValidity();
    }
}
<form [formGroup]="loginForm" id="loginForm">
    <div>
        <input formControlName="username" placeholder="Username" required id="usernameInput"/>
    </div>
    <div>
        <input formControlName="password" placeholder="Password" type="password"/>
    </div>
</form>

这是我的测试:

import { ComponentFixture, TestBed } from '@angular/core/testing';
import { FormBuilder } from '@angular/forms';
import { UserComponent } from './user.component';

describe('UserComponent', () => {
    let component: UserComponent;
    let fixture: ComponentFixture<UserComponent>;

    beforeEach(async () => {
        await TestBed.configureTestingModule({
            declarations: [UserComponent],
            providers: [FormBuilder]
        })
            .compileComponents();
    });

    beforeEach(() => {
        fixture = TestBed.createComponent(UserComponent);
        component = fixture.componentInstance;
        fixture.detectChanges();
    });

    it('should have same values between control and UI', () => {
        const loginFormUserElement: HTMLInputElement = fixture.debugElement.nativeElement.querySelector('#loginForm').querySelector('#usernameInput');
        const userNameValueFromGroup = component.loginForm.get('username');
        expect(loginFormUserElement.value).toEqual(userNameValueFromGroup?.value);
    });

    it('should accept username', () => {
        const loginFormUserElement: HTMLInputElement = fixture.debugElement.nativeElement.querySelector('#loginForm').querySelector('#usernameInput');
        loginFormUserElement.value = 'joe';
        loginFormUserElement.dispatchEvent(new Event('input'));
        fixture.detectChanges();
        fixture.whenStable().then(() => {
            const userNameValueFromGroup = component.loginForm.get('username');
            expect(loginFormUserElement.value).toEqual('joe');
            expect(loginFormUserElement.value).toEqual(userNameValueFromGroup?.value);
        });
    });
});

这是结果。 设置默认值不会更新 UI。 调用setValue不会更新 UI。 在 UI 元素上设置值不会更新控件。

在此处输入图像描述

它必须是基本的东西。 我错过了什么?

编辑:我试图根据一个独立的单元测试示例设置一个 StackBlitz 实现,但formGroup指令似乎无法在 Jasmine 下被识别; 我正在将ReactiveFormsModule导入到 app.module 中。 该链接位于此处,以防任何人可以提供有关我在这方面缺少的内容的见解。

编辑:这里的问题似乎实际上与我的单元测试工作有关。 当我直接使用该组件时,默认值正确显示在 HTML 输入元素上; 但是,在运行 Karma 的 Chrome 上通过控制台检查消息时,我看到相同的错误消息,指示无法识别formGroup指令。 我已经相应地更新了标题。

这个问题实际上纯粹是在单元测试的设置中,当组件被实时使用时,控件和表单之间的绑定工作正常。 单元测试中的问题是ReactiveFormsModule必须通过TestBed.configureTestingModule导入; 我在安装过程中也没有调用ngOnInit ,尽管它的缺失似乎没有任何效果(目前)。

我更新的设置代码如下。 奇怪的是,尝试在 StackBlitz 设置中以相同的方式导入ReactiveFormsModule会产生错误“超出最大调用堆栈大小”,但它在我的机器上运行良好。

    beforeEach(() => {
        TestBed.configureTestingModule({
            declarations: [TimeRangeSelectorComponent],
            imports: [ReactiveFormsModule], // This is new
            providers: [
                FormBuilder,
                { provide: TimeRangeSelectorDefaultsBase, useValue: new TestTimeRangeSelectorDefaults() },
            ]
        });
        fixture = TestBed.createComponent(TimeRangeSelectorComponent);
        component = fixture.componentInstance;
        component.ngOnInit(); // This is new
        fixture.detectChanges();
    });

事实证明这是一个骗局 将此问题保留在这里,以防对其他人有帮助; 直到我开始搜索与测试相关的问题,我才发现另一个。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM