4325 lines
192 KiB
Plaintext
4325 lines
192 KiB
Plaintext
block includes
|
||
include ../_util-fns
|
||
|
||
:marked
|
||
This guide offers tips and techniques for testing Angular applications.
|
||
Though this page includes some general testing principles and techniques,
|
||
the focus is on testing applications written with Angular.
|
||
|
||
本章提供了一些测试Angular应用的提示和技巧。虽然这里讲述了一些常规测试理念和技巧,但是其重点是测试用Angular编写的应用。
|
||
|
||
a#top
|
||
:marked
|
||
# Contents
|
||
|
||
# 目录
|
||
|
||
* [Live examples](#live-examples "Live examples of the tests in this guide")
|
||
|
||
[在线例子](#live-examples "本章这些测试的在线例子")
|
||
|
||
* [Introduction to Angular testing](#testing-intro)
|
||
|
||
[Angular测试简介](#testing-intro)
|
||
|
||
* [Tools and technologies](#tools-and-tech)
|
||
|
||
[工具与技术](#tools-and-tech)
|
||
|
||
* [Setup](#setup)
|
||
|
||
[环境设置](#setup)
|
||
|
||
* [Isolated unit tests vs. the Angular testing utilities](#isolated-v-testing-utilities)
|
||
|
||
[独立的单元测试 vs. Angular测试工具集](#isolated-v-testing-utilities)
|
||
|
||
|
||
* [The first karma test](#1st-karma-test)
|
||
|
||
[第一个Karma测试](#1st-karma-test)
|
||
|
||
* [Run with karma](#run-karma)
|
||
|
||
[运行Karma](#run-karma)
|
||
|
||
* [Test debugging](#test-debugging)
|
||
|
||
[调试测试代码](#test-debugging)
|
||
|
||
* [Try the live example](#live-karma-example)
|
||
|
||
[试用在线例子](#live-karma-example)
|
||
|
||
|
||
* [Test a component](#simple-component-test)
|
||
|
||
[测试一个组件](#simple-component-test)
|
||
|
||
* [`TestBed`](#testbed)
|
||
|
||
[`TestBed`(测试台)](#testbed)
|
||
|
||
* [`createComponent`](#create-component)
|
||
|
||
* [`ComponentFixture`, `DebugElement`, and `query(By.css)`](#component-fixture)
|
||
|
||
[`ComponentFixture`, `DebugElement`, 和 `query(By.css)`](#component-fixture)
|
||
|
||
* [The tests](#the-tests)
|
||
|
||
[测试](#the-tests)
|
||
|
||
* [`detectChanges`: Angular change detection within a test](#detect-changes)
|
||
|
||
[`detectChanges`: 在测试中控制Angular的变更检测](#detect-changes)
|
||
|
||
* [Try the live example](#try-example)
|
||
|
||
[试试在线例子](#try-example)
|
||
|
||
* [Automatic change detection](#auto-detect-changes)
|
||
|
||
[自动变更检测](#auto-detect-changes)
|
||
|
||
|
||
* [Test a component with an external template](#component-with-external-template)
|
||
|
||
[测试带有外部模板的组件](#component-with-external-template)
|
||
|
||
* [The first asynchronous `beforeEach`](#async-in-before-each)
|
||
|
||
[第一处异步代码:`beforeEach`](#async-in-before-each)
|
||
|
||
* [`compileComponents`](#compile-components)
|
||
|
||
* [The second synchronous `beforeEach`](#second-before-each)
|
||
|
||
[第二处同步代码:`beforeEach`](#second-before-each)
|
||
|
||
* [Waiting for `compileComponents`](#waiting-compile-components)
|
||
|
||
[等待 `compileComponents`](#waiting-compile-components)
|
||
|
||
* [Try the live example](#live-external-template-example)
|
||
|
||
[试试在线例子](#live-external-template-example)
|
||
|
||
|
||
* [Test a component with a service dependency](#component-with-dependency)
|
||
|
||
[测试依赖服务的组件](#component-with-dependency)
|
||
|
||
* [Provide service test doubles](#service-test-doubles)
|
||
|
||
[测试替身](#service-test-doubles)
|
||
|
||
* [Get injected services](#get-injected-service)
|
||
|
||
[获取注入的服务](#get-injected-service)
|
||
|
||
* [`TestBed.get`](#testbed-get)
|
||
|
||
* [Always get the service from an injector](#service-from-injector)
|
||
|
||
[总是从注入器中取得服务](#service-from-injector)
|
||
|
||
* [Final setup and tests](#welcome-spec-setup)
|
||
|
||
[最终设置与测试](#welcome-spec-setup)
|
||
|
||
|
||
* [Test a component with an async service](#component-with-async-service)
|
||
|
||
[测试带有异步服务的组件](#component-with-async-service)
|
||
|
||
* [Spying on the real service](#service-spy)
|
||
|
||
[监听真实的服务](#service-spy)
|
||
|
||
* [Synchronous tests](#sync-tests)
|
||
|
||
[同步测试](#sync-tests)
|
||
|
||
* [The `async` funciton in it](#async)
|
||
|
||
[`it`中的`async`函数](#async)
|
||
|
||
* [`whenStable`](#when-stable)
|
||
|
||
* [The `fakeAsync` function](#fake-async)
|
||
|
||
[`fakeAsync` 函数](#fake-async)
|
||
|
||
* [The `tick` function](#tick)
|
||
|
||
[`tick`函数](#tick)
|
||
|
||
* [`jasmine.done`](#jasmine-done)
|
||
|
||
* [Test a component with inputs and outputs](#component-with-input-output)
|
||
|
||
[测试带有输入属性和输出属性的组件](#component-with-input-output)
|
||
|
||
* [Test `DashboardHeroComponent` stand-alone](#dashboard-standalone)
|
||
|
||
[单独测试`DashboardHeroComponent`](#dashboard-standalone)
|
||
|
||
* [`triggerEventHandler`](#trigger-event-handler)
|
||
|
||
* [Test a component inside a test host component](#component-inside-test-host)
|
||
|
||
[在宿主组件中测试组件](#component-inside-test-host)
|
||
|
||
* [Test a routed component](#routed-component)
|
||
|
||
[测试路由组件](#routed-component)
|
||
|
||
* [The `inject` helper function](#inject)
|
||
|
||
[`inject`助手函数](#inject)
|
||
|
||
* [Test a routed component with parameters](#routed-component-w-param)
|
||
|
||
[测试带参数的路由组件](#routed-component-w-param)
|
||
|
||
* [Create an `Observable` test double](#stub-observable)
|
||
|
||
[创建`Observable`测试桩(Stub)](#stub-observable)
|
||
|
||
* [Testing with the `Observable` test double](#tests-w-observable-double)
|
||
|
||
[使用`Observable`测试桩进行测试](#tests-w-observable-double)
|
||
|
||
* [Use a `page` object to simplify setup](#page-object)
|
||
|
||
[使用`page`对象来简化设置](#page-object)
|
||
|
||
* [Set up with module imports](#import-module)
|
||
|
||
[设置导入了其它模块的模块](#import-module)
|
||
|
||
* [Import the feature module](#feature-module-import)
|
||
|
||
[导入特性模块](#feature-module-import)
|
||
|
||
* [Override a component's providers](#component-override)
|
||
|
||
[改写(Override)组件的提供商](#component-override)
|
||
|
||
* [The `overrideComponent` method](#override-component-method)
|
||
|
||
[`overrideComponent`方法](#override-component-method)
|
||
|
||
* [Provide a _spy-stub (HeroDetailServiceSpy)_](#spy-stub)
|
||
|
||
[提供一个 *监听桩*(`HeroDetailServiceSpy`)](#spy-stub)
|
||
|
||
* [The override tests](#override-tests)
|
||
|
||
[改写测试](#override-tests)
|
||
|
||
* [More overrides](#more-overrides)
|
||
|
||
[更多的改写](#more-overrides)
|
||
|
||
|
||
* [Test a `RouterOutlet` component](#router-outlet-component)
|
||
|
||
[测试`RouterOutlet`组件](#router-outlet-component)
|
||
|
||
* [Stubbing unneeded components](#stub-component)
|
||
|
||
[模拟不需要的组件](#stub-component)
|
||
|
||
* [Stubbing the `RouterLink`](#router-link-stub)
|
||
|
||
[模拟`RouterLink`](#router-link-stub)
|
||
|
||
* [`By.directive` and injected directives](#by-directive)
|
||
|
||
[`By.directive`与注入的指令](#by-directive)
|
||
|
||
* [What good are these tests?](#why-stubbed-routerlink-tests)
|
||
|
||
[为什么要这样写测试?](#why-stubbed-routerlink-tests)
|
||
|
||
|
||
* ["Shallow component tests" with *NO_ERRORS_SCHEMA*](#shallow-component-test)
|
||
|
||
[使用`NO_ERRORS_SCHEMA`进行 "浅组件测试" ](#shallow-component-test)
|
||
|
||
* [Test an attribute directive](#attribute-directive)
|
||
|
||
[测试属性型指令](#attribute-directive)
|
||
|
||
* [Isolated unit tests](#isolated-unit-tests "Unit testing without the Angular testing utilities")
|
||
|
||
[独立的单元测试](#isolated-unit-tests "不使用Angular测试工具集的单元测试方式")
|
||
|
||
* [Services](#isolated-service-tests)
|
||
|
||
[服务](#isolated-service-tests)
|
||
|
||
* [Services with dependencies](#services-with-dependencies)
|
||
|
||
[带依赖的服务](#services-with-dependencies)
|
||
|
||
* [Pipes](#isolated-pipe-tests)
|
||
|
||
[管道](#isolated-pipe-tests)
|
||
|
||
* [Write Angular tests too](#write-tests)
|
||
|
||
[写Angular测试](#write-tests)
|
||
|
||
* [Components](#isolated-component-tests)
|
||
|
||
[组件](#isolated-component-tests)
|
||
|
||
|
||
* [Angular testing utility APIs](#atu-apis)
|
||
|
||
[Angular单元测试工具类 API ](#atu-apis)
|
||
|
||
* [`TestBed` class summary](#testbed-class-summary)
|
||
|
||
[`TestBed`总结](#testbed-class-summary)
|
||
|
||
* [The `ComponentFixture`](#component-fixture-api-summary)
|
||
|
||
[`ComponentFixture`总结](#component-fixture-api-summary)
|
||
|
||
* [`ComponentFixture` properties](#component-fixture-properties)
|
||
|
||
[`ComponentFixture`的属性](#component-fixture-properties)
|
||
|
||
* [The `ComponentFixture` methods](#component-fixture-methods)
|
||
|
||
[`ComponentFixture`方法](#component-fixture-methods)
|
||
|
||
* [`DebugElement`](#debug-element-details)
|
||
|
||
[`DebugElement`详情](#debug-element-details)
|
||
|
||
|
||
* [Test environment setup files](#setup-files)
|
||
|
||
[测试环境设置所需的文件](#setup-files)
|
||
|
||
* [npm packages](#npm-packages)
|
||
|
||
[npm包](#npm-packages)
|
||
|
||
|
||
* [FAQ: Frequently asked questions](#faq "Frequently asked questions")
|
||
|
||
[常见问题(FAQ)](#faq "常见问题")
|
||
|
||
:marked
|
||
It’s a big agenda. Fortunately, you can learn a little bit at a time and put each lesson to use.
|
||
|
||
以上主题繁多。幸运的是,你可以慢慢地阅读并立刻应用每一个主题。
|
||
|
||
## Live examples
|
||
|
||
## 在线例子
|
||
|
||
This guide presents tests of a sample application that is much like the [_Tour of Heroes_ tutorial](../tutorial).
|
||
The sample application and all tests in this guide are available as live examples for inspection, experiment, and download:
|
||
|
||
这篇指南会展示一个范例应用的所有测试,这个范例应用和[《英雄指南》教程](../tutorial)非常像。
|
||
本章中的这个范例应用及其所有测试都有在线例子,以供查看、试验和下载。
|
||
|
||
* <live-example plnkr="1st-specs" embedded-style>A spec to verify the test environment</live-example>.
|
||
|
||
<live-example plnkr="1st-specs" embedded-style>用于验证测试环境的规约</live-example>。
|
||
|
||
* <live-example plnkr="banner-inline-specs" embedded-style>The first component spec with inline template</live-example>.
|
||
|
||
<live-example plnkr="banner-inline-specs" embedded-style>第一个带内联模板的组件规约</live-example>。
|
||
|
||
* <live-example plnkr="banner-specs" embedded-style>A component spec with external template</live-example>.
|
||
|
||
<live-example plnkr="banner-specs" embedded-style>带外部模板的组件规约</live-example>。
|
||
|
||
* <live-example name="setup" plnkr="quickstart-specs" embedded-style>The QuickStart seed's AppComponent spec</live-example>.
|
||
|
||
<live-example name="setup" plnkr="quickstart-specs" embedded-style>快速起步种子工程的`AppComponent`规约</live-example>。
|
||
|
||
* <live-example embedded-style>The sample application to be tested</live-example>.
|
||
|
||
<live-example embedded-style>所要测试的范例应用</live-example>。
|
||
|
||
* <live-example plnkr="app-specs" embedded-style>All specs that test the sample application</live-example>.
|
||
|
||
<live-example plnkr="app-specs" embedded-style>本范例应用的所有规约</live-example>。
|
||
|
||
* <live-example plnkr="bag-specs" embedded-style>A grab bag of additional specs</live-example>.
|
||
|
||
<live-example plnkr="bag-specs" embedded-style>其它规约汇总</live-example>。
|
||
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
a#testing-intro
|
||
:marked
|
||
## Introduction to Angular Testing
|
||
|
||
## Angular测试入门
|
||
|
||
This page guides you through writing tests to explore
|
||
and confirm the behavior of the application. Testing
|
||
does the following:
|
||
|
||
本章教你如何编写测试程序来探索和确认应用的行为。测试的作用有:
|
||
|
||
1. Guards against changes that break existing code (“regressions”).
|
||
|
||
测试**守护**由于代码变化而打破已有代码(“回归”)的情况。
|
||
|
||
1. Clarifies what the code does both when used as intended and when faced with deviant conditions.
|
||
|
||
不管代码被正确使用还是错误使用,测试程序起到**澄清**代码的作用。
|
||
|
||
1. Reveals mistakes in design and implementation.
|
||
Tests shine a harsh light on the code from many angles.
|
||
When a part of the application seems hard to test, the root cause is often a design flaw,
|
||
something to cure now rather than later when it becomes expensive to fix.
|
||
|
||
测试程序**暴露**设计和实现可能出现的错误。测试程序从很多角度为代码亮出警报灯。当应用程序很难被测试时,
|
||
其根本原因一般都是设计缺陷,这种缺陷最好立刻被修正,不要等到它变得很难被修复的时候才行动。
|
||
|
||
This chapter assumes that you know something about testing. Don't worry if you don't.
|
||
There are plenty of books and online resources to get up to speed.
|
||
|
||
本章假设你熟悉测试。但是如果你不熟悉也没有关系。有很多书本和在线资源可以帮助你。
|
||
|
||
<!-- TODO
|
||
:marked
|
||
### Learn more
|
||
Learn more about basic Jasmine testing here
|
||
[Resources TBD](./#)
|
||
-->
|
||
|
||
a#tools-and-tech
|
||
:marked
|
||
### Tools and technologies
|
||
|
||
### 工具与技术
|
||
|
||
You can write and run Angular tests with a variety of tools and technologies.
|
||
This guide describes specific choices that are known to work well.
|
||
|
||
你可以用多种工具和技术来编写和运行Angular测试程序。本章介绍了一些大家已经知道能良好工作的选择。
|
||
|
||
table(width="100%")
|
||
col(width="20%")
|
||
col(width="80%")
|
||
tr
|
||
th
|
||
p Technology
|
||
p 技术
|
||
th
|
||
p Purpose
|
||
p 目的
|
||
tr(style=top)
|
||
td(style="vertical-align: top") Jasmine
|
||
td
|
||
:marked
|
||
The [Jasmine test framework](http://jasmine.github.io/2.4/introduction.html)
|
||
provides everything needed to write basic tests.
|
||
It ships with an HTML test runner that executes tests in the browser.
|
||
|
||
[Jasmine测试框架](http://jasmine.github.io/2.4/introduction.html)提供了所有编写基本测试的工具。
|
||
它自带HTML测试运行器,用来在浏览器中执行测试程序。
|
||
|
||
tr(style=top)
|
||
td(style="vertical-align: top")
|
||
p Angular testing utilities
|
||
p Angular测试工具
|
||
td
|
||
:marked
|
||
Angular testing utilities create a test environment
|
||
for the Angular application code under test.
|
||
Use them to condition and control parts of the application as they
|
||
interact _within_ the Angular environment.
|
||
|
||
Angular测试工具为被测试的Angular应用代码创建测试环境。在应用代码与Angular环境互动时,使用Angular测试工具来限制和控制应用的部分代码。
|
||
|
||
tr(style=top)
|
||
td(style="vertical-align: top") Karma
|
||
td
|
||
:marked
|
||
The [karma test runner](https://karma-runner.github.io/1.0/index.html)
|
||
is ideal for writing and running unit tests while developing the application.
|
||
It can be an integral part of the project's development and continuous integration processes.
|
||
This guide describes how to set up and run tests with karma.
|
||
|
||
[karma测试运行器](https://karma-runner.github.io/1.0/index.html)是在开发应用的过程中
|
||
编写和运行单元测试的理想工具。
|
||
它能成为项目开发和连续一体化进程的不可分割的一部分。本章讲述了如何用Karma设置和运行测试程序。
|
||
tr(style=top)
|
||
td(style="vertical-align: top") Protractor
|
||
td
|
||
:marked
|
||
Use protractor to write and run _end-to-end_ (e2e) tests.
|
||
End-to-end tests explore the application _as users experience it_.
|
||
In e2e testing, one process runs the real application
|
||
and a second process runs protractor tests that simulate user behavior
|
||
and assert that the application respond in the browser as expected.
|
||
|
||
使用`Protractor`来编写和运行_端对端(e2e)_测试程序。端对端测试程序**像用户体验应用程序那样**探索它。
|
||
在端对端测试中,一条进程运行真正的应用,另一条进程运行Protractor测试程序,模拟用户行为,判断应用在浏览器中的反应是否正确。
|
||
|
||
a#setup
|
||
:marked
|
||
### Setup
|
||
|
||
### 环境设置
|
||
|
||
There are two fast paths to getting started with unit testing.
|
||
|
||
要开始单元测试,有两条捷径:
|
||
|
||
1. Start a new project following the instructions in [Setup](setup.html "Setup").
|
||
|
||
遵循[环境设置](setup.html "环境设置")中给出的步骤开始一个新项目。
|
||
|
||
1. Start a new project with the
|
||
<a href="https://github.com/angular/angular-cli/blob/master/README.md" target="_blank" title="Angular CLI">Angular CLI</a>.
|
||
|
||
使用[Angular CLI](https://github.com/angular/angular-cli/blob/master/README.md)创建新的项目。
|
||
|
||
Both approaches install npm packages, files, and scripts pre-configured for applications
|
||
built in their respective modalities.
|
||
Their artifacts and procedures differ slightly but their essentials are the same
|
||
and there are no differences in the test code.
|
||
|
||
以上两种方法都安装在各自的模式下为应用预先配置的**npm包、文件和脚本**。它们的文件和规程有一点不同,但是它们的核心部分是一样的,并且在测试代码方面没有任何区别。
|
||
|
||
In this guide, the application and its tests are based on the [setup instructions](setup.html "Setup").
|
||
For a discussion of the unit testing setup files, [see below](#setup-files).
|
||
|
||
本章中,该应用及其测试都是基于[环境设置步骤](setup.html "Setup")的。
|
||
对单元测试的环境设置文件的讨论,[参见后面](#setup-files)。
|
||
|
||
a#isolated-v-testing-utilities
|
||
:marked
|
||
### Isolated unit tests vs. the Angular testing utilites
|
||
|
||
### 独立单元测试 vs. Angular测试工具集
|
||
|
||
[Isolated unit tests](#isolated-unit-tests "Unit testing without the Angular testing utilities")
|
||
examine an instance of a class all by itself without any dependence on Angular or any injected values.
|
||
The tester creates a test instance of the class with `new`, supplying test doubles for the constructor parameters as needed, and
|
||
then probes the test instance API surface.
|
||
|
||
[独立单元测试](#isolated-unit-tests "不使用Angular测试工具集的单元测试")用于测试那些完全不依赖Angular或不需要注入值的类实例。
|
||
测试程序是所有`new`创建该类的实例,为构造函数参数提供所需的测试替身,然后测试该实例的API接口。
|
||
|
||
*You should write isolated unit tests for pipes and services.*
|
||
|
||
*我们应该为管道和服务书写独立单元测试。*
|
||
|
||
You can test components in isolation as well.
|
||
However, isolated unit tests don't reveal how components interact with Angular.
|
||
In particular, they can't reveal how a component class interacts with its own template or with other components.
|
||
|
||
我们也同样可以对组件写独立单元测试。
|
||
不过,独立单元测试无法体现组件与Angular的交互。
|
||
具体来说,就是不能发现组件类如何与它的模板或其它组件交互。
|
||
|
||
Such tests require the **Angular testing utilities**.
|
||
The Angular testing utilities include the `TestBed` class and several helper functions from `@angular/core/testing`.
|
||
They are the main focus of this guide and you'll learn about them
|
||
when you write your [first component test](#simple-component-test).
|
||
A comprehensive review of the Angular testing utilities appears [later in this guide](#atu-apis).
|
||
|
||
这时你需要*Angular测试工具集*。
|
||
Angular测试工具集包括`TestBed`类和一些来自`@angular/core/testing`的助手函数。
|
||
本章将会重点讲解它们,通过[第一个组件测试](#simple-component-test)来讲解。
|
||
[本章稍后的部分](#atu-apis)将展示Angular测试工具集的全貌。
|
||
|
||
But first you should write a dummy test to verify that your test environment is set up properly
|
||
and to lock in a few basic testing skills.
|
||
|
||
但首先,我们要先随便写一个测试来验证测试环境是否已经就绪了,并掌握一些基础的测试技术。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#1st-karma-test
|
||
:marked
|
||
## The first karma test
|
||
|
||
## 第一个`karma`测试
|
||
|
||
Start with a simple test to make sure that the setup works properly.
|
||
|
||
编写简单的测试程序,来确认以上的配置是否工作正常。
|
||
|
||
Create a new file called `1st.spec.ts` in the application root folder, `src/app/`
|
||
|
||
在应用的根目录`app/`创建新文件,名叫`1st.spec.ts`。
|
||
|
||
.alert.is-important
|
||
:marked
|
||
Tests written in Jasmine are called _specs_ .
|
||
**The filename extension must be `.spec.ts`**,
|
||
the convention adhered to by `karma.conf.js` and other tooling.
|
||
|
||
用Jasmine编写的测试程序都被叫做**specs**。**文件名后缀必须是`.spec.ts`**,这是`karma.conf.js`和其它工具所坚持和遵守的规约。
|
||
|
||
:marked
|
||
**Put spec files somewhere within the `src/app/` folder.**
|
||
The `karma.conf.js` tells karma to look for spec files there,
|
||
for reasons explained [below](#q-spec-file-location).
|
||
|
||
**将测试程序spec放到`app/`文件夹下的任何位置。**
|
||
`karma.conf.js`告诉`Karma`在这个文件夹中寻找测试程序spec文件,原因在 [这里](#spec-file-location) 有所解释。
|
||
|
||
Add the following code to `src/app/1st.spec.ts`.
|
||
|
||
添加下面的代码到`app/1st.spec.ts`。
|
||
+makeExample('testing/ts/src/app/1st.spec.ts', '', 'src/app/1st.spec.ts')(format='.')
|
||
|
||
a#run-karma
|
||
:marked
|
||
### Run with karma
|
||
|
||
### 运行Karma
|
||
|
||
Compile and run it in karma from the command line using the following command:
|
||
|
||
使用下面的命令从命令行中编译并在`Karma`中运行上面的测试程序。
|
||
|
||
code-example(format="." language="bash").
|
||
npm test
|
||
:marked
|
||
The command compiles the application and test code and starts karma.
|
||
Both processes watch pertinent files, write messages to the console, and re-run when they detect changes.
|
||
|
||
该命令编译应用及其测试代码,并启动Karma。
|
||
两个进程都监视相关文件,往控制台输入信息和检测到变化时自动重新运行。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The documentation setup defines the `test` command in the `scripts` section of npm's `package.json`.
|
||
The Angular CLI has different commands to do the same thing. Adjust accordingly.
|
||
|
||
《快速起步》在npm的`package.json`中的`scripts`里定义了`test`命令。
|
||
Angular CLI使用不同的命令来做同样的事情。对不同的环境采取不同的方案。
|
||
:marked
|
||
After a few moments, karma opens a browser and starts writing to the console.
|
||
|
||
等一小段时间后,Karma便打开浏览器并开始向控制台输出。
|
||
figure.image-display
|
||
img(src='/resources/images/devguide/testing/karma-browser.png' style="width:400px;" alt="Karma browser")
|
||
:marked
|
||
Hide (don't close!) the browser and focus on the console output, which
|
||
should look something like this:
|
||
|
||
隐藏(不要关闭)浏览器,查看控制台的输出,应该是这样的:
|
||
|
||
code-example(format="." language="bash").
|
||
> npm test
|
||
...
|
||
[0] 1:37:03 PM - Compilation complete. Watching for file changes.
|
||
...
|
||
[1] Chrome 51.0.2704: Executed 0 of 0 SUCCESS
|
||
Chrome 51.0.2704: Executed 1 of 1 SUCCESS
|
||
SUCCESS (0.005 secs / 0.005 secs)
|
||
|
||
:marked
|
||
Both the compiler and karma continue to run. The compiler output is preceded by `[0]`;
|
||
the karma output by `[1]`.
|
||
|
||
编译器和`Karma`都会持续运行。编译器的输入信息前面有`[0]`,`Karma`的输出前面有`[1]`。
|
||
|
||
Change the expectation from `true` to `false`.
|
||
|
||
将期望从`true`变换为`false`。
|
||
|
||
The _compiler_ watcher detects the change and recompiles.
|
||
|
||
**编译器**监视器检测到这个变化并重新编译。
|
||
|
||
code-example(format="." language="bash").
|
||
[0] 1:49:21 PM - File change detected. Starting incremental compilation...
|
||
[0] 1:49:25 PM - Compilation complete. Watching for file changes.
|
||
|
||
:marked
|
||
The _karma_ watcher detects the change to the compilation output and re-runs the test.
|
||
|
||
**`Karma`**监视器检测到编译器输出的变化,并重新运行测试。
|
||
|
||
code-example(format="." language="bash").
|
||
[1] Chrome 51.0.2704 1st tests true is true FAILED
|
||
[1] Expected false to equal true.
|
||
[1] Chrome 51.0.2704: Executed 1 of 1 (1 FAILED) (0.005 secs / 0.005 secs)
|
||
|
||
:marked
|
||
It fails of course.
|
||
|
||
正如所料,测试结果是**失败**。
|
||
|
||
Restore the expectation from `false` back to `true`.
|
||
Both processes detect the change, re-run, and karma reports complete success.
|
||
|
||
将期望从`false`恢复为`true`。两个进程都检测到这个变化,自动重新运行,`Karma`报告测试成功。
|
||
|
||
.alert.is-helpful
|
||
:marked
|
||
The console log can be quite long. Keep your eye on the last line.
|
||
When all is well, it reads `SUCCESS`.
|
||
|
||
控制台的日志可能会非常长。注意最后一样。当一切正常时,它会显示`SUCCESS`。
|
||
|
||
a#test-debugging
|
||
:marked
|
||
### Test debugging
|
||
|
||
### 调试测试程序
|
||
|
||
Debug specs in the browser in the same way thatyou debug an application.
|
||
|
||
在浏览器中,像调试应用一样调试测试程序spec。
|
||
|
||
1. Reveal the karma browser window (hidden earlier).
|
||
|
||
显示`Karma`的浏览器窗口(之前被隐藏了)。
|
||
|
||
1. Click the **DEBUG** button; it opens a new browser tab and re-runs the tests.
|
||
|
||
点击“DEBUG”按钮;它打开一页新浏览器标签并重新开始运行测试程序
|
||
|
||
1. Open the browser's “Developer Tools” (`Ctrl-Shift-I` on windows; `Command-Option-I` in OSX).
|
||
|
||
打开浏览器的“Developer Tools”(Windows上的Ctrl-Shift-I或者OSX上的`Command-Option-I)。
|
||
|
||
1. Pick the "sources" section.
|
||
|
||
选择“sources”页
|
||
|
||
1. Open the `1st.spec.ts` test file (Control/Command-P, then start typing the name of the file).
|
||
|
||
打开`1st.spec.ts`测试文件(Control/Command-P, 然后输入文件名字)。
|
||
|
||
1. Set a breakpoint in the test.
|
||
|
||
在测试程序中设置断点。
|
||
|
||
1. Refresh the browser, and it stops at the breakpoint.
|
||
|
||
刷新浏览器...然后它就会停在断点上。
|
||
|
||
figure.image-display
|
||
img(src='/resources/images/devguide/testing/karma-1st-spec-debug.png' style="width:700px;" alt="Karma debugging")
|
||
|
||
a#live-karma-example
|
||
:marked
|
||
### Try the live example
|
||
|
||
### 试试这个在线例子
|
||
|
||
You can also try this test as a <live-example plnkr="1st-specs" title="First spec" embedded-style></live-example> in plunker.
|
||
All of the tests in this guide are available as [live examples](#live-examples "Live examples of these tests").
|
||
|
||
你还可以在plunker的<live-example plnkr="1st-specs" title="First spec" embedded-style></live-example>中试运行这个测试。
|
||
本章的所有测试都有相应的[在线例子](#live-examples "Live examples of these tests")。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#simple-component-test
|
||
:marked
|
||
## Test a component
|
||
|
||
## 测试一个组件
|
||
|
||
An Angular component is the first thing most developers want to test.
|
||
The `BannerComponent` in `src/app/banner-inline.component.ts` is the simplest component in this application and
|
||
a good place to start.
|
||
It presents the application title at the top of the screen within an `<h1>` tag.
|
||
|
||
大多数开发人员首先要测试的就是Angular组件。
|
||
`src/app/banner-inline.component.ts`中的`BannerComponent`是这个应用中最简单的组件,也是一个好的起点。
|
||
它所表示的是屏幕顶部`<h1>`标签中的应用标题。
|
||
|
||
+makeExample('testing/ts/src/app/banner-inline.component.ts', '', 'src/app/banner-inline.component.ts')(format='.')
|
||
|
||
:marked
|
||
This version of the `BannerComponent` has an inline template and an interpolation binding.
|
||
The component is probably too simple to be worth testing in real life but
|
||
it's perfect for a first encounter with the Angular testing utilities.
|
||
|
||
这个版本的`BannerComponent`有一个内联模板和一个插值表达式绑定。
|
||
这个组件可能太简单,以至于在真实的项目中都不值得测试,但它却是首次接触Angular测试工具集时的完美例子。
|
||
|
||
The corresponding `src/app/banner-inline.component.spec.ts` sits in the same folder as the component,
|
||
for reasons explained in the [FAQ](#faq) answer to
|
||
["Why put specs next to the things they test?"](#q-spec-file-location).
|
||
|
||
组件对应的`src/app/banner-inline.component.spec.ts`文件与该组件位于同一个目录中,原因详见[FAQ](#faq)中的
|
||
[为什么要把测试规约文件放在被测试对象旁边?](#q-spec-file-location)
|
||
|
||
Start with ES6 import statements to get access to symbols referenced in the spec.
|
||
|
||
在测试文件中,我们先用ES6的`import`语句来引入测试所需的符号。
|
||
|
||
+makeExample('testing/ts/src/app/banner-inline.component.spec.ts', 'imports', 'src/app/banner-inline.component.spec.ts (imports)')(format='.')
|
||
|
||
a#configure-testing-module
|
||
:marked
|
||
Here's the `describe` and the `beforeEach` that precedes the tests:
|
||
|
||
测试前面的`describe`和`beforeEach`如下:
|
||
|
||
+makeExample('testing/ts/src/app/banner-inline.component.spec.ts', 'setup', 'src/app/banner-inline.component.spec.ts (beforeEach)')(format='.')
|
||
|
||
a#testbed
|
||
:marked
|
||
### _TestBed_
|
||
|
||
`TestBed` is the first and most important of the Angular testing utilities.
|
||
It creates an Angular testing module—an `@NgModule` class—that
|
||
you configure with the `configureTestingModule` method to produce the module environment for the class you want to test.
|
||
In effect, you detach the tested component from its own application module
|
||
and re-attach it to a dynamically-constructed Angular test module
|
||
tailored specifically for this battery of tests.
|
||
|
||
`TestBed`(测试台)是Angular测试工具集中的首要概念。
|
||
它创建Angular测试模块(一个`@NgModule`类),我们可以通过调用它的`configureTestingModule`方法来为要测试的类生成模块环境。
|
||
其效果是,你可以把被测试的组件从原有的应用模块中剥离出来,把它附加到一个动态生成的Angular测试模块上,而该测试模块可以为这些测试进行特殊裁剪。
|
||
|
||
The `configureTestingModule` method takes an `@NgModule`-like metadata object.
|
||
The metadata object can have most of the properties of a normal [Angular module](ngmodule.html).
|
||
|
||
`configureTestingModule`方法接受一个类似`@NgModule`的元数据对象。这个元数据对象具有标准[Angular模块](ngmodule.html)的大多数属性。
|
||
|
||
_This metadata object_ simply declares the component to test, `BannerComponent`.
|
||
The metadata lack `imports` because (a) the default testing module configuration already has what `BannerComponent` needs
|
||
and (b) `BannerComponent` doesn't interact with any other components.
|
||
|
||
*这里的元数据对象*只是声明了要测试的组件`BannerComponent`。
|
||
这个元数据中没有`imports`属性,这是因为:(a) 默认的测试模块配置中已经有了`BannerComponent`所需的一切,(b) `BannerComponent`不需要与任何其它组件交互。
|
||
|
||
Call `configureTestingModule` within a `beforeEach` so that
|
||
`TestBed` can reset itself to a base state before each test runs.
|
||
|
||
在`beforeEach`中调用`configureTestingModule`,以便`TestBed`可以在运行每个测试之前都把自己重置回它的基础状态。
|
||
|
||
The base state includes a default testing module configuration consisting of the
|
||
declarables (components, directives, and pipes) and providers (some of them mocked)
|
||
that almost everyone needs.
|
||
|
||
基础状态中包含一个默认的测试模块配置,它包含每个测试都需要的那些声明(组件、指令和管道)以及服务提供商(有些是Mock版)。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The testing shims mentioned [later](#testbed-methods) initialize the testing module configuration
|
||
to something like the `BrowserModule` from `@angular/platform-browser`.
|
||
|
||
[之前](#setup)提到的测试垫片初始化测试模块配置到一个模块,这个模块和`@angular/platform-browser`中的`BrowserModule`类似。
|
||
|
||
:marked
|
||
This default configuration is merely a _foundation_ for testing an app.
|
||
Later you'll call `TestBed.configureTestingModule` with more metadata that define additional
|
||
imports, declarations, providers, and schemas to fit your application tests.
|
||
Optional `override` methods can fine-tune aspects of the configuration.
|
||
|
||
这个默认的配置只是测试的*基础性*工作。稍后我们会调用`TestBed.configureTestingModule`来传入更多元数据,这些元数据定义了额外的
|
||
`imports`、`declarations`、`providers`和试用于这些测试的概要(Schema)。
|
||
可选的`override`方法可以微调配置的各个方面。
|
||
|
||
a#create-component
|
||
:marked
|
||
### _createComponent_
|
||
|
||
After configuring `TestBed`, you tell it to create an instance of the _component-under-test_.
|
||
In this example, `TestBed.createComponent` creates an instance of `BannerComponent` and
|
||
returns a [_component test fixture_](#component-fixture).
|
||
|
||
在配置好`TestBed`之后,我们可以告诉它创建一个*待测组件*的实例。
|
||
在这个例子中,`TestBed.createComponent`创建了一个`BannerComponent`的实例,并返回一个[*组件测试夹具*](#component-fixture)。
|
||
|
||
.alert.is-important
|
||
:marked
|
||
Do not re-configure `TestBed` after calling `createComponent`.
|
||
|
||
在调用了`createComponent`之后就不要再重新配置`TestBed`了。
|
||
|
||
:marked
|
||
The `createComponent` method closes the current `TestBed` instance to further configuration.
|
||
You cannot call any more `TestBed` configuration methods, not `configureTestingModule`
|
||
nor any of the `override...` methods. If you try, `TestBed` throws an error.
|
||
|
||
`createComponent`方法封闭了当前的`TestBed`实例,以免将来再配置它。
|
||
我们不能再调用任何`TestBed`的方法修改配置:不能调用`configureTestingModule`或任何`override...`方法。如果这么做,`TestBed`就会抛出错误。
|
||
|
||
a#component-fixture
|
||
:marked
|
||
### _ComponentFixture_, _DebugElement_, and _query(By.css)_
|
||
|
||
### `ComponentFixture`、`DebugElement` 和 `query(By.css)`
|
||
|
||
The `createComponent` method returns a **`ComponentFixture`**, a handle on the test environment surrounding the created component.
|
||
The fixture provides access to the component instance itself and
|
||
to the **`DebugElement`**, which is a handle on the component's DOM element.
|
||
|
||
`createComponent`方法返回**`ComponentFixture`**,用来控制和访问已创建的组件所在的测试环境。
|
||
这个fixture提供了对组件实例自身的访问,同时还提供了用来访问组件的DOM元素的**`DebugElement`**对象。
|
||
|
||
The `title` property value is interpolated into the DOM within `<h1>` tags.
|
||
Use the fixture's `DebugElement` to `query` for the `<h1>` element by CSS selector.
|
||
|
||
`title`属性被插值到DOM的`<h1>`标签中。
|
||
用CSS选择器从fixture的`DebugElement`中`query``<h1>`元素。
|
||
|
||
The **`query`** method takes a predicate function and searches the fixture's entire DOM tree for the
|
||
_first_ element that satisfies the predicate.
|
||
The result is a _different_ `DebugElement`, one associated with the matching DOM element.
|
||
|
||
**`query`**方法接受predicate函数,并搜索fixture的整个DOM树,试图寻找**第一个**满足predicate函数的元素。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The `queryAll` method returns an array of _all_ `DebugElements` that satisfy the predicate.
|
||
|
||
`queryAll`方法返回一列数组,包含所有`DebugElement`中满足predicate的元素。
|
||
|
||
A _predicate_ is a function that returns a boolean.
|
||
A query predicate receives a `DebugElement` and returns `true` if the element meets the selection criteria.
|
||
|
||
**predicate**是返回布尔值的函数。
|
||
predicate查询接受`DebugElement`参数,如果元素符合选择条件便返回`true`。
|
||
|
||
:marked
|
||
The **`By`** class is an Angular testing utility that produces useful predicates.
|
||
Its `By.css` static method produces a
|
||
<a href="https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Getting_started/Selectors" target="_blank">standard CSS selector</a>
|
||
predicate that filters the same way as a jQuery selector.
|
||
|
||
**`By`**类是Angular测试工具之一,它生成有用的predicate。
|
||
它的`By.css`静态方法产生<a href="https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Getting_started/Selectors" target="_blank">标准CSS选择器</a>
|
||
predicate,与JQuery选择器相同的方式过滤。
|
||
|
||
Finally, the setup assigns the DOM element from the `DebugElement` **`nativeElement`** property to `el`.
|
||
The tests assert that `el` contains the expected title text.
|
||
|
||
最后,这个配置把`DebugElement`中的**`nativeElement`**DOM元素赋值给`el`属性。
|
||
测试程序将判断`el`是否包含期待的标题文本。
|
||
|
||
a#the-tests
|
||
:marked
|
||
### The tests
|
||
|
||
### 测试程序
|
||
|
||
Jasmine runs the `beforeEach` function before each of these tests
|
||
|
||
再每个测试程序之前,Jasmin都一次运行`beforeEach`函数:
|
||
|
||
+makeExample('testing/ts/src/app/banner-inline.component.spec.ts', 'tests', 'src/app/banner-inline.component.spec.ts (tests)')(format='.')
|
||
|
||
:marked
|
||
These tests ask the `DebugElement` for the native HTML element to satisfy their expectations.
|
||
|
||
这些测试程序向`DebugElement`获取原生HTML元素,来满足自己的期望。
|
||
|
||
a#detect-changes
|
||
:marked
|
||
### _detectChanges_: Angular change detection within a test
|
||
|
||
### **detectChanges**:在测试中的Angular变更检测
|
||
|
||
Each test tells Angular when to perform change detection by calling `fixture.detectChanges()`.
|
||
The first test does so immediately, triggering data binding and propagation of the `title` property
|
||
to the DOM element.
|
||
|
||
每个测试程序都通过调用`fixture.detectChanges()`来通知Angular执行变更检测。第一个测试程序立刻这么做,触发数据绑定和并将`title`属性发送到DOM元素中。
|
||
|
||
The second test changes the component's `title` property _and only then_ calls `fixture.detectChanges()`;
|
||
the new value appears in the DOM element.
|
||
|
||
第二个测试程序在更改组件的`title`属性**之后**才调用`fixture.detectChanges()`。新值出现在DOM元素中。
|
||
|
||
In production, change detection kicks in automatically
|
||
when Angular creates a component or the user enters a keystroke or
|
||
an asynchronous activity (e.g., AJAX) completes.
|
||
|
||
在产品阶段,当Angular创建组件、用户输入或者异步动作(比如AJAX)完成时,自动触发变更检测。
|
||
|
||
The `TestBed.createComponent` does _not_ trigger change detection.
|
||
The fixture does not automatically push the component's `title` property value into the data bound element,
|
||
a fact demonstrated in the following test:
|
||
|
||
`TestBed.createComponent`**不会**触发变更检测。该工具不会自动将组件的`title`属性值推送到数据绑定的元素,下面的测试程序展示了这个事实:
|
||
|
||
+makeExample('testing/ts/src/app/banner-inline.component.spec.ts', 'test-w-o-detect-changes', 'src/app/banner-inline.component.spec.ts (no detectChanges)')(format='.')
|
||
|
||
:marked
|
||
This behavior (or lack of it) is intentional.
|
||
It gives the tester an opportunity to inspect or change the state of
|
||
the component _before Angular initiates data binding or calls lifecycle hooks_.
|
||
|
||
这种行为(或者缺乏的行为)是有意为之。**在Angular初始化数据绑定或者调用生命周期钩子**之前,它给测试者机会来查看或者改变组件的状态。
|
||
|
||
a#try-example
|
||
:marked
|
||
### Try the live example
|
||
|
||
### 试试在线例子
|
||
|
||
Take a moment to explore this component spec as a <live-example plnkr="banner-inline-specs" title="Spec for component with inline template" embedded-style></live-example> and
|
||
lock in these fundamentals of component unit testing.
|
||
|
||
花点时间来浏览一下该组件的规约,比如<live-example plnkr="banner-inline-specs" title="Spec for component with inline template" embedded-style></live-example>,深入理解组件单元测试的这些基本原理。
|
||
|
||
a#auto-detect-changes
|
||
:marked
|
||
### Automatic change detection
|
||
|
||
### 自动变更检测
|
||
|
||
The `BannerComponent` tests frequently call `detectChanges`.
|
||
Some testers prefer that the Angular test environment run change detection automatically.
|
||
That's possible by configuring the `TestBed` with the `ComponentFixtureAutoDetect` provider .
|
||
First import it from the testing utility library :
|
||
|
||
`BannerComponent`的测试频繁调用`detectChanges`。
|
||
有些测试人员更希望Angular的测试环境自动进行变更检测。
|
||
这可以通过为`TestBed`配置上`ComponentFixtureAutoDetect`提供商来做到。首先从测试工具库中导入它:
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.detect-changes.spec.ts', 'import-ComponentFixtureAutoDetect', 'src/app/banner.component.detect-changes.spec.ts (import)')(format='.')
|
||
|
||
:marked
|
||
Then add it to the `providers` array of the testing module configuration:
|
||
|
||
然后把它添加到测试模块配置的`providers`数组中:
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.detect-changes.spec.ts', 'auto-detect', 'src/app/banner.component.detect-changes.spec.ts (AutoDetect)')(format='.')
|
||
|
||
:marked
|
||
Here are three tests that illustrate how automatic change detection works.
|
||
|
||
下列测试阐明了自动变更检测的工作原理。
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.detect-changes.spec.ts', 'auto-detect-tests', 'src/app/banner.component.detect-changes.spec.ts (AutoDetect Tests)')(format='.')
|
||
|
||
:marked
|
||
The first test shows the benefit of automatic change detection.
|
||
|
||
第一个测试程序展示了自动检测的好处。
|
||
|
||
The second and third test reveal an important limitation.
|
||
The Angular testing environment does _not_ know that the test changed the component's `title`.
|
||
The `ComponentFixtureAutoDetect` service responds to _asynchronous activities_ such as promise resolution, timers, and DOM events.
|
||
But a direct, synchronous update of the component property is invisible.
|
||
The test must call `fixture.detectChanges()` manually to trigger another cycle of change detection.
|
||
|
||
第二和第三个测试程序显示了一个重要的局限性。
|
||
Angular测试环境**不会**知道测试程序改变了组件的`title`属性。
|
||
自动检测只对异步行为比如承诺的解析、计时器和DOM时间作出反应。
|
||
但是直接修改组件属性值的这种同步更新是不会触发**自动检测**的。
|
||
测试程序必须手动调用`fixture.detectChange()`,来触发新一轮的变更检测周期。
|
||
|
||
.alert.is-helpful
|
||
:marked
|
||
Rather than wonder when the test fixture will or won't perform change detection,
|
||
the samples in this guide _always call_ `detectChanges()` _explicitly_.
|
||
There is no harm in calling `detectChanges()` more often than is strictly necessary.
|
||
|
||
与其怀疑测试工具会不会执行变更检测,本章中的例子**总是显式**调用`detectChanges()`。
|
||
即使是在不需要的时候,频繁调用`detectChanges()`没有任何什么坏处。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#component-with-external-template
|
||
:marked
|
||
## Test a component with an external template
|
||
|
||
## 测试带有外部模板的组件
|
||
|
||
The application's actual `BannerComponent` behaves the same as the version above but is implemented differently.
|
||
It has _external_ template and css files, specified in `templateUrl` and `styleUrls` properties.
|
||
|
||
在实际应用中,`BannerComponent`的行为和刚才的版本相同,但是实现方式不同。
|
||
它有一个*外部*模板和CSS文件,通过`templateUrl`和`styleUrls`属性来指定。
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.ts', '', 'src/app/banner.component.ts')(format='.')
|
||
|
||
:marked
|
||
That's a problem for the tests.
|
||
The `TestBed.createComponent` method is synchronous.
|
||
But the Angular template compiler must read the external files from the file system before it can create a component instance.
|
||
That's an asynchronous activity.
|
||
The previous setup for testing the inline component won't work for a component with an external template.
|
||
|
||
|
||
这些测试有一个问题。
|
||
`TestBed.createComponent`方法是同步的。
|
||
但是Angular模板编译器必须在创建组件实例之前先从文件系统中读取这些值,而这是异步的。
|
||
以前测试内联模板时使用的设置方式不适用于外部模板。
|
||
|
||
|
||
#async-in-before-each
|
||
:marked
|
||
### The first asynchronous _beforeEach_
|
||
|
||
### 第一个异步的`beforeEach`
|
||
|
||
The test setup for `BannerComponent` must give the Angular template compiler time to read the files.
|
||
The logic in the `beforeEach` of the previous spec is split into two `beforeEach` calls.
|
||
The first `beforeEach` handles asynchronous compilation.
|
||
|
||
`BannerComponent`测试的设置方式必须给Angular模板编译器一些时间来读取文件。
|
||
以前放在`beforeEach`中的逻辑被拆分成了两个`beforeEach`调用。
|
||
第一个`beforeEach`处理异步编译工作。
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.spec.ts', 'async-before-each', 'src/app/banner.component.spec.ts (first beforeEach)')(format='.')
|
||
|
||
:marked
|
||
Notice the `async` function called as the argument to `beforeEach`.
|
||
The `async` function is one of the Angular testing utilities and
|
||
has to be imported.
|
||
|
||
注意`async`函数被用作调用`beforeEach`的参数。
|
||
`async`函数是Angular测试工具集的一部分,这里必须引入它。
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.detect-changes.spec.ts', 'import-async')(format='.')
|
||
|
||
:marked
|
||
It takes a parameterless function and _returns a function_
|
||
which becomes the true argument to the `beforeEach`.
|
||
|
||
它接收一个无参数的函数,并*返回一个函数*,这个函数会作为实参传给`beforeEach`。
|
||
|
||
The body of the `async` argument looks much like the body of a synchronous `beforeEach`.
|
||
There is nothing obviously asynchronous about it.
|
||
For example, it doesn't return a promise and
|
||
there is no `done` function to call as there would be in standard Jasmine asynchronous tests.
|
||
Internally, `async` arranges for the body of the `beforeEach` to run in a special _async test zone_
|
||
that hides the mechanics of asynchronous execution.
|
||
|
||
`async`参数的内容看起来非常像同步版`beforeEach`的函数体。
|
||
它并不能很明显的看出来这是异步函数。
|
||
比如它不返回承诺(Promise),并且也没有标准Jasmine异步测试时常用的`done`函数作为参数。
|
||
内部实现上,`async`会把`beforeEach`的函数体放进一个特殊的*异步测试区(async test zone)*,它隐藏了异步执行的内部机制。
|
||
|
||
All this is necessary in order to call the asynchronous `TestBed.compileComponents` method.
|
||
|
||
这就是为了调用异步的`TestBed.compileComponents`方法所要做的一切。
|
||
|
||
a#compile-components
|
||
:marked
|
||
### _compileComponents_
|
||
|
||
The `TestBed.configureTestingModule` method returns the `TestBed` class so you can chain
|
||
calls to other `TestBed` static methods such as `compileComponents`.
|
||
|
||
`TestBed.configureTestingModule`方法返回`TestBed`类,以便你可以链式调用`TestBed`的其它静态方法,比如`compileComponents`。
|
||
|
||
The `TestBed.compileComponents` method asynchronously compiles all the components configured in the testing module.
|
||
In this example, the `BannerComponent` is the only component to compile.
|
||
When `compileComponents` completes, the external templates and css files have been "inlined"
|
||
and `TestBed.createComponent` can create new instances of `BannerComponent` synchronously.
|
||
|
||
`TestBed.compileComponents`方法会异步编译这个测试模块中配置的所有组件。
|
||
在这个例子中,`BannerComponent`是唯一要编译的组件。
|
||
当`compileComponents`完成时,外部组件和css文件会被“内联”,而`TestBed.createComponent`会用同步的方式创建一个`BannerComponent`的新实例。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
WebPack developers need not call `compileComponents` because it inlines templates and css
|
||
as part of the automated build process that precedes running the test.
|
||
|
||
WebPack用户不用调用`compileComponents`,因为它会在构建过程中自动内联模板和css,然后执行测试
|
||
|
||
:marked
|
||
In this example, `TestBed.compileComponents` only compiles the `BannerComponent`.
|
||
Tests later in the guide declare multiple components and
|
||
a few specs import entire application modules that hold yet more components.
|
||
Any of these components might have external templates and css files.
|
||
`TestBed.compileComponents` compiles all of the declared components asynchronously at one time.
|
||
|
||
在这个例子中,`TestBed.compileComponents`只会编译`BannerComponent`。
|
||
本章稍后的测试中会声明多个组件,并且少量规约中会导入包含多个组件的应用模块。所有这些组件都可能含有外部模板和css文件。
|
||
`TestBed.compileComponents`会同时异步编译所有这些声明的组件。
|
||
|
||
.alert.is-important
|
||
:marked
|
||
Do not configure the `TestBed` after calling `compileComponents`.
|
||
Make `compileComponents` the last step
|
||
before calling `TestBed.createComponent` to instantiate the _component-under-test_.
|
||
|
||
调用了`compileComponents`之后就不能再配置`TestBed`了。
|
||
务必确保`compileComponents`是调用`TestBed.createComponent`来实例化*待测组件*之前的最后一步。
|
||
|
||
:marked
|
||
Calling `compileComponents` closes the current `TestBed` instance is further configuration.
|
||
You cannot call any more `TestBed` configuration methods, not `configureTestingModule`
|
||
nor any of the `override...` methods. The `TestBed` throws an error if you try.
|
||
|
||
`compileComponents`方法封闭了当前的`TestBed`实例,以免将来再配置它。
|
||
我们不能再调用任何`TestBed`的方法修改配置:不能调用`configureTestingModule`或任何`override...`方法。如果这么做,`TestBed`就会抛出错误。
|
||
|
||
a#second-before-each
|
||
:marked
|
||
### The second synchronous _beforeEach_
|
||
|
||
### 第二个同步`beforeEach`
|
||
|
||
A _synchronous_ `beforeEach` containing the remaining setup steps follows the asynchronous `beforeEach`.
|
||
|
||
这个同步的`beforeEach`包含异步`beforeEach`之后的其余步骤。
|
||
|
||
+makeExample('testing/ts/src/app/banner.component.spec.ts', 'sync-before-each', 'src/app/banner.component.spec.ts (second beforeEach)')(format='.')
|
||
|
||
:marked
|
||
These are the same steps as in the original `beforeEach`.
|
||
They include creating an instance of the `BannerComponent` and querying for the elements to inspect.
|
||
|
||
这些步骤和原来的`beforeEach`中相同。
|
||
包括创建`BannerComponent`实例和查询要审查的元素。
|
||
|
||
You can count on the test runner to wait for the first asynchronous `beforeEach` to finish before calling the second.
|
||
|
||
测试运行器(runner)会先等待第一个异步`beforeEach`函数执行完再调用第二个。
|
||
|
||
a#waiting-compile-components
|
||
:marked
|
||
### Waiting for _compileComponents_
|
||
|
||
### 等待`compileComponents`
|
||
|
||
The `compileComponents` method returns a promise so you can perform additional tasks _immediately after_ it finishes.
|
||
For example, you could move the synchronous code in the second `beforeEach`
|
||
into a `compileComponents().then(...)` callback and write only one `beforeEach`.
|
||
|
||
`compileComponents`方法返回一个承诺,来让我们可以在它完成之后*立即*执行额外的任务。
|
||
比如,我们可以把第二个`beforeEach`中的同步代码移到一个`compileComponents().then(...)`回调中,从而只需要写一个`beforeEach`。
|
||
|
||
Most developers find that hard to read.
|
||
The two `beforeEach` calls are widely preferred.
|
||
|
||
大多数开发人员会觉得这样不易读,因此,更多采用的还是写两个`beforeEach`调用的方式。
|
||
|
||
### Try the live example
|
||
|
||
### 试试在线例子
|
||
|
||
Take a moment to explore this component spec as a <live-example plnkr="banner-specs" title="Spec for component with external template" embedded-style></live-example>.
|
||
|
||
稍微花点时间,在<live-example plnkr="banner-specs" title="Spec for component with external template" embedded-style></live-example>中看看该组件的规约。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The [Quickstart seed](setup.html) provides a similar test of its `AppComponent`
|
||
as you can see in _this_ <live-example name="setup" plnkr="quickstart-specs" title="QuickStart seed spec" embedded-style></live-example>.
|
||
It too calls `compileComponents` although it doesn't have to because the `AppComponent`'s template is inline.
|
||
|
||
[“快速起步” 种子工程](setup.html)为其`AppComponent`提供了简单的测试,在<live-example name="setup" plnkr="quickstart-specs" title="QuickStart seed spec" embedded-style></live-example>中可以看到。
|
||
它也调用了`compileComponents`,不过它并不是必须这么做,因为`AppComponent`的模板是内联的。
|
||
|
||
There's no harm in it and you might call `compileComponents` anyway
|
||
in case you decide later to re-factor the template into a separate file.
|
||
The tests in this guide only call `compileComponents` when necessary.
|
||
|
||
这样做也没坏处,如果你将来可能会把模板重构到独立的文件中去,那就可以调用`compileComponents`。
|
||
不过本章中的这些测试只会在必要时才调用`compileComponents`。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#component-with-dependency
|
||
:marked
|
||
## Test a component with a dependency
|
||
|
||
## 测试有依赖的组件
|
||
|
||
Components often have service dependencies.
|
||
|
||
组件经常依赖其他服务。
|
||
|
||
The `WelcomeComponent` displays a welcome message to the logged in user.
|
||
It knows who the user is based on a property of the injected `UserService`:
|
||
|
||
`WelcomeComponent`为登陆的用户显示一条欢迎信息。它从注入的`UserService`的属性得知用户的身份:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.ts', '', 'src/app/welcome.component.ts')(format='.')
|
||
|
||
:marked
|
||
The `WelcomeComponent` has decision logic that interacts with the service, logic that makes this component worth testing.
|
||
Here's the testing module configuration for the spec file, `src/app/welcome.component.spec.ts`:
|
||
|
||
`WelcomeComponent`有与服务进行交互的决策逻辑,这样的逻辑让这个组件值得测试。下面是spec文件的测试模块配置,`src/app/welcome.component.spec.ts`:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'config-test-module', 'src/app/welcome.component.spec.ts')(format='.')
|
||
|
||
:marked
|
||
This time, in addition to declaring the _component-under-test_,
|
||
the configuration adds a `UserService` provider to the `providers` list.
|
||
But not the real `UserService`.
|
||
|
||
这次,在测试配置里不但声明了被测试的组件,而且在`providers`数组中添加了`UserService`依赖。但不是真实的`UserService`。
|
||
|
||
a#service-test-doubles
|
||
:marked
|
||
### Provide service test doubles
|
||
|
||
### 提供服务替身
|
||
|
||
A _component-under-test_ doesn't have to be injected with real services.
|
||
In fact, it is usually better if they are test doubles (stubs, fakes, spies, or mocks).
|
||
The purpose of the spec is to test the component, not the service,
|
||
and real services can be trouble.
|
||
|
||
被测试的组件不一定要注入真正的服务。实际上,服务的替身(stubs, fakes, spies或者mocks)通常会更加合适。
|
||
spec的主要目的是测试组件,而不是服务。真实的服务可能自身有问题。
|
||
|
||
Injecting the real `UserService` could be a nightmare.
|
||
The real service might ask the user for login credentials and
|
||
attempt to reach an authentication server.
|
||
These behaviors can be hard to intercept.
|
||
It is far easier and safer to create and register a test double in place of the real `UserService`.
|
||
|
||
注入真实的`UserService`有可能很麻烦。真实的服务可能询问用户登录凭据,也可能试图连接认证服务器。
|
||
可能很难处理这些行为。所以在真实的`UserService`的位置创建和注册`UserService`替身,会让测试更加容易和安全。
|
||
|
||
This particular test suite supplies a minimal `UserService` stub that satisfies the needs of the `WelcomeComponent`
|
||
and its tests:
|
||
|
||
这个测试套件提供了最小化的`UserService`stub类,用来满足`WelcomeComponent`和它的测试的需求:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'user-service-stub')(format='.')
|
||
|
||
a#get-injected-service
|
||
:marked
|
||
### Get injected services
|
||
|
||
### 获取注入的服务
|
||
The tests need access to the (stub) `UserService` injected into the `WelcomeComponent`.
|
||
|
||
测试程序需要访问被注入到`WelcomeComponent`中的`UserService`(stub类)。
|
||
|
||
Angular has a hierarchical injection system.
|
||
There can be injectors at multiple levels, from the root injector created by the `TestBed`
|
||
down through the component tree.
|
||
|
||
Angular的注入系统是层次化的。
|
||
可以有很多层注入器,从根`TestBed`创建的注入器下来贯穿整个组件树。
|
||
|
||
The safest way to get the injected service, the way that **_always works_**,
|
||
is to **get it from the injector of the _component-under-test_**.
|
||
The component injector is a property of the fixture's `DebugElement`.
|
||
|
||
最安全并总是有效的获取注入服务的方法,是从被测试的组件的注入器获取。
|
||
组件注入器是fixture的`DebugElement`的属性。
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'injected-service', 'WelcomeComponent\'s injector')(format='.')
|
||
|
||
a#testbed-get
|
||
:marked
|
||
### _TestBed.get_
|
||
|
||
You _may_ also be able to get the service from the root injector via `TestBed.get`.
|
||
This is easier to remember and less verbose.
|
||
But it only works when Angular injects the component with the service instance in the test's root injector.
|
||
Fortunately, in this test suite, the _only_ provider of `UserService` is the root testing module,
|
||
so it is safe to call `TestBed.get` as follows:
|
||
|
||
你**可以**通过`TestBed.get`方法来从根注入器中获取服务。
|
||
它更容易被记住,也更加简介。
|
||
但是只有在Angular使用测试的根注入器中的那个服务实例来注入到组件时,它才有效。
|
||
幸运的是,在这个测试套件中,**唯一**的`UserService`提供商就是根测试模块,所以像下面这样调用`TestBed.get`很安全:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'inject-from-testbed', 'TestBed injector')(format='.')
|
||
.l-sub-section
|
||
:marked
|
||
The [`inject`](#inject) utility function is another way to get one or more services from the test root injector.
|
||
|
||
[`inject`](#inject)辅助函数方法是另外一种从测试的根注入器注入一个或多个服务到测试的方法。
|
||
|
||
For a use case in which `inject` and `TestBed.get` do not work,
|
||
see the section [_Override a component's providers_](#component-override), which
|
||
explains why you must get the service from the component's injector instead.
|
||
|
||
如果遇到了`inject`和`TestBed.get`无效,的情况,请到“[**重载组件提供商**](#component-override)”一节。那里会解释为什么要改用组件的注入器来获取服务。
|
||
|
||
a#service-from-injector
|
||
:marked
|
||
### Always get the service from an injector
|
||
|
||
### 总是从注入器获取服务
|
||
|
||
Do _not_ reference the `userServiceStub` object
|
||
that's provided to the testing module in the body of your test.
|
||
**It does not work!**
|
||
The `userService` instance injected into the component is a completely _different_ object,
|
||
a clone of the provided `userServiceStub`.
|
||
|
||
请不要引用测试代码里提供给测试模块的`userServiceStub`对象。**这样不行!**
|
||
被注入组件的`userService`实例是完全**不一样**的对象,它提供的是`userServiceStub`的克隆。
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'stub-not-injected')(format='.')
|
||
|
||
a#welcome-spec-setup
|
||
:marked
|
||
### Final setup and tests
|
||
|
||
### 最后的设置和测试程序
|
||
|
||
Here's the complete `beforeEach` using `TestBed.get`:
|
||
|
||
这里是使用`TestBed.get`的完整`beforeEach`:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'setup', 'src/app/welcome.component.spec.ts')(format='.')
|
||
:marked
|
||
And here are some tests:
|
||
|
||
下面是一些测试程序:
|
||
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'tests', 'src/app/welcome.component.spec.ts')(format='.')
|
||
:marked
|
||
The first is a sanity test; it confirms that the stubbed `UserService` is called and working.
|
||
|
||
第一个测试程序是合法测试程序,它确认这个被模拟的`UserService`是否被调用和工作正常。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The second parameter to the Jasmine `it` (e.g., `'expected name'`) is an optional addendum.
|
||
If the expectation fails, Jasmine displays this addendum after the expectation failure message.
|
||
In a spec with multiple expectations, it can help clarify what went wrong and which expectation failed .
|
||
|
||
Jasmine的`it`方法的第二个参数(比如`'expected name'`)是可选附加参数。
|
||
如果这个期待失败了,Jasmine在期待失败信息后面显示这个附加参数。
|
||
在拥有多个期待的spec中,它可以帮助澄清发生了什么错误,哪个期待失败了。
|
||
|
||
:marked
|
||
The remaining tests confirm the logic of the component when the service returns different values.
|
||
The second test validates the effect of changing the user name.
|
||
The third test checks that the component displays the proper message when there is no logged-in user.
|
||
|
||
接下来的测试程序确认当服务返回不同的值时组件的逻辑是否工作正常。
|
||
第二个测试程序验证变换用户名字的效果。
|
||
第三个测试程序检查如果用户没有登录,组件是否显示正确消息。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#component-with-async-service
|
||
:marked
|
||
## Test a component with an async service
|
||
|
||
## 测试有异步服务的组件
|
||
|
||
Many services return values asynchronously.
|
||
Most data services make an HTTP request to a remote server and the response is necessarily asynchronous.
|
||
|
||
许多服务异步返回值。大部分数据服务向远程服务器发起HTTP请求,响应必然是异步的。
|
||
|
||
The "About" view in this sample displays Mark Twain quotes.
|
||
The `TwainComponent` handles the display, delegating the server request to the `TwainService`.
|
||
|
||
本例的`About`视图显示马克吐温的名言。
|
||
`TwainComponent`组件处理视图,并委派`TwainService`向服务器发起请求。
|
||
|
||
Both are in the `src/app/shared` folder because the author intends to display Twain quotes on other pages someday.
|
||
Here is the `TwainComponent`.
|
||
|
||
两者都在`app/shared`目录里,因为作者计划将来在其它页面也显示马克吐温的名言。
|
||
下面是`TwainComponent`:
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.ts', 'component', 'src/app/shared/twain.component.ts')(format='.')
|
||
|
||
:marked
|
||
The `TwainService` implementation is irrelevant for this particular test.
|
||
It is sufficient to see within `ngOnInit` that `twainService.getQuote` returns a promise, which means it is asynchronous.
|
||
|
||
`TwainService`的实现细节现在并不重要。
|
||
`ngOnInit`的`twainService.getQuote`返回承诺,所以显然它是异步的。
|
||
|
||
In general, tests should not make calls to remote servers.
|
||
They should emulate such calls. The setup in this `src/app/shared/twain.component.spec.ts` shows one way to do that:
|
||
|
||
一般来讲,测试程序不应该向远程服务器发请求。
|
||
它们应该仿真这样的请求。`src/app/shared/twain.component.spec.ts`里的配置是其中一种伪造方法:
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'setup', 'src/app/shared/twain.component.spec.ts (setup)')(format='.')
|
||
|
||
a#service-spy
|
||
:marked
|
||
### Spying on the real service
|
||
|
||
### 刺探(Spy)真实服务
|
||
|
||
This setup is similar to the [`welcome.component.spec` setup](#welcome-spec-setup).
|
||
But instead of creating a stubbed service object, it injects the _real_ service (see the testing module `providers`) and
|
||
replaces the critical `getQuote` method with a Jasmine spy.
|
||
|
||
本配置与[`welcome.component.spec`配置](#welcome-spec-setup)类似。
|
||
但是与其伪造服务对象,它注入了真实的服务(参见测试模块的`providers`),并用Jasmine的`spy`替换关键的`getQuote`方法。
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'spy')(format='.')
|
||
|
||
:marked
|
||
The spy is designed such that any call to `getQuote` receives an immediately resolved promise with a test quote.
|
||
The spy bypasses the actual `getQuote` method and therefore does not contact the server.
|
||
|
||
这个Spy的设计是,所有调用`getQuote`的方法都会收到立刻解析的承诺,得到一条预设的名言。Spy拦截了实际`getQuote`方法,所有它不会联系服务。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
Faking a service instance and spying on the real service are _both_ great options.
|
||
Pick the one that seems easiest for the current test suite.
|
||
Don't be afraid to change your mind.
|
||
|
||
伪造服务实例和刺探真实服务都是好方法。挑选一种对当前测试套件最简单的方法。你可以随时改变主意。
|
||
|
||
Spying on the real service isn't always easy, especially when the real service has injected dependencies.
|
||
You can _stub and spy_ at the same time, as shown in [an example below](#spy-stub).
|
||
|
||
刺探真实的服务往往并不容易,特别是真实的服务依赖其它服务时。
|
||
我们可以同时*打桩和刺探*,就像[后面的例子](#spy-stub)那样。
|
||
|
||
:marked
|
||
Here are the tests with commentary to follow:
|
||
|
||
下面是接下来带有注解的测试程序:
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'tests', 'src/app/shared/twain.component.spec.ts (tests)')
|
||
|
||
a#sync-tests
|
||
:marked
|
||
### Synchronous tests
|
||
|
||
### 同步测试程序
|
||
|
||
The first two tests are synchronous.
|
||
Thanks to the spy, they verify that `getQuote` is called _after_
|
||
the first change detection cycle during which Angular calls `ngOnInit`.
|
||
|
||
前两个测试程序是同步的。
|
||
在Spy的帮助下,它们验证了在Angular调用`ngOnInit`期间发生的第一次变更检测后,`getQuote`被调用了。
|
||
|
||
Neither test can prove that a value from the service is displayed.
|
||
The quote itself has not arrived, despite the fact that the spy returns a resolved promise.
|
||
|
||
两者都不能证明被显示的值是服务提供的。
|
||
虽然spy返回了解析的承诺,名言本身还没有到来。
|
||
|
||
This test must wait at least one full turn of the JavaScript engine before the
|
||
value becomes available. The test must become _asynchronous_.
|
||
|
||
这个测试程序必须等待JavaScript引擎一整个回合,返回值才会有效。该测试程序必须要变成**异步的**。
|
||
|
||
a#async
|
||
:marked
|
||
### The _async_ function in _it_
|
||
|
||
### **it**里的**async**函数方法
|
||
|
||
Notice the `async` in the third test.
|
||
|
||
注意第三个测试程序的`async`方法。
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'async-test', 'src/app/shared/twain.component.spec.ts (async test)')(format='.')
|
||
:marked
|
||
The `async` function is one of the Angular testing utilities.
|
||
It simplifies coding of asynchronous tests by arranging for the tester's code to run in a special _async test zone_
|
||
as [discussed earlier](#async-in-before-each) when it was called in a `beforeEach`.
|
||
|
||
`async`函数是**Angular TestBed**的一部分。通过将测试代码放到特殊的**异步测试区域**来运行,`async`函数简化了异步测试程序的代码。就像[以前讨论过的](#async-in-before-each),它会在`beforeEach`中被调用。
|
||
|
||
Although `async` does a great job of hiding asynchronous boilerplate,
|
||
some functions called within a test (such as `fixture.whenStable`) continue to reveal their asynchronous behavior.
|
||
|
||
虽然`async`做了很多工作来尽量隐藏异步特性,但在测试程序(比如`fixture.whenStable`)里面调用函数时,有时还是会体现它们的异步行为。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The `fakeAsync` alternative, [covered below](#fake-async), removes this artifact and affords a more linear coding experience.
|
||
|
||
`fakeAsync`可选方法,[正如下面解释的](#fake-async),进一步移除了异步行为,提供了更加直观的代码经验。
|
||
|
||
a#when-stable
|
||
:marked
|
||
### _whenStable_
|
||
|
||
The test must wait for the `getQuote` promise to resolve in the next turn of the JavaScript engine.
|
||
|
||
测试程序必须等待`getQuote`在JavaScript引擎的下一回合中被解析。
|
||
|
||
This test has no direct access to the promise returned by the call to `twainService.getQuote`
|
||
because it is buried inside `TwainComponent.ngOnInit` and therefore inaccessible to a test that
|
||
probes only the component API surface.
|
||
|
||
本测试对`twainService.getQuote`返回的承诺没有直接的访问,因为它被埋没在`TwainComponent.ngOnInit`里,
|
||
所以对于只测试组件API表面的测试来说,它是无法被访问的。
|
||
|
||
Fortunately, the `getQuote` promise is accessible to the _async test zone_ ,
|
||
which intercepts all promises issued within the _async_ method call _no matter where they occur_.
|
||
|
||
幸运的是,**异步测试区域**可以访问`getQuote`承诺,因为它拦截所有调用**异步**方法所发出的承诺,不管它们在哪儿。
|
||
|
||
The `ComponentFixture.whenStable` method returns its own promise, which resolves when the `getQuote` promise finishes.
|
||
In fact, the _whenStable_ promise resolves when _all pending asynchronous activities within this test_ complete — the definition of "stable."
|
||
|
||
`ComponentFixture.whenStable`方法返回它自己的承诺,它在`getQuote`承诺完成时被解析。实际上,“stable”的意思是当**所有待处理异步行为**完成时的状态,在“stable”后**whenStable**承诺被解析。
|
||
|
||
Then the test resumes and kicks off another round of change detection (`fixture.detectChanges`),
|
||
which tells Angular to update the DOM with the quote.
|
||
The `getQuote` helper method extracts the display element text and the expectation confirms that the text matches the test quote.
|
||
|
||
然后测试程序继续运行,并开始另一轮的变更检测(`fixture.detectChanges`),通知Angular使用名言来更新DOM。
|
||
`getQuote`辅助方法提取出显示元素文本,然后expect语句确认这个文本与预备的名言相符。
|
||
|
||
a#fakeAsync
|
||
a#fake-async
|
||
:marked
|
||
### The _fakeAsync_ function
|
||
|
||
### **fakeAsync**函数方法
|
||
|
||
The fourth test verifies the same component behavior in a different way.
|
||
|
||
第四个测试程序用不同的方法验证同样的组件行为。
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'fake-async-test', 'src/app/shared/twain.component.spec.ts (fakeAsync test)')(format='.')
|
||
:marked
|
||
Notice that `fakeAsync` replaces `async` as the `it` argument.
|
||
The `fakeAsync` function is another of the Angular testing utilities.
|
||
|
||
注意,在`it`的参数中,`async`被`faceAsync`替换。
|
||
`fakeAsync`是另一种Angular测试工具。
|
||
|
||
Like [async](#async), it _takes_ a parameterless function and _returns_ a function
|
||
that becomes the argument to the Jasmine `it` call.
|
||
|
||
和[async](#async)一样,它也**接受**无参数函数并**返回**一个函数,变成Jasmine的`it`函数的参数。
|
||
|
||
The `fakeAsync` function enables a linear coding style by running the test body in a special _fakeAsync test zone_.
|
||
|
||
`fakeAsync`函数通过在特殊的**fakeAsync测试区域**运行测试程序,让测试代码更加简单直观。
|
||
|
||
The principle advantage of `fakeAsync` over `async` is that the test appears to be synchronous.
|
||
There is no `then(...)` to disrupt the visible flow of control.
|
||
The promise-returning `fixture.whenStable` is gone, replaced by `tick()`.
|
||
|
||
对于`async`来说,`fakeAsync`最重要的好处是测试程序看起来像同步的。里面没有任何承诺。
|
||
没有`then(...)`链来打断控制流。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
There _are_ limitations. For example, you cannot make an XHR call from within a `fakeAsync`.
|
||
|
||
但是`fakeAsync`有局限性。比如,你不能从`fakeAsync`发起XHR请求。
|
||
|
||
a#tick
|
||
:marked
|
||
### The _tick_ function
|
||
|
||
### **tick**函数
|
||
|
||
The `tick` function is one of the Angular testing utilities and a companion to `fakeAsync`.
|
||
You can only call it within a `fakeAsync` body.
|
||
|
||
`tick`函数是Angular测试工具之一,是`fakeAsync`的同伴。
|
||
它只能在`fakeAsync`的主体中被调用。
|
||
|
||
Calling `tick()` simulates the passage of time until all pending asynchronous activities finish,
|
||
including the resolution of the `getQuote` promise in this test case.
|
||
|
||
调用`tick()`模拟时间的推移,直到全部待处理的异步任务都已完成,在这个测试案例中,包含`getQuote`承诺的解析。
|
||
|
||
It returns nothing. There is no promise to wait for.
|
||
Proceed with the same test code that appeared in the `whenStable.then()` callback.
|
||
|
||
它不返回任何结果。没有任何承诺需要等待。
|
||
直接执行与之前在`whenStable.then()`的回调函数里相同的代码。
|
||
|
||
Even this simple example is easier to read than the third test.
|
||
To more fully appreciate the improvement, imagine a succession of asynchronous operations,
|
||
chained in a long sequence of promise callbacks.
|
||
|
||
虽然这个例子非常简单,但是它已经比第三个测试程序更易阅读。
|
||
为了更充分的体会`fakeAsync`的好处,试想一下一连串的异步操作,被一长串的承诺回调链在一起。
|
||
|
||
a#jasmine-done
|
||
:marked
|
||
### _jasmine.done_
|
||
|
||
While the `async` and `fakeAsync` functions greatly
|
||
simplify Angular asynchronous testing,
|
||
you can still fall back to the traditional Jasmine asynchronous testing technique.
|
||
|
||
虽然`async`和`fakeAsync`函数大大的简化了异步测试,你仍然可以回退到传统的Jasmine异步测试技术上。
|
||
|
||
You can still pass `it` a function that takes a
|
||
[`done` callback](http://jasmine.github.io/2.0/introduction.html#section-Asynchronous_Support).
|
||
Now you are responsible for chaining promises, handling errors, and calling `done` at the appropriate moment.
|
||
|
||
你仍然可以将接受 [`done`回调](http://jasmine.github.io/2.0/introduction.html#section-Asynchronous_Support)的函数传给`it`。
|
||
但是,你必须链接承诺、处理错误,并在适当的时候调用`done`。
|
||
|
||
Here is a `done` version of the previous two tests:
|
||
|
||
下面是上面两个测试程序的`done`版本:
|
||
|
||
+makeExample('testing/ts/src/app/shared/twain.component.spec.ts', 'done-test', 'src/app/shared/twain.component.spec.ts (done test)')(format='.')
|
||
|
||
:marked
|
||
Although there is no direct access to the `getQuote` promise inside `TwainComponent`,
|
||
the spy has direct access, which makes it possible to wait for `getQuote` to finish.
|
||
|
||
虽然我们对`TwainComponent`里的`getQuote`承诺没有直接访问,但是Spy有,所以才可能等待`getQuote`完成。
|
||
|
||
Writing test functions with `done`, while more cumbersome than `async`
|
||
and `fakeAsync`, is a viable and occasionally necessary technique.
|
||
For example, you can't call `async` or `fakeAsync` when testing
|
||
code that involves the `intervalTimer`, as is common when
|
||
testing async `Observable` methods.
|
||
|
||
写带有`done`回调的测试函数,虽然比`async`和`fakeAsync`函数笨拙,但是在少数偶然情况下却是很有必要的技巧。比如,当测试涉及`intervalTimer`的代码时,你就没法调用`async`和`fakeAsync`函数,在测试异步`Observable`函数时也一样。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#component-with-input-output
|
||
:marked
|
||
## Test a component with inputs and outputs
|
||
|
||
## 测试带有导入inputs和导出outputs的组件
|
||
|
||
A component with inputs and outputs typically appears inside the view template of a host component.
|
||
The host uses a property binding to set the input property and an event binding to
|
||
listen to events raised by the output property.
|
||
|
||
带有导入和导出的组件通常出现在宿主组件的视图模板中。
|
||
宿主使用属性绑定来设置输入属性,使用事件绑定来监听输出属性触发的事件。
|
||
|
||
The testing goal is to verify that such bindings work as expected.
|
||
The tests should set input values and listen for output events.
|
||
|
||
测试的目的是验证这样的绑定和期待的那样正常工作。
|
||
测试程序应该设置导入值并监听导出事件。
|
||
|
||
The `DashboardHeroComponent` is a tiny example of a component in this role.
|
||
It displays an individual hero provided by the `DashboardComponent`.
|
||
Clicking that hero tells the `DashboardComponent` that the user has selected the hero.
|
||
|
||
`DashboardHeroComponent`是非常小的这种类型的例子组件。
|
||
它显示由`DashboardCompoent`提供的英雄个体。
|
||
点击英雄告诉`DashbaordComponent`用户已经选择了这个英雄。
|
||
|
||
The `DashboardHeroComponent` is embedded in the `DashboardComponent` template like this:
|
||
|
||
`DashboardHeroComponent`是这样内嵌在`DashboardCompoent`的模板中的:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.html', 'dashboard-hero', 'src/app/dashboard/dashboard.component.html (excerpt)')(format='.')
|
||
:marked
|
||
The `DashboardHeroComponent` appears in an `*ngFor` repeater, which sets each component's `hero` input property
|
||
to the looping value and listens for the component's `selected` event.
|
||
|
||
`DashboardHeroComponent`在`*ngFor`循环中出现,设置每个组件的`hero`input属性到迭代的值,并监听组件的`selected`事件。
|
||
|
||
Here's the component's definition:
|
||
|
||
下面是组件的定义:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.ts', 'component', 'src/app/dashboard/dashboard-hero.component.ts (component)')(format='.')
|
||
|
||
:marked
|
||
While testing a component this simple has little intrinsic value, it's worth knowing how.
|
||
You can use one of these approaches:
|
||
|
||
虽然测试这么简单的组件没有什么内在价值,但是它的测试程序是值得学习的。
|
||
有下列候选测试方案:
|
||
|
||
* Test it as used by `DashboardComponent`.
|
||
|
||
把它当作被`DashbaordComponent`使用的组件来测试
|
||
|
||
* Test it as a stand-alone component.
|
||
|
||
把它当作独立的组件来测试
|
||
|
||
* Test it as used by a substitute for `DashboardComponent`.
|
||
|
||
把它当作被`DashbaordComponent`的替代组件使用的组件来测试
|
||
|
||
A quick look at the `DashboardComponent` constructor discourages the first approach:
|
||
|
||
简单看看`DashbaordComponent`的构造函数就否决了第一种方案:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.ts', 'ctor', 'src/app/dashboard/dashboard.component.ts (constructor)')(format='.')
|
||
:marked
|
||
The `DashboardComponent` depends on the Angular router and the `HeroService`.
|
||
You'd probably have to replace them both with test doubles, which is a lot of work.
|
||
The router seems particularly challenging.
|
||
|
||
`DashbaordComponent`依赖Angular路由器和`HeroService`服务。
|
||
你必须使用测试替身替换它们两个,似乎过于复杂了。
|
||
路由器尤其具有挑战性。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The [discussion below](#routed-component) covers testing components that require the router.
|
||
|
||
[下面](#routed-component) 覆盖了如何测试带有路由器的组件。
|
||
:marked
|
||
The immediate goal is to test the `DashboardHeroComponent`, not the `DashboardComponent`,
|
||
so, try the second and third options.
|
||
|
||
当前的任务是测试`DashboardHeroComponent`组件,而非`DashbaordComponent`,所以无需做不必要的努力。
|
||
让我们尝试第二和第三种方案。
|
||
|
||
a#dashboard-standalone
|
||
:marked
|
||
### Test _DashboardHeroComponent_ stand-alone
|
||
|
||
### 独立测试_DashboardHeroComponent_
|
||
|
||
Here's the spec file setup.
|
||
|
||
下面是spec文件的设置。
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'setup', 'src/app/dashboard/dashboard-hero.component.spec.ts (setup)')(format='.')
|
||
|
||
|
||
:marked
|
||
The async `beforeEach` was discussed [above](#component-with-external-template).
|
||
Having compiled the components asynchronously with `compileComponents`, the rest of the setup
|
||
proceeds _synchronously_ in a _second_ `beforeEach`, using the basic techniques described [earlier](#simple-component-test).
|
||
|
||
异步`beforeEach`已经在[上面](#component-with-external-template)讨论过。
|
||
在使用`compileComponents`异步编译完组件后,接下来的设置执行另一个**同步**的`beforeEach`,使用[之前](#simple-component-test)解释过的基本知识。
|
||
|
||
Note how the setup code assigns a test hero (`expectedHero`) to the component's `hero` property, emulating
|
||
the way the `DashboardComponent` would set it via the property binding in its repeater.
|
||
|
||
注意代码是如何将模拟英雄(`expectedHero`)赋值给组件的`hero`属性的,模拟了`DashbaordComponent`在它的迭代器中通过属性绑定的赋值方式。
|
||
|
||
The first test follows:
|
||
|
||
紧接着第一个测试程序:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'name-test', 'src/app/dashboard/dashboard-hero.component.spec.ts (name test)')(format='.')
|
||
|
||
:marked
|
||
It verifies that the hero name is propagated to template with a binding.
|
||
Because the template passes the hero name through the Angular `UpperCasePipe`,
|
||
the test must match the element value with the uppercased name:
|
||
|
||
它验证了英雄名字通过绑定被传递到模板了。这里有个额外步骤。模板将英雄名字传给Angular的`UpperCasePipe`,
|
||
所以测试程序必须使用大写名字来匹配元素的值:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.html')(format='.')
|
||
|
||
:marked
|
||
.alert.is-helpful
|
||
:marked
|
||
This small test demonstrates how Angular tests can verify a component's visual
|
||
representation—something not possible with
|
||
[isolated unit tests](#isolated-component-tests)—at
|
||
low cost and without resorting to much slower and more complicated end-to-end tests.
|
||
|
||
这个小测试演示了Angular测试是如何验证组件的视图表现的 —— 这是[孤立的单元测试](#isolated-component-tests)无法实现的
|
||
—— 它成本低,而且无需依靠更慢、更复杂的端对端测试。
|
||
|
||
:marked
|
||
The second test verifies click behavior. Clicking the hero should raise a `selected` event that the
|
||
host component (`DashboardComponent` presumably) can hear:
|
||
|
||
第二个测试程序验证点击行为。点击英雄应该出发`selected`事件,可供宿主组件(`DashbaordComponent`)监听:
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'click-test', 'src/app/dashboard/dashboard-hero.component.spec.ts (click test)')(format='.')
|
||
:marked
|
||
The component exposes an `EventEmitter` property. The test subscribes to it just as the host component would do.
|
||
|
||
这个组件公开`EventEmitter`属性。测试程序像宿主组件那样来描述它。
|
||
|
||
The `heroEl` is a `DebugElement` that represents the hero `<div>`.
|
||
The test calls `triggerEventHandler` with the "click" event name.
|
||
The "click" event binding responds by calling `DashboardHeroComponent.click()`.
|
||
|
||
`heroEl`是个`DebugElement`,它代表了英雄所在的`<div>`。
|
||
测试程序用“click”事件名字来调用`triggerEventHandler`。
|
||
调用`DashboardHeroComponent.click()`时,“click”事件绑定作出响应。
|
||
|
||
If the component behaves as expected, `click()` tells the component's `selected` property to emit the `hero` object,
|
||
the test detects that value through its subscription to `selected`, and the test should pass.
|
||
|
||
如果组件像期待的那样工作,`click()`通知组件的`selected`属性就会发出`hero`对象,测试程序通过订阅`selected`事件而检测到这个值,所以测试应该成功。
|
||
|
||
a#trigger-event-handler
|
||
:marked
|
||
### _triggerEventHandler_
|
||
|
||
The Angular `DebugElement.triggerEventHandler` can raise _any data-bound event_ by its _event name_.
|
||
The second parameter is the event object passed to the handler.
|
||
|
||
Angular的`DebugElement.triggerEventHandler`可以用**事件的名字**触发**任何数据绑定事件**。
|
||
第二个参数是传递给事件处理器的事件对象。
|
||
|
||
In this example, the test triggers a "click" event with a null event object.
|
||
|
||
本例中,测试程序用null事件对象触发“click”事件。
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'trigger-event-handler')(format='.')
|
||
:marked
|
||
The test assumes (correctly in this case) that the runtime
|
||
event handler—the component's `click()` method—doesn't
|
||
care about the event object.
|
||
|
||
测试程序假设(在这里应该这样)运行时间的事件处理器——组件的`click()`方法——不关心事件对象。
|
||
|
||
Other handlers are less forgiving. For example, the `RouterLink`
|
||
directive expects an object with a `button` property
|
||
that identifies which mouse button was pressed.
|
||
This directive throws an error if the event object doesn't do this correctly.
|
||
|
||
其它处理器将会更加严格。
|
||
比如,`RouterLink`指令期待事件对象,并且该对象具有`button`属性,代表了已被按下的鼠标按钮。
|
||
如果该事件对象不具备上面的条件,指令便会抛出错误。
|
||
|
||
a#click-helper
|
||
:marked
|
||
Clicking a button, an anchor, or an arbitrary HTML element is a common test task.
|
||
|
||
点击按钮、链接或者任意HTML元素是很常见的测试任务。
|
||
|
||
Make that easy by encapsulating the _click-triggering_ process in a helper such as the `click` function below:
|
||
|
||
把**click触发**过程封装到辅助方法中可以简化这个任务,比如下面的`click`辅助方法:
|
||
|
||
+makeExample('testing/ts/src/testing/index.ts', 'click-event', 'testing/index.ts (click helper)')(format='.')
|
||
|
||
:marked
|
||
The first parameter is the _element-to-click_. If you wish, you can pass a
|
||
custom event object as the second parameter. The default is a (partial)
|
||
<a href="https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button" target="_blank">left-button mouse event object</a>
|
||
accepted by many handlers including the `RouterLink` directive.
|
||
|
||
第一个参数是**用来点击的元素**。如果你愿意,可以将自定义的事件对象传递给第二个参数。
|
||
默认的是(局部的)<a href="https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button" target="_blank">鼠标左键事件对象</a>,
|
||
它被许多事件处理器接受,包括`RouterLink`指令。
|
||
|
||
.callout.is-critical
|
||
header click() is not an Angular testing utility
|
||
header click()不是Angular测试工具
|
||
:marked
|
||
The `click()` helper function is **not** one of the Angular testing utilities.
|
||
It's a function defined in _this guide's sample code_.
|
||
All of the sample tests use it.
|
||
If you like it, add it to your own collection of helpers.
|
||
|
||
`click()`辅助函数**不是**Angular测试工具之一。
|
||
它是在**本章的例子代码**中定义的函数方法,被所有测试例子所用。
|
||
如果你喜欢它,将它添加到你自己的辅助函数集。
|
||
:marked
|
||
Here's the previous test, rewritten using this click helper.
|
||
|
||
下面是使用了click辅助函数重新编写的上一个测试程序:
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'click-test-2', 'src/app/dashboard/dashboard-hero.component.spec.ts (click test revised)')(format='.')
|
||
|
||
.l-hr
|
||
|
||
a#component-inside-test-host
|
||
:marked
|
||
## Test a component inside a test host component
|
||
|
||
## 在测试宿主组件中测试组件
|
||
|
||
# 在测试宿主组件中测试组件
|
||
|
||
In the previous approach, the tests themselves played the role of the host `DashboardComponent`.
|
||
But does the `DashboardHeroComponent` work correctly when properly data-bound to a host component?
|
||
|
||
在前面的方法中,测试本身扮演了宿主组件`DashbaordComponent`的角色。
|
||
一种挥之不去的疑虑仍然存在:当正常数据绑定到宿主组件时,`DashboardHeroComponent`还会正常工作吗?
|
||
|
||
Testing with the actual `DashboardComponent` host is doable but seems more trouble than its worth.
|
||
It's easier to emulate the `DashboardComponent` host with a _test host_ like this one:
|
||
|
||
使用实际的`DashbaordComponent`宿主来测试是可行的,但是这么做似乎不合算。
|
||
像下面这样使用**测试宿主组件**来模拟`DashbaordComponent`显得更加容易:
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'test-host', 'src/app/dashboard/dashboard-hero.component.spec.ts (test host)')(format='.')
|
||
:marked
|
||
The test host binds to `DashboardHeroComponent` as the `DashboardComponent` would but without
|
||
the distraction of the `Router`, the `HeroService`, or even the `*ngFor` repeater.
|
||
|
||
测试宿主组件和`DashboardComponent`一样绑定`DashboardHeroComponent`,但是不用理会`Router`、`HeroService`服务,甚至`*ngFor`循环。
|
||
|
||
The test host sets the component's `hero` input property with its test hero.
|
||
It binds the component's `selected` event with its `onSelected` handler,
|
||
which records the emitted hero
|
||
in its `selectedHero` property. Later, the tests check that property to verify that the
|
||
`DashboardHeroComponent.selected` event emitted the right hero.
|
||
|
||
测试宿主将组件的`hero`导入属性设置为它的模拟英雄。
|
||
它将组件的`selected`事件绑定到它的`onSelected`处理器,使用`selectedHero`属性来记录发送来的英雄。
|
||
然后测试检查这个属性来验证`DashboardHeroComponent.selected`事件确实发送了正确的英雄。
|
||
|
||
The setup for the test-host tests is similar to the setup for the stand-alone tests:
|
||
|
||
配置使用测试宿主的测试程序与配置孤立测试相似:
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'test-host-setup', 'src/app/dashboard/dashboard-hero.component.spec.ts (test host setup)')(format='.')
|
||
:marked
|
||
This testing module configuration shows two important differences:
|
||
|
||
这个测试模块配置展示了两个非常重要的区别:
|
||
|
||
1. It _declares_ both the `DashboardHeroComponent` and the `TestHostComponent`.
|
||
|
||
它同时**声明**了`DashboardHeroComponent`和`TestHostComponent`。
|
||
|
||
1. It _creates_ the `TestHostComponent` instead of the `DashboardHeroComponent`.
|
||
|
||
它**创建**了`TestHostComponent`,而非`DashboardHeroComponent`。
|
||
|
||
The `createComponent` returns a `fixture` that holds an instance of `TestHostComponent` instead of an instance of `DashboardHeroComponent`.
|
||
|
||
`createComponent`返回的`fixture`里有`TestHostComponent`实例,而非`DashboardHeroComponent`组件实例。
|
||
|
||
Creating the `TestHostComponent` has the side-effect of creating a `DashboardHeroComponent`
|
||
because the latter appears within the template of the former.
|
||
The query for the hero element (`heroEl`) still finds it in the test DOM,
|
||
albeit at greater depth in the element tree than before.
|
||
|
||
当然,创建`TestHostComponent`有创建`DashboardHeroComponent`的副作用,因为后者出现在前者的模板中。
|
||
英雄元素(`heroEl`)的查询语句仍然可以在测试DOM中找到它,尽管元素树比以前更深。
|
||
|
||
The tests themselves are almost identical to the stand-alone version:
|
||
|
||
这些测试本身和它们的孤立版本几乎相同:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'test-host-tests', 'src/app/dashboard/dashboard-hero.component.spec.ts (test-host)')(format='.')
|
||
|
||
:marked
|
||
Only the selected event test differs. It confirms that the selected `DashboardHeroComponent` hero
|
||
really does find its way up through the event binding to the host component.
|
||
|
||
只有selected事件的测试不一样。它确保被选择的`DashboardHeroComponent`英雄确实通过事件绑定被传递到宿主组件。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#routed-component
|
||
:marked
|
||
## Test a routed component
|
||
|
||
## 测试带路由器的组件
|
||
|
||
Testing the actual `DashboardComponent` seemed daunting because it injects the `Router`.
|
||
|
||
测试实际的`DashbaordComponent`似乎令人生畏,因为它注入了`Router`。
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.ts', 'ctor', 'src/app/dashboard/dashboard.component.ts (constructor)')(format='.')
|
||
:marked
|
||
It also injects the `HeroService`, but faking that is a [familiar story](#component-with-async-service).
|
||
The `Router` has a complicated API and is entwined with other services and application preconditions.
|
||
|
||
它同时还注入了`HeroService`,但是我们已经知道如何[伪造](#component-with-async-servic)它。
|
||
`Router`的API非常复杂,并且它缠绕了其它服务和许多应用的先决条件。
|
||
|
||
Fortunately, the `DashboardComponent` isn't doing much with the `Router`
|
||
|
||
幸运的是,`DashbaordComponent`没有使用`Router`做很多事情。
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.ts', 'goto-detail', 'src/app/dashboard/dashboard.component.ts (goToDetail)')(format='.')
|
||
|
||
:marked
|
||
This is often the case.
|
||
As a rule you test the component, not the router,
|
||
and care only if the component navigates with the right address under the given conditions.
|
||
Stubbing the router with a test implementation is an easy option. This should do the trick:
|
||
|
||
通常都是这样的。原则上,你测试的是组件,不是路由器,应该只关心在指定的条件下,组件是否导航到正确的地址。
|
||
用模拟类来替换路由器是一种简单的方案。下面的代码应该可以:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.spec.ts', 'router-stub', 'src/app/dashboard/dashboard.component.spec.ts (Router Stub)')(format='.')
|
||
|
||
:marked
|
||
Now set up the testing module with the test stubs for the `Router` and `HeroService`, and
|
||
create a test instance of the `DashboardComponent` for subsequent testing.
|
||
|
||
现在我们来利用`Router`和`HeroService`的测试stub类来配置测试模块,并为接下来的测试创建`DashboardComponent`的测试实例。
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.spec.ts', 'compile-and-create-body', 'src/app/dashboard/dashboard.component.spec.ts (compile and create)')(format='.')
|
||
|
||
:marked
|
||
The following test clicks the displayed hero and confirms (with the help of a spy) that `Router.navigateByUrl` is called with the expected url.
|
||
|
||
下面的测试程序点击显示的英雄,并利用spy来确认`Router.navigateByUrl`被调用了,而且传进的url是所期待的值。
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.spec.ts', 'navigate-test', 'src/app/dashboard/dashboard.component.spec.ts (navigate test)')(format='.')
|
||
|
||
|
||
a#inject
|
||
:marked
|
||
### The _inject_ function
|
||
|
||
### _inject_函数
|
||
|
||
Notice the `inject` function in the second `it` argument.
|
||
|
||
注意第二个`it`参数里面的`inject`函数。
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard.component.spec.ts', 'inject')(format='.')
|
||
|
||
:marked
|
||
The `inject` function is one of the Angular testing utilities.
|
||
It injects services into the test function where you can alter, spy on, and manipulate them.
|
||
|
||
`inject`函数是Angular测试工具之一。
|
||
它注入服务到测试函数,以供修改、监视和操纵。
|
||
|
||
The `inject` function has two parameters:
|
||
|
||
`inject`函数有两个参数:
|
||
|
||
1. An array of Angular dependency injection tokens.
|
||
|
||
一列数组,包含了Angular依赖注入令牌
|
||
|
||
1. A test function whose parameters correspond exactly to each item in the injection token array.
|
||
|
||
一个测试函数,它的参数与注入令牌数组里的每个项目严格的一一对应。
|
||
|
||
.callout.is-important
|
||
header inject uses the TestBed Injector
|
||
|
||
header 使用TestBed注入器来注入
|
||
|
||
:marked
|
||
The `inject` function uses the current `TestBed` injector and can only return services provided at that level.
|
||
It does not return services from component providers.
|
||
|
||
`inject`函数使用当前`TestBed`注入器,并且只返回这个级别提供的服务。
|
||
它不会返回组件提供商提供的服务。
|
||
|
||
:marked
|
||
This example injects the `Router` from the current `TestBed` injector.
|
||
That's fine for this test because the `Router` is, and must be, provided by the application root injector.
|
||
|
||
这个例子通过当前的`TestBed`注入器来注入`Router`。
|
||
对这个测试程序来说,这是没问题的,因为`Router`是(也必须是)由应用的根注入器来提供。
|
||
|
||
If you need a service provided by the component's _own_ injector, call `fixture.debugElement.injector.get` instead:
|
||
|
||
如果你需要组件自己的注入器提供的服务,调用`fixture.debugElement.injector.get`:
|
||
+makeExample('testing/ts/src/app/welcome.component.spec.ts', 'injected-service', 'Component\'s injector')(format='.')
|
||
.alert.is-important
|
||
:marked
|
||
Use the component's own injector to get the service actually injected into the component.
|
||
|
||
使用组件自己的注入器来获取实际注入到组件的服务。
|
||
|
||
:marked
|
||
The `inject` function closes the current `TestBed` instance to further configuration.
|
||
You cannot call any more `TestBed` configuration methods, not `configureTestingModule`
|
||
nor any of the `override...` methods. The `TestBed` throws an error if you try.
|
||
|
||
`inject`函数关闭当前`TestBed`实例,使它无法再被配置。
|
||
你不能再调用任何`TestBed`配置方法、`configureTestModule`或者任何`override...`方法,否则`TestBed`将抛出错误。
|
||
|
||
.alert.is-important
|
||
:marked
|
||
Do not configure the `TestBed` after calling `inject`.
|
||
|
||
不要在调用`inject`以后再试图配置`TestBed`。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
a#routed-component-w-param
|
||
:marked
|
||
### Test a routed component with parameters
|
||
### 测试带有路由和路由参数的组件
|
||
Clicking a _Dashboard_ hero triggers navigation to `heroes/:id`, where `:id`
|
||
is a route parameter whose value is the `id` of the hero to edit.
|
||
That URL matches a route to the `HeroDetailComponent`.
|
||
|
||
点击**Dashboard**英雄触发导航到`heros/:id`,其中`:id`是路由参数,它的值是进行编辑的英雄的`id`。
|
||
这个URL匹配到`HeroDetailComponent`的路由。
|
||
|
||
The router pushes the `:id` token value into the `ActivatedRoute.params` _Observable_ property,
|
||
Angular injects the `ActivatedRoute` into the `HeroDetailComponent`,
|
||
and the component extracts the `id` so it can fetch the corresponding hero via the `HeroDetailService`.
|
||
Here's the `HeroDetailComponent` constructor:
|
||
|
||
路由器将`:id`令牌的值推送到`ActivatedRoute.params`**可观察**属性里,
|
||
Angular注入`ActivatedRoute`到`HeroDetailComponent`中,
|
||
然后组件提取`id`,这样它就可以通过`HeroDetailService`获取相应的英雄。
|
||
下面是`HeroDetailComponent`的构造函数:
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.ts', 'ctor', 'src/app/hero/hero-detail.component.ts (constructor)')(format='.')
|
||
:marked
|
||
`HeroDetailComponent` subscribes to `ActivatedRoute.params` changes in its `ngOnInit` method.
|
||
|
||
`HeroDetailComponent`在它的`ngOnInit`方法中监听`ActivatedRoute.params`的变化。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.ts', 'ng-on-init', 'src/app/hero/hero-detail.component.ts (ngOnInit)')(format='.')
|
||
.l-sub-section
|
||
:marked
|
||
The expression after `route.params` chains an _Observable_ operator that _plucks_ the `id` from the `params`
|
||
and then chains a `forEach` operator to subscribe to `id`-changing events.
|
||
The `id` changes every time the user navigates to a different hero.
|
||
|
||
`route.params`之后的表达式链接了**可观察**操作符,它从`params`中提取`id`,然后链接`forEach`操作符来订阅`id`变化事件。
|
||
每次`id`变化时,用户被导航到不同的英雄。
|
||
|
||
The `forEach` passes the new `id` value to the component's `getHero` method (not shown)
|
||
which fetches a hero and sets the component's `hero` property.
|
||
If the`id` parameter is missing, the `pluck` operator fails and the `catch` treats failure as a request to edit a new hero.
|
||
|
||
`forEach`将新的`id`值传递到组件的`getHero`方法(这里没有列出来),它获取英雄并将它赋值到组件的`hero`属性。
|
||
如果`id`参数无效,`pluck`操作符就会失败,`catch`将失败当作创建新英雄来处理。
|
||
|
||
The [Router](router.html#route-parameters) guide covers `ActivatedRoute.params` in more detail.
|
||
|
||
[路由器](router.html#route-parameters)章更详尽的讲述了`ActivatedRoute.params`。
|
||
|
||
:marked
|
||
A test can explore how the `HeroDetailComponent` responds to different `id` parameter values
|
||
by manipulating the `ActivatedRoute` injected into the component's constructor.
|
||
|
||
通过操纵被注入到组件构造函数的`ActivatedRoute`服务,测试程序可以探索`HeroDetailComponent`是如何对不同的`id`参数值作出响应的。
|
||
|
||
By now you know how to stub the `Router` and a data service.
|
||
Stubbing the `ActivatedRoute` follows the same pattern except for a complication:
|
||
the `ActivatedRoute.params` is an _Observable_.
|
||
|
||
现在,你已经知道如何模拟`Router`和数据服务。
|
||
模拟`ActivatedRoute`遵循类似的模式,但是有个额外枝节:`ActivatedRoute.params`是**可观察对象**。
|
||
|
||
a#stub-observable
|
||
:marked
|
||
### Create an _Observable_ test double
|
||
|
||
### **可观察对象**的测试替身
|
||
|
||
The `hero-detail.component.spec.ts` relies on an `ActivatedRouteStub` to set `ActivatedRoute.params` values for each test.
|
||
This is a cross-application, re-usable _test helper class_.
|
||
Consider placing such helpers in a `testing` folder sibling to the `app` folder.
|
||
This sample keeps `ActivatedRouteStub` in `testing/router-stubs.ts`:
|
||
|
||
`hero-detail.component.spec.ts`依赖`ActivatedRouteStub`来为每个测试程序设置`ActivatedRoute.params`值。
|
||
它是跨应用、可复用的**测试辅助类**。
|
||
我们建议将这样的辅助类放到`app`目录下的名为`testing`的目录。
|
||
本例把`ActivatedRouteStub`放到`testing/router-stubs.ts`:
|
||
|
||
+makeExample('testing/ts/src/testing/router-stubs.ts', 'activated-route-stub', 'testing/router-stubs.ts (ActivatedRouteStub)')(format='.')
|
||
:marked
|
||
Notable features of this stub are:
|
||
|
||
这个stub类有下列值得注意的特征:
|
||
|
||
* The stub implements only two of the `ActivatedRoute` capabilities: `params` and `snapshot.params`.
|
||
|
||
这个stub类只实现`ActivatedRoute`的两个功能:`params`和`snapshot.params`。
|
||
|
||
* <a href="https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/behaviorsubject.md" target="_blank">_BehaviorSubject_</a>
|
||
drives the stub's `params` _Observable_ and returns the same value to every `params` subscriber until it's given a new value.
|
||
|
||
<a href="https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/behaviorsubject.md" target="_blank">_BehaviorSubject_</a>驱使这个stub类的`params`可观察对象,并为每个`params`的订阅者返回同样的值,直到它接受到新值。
|
||
|
||
* The `HeroDetailComponent` chains its expressions to this stub `params` _Observable_ which is now under the tester's control.
|
||
|
||
`HeroDetailComponent`链接它的表达式到这个stub类的`params`可观察对象,该对象现在被测试者的控制之下。
|
||
|
||
* Setting the `testParams` property causes the `subject` to push the assigned value into `params`.
|
||
That triggers the `HeroDetailComponent` _params_ subscription, described above, in the same way that navigation does.
|
||
|
||
设置`testParams`属性导致`subject`将指定的值推送进`params`。它触发上面描述过的`HeroDetailComponent`的`params`订阅,和导航的方式一样。
|
||
|
||
* Setting the `testParams` property also updates the stub's internal value for the `snapshot` property to return.
|
||
|
||
设置`testParams`属性同时更新这个stub类内部值,用于`snapshot`属性的返回。
|
||
|
||
.l-sub-section(style="margin-left:30px")
|
||
:marked
|
||
The [_snapshot_](router.html#snapshot "Router guide: snapshot") is another popular way for components to consume route parameters.
|
||
|
||
[_snapshot_](router.html#snapshot "Router Chapter: snapshot")是组件使用路由参数的另一种流行的方法。
|
||
.callout.is-helpful
|
||
:marked
|
||
The router stubs in this guide are meant to inspire you. Create your own stubs to fit your testing needs.
|
||
|
||
本章的路由器stub类是为了给你灵感。创建你自己的stub类,以适合你的测试需求。
|
||
|
||
a#tests-w-observable-double
|
||
:marked
|
||
### Testing with the _Observable_ test double
|
||
|
||
### 测试**可观察对象**的替身
|
||
|
||
Here's a test demonstrating the component's behavior when the observed `id` refers to an existing hero:
|
||
|
||
下面的测试程序是演示组件在被观察的`id`指向现有英雄时的行为:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'route-good-id', 'src/app/hero/hero-detail.component.spec.ts (existing id)')(format='.')
|
||
.l-sub-section
|
||
:marked
|
||
The `createComponent` method and `page` object are discussed [in the next section](#page-object).
|
||
Rely on your intuition for now.
|
||
|
||
[下一节](#page-object)将解释`createComponent`方法和`page`对象,现在暂时跟着自己的直觉走。
|
||
|
||
:marked
|
||
When the `id` cannot be found, the component should re-route to the `HeroListComponent`.
|
||
The test suite setup provided the same `RouterStub` [described above](#routed-component) which spies on the router without actually navigating.
|
||
This test supplies a "bad" id and expects the component to try to navigate.
|
||
|
||
当无法找到`id`时,组件应该重新导航到`HeroListComponent`。
|
||
该测试套件配置与[上面描述](#routed-component)的`RouterStub`一样,它在不实际导航的情况下刺探路由器。
|
||
该测试程序提供了“坏”的id,期望组件尝试导航。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'route-bad-id', 'src/app/hero/hero-detail.component.spec.ts (bad id)')(format='.')
|
||
:marked
|
||
:marked
|
||
While this app doesn't have a route to the `HeroDetailComponent` that omits the `id` parameter, it might add such a route someday.
|
||
The component should do something reasonable when there is no `id`.
|
||
|
||
虽然本应用没有在缺少`id`参数的时候,继续导航到`HeroDetailComponent`的路由,但是,将来它可能会添加这样的路由。
|
||
当没有`id`时,该组件应该作出合理的反应。
|
||
|
||
In this implementation, the component should create and display a new hero.
|
||
New heroes have `id=0` and a blank `name`. This test confirms that the component behaves as expected:
|
||
|
||
在本例中,组件应该创建和显示新英雄。
|
||
新英雄的`id`为零,`name`为空。本测试程序确认组件是按照预期的这样做的:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'route-no-id', 'src/app/hero/hero-detail.component.spec.ts (no id)')(format='.')
|
||
:marked
|
||
.callout.is-helpful
|
||
:marked
|
||
Inspect and download _all_ of the guide's application test code with this <live-example plnkr="app-specs" embedded-style>live example</live-example>.
|
||
|
||
到<live-example plnkr="app-specs">在线例子</live-example>查看和下载**所有**本章应用程序测试代码。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#page-object
|
||
:marked
|
||
## Use a _page_ object to simplify setup
|
||
|
||
## 使用**page**对象来简化配置
|
||
|
||
The `HeroDetailComponent` is a simple view with a title, two hero fields, and two buttons.
|
||
|
||
`HeroDetailComponent`是带有标题、两个英雄字段和两个按钮的简单视图。
|
||
|
||
figure.image-display
|
||
img(src='/resources/images/devguide/testing/hero-detail.component.png' alt="HeroDetailComponent in action")
|
||
|
||
:marked
|
||
But there's already plenty of template complexity.
|
||
|
||
但是它已经有很多模板复杂性。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.html', '', 'src/app/hero/hero-detail.component.html')(format='.')
|
||
|
||
:marked
|
||
To fully exercise the component, the test needs a lot of setup:
|
||
|
||
要彻底测试该组件,测试程序需要一系列设置:
|
||
|
||
* It must wait until a hero arrives before `*ngIf` allows any element in DOM.
|
||
|
||
它必须在`*ngIf`允许元素进入DOM之前,等待`hero`的到来
|
||
|
||
* It needs references to the title `<span>` and the name `<input>` so it can inspect their values.
|
||
|
||
它需要标题名字span和名字输入框元素的引用,用来检查它们的值
|
||
|
||
* It needs references to the two buttons so it can click them.
|
||
|
||
它需要两个按钮的引用,以便点击它们
|
||
|
||
* It needs spies for some of the component and router methods.
|
||
|
||
刺探(spy)组件和路由器的方法
|
||
|
||
Even a small form such as this one can produce a mess of tortured conditional setup and CSS element selection.
|
||
|
||
即使是像这样一个很小的表单,也能产生令人疯狂的错综复杂的条件设置和CSS元素选择。
|
||
|
||
Tame the madness with a `Page` class that simplifies access to component properties and encapsulates the logic that sets them.
|
||
Here's the `Page` class for the `hero-detail.component.spec.ts`
|
||
|
||
通过简化组件属性的访问和封装设置属性的逻辑,`Page`类可以轻松解决这个令人抓狂的难题。
|
||
下面是为`hero-detail.component.spec.ts`准备的`page`类:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'page', 'src/app/hero/hero-detail.component.spec.ts (Page)')(format='.')
|
||
:marked
|
||
Now the important hooks for component manipulation and inspection are neatly organized and accessible from an instance of `Page`.
|
||
|
||
现在,用来操作和检查组件的重要钩子都被井然有序的组织起来了,可以通过`page`实例来使用它们。
|
||
|
||
A `createComponent` method creates a `page` objectand fills in the blanks once the `hero` arrives.
|
||
|
||
`createComponent`方法创建`page`,在`hero`到来时,自动填补空白。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'create-component', 'src/app/hero/hero-detail.component.spec.ts (createComponent)')(format='.')
|
||
|
||
:marked
|
||
The [observable tests](#tests-w-observable-double) in the previous section demonstrate how `createComponent` and `page`
|
||
keep the tests short and _on message_.
|
||
There are no distractions: no waiting for promises to resolve and no searching the DOM for element values to compare.
|
||
|
||
上一节的[可观察对象测试](#tests-w-observable-double)展示了`createComponent`和`page`如何让测试程序简短和即时。
|
||
没有任何干扰:无需等待承诺的解析,也没有搜索DOM元素值进行比较。
|
||
|
||
Here are a few more `HeroDetailComponent` tests to drive the point home.
|
||
|
||
这里是一些更多的`HeroDetailComponent`测试程序,进一步的展示了这一点。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'selected-tests', 'src/app/hero/hero-detail.component.spec.ts (selected tests)')(format='.')
|
||
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
.l-hr
|
||
|
||
a#import-module
|
||
:marked
|
||
## Setup with module imports
|
||
|
||
## 模块导入imports的配置
|
||
|
||
Earlier component tests configured the testing module with a few `declarations` like this:
|
||
|
||
此前的组件测试程序使用了一些`declarations`来配置模块,就像这样:
|
||
|
||
+makeExample('testing/ts/src/app/dashboard/dashboard-hero.component.spec.ts', 'compile-components', 'src/app/dashboard/dashboard-hero.component.spec.ts (config)')(format='.')
|
||
:marked
|
||
The `DashboardComponent` is simple. It needs no help.
|
||
But more complex components often depend on other components, directives, pipes, and providers
|
||
and these must be added to the testing module too.
|
||
|
||
`DashbaordComponent`非常简单。它不需要帮助。
|
||
但是更加复杂的组件通常依赖其它组件、指令、管道和提供商,
|
||
所以这些必须也被添加到测试模块中。
|
||
|
||
Fortunately, the `TestBed.configureTestingModule` parameter parallels
|
||
the metadata passed to the `@NgModule` decorator
|
||
which means you can also specify `providers` and `imports`.
|
||
|
||
幸运的是,`TestBed.configureTestingModule`参数与传入`@NgModule`装饰器的元数据一样,也就是所你也可以指定`providers`和`imports`.
|
||
|
||
The `HeroDetailComponent` requires a lot of help despite its small size and simple construction.
|
||
In addition to the support it receives from the default testing module `CommonModule`, it needs:
|
||
|
||
虽然`HeroDetailComponent`很小,结构也很简单,但是它需要很多帮助。
|
||
除了从默认测试模块`CommonModule`中获得的支持,它还需要:
|
||
|
||
* `NgModel` and friends in the `FormsModule` to enable two-way data binding.
|
||
|
||
`FormsModule`里的`NgModel`和其它,来进行双向数据绑定
|
||
|
||
* The `TitleCasePipe` from the `shared` folder.
|
||
|
||
`shared`目录里的`TitleCasePipe`
|
||
|
||
* Router services (which these tests are stubbing).
|
||
|
||
一些路由器服务(测试程序将stub伪造它们)
|
||
|
||
* Hero data access services (also stubbed).
|
||
|
||
英雄数据访问服务(同样被stub伪造了)
|
||
|
||
One approach is to configure the testing module from the individual pieces as in this example:
|
||
|
||
一种方法是在测试模块中一一配置,就像这样:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'setup-forms-module', 'src/app/hero/hero-detail.component.spec.ts (FormsModule setup)')(format='.')
|
||
|
||
:marked
|
||
Because many app components need the `FormsModule` and the `TitleCasePipe`, the developer created
|
||
a `SharedModule` to combine these and other frequently requested parts.
|
||
The test configuration can use the `SharedModule` too as seen in this alternative setup:
|
||
|
||
因为许多应用组件需要`FormsModule`和`TitleCasePipe`,所以开发者创建了`SharedModule`来合并它们和一些频繁需要的部件。
|
||
测试配置也可以使用`SharedModule`,请看下面另一种配置:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'setup-shared-module', 'src/app/hero/hero-detail.component.spec.ts (SharedModule setup)')(format='.')
|
||
|
||
:marked
|
||
It's a bit tighter and smaller, with fewer import statements (not shown).
|
||
|
||
它的导入声明少一些(未显示),稍微干净一些,小一些。
|
||
|
||
a#feature-module-import
|
||
:marked
|
||
### Import the feature module
|
||
|
||
### 导入特性模块
|
||
|
||
The `HeroDetailComponent` is part of the `HeroModule` [Feature Module](ngmodule.html#feature-modules) that aggregates more of the interdependent pieces
|
||
including the `SharedModule`.
|
||
Try a test configuration that imports the `HeroModule` like this one:
|
||
|
||
`HeroDetailComponent`是`HeroModule`[特性模块](ngmodule.html#feature-modules)的一部分,它组合了更多互相依赖的部件,包括`SharedModule`。
|
||
试试下面这个导入`HeroModule`的测试配置:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'setup-hero-module', 'src/app/hero/hero-detail.component.spec.ts (HeroModule setup)')(format='.')
|
||
|
||
:marked
|
||
That's _really_ crisp. Only the _test doubles_ in the `providers` remain. Even the `HeroDetailComponent` declaration is gone.
|
||
|
||
这样特别清爽。只有`providers`里面的测试替身被保留。连`HeroDetailComponent`声明都消失了。
|
||
.l-sub-section
|
||
:marked
|
||
In fact, if you try to declare it, Angular throws an error because
|
||
`HeroDetailComponent` is declared in both the `HeroModule` and the `DynamicTestModule` (the testing module).
|
||
|
||
事实上,如果里试图声明它,Angular会抛出错误,因为`HeroDetailComponent`已经在`HeroModule`和测试模块的`DynamicTestModule`中声明。
|
||
|
||
.alert.is-helpful
|
||
:marked
|
||
Importing the component's feature module is often the easiest way to configure the tests,
|
||
especially when the feature module is small and mostly self-contained, as feature modules should be.
|
||
|
||
导入组件的特性模块通常是最简单的配置测试的方法,
|
||
尤其是当特性模块很小而且几乎自包含时...特性模块应该是自包含的。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
.l-hr
|
||
|
||
a#component-override
|
||
:marked
|
||
## Override a component's providers
|
||
|
||
## 重载组件的提供商
|
||
|
||
The `HeroDetailComponent` provides its own `HeroDetailService`.
|
||
|
||
`HeroDetailComponent`提供自己的`HeroDetailService`服务。
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.ts', 'prototype', 'src/app/hero/hero-detail.component.ts (prototype)')(format='.')
|
||
|
||
:marked
|
||
It's not possible to stub the component's `HeroDetailService` in the `providers` of the `TestBed.configureTestingModule`.
|
||
Those are providers for the _testing module_, not the component. They prepare the dependency injector at the _fixture level_.
|
||
|
||
在`TestBed.configureTestingModule`的`providers`中stub伪造组件的`HeroDetailService`是不可行的。
|
||
这些是**测试模块**的提供商,而非组件的。组件级别的供应商应该在**fixture级别**准备的依赖注入器。
|
||
|
||
Angular creates the component with its _own_ injector, which is a _child_ of the fixture injector.
|
||
It registers the component's providers (the `HeroDetailService` in this case) with the child injector.
|
||
A test cannot get to child injector services from the fixture injector.
|
||
And `TestBed.configureTestingModule` can't configure them either.
|
||
|
||
Angular创建组件时,该组件有自己的注入器,它是fixture注入器的子级。
|
||
Angular使用这个子级注入器来注册组件的提供商(也就是`HeroDetailService`)。
|
||
测试程序无法从fixture的注入器获取这个子级注入器。
|
||
而且`TestBed.configureTestingModule`也无法配置它们。
|
||
|
||
Angular has been creating new instances of the real `HeroDetailService` all along!
|
||
|
||
Angular始终都在创建真实`HeroDetailService`的实例。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
These tests could fail or timeout if the `HeroDetailService` made its own XHR calls to a remote server.
|
||
There might not be a remote server to call.
|
||
|
||
如果`HeroDetailService`向远程服务器发出自己的XHR请求,这些测试可能会失败或者超时。
|
||
这个远程服务器可能根本不存在。
|
||
|
||
Fortunately, the `HeroDetailService` delegates responsibility for remote data access to an injected `HeroService`.
|
||
|
||
幸运的是,`HeroDetailService`将远程数据访问的责任交给了注入进来的`HeroService`。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.service.ts', 'prototype', 'src/app/hero/hero-detail.service.ts (prototype)')(format='.')
|
||
:marked
|
||
The [previous test configuration](#feature-module-import) replaces the real `HeroService` with a `FakeHeroService`
|
||
that intercepts server requests and fakes their responses.
|
||
|
||
[之前的测试配置](#feature-module-import)将真实的`HeroService`替换为`FakeHeroService`,拦截了服务起请求,伪造了它们的响应。
|
||
|
||
:marked
|
||
What if you aren't so lucky. What if faking the `HeroService` is hard?
|
||
What if `HeroDetailService` makes its own server requests?
|
||
|
||
如果我们没有这么幸运怎么办?如果伪造`HeroService`很难怎么办?如果`HeroDetailService`自己发出服务器请求怎么办?
|
||
|
||
The `TestBed.overrideComponent` method can replace the component's `providers` with easy-to-manage _test doubles_
|
||
as seen in the following setup variation:
|
||
|
||
`TestBed.overrideComponent`方法可以将组件的`providers`替换为容易管理的**测试替身**,参见下面的设置变化:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'setup-override', 'src/app/hero/hero-detail.component.spec.ts (Override setup)')(format='.')
|
||
|
||
:marked
|
||
Notice that `TestBed.configureTestingModule` no longer provides a (fake) `HeroService` because it's [not needed](#spy-stub).
|
||
|
||
注意,`TestBed.configureTestingModule`不再提供(伪造)`HeroService`,因为已经[没有必要了](#stub-hero-detail-service)。
|
||
|
||
a#override-component-method
|
||
:marked
|
||
### The _overrideComponent_ method
|
||
|
||
### **overrideComponent**方法
|
||
|
||
Focus on the `overrideComponent` method.
|
||
|
||
注意这个`overrideComponent`方法。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'override-component-method', 'src/app/hero/hero-detail.component.spec.ts (overrideComponent)')(format='.')
|
||
:marked
|
||
It takes two arguments: the component type to override (`HeroDetailComponent`) and an override metadata object.
|
||
The [overide metadata object](#metadata-override-object) is a generic defined as follows:
|
||
|
||
它接受两个参数:要重载的组件类型(`HeroDetailComponent`)和用于重载的元数据对象。
|
||
[重载元数据对象](#metadata-override-object)是泛型类,就像这样:
|
||
|
||
code-example(format="." language="javascript").
|
||
type MetadataOverride<T> = {
|
||
add?: T;
|
||
remove?: T;
|
||
set?: T;
|
||
};
|
||
:marked
|
||
A metadata override object can either add-and-remove elements in metadata properties or completely reset those properties.
|
||
This example resets the component's `providers` metadata.
|
||
|
||
元数据重载对象可以添加和删除元数据属性的项目,也可以彻底重设这些属性。
|
||
这个例子重新设置了组件的`providers`元数据。
|
||
|
||
The type parameter, `T`, is the kind of metadata you'd pass to the `@Component` decorator:
|
||
|
||
这个类型参数,`T`,是你会传递给`@Component`装饰器的元数据的类型。
|
||
code-example(format="." language="javascript").
|
||
selector?: string;
|
||
template?: string;
|
||
templateUrl?: string;
|
||
providers?: any[];
|
||
...
|
||
|
||
a#spy-stub
|
||
:marked
|
||
### Provide a _spy stub_ (_HeroDetailServiceSpy_)
|
||
|
||
### 提供一个*刺探桩(Spy stub)*(`HeroDetailServiceSpy`)
|
||
|
||
This example completely replaces the component's `providers` array with a new array containing a `HeroDetailServiceSpy`.
|
||
|
||
这个例子把组件的`providers`数组完全替换成了一个包含`HeroDetailServiceSpy`的新数组。
|
||
|
||
The `HeroDetailServiceSpy` is a stubbed version of the real `HeroDetailService`
|
||
that fakes all necessary features of that service.
|
||
It neither injects nor delegates to the lower level `HeroService`
|
||
so there's no need to provide a test double for that.
|
||
|
||
`HeroDetailServiceSpy`是实际`HeroDetailService`服务的桩版本,它伪造了该服务的所有必要特性。
|
||
但它既不需要注入也不会委托给低层的`HeroService`服务,因此我们不用为`HeroService`提供测试替身。
|
||
|
||
The related `HeroDetailComponent` tests will assert that methods of the `HeroDetailService`
|
||
were called by spying on the service methods.
|
||
Accordingly, the stub implements its methods as spies:
|
||
|
||
通过对该服务的方法进行刺探,`HeroDetailComponent`的关联测试将会对`HeroDetailService`是否被调用过进行断言。
|
||
因此,这个桩类会把它的方法实现为刺探方法:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'hds-spy', 'src/app/hero/hero-detail.component.spec.ts (HeroDetailServiceSpy)')(format='.')
|
||
|
||
a#override-tests
|
||
:marked
|
||
### The override tests
|
||
|
||
### 重载的测试程序
|
||
|
||
Now the tests can control the component's hero directly by manipulating the spy-stub's `testHero`
|
||
and confirm that service methods were called.
|
||
|
||
现在,测试程序可以通过操控stub的`testHero`,直接控制组件的英雄,并确保服务的方法被调用过。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'override-tests', 'src/app/hero/hero-detail.component.spec.ts (override tests)')(format='.')
|
||
|
||
a#more-overrides
|
||
|
||
:marked
|
||
### More overrides
|
||
|
||
### 更多重载
|
||
|
||
The `TestBed.overrideComponent` method can be called multiple times for the same or different components.
|
||
The `TestBed` offers similar `overrideDirective`, `overrideModule`, and `overridePipe` methods
|
||
for digging into and replacing parts of these other classes.
|
||
|
||
`TestBed.overrideComponent`方法可以在相同或不同的组件中被反复调用。
|
||
`TestBed`还提供了类似的`overrideDirective`、`overrideModule`和`overridePipe`方法,用来深入并重载这些其它类的部件。
|
||
|
||
Explore the options and combinations on your own.
|
||
|
||
自己探索这些选项和组合。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
.l-hr
|
||
|
||
a#router-outlet-component
|
||
:marked
|
||
## Test a _RouterOutlet_ component
|
||
|
||
## 测试带有_RouterOutlet_的组件
|
||
|
||
The `AppComponent` displays routed components in a `<router-outlet>`.
|
||
It also displays a navigation bar with anchors and their `RouterLink` directives.
|
||
|
||
`AppComponent`在`<router-outlet>`中显示导航组件。
|
||
它还显示了导航条,包含了链接和它们的`RouterLink`指令。
|
||
|
||
a#app-component-html
|
||
|
||
+makeExample('testing/ts/src/app/app.component.html', '', 'src/app/app.component.html')(format='.')
|
||
|
||
:marked
|
||
The component class does nothing.
|
||
|
||
组件的类没有做任何事。
|
||
|
||
+makeExample('testing/ts/src/app/app.component.ts', '', 'src/app/app.component.ts')(format='.')
|
||
|
||
:marked
|
||
Unit tests can confirm that the anchors are wired properly without engaging the router.
|
||
See why this is worth doing [below](#why-stubbed-routerlink-tests).
|
||
|
||
在不涉及路由的情况下,单元测试可以确认链接的设置是否正确。
|
||
参见[下面的内容](#why-stubbed-routerlink-tests),了解为什么值得这么做。
|
||
|
||
a#stub-component
|
||
:marked
|
||
### Stubbing unneeded components
|
||
|
||
### stub伪造不需要的组件
|
||
|
||
The test setup should look familiar.
|
||
|
||
该测试配置应该看起来很眼熟:
|
||
|
||
+makeExample('testing/ts/src/app/app.component.spec.ts', 'setup-stubs', 'src/app/app.component.spec.ts (Stub Setup)')(format='.')
|
||
|
||
:marked
|
||
The `AppComponent` is the declared test subject.
|
||
|
||
`AppComponent`是被声明的测试对象。
|
||
|
||
The setup extends the default testing module with one real component (`BannerComponent`) and several stubs.
|
||
|
||
使用一个真实的组件(`BannerComponent`)和几个stub,该配置扩展了默认测试模块。
|
||
|
||
* `BannerComponent` is simple and harmless to use as is.
|
||
|
||
原样使用`BannerComponent`非常简单而且无害。
|
||
|
||
* The real `WelcomeComponent` has an injected service. `WelcomeStubComponent` is a placeholder with no service to worry about.
|
||
|
||
真实的`WelcomeComponent`有被注入的服务。`WelcomeStubComponent`是无服务的替代品。
|
||
|
||
* The real `RouterOutlet` is complex and errors easily.
|
||
The `RouterOutletStubComponent` (in `testing/router-stubs.ts`) is safely inert.
|
||
|
||
真实的`RouterOutlet`很复杂而且容易出错。
|
||
`testing/router-stubs.ts`里的`RouterOutletStubComponent`是安全的替代品。
|
||
|
||
The component stubs are essential.
|
||
Without them, the Angular compiler doesn't recognize the `<app-welcome>` and `<router-outlet>` tags
|
||
and throws an error.
|
||
|
||
组件stub替代品很关键。
|
||
没有它们,Angular编译器无法识别`<app-welcome`和`<router-outlet>`标签,抛出错误。
|
||
|
||
a#router-link-stub
|
||
:marked
|
||
### Stubbing the _RouterLink_
|
||
|
||
### Stub伪造_RouterLink_
|
||
|
||
The `RouterLinkStubDirective` contributes substantively to the test:
|
||
|
||
`RouterLinkStubDirective`为测试作出了重要的贡献:
|
||
|
||
+makeExample('testing/ts/src/testing/router-stubs.ts', 'router-link', 'testing/router-stubs.ts (RouterLinkStubDirective)')(format='.')
|
||
:marked
|
||
The `host` metadata property wires the click event of the host element (the `<a>`) to the directive's `onClick` method.
|
||
The URL bound to the `[routerLink]` attribute flows to the directive's `linkParams` property.
|
||
Clicking the anchor should trigger the `onClick` method which sets the telltale `navigatedTo` property.
|
||
Tests can inspect that property to confirm the expected _click-to-navigation_ behavior.
|
||
|
||
`host`元数据属性将宿主元素(`<a>`)的click事件与指令的`onClick`方法关联起来。
|
||
绑定到`[routerLink]`的URL属性被传递到指令的`linkParams`属性。
|
||
点击这个链接应该能触发`onClick`方法,从而设置`navigatedTo`属性。
|
||
测试程序可以查看这个属性,来确认期望的**点击导航**行为。
|
||
|
||
a#by-directive
|
||
a#inject-directive
|
||
:marked
|
||
### _By.directive_ and injected directives
|
||
|
||
### _By.directive_和注入的指令
|
||
|
||
A little more setup triggers the initial data binding and gets references to the navigation links:
|
||
|
||
再一步配置触发了数据绑定的初始化,获取导航链接的引用:
|
||
+makeExample('testing/ts/src/app/app.component.spec.ts', 'test-setup', 'src/app/app.component.spec.ts (test setup)')(format='.')
|
||
:marked
|
||
Two points of special interest:
|
||
|
||
特别值得注意的两点:
|
||
|
||
1. You can locate elements _by directive_, using `By.directive`, not just by css selectors.
|
||
|
||
你还可以按指令定位元素,使用`By.directive`,而不仅仅是通过CSS选择器。
|
||
|
||
1. You can use the component's dependency injector to get an attached directive because
|
||
Angular always adds attached directives to the component's injector.
|
||
|
||
你可以使用组件的依赖注入器来获取附加的指令,因为Angular总是将附加组件添加到组件的注入器中。
|
||
|
||
a#app-component-tests
|
||
:marked
|
||
Here are some tests that leverage this setup:
|
||
|
||
下面是一些使用这个配置的测试程序:
|
||
|
||
+makeExample('testing/ts/src/app/app.component.spec.ts', 'tests', 'src/app/app.component.spec.ts (selected tests)')(format='.')
|
||
.l-sub-section
|
||
:marked
|
||
The "click" test _in this example_ is worthless.
|
||
It works hard to appear useful when in fact it
|
||
tests the `RouterLinkStubDirective` rather than the _component_.
|
||
This is a common failing of directive stubs.
|
||
|
||
本例中的“click”测试程序其实毫无价值。
|
||
它显得很有用,但是事实上,它测试的是`RouterLinkStubDirective`,而非测试组件。
|
||
这是指令stub的通病。
|
||
|
||
It has a legitimate purpose in this guide.
|
||
It demonstrates how to find a `RouterLink` element, click it, and inspect a result,
|
||
without engaging the full router machinery.
|
||
This is a skill you may need to test a more sophisticated component, one that changes the display,
|
||
re-calculates parameters, or re-arranges navigation options when the user clicks the link.
|
||
|
||
在本章中,它有存在的必要。
|
||
它演示了如何在不涉及完整路由器机制的情况下,如何找到`RouterLink`元素、点击它并检查结果。
|
||
要测试更复杂的组件,你可能需要具备这样的能力,能改变视图和重新计算参数,或者当用户点击链接时,有能力重新安排导航选项。
|
||
|
||
a#why-stubbed-routerlink-tests
|
||
:marked
|
||
### What good are these tests?
|
||
|
||
### 这些测试有什么好处?
|
||
|
||
Stubbed `RouterLink` tests can confirm that a component with links and an outlet is setup properly,
|
||
that the component has the links it should have, and that they are all pointing in the expected direction.
|
||
These tests do not concern whether the app will succeed in navigating to the target component when the user clicks a link.
|
||
|
||
stub伪造的`RouterLink`测试可以确认带有链接和outlet的组件的设置的正确性,确认组件有应该有的链接,确认它们都指向了正确的方向。
|
||
这些测试程序不关心用户点击链接时,应用是否会成功的导航到目标组件。
|
||
|
||
Stubbing the RouterLink and RouterOutlet is the best option for such limited testing goals.
|
||
Relying on the real router would make them brittle.
|
||
They could fail for reasons unrelated to the component.
|
||
For example, a navigation guard could prevent an unauthorized user from visiting the `HeroListComponent`.
|
||
That's not the fault of the `AppComponent` and no change to that component could cure the failed test.
|
||
|
||
对于这样局限的测试目标,stub伪造RouterLink和RouterOutlet是最佳选择。
|
||
依靠真正的路由器会让它们很脆弱。
|
||
它们可能因为与组件无关的原因而失败。
|
||
例如,一个导航守卫可能防止没有授权的用户访问`HeroListComponent`。
|
||
这并不是`AppComponent`的过错,并且无论该组件怎么改变都无法修复这个失败的测试程序。
|
||
|
||
A _different_ battery of tests can explore whether the application navigates as expected
|
||
in the presence of conditions that influence guards such as whether the user is authenticated and authorized.
|
||
|
||
不同的测试程序可以探索在不同条件下(比如像检查用户是否认证),该应用是否和期望的那样导航。
|
||
|
||
.alert.is-helpful
|
||
:marked
|
||
A future guide update will explain how to write such tests with the `RouterTestingModule`.
|
||
|
||
未来本章的更新将介绍如何使用`RouterTestingModule`来编写这样的测试程序。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
.l-hr
|
||
|
||
a#shallow-component-test
|
||
:marked
|
||
## "Shallow component tests" with *NO\_ERRORS\_SCHEMA*
|
||
|
||
## 使用*NO\_ERRORS\_SCHEMA*来“浅化”组件测试程序
|
||
|
||
The [previous setup](#stub-component) declared the `BannerComponent` and stubbed two other components
|
||
for _no reason other than to avoid a compiler error_.
|
||
|
||
[以前的配置](#stub-component)声明了`BannerComponent`,并stub伪造了两个其它组件,**仅仅是为了避免编译错误,不是为别的原因**。
|
||
|
||
Without them, the Angular compiler doesn't recognize the `<app-banner>`, `<app-welcome>` and `<router-outlet>` tags
|
||
in the [_app.component.html_](#app-component-html) template and throws an error.
|
||
|
||
没有它们,Angular编译器无法识别[_app.component.html_](#app-component-html)模板里的`<app-banner>`、`<app-welcome>`和`<router-outlet>`标签,并抛出错误。
|
||
|
||
Add `NO_ERRORS_SCHEMA` to the testing module's `schemas` metadata
|
||
to tell the compiler to ignore unrecognized elements and attributes.
|
||
You no longer have to declare irrelevant components and directives.
|
||
|
||
添加`NO_ERRORS_SCHEMA`到测试模块的`schemas`元数据中,告诉编译器忽略不认识的元素和属性。
|
||
这样你不再需要声明无关组件和指令。
|
||
|
||
These tests are ***shallow*** because they only "go deep" into the components you want to test.
|
||
Here is a setup, with `import` statements, that demonstrates the improved simplicity of _shallow_ tests, relative to the stubbing setup.
|
||
|
||
这些测试程序比较**浅**,因为它们只“深入”到你要测试的组件。
|
||
这里是一套配置(拥有`import`语句),体现了相比使用stub伪造的配置来说,**浅**测试程序的简单性。
|
||
|
||
+makeTabs('testing/ts/src/app/app.component.spec.ts, testing/ts/src/app/app.component.spec.ts',
|
||
'setup-schemas, setup-stubs-w-imports',
|
||
'src/app/app.component.spec.ts (NO_ERRORS_SCHEMA), src/app/app.component.spec.ts (Stubs)')(format='.')
|
||
|
||
:marked
|
||
The _only_ declarations are the _component-under-test_ (`AppComponent`) and the `RouterLinkStubDirective`
|
||
that contributes actively to the tests.
|
||
The [tests in this example](#app-component-tests) are unchanged.
|
||
|
||
这里**唯一**声明的是**被测试的组件**(`AppComponent`)和测试需要的`RouterLinkStubDirective`。
|
||
没有改变任何[原测试程序](#app-component-tests)。
|
||
|
||
.alert.is-important
|
||
:marked
|
||
_Shallow component tests_ with `NO_ERRORS_SCHEMA` greatly simplify unit testing of complex templates.
|
||
However, the compiler no longer alerts you to mistakes
|
||
such as misspelled or misused components and directives.
|
||
|
||
使用`NO_ERRORS_SCHEMA`的**浅组件测试程序**很大程度上简化了拥有复杂模板组件的单元测试。
|
||
但是,编译器将不再提醒你一些错误,比如模板中拼写错误或者误用的组件和指令。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#attribute-directive
|
||
:marked
|
||
## Test an attribute directive
|
||
|
||
## 测试属性指令
|
||
|
||
An _attribute directive_ modifies the behavior of an element, component or another directive.
|
||
Its name reflects the way the directive is applied: as an attribute on a host element.
|
||
|
||
**属性指令**修改元素、组件和其它指令的行为。正如它们的名字所示,它们是作为宿主元素的属性来被使用的。
|
||
|
||
The sample application's `HighlightDirective` sets the background color of an element
|
||
based on either a data bound color or a default color (lightgray).
|
||
It also sets a custom property of the element (`customProperty`) to `true`
|
||
for no reason other than to show that it can.
|
||
|
||
本例子应用的`HighlightDirective`使用数据绑定的颜色或者默认颜色来设置元素的背景色。
|
||
它同时设置元素的`customProperty`属性为`true`,这里仅仅是为了显示它能这么做而已,并无其它原因。
|
||
+makeExample('testing/ts/src/app/shared/highlight.directive.ts', '', 'src/app/shared/highlight.directive.ts')(format='.')
|
||
:marked
|
||
It's used throughout the application, perhaps most simply in the `AboutComponent`:
|
||
|
||
它的使用贯穿整个应用,也许最简单的使用在`AboutComponent`里:
|
||
+makeExample('testing/ts/src/app/about.component.ts', '', 'src/app/about.component.ts')(format='.')
|
||
:marked
|
||
Testing the specific use of the `HighlightDirective` within the `AboutComponent` requires only the
|
||
techniques explored above (in particular the ["Shallow test"](#shallow-component-test) approach).
|
||
|
||
使用`AboutComponent`来测试这个`HightlightDirective`的使用,只需要上面解释过的知识就够了,(尤其是["浅测试程序"](#shallow-component-test)方法)。
|
||
+makeExample('testing/ts/src/app/about.component.spec.ts', 'tests', 'src/app/about.component.spec.ts')(format='.')
|
||
:marked
|
||
However, testing a single use case is unlikely to explore the full range of a directive's capabilities.
|
||
Finding and testing all components that use the directive is tedious, brittle, and almost as unlikely to afford full coverage.
|
||
|
||
但是,测试单一的用例一般无法探索该指令的全部能力。
|
||
查找和测试所有使用该指令的组件非常繁琐和脆弱,并且通常无法覆盖所有组件。
|
||
|
||
[Isolated unit tests](#isolated-unit-tests) might be helpful,
|
||
but attribute directives like this one tend to manipulate the DOM.
|
||
Isolated unit tests don't touch the DOMand, therefore ,
|
||
do not inspire confidence in the directive's efficacy.
|
||
|
||
[孤立单元测试](#isolated-unit-tests)可能有用。
|
||
但是像这样的属性指令一般都操纵DOM。孤立单元测试不能控制DOM,所以不推荐用它测试指令的功能。
|
||
|
||
A better solution is to create an artificial test component that demonstrates all ways to apply the directive.
|
||
|
||
更好的方法是创建一个展示所有使用该组件的方法的人工测试组件。
|
||
|
||
+makeExample('testing/ts/src/app/shared/highlight.directive.spec.ts', 'test-component', 'src/app/shared/highlight.directive.spec.ts (TestComponent)')(format='.')
|
||
|
||
figure.image-display
|
||
img(src='/resources/images/devguide/testing/highlight-directive-spec.png' width="200px" alt="HighlightDirective spec in action")
|
||
.l-sub-section
|
||
:marked
|
||
The `<input>` case binds the `HighlightDirective` to the name of a color value in the input box.
|
||
The initial value is the word "cyan" which should be the background color of the input box.
|
||
|
||
`<input>`用例将`HighlightDirective`绑定到输入框里输入的颜色名字。
|
||
初始只是单词“cyan”,所以输入框的背景色应该是cyan。
|
||
:marked
|
||
Here are some tests of this component:
|
||
|
||
下面是一些该组件的测试程序:
|
||
+makeExample('testing/ts/src/app/shared/highlight.directive.spec.ts', 'selected-tests', 'src/app/shared/highlight.directive.spec.ts (selected tests)')
|
||
:marked
|
||
A few techniques are noteworthy:
|
||
|
||
一些技巧值得注意:
|
||
|
||
* The `By.directive` predicate is a great way to get the elements that have this directive _when their element types are unknown_.
|
||
|
||
当**已知元素类型**时,`By.directive`是一种获取拥有这个指令的元素的好方法。
|
||
|
||
* The <a href="https://developer.mozilla.org/en-US/docs/Web/CSS/:not" target="_blank">`:not` pseudo-class</a>
|
||
in `By.css('h2:not([highlight])')` helps find `<h2>` elements that _do not_ have the directive.
|
||
`By.css('*:not([highlight])')` finds _any_ element that does not have the directive.
|
||
|
||
`By.css('h2:not([highlight])')`里的<a href="https://developer.mozilla.org/en-US/docs/Web/CSS/:not" target="_blank">`:not`伪类(pseudo-class)</a>帮助查找**不带**该指令的`<h2>`元素。`By.css('*:not([highlight])')`查找**所有**不带该指令的元素。
|
||
|
||
* `DebugElement.styles` affords access to element styles even in the absence of a real browser, thanks to the `DebugElement` abstraction.
|
||
But feel free to exploit the `nativeElement` when that seems easier or more clear than the abstraction.
|
||
|
||
`DebugElement.styles`让我们不借助真实的浏览器也可以访问元素的样式,感谢`DebugElement`提供的这层抽象!
|
||
但是如果直接使用`nativeElement`会比这层抽象更简单、更清晰,也可以放心大胆的使用它。
|
||
|
||
* Angular adds a directive to the injector of the element to which it is applied.
|
||
The test for the default color uses the injector of the second `<h2>` to get its `HighlightDirective` instance
|
||
and its `defaultColor`.
|
||
|
||
Angular将指令添加到它的元素的注入器中。默认颜色的测试程序使用第二个`<h2>`的注入器来获取它的`HighlightDirective`实例以及它的`defaultColor`。
|
||
|
||
* `DebugElement.properties` affords access to the artificial custom property that is set by the directive.
|
||
|
||
`DebugElement.properties`让我们可以访问由指令设置的自定义属性。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#isolated-unit-tests
|
||
:marked
|
||
## Isolated Unit Tests
|
||
|
||
## 孤立的单元测试
|
||
|
||
Testing applications with the help of the Angular testing utilities is the main focus of this guide.
|
||
|
||
使用Angular测试工具测试应用程序是本章的重点。
|
||
|
||
However, it's often more productive to explore the inner logic of application classes
|
||
with _isolated_ unit tests that don't depend upon Angular.
|
||
Such tests are often smaller and easier to read, write, and maintain.
|
||
|
||
但是,使用**孤立**单元测试来探索应用类的内在逻辑往往更加有效率,它不依赖Angular。
|
||
这种测试程序通常比较小、更易阅读、编写和维护。
|
||
|
||
They don't carry extra baggage:
|
||
|
||
它们不用背负额外的包袱:
|
||
|
||
* Import from the Angular test libraries.
|
||
|
||
从Angular测试库导入
|
||
|
||
* Configure a module.
|
||
|
||
配置模块
|
||
|
||
* Prepare dependency injection `providers`.
|
||
|
||
准备依赖注入`providers`
|
||
|
||
* Call `inject` or `async` or `fakeAsync`.
|
||
|
||
调用`inject`,或者`async`,或者`fakeAsync`
|
||
|
||
They follow patterns familiar to test developers everywhere:
|
||
|
||
它们会遵循测试时众所周知的模式:
|
||
|
||
* Exhibit standard, Angular-agnostic testing techniques.
|
||
|
||
使用标准的、与Angular无关的测试技巧
|
||
|
||
* Create instances directly with `new`.
|
||
|
||
直接使用`new`创建实例
|
||
|
||
* Substitute test doubles (stubs, spys, and mocks) for the real dependencies.
|
||
|
||
用测试替身(stub,spy和mock)替代真正的依赖
|
||
|
||
.callout.is-important
|
||
header Write both kinds of tests
|
||
|
||
header 同时采用这两种测试程序
|
||
:marked
|
||
Good developers write both kinds of tests for the same application part, often in the same spec file.
|
||
Write simple _isolated_ unit tests to validate the part in isolation.
|
||
Write _Angular_ tests to validate the part as it interacts with Angular,
|
||
updates the DOM, and collaborates with the rest of the application.
|
||
|
||
优秀的开发者同时编写这两种测试程序来测试相同的应用部件,往往在同一个spec文件。
|
||
编写简单的**孤立**单元测试程序来验证孤立的部分。
|
||
编写**Angular**测试程序来验证与Angular互动、更新DOM、以及与应用其它部分互动的部分。
|
||
|
||
a#isolated-service-tests
|
||
:marked
|
||
### Services
|
||
|
||
### 服务
|
||
|
||
Services are good candidates for isolated unit testing.
|
||
Here are some synchronous and asynchronous unit tests of the `FancyService`
|
||
written without assistance from Angular testing utilities.
|
||
|
||
服务是应用孤立测试的好例子。
|
||
下面是未使用Angular测试工具的一些`FancyService`的同步和异步单元测试:
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.no-testbed.spec.ts', 'FancyService', 'src/app/bag/bag.no-testbed.spec.ts')
|
||
:marked
|
||
A rough line count suggests that these isolated unit tests are about 25% smaller than equivalent Angular tests.
|
||
That's telling but not decisive.
|
||
The benefit comes from reduced setup and code complexity.
|
||
|
||
粗略行数表明,这些孤立单元测试比同等的Angular测试小25%。
|
||
这表明了它的好处,但是不是最关键的。
|
||
主要的好处来自于缩减的配置和代码的复杂性。
|
||
|
||
Compare these equivalent tests of `FancyService.getTimeoutValue`.
|
||
|
||
比较下面两个同等的`FancyService.getTimeoutValue`测试程序:
|
||
|
||
+makeTabs(
|
||
`testing/ts/src/app/bag/bag.no-testbed.spec.ts, testing/ts/src/app/bag/bag.spec.ts`,
|
||
'getTimeoutValue, getTimeoutValue',
|
||
`src/app/bag/bag.no-testbed.spec.ts (Isolated), src/app/bag/bag.spec.ts (with Angular testing utilities)`)
|
||
:marked
|
||
They have about the same line-count, but the Angular-dependent version
|
||
has more moving parts including a couple of utility functions (`async` and `inject`).
|
||
Both approaches work and it's not much of an issue if you're using the
|
||
Angular testing utilities nearby for other reasons.
|
||
On the other hand, why burden simple service tests with added complexity?
|
||
|
||
它们有类似的行数。
|
||
但是,依赖Angular的版本有更多活动的部分,包括一些工具函数(`async`和`inject`)。
|
||
两种方法都可行,而且如果你为了某些原因使用Angular测试工具,也并没有什么问题。
|
||
反过来,为什么要为简单的服务测试程序添加复杂度呢?
|
||
|
||
Pick the approach that suits you.
|
||
|
||
选择你喜欢的方法。
|
||
|
||
a#services-with-dependencies
|
||
:marked
|
||
### Services with dependencies
|
||
|
||
### 带依赖的服务
|
||
|
||
Services often depend on other services that Angular injects into the constructor.
|
||
You can test these services _without_ the `TestBed`.
|
||
In many cases, it's easier to create and _inject_ dependencies by hand.
|
||
|
||
服务通常依赖其它服务,Angular通过构造函数注入它们。
|
||
你可以**不使用**TestBed测试这些服务。
|
||
在许多情况下,创建和手动**注入**依赖来的更加容易。
|
||
|
||
The `DependentService` is a simple example:
|
||
|
||
`DependentService`是一个简单的例子:
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.ts', 'DependentService', 'src/app/bag/bag.ts')(format='.')
|
||
|
||
:marked
|
||
It delegates its only method, `getValue`, to the injected `FancyService`.
|
||
|
||
它将唯一的方法,`getValue`,委托给了注入的`FancyService`。
|
||
|
||
Here are several ways to test it.
|
||
|
||
这里是几种测试它的方法。
|
||
+makeExample('testing/ts/src/app/bag/bag.no-testbed.spec.ts', 'DependentService', 'src/app/bag/bag.no-testbed.spec.ts')
|
||
:marked
|
||
The first test creates a `FancyService` with `new` and passes it to the `DependentService` constructor.
|
||
|
||
第一个测试程序使用`new`创建`FancyService`实例,并将它传递给`DependentService`构造函数。
|
||
|
||
However, it's rarely that simple. The injected service can be difficult to create or control.
|
||
You can mock the dependency, use a dummy value, or stub the pertinent service method
|
||
with a substitute method that 's easy to control.
|
||
|
||
很少有这么简单的,注入的服务有可能很难创建和控制。
|
||
你可以mock依赖,或者使用假值,或者用易于控制的替代品stub伪造相关服务。
|
||
|
||
These _isolated_ unit testing techniques are great for exploring the inner logic of a service or its
|
||
simple integration with a component class.
|
||
Use the Angular testing utilities when writing tests that validate how a service interacts with components
|
||
_within the Angular runtime environment_.
|
||
|
||
这些**孤立**单元测试技巧是一个很好的方法,用来探索服务的内在逻辑,以及它与组件类简单的集成。
|
||
当在**运行时间环境下**,使用Angular测试工具来验证一个服务是如何与组件互动的。
|
||
|
||
a#isolated-pipe-tests
|
||
:marked
|
||
### Pipes
|
||
|
||
### 管道
|
||
Pipes are easy to test without the Angular testing utilities.
|
||
|
||
管道很容易测试,无需Angular测试工具。
|
||
|
||
A pipe class has one method, `transform`, that manipulates the input
|
||
value into a transformed output value.
|
||
The `transform` implementation rarely interacts with the DOM.
|
||
Most pipes have no dependence on Angular other than the `@Pipe`
|
||
metadata and an interface.
|
||
|
||
管道类有一个方法,`transform`,用来转换输入值到输出值。
|
||
`transform`的实现很少与DOM交互。
|
||
除了`@Pipe`元数据和一个接口外,大部分管道不依赖Angular。
|
||
|
||
Consider a `TitleCasePipe` that capitalizes the first letter of each word.
|
||
Here's a naive implementation with a regular expression.
|
||
|
||
假设`TitleCasePipe`将每个单词的第一个字母变成大写。
|
||
下面是使用正则表达式实现的简单代码:
|
||
|
||
+makeExample('testing/ts/src/app/shared/title-case.pipe.ts', '', 'src/app/shared/title-case.pipe.ts')(format='.')
|
||
|
||
:marked
|
||
Anything that uses a regular expression is worth testing thoroughly.
|
||
Use simple Jasmine to explore the expected cases and the edge cases.
|
||
|
||
任何使用正则表达式的类都值得彻底的进行测试。
|
||
使用Jasmine来探索预期的用例和极端的用例。
|
||
|
||
+makeExample('testing/ts/src/app/shared/title-case.pipe.spec.ts', 'excerpt', 'src/app/shared/title-case.pipe.spec.ts')
|
||
|
||
a#write-tests
|
||
:marked
|
||
### Write Angular tests too
|
||
|
||
### 同时也编写Angular测试
|
||
|
||
These are tests of the pipe _in isolation_.
|
||
They can't tell if the `TitleCasePipe` is working properly as applied in the application components.
|
||
|
||
有些管道的测试程序是**孤立的**。
|
||
它们不能验证`TitleCasePipe`是否在应用到组件上时是否工作正常。
|
||
|
||
Consider adding component tests such as this one:
|
||
|
||
考虑像这样添加组件测试程序:
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-detail.component.spec.ts', 'title-case-pipe', 'src/app/hero/hero-detail.component.spec.ts (pipe test)')
|
||
|
||
a#isolated-component-tests
|
||
:marked
|
||
### Components
|
||
|
||
### 组件
|
||
|
||
Component tests typically examine how a component class interacts with its own template or with collaborating components.
|
||
The Angular testing utilities are specifically designed to facilitate such tests.
|
||
|
||
组件测试通常检查该组件类是如何与自己的模板或者其它合作组件交互的。
|
||
Angular测试工具是专门为这种测试设计的。
|
||
|
||
Consider this `ButtonComp` component.
|
||
|
||
考虑这个`ButtonComp`组件。
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.ts', 'ButtonComp', 'src/app/bag/bag.ts (ButtonComp)')(format='.')
|
||
|
||
:marked
|
||
The following Angular test demonstrates that clicking a button in the template leads
|
||
to an update of the on-screen message.
|
||
|
||
下面的Angular测试演示点击模板里的按钮后,引起了屏幕上的消息的更新。
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.spec.ts', 'ButtonComp', 'src/app/bag/bag.spec.ts (ButtonComp)')(format='.')
|
||
|
||
:marked
|
||
The assertions verify that the data values flow from one HTML control (the `<button>`) to the component and
|
||
from the component back to a _different_ HTML control (the `<span>`).
|
||
A passing test means the component and its template are wired correctly.
|
||
|
||
该判断验证了数据绑定从一个HTML控件(`<button>`)流动到组件,以及从组件回到**不同**的HTML控件(`<span>`)。
|
||
通过的测试程序说明组件和它的模块是否设置正确。
|
||
|
||
Isolated unit tests can more rapidly probe a component at its API boundary,
|
||
exploring many more conditions with less effort.
|
||
|
||
孤立单元测试可以更快的在API边界探测组件,更轻松的探索更多条件。
|
||
|
||
Here are a set of unit tests that verify the component's outputs in the face of a variety of
|
||
component inputs.
|
||
|
||
下面是一套单元测试程序,用来验证面对多种输入时组件的输出。
|
||
+makeExample('testing/ts/src/app/bag/bag.no-testbed.spec.ts', 'ButtonComp', 'src/app/bag/bag.no-testbed.spec.ts (ButtonComp)')(format='.')
|
||
:marked
|
||
Isolated component tests offer a lot of test coverage with less code and almost no setup.
|
||
This is even more of an advantage with complex components, which
|
||
may require meticulous preparation with the Angular testing utilities.
|
||
|
||
孤立组件单元测试使用更少的代码以及几乎不存在的配置,提供了很多测试覆盖率。
|
||
在测试复杂的组件时,这个优势显得更加明显,因为可能需要使用Angular测试工具进行精心准备。
|
||
|
||
On the other hand, isolated unit tests can't confirm that the `ButtonComp` is
|
||
properly bound to its template or even data bound at all.
|
||
Use Angular tests for that.
|
||
|
||
但是,孤立测试无法确认`ButtonComp`是否与其模板正确的绑定,或者是否有数据绑定。
|
||
使用Angular测试来应对它们。
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l-hr
|
||
|
||
a#atu-apis
|
||
:marked
|
||
## Angular testing utility APIs
|
||
|
||
## Angular测试工具API
|
||
|
||
This section takes inventory of the most useful Angular testing features and summarizes what they do.
|
||
|
||
本节将最有用的Angular测试功能提取出来,并总结了它们的作用。
|
||
|
||
The Angular testing utilities include the `TestBed`, the `ComponentFixture`, and a handful of functions that control the test environment.
|
||
The [_TestBed_](#testbed-api-summary) and [_ComponentFixture_](#component-fixture-api-summary) classes are covered separately.
|
||
|
||
Angular测试工具包括`TestBed`、`ComponentFixture`和一些其他函数,用来控制测试环境。
|
||
[_TestBed_](#testbed-api-summary)和[_ComponentFixture_](#component-fixture-api-summary)在这里分别解释了。
|
||
|
||
Here's a summary of the stand-alone functions, in order of likely utility:
|
||
|
||
下面是一些独立函数的总结,以使用频率排序:
|
||
|
||
table
|
||
tr
|
||
th
|
||
p Function
|
||
p 函数
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>async</code>
|
||
td
|
||
:marked
|
||
Runs the body of a test (`it`) or setup (`beforeEach`) function within a special _async test zone_.
|
||
See [discussion above](#async).
|
||
|
||
在特殊的**async测试区域**运行测试程序(`it`)或者设置(`beforeEach`)的主体。
|
||
参见[上面的讨论](#async).
|
||
tr
|
||
td(style="vertical-align: top") <code>fakeAsync</code>
|
||
td
|
||
:marked
|
||
Runs the body of a test (`it`) within a special _fakeAsync test zone_, enabling
|
||
a linear control flow coding style. See [discussion above](#fake-async).
|
||
|
||
在特殊的**fakeAsync测试区域**运行测试程序(`it`)的主体,造就控制流更加线性的代码风格。
|
||
参见[上面的讨论](#fake-async).
|
||
tr
|
||
td(style="vertical-align: top") <code>tick</code>
|
||
td
|
||
:marked
|
||
Simulates the passage of time and the completion of pending asynchronous activities
|
||
by flushing both _timer_ and _micro-task_ queues within the _fakeAsync test zone_.
|
||
|
||
在**fakeAsync测试区域**内触发**计时器**和**微任务**队列,以模拟时间的推移和未完成异步任务的完成。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The curious, dedicated reader might enjoy this lengthy blog post,
|
||
"<a href="https://jakearchibald.com/2015/tasks-microtasks-queues-and-schedules/"
|
||
target="_blank">_Tasks, microtasks, queues and schedules_</a>".
|
||
|
||
好奇和执着的读者可能会喜欢这篇长博客:
|
||
"<a href="https://jakearchibald.com/2015/tasks-microtasks-queues-and-schedules/"
|
||
target="_blank">_Tasks, microtasks, queues and schedules_</a>".
|
||
:marked
|
||
Accepts an optional argument that moves the virtual clock forward
|
||
by the specified number of milliseconds,
|
||
clearing asynchronous activities scheduled within that timeframe.
|
||
See [discussion above](#tick).
|
||
|
||
接受一个可选参数,往前推移虚拟时间提供数字的毫秒数,清除在这段时间内的异步行为。
|
||
参见[上面的讨论](#tick)
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>inject</code>
|
||
td
|
||
:marked
|
||
Injects one or more services from the current `TestBed` injector into a test function.
|
||
See [above](#inject).
|
||
|
||
从当前`TestBed`注入器注入一个或多个服务到测试函数。参见[上面](#inject)。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>discardPeriodicTasks</code>
|
||
td
|
||
:marked
|
||
When a `fakeAsync` test ends with pending timer event _tasks_ (queued `setTimeOut` and `setInterval` callbacks),
|
||
the test fails with a clear error message.
|
||
|
||
当`fakeAsync`测试程序以正在运行的计时器事件**任务**(排队中的`setTimeOut`和`setInterval`的回调)结束时,
|
||
测试会失败,并显示一条明确的错误信息。
|
||
|
||
In general, a test should end with no queued tasks.
|
||
When pending timer tasks are expected, call `discardPeriodicTasks` to flush the _task_ queue
|
||
and avoid the error.
|
||
|
||
一般来讲,测试程序应该以无排队任务结束。
|
||
当待执行计时器任务存在时,调用`discardPeriodicTasks`来触发**任务**队列,防止该错误发生。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>flushMicrotasks</code>
|
||
td
|
||
:marked
|
||
When a `fakeAsync` test ends with pending _micro-tasks_ such as unresolved promises,
|
||
the test fails with a clear error message.
|
||
|
||
当`fakeAsync`测试程序以待执行**微任务**(比如未解析的承诺)结束时,测试会失败并显示明确的错误信息。
|
||
|
||
In general, a test should wait for micro-tasks to finish.
|
||
When pending microtasks are expected, call `flushMicrotasks` to flush the _micro-task_ queue
|
||
and avoid the error.
|
||
|
||
一般来说,测试应该等待微任务结束。
|
||
当待执行微任务存在时,调用`flushMicrotasks`来触发**微任务**队列,防止该错误发生。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>ComponentFixtureAutoDetect</code>
|
||
td
|
||
:marked
|
||
A provider token for a service that turns on [automatic change detection](#automatic-change-detection).
|
||
|
||
一个提供商令牌,用来设置**auto-changeDetect**的值,它默认值为`false`。
|
||
参见[自动变更检测](#automatic-change-detection)
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>getTestBed</code>
|
||
td
|
||
:marked
|
||
Gets the current instance of the `TestBed`.
|
||
Usually unnecessary because the static class methods of the `TestBed` class are typically sufficient.
|
||
The `TestBed` instance exposes a few rarely used members that are not available as
|
||
static methods.
|
||
|
||
获取当前`TestBed`实例。
|
||
通常用不上,因为`TestBed`的静态类方法已经够用。
|
||
`TestBed`实例有一些很少需要用到的方法,它们没有对应的静态方法。
|
||
|
||
.l-hr
|
||
|
||
a#testbed-class-summary
|
||
:marked
|
||
### _TestBed_ class summary
|
||
|
||
### _TestBed_ 类总结
|
||
|
||
The `TestBed` class is one of the principal Angular testing utilities.
|
||
Its API is quite large and can be overwhelming until you've explored it,
|
||
a little at a time. Read the early part of this guide first
|
||
to get the basics before trying to absorb the full API.
|
||
|
||
`TestBed`类是Angular测试工具的主要类之一。它的API很庞大,可能有点过于复杂,直到你一点一点的探索它们。
|
||
阅读本章前面的部分,了解了基本的知识以后,再试着了解完整API。
|
||
|
||
The module definition passed to `configureTestingModule`
|
||
is a subset of the `@NgModule` metadata properties.
|
||
|
||
传递给`configureTestingModule`的模块定义是`@NgModule`元数据属性的子集。
|
||
|
||
code-example(format="." language="javascript").
|
||
type TestModuleMetadata = {
|
||
providers?: any[];
|
||
declarations?: any[];
|
||
imports?: any[];
|
||
schemas?: Array<SchemaMetadata | any[]>;
|
||
};
|
||
|
||
a#metadata-override-object
|
||
:marked
|
||
Each override method takes a `MetadataOverride<T>` where `T` is the kind of metadata
|
||
appropriate to the method, that is, the parameter of an `@NgModule`,
|
||
`@Component`, `@Directive`, or `@Pipe`.
|
||
|
||
每一个重载方法接受一个`MetadataOverride<T>`,这里`T`是适合这个方法的元数据类型,也就是`@NgModule`、`@Component`、`@Directive`或者`@Pipe`的参数。
|
||
|
||
code-example(format="." language="javascript").
|
||
type MetadataOverride<T> = {
|
||
add?: T;
|
||
remove?: T;
|
||
set?: T;
|
||
};
|
||
|
||
:marked
|
||
a#testbed-methods
|
||
:marked
|
||
The `TestBed` API consists of static class methods that either update or reference a _global_ instance of the`TestBed`.
|
||
|
||
`TestBed`的API包含了一系列静态类方法,它们更新或者引用**全局**的`TestBed`实例。
|
||
|
||
Internally, all static methods cover methods of the current runtime `TestBed` instance ,
|
||
which is also returned by the `getTestBed()` function.
|
||
|
||
在内部,所有静态方法在`getTestBed()`函数返回的当前运行时间的`TestBed`实例上都有对应的方法。
|
||
|
||
Call `TestBed` methods _within_ a `beforeEach()` to ensure a fresh start before each individual test.
|
||
|
||
在`BeforeEach()`内调用`TestBed`方法,这样确保在运行每个单独测试时,都有崭新的开始。
|
||
|
||
Here are the most important static methods, in order of likely utility.
|
||
|
||
这里列出了最重要的静态方法,以使用频率排序:
|
||
|
||
table
|
||
tr
|
||
th
|
||
p Methods
|
||
p 方法
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>configureTestingModule</code>
|
||
td
|
||
:marked
|
||
The testing shims (`karma-test-shim`, `browser-test-shim`)
|
||
establish the [initial test environment](##testbed-initTestEnvironment) and a default testing module.
|
||
The default testing module is configured with basic declaratives and some Angular service substitutes that every tester needs.
|
||
|
||
测试垫片(`karma-test-shim`, `browser-test-shim`)创建了[初始测试环境](##testbed-initTestEnvironment)和默认测试模块。
|
||
默认测试模块是使用基本声明和一些Angular服务替代品,它们是所有测试程序都需要的。
|
||
|
||
Call `configureTestingModule` to refine the testing module configuration for a particular set of tests
|
||
by adding and removing imports, declarations (of components, directives, and pipes), and providers.
|
||
|
||
调用`configureTestingModule`来为一套特定的测试定义测试模块配置,添加和删除导入、(组件、指令和管道的)声明和服务提供商。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>compileComponents</code>
|
||
td
|
||
:marked
|
||
Compile the testing module asynchronously after you've finished configuring it.
|
||
You **must** call this method if _any_ of the testing module components have a `templateUrl`
|
||
or `styleUrls` because fetching component template and style files is necessarily asynchronous.
|
||
See [above](#compile-components).
|
||
|
||
在你完成配置以后异步编译测试模块。
|
||
如果**任何**测试组件有`templateUrl`或`styleUrls`,那么你**必须**调用这个方法。因为获取组件模块和样式文件必须是异步的。
|
||
参见[上面的描述](#compile-components)。
|
||
|
||
After calling `compileComponents`, the `TestBed` configuration is frozen for the duration of the current spec.
|
||
|
||
调用完`compileComponents`之后,`TestBed`的配置就会在当前测试期间被冻结。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>createComponent<T></code>
|
||
td
|
||
:marked
|
||
Create an instance of a component of type `T` based on the current `TestBed` configuration.
|
||
After calling `compileComponent`, the `TestBed` configuration is frozen for the duration of the current spec.
|
||
|
||
基于当前`TestBed`的配置创建一个类型为T的组件实例。
|
||
一旦调用,`TestBed`的配置就会在当前测试期间被冻结。
|
||
tr
|
||
td(style="vertical-align: top") <code>overrideModule</code>
|
||
td
|
||
:marked
|
||
Replace metadata for the given `NgModule`. Recall that modules can import other modules.
|
||
The `overrideModule` method can reach deeply into the current testing module to
|
||
modify one of these inner modules.
|
||
|
||
替换指定的`NgModule`的元数据。回想一下,模块可以导入其他模块。
|
||
`overrideModule`方法可以深入到当前测试模块深处,修改其中一个内部模块。
|
||
tr
|
||
td(style="vertical-align: top") <code>overrideComponent</code>
|
||
td
|
||
:marked
|
||
Replace metadata for the given component class, which could be nested deeply
|
||
within an inner module.
|
||
|
||
替换指定组件类的元数据,该组件类可能嵌套在一个很深的内部模块中。
|
||
tr
|
||
td(style="vertical-align: top") <code>overrideDirective</code>
|
||
td
|
||
:marked
|
||
Replace metadata for the given directive class, which could be nested deeply
|
||
within an inner module.
|
||
|
||
替换指定指令类的元数据,该指令可能嵌套在一个很深的内部模块中。
|
||
tr
|
||
td(style="vertical-align: top") <code>overridePipe</code>
|
||
td
|
||
:marked
|
||
Replace metadata for the given pipe class, which could be nested deeply
|
||
within an inner module.
|
||
|
||
替换指定管道类的元数据,该管道可能嵌套在一个很深的内部模块中。
|
||
tr
|
||
td(style="vertical-align: top").
|
||
<a id="testbed-get"></a>
|
||
<code>get</code>
|
||
td
|
||
:marked
|
||
Retrieve a service from the current `TestBed` injector.
|
||
|
||
从当前`TestBed`注入器获取一个服务。
|
||
|
||
The `inject` function is often adequate for this purpose.
|
||
But `inject` throws an error if it can't provide the service.
|
||
What if the service is optional?
|
||
|
||
`inject`函数通常很适合这个任务。
|
||
但是如果`inject`不能提供服务,它会抛出错误。
|
||
如果服务是可选的呢?
|
||
|
||
The `TestBed.get` method takes an optional second parameter,
|
||
the object to return if Angular can't find the provider
|
||
(`null` in this example):
|
||
|
||
`TestBed.get`方法接受一个可选的第二参数,它是在Angular找不到所需提供商时返回的对象。(在本例中为`null`):
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.spec.ts', 'testbed-get')(format=".")
|
||
:marked
|
||
After calling `get`, the `TestBed` configuration is frozen for the duration of the current spec.
|
||
|
||
一旦调用,`TestBed`的配置就会在当前测试期间被冻结。
|
||
|
||
tr
|
||
td(style="vertical-align: top").
|
||
<a id="testbed-initTestEnvironment"></a>
|
||
<code>initTestEnvironment</code>
|
||
td
|
||
:marked
|
||
Initialize the testing environment for the entire test run.
|
||
|
||
为整套测试的运行初始化测试环境。
|
||
|
||
The testing shims (`karma-test-shim`, `browser-test-shim`) call it for you
|
||
so there is rarely a reason for you to call it yourself.
|
||
|
||
测试垫片(`karma-test-shim`, `browser-test-shim`)会为你调用它,所以你很少需要自己调用它。
|
||
|
||
You may call this method _exactly once_. If you must change
|
||
this default in the middle of your test run, call `resetTestEnvironment` first.
|
||
|
||
这个方法只能被调用**一次**。如果确实需要在测试程序运行期间变换这个默认设置,那么先调用`resetTestEnvironment`。
|
||
|
||
Specify the Angular compiler factory, a `PlatformRef`, and a default Angular testing module.
|
||
Alternatives for non-browser platforms are available in the general form
|
||
`@angular/platform-<platform_name>/testing/<platform_name>`.
|
||
|
||
指定Angular编译器工厂,`PlatformRef`,和默认Angular测试模块。
|
||
以`@angular/platform-<platform_name>/testing/<platform_name>`的形式提供非浏览器平台的替代品。
|
||
tr
|
||
td(style="vertical-align: top") <code>resetTestEnvironment</code>
|
||
td
|
||
:marked
|
||
Reset the initial test environment, including the default testing module.
|
||
|
||
重设初始测试环境,包括默认测试模块在内。
|
||
|
||
:marked
|
||
A few of the `TestBed` instance methods are not covered by static `TestBed` _class_ methods.
|
||
These are rarely needed.
|
||
|
||
少数`TestBed`实例方法没有对应的静态方法。它们很少被使用。
|
||
|
||
a#component-fixture-api-summary
|
||
:marked
|
||
### The _ComponentFixture_
|
||
|
||
### _ComponentFixture_对象
|
||
|
||
The `TestBed.createComponent<T>`
|
||
creates an instance of the component `T`
|
||
and returns a strongly typed `ComponentFixture` for that component.
|
||
|
||
`TestBed.createComponent<T>`创建一个组件`T`的实例,并为该组件返回一个强类型的`ComponentFixture`。
|
||
|
||
The `ComponentFixture` properties and methods provide access to the component,
|
||
its DOM representation, and aspects of its Angular environment.
|
||
|
||
`ComponentFixture`的属性和方法提供了对组件、它的DOM和它的Angular环境方面的访问。
|
||
|
||
a#component-fixture-properties
|
||
:marked
|
||
### _ComponentFixture_ properties
|
||
|
||
### _ComponentFixture_的属性
|
||
|
||
Here are the most important properties for testers, in order of likely utility.
|
||
|
||
下面是对测试最重要的属性,以使用频率排序:
|
||
|
||
table
|
||
tr
|
||
th
|
||
p Properties
|
||
p 属性
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>componentInstance</code>
|
||
td
|
||
:marked
|
||
The instance of the component class created by `TestBed.createComponent`.
|
||
|
||
被`TestBed.createComponent`创建的组件类实例。
|
||
tr
|
||
td(style="vertical-align: top") <code>debugElement</code>
|
||
td
|
||
:marked
|
||
The `DebugElement` associated with the root element of the component.
|
||
|
||
与组件根元素关联的`DebugElement`。
|
||
|
||
The `debugElement` provides insight into the component and its DOM element during test and debugging.
|
||
It's a critical property for testers. The most interesting members are covered [below](#debug-element-details).
|
||
|
||
`debugElement`在测试和调试期间,提供对组件及其DOM元素的访问。
|
||
它是测试者至关重要的属性。它最有用的成员在[下面](#debug-element-details)有所介绍。
|
||
tr
|
||
td(style="vertical-align: top") <code>nativeElement</code>
|
||
td
|
||
:marked
|
||
The native DOM element at the root of the component.
|
||
|
||
组件的原生根DOM元素。
|
||
tr
|
||
td(style="vertical-align: top") <code>changeDetectorRef</code>
|
||
td
|
||
:marked
|
||
The `ChangeDetectorRef` for the component.
|
||
|
||
组件的`ChangeDetectorRef`。
|
||
|
||
The `ChangeDetectorRef` is most valuable when testing a
|
||
component that has the `ChangeDetectionStrategy.OnPush` method
|
||
or the component's change detection is under your programmatic control.
|
||
|
||
在测试一个拥有`ChangeDetectionStrategy.OnPush`的组件,或者在组件的变化测试在你的程序控制下时,`ChangeDetectorRef`是最重要的。
|
||
|
||
a#component-fixture-methods
|
||
:marked
|
||
### _ComponentFixture_ methods
|
||
|
||
### _ComponentFixture_的方法
|
||
|
||
The _fixture_ methods cause Angular to perform certain tasks on the component tree.
|
||
Call these method to trigger Angular behavior in response to simulated user action.
|
||
|
||
**fixture**方法使Angular对组件树执行某些任务。
|
||
在触发Angular行为来模拟的用户行为时,调用这些方法。
|
||
|
||
Here are the most useful methods for testers.
|
||
|
||
下面是对测试最有用的方法。
|
||
|
||
table
|
||
tr
|
||
th
|
||
p Methods
|
||
p 方法
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>detectChanges</code>
|
||
td
|
||
:marked
|
||
Trigger a change detection cycle for the component.
|
||
|
||
为组件触发一轮变化检查。
|
||
|
||
Call it to initialize the component (it calls `ngOnInit`) and after your
|
||
test code, change the component's data bound property values.
|
||
Angular can't see that you've changed `personComponent.name` and won't update the `name`
|
||
binding until you call `detectChanges`.
|
||
|
||
调用它来初始化组件(它调用`ngOnInit`)。或者在你的测试代码改变了组件的数据绑定属性值后调用它。
|
||
Angular不能检测到你已经改变了`personComponent.name`属性,也不会更新`name`的绑定,直到你调用了`detectChanges`。
|
||
|
||
Runs `checkNoChanges`afterwards to confirm that there are no circular updates unless
|
||
called as `detectChanges(false)`;
|
||
|
||
之后,运行`checkNoChanges`,来确认没有循环更新,除非它被这样调用:`detectChanges(false)`。
|
||
tr
|
||
td(style="vertical-align: top") <code>autoDetectChanges</code>
|
||
td
|
||
:marked
|
||
Set this to `true` when you want the fixture to detect changes automatically.
|
||
|
||
设置fixture是否应该自动试图检测变化。
|
||
|
||
When autodetect is `true`, the test fixture calls `detectChanges` immediately
|
||
after creating the component. Then it listens for pertinent zone events
|
||
and calls `detectChanges` accordingly.
|
||
When your test code modifies component property values directly,
|
||
you probably still have to call `fixture.detectChanges` to trigger data binding updates.
|
||
|
||
当自动检测打开时,测试fixture监听**zone**事件,并调用`detectChanges`。
|
||
当你的测试代码直接修改了组件属性值时,你还是要调用`fixture.detectChanges`来触发数据绑定更新。
|
||
|
||
The default is `false`. Testers who prefer fine control over test behavior
|
||
tend to keep it `false`.
|
||
|
||
默认值是`false`,喜欢对测试行为进行精细控制的测试者一般保持它为`false`。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>checkNoChanges</code>
|
||
td
|
||
:marked
|
||
Do a change detection run to make sure there are no pending changes.
|
||
Throws an exceptions if there are.
|
||
|
||
运行一次变更检测来确认没有待处理的变化。如果有未处理的变化,它将抛出一个错误。
|
||
tr
|
||
td(style="vertical-align: top") <code>isStable</code>
|
||
td
|
||
:marked
|
||
If the fixture is currently _stable_, returns `true`.
|
||
If there are async tasks that have not completed, returns `false`.
|
||
|
||
如果fixture当前是**稳定的**,则返回`true`。
|
||
如果有异步任务没有完成,则返回`false`。
|
||
tr
|
||
td(style="vertical-align: top") <code>whenStable</code>
|
||
td
|
||
:marked
|
||
Returns a promise that resolves when the fixture is stable.
|
||
|
||
返回一个承诺,在fixture稳定时解析。
|
||
|
||
To resume testing after completion of asynchronous activity or
|
||
asynchronous change detection, hook that promise.
|
||
See [above](#when-stable).
|
||
|
||
钩住这个承诺,以在异步行为或者异步变更检测之后继续测试。参见[上面](#when-stable)。
|
||
tr
|
||
td(style="vertical-align: top") <code>destroy</code>
|
||
td
|
||
:marked
|
||
Trigger component destruction.
|
||
|
||
触发组件的销毁。
|
||
|
||
a#debug-element-details
|
||
:marked
|
||
### _DebugElement_
|
||
|
||
The `DebugElement` provides crucial insights into the component's DOM representation.
|
||
|
||
`DebugElement`提供了对组件的DOM的访问。
|
||
|
||
From the test root component's `DebugElement` returned by `fixture.debugElement`,
|
||
you can walk (and query) the fixture's entire element and component subtrees.
|
||
|
||
`fixture.debugElement`返回测试根组件的`DebugElement`,通过它你可以访问(查询)fixture的整个元素和组件子树。
|
||
|
||
Here are the most useful `DebugElement` members for testers, in approximate order of utility:
|
||
|
||
下面是`DebugElement`最有用的成员,以使用频率排序。
|
||
|
||
table
|
||
tr
|
||
th
|
||
p Member
|
||
p 成员
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>nativeElement</code>
|
||
td
|
||
:marked
|
||
The corresponding DOM element in the browser (null for WebWorkers).
|
||
|
||
与浏览器中DOM元素对应(WebWorkers时,值为null)。
|
||
tr
|
||
td(style="vertical-align: top") <code>query</code>
|
||
td
|
||
:marked
|
||
Calling `query(predicate: Predicate<DebugElement>)` returns the first `DebugElement`
|
||
that matches the [predicate](#query-predicate) at any depth in the subtree.
|
||
|
||
调用`query(predicate: Predicate<DebugElement>)`返回子树所有层中第一个匹配[predicate](#query-predicate)的`DebugElement`。
|
||
tr
|
||
td(style="vertical-align: top") <code>queryAll</code>
|
||
td
|
||
:marked
|
||
Calling `queryAll(predicate: Predicate<DebugElement>)` returns all `DebugElements`
|
||
that matches the [predicate](#query-predicate) at any depth in subtree.
|
||
|
||
调用`query(predicate: Predicate<DebugElement>)`返回子树所有层中所有匹配[predicate](#query-predicate)`DebugElement`。
|
||
tr
|
||
td(style="vertical-align: top") <code>injector</code>
|
||
td
|
||
:marked
|
||
The host dependency injector.
|
||
For example, the root element's component instance injector.
|
||
|
||
宿主依赖注入器。
|
||
比如,根元素的组件实例注入器。
|
||
tr
|
||
td(style="vertical-align: top") <code>componentInstance</code>
|
||
td
|
||
:marked
|
||
The element's own component instance, if it has one.
|
||
|
||
元素自己的组件实例(如果有)。
|
||
tr
|
||
td(style="vertical-align: top") <code>context</code>
|
||
td
|
||
:marked
|
||
An object that provides parent context for this element.
|
||
Often an ancestor component instance that governs this element.
|
||
|
||
为元素提供父级上下文的对象。
|
||
通常是控制该元素的祖级组件实例。
|
||
|
||
When an element is repeated within `*ngFor`, the context is an `NgForRow` whose `$implicit`
|
||
property is the value of the row instance value.
|
||
For example, the `hero` in `*ngFor="let hero of heroes"`.
|
||
|
||
当一个元素被`*ngFor`重复,它的上下文为`NgForRow`,它的`$implicit`属性值是该行的实例值。
|
||
比如,`*ngFor="let hero of heroes"`里的`hero`。
|
||
tr
|
||
td(style="vertical-align: top") <code>children</code>
|
||
td
|
||
:marked
|
||
The immediate `DebugElement` children. Walk the tree by descending through `children`.
|
||
|
||
`DebugElement`的直接子级。通过`children`来降序探索元素树。
|
||
|
||
.l-sub-section
|
||
:marked
|
||
`DebugElement` also has `childNodes`, a list of `DebugNode` objects.
|
||
`DebugElement` derives from `DebugNode` objects and there are often
|
||
more nodes than elements. Testers can usually ignore plain nodes.
|
||
|
||
`DebugElement`还有`childNodes`,即`DebugNode`对象列表。
|
||
`DebugElement`从`DebugNode`对象衍生,而且通常节点(node)比元素多。测试者通常忽略赤裸节点。
|
||
tr
|
||
td(style="vertical-align: top") <code>parent</code>
|
||
td
|
||
:marked
|
||
The `DebugElement` parent. Null if this is the root element.
|
||
|
||
`DebugElement`的父级。如果`DebugElement`是根元素,`parent`为null。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>name</code>
|
||
td
|
||
:marked
|
||
The element tag name, if it is an element.
|
||
|
||
元素的标签名字,如果它是一个元素的话。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>triggerEventHandler</code>
|
||
td
|
||
:marked
|
||
Triggers the event by its name if there is a corresponding listener
|
||
in the element's `listeners` collection.
|
||
The second parameter is the _event object_ expected by the handler.
|
||
See [above](#trigger-event-handler).
|
||
|
||
如果在元素的`listeners`列表中有一个对应的`listener`,则以事件名字触发。
|
||
第二个参数是**事件对象**,一般为事件处理器。
|
||
参见[上面](#trigger-event-handler)。
|
||
|
||
If the event lacks a listener or there's some other problem,
|
||
consider calling `nativeElement.dispatchEvent(eventObject)`.
|
||
|
||
如果事件缺乏监听器,或者有其它问题,考虑调用`nativeElement.dispatchEvent(eventObject)`。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>listeners</code>
|
||
td
|
||
:marked
|
||
The callbacks attached to the component's `@Output` properties and/or the element's event properties.
|
||
|
||
元素的`@Output`属性以及/或者元素的事件属性所附带的回调函数。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>providerTokens</code>
|
||
td
|
||
:marked
|
||
This component's injector lookup tokens.
|
||
Includes the component itself plus the tokens that the component lists in its `providers` metadata.
|
||
|
||
组件注入器的查询令牌。
|
||
包括组件自己的令牌和组件的`providers`元数据中列出来的令牌。
|
||
tr
|
||
td(style="vertical-align: top") <code>source</code>
|
||
td
|
||
:marked
|
||
Where to find this element in the source component template.
|
||
|
||
source是在源组件模板中查询这个元素的处所。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>references</code>
|
||
td
|
||
:marked
|
||
Dictionary of objects associated with template local variables (e.g. `#foo`),
|
||
keyed by the local variable name.
|
||
|
||
与模板本地变量(比如`#foo`)关联的词典对象,关键字与本地变量名字配对。
|
||
|
||
a#query-predicate
|
||
:marked
|
||
The `DebugElement.query(predicate)` and `DebugElement.queryAll(predicate)` methods take a
|
||
predicate that filters the source element's subtree for matching `DebugElement`.
|
||
|
||
`DebugElement.query(predicate)`和`DebugElement.queryAll(predicate)`方法接受一个条件方法,
|
||
它过滤源元素的子树,返回匹配的`DebugElement`。
|
||
|
||
The predicate is any method that takes a `DebugElement` and returns a _truthy_ value.
|
||
The following example finds all `DebugElements` with a reference to a template local variable named "content":
|
||
|
||
这个条件方法是任何接受一个`DebugElement`并返回真值的方法。
|
||
下面的例子查询所有拥有名为`content`的模块本地变量的所有`DebugElement`:
|
||
+makeExample('testing/ts/src/app/bag/bag.spec.ts', 'custom-predicate')(format=".")
|
||
|
||
:marked
|
||
The Angular `By` class has three static methods for common predicates:
|
||
|
||
Angular的`By`类为常用条件方法提供了三个静态方法:
|
||
|
||
* `By.all` - return all elements.
|
||
|
||
`By.all` - 返回所有元素
|
||
|
||
* `By.css(selector)` - return elements with matching CSS selectors.
|
||
|
||
`By.css(selector)` - 返回符合CSS选择器的元素。
|
||
|
||
* `By.directive(directive)` - return elements that Angular matched to an instance of the directive class.
|
||
|
||
`By.directive(directive)` - 返回Angular能匹配一个指令类实例的所有元素。
|
||
|
||
+makeExample('testing/ts/src/app/hero/hero-list.component.spec.ts', 'by', 'src/app/hero/hero-list.component.spec.ts')(format=".")
|
||
|
||
// Removed on 12/02/2016 when ceased public discussion of the `Renderer`. Revive in future?
|
||
#renderer-tests
|
||
:marked
|
||
Many custom application directives inject the `Renderer` and call one of its `set...` methods.
|
||
|
||
很多制定应用程序指令注入`Renderer`并调用它其中一个方法`set...`。
|
||
|
||
The test environment substitutes the `DebugDomRender` for the runtime `Renderer`.
|
||
The `DebugDomRender` updates additional dictionary properties of the `DebugElement`
|
||
when something calls a `set...` method.
|
||
|
||
运行时的`Renderer`在测试环境中的替代品为`DebugDomRender`。
|
||
在调用`set...`方法时,`DebugDomRender`更新`DebugElement`额外词典属性。
|
||
|
||
These dictionary properties are primarily of interest to authors of Angular DOM inspection tools
|
||
but they may provide useful insights to testers as well.
|
||
|
||
这些词典属性主要是为Angular DOM检测工具准备的,但是它们可能对测试者提供有用的信息。
|
||
|
||
table
|
||
tr
|
||
thp Dictionary
|
||
p 词典thp Description
|
||
p 描述tr
|
||
td(style="vertical-align: top") <code>properties</code>
|
||
td
|
||
:marked
|
||
Updated by `Renderer.setElementProperty`.
|
||
Many Angular directives call it, including `NgModel`.
|
||
|
||
被`Renderer.setElementProperty`更新。
|
||
很多Angular指令调用它,包括`NgModel`。
|
||
tr
|
||
td(style="vertical-align: top") <code>attributes</code>
|
||
td
|
||
:marked
|
||
Updated by `Renderer.setElementAttribute`.
|
||
Angular `[attribute]` bindings call it.
|
||
|
||
被`Renderer.setElementAttribute`更新。
|
||
Angular的`[attribute]`绑定调用它。
|
||
tr
|
||
td(style="vertical-align: top") <code>classes</code>
|
||
td
|
||
:marked
|
||
Updated by `Renderer.setElementClass`.
|
||
Angular `[class]` bindings call it.
|
||
|
||
被`Renderer.setElementClass`更新。
|
||
Angular的`[class]`绑定调用它。
|
||
tr
|
||
td(style="vertical-align: top") <code>styles</code>
|
||
td
|
||
:marked
|
||
Updated by `Renderer.setElementStyle`.
|
||
Angular `[style]` bindings call it.
|
||
|
||
被`Renderer.setElementStyle`更新。
|
||
Angular的 `[style]`绑定调用它。
|
||
:marked
|
||
Here's an example of `Renderer` tests from the <live-example plnkr="bag-specs">live "Specs Bag" sample</live-example>.
|
||
|
||
下面是<live-example plnkr="bag-specs">在线“Specs Bag”例子</live-example>中`Renderer`测试程序的例子
|
||
|
||
+makeExample('testing/ts/src/app/bag/bag.spec.ts', 'dom-attributes')(format=".")
|
||
|
||
a(href="#top").to-top Back to top
|
||
a(href="#top").to-top 回到顶部
|
||
|
||
.l.hr
|
||
|
||
a#setup-files
|
||
:marked
|
||
## Test environment setup files
|
||
|
||
## 测试环境的设置文件
|
||
|
||
Unit testing requires some configuration and bootstrapping that is captured in _setup files_.
|
||
The setup files for this guide are provided for you when you follow the [Setup](setup.html) instructions.
|
||
The CLI delivers similar files with the same purpose.
|
||
|
||
单元测试需要一些配置和启动代码,它们被收集到了这些*设置文件*中。
|
||
当你遵循[环境设置](setup.html)中的步骤操作时,就会得到这些设置文件。
|
||
CLI工具也会生成类似的文件。
|
||
|
||
Here's a brief description of this guide's setup files:
|
||
|
||
下面是对本章中这些设置文件的简短说明:
|
||
|
||
.l-sub-section
|
||
:marked
|
||
The deep details of these files and how to reconfigure them for your needs
|
||
is a topic beyond the scope of this guide .
|
||
|
||
本章不会深入讲解这些文件的详情以及如何根据需要重新配置它们,那超出了本章的范围。
|
||
|
||
table(width="100%")
|
||
col(width="20%")
|
||
col(width="80%")
|
||
tr
|
||
th
|
||
p File
|
||
p 文件
|
||
th
|
||
p Description
|
||
p 描述
|
||
tr
|
||
td(style="vertical-align: top") <code>karma.conf.js</code>
|
||
td
|
||
:marked
|
||
The karma configuration file that specifies which plug-ins to use,
|
||
which application and test files to load, which browser(s) to use,
|
||
and how to report test results.
|
||
|
||
这个karma配置文件指定了要使用那些插件、要加载那些应用文件和测试文件、要使用哪些浏览器以及如何报告测试结果。
|
||
|
||
It loads three other setup files:
|
||
|
||
它加载了下列设置文件:
|
||
|
||
* `systemjs.config.js`
|
||
* `systemjs.config.extras.js`
|
||
* `karma-test-shim.js`
|
||
tr
|
||
td(style="vertical-align: top") <code>karma-test-shim.js</code>
|
||
td
|
||
:marked
|
||
This shim prepares karma specifically for the Angular test environment
|
||
and launches karma itself.
|
||
It loads the `systemjs.config.js` file as part of that process.
|
||
|
||
这个垫片(shim)文件为karma准备Angular特有的测试环境,并启动karma自身。
|
||
这期间,它还加载`systemjs.config.js`文件。
|
||
|
||
tr
|
||
td(style="vertical-align: top") <code>systemjs.config.js</code>
|
||
td
|
||
:marked
|
||
[SystemJS](https://github.com/systemjs/systemjs/blob/master/README.md)
|
||
loads the application and test files.
|
||
This script tells SystemJS where to find those files and how to load them.
|
||
It's the same version of `systemjs.config.js` you installed during [setup](#setup).
|
||
|
||
[SystemJS](https://github.com/systemjs/systemjs/blob/master/README.md)加载应用文件和测试文件。
|
||
这个脚本告诉SystemJS到哪里去找那些文件,以及如何加载它们。
|
||
它和你在[环境设置](#setup)期间安装的那个`systemjs.config.js`是同一个版本。
|
||
tr
|
||
td(style="vertical-align: top") <code>systemjs.config.extras.js</code>
|
||
td
|
||
:marked
|
||
An optional file that supplements the SystemJS configuration in `systemjs.config.js` with
|
||
configuration for the specific needs of the application itself.
|
||
|
||
一个可选的文件,它会为`systemjs.config.js`中提供SystemJS的配置加上应用自身需要的特殊配置。
|
||
|
||
A stock `systemjs.config.js` can't anticipate those needs.
|
||
You fill the gaps here.
|
||
|
||
常规的`systemjs.config.js`文件无法满足那些需求,我们需要自己填充它。
|
||
|
||
The sample version for this guide adds the **model barrel**
|
||
to the SystemJs `packages` configuration.
|
||
|
||
本章的例子中把**模型桶(barrel)*添加到了SystemJS的`packages`配置中。
|
||
tr
|
||
td(colspan="2")
|
||
+makeExample('testing/ts/src/systemjs.config.extras.js', '', 'systemjs.config.extras.js')(format='.')
|
||
|
||
:marked
|
||
### npm packages
|
||
|
||
### npm包
|
||
|
||
The sample tests are written to run in Jasmine and karma.
|
||
The two "fast path" setups added the appropriate Jasmine and karma npm packages to the
|
||
`devDependencies` section of the `package.json`.
|
||
They're installed when you run `npm install`.
|
||
|
||
这些范例测试是为在Jasmine和karma而写的。
|
||
那两条“捷径”设置会把适当的Jasmine和Karma包添加到`package.json`的`devDependencies`区。
|
||
当我们运行`npm install`时,它们就会被安装上。
|
||
|
||
a(href="#top").to-top Back to top
|
||
|
||
a(href="#top").to-top 返回顶部
|
||
|
||
.l.hr
|
||
|
||
#faq
|
||
.l-main-section
|
||
:marked
|
||
## FAQ: Frequently Asked Questions
|
||
|
||
## 常见问题
|
||
//
|
||
:marked
|
||
General
|
||
|
||
* [When are end-to-end (e2e) tests a good choice?](#q-when-e2e)
|
||
* [When to use the _TestBed_?](#q-why-testbed)
|
||
* [When to write isolated unit tests without the _TestBed_?](#q-when-no-testbed)
|
||
* [When can I skip _TestBed.compileComponents_?](#q-when-no-compile-components)
|
||
* [Why must _TestBed.compileComponents_ be called last?](#q-why-compile-components-is-last)
|
||
* [Why must _inject_ be called last?](#q-why-last-last)
|
||
* [What's the difference between _async_ and _fakeAsync_?](#q-async-vs-fake-async)
|
||
* [What's the difference between _whenStable_ and _tick_?](#q-when-stable-vs-tick)
|
||
* [How do I get something from the component's injector?](#q-component-injector)
|
||
* [Why do feature modules make testing easier?](#q-why-feature-modules)
|
||
* [When should I prefer the _DynamicTestModule_?](#q-dynamic-test-module)
|
||
* [How do I know if an injected service method was called?](#q-spy-on-service)
|
||
* [When must I call _detectChanges_ and why?](#q-detect-changes)
|
||
* [What's the difference between _triggerEventHandler_ and _dispatchEvent_?](#q-trigger-event-handler-vs-dispatch-event)
|
||
* [How do I find an element by directive?](#q-by-directive)
|
||
* [How do I extend Jasmine matchers?](#q-jasmine-matchers)
|
||
* [Why would I add a test folder and how?](#q-test-folder)
|
||
* [Why put specs next to the things they test?](#q-spec-file-location)
|
||
* [When would I put specs in a test folder?](#q-specs-in-test-folder)
|
||
* [How do I use the Jasmine HTML TestRunner in the browser?](#q-jasmine-browser-test-runner)
|
||
|
||
Resources
|
||
|
||
* [Where can I learn more about unit testing in JavaScript?](#q-js-unit-testing-resources)
|
||
* [Where can I learn more about testing with Jasmine?](#q-jasmine-resources)
|
||
* [Where can I learn more about testing with karma?](#q-karma-resources)
|
||
* [Where can I learn more about e2e testing with protractor?](#q-protractor-resources)
|
||
|
||
a(href="#top").to-top Back to top
|
||
|
||
.l-hr
|
||
|
||
#q-spec-file-location
|
||
:marked
|
||
### Why put specs next to the things they test?
|
||
|
||
### 为何将测试的spec配置文件放置到被测试文件的傍边?
|
||
|
||
It's a good idea to put unit test spec files in the same folder
|
||
as the application source code files that they test:
|
||
|
||
我们推荐将单元测试的spec配置文件放到与应用程序源代码文件所在的同一个文件夹中,因为:
|
||
|
||
* Such tests are easy to find.
|
||
|
||
这样的测试程序很容易被找到
|
||
|
||
* You see at a glance if a part of your application lacks tests.
|
||
|
||
你可以一眼看出应用程序的那些部分缺乏测试程序。
|
||
|
||
* Nearby tests can reveal how a part works in context.
|
||
|
||
临近的测试程序可以展示代码是如何在上下文中工作的
|
||
|
||
* When you move the source (inevitable), you remember to move the test.
|
||
|
||
当你移动代码(无可避免)时,你记得一起移动测试程序
|
||
|
||
* When you rename the source file (inevitable), you remember to rename the test file.
|
||
|
||
当你重命名源代码文件(无可避免),你记得重命名测试程序文件。
|
||
|
||
.l-hr
|
||
|
||
#q-specs-in-test-folder
|
||
:marked
|
||
### When would I put specs in a test folder?
|
||
|
||
### 什么时候我应该把测试spec文件放到测试目录中?
|
||
|
||
Application integration specs can test the interactions of multiple parts
|
||
spread across folders and modules.
|
||
They don't really belong to any part in particular, so they don't have a
|
||
natural home next to any one file.
|
||
|
||
应用程序的整合测试spec文件可以测试横跨多个目录和模块的多个部分之间的互动。
|
||
它们不属于任何部分,很自然,没有特别的地方存放它们。
|
||
|
||
It's often better to create an appropriate folder for them in the `tests` directory.
|
||
|
||
通常,在`test`目录中为它们创建一个合适的目录比较好。
|
||
|
||
Of course specs that test the test helpers belong in the `test` folder,
|
||
next to their corresponding helper files.
|
||
|
||
当然,**测试助手对象**的测试spec文件也属于`test`目录,与它们对应的助手文件相邻。
|