Как найти наличие дочернего компонента в angular с помощью `testing-library`

В моем component.html я сохраняю некоторые необходимые дочерние компоненты. Я хотел бы проверить их существование. Я также используюCUSTOM_ELEMENTS_SCHEMAна моем стенде. если я не использую это, мне нужно добавить еще 20 компонентов в тест, который напрямую не используется тестовым компонентом.

В этом сценарии, как я могу определить наличие дочерних компонентов при тестировании?

Вот мой html:

<section>
    <div class="inner-section">
        <div class="sub-title">
            <h2>Subsystem</h2>
            <button data-testid="btn-addRow" (click)="addRow()" [disabled]="crud.isCreate || crud.isUpdate">
                Add Subsystem
            </button>
        </div>

        <shell-page-navigation
        data-testid="page-navigation"
        *ngIf="pagination"
        [configPageNator]="pageNavigator"
        (pageOptionChange)="pageOptionChange($event)"
        (turnPage)="turnPage($event)"
        (setCurrentPage)="setCurrentPage($event)"
        ></shell-page-navigation>

        <form data-testid="form-subsystem" [formGroup]="createForm" (ngSubmit)="onCreateFormSubmit()" >
            <span class="form-indication"><sup>*</sup> Indicates required</span>
            <!-- <span class="error-resubmit" *ngIf="reSubmits">These fields have errors which need to be resolved before you can submit the configuration</span> -->

            <table class="ibo-table">
                <thead>
                    <tr class="row-head">
                        <th *ngFor="let col of columns;" [ngClass]="col.class">
                            {{col.title}}
                            <ng-container *ngIf="col.data === 'Name'"><sup>*</sup></ng-container>
                            <div class="row-header-menu" *ngIf="col.filter">
                                <a href="#" class="row-head-menu" (click)="toggleFilterMenu(col.title); false"></a>
                                <div class="menu-content" *ngIf="showFilter === col.title">
                                    <ul class="header-menu">
                                        <li class="sort-group divide">
                                            <label>
                                                <input type="radio" value="asc" aria-label="Sort by ascending order" (change)="setSort($event.target.value, col.data)" name="sortOrder">
                                                <span class="ibo-radio" [ngClass]="{'active':position && sortBy === col.data}"></span>
                                                Sort by ascending order
                                            </label>
                                        </li>
                                        <li class="sort-group divide">
                                            <label>
                                                <input type="radio" value="des" aria-label="Sort by descending order" (change)="setSort($event.target.value, col.data)" name="sortOrder">
                                                <span class="ibo-radio" [ngClass]="{'active':!position && sortBy === col.data }"></span>
                                                Sort by descending order
                                            </label>
                                        </li>

                                        <li class="ibo-btn">
                                            <button (click)="applyFilterOptions(); false"  aria-label="Apply"  class="ibo-btn-true">Apply</button>
                                            <button (click)="toggleFilterMenu(''); false"  aria-label="Cancel"  class="ibo-btn-false">Cancel</button>
                                        </li>
                                    </ul>
                                </div>
                            </div>
                        </th>
                    </tr>
                </thead>

                <tbody>
                    <tr *ngIf="crud.isCreate" [ngClass]="{'create' : crud?.isCreate}">
                        <td *ngFor="let col of columns;" [ngClass]="col.class">
                            <ng-container *ngIf="col.data !== ''">
                                <input id="name" type="text"
                                class="form-control"
                                formControlName={{col.data}}
                                placeholder={{col.placeholder}}>
                                <control-messages [control]="createForm.controls[col.data]"></control-messages>
                            </ng-container>
                            <ng-container *ngIf="col.data === ''">
                                <button type="submit" aria-label="Ok" class="apply-item"></button>
                                <button type="button" aria-label="Cancel" (click)="confirmCancel(); false" class="cancel-apply"></button>
                            </ng-container>
                        </td>
                    </tr>
                    <tr *ngFor="let data of subsystems;" [ngClass]="{'edit': data.Id === crud.updateData?.Id}">
                        <td *ngFor="let col of columns" [ngClass]="col.class">
                            <ng-container *ngIf="data.Id !== crud.updateData?.Id">
                                <ng-container *ngIf="col.data !== ''">
                                    {{getData(data,col.data)}}
                                </ng-container>
                                <ng-container *ngIf="col.data === ''">
                                    <button type="button" aria-label="Edit" class="edit-item" (click)="editSubsytem(data)"></button>
                                    <button type="button" aria-label="Delete" *ngIf="data.IsDeletePossible" (click)="confirmRemoveSubsytem(data); false" class="delete-item"></button>
                                </ng-container>
                            </ng-container>
                            <ng-container *ngIf="data.Id === crud.updateData?.Id">
                                <ng-container *ngIf="col.data !== ''">
                                    <input id="name" type="text" class="form-control"
                                    formControlName={{col.data}} placeholder={{col.title}}>
                                    <control-messages [control]="createForm.controls[col.data]"></control-messages>
                                </ng-container>
                                <ng-container *ngIf="col.data === ''">
                                    <button type="submit" aria-label="Ok" class="apply-item"></button>
                                    <button type="button" aria-label="Cancel" (click)="confirmCancel(); false" class="cancel-apply"></button>
                                </ng-container>
                            </ng-container>

                        </td>
                    </tr>
                </tbody>
            </table>
        </form>

        <shell-page-navigation
        *ngIf="pagination"
        [configPageNator]="pageNavigator"
        (pageOptionChange)="pageOptionChange($event)"
        (turnPage)="turnPage($event)"
        (setCurrentPage)="setCurrentPage($event)"
        ></shell-page-navigation>

    </div>
</section>

<shell-modals-popup [popConfig]='popConfig' (reset)="reset()"></shell-modals-popup>

мой тестовый файл спецификации:

import { render } from '@testing-library/angular';
import { getByTestId } from '@testing-library/dom';

import { CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
import { HttpClientTestingModule } from '@angular/common/http/testing';
import { SubSystemComponent } from './sub-system.component';
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
import { StoreModule } from '@ngrx/store';
import { reducer } from './../../../state/reducers/reducer';
import { EffectsModule } from '@ngrx/effects';
import { EffectsSubSystem } from './../../../state/effects/subSystem.effects';
import { ModelSubSystem } from './../../../models/model.subSystem';

test('renders the current value', async () => {

    const data1 = {
        Id: 1,
        Name: 'subsystem1',
        IsDeletePossible: true,
        CreatedBy: '',
        CreatedDate: new Date(),
        UpdatedBy: '',
        UpdatedDate: new Date(),
        UpdatedByName: '',
        CreatedByName: ''
    } as ModelSubSystem;

    const data2 = {
        Id: 2,
        Name: 'subsystem2',
        IsDeletePossible: true,
        CreatedBy: '',
        CreatedDate: new Date(),
        UpdatedBy: '',
        UpdatedDate: new Date(),
        UpdatedByName: '',
        CreatedByName: ''
    } as ModelSubSystem;

    const component = await render(SubSystemComponent, {

        schemas: [CUSTOM_ELEMENTS_SCHEMA],
        imports: [
            HttpClientTestingModule,
            FormsModule,
            ReactiveFormsModule,
            StoreModule.forRoot({}, { runtimeChecks: { strictStateImmutability: true, strictActionImmutability: true } }),
            StoreModule.forFeature('pfservice', reducer),
            EffectsModule.forRoot([]),
            EffectsModule.forFeature([EffectsSubSystem])
        ],
        componentProperties: {
            subsystems : [data1, data2]
        }

    });

    const container = document.body;
    expect(getByTestId(container, 'btn-addRow')).toBeTruthy();
    expect(getByTestId(container, 'page-navigation')).toBeTruthy(); //fails..
    expect(getByTestId(container, 'form-subsystem')).toBeTruthy();

});

Кто-нибудь, пожалуйста, помогите мне?

2 ответа

Решение

Вам нужно будет добавить в объявления дочерний компонент:


    const component = await render(SubSystemComponent, {
        declarations: [ChildComponent],
        ...
    });

Есть отличная библиотека для ваших нужд: ng-mocks

Я использую его, и очень легко имитировать ваши компоненты и играть с ними.

Для имитации всех ваших дочерних компонентов вам просто понадобится следующее:

MockComponents(ChildComponent1, ChildComponent2, ChildComponent3 ...),

И чтобы пойти дальше, вы можете найти ниже общую функцию, чтобы легко получить нужный вам компонент:

export function getComponent<T>(fixture: ComponentFixture<any>, comp: Type<T>, index: number = 0): T {
  return fixture.debugElement
    .queryAll(By.directive(comp))[index]
    .componentInstance as T;
}

поэтому в своем тесте вы можете:

getComponent(fixture, Component1).valueChangeB.emit( XXX )

expect(getComponent(fixture, Component1).valueA).toBe('some-value')
Другие вопросы по тегам