我有打开,确认和关闭对话框的模态服务,我正在制作单元测试文件,但我得到了Angular的错误,这就是代码。
modal.service.ts
@Injectable()
export class ModalService {
constructor(private dialog: MatDialog) { }
public open<modalType>(modalComponent: ComponentType<modalType>): Observable<any> {
let dialogRef: MatDialogRef<any>;
dialogRef = this.dialog.open(modalComponent, {
maxWidth: '100vw'
});
console.log(dialogRef)
dialogRef.componentInstance.body = body;
return dialogRef.afterClosed().pipe(map(result => console.log('test'); );
}
}
modal.service.spec.ts
export class TestComponent {}
describe('ModalService', () => {
let modalService: ModalService;
const mockDialogRef = {
open: jasmine.createSpy('open')
};
beforeEach(async(() => {
TestBed.configureTestingModule({
imports: [ MatDialogModule ],
providers: [
ModalService,
MatDialogRef,
{ provide: MatDialog, useClass: MatDialogStub }
]
}).compileComponents();
modalService = TestBed.get(ModalService);
}));
it('open modal', () => {
modalService.open(DummyComponent, '300px');
expect(modalService.open).toHaveBeenCalled();
});
});
因此,使用该代码的错误是
TypeError: Cannot read property 'componentInstance' of undefined
你能帮我解决这个问题吗?非常感谢帮助。
测试mat-dialogs可能很棘手。我倾向于使用间谍对象从打开的对话框返回(下面的dialogRefSpyObj),这样我就可以更轻松地跟踪和控制测试。在您的情况下,它可能类似于以下内容:
describe('ModalService', () => {
let modalService: ModalService;
let dialogSpy: jasmine.Spy;
let dialogRefSpyObj = jasmine.createSpyObj({ afterClosed : of({}), close: null });
dialogRefSpyObj.componentInstance = { body: '' }; // attach componentInstance to the spy object...
beforeEach(() => {
TestBed.configureTestingModule({
imports: [MatDialogModule],
providers: [ModalService]
});
modalService = TestBed.get(ModalService);
});
beforeEach(() => {
dialogSpy = spyOn(TestBed.get(MatDialog), 'open').and.returnValue(dialogRefSpyObj);
});
it('open modal ', () => {
modalService.open(TestComponent, '300px');
expect(dialogSpy).toHaveBeenCalled();
// You can also do things with this like:
expect(dialogSpy).toHaveBeenCalledWith(TestComponent, { maxWidth: '100vw' });
// and ...
expect(dialogRefSpyObj.afterClosed).toHaveBeenCalled();
});
});
import { BeforeLogOutComponent } from '@app-global/components/before-log-out/before-log-out.component';
/**
* The method triggers before the logout.
* Opens the dialog and warns the user before log Out.
*/
public beforeLogOut(): void {
this._dialog.open(BeforeLogOutComponent, { width: '400px', disableClose: false, panelClass: 'dialog_before_log_out' })
.afterClosed()
.subscribe((res) => {
if (res && res.action === true) { this.loggedOut(); }
}, err => {
console.error(err);
});
}
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
import { MatDialog } from '@angular/material';
import { Observable, of } from 'rxjs';
<<-- Create a MatDialog mock class -->>
export class MatDialogMock {
// When the component calls this.dialog.open(...) we'll return an object
// with an afterClosed method that allows to subscribe to the dialog result observable.
open() {
return {
afterClosed: () => of({action: true})
};
}
}
describe('HeaderComponent', () => {
let component: HeaderComponent;
let fixture: ComponentFixture<HeaderComponent>;
beforeEach(async(() => {
TestBed.configureTestingModule({
imports: [
MaterialModule, RouterTestingModule, HttpModule, BrowserAnimationsModule,
HttpClientModule, FlexLayoutModule,
],
declarations: [
HeaderComponent,
],
providers: [
{ provide: MatDialog, useClass: MatDialogMock } <<-- look this
]
})
.compileComponents();
}));
beforeEach(async() => {
fixture = TestBed.createComponent(HeaderComponent);
component = fixture.componentInstance;
component.ngOnInit();
component.ngAfterViewInit();
await fixture.whenStable();
fixture.detectChanges();
});
fit('should create', () => {
expect(component).toBeTruthy();
});
// I test the dialog here.
fit('should open the dialog', () => {
component.beforeLogOut();
});
}
我没有你的确切答案,但我也对MatDialog
做了一些测试。我可以告诉你我做了什么。也许看看inject()
部分:
(为了清晰和保密,我删除了一些内容)
describe('MyDialogComponent', () => {
let dialog: MatDialog;
let overlayContainer: OverlayContainer;
let component: MyDialogComponent;
let fixture: ComponentFixture<MyDialogComponent>;
const mockDialogRef = {
close: jasmine.createSpy('close')
};
beforeEach(async(() => {
TestBed.configureTestingModule({
imports: [
BrowserAnimationsModule,
ReactiveFormsModule,
AngularMaterialModule,
],
providers: [
{ provide: MatDialogRef, useValue: mockDialogRef },
{
provide: MAT_DIALOG_DATA,
useValue: {
title: 'myTitle',
}
}
],
declarations: [MyDialogComponent],
});
TestBed.overrideModule(BrowserDynamicTestingModule, {
set: {
entryComponents: [MyDialogComponent]
}
});
TestBed.compileComponents();
}));
beforeEach(inject([MatDialog, OverlayContainer],
(d: MatDialog, oc: OverlayContainer) => {
dialog = d;
overlayContainer = oc;
})
);
afterEach(() => {
overlayContainer.ngOnDestroy();
});
beforeEach(() => {
fixture = TestBed.createComponent(MyDialogComponent);
component = fixture.componentInstance;
fixture.detectChanges();
});
it('should create', () => {
expect(component).toBeTruthy();
});
it('onCancel should close the dialog', () => {
component.onCancel();
expect(mockDialogRef.close).toHaveBeenCalled();
});
});