API
Angular Testing Library
重新导出了 DOM Testing Library
的所有内容以及
render
方法。
以下的重新导出是经过 patch 的,使其更容易与 Angular 一起使用:
fireEvent
上的事件在事件被触发后自动调用一个变化检测周期findBy
查询会在调用函数之前自动调用一个变化检测周期waitFor
函数在调用回调函数之前自动调用一个变化检测周期
render
通过 Angular Testing Library,可以用两种方式渲染组件,通过组件的类型或用模板。
默认情况下,
render
也会导入NoopAnimationsModule
。
Type
要渲染一个组件,你需要将组件的类型传递给 render
方法。对于不使用你的应用程序的
其他部分的组件(例如设计模块或服务),渲染一个组件可以像下面的例子一样简单。
await render(AppComponent)
template
你也可以提供一个模板,而不是把组件的类型作为第一个参数传递。这种做法是渲染指令所
需要的,但也可以应用于组件,它甚至可能更有用。然后,指令的(或组件的)类型必须被
添加到 declarations
中。
指令示例:
await render('<div appSpoiler></div>', {
declarations: [SpoilerDirective],
})
组件示例:
await render(
'<app-component [value]="47" [otherValue]="anotherValue" (sendValue)="sendValue($event)"></app-component>',
{
declarations: [AppComponent],
componentProperties: {
anotherValue: 'valueOfAnotherProperty',
sendValue: jest.fn(),
},
},
)
export async function render<ComponentType>(
component: Type<ComponentType>,
renderOptions?: RenderComponentOptions<ComponentType>,
): Promise<RenderResult<ComponentType, ComponentType>>
export async function render<WrapperType = WrapperComponent>(
template: string,
renderOptions?: RenderTemplateOptions<WrapperType>,
): Promise<RenderResult<WrapperType>>
Component RenderOptions
componentInputs
用于设置组件的@Input
属性的对象。使用setInput
设置输入属性。如果组件属性未使
用@Input
属性进行注释,则抛出。
默认值 : {}
示例:
await render(AppComponent, {
componentInputs: {
counterValue: 10,
},
})
componentOutputs
用于设置组件的@Output
属性的对象。
默认值 : {}
示例:
await render(AppComponent, {
componentOutputs: {
clicked: (value) => { ... }
}
})
componentProperties
一个用于设置组件的 @Input
和 @Output
属性的对象。没有像
componentInputs
和componentOutputs
那样的细粒度控制。
默认 : {}
示例:
await render(AppComponent, {
// an input
counterValue: 10,
// an output
send: (value) => { ... }
// a public property
name: 'Tim'
})
declarations
渲染该组件所需的组件、指令和管道的集合。例如,组件的嵌套组件。
更多信息见Angular 文档。
默认 : []
示例:
await render(AppComponent, {
declarations: [CustomerDetailComponent, ButtonComponent],
})
componentProviders
通过依赖注入渲染组件所需的提供者的集合。
这些将在组件级别上提供。要想在模块层面上注入依赖关系,请使
用providers
。
更多信息请参见Angular 文档。
默认 : []
示例:
await render(AppComponent, {
componentProviders: [AppComponentService],
})
ɵcomponentImports
一个导入的集合,用来覆盖一个独立组件的导入。
默认 : undefined
示例:
await render(AppComponent, {
ɵcomponentImports: [
componentImports: [MockChildComponent],
]
})
childComponentOverrides
用于覆盖的子组件指定提供者的集合。
默认 : undefined
示例:
await render(AppComponent, {
childComponentOverrides: [
{
component: ChildOfAppComponent,
providers: [{provide: ChildService, useValue: {hello: 'world'}}],
},
],
})
detectChangesOnRender
当组件被渲染时,将调用 detectChanges
。
默认 : true
示例:
await render(AppComponent, {detectChangesOnRender: false})
autoDetectChanges
像“实际”运行的组件一样自动检测更改。例如,在发生事件时运行更改检测周期。
默认 : true
示例:
await render(AppComponent, {
autoDetectChanges: false,
})
excludeComponentDeclaration
排除被自动添加为声明的组件。当组件在导入的模块中被声明时需要这样做,例如用 SCAMs。
默认 : false
示例:
await render(AppComponent, {
imports: [AppModule], // a module that includes AppComponent
excludeComponentDeclaration: true,
})
imports
渲染组件所需的导入集合,例如,共享模块。如果 BrowserAnimationsModule
没有被添
加到集合中,默认会添加 NoopAnimationsModule
。
更多信息见Angular 文档。
默认 : [NoopAnimationsModule]
示例:
await render(AppComponent, {
imports: [AppSharedModule, MaterialModule],
})
providers
通过依赖性注入渲染组件所需的提供者集合。
这些将在模块级别上提供。要在组件级别上注入依赖关系,请使用
componentProviders
更多信息请参见Angular 文档。
默认 : []
示例:
await render(AppComponent, {
providers: [
CustomersService,
{
provide: MAX_CUSTOMERS_TOKEN,
useValue: 10,
},
],
})
queries
要绑定的查询。覆盖 DOM 测试库的默认设置,除非合并。
默认 : undefined
示例:
await render(AppComponent, {
queries: {...queries, ...customQueries},
})
routes
通过 RouterTestingModule.withRoutes
设置路由器服务的路由配置。更多信息
见Angular Routes 文档。
默认 : []
示例:
await render(AppComponent, {
declarations: [ChildComponent],
routes: [
{
path: '',
children: [
{
path: 'child/:id',
component: ChildComponent,
},
],
},
],
})
schemas
渲染该组件所需的模式集合。允许的值是 NO_ERRORS_SCHEMA
和
CUSTOM_ELEMENTS_SCHEMA
。
更多信息见Angular 文档。
默认 : []
示例:
await render(AppComponent, {
schemas: [NO_ERRORS_SCHEMA],
})
removeAngularAttributes
移除夹具中的 Angular 属性(ng-version 和 root-id)。
默认 : false
示例:
await render(AppComponent, {
removeAngularAttributes: true,
})
RenderResult
container
你渲染的 Angular Component 的包含 DOM 节点。这是一个普通的 DOM 节点,所以你可以
调用 container.querySelector
等来检查子节点。
debug
打印出组件的 DOM,并有语法高亮显示。接受一个可选的参数,以打印出一个特定的 DOM 节点。
const {debug} = await render(AppComponent)
debug()
change
改变组件的输入。这在道具更新后调用 detectChanges
。
const {change} = await render(Counter, {
componentProperties: {count: 4, name: 'Sarah'},
})
expect(screen.getByTestId('count-value').textContent).toBe('4')
expect(screen.getByTestId('name-value').textContent).toBe('Sarah')
change({count: 7})
// count updated to 7
expect(screen.getByTestId('count-value').textContent).toBe('7')
// name keeps the same value
expect(screen.getByTestId('name-value').textContent).toBe('Sarah')
rerender
创建并渲染一个新的组件实例。没有定义的输入属性会被清除。要保留它们,请使 用change。
const {rerender} = await render(Counter, {
componentProperties: {count: 4, name: 'Sarah'},
})
expect(screen.getByTestId('count-value').textContent).toBe('4')
expect(screen.getByTestId('name-value').textContent).toBe('Sarah')
await rerender({count: 7})
// count updated to 7
expect(screen.getByTestId('count-value').textContent).toBe('7')
// name is undefined because it's not provided in rerender
expect(screen.getByTestId('name-value').textContent).toBeUndefined()
detectChanges
触发该组件的变化检测周期。
更多信息 见Angular 文档。
debugElement
该组件的 Angular DebugElement
。
更多信息请参见Angular 文档。
fixture
该组件的 Angular ComponentFixture
。
更多信息请参 见Angular 文档。
const {fixture} = await render(AppComponent)
// componentInstance is typed as AppComponent
const componentInstance = fixture.componentInstance
🚨 如果你发现自己使用
fixture
来访问组件的内部值,你应该重新考虑!这可能意味 着你正在测试实现细节。
navigate
接受一个 DOM 元素或一个路径作为参数。如果一个元素被传递,navigate
将导航到该元
素的 href
值。如果传递了一个路径,navigate
将导航到该路径。
const { navigate } = await render(AppComponent, {
routes: [...]
})
await navigate(component.getByLabelText('To details'))
await navigate('details/3')
...queries
render
最重要的特点是,DOM Testing Library
的查询会自动返回,其第一个参数与被测组件绑定。
完整的列表见查询。
示例:
const {getByText, queryByLabelText} = await render(AppComponent)
screen.getByRole('heading', {
name: /api/i,
})
queryByLabelText(/First name/i')