gpt4 book ai didi

unit-testing - 从父组件传递到子组件的 Angular 2 单元测试数据

转载 作者:太空狗 更新时间:2023-10-29 17:24:52 25 4
gpt4 key购买 nike

我对我看到的(极少数)测试从父组件传递到子组件的数据的方式有疑问。目前,在 Angular2 docs ,他们正在通过检查子组件的 dom 值 来测试数据是否已从父组件传递到子组件。我对这种方法的问题是它强制父规范知道子组件的 html 结构。父组件的工作只是将数据传递给子组件。一个例子...

我有一个故事组件如下:

'use strict';

import {Component, OnInit, Input} from '@angular/core';
import {StoryService} from '../../services/story.service';
import {StoryModel} from '../../models/story-model';
import {AlbumCover} from './album-cover/album-cover';
import {Author} from "./author/author";
import {StoryDuration} from "./story-duration/story-duration";

@Component({
selector: 'story',
templateUrl: 'build/components/story/story.html',
providers: [StoryService],
directives: [AlbumCover, Author, StoryDuration]
})

export class Story implements OnInit {
@Input('id') id:number;
public story:StoryModel;

constructor(private storyService:StoryService) {}

ngOnInit() {
this.getStory();
}

private getStory() {
this.storyService.getStory(this.id).subscribe(story => this.story = story);
}
}

注意它有一个 AlbumCover directives 中的组件依赖性@Component 中的数组装饰器。

这是我的故事模板:

<div *ngIf="story">
<album-cover [image]="story.albumCover" [title]="story.title"></album-cover>
<div class="author-duration-container">
<author [avatar]="story.author.avatar" [name]="story.author.name"></author>
<story-duration [word-count]="story.wordCount"></story-duration>
</div>
</div>

注意 <album-cover [image]="story.albumCover" [title]="story.title"></album-cover>我绑定(bind) story.albumCover 的行来自 Story Controller 到 image AlbumCover 的属性(property).这一切都很完美。现在进行测试:

import {provide} from '@angular/core';
import {beforeEach, beforeEachProviders, describe, expect, injectAsync, it, setBaseTestProviders, resetBaseTestProviders} from '@angular/core/testing';
import {HTTP_PROVIDERS} from '@angular/http';
import {BROWSER_APP_DYNAMIC_PROVIDERS} from "@angular/platform-browser-dynamic";
import {TEST_BROWSER_STATIC_PLATFORM_PROVIDERS, ADDITIONAL_TEST_BROWSER_PROVIDERS} from '@angular/platform-browser/testing';
import {ComponentFixture, TestComponentBuilder} from '@angular/compiler/testing';
import {Observable} from 'rxjs/Observable';

// TODO: this pattern of importing 'of' can probably go away once rxjs is fixed
// https://github.com/ReactiveX/rxjs/issues/1713
import 'rxjs/add/observable/of';

resetBaseTestProviders();
setBaseTestProviders(
TEST_BROWSER_STATIC_PLATFORM_PROVIDERS,
[BROWSER_APP_DYNAMIC_PROVIDERS, ADDITIONAL_TEST_BROWSER_PROVIDERS]
);

import {Story} from './story';
import {StoryModel} from '../../models/story-model';
import {StoryService} from '../../services/story.service';

var mockStory = {
id: 1,
title: 'Benefit',
albumCover: 'images/placeholders/story-4.jpg',
author: {
id: 2,
name: 'Brett Beach',
avatar: 'images/placeholders/author-1.jpg'
},
wordCount: 4340,
content: '<p>This is going to be a great book! I <strong>swear!</strong></p>'
};

class MockStoryService {
public getStory(id):Observable<StoryModel> {
return Observable.of(mockStory);
}
}

describe('Story', () => {
var storyFixture,
story,
storyEl;

beforeEachProviders(() => [
HTTP_PROVIDERS
]);

beforeEach(injectAsync([TestComponentBuilder], (tcb:TestComponentBuilder) => {
return tcb
.overrideProviders(Story, [
provide(StoryService, {
useClass: MockStoryService
})
])
.createAsync(Story)
.then((componentFixture:ComponentFixture<Story>) => {
storyFixture = componentFixture;
story = componentFixture.componentInstance;
storyEl = componentFixture.nativeElement;
componentFixture.detectChanges();
});
}));

describe(`ngOnInit`, () => {
describe(`storyService.getStory`, () => {
it(`should be called, and on success, set this.story`, () => {
spyOn(story.storyService, 'getStory').and.callThrough();
story.ngOnInit();
expect(story.storyService.getStory).toHaveBeenCalled();
expect(story.story.title).toBe('Benefit');
});
});
});

it('should not show the story component if story does not exist', () => {
story.story = null;
storyFixture.detectChanges();
expect(storyEl.children.length).toBe(0);
});

it('should show the story component if story exists', () => {
story.story = mockStory;
storyFixture.detectChanges();
expect(storyEl.children.length).not.toBe(0);
});

describe('story components', () => {
beforeEach(() => {
story.story = mockStory;
storyFixture.detectChanges();
});

describe('album cover', () => {
var element,
img;

beforeEach(() => {
element = storyEl.querySelector('album-cover');
img = element.querySelector('img');
});

it(`should be passed the story albumCover and title to the album cover component`, () => {
expect(img.attributes.src.value).toBe(mockStory.albumCover);
expect(img.attributes.alt.value).toBe(mockStory.title);
});
});

describe('author', () => {
var element,
img,
nameEl;

beforeEach(() => {
element = storyEl.querySelector('author');
img = element.querySelector('img');
nameEl = element.querySelector('.name');
});

it(`should be passed the author name and avatar`, () => {
expect(img.attributes.src.value).toBe(story.story.author.avatar);
expect(img.attributes.alt.value).toBe(story.story.author.name);
expect(nameEl.innerText).toBe(story.story.author.name);
});
});

describe('story duration', () => {
var element;

beforeEach(() => {
element = storyEl.querySelector('.story-duration');
});

it(`should be passed the word count to generate the total read time`, () => {
story.story.wordCount = 234234;
storyFixture.detectChanges();
expect(element.innerText).toBe(`852 min read`);
});
});
});
});

看看我的describe('album cover'... .我通过这种期望的方式是我找到了 <album-cover>元素,然后找到 <img>在其中标记,然后检查 <img>的 DOM 属性。对我来说,这个期望应该在 album-cover.spec.ts 内- 不是 story.spec.ts .

我的问题是:有没有一种方法可以在不依赖读取 dom 值的情况下测试父组件是否将数据传递给子组件?

最佳答案

您可以使用overrideTemplate 传递一个仅用于测试的 View 。

   return tcb
.overrideTemplate(AlbumCover, '<div>{{valueFromParent}}</div>')
.overrideProviders(Story, [

关于unit-testing - 从父组件传递到子组件的 Angular 2 单元测试数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37982544/

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