From 644e7a28d8fab1683e3c70d4e8bed86348460ef1 Mon Sep 17 00:00:00 2001 From: Judy Bogart Date: Mon, 23 Jul 2018 17:11:30 -0700 Subject: [PATCH] docs: add di-related api doc (#27731) PR Close #27731 --- .../guide/dependency-injection-in-action.md | 73 ++++++++++--------- .../common/src/location/location_strategy.ts | 13 ++-- packages/core/src/di/injectable.ts | 23 ++++-- packages/core/src/di/metadata.ts | 62 +++++++++++----- packages/core/src/metadata/di.ts | 53 ++++++++++++-- 5 files changed, 150 insertions(+), 74 deletions(-) diff --git a/aio/content/guide/dependency-injection-in-action.md b/aio/content/guide/dependency-injection-in-action.md index d622fde4d1..3e94cbdeb4 100644 --- a/aio/content/guide/dependency-injection-in-action.md +++ b/aio/content/guide/dependency-injection-in-action.md @@ -52,16 +52,16 @@ When all dependencies are in place, `AppComponent` displays the user information ## Limit service scope to a component subtree -An Angular application has multiple injectors, arranged in a tree hierarchy that parallels the component tree. -Each injector creates a singleton instance of a dependency. +An Angular application has multiple injectors, arranged in a tree hierarchy that parallels the component tree. +Each injector creates a singleton instance of a dependency. That same instance is injected wherever that injector provides that service. A particular service can be provided and created at any level of the injector hierarchy, which means that there can be multiple instances of a service if it is provided by multiple injectors. -Dependencies provided by the root injector can be injected into *any* component *anywhere* in the application. -In some cases, you might want to restrict service availability to a particular region of the application. +Dependencies provided by the root injector can be injected into *any* component *anywhere* in the application. +In some cases, you might want to restrict service availability to a particular region of the application. For instance, you might want to let users explicitly opt in to use a service, -rather than letting the root injector provide it automatically. +rather than letting the root injector provide it automatically. You can limit the scope of an injected service to a *branch* of the application hierarchy by providing that service *at the sub-root component for that branch*. @@ -146,34 +146,34 @@ and confirm that the three `HeroBioComponent` instances have their own cached he When a class requires a dependency, that dependency is added to the constructor as a parameter. When Angular needs to instantiate the class, it calls upon the DI framework to supply the dependency. By default, the DI framework searches for a provider in the injector hierarchy, -starting at the component's local injector of the component, and if necessary bubbling up +starting at the component's local injector of the component, and if necessary bubbling up through the injector tree until it reaches the root injector. -* The first injector configured with a provider supplies the dependency (a service instance or value) to the constructor. +* The first injector configured with a provider supplies the dependency (a service instance or value) to the constructor. * If no provider is found in the root injector, the DI framework returns null to the constructor. There are a number of options for modifying the default search behavior, using _parameter decorators_ -on the service-valued parameters of a class constructor. +on the service-valued parameters of a class constructor. {@a optional} ### Make a dependency `@Optional` and limit search with `@Host` -Dependencies can be registered at any level in the component hierarchy. -When a component requests a dependency, Angular starts with that component's injector -and walks up the injector tree until it finds the first suitable provider. +Dependencies can be registered at any level in the component hierarchy. +When a component requests a dependency, Angular starts with that component's injector +and walks up the injector tree until it finds the first suitable provider. Angular throws an error if it can't find the dependency during that walk. In some cases, you need to limit the search or accommodate a missing dependency. You can modify Angular's search behavior with the `@Host` and `@Optional` qualifying -decorators on a service-valued parameter of the component's constructor. +decorators on a service-valued parameter of the component's constructor. * The `@Optional` property decorator tells Angular to return null when it can't find the dependency. -* The `@Host` property decorator stops the upward search at the *host component*. -The host component is typically the component requesting the dependency. -However, when this component is projected into a *parent* component, +* The `@Host` property decorator stops the upward search at the *host component*. +The host component is typically the component requesting the dependency. +However, when this component is projected into a *parent* component, that parent component becomes the host. The following example covers this second case. These decorators can be used individually or together, as shown in the example. @@ -238,8 +238,8 @@ Here's `HeroBiosAndContactsComponent` in action. If you comment out the `@Host()` decorator, Angular walks up the injector ancestor tree -until it finds the logger at the `AppComponent` level. -The logger logic kicks in and the hero display updates +until it finds the logger at the `AppComponent` level. +The logger logic kicks in and the hero display updates with the "!!!" marker to indicate that the logger was found.
@@ -254,7 +254,7 @@ the app throws an exception when it cannot find the required logger at the host ### Supply a custom provider with `@Inject` -Using a custom provider allows you to provide a concrete implementation for implicit dependencies, such as built-in browser APIs. The following example uses an `InjectionToken` to provide the [localStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage) browser API as a dependency in the `BrowserStorageService`. +Using a custom provider allows you to provide a concrete implementation for implicit dependencies, such as built-in browser APIs. The following example uses an `InjectionToken` to provide the [localStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage) browser API as a dependency in the `BrowserStorageService`. @@ -262,6 +262,8 @@ Using a custom provider allows you to provide a concrete implementation for impl The `factory` function returns the `localStorage` property that is attached to the browser window object. The `Inject` decorator is a constructor parameter used to specify a custom provider of a dependency. This custom provider can now be overridden during testing with a mock API of `localStorage` instead of interactive with real browser APIs. +{@a skip} + ### Modify the provider search with `@Self` and `@SkipSelf` Providers can also be scoped by injector through constructor parameter decorators. The following example overrides the `BROWSER_STORAGE` token in the `Component` class `providers` with the `sessionStorage` browser API. The same `BrowserStorageService` is injected twice in the constructor, decorated with `@Self` and `@SkipSelf` to define which injector handles the provider dependency. @@ -291,7 +293,7 @@ The directive sets the background to a highlight color when the user mouses over DOM element to which the directive is applied. Angular sets the constructor's `el` parameter to the injected `ElementRef`. -(An `ElementRef` is a wrapper around a DOM element, +(An `ElementRef` is a wrapper around a DOM element, whose `nativeElement` property exposes the DOM element for the directive to manipulate.) The sample code applies the directive's `myHighlight` attribute to two `
` tags, @@ -332,7 +334,7 @@ Angular asks the injector for the service associated with `LoggerService` and assigns the returned value to the `logger` parameter. If the injector has already cached an instance of the service associated with the token, -it provides that instance. +it provides that instance. If it doesn't, it needs to make one using the provider associated with the token.
@@ -346,7 +348,7 @@ If the search fails, the injector throws an error—unless the request was [ A new injector has no providers. Angular initializes the injectors it creates with a set of preferred providers. -You have to configure providers for your own app-specific dependencies. +You have to configure providers for your own app-specific dependencies. {@a defining-providers} @@ -355,7 +357,7 @@ You have to configure providers for your own app-specific dependencies. ### Defining providers A dependency can't always be created by the default method of instantiating a class. -You learned about some other methods in [Dependency Providers](guide/dependency-injection-providers). +You learned about some other methods in [Dependency Providers](guide/dependency-injection-providers). The following `HeroOfTheMonthComponent` example demonstrates many of the alternatives and why you need them. It's visually simple: a few properties and the logs produced by a logger. @@ -364,7 +366,7 @@ It's visually simple: a few properties and the logs produced by a logger.
The code behind it customizes how and where the DI framework provides dependencies. -The use cases illustrate different ways to use the [*provide* object literal](guide/dependency-injection-providers#provide) to associate a definition object with a DI token. +The use cases illustrate different ways to use the [*provide* object literal](guide/dependency-injection-providers#provide) to associate a definition object with a DI token. @@ -389,13 +391,13 @@ The `HeroOfTheMonthComponent` example has two value providers. * The first provides an existing instance of the `Hero` class to use for the `Hero` token, rather than -requiring the injector to create a new instance with `new` or use its own cached instance. +requiring the injector to create a new instance with `new` or use its own cached instance. Here, the token is the class itself. * The second specifies a literal string resource to use for the `TITLE` token. The `TITLE` provider token is *not* a class, but is instead a special kind of provider lookup key called an [injection token](guide/dependency-injection-in-action#injection-token), represented by -an `InjectionToken` instance. +an `InjectionToken` instance. You can use an injection token for any kind of provider but it's particularly helpful when the dependency is a simple value like a string, a number, or a function. @@ -414,12 +416,12 @@ Other types of providers can create their values *lazily*; that is, when they're {@a useclass} -#### Class providers: `useClass` +#### Class providers: `useClass` The `useClass` provider key lets you create and return a new instance of the specified class. You can use this type of provider to substitute an *alternative implementation* -for a common or default class. +for a common or default class. The alternative implementation could, for example, implement a different strategy, extend the default class, or emulate the behavior of the real class in a test case. @@ -502,7 +504,7 @@ This is illustrated in the following image, which displays the logging date. {@a usefactory} -#### Factory providers: `useFactory` +#### Factory providers: `useFactory` The `useFactory` provider key lets you create a dependency object by calling a factory function, as in the following example. @@ -537,7 +539,7 @@ the passed-in state value and the injected services `Hero` and `HeroService`. The provider factory function (returned by `runnersUpFactory()`) returns the actual dependency object, the string of names. -* The function takes a winning `Hero` and a `HeroService` as arguments. +* The function takes a winning `Hero` and a `HeroService` as arguments. Angular supplies these arguments from injected values identified by the two *tokens* in the `deps` array. @@ -588,7 +590,7 @@ But they did neither. `MinimalLogger` is used only as a dependency injection token. When you use a class this way, it's called a *class interface*. - + As mentioned in [DI Providers](guide/dependency-injection-providers#interface-not-valid-token), an interface is not a valid DI token because it is a TypeScript artifact that doesn't exist at run time. Use this abstract class interface to get the strong typing of an interface, @@ -609,7 +611,7 @@ The `MinimalLogger` transpiles to this unoptimized, pre-minified JavaScript for Notice that it doesn't have any members. It never grows no matter how many members you add to the class, -as long as those members are typed but not implemented. +as long as those members are typed but not implemented. Look again at the TypeScript `MinimalLogger` class to confirm that it has no implementation. @@ -736,7 +738,7 @@ Break the circularity with `forwardRef`. - - diff --git a/packages/common/src/location/location_strategy.ts b/packages/common/src/location/location_strategy.ts index f608180acd..2cbf0de24e 100644 --- a/packages/common/src/location/location_strategy.ts +++ b/packages/common/src/location/location_strategy.ts @@ -40,16 +40,15 @@ export abstract class LocationStrategy { /** - * The `APP_BASE_HREF` token represents the base href to be used with the - * {@link PathLocationStrategy}. - * - * If you're using {@link PathLocationStrategy}, you must provide a provider to a string - * representing the URL prefix that should be preserved when generating and recognizing - * URLs. + * A predefined [DI token](guide/glossary#di-token) for the base href + * to be used with the `PathLocationStrategy`. + * The base href is the URL prefix that should be preserved when generating + * and recognizing URLs. * * @usageNotes * - * ### Example + * The following example shows how to use this token to configure the root app injector + * with a base href value, so that the DI framework can supply the dependency anywhere in the app. * * ```typescript * import {Component, NgModule} from '@angular/core'; diff --git a/packages/core/src/di/injectable.ts b/packages/core/src/di/injectable.ts index c628ff2938..772dc88e40 100644 --- a/packages/core/src/di/injectable.ts +++ b/packages/core/src/di/injectable.ts @@ -30,17 +30,20 @@ export type InjectableProvider = ValueSansProvider | ExistingSansProvider | */ export interface InjectableDecorator { /** - * A marker metadata that marks a class as available to `Injector` for creation. + * Marks a class as available to `Injector` for creation. * - * For more details, see the ["Dependency Injection Guide"](guide/dependency-injection). + * @see [Introduction to Services and DI](guide/architecture-services) + * @see [Dependency Injection Guide](guide/dependency-injection) * * @usageNotes - * ### Example + * + * The following example shows how service classes are properly marked as + * injectable. * * {@example core/di/ts/metadata_spec.ts region='Injectable'} * - * `Injector` will throw an error when trying to instantiate a class that - * does not have `@Injectable` marker, as shown in the example below. + * `Injector` throws an error if it tries to instantiate a class that + * is not decorated with `@Injectable`, as shown in the following example. * * {@example core/di/ts/metadata_spec.ts region='InjectableThrows'} * @@ -56,7 +59,15 @@ export interface InjectableDecorator { * * @publicApi */ -export interface Injectable { providedIn?: Type|'root'|null; } +export interface Injectable { + /** + * Determines which injectors will provide the injectable, + * by either associating it with an @NgModule or other `InjectorType`, + * or by specifying that this injectable should be provided in the + * 'root' injector, which will be the application-level injector in most apps. + */ + providedIn?: Type|'root'|null; +} /** * Injectable decorator and metadata. diff --git a/packages/core/src/di/metadata.ts b/packages/core/src/di/metadata.ts index 5537a4e7db..95c090bda3 100644 --- a/packages/core/src/di/metadata.ts +++ b/packages/core/src/di/metadata.ts @@ -17,10 +17,10 @@ import {makeParamDecorator} from '../util/decorators'; */ export interface InjectDecorator { /** - * A constructor parameter decorator that specifies a - * custom provider of a dependency. + * A parameter decorator on a dependency parameter of a class constructor + * that specifies a custom provider of the dependency. * - * @see ["Dependency Injection Guide"](guide/dependency-injection). + * Learn more in the ["Dependency Injection Guide"](guide/dependency-injection). * * @usageNotes * The following example shows a class constructor that specifies a @@ -28,7 +28,7 @@ export interface InjectDecorator { * * {@example core/di/ts/metadata_spec.ts region='Inject'} * - * When `@Inject()` is not present, the `Injector` uses the type annotation of the + * When `@Inject()` is not present, the injector uses the type annotation of the * parameter as the provider. * * {@example core/di/ts/metadata_spec.ts region='InjectWithoutDecorator'} @@ -44,7 +44,7 @@ export interface InjectDecorator { */ export interface Inject { /** - * Injector token that maps to the dependency to be injected. + * A [DI token](guide/glossary#di-token) that maps to the dependency to be injected. */ token: any; } @@ -65,14 +65,21 @@ export const Inject: InjectDecorator = makeParamDecorator('Inject', (token: any) */ export interface OptionalDecorator { /** - * A constructor parameter decorator that marks a dependency as optional. - * + * A parameter decorator to be used on constructor parameters, + * which marks the parameter as being an optional dependency. * The DI framework provides null if the dependency is not found. - * For example, the following code allows the possibility of a null result: + * + * Can be used together with other parameter decorators + * that modify how dependency injection operates. + * + * Learn more in the ["Dependency Injection Guide"](guide/dependency-injection). + * + * @usageNotes + * + * The following code allows the possibility of a null result: * * {@example core/di/ts/metadata_spec.ts region='Optional'} * - * @see ["Dependency Injection Guide"](guide/dependency-injection). */ (): any; new (): Optional; @@ -100,8 +107,13 @@ export const Optional: OptionalDecorator = makeParamDecorator('Optional'); */ export interface SelfDecorator { /** - * A constructor parameter decorator that tells the DI framework - * to retrieve a dependency only from the local injector. + * A parameter decorator to be used on constructor parameters, + * which tells the DI framework to start dependency resolution from the local injector. + * + * Resolution works upward through the injector hierarchy, so the children + * of this class must configure their own providers or be prepared for a null result. + * + * @usageNotes * * In the following example, the dependency can be resolved * by the local injector when instantiating the class itself, but not @@ -109,8 +121,8 @@ export interface SelfDecorator { * * {@example core/di/ts/metadata_spec.ts region='Self'} * - * @see ["Dependency Injection Guide"](guide/dependency-injection). - * + * @see `SkipSelf` + * @see `Optional` * */ (): any; @@ -140,16 +152,23 @@ export const Self: SelfDecorator = makeParamDecorator('Self'); */ export interface SkipSelfDecorator { /** - * A constructor parameter decorator that tells the DI framework - * that dependency resolution should start from the parent injector. + * A parameter decorator to be used on constructor parameters, + * which tells the DI framework to start dependency resolution from the parent injector. + * Resolution works upward through the injector hierarchy, so the local injector + * is not checked for a provider. + * + * @usageNotes * * In the following example, the dependency can be resolved when * instantiating a child, but not when instantiating the class itself. * * {@example core/di/ts/metadata_spec.ts region='SkipSelf'} * - * @see ["Dependency Injection Guide"](guide/dependency-injection). + * Learn more in the + * [Dependency Injection guide](guide/dependency-injection-in-action#skip). * + * @see `Self` + * @see `Optional` * */ (): any; @@ -178,14 +197,17 @@ export const SkipSelf: SkipSelfDecorator = makeParamDecorator('SkipSelf'); */ export interface HostDecorator { /** - * A constructor parameter decorator that tells the DI framework - * to retrieve a dependency from any injector until - * reaching the host element of the current component. + * A parameter decorator on a view-provider parameter of a class constructor + * that tells the DI framework to resolve the view by checking injectors of child + * elements, and stop when reaching the host element of the current component. * - * @see ["Dependency Injection Guide"](guide/dependency-injection). + * For an extended example, see + * ["Dependency Injection Guide"](guide/dependency-injection-in-action#optional). * * @usageNotes * + * The following shows use with the `@Optional` decorator, and allows for a null result. + * * {@example core/di/ts/metadata_spec.ts region='Host'} */ (): any; diff --git a/packages/core/src/metadata/di.ts b/packages/core/src/metadata/di.ts index e01e592076..867259c416 100644 --- a/packages/core/src/metadata/di.ts +++ b/packages/core/src/metadata/di.ts @@ -11,15 +11,16 @@ import {Type} from '../interface/type'; import {makePropDecorator} from '../util/decorators'; /** - * This token can be used to create a virtual provider that will populate the - * `entryComponents` fields of components and ng modules based on its `useValue`. + * A DI token that you can use to create a virtual [provider](guide/glossary#provider) + * that will populate the `entryComponents` field of components and NgModules + * based on its `useValue` property value. * All components that are referenced in the `useValue` value (either directly - * or in a nested array or map) will be added to the `entryComponents` property. + * or in a nested array or map) are added to the `entryComponents` property. * * @usageNotes - * ### Example + * * The following example shows how the router can populate the `entryComponents` - * field of an NgModule based on the router configuration which refers + * field of an NgModule based on a router configuration that refers * to components. * * ```typescript @@ -47,6 +48,48 @@ import {makePropDecorator} from '../util/decorators'; */ export const ANALYZE_FOR_ENTRY_COMPONENTS = new InjectionToken('AnalyzeForEntryComponents'); +/** + * Type of the `Attribute` decorator / constructor function. + * + * @publicApi + */ +export interface AttributeDecorator { + /** + * Specifies that a constant attribute value should be injected. + * + * The directive can inject constant string literals of host element attributes. + * + * @usageNotes + * ### Example + * + * Suppose we have an `` element and want to know its `type`. + * + * ```html + * + * ``` + * + * A decorator can inject string literal `text` as in the following example. + * + * {@example core/ts/metadata/metadata.ts region='attributeMetadata'} + * + * @publicApi + */ + (name: string): any; + new (name: string): Attribute; +} + + +/** + * Type of the Attribute metadata. + * + * @publicApi + */ +export interface Attribute { + /** + * The name of the attribute to be injected into the constructor. + */ + attributeName?: string; +} /** * Type of the Query metadata.