docs: several minor `NgModule` guide fixes/improvements (#21589)
PR Close #21589
This commit is contained in:
parent
bf248792eb
commit
81e87095b4
|
@ -15,8 +15,7 @@ const routes: Routes = [
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [RouterModule.forChild(routes)],
|
imports: [RouterModule.forChild(routes)],
|
||||||
exports: [RouterModule],
|
exports: [RouterModule]
|
||||||
providers: []
|
|
||||||
})
|
})
|
||||||
export class CustomersRoutingModule { }
|
export class CustomersRoutingModule { }
|
||||||
// #enddocregion customers-routing-module
|
// #enddocregion customers-routing-module
|
||||||
|
|
|
@ -6,23 +6,17 @@ import { Routes, RouterModule } from '@angular/router';
|
||||||
// #docregion orders-routing-module-detail
|
// #docregion orders-routing-module-detail
|
||||||
import { OrderListComponent } from './order-list/order-list.component';
|
import { OrderListComponent } from './order-list/order-list.component';
|
||||||
|
|
||||||
|
|
||||||
const routes: Routes = [
|
const routes: Routes = [
|
||||||
{
|
{
|
||||||
path: '',
|
path: '',
|
||||||
component: OrderListComponent
|
component: OrderListComponent
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// #enddocregion orders-routing-module-detail
|
// #enddocregion orders-routing-module-detail
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [
|
imports: [RouterModule.forChild(routes)],
|
||||||
RouterModule.forChild(routes)
|
exports: [RouterModule]
|
||||||
],
|
|
||||||
exports: [
|
|
||||||
RouterModule
|
|
||||||
]
|
|
||||||
})
|
})
|
||||||
export class OrdersRoutingModule { }
|
export class OrdersRoutingModule { }
|
||||||
// #enddocregion orders-routing-module
|
// #enddocregion orders-routing-module
|
||||||
|
|
|
@ -7,7 +7,7 @@ A basic understanding of the following concepts:
|
||||||
|
|
||||||
<hr />
|
<hr />
|
||||||
|
|
||||||
An entry component is any component that Angular loads imperatively, (which means you’re not referencing it in the template), by type. You specify an entry component by bootstrapping it in an NgModule,or including it in a routing definition.
|
An entry component is any component that Angular loads imperatively, (which means you’re not referencing it in the template), by type. You specify an entry component by bootstrapping it in an NgModule, or including it in a routing definition.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
|
@ -28,30 +28,31 @@ There are two main kinds of entry components:
|
||||||
The following is an example of specifying a bootstrapped component,
|
The following is an example of specifying a bootstrapped component,
|
||||||
`AppComponent`, in a basic `app.module.ts`:
|
`AppComponent`, in a basic `app.module.ts`:
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
AppComponent
|
AppComponent
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
FormsModule,
|
FormsModule,
|
||||||
HttpModule,
|
HttpModule,
|
||||||
AppRoutingModule
|
AppRoutingModule
|
||||||
],
|
],
|
||||||
providers: [],
|
providers: [],
|
||||||
bootstrap: [AppComponent] // bootstrapped entry component
|
bootstrap: [AppComponent] // bootstrapped entry component
|
||||||
})
|
})
|
||||||
```
|
```
|
||||||
|
|
||||||
A bootstrapped component is an entry component
|
A bootstrapped component is an entry component
|
||||||
that Angular loads into the DOM during the bootstrap (application launch), process.
|
that Angular loads into the DOM during the bootstrap process (application launch).
|
||||||
Other entry components are loaded dynamically by other means, such as with the router.
|
Other entry components are loaded dynamically by other means, such as with the router.
|
||||||
|
|
||||||
Angular loads a root `AppComponent` dynamically because it's listed by type in `@NgModule.bootstrap`.
|
Angular loads a root `AppComponent` dynamically because it's listed by type in `@NgModule.bootstrap`.
|
||||||
|
|
||||||
<div class="alert is-helpful">
|
<div class="alert is-helpful">
|
||||||
|
|
||||||
A component can also be bootstrapped imperatively with the module's `ngDoBootstrap()` method.
|
A component can also be bootstrapped imperatively in the module's `ngDoBootstrap()` method.
|
||||||
The `@NgModule.bootstrap` property tells the compiler that this is an entry component and
|
The `@NgModule.bootstrap` property tells the compiler that this is an entry component and
|
||||||
it should generate code to bootstrap the application with this component.
|
it should generate code to bootstrap the application with this component.
|
||||||
|
|
||||||
|
@ -66,18 +67,18 @@ A bootstrapped component is necessarily an entry component because bootstrapping
|
||||||
The second kind of entry component occurs in a route definition like
|
The second kind of entry component occurs in a route definition like
|
||||||
this:
|
this:
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
const routes: Routes = [
|
const routes: Routes = [
|
||||||
{
|
{
|
||||||
path: '',
|
path: '',
|
||||||
component: CustomerListComponent
|
component: CustomerListComponent
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
```
|
```
|
||||||
|
|
||||||
A route definition refers to a component by its type with `component: CustomerListComponent`.
|
A route definition refers to a component by its type with `component: CustomerListComponent`.
|
||||||
|
|
||||||
All router components must be `entryComponents`. Because this would require you to add the component in two places (router and `entryComponent`) the Compiler is smart enough to recognize that this is a router definition and automatically add the router component into `entryComponents`.
|
All router components must be entry components. Because this would require you to add the component in two places (router and `entryComponents`) the Compiler is smart enough to recognize that this is a router definition and automatically add the router component into `entryComponents`.
|
||||||
|
|
||||||
|
|
||||||
## The `entryComponents` array
|
## The `entryComponents` array
|
||||||
|
@ -93,11 +94,11 @@ The code should contain only the classes that you actually need and
|
||||||
exclude components that are never used. For this reason, the Angular compiler only generates code for components which are reachable from the `entryComponents`; This means that adding more references to `@NgModule.declarations` does not imply that they will necessarily be included in the final bundle.
|
exclude components that are never used. For this reason, the Angular compiler only generates code for components which are reachable from the `entryComponents`; This means that adding more references to `@NgModule.declarations` does not imply that they will necessarily be included in the final bundle.
|
||||||
|
|
||||||
In fact, many libraries declare and export components you'll never use.
|
In fact, many libraries declare and export components you'll never use.
|
||||||
For example, a material design library will export all components because it doesn’t know which ones the you will use. However, it is unlikely that the you will use them all.
|
For example, a material design library will export all components because it doesn’t know which ones you will use. However, it is unlikely that you will use them all.
|
||||||
For the ones you don't reference, the tree shaker drops these components from the final code package.
|
For the ones you don't reference, the tree shaker drops these components from the final code package.
|
||||||
|
|
||||||
If a component isn't an _entry component_ or isn't found in a template,
|
If a component isn't an _entry component_ or isn't found in a template,
|
||||||
The tree shaker will throw it away. So, it's best to add only the components that are truly entry components to help keep your app
|
the tree shaker will throw it away. So, it's best to add only the components that are truly entry components to help keep your app
|
||||||
as trim as possible.
|
as trim as possible.
|
||||||
|
|
||||||
|
|
||||||
|
@ -110,7 +111,3 @@ You may also be interested in the following:
|
||||||
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [NgModules FAQ](guide/ngmodule-faq).
|
* [NgModules FAQ](guide/ngmodule-faq).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -36,7 +36,7 @@ pipes that it shares.
|
||||||
Assuming you already have a CLI generated app, create a feature
|
Assuming you already have a CLI generated app, create a feature
|
||||||
module using the CLI by entering the following command in the
|
module using the CLI by entering the following command in the
|
||||||
root project directory. Replace `CustomerDashboard` with the
|
root project directory. Replace `CustomerDashboard` with the
|
||||||
name of your module. You can omit the word module because the CLI appends it:
|
name of your module. You can omit the "Module" suffix from the name because the CLI appends it:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
ng generate module CustomerDashboard
|
ng generate module CustomerDashboard
|
||||||
|
@ -46,21 +46,17 @@ ng generate module CustomerDashboard
|
||||||
|
|
||||||
This causes the CLI to create a folder called `customer-dashboard` with a file inside called `customer-dashboard.module.ts` with the following contents:
|
This causes the CLI to create a folder called `customer-dashboard` with a file inside called `customer-dashboard.module.ts` with the following contents:
|
||||||
|
|
||||||
```ts
|
```typescript
|
||||||
|
|
||||||
import { NgModule } from '@angular/core';
|
import { NgModule } from '@angular/core';
|
||||||
import { CommonModule } from '@angular/common';
|
import { CommonModule } from '@angular/common';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [
|
imports: [
|
||||||
CommonModule
|
CommonModule
|
||||||
],
|
],
|
||||||
declarations: []
|
declarations: []
|
||||||
})
|
})
|
||||||
|
|
||||||
export class CustomerDashboardModule { }
|
export class CustomerDashboardModule { }
|
||||||
|
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
The structure of an NgModule is the same whether it is a root module or a feature module. In the CLI generated feature module, there are two JavaScript import statements at the top of the file: the first imports `NgModule`, which, like the root module, lets you use the `@NgModule` decorator; the second imports `CommonModule`, which contributes many common directives such as `ngIf` and `ngFor`. Feature modules import `CommonModule` instead of `BrowserModule`, which is only imported once in the root module. `CommonModule` only contains information for common directives such as `ngIf` and `ngFor` which are needed in most templates, whereas `BrowserModule` configures the Angular app for the browser which needs to be done only once.
|
The structure of an NgModule is the same whether it is a root module or a feature module. In the CLI generated feature module, there are two JavaScript import statements at the top of the file: the first imports `NgModule`, which, like the root module, lets you use the `@NgModule` decorator; the second imports `CommonModule`, which contributes many common directives such as `ngIf` and `ngFor`. Feature modules import `CommonModule` instead of `BrowserModule`, which is only imported once in the root module. `CommonModule` only contains information for common directives such as `ngIf` and `ngFor` which are needed in most templates, whereas `BrowserModule` configures the Angular app for the browser which needs to be done only once.
|
||||||
|
@ -76,7 +72,7 @@ ng generate component customer-dashboard/CustomerDashboard
|
||||||
This generates a folder for the new component within the customer-dashboard folder and updates the feature module with the `CustomerDashboardComponent` info:
|
This generates a folder for the new component within the customer-dashboard folder and updates the feature module with the `CustomerDashboardComponent` info:
|
||||||
|
|
||||||
|
|
||||||
<code-example path="feature-modules/src/app/customer-dashboard/customer-dashboard.module.ts" region="customer-dashboard-component" title="src/app/customer-dashboard.module.ts" linenums="false">
|
<code-example path="feature-modules/src/app/customer-dashboard/customer-dashboard.module.ts" region="customer-dashboard-component" title="src/app/customer-dashboard/customer-dashboard.module.ts" linenums="false">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
|
||||||
|
@ -85,7 +81,7 @@ The `CustomerDashboardComponent` is now in the JavaScript import list at the top
|
||||||
|
|
||||||
## Importing a feature module
|
## Importing a feature module
|
||||||
|
|
||||||
To incorporate the feature module into your app, you have to let the root module, `app.module.ts`, know about it. Notice the `CustomerDashboardModule` export at the bottom of `customer-dashboard.module.ts`. This exposes it so that other modules can get to it. To import it into the AppModule, add it to the imports in `app.module.ts` and to the imports array:
|
To incorporate the feature module into your app, you have to let the root module, `app.module.ts`, know about it. Notice the `CustomerDashboardModule` export at the bottom of `customer-dashboard.module.ts`. This exposes it so that other modules can get to it. To import it into the `AppModule`, add it to the imports in `app.module.ts` and to the `imports` array:
|
||||||
|
|
||||||
<code-example path="feature-modules/src/app/app.module.ts" region="app-module" title="src/app/app.module.ts" linenums="false">
|
<code-example path="feature-modules/src/app/app.module.ts" region="app-module" title="src/app/app.module.ts" linenums="false">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
@ -104,7 +100,7 @@ When the CLI generated the `CustomerDashboardComponent` for the feature module,
|
||||||
|
|
||||||
To see this HTML in the `AppComponent`, you first have to export the `CustomerDashboardComponent` in the `CustomerDashboardModule`. In `customer-dashboard.module.ts`, just beneath the declarations array, add an exports array containing `CustomerDashboardModule`:
|
To see this HTML in the `AppComponent`, you first have to export the `CustomerDashboardComponent` in the `CustomerDashboardModule`. In `customer-dashboard.module.ts`, just beneath the declarations array, add an exports array containing `CustomerDashboardModule`:
|
||||||
|
|
||||||
<code-example path="feature-modules/src/app/customer-dashboard/customer-dashboard.module.ts" region="component-exports" title="src/app/customer-dashboard.module.ts" linenums="false">
|
<code-example path="feature-modules/src/app/customer-dashboard/customer-dashboard.module.ts" region="component-exports" title="src/app/customer-dashboard/customer-dashboard.module.ts" linenums="false">
|
||||||
</code-example>
|
</code-example>
|
||||||
|
|
||||||
|
|
||||||
|
@ -129,6 +125,4 @@ Now, in addition to the title that renders by default, the `CustomerDashboardCom
|
||||||
You may also be interested in the following:
|
You may also be interested in the following:
|
||||||
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [Types of NgModules](guide/module-types).
|
* [Types of Feature Modules](guide/module-types).
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -55,13 +55,13 @@ of some of the things they contain:
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>RouterModule</code></td>
|
<td><code>RouterModule</code></td>
|
||||||
<td><code>@angular/forms</code></td>
|
<td><code>@angular/router</code></td>
|
||||||
<td>For Routing and when you want to use <code>RouterLink</code>,<code>.forRoot()</code>, and <code>.forChild()</code></td>
|
<td>For Routing and when you want to use <code>RouterLink</code>,<code>.forRoot()</code>, and <code>.forChild()</code></td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>HttpModule</code></td>
|
<td><code>HttpClientModule</code></td>
|
||||||
<td><code>@angular/http</code></td>
|
<td><code>@angular/common/http</code></td>
|
||||||
<td>When you to talk to a server</td>
|
<td>When you to talk to a server</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
@ -73,29 +73,25 @@ When you use these Angular modules, import them in `AppModule`,
|
||||||
or your feature module as appropriate, and list them in the `@NgModule`
|
or your feature module as appropriate, and list them in the `@NgModule`
|
||||||
`imports` array. For example, in the basic app generated by the CLI,
|
`imports` array. For example, in the basic app generated by the CLI,
|
||||||
`BrowserModule` is the first import at the top of the `AppModule`,
|
`BrowserModule` is the first import at the top of the `AppModule`,
|
||||||
`app.module.ts`. Notice that this is the same case for `FormsModule` and `HttpModule`.
|
`app.module.ts`.
|
||||||
|
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
/* import modules so that AppModule can access them */
|
/* import modules so that AppModule can access them */
|
||||||
import { BrowserModule } from '@angular/platform-browser';
|
import { BrowserModule } from '@angular/platform-browser';
|
||||||
import { NgModule } from '@angular/core';
|
import { NgModule } from '@angular/core';
|
||||||
import { FormsModule } from '@angular/forms';
|
|
||||||
import { HttpModule } from '@angular/http';
|
|
||||||
|
|
||||||
import { AppComponent } from './app.component';
|
import { AppComponent } from './app.component';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
AppComponent
|
AppComponent
|
||||||
],
|
],
|
||||||
imports: [ /* add modules here so Angular knows to use them */
|
imports: [ /* add modules here so Angular knows to use them */
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
FormsModule,
|
],
|
||||||
HttpModule
|
providers: [],
|
||||||
],
|
bootstrap: [AppComponent]
|
||||||
providers: [],
|
|
||||||
bootstrap: [AppComponent]
|
|
||||||
})
|
})
|
||||||
export class AppModule { }
|
export class AppModule { }
|
||||||
```
|
```
|
||||||
|
@ -135,4 +131,3 @@ You may also be interested in the following:
|
||||||
* [Bootstrapping](guide/bootstrapping).
|
* [Bootstrapping](guide/bootstrapping).
|
||||||
* [NgModules](guide/ngmodules).
|
* [NgModules](guide/ngmodules).
|
||||||
* [JavaScript Modules vs. NgModules](guide/ngmodule-vs-jsmodule).
|
* [JavaScript Modules vs. NgModules](guide/ngmodule-vs-jsmodule).
|
||||||
|
|
||||||
|
|
|
@ -5,10 +5,11 @@ A basic understanding of the following:
|
||||||
* [Feature Modules](guide/feature-modules).
|
* [Feature Modules](guide/feature-modules).
|
||||||
* [JavaScript Modules vs. NgModules](guide/ngmodule-vs-jsmodule).
|
* [JavaScript Modules vs. NgModules](guide/ngmodule-vs-jsmodule).
|
||||||
* [Frequently Used Modules](guide/frequent-ngmodules).
|
* [Frequently Used Modules](guide/frequent-ngmodules).
|
||||||
* [Types of Modules](guide/module-types).
|
* [Types of Feature Modules](guide/module-types).
|
||||||
* [Routing and Navigation](guide/router).
|
* [Routing and Navigation](guide/router).
|
||||||
|
|
||||||
For the final sample app with two lazy loaded modules that this page describes, see the <live-example></live-example>.
|
For the final sample app with two lazy loaded modules that this page describes, see the
|
||||||
|
<live-example></live-example>.
|
||||||
|
|
||||||
<hr>
|
<hr>
|
||||||
|
|
||||||
|
@ -33,7 +34,7 @@ ng new customer-app --routing
|
||||||
|
|
||||||
This creates an app called `customer-app` and the `--routing` flag
|
This creates an app called `customer-app` and the `--routing` flag
|
||||||
generates a file called `app-routing.module.ts`, which is one of
|
generates a file called `app-routing.module.ts`, which is one of
|
||||||
the files you need for setting up lazy loading for your feature module.
|
the files you need for setting up lazy loading for your feature module.
|
||||||
Navigate into the project by issuing the command `cd customer-app`.
|
Navigate into the project by issuing the command `cd customer-app`.
|
||||||
|
|
||||||
## Create a feature module with routing
|
## Create a feature module with routing
|
||||||
|
@ -64,7 +65,7 @@ ng generate component customers/customer-list
|
||||||
```
|
```
|
||||||
|
|
||||||
This creates a folder inside of `customers` called `customer-list`
|
This creates a folder inside of `customers` called `customer-list`
|
||||||
with the four files that make up the component.
|
with the four files that make up the component.
|
||||||
|
|
||||||
<!-- For more information
|
<!-- For more information
|
||||||
about components, see [Components](). -->
|
about components, see [Components](). -->
|
||||||
|
@ -92,8 +93,8 @@ ng generate component orders/order-list
|
||||||
## Set up the UI
|
## Set up the UI
|
||||||
|
|
||||||
Though you can type the URL into the address bar, a nav
|
Though you can type the URL into the address bar, a nav
|
||||||
is easier for the user and more common. Replace the default
|
is easier for the user and more common. Replace the default
|
||||||
placeholder markup in `app.component.html` with a custom nav
|
placeholder markup in `app.component.html` with a custom nav
|
||||||
so you can easily navigate to your modules in the browser:
|
so you can easily navigate to your modules in the browser:
|
||||||
|
|
||||||
|
|
||||||
|
@ -224,7 +225,4 @@ knows that the route list is only responsible for providing additional routes an
|
||||||
You may also be interested in the following:
|
You may also be interested in the following:
|
||||||
* [Routing and Navigation](guide/router).
|
* [Routing and Navigation](guide/router).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [Types of NgModules](guide/module-types).
|
* [Types of Feature Modules](guide/module-types).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -62,7 +62,7 @@ typical characteristics, in real world apps, you may see hybrids.
|
||||||
|
|
||||||
A lazy-loaded routed feature module should not be imported by any module. Doing so would trigger an eager load, defeating the purpose of lazy loading.That means you won’t see them mentioned among the `AppModule` imports. An eager loaded routed feature module must be imported by another module so that the compiler learns about its components.
|
A lazy-loaded routed feature module should not be imported by any module. Doing so would trigger an eager load, defeating the purpose of lazy loading.That means you won’t see them mentioned among the `AppModule` imports. An eager loaded routed feature module must be imported by another module so that the compiler learns about its components.
|
||||||
|
|
||||||
Routed feature modules rarely have providers for reasons explained in Lazy Loading Feature Modules(page forthcoming). When they do, the lifetime of the provided services should be the same as the lifetime of the module. Don't provide application-wide singleton services in a routed feature module or in a module that the routed module imports.
|
Routed feature modules rarely have providers for reasons explained in [Lazy Loading Feature Modules](/guide/lazy-loading-ngmodules). When they do, the lifetime of the provided services should be the same as the lifetime of the module. Don't provide application-wide singleton services in a routed feature module or in a module that the routed module imports.
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
@ -92,7 +92,7 @@ typical characteristics, in real world apps, you may see hybrids.
|
||||||
<td>Service</td>
|
<td>Service</td>
|
||||||
<td>
|
<td>
|
||||||
|
|
||||||
Service modules provide utility services such as data access and messaging. Ideally, they consist entirely of providers and have no declarations. Angular's `HttpModule` is a good example of a service module.
|
Service modules provide utility services such as data access and messaging. Ideally, they consist entirely of providers and have no declarations. Angular's `HttpClientModule` is a good example of a service module.
|
||||||
|
|
||||||
The root `AppModule` is the only module that should import service modules.
|
The root `AppModule` is the only module that should import service modules.
|
||||||
|
|
||||||
|
@ -110,7 +110,7 @@ typical characteristics, in real world apps, you may see hybrids.
|
||||||
A widget module should rarely have providers.
|
A widget module should rarely have providers.
|
||||||
|
|
||||||
Import widget modules in any module whose component templates need the widgets.
|
Import widget modules in any module whose component templates need the widgets.
|
||||||
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
@ -189,4 +189,3 @@ The following table summarizes the key characteristics of each feature module gr
|
||||||
You may also be interested in the following:
|
You may also be interested in the following:
|
||||||
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
* [Lazy Loading Modules with the Angular Router](guide/lazy-loading-ngmodules).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [Types of NgModules](guide/module-types).
|
|
||||||
|
|
|
@ -19,213 +19,214 @@ into three categories:
|
||||||
* **Runtime:** Injector configuration via the `providers` array.
|
* **Runtime:** Injector configuration via the `providers` array.
|
||||||
* **Composability/Grouping:** Bringing NgModules together and making them available via the `imports` and `exports` arrays.
|
* **Composability/Grouping:** Bringing NgModules together and making them available via the `imports` and `exports` arrays.
|
||||||
|
|
||||||
```ts
|
```typescript
|
||||||
@NgModule({
|
@NgModule({
|
||||||
// Static, that is compiler configuration
|
// Static, that is compiler configuration
|
||||||
declarations: [], // Configure the selectors
|
declarations: [], // Configure the selectors
|
||||||
entryComponents: [], // Generate the host factory
|
entryComponents: [], // Generate the host factory
|
||||||
|
|
||||||
// Runtime, or injector configuration
|
// Runtime, or injector configuration
|
||||||
providers: [], // Runtime injector configuration
|
providers: [], // Runtime injector configuration
|
||||||
|
|
||||||
// Composability / Grouping
|
// Composability / Grouping
|
||||||
imports: [], // composing NgModules together
|
imports: [], // composing NgModules together
|
||||||
exports: [] // making NgModules available to other parts of the app
|
exports: [] // making NgModules available to other parts of the app
|
||||||
})
|
})
|
||||||
```
|
```
|
||||||
|
|
||||||
## `@NgModule` metadata
|
## `@NgModule` metadata
|
||||||
|
|
||||||
The following table summarizes the `NgModule` metadata properties.
|
The following table summarizes the `@NgModule` metadata properties.
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
|
|
||||||
<th>
|
<th>
|
||||||
Property
|
Property
|
||||||
</th>
|
</th>
|
||||||
|
|
||||||
<th>
|
<th>
|
||||||
Description
|
Description
|
||||||
</th>
|
</th>
|
||||||
|
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
<td style="vertical-align: top">
|
||||||
<code>declarations</code>
|
<code>declarations</code>
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td>
|
<td>
|
||||||
|
|
||||||
|
A list of [declarable](guide/ngmodule-faq#q-declarable) classes,
|
||||||
|
(*components*, *directives*, and *pipes*) that _belong to this module_.
|
||||||
|
|
||||||
A list of [declarable](guide/ngmodule-faq#q-declarable) classes,
|
<ol>
|
||||||
(*components*, *directives*, and *pipes*) that _belong to this module_.
|
<li>When compiling a template, you need to determine a set of selectors which should be used for triggering their corresponding directives.</li>
|
||||||
|
<li>
|
||||||
|
The template is compiled within the context of an NgModule—the NgModule within which the template's component is declared—which determines the set of selectors using the following rules:
|
||||||
|
<ul>
|
||||||
|
<li>All selectors of directives listed in `declarations`.</li>
|
||||||
|
<li>All selectors of directives exported from imported NgModules.</li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
</ol>
|
||||||
|
|
||||||
1) When compiling a template, you need to determine a set of selectors which should be used for triggering their corresponding directives.
|
Components, directives, and pipes must belong to _exactly_ one module.
|
||||||
2) The template is compiled within a context of an `NgModule`—the `NgModule` which this template's component is declared in—which determines the set of selectors using the following rules:
|
The compiler emits an error if you try to declare the same class in more than one module.
|
||||||
a) All selectors of directives listed in `declarations`
|
|
||||||
b) All exported selectors of imported `NgModules`.
|
|
||||||
|
|
||||||
|
Don't re-declare a class imported from another module.
|
||||||
|
|
||||||
Components, directives, and pipes must belong to _exactly_ one module.
|
</td>
|
||||||
The compiler emits an error if you try to declare the same class in more than one module.
|
|
||||||
|
|
||||||
Don't re-declare a class imported from another module.
|
</tr>
|
||||||
|
|
||||||
</td>
|
<tr>
|
||||||
|
|
||||||
</tr>
|
<td style="vertical-align: top">
|
||||||
|
<code>providers</code>
|
||||||
|
</td>
|
||||||
|
|
||||||
<tr>
|
<td>
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
A list of dependency-injection providers.
|
||||||
<code>providers</code>
|
|
||||||
</td>
|
|
||||||
|
|
||||||
<td>
|
Angular registers these providers with the NgModule's injector.
|
||||||
|
If it is the NgModule used for bootstrapping then it is the root injector.
|
||||||
|
|
||||||
|
These services become available for injection into any component, directive, pipe or service which is a child of this injector.
|
||||||
|
|
||||||
A list of dependency-injection providers.
|
A lazy-loaded module has its own injector which
|
||||||
|
is typically a child of the application root injector.
|
||||||
|
|
||||||
Angular registers these providers with the NgModule's injector.
|
Lazy-loaded services are scoped to the lazy module's injector.
|
||||||
If it is the NgModule used for bootstrapping that it is the root injector.
|
If a lazy-loaded module also provides the `UserService`,
|
||||||
|
any component created within that module's context (such as by router navigation)
|
||||||
|
gets the local instance of the service, not the instance in the root application injector.
|
||||||
|
|
||||||
These services become available for injection into any component, directive, pipe or service which is a child of this injector.
|
Components in external modules continue to receive the instance provided by their injectors.
|
||||||
|
|
||||||
A lazy-loaded module has its own injector which
|
For more information on injector hierarchy and scoping, see [Providers](guide/providers).
|
||||||
is typically a child of the application root injector.
|
|
||||||
|
|
||||||
Lazy-loaded services are scoped to the lazy module's injector.
|
</td>
|
||||||
If a lazy-loaded module also provides the `UserService`,
|
|
||||||
any component created within that module's context (such as by router navigation)
|
|
||||||
gets the local instance of the service, not the instance in the root application injector.
|
|
||||||
|
|
||||||
Components in external modules continue to receive the instance provided by their injectors.
|
</tr>
|
||||||
|
|
||||||
For more information on injector hierarchy and scoping, see [Providers](guide/providers).
|
<tr>
|
||||||
|
|
||||||
</td>
|
<td style="vertical-align: top">
|
||||||
|
<code>imports</code>
|
||||||
|
</td>
|
||||||
|
|
||||||
</tr>
|
<td>
|
||||||
|
|
||||||
<tr>
|
A list of modules which should be folded into this module. Folded means it is
|
||||||
|
as if all the imported NgModule's exported properties were declared here.
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
Specifically, it is as if the list of modules whose exported components, directives, or pipes
|
||||||
<code>imports</code>
|
are referenced by the component templates were declared in this module.
|
||||||
</td>
|
|
||||||
|
|
||||||
<td>
|
A component template can [reference](guide/ngmodule-faq#q-template-reference) another component, directive, or pipe
|
||||||
|
when the reference is declared in this module or if the imported module has exported it.
|
||||||
|
For example, a component can use the `NgIf` and `NgFor` directives only if the
|
||||||
|
module has imported the Angular `CommonModule` (perhaps indirectly by importing `BrowserModule`).
|
||||||
|
|
||||||
|
You can import many standard directives from the `CommonModule`
|
||||||
|
but some familiar directives belong to other modules.
|
||||||
|
For example, you can use `[(ngModel)]` only
|
||||||
|
after importing the Angular `FormsModule`.
|
||||||
|
|
||||||
A list of modules which should be folded into this module. Folded means it is
|
</td>
|
||||||
as if all of the imported NgModule properties were declared here.
|
|
||||||
|
|
||||||
Specifically, it is as if the list of modules whose exported components, directives, or pipes
|
</tr>
|
||||||
are referenced by the component templates were declared in this module.
|
|
||||||
|
|
||||||
A component template can [reference](guide/ngmodule-faq#q-template-reference) another component, directive, or pipe
|
<tr>
|
||||||
when the reference is declared in this module
|
|
||||||
or if the imported module has exported it.
|
|
||||||
For example, a component can use the `NgIf` and `NgFor` directives only if the
|
|
||||||
module has imported the Angular `CommonModule` (perhaps indirectly by importing `BrowserModule`).
|
|
||||||
|
|
||||||
You can import many standard directives from the `CommonModule`
|
<td style="vertical-align: top">
|
||||||
but some familiar directives belong to other modules.
|
<code>exports</code>
|
||||||
For example, you can use `[(ngModel)]` only
|
</td>
|
||||||
after importing the Angular `FormsModule`.
|
|
||||||
</td>
|
|
||||||
|
|
||||||
</tr>
|
<td>
|
||||||
|
|
||||||
<tr>
|
A list of declarations—*component*, *directive*, and *pipe* classes—that
|
||||||
|
an importing module can use.
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
Exported declarations are the module's _public API_.
|
||||||
<code>exports</code>
|
A component in another module can [use](guide/ngmodule-faq#q-template-reference) _this_
|
||||||
</td>
|
module's `UserComponent` if it imports this module and this module exports `UserComponent`.
|
||||||
|
|
||||||
<td>
|
Declarations are private by default.
|
||||||
|
If this module does _not_ export `UserComponent`, then only the components within _this_
|
||||||
|
module can use `UserComponent`.
|
||||||
|
|
||||||
|
Importing a module does _not_ automatically re-export the imported module's imports.
|
||||||
|
Module 'B' can't use `ngIf` just because it imported module 'A' which imported `CommonModule`.
|
||||||
|
Module 'B' must import `CommonModule` itself.
|
||||||
|
|
||||||
A list of declarations—*component*, *directive*, and *pipe* classes—that
|
A module can list another module among its `exports`, in which case
|
||||||
an importing module can use.
|
all of that module's public components, directives, and pipes are exported.
|
||||||
|
|
||||||
Exported declarations are the module's _public API_.
|
[Re-export](guide/ngmodule-faq#q-reexport) makes module transitivity explicit.
|
||||||
A component in another module can [use](guide/ngmodule-faq#q-template-reference) _this_ module's `UserComponent`
|
If Module 'A' re-exports `CommonModule` and Module 'B' imports Module 'A',
|
||||||
if it imports this module and this module exports `UserComponent`.
|
Module 'B' components can use `ngIf` even though 'B' itself didn't import `CommonModule`.
|
||||||
|
|
||||||
Declarations are private by default.
|
</td>
|
||||||
If this module does _not_ export `UserComponent`, than only the components within where the `UserComponent` has been declared can use `UserComponent.
|
|
||||||
|
|
||||||
Importing a module does _not_ automatically re-export the imported module's imports.
|
</tr>
|
||||||
Module 'B' can't use `ngIf` just because it imported module `A` which imported `CommonModule`.
|
|
||||||
Module 'B' must import `CommonModule` itself.
|
|
||||||
|
|
||||||
A module can list another module among its `exports`, in which case
|
<tr>
|
||||||
all of that module's public components, directives, and pipes are exported.
|
|
||||||
|
|
||||||
[Re-export](guide/ngmodule-faq#q-reexport) makes module transitivity explicit.
|
<td style="vertical-align: top">
|
||||||
If Module 'A' re-exports `CommonModule` and Module 'B' imports Module 'A',
|
<code>bootstrap</code>
|
||||||
Module 'B' components can use `ngIf` even though 'B' itself didn't import `CommonModule`.
|
</td>
|
||||||
|
|
||||||
</td>
|
<td>
|
||||||
|
|
||||||
</tr>
|
A list of components that are automatically bootstrapped.
|
||||||
|
|
||||||
<tr>
|
Usually there's only one component in this list, the _root component_ of the application.
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
Angular can launch with multiple bootstrap components,
|
||||||
<code>bootstrap</code>
|
each with its own location in the host web page.
|
||||||
</td>
|
|
||||||
|
|
||||||
<td>
|
A bootstrap component is automatically added to `entryComponents`.
|
||||||
|
|
||||||
|
</td>
|
||||||
|
|
||||||
A list of components that are automatically bootstrapped.
|
</tr>
|
||||||
|
|
||||||
Usually there's only one component in this list, the _root component_ of the application.
|
<tr>
|
||||||
|
|
||||||
Angular can launch with multiple bootstrap components,
|
<td style="vertical-align: top">
|
||||||
each with its own location in the host web page.
|
<code>entryComponents</code>
|
||||||
|
</td>
|
||||||
|
|
||||||
A bootstrap component is automatically added to `entryComponents`.
|
<td>
|
||||||
|
|
||||||
</td>
|
A list of components that can be dynamically loaded into the view.
|
||||||
|
|
||||||
</tr>
|
By default, an Angular app always has at least one entry component, the root component, `AppComponent`. Its purpose is to serve as a point of entry into the app, that is, you bootstrap it to launch the app.
|
||||||
|
|
||||||
<tr>
|
Routed components are also _entry components_ because they need to be loaded dynamically.
|
||||||
|
The router creates them and drops them into the DOM near a `<router-outlet>`.
|
||||||
|
|
||||||
<td style="vertical-align: top">
|
While the bootstrapped and routed components are _entry components_,
|
||||||
<code>entryComponents</code>
|
you don't have to add them to a module's `entryComponents` list,
|
||||||
</td>
|
as they are added implicitly.
|
||||||
|
|
||||||
<td>
|
Angular automatically adds components in the module's `bootstrap` and route definitions into the `entryComponents` list.
|
||||||
|
|
||||||
|
That leaves only components bootstrapped using one of the imperative techniques, such as [`ViewComponentRef.createComponent()`](https://angular.io/api/core/ViewContainerRef#createComponent) as undiscoverable.
|
||||||
|
|
||||||
A list of components that can be dynamically loaded into the view.
|
Dynamic component loading is not common in most apps beyond the router. If you need to dynamically load components, you must add these components to the `entryComponents` list yourself.
|
||||||
|
|
||||||
By default, an Angular app always has at least one entry component, the root component, `AppComponent`. Its purpose is to serve as a point of entry into the app, that is, you bootstrap it to launch the app.
|
For more information, see [Entry Components](guide/entry-components).
|
||||||
|
|
||||||
Routed components are also _entry components_ because they need to be loaded dynamically.
|
</td>
|
||||||
The router creates them and drops them into the DOM near a `<router-outlet>`.
|
|
||||||
|
|
||||||
While the bootstrapped and routed components are _entry components_,
|
</tr>
|
||||||
you don't have to add them to a module's `entryComponents` list,
|
|
||||||
as they are added implicitly.
|
|
||||||
|
|
||||||
Angular automatically adds components in the module's `bootstrap` and route definitions into the `entryComponents` list.
|
|
||||||
|
|
||||||
That leaves only components bootstrapped using one of the imperative techniques, such as [`ViewComponentRef.createComponent()`](https://angular.io/api/core/ViewContainerRef#createComponent) as undiscoverable.
|
|
||||||
|
|
||||||
Dynamic component loading is not common in most apps beyond the router. If you need to dynamically load components, you must add these components to the `entryComponents` list yourself.
|
|
||||||
|
|
||||||
For more information, see [Entry Components](guide/entry-components).
|
|
||||||
</td>
|
|
||||||
|
|
||||||
</tr>
|
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
@ -238,7 +239,4 @@ You may also be interested in the following:
|
||||||
* [Feature Modules](guide/feature-modules).
|
* [Feature Modules](guide/feature-modules).
|
||||||
* [Entry Components](guide/entry-components).
|
* [Entry Components](guide/entry-components).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [Types of NgModules](guide/module-types).
|
* [Types of Feature Modules](guide/module-types).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,7 @@ Do *not* declare the following:
|
||||||
|
|
||||||
* A class that's already declared in another module, whether an app module, @NgModule, or third-party module.
|
* A class that's already declared in another module, whether an app module, @NgModule, or third-party module.
|
||||||
* An array of directives imported from another module.
|
* An array of directives imported from another module.
|
||||||
For example, don't declare `FORMS_DIRECTIVES` from `@angular/forms` because the `FormsModule` already declares it.
|
For example, don't declare `FORMS_DIRECTIVES` from `@angular/forms` because the `FormsModule` already declares it.
|
||||||
|
|
||||||
* Module classes.
|
* Module classes.
|
||||||
* Service classes.
|
* Service classes.
|
||||||
|
@ -120,7 +120,7 @@ Importing `CommonModule` also frees feature modules for use on _any_ target plat
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-reimport"></a>
|
{@a q-reimport}
|
||||||
|
|
||||||
## What if I import the same module twice?
|
## What if I import the same module twice?
|
||||||
|
|
||||||
|
@ -136,13 +136,13 @@ Angular doesn't like NgModules with circular references, so don't let Module 'A'
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-reexport"></a>
|
{@a q-reexport}
|
||||||
|
|
||||||
## What should I export?
|
## What should I export?
|
||||||
|
|
||||||
Export [declarable](guide/bootstrapping#the-declarations-array) classes that components in _other_ NgModules
|
Export [declarable](guide/bootstrapping#the-declarations-array) classes that components in _other_ NgModules
|
||||||
are able to reference in their templates. These are your _public_ classes.
|
are able to reference in their templates. These are your _public_ classes.
|
||||||
If you don't export a class, it stays _private_, visible only to other component
|
If you don't export a declarable class, it stays _private_, visible only to other components
|
||||||
declared in this NgModule.
|
declared in this NgModule.
|
||||||
|
|
||||||
You _can_ export any declarable class—components, directives, and pipes—whether
|
You _can_ export any declarable class—components, directives, and pipes—whether
|
||||||
|
@ -164,7 +164,7 @@ If you don't want another NgModule to see it, don't export it.
|
||||||
Such [entry components](guide/ngmodule-faq#q-entry-component-defined) can never be selected in another component's template.
|
Such [entry components](guide/ngmodule-faq#q-entry-component-defined) can never be selected in another component's template.
|
||||||
While there's no harm in exporting them, there's also no benefit.
|
While there's no harm in exporting them, there's also no benefit.
|
||||||
* Pure service modules that don't have public (exported) declarations.
|
* Pure service modules that don't have public (exported) declarations.
|
||||||
For example, there's no point in re-exporting `HttpModule` because it doesn't export anything.
|
For example, there's no point in re-exporting `HttpClientModule` because it doesn't export anything.
|
||||||
It's only purpose is to add http service providers to the application as a whole.
|
It's only purpose is to add http service providers to the application as a whole.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
@ -183,14 +183,13 @@ Angular's own `BrowserModule` exports a couple of NgModules like this:
|
||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
exports: [CommonModule, ApplicationModule]
|
exports: [CommonModule, ApplicationModule]
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
An NgModule can export a combination of its own declarations, selected imported classes, and imported NgModules.
|
An NgModule can export a combination of its own declarations, selected imported classes, and imported NgModules.
|
||||||
|
|
||||||
Don't bother re-exporting pure service modules.
|
Don't bother re-exporting pure service modules.
|
||||||
Pure service modules don't export [declarable](guide/bootstrapping#the-declarations-array) classes that another NgModule could use.
|
Pure service modules don't export [declarable](guide/bootstrapping#the-declarations-array) classes that another NgModule could use.
|
||||||
For example, there's no point in re-exporting `HttpModule` because it doesn't export anything.
|
For example, there's no point in re-exporting `HttpClientModule` because it doesn't export anything.
|
||||||
It's only purpose is to add http service providers to the application as a whole.
|
It's only purpose is to add http service providers to the application as a whole.
|
||||||
|
|
||||||
|
|
||||||
|
@ -209,6 +208,7 @@ You add that result to the `imports` list of the root `AppModule`.
|
||||||
Only call and import a `.forRoot()` result in the root application module, `AppModule`.
|
Only call and import a `.forRoot()` result in the root application module, `AppModule`.
|
||||||
Importing it in any other module, particularly in a lazy-loaded module,
|
Importing it in any other module, particularly in a lazy-loaded module,
|
||||||
is contrary to the intent and will likely produce a runtime error.
|
is contrary to the intent and will likely produce a runtime error.
|
||||||
|
For more information, see [Singleton Services](guide/singleton-services).
|
||||||
|
|
||||||
`RouterModule` also offers a `forChild` static method for configuring the routes of lazy-loaded modules.
|
`RouterModule` also offers a `forChild` static method for configuring the routes of lazy-loaded modules.
|
||||||
|
|
||||||
|
@ -237,7 +237,7 @@ This is by design.
|
||||||
Extensibility through NgModule imports is a primary goal of the NgModule system.
|
Extensibility through NgModule imports is a primary goal of the NgModule system.
|
||||||
Merging NgModule providers into the application injector
|
Merging NgModule providers into the application injector
|
||||||
makes it easy for a module library to enrich the entire application with new services.
|
makes it easy for a module library to enrich the entire application with new services.
|
||||||
By adding the `HttpModule` once, every application component can make HTTP requests.
|
By adding the `HttpClientModule` once, every application component can make HTTP requests.
|
||||||
|
|
||||||
However, this might feel like an unwelcome surprise if you expect the module's services
|
However, this might feel like an unwelcome surprise if you expect the module's services
|
||||||
to be visible only to the components declared by that feature module.
|
to be visible only to the components declared by that feature module.
|
||||||
|
@ -247,9 +247,7 @@ not just the classes declared in the `HeroModule`.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-lazy-loaded-module-provider-visibility"></a>
|
{@a q-lazy-loaded-module-provider-visibility}
|
||||||
{@ q-lazy-loaded-module-provider-visibility}
|
|
||||||
|
|
||||||
|
|
||||||
## Why is a service provided in a lazy-loaded module visible only to that module?
|
## Why is a service provided in a lazy-loaded module visible only to that module?
|
||||||
|
|
||||||
|
@ -303,10 +301,10 @@ That's also usually the best place to configure, wrap, and override them.
|
||||||
|
|
||||||
Suppose a module requires a customized `HttpBackend` that adds a special header for all Http requests.
|
Suppose a module requires a customized `HttpBackend` that adds a special header for all Http requests.
|
||||||
If another module elsewhere in the application also customizes `HttpBackend`
|
If another module elsewhere in the application also customizes `HttpBackend`
|
||||||
or merely imports the `HttpModule`, it could override this module's `HttpBackend` provider,
|
or merely imports the `HttpClientModule`, it could override this module's `HttpBackend` provider,
|
||||||
losing the special header. The server will reject http requests from this module.
|
losing the special header. The server will reject http requests from this module.
|
||||||
|
|
||||||
To avoid this problem, import the `HttpModule` only in the `AppModule`, the application _root module_.
|
To avoid this problem, import the `HttpClientModule` only in the `AppModule`, the application _root module_.
|
||||||
|
|
||||||
If you must guard against this kind of "provider corruption", *don't rely on a launch-time module's `providers`.*
|
If you must guard against this kind of "provider corruption", *don't rely on a launch-time module's `providers`.*
|
||||||
|
|
||||||
|
@ -337,9 +335,8 @@ Define child routes and let the router load module components into that outlet.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-root-component-or-module"></a>
|
{@a q-root-component-or-module}
|
||||||
|
|
||||||
{@ q-root-component-or-module}
|
|
||||||
|
|
||||||
## Should I add application-wide providers to the root `AppModule` or the root `AppComponent`?
|
## Should I add application-wide providers to the root `AppModule` or the root `AppComponent`?
|
||||||
|
|
||||||
|
@ -356,7 +353,7 @@ More generally, [prefer registering providers in NgModules](guide/ngmodule-faq#q
|
||||||
<h3 class="no-toc">Discussion</h3>
|
<h3 class="no-toc">Discussion</h3>
|
||||||
|
|
||||||
Angular registers all startup module providers with the application root injector.
|
Angular registers all startup module providers with the application root injector.
|
||||||
The services that root injector providers create have application scope, which
|
The services that root injector providers create have application scope, which
|
||||||
means they are available to the entire application.
|
means they are available to the entire application.
|
||||||
|
|
||||||
Certain services, such as the `Router`, only work when you register them in the application root injector.
|
Certain services, such as the `Router`, only work when you register them in the application root injector.
|
||||||
|
@ -367,14 +364,13 @@ They have component scope.
|
||||||
|
|
||||||
The `AppComponent`'s injector is a child of the root injector, one down in the injector hierarchy.
|
The `AppComponent`'s injector is a child of the root injector, one down in the injector hierarchy.
|
||||||
For applications that don't use the router, that's almost the entire application.
|
For applications that don't use the router, that's almost the entire application.
|
||||||
But in routed applications, routing operates at the root level
|
But in routed applications, routing operates at the root level
|
||||||
where `AppComponent` services don't exist.
|
where `AppComponent` services don't exist.
|
||||||
This means that lazy-loaded modules can't reach them.
|
This means that lazy-loaded modules can't reach them.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
<a id="q-component-or-module"></a>
|
|
||||||
|
|
||||||
{@ q-component-or-module}
|
{@a q-component-or-module}
|
||||||
|
|
||||||
## Should I add other providers to a module or a component?
|
## Should I add other providers to a module or a component?
|
||||||
|
|
||||||
|
@ -395,15 +391,14 @@ not the root `AppComponent`.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-why-bad"></a>
|
{@a q-why-bad}
|
||||||
|
|
||||||
{@ q-why-bad}
|
|
||||||
|
|
||||||
## Why is it bad if a shared module provides a service to a lazy-loaded module?
|
## Why is it bad if a shared module provides a service to a lazy-loaded module?
|
||||||
|
|
||||||
### The eagerly loaded scenario
|
### The eagerly loaded scenario
|
||||||
When an eagerly loaded module provides a service, for example a `UserService`, that service is available application-wide. If the root module provides `UserService` and
|
When an eagerly loaded module provides a service, for example a `UserService`, that service is available application-wide. If the root module provides `UserService` and
|
||||||
imports another module that provides the same `UserService`, Angular registers one of
|
imports another module that provides the same `UserService`, Angular registers one of
|
||||||
them in the root app injector (see [What if I import the same module twice?](guide/ngmodule-faq#q-reimport)).
|
them in the root app injector (see [What if I import the same module twice?](guide/ngmodule-faq#q-reimport)).
|
||||||
|
|
||||||
Then, when some component injects `UserService`, Angular finds it in the app root injector,
|
Then, when some component injects `UserService`, Angular finds it in the app root injector,
|
||||||
|
@ -480,7 +475,7 @@ Here is a custom constructor for an NgModule called `CoreModule`.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
{@a q-entry-component-defined}
|
{@a q-entry-component-defined}
|
||||||
|
|
||||||
## What is an `entry component`?
|
## What is an `entry component`?
|
||||||
|
|
||||||
|
@ -488,7 +483,7 @@ An entry component is any component that Angular loads _imperatively_ by type.
|
||||||
|
|
||||||
A component loaded _declaratively_ via its selector is _not_ an entry component.
|
A component loaded _declaratively_ via its selector is _not_ an entry component.
|
||||||
|
|
||||||
Angular loads a component declaratively when
|
Angular loads a component declaratively when
|
||||||
using the component's selector to locate the element in the template.
|
using the component's selector to locate the element in the template.
|
||||||
Angular then creates the HTML representation of the component and inserts it into the DOM at the selected element. These aren't entry components.
|
Angular then creates the HTML representation of the component and inserts it into the DOM at the selected element. These aren't entry components.
|
||||||
|
|
||||||
|
@ -509,13 +504,13 @@ For more information, see [Entry Components](guide/entry-components).
|
||||||
## What's the difference between a _bootstrap_ component and an _entry component_?
|
## What's the difference between a _bootstrap_ component and an _entry component_?
|
||||||
|
|
||||||
A bootstrapped component _is_ an [entry component](guide/ngmodule-faq#q-entry-component-defined)
|
A bootstrapped component _is_ an [entry component](guide/ngmodule-faq#q-entry-component-defined)
|
||||||
that Angular loads into the DOM during the bootstrap (application launch) process.
|
that Angular loads into the DOM during the bootstrap process (application launch).
|
||||||
Other entry components are loaded dynamically by other means, such as with the router.
|
Other entry components are loaded dynamically by other means, such as with the router.
|
||||||
|
|
||||||
The `@NgModule.bootstrap` property tells the compiler that this is an entry component _and_
|
The `@NgModule.bootstrap` property tells the compiler that this is an entry component _and_
|
||||||
it should generate code to bootstrap the application with this component.
|
it should generate code to bootstrap the application with this component.
|
||||||
|
|
||||||
There's no need to list a component in both the `bootstrap` and `entryComponent` lists,
|
There's no need to list a component in both the `bootstrap` and `entryComponents` lists,
|
||||||
although doing so is harmless.
|
although doing so is harmless.
|
||||||
|
|
||||||
For more information, see [Entry Components](guide/entry-components).
|
For more information, see [Entry Components](guide/entry-components).
|
||||||
|
@ -587,7 +582,7 @@ Import the `SharedModule` in your _feature_ modules,
|
||||||
both those loaded when the app starts and those you lazy load later.
|
both those loaded when the app starts and those you lazy load later.
|
||||||
|
|
||||||
### `CoreModule`
|
### `CoreModule`
|
||||||
`CoreModule` is a conventional name for an `NgModule` with `providers` for
|
`CoreModule` is a conventional name for an `NgModule` with `providers` for
|
||||||
the singleton services you load when the application starts.
|
the singleton services you load when the application starts.
|
||||||
|
|
||||||
Import `CoreModule` in the root `AppModule` only.
|
Import `CoreModule` in the root `AppModule` only.
|
||||||
|
@ -596,20 +591,20 @@ Never import `CoreModule` in any other module.
|
||||||
Consider making `CoreModule` a pure services module
|
Consider making `CoreModule` a pure services module
|
||||||
with no `declarations`.
|
with no `declarations`.
|
||||||
|
|
||||||
For more information, see [Sharing NgModules](guide/sharing-ngmodules)
|
For more information, see [Sharing NgModules](guide/sharing-ngmodules)
|
||||||
and [Singleton Services](guide/singleton-services).
|
and [Singleton Services](guide/singleton-services).
|
||||||
|
|
||||||
### Feature Modules
|
### Feature Modules
|
||||||
|
|
||||||
Feature modules are modules you create around specific application business domains, user workflows, and utility collections. They support your app by containing a particular feature,
|
Feature modules are modules you create around specific application business domains, user workflows, and utility collections. They support your app by containing a particular feature,
|
||||||
such as routes, services, widgets, etc. To conceptualize what a feature module might be in your
|
such as routes, services, widgets, etc. To conceptualize what a feature module might be in your
|
||||||
app, consider that if you would put the files related to a certain functionality, like a search,
|
app, consider that if you would put the files related to a certain functionality, like a search,
|
||||||
in one folder, that the contents of that folder would be a feature module that you might call
|
in one folder, that the contents of that folder would be a feature module that you might call
|
||||||
your `SearchModule`. It would contain all of the components, routing, and templates that
|
your `SearchModule`. It would contain all of the components, routing, and templates that
|
||||||
would make up the search functionality.
|
would make up the search functionality.
|
||||||
|
|
||||||
For more information, see [Feature Modules](guide/feature-modules) and
|
For more information, see [Feature Modules](guide/feature-modules) and
|
||||||
[Module Types](guide/module-types)
|
[Module Types](guide/module-types)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -621,7 +616,7 @@ In modern JavaScript, every file is a module
|
||||||
(see the [Modules](http://exploringjs.com/es6/ch_modules.html) page of the Exploring ES6 website).
|
(see the [Modules](http://exploringjs.com/es6/ch_modules.html) page of the Exploring ES6 website).
|
||||||
Within each file you write an `export` statement to make parts of the module public.
|
Within each file you write an `export` statement to make parts of the module public.
|
||||||
|
|
||||||
An Angular NgModule is a class with the `@NgModule` decorator—JavaScript modules
|
An Angular NgModule is a class with the `@NgModule` decorator—JavaScript modules
|
||||||
don't have to have the `@NgModule` decorator. Angular's `NgModule` has `imports` and `exports` and they serve a similar purpose.
|
don't have to have the `@NgModule` decorator. Angular's `NgModule` has `imports` and `exports` and they serve a similar purpose.
|
||||||
|
|
||||||
You _import_ other NgModules so you can use their exported classes in component templates.
|
You _import_ other NgModules so you can use their exported classes in component templates.
|
||||||
|
@ -631,7 +626,7 @@ For more information, see [JavaScript Modules vs. NgModules](guide/ngmodule-vs-j
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-template-reference"></a>
|
{@a q-template-reference}
|
||||||
|
|
||||||
## How does Angular find components, directives, and pipes in a template?<br>What is a <i><b>template reference</b></i>?
|
## How does Angular find components, directives, and pipes in a template?<br>What is a <i><b>template reference</b></i>?
|
||||||
|
|
||||||
|
@ -647,9 +642,7 @@ or exported by a module that this module imports.
|
||||||
|
|
||||||
<hr/>
|
<hr/>
|
||||||
|
|
||||||
<a id="q-angular-compiler"></a>
|
{@a q-angular-compiler}
|
||||||
|
|
||||||
{@ q-angular-compiler}
|
|
||||||
|
|
||||||
## What is the Angular compiler?
|
## What is the Angular compiler?
|
||||||
|
|
||||||
|
@ -671,4 +664,3 @@ the Angular compiler incorporates them into compiled component code too.
|
||||||
|
|
||||||
`@NgModule` metadata tells the Angular compiler what components to compile for this module and
|
`@NgModule` metadata tells the Angular compiler what components to compile for this module and
|
||||||
how to link this module with other modules.
|
how to link this module with other modules.
|
||||||
|
|
||||||
|
|
|
@ -12,14 +12,14 @@ though they organize it differently, Angular apps rely on both.
|
||||||
|
|
||||||
In JavaScript, modules are individual files with JavaScript code in them. To make what’s in them available, you write an export statement, usually after the relevant code, like this:
|
In JavaScript, modules are individual files with JavaScript code in them. To make what’s in them available, you write an export statement, usually after the relevant code, like this:
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
export class AppComponent { ... }
|
export class AppComponent { ... }
|
||||||
```
|
```
|
||||||
|
|
||||||
Then, when you need that file’s code in another file, you import it like this:
|
Then, when you need that file’s code in another file, you import it like this:
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
import { AppComponent } from './app.component';
|
import { AppComponent } from './app.component';
|
||||||
```
|
```
|
||||||
|
|
||||||
JavaScript modules help you namespace, preventing accidental global variables.
|
JavaScript modules help you namespace, preventing accidental global variables.
|
||||||
|
@ -31,27 +31,23 @@ NgModules are classes decorated with `@NgModule`. The `@NgModule` decorator’s
|
||||||
|
|
||||||
The `AppModule` generated from the Angular CLI demonstrates both kinds of modules in action:
|
The `AppModule` generated from the Angular CLI demonstrates both kinds of modules in action:
|
||||||
|
|
||||||
```javascript
|
```typescript
|
||||||
/* These are JavaScript import statements. Angular doesn’t know anything about these. */
|
/* These are JavaScript import statements. Angular doesn’t know anything about these. */
|
||||||
import { BrowserModule } from '@angular/platform-browser';
|
import { BrowserModule } from '@angular/platform-browser';
|
||||||
import { NgModule } from '@angular/core';
|
import { NgModule } from '@angular/core';
|
||||||
import { FormsModule } from '@angular/forms';
|
|
||||||
import { HttpModule } from '@angular/http';
|
|
||||||
|
|
||||||
import { AppComponent } from './app.component';
|
import { AppComponent } from './app.component';
|
||||||
|
|
||||||
/* The @NgModule decorator lets Angular know that this is an NgModule. */
|
/* The @NgModule decorator lets Angular know that this is an NgModule. */
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [
|
declarations: [
|
||||||
AppComponent
|
AppComponent
|
||||||
],
|
],
|
||||||
imports: [ /* These are NgModule imports. */
|
imports: [ /* These are NgModule imports. */
|
||||||
BrowserModule,
|
BrowserModule
|
||||||
FormsModule,
|
],
|
||||||
HttpModule
|
providers: [],
|
||||||
],
|
bootstrap: [AppComponent]
|
||||||
providers: [],
|
|
||||||
bootstrap: [AppComponent]
|
|
||||||
})
|
})
|
||||||
export class AppModule { }
|
export class AppModule { }
|
||||||
```
|
```
|
||||||
|
@ -76,5 +72,3 @@ For more information on NgModules, see:
|
||||||
* [Bootstrapping](guide/bootstrapping).
|
* [Bootstrapping](guide/bootstrapping).
|
||||||
* [Frequently used modules](guide/frequent-ngmodules).
|
* [Frequently used modules](guide/frequent-ngmodules).
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -8,12 +8,12 @@ A basic understanding of the following concepts:
|
||||||
|
|
||||||
<hr>
|
<hr>
|
||||||
|
|
||||||
**NgModules** configure the injector and the compiler and help organize related things together
|
**NgModules** configure the injector and the compiler and help organize related things together.
|
||||||
|
|
||||||
An NgModule is a class marked by the `@NgModule` decorator.
|
An NgModule is a class marked by the `@NgModule` decorator.
|
||||||
`@NgModule` takes a metadata object that describes how to compile a component's templates and how to create an injector at runtime.
|
`@NgModule` takes a metadata object that describes how to compile a component's template and how to create an injector at runtime.
|
||||||
It identifies the module's own components, directives, and pipes,
|
It identifies the module's own components, directives, and pipes,
|
||||||
making some of them public, through the `exports` property public, so that external components can use them.
|
making some of them public, through the `exports` property, so that external components can use them.
|
||||||
`@NgModule` can also add service providers to the application dependency injectors.
|
`@NgModule` can also add service providers to the application dependency injectors.
|
||||||
|
|
||||||
For an example app showcasing all the techniques that NgModules related pages
|
For an example app showcasing all the techniques that NgModules related pages
|
||||||
|
@ -25,7 +25,7 @@ section.
|
||||||
|
|
||||||
Modules are a great way to organize an application and extend it with capabilities from external libraries.
|
Modules are a great way to organize an application and extend it with capabilities from external libraries.
|
||||||
|
|
||||||
Angular libraries are NgModules, such as `FormsModule`, `HttpModule`, and `RouterModule`.
|
Angular libraries are NgModules, such as `FormsModule`, `HttpClientModule`, and `RouterModule`.
|
||||||
Many third-party libraries are available as NgModules such as
|
Many third-party libraries are available as NgModules such as
|
||||||
<a href="https://material.angular.io/">Material Design</a>,
|
<a href="https://material.angular.io/">Material Design</a>,
|
||||||
<a href="http://ionicframework.com/">Ionic</a>, and
|
<a href="http://ionicframework.com/">Ionic</a>, and
|
||||||
|
|
|
@ -21,15 +21,17 @@ Consider the following module from an imaginary app:
|
||||||
|
|
||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
import { NgModule } from '@angular/core';
|
import { CommonModule } from '@angular/common';
|
||||||
import { CommonModule } from '@angular/common';
|
import { NgModule } from '@angular/core';
|
||||||
import { FormsModule } from '@angular/forms';
|
import { FormsModule } from '@angular/forms';
|
||||||
import { OrdersPipe } from './orders.pipe';
|
import { CustomerComponent } from './customer.component';
|
||||||
import { NewItemDirective } from './newitem.directive';
|
import { NewItemDirective } from './new-item.directive';
|
||||||
|
import { OrdersPipe } from './orders.pipe';
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [ CommonModule ],
|
imports: [ CommonModule ],
|
||||||
declarations: [ OrdersPipe, NewItemDirective ],
|
declarations: [ CustomerComponent, NewItemDirective, OrdersPipe ],
|
||||||
exports: [ OrdersPipe, NewItemDirective,
|
exports: [ CustomerComponent, NewItemDirective, OrdersPipe,
|
||||||
CommonModule, FormsModule ]
|
CommonModule, FormsModule ]
|
||||||
})
|
})
|
||||||
export class SharedModule { }
|
export class SharedModule { }
|
||||||
|
@ -56,9 +58,9 @@ having to import it directly into the `@NgModule` decorator.
|
||||||
|
|
||||||
There is an important distinction between using another module's component and
|
There is an important distinction between using another module's component and
|
||||||
using a service from another module. Import modules when you want to use
|
using a service from another module. Import modules when you want to use
|
||||||
directives, pipes, and components. Importing a module with services means that you will have a new instance of that service, which typically is not what you need, (typically one wants to reuse an existing service.) Use module imports to control service instantiation.
|
directives, pipes, and components. Importing a module with services means that you will have a new instance of that service, which typically is not what you need (typically one wants to reuse an existing service). Use module imports to control service instantiation.
|
||||||
|
|
||||||
The most common way to get a hold of sharedservices is through Angular
|
The most common way to get a hold of shared services is through Angular
|
||||||
[dependency injection](guide/dependency-injection), rather than through the module system (importing a module will result in a new service instance, which is not a typical usage).
|
[dependency injection](guide/dependency-injection), rather than through the module system (importing a module will result in a new service instance, which is not a typical usage).
|
||||||
|
|
||||||
To read about sharing services, see [Providers](guide/providers).
|
To read about sharing services, see [Providers](guide/providers).
|
||||||
|
@ -70,7 +72,4 @@ To read about sharing services, see [Providers](guide/providers).
|
||||||
|
|
||||||
You may also be interested in the following:
|
You may also be interested in the following:
|
||||||
* [Providers](guide/providers).
|
* [Providers](guide/providers).
|
||||||
* [Types of Modules](guide/module-types).
|
* [Types of Feature Modules](guide/module-types).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,13 @@ showcasing all the documented features of NgModules.
|
||||||
|
|
||||||
## Providing a singleton service
|
## Providing a singleton service
|
||||||
|
|
||||||
An injector created from a module definition will have services which are singletons with respect to that injector. To control the lifetime of services, one controls the creation and destruction of injectors. For example, a route will have an associated module. When the route is activated, an injector is created from that module as a child of the current injector. When you navigate away from the route, the injector is destroyed. This means that services declared in a route module will have a lifetime equal to that of the route. Similarly, services provided in an application module will have the same lifetime of the application, hence singleton.
|
An injector created from a module definition will have services which are singletons with respect to
|
||||||
|
that injector. To control the lifetime of services, one controls the creation and destruction of
|
||||||
|
injectors. For example, a route will have an associated module. When the route is activated, an
|
||||||
|
injector is created from that module as a child of the current injector. When you navigate away from
|
||||||
|
the route, the injector is destroyed. This means that services declared in a route module will have
|
||||||
|
a lifetime equal to that of the route. Similarly, services provided in an application module will
|
||||||
|
have the same lifetime of the application, hence singleton.
|
||||||
|
|
||||||
The following example module is called, as a convention, `CoreModule`. This use of `@NgModule` creates organizational infrastructure and gives you
|
The following example module is called, as a convention, `CoreModule`. This use of `@NgModule` creates organizational infrastructure and gives you
|
||||||
a way of providing services from a designated NgModule.
|
a way of providing services from a designated NgModule.
|
||||||
|
@ -47,8 +53,8 @@ As a general rule, import modules with providers _exactly once_,
|
||||||
preferably in the application's _root module_.
|
preferably in the application's _root module_.
|
||||||
That's also usually the best place to configure, wrap, and override them.
|
That's also usually the best place to configure, wrap, and override them.
|
||||||
|
|
||||||
For more detailed information on services, see
|
For more detailed information on services, see the [Services](tutorial/toh-pt4) chapter of the
|
||||||
[part 5](tutorial/toh-pt4) of the [Tour of Heroes tutorial](tutorial).
|
[Tour of Heroes tutorial](tutorial).
|
||||||
|
|
||||||
|
|
||||||
## `forRoot()`
|
## `forRoot()`
|
||||||
|
@ -171,9 +177,6 @@ Here are the two files in their entirety for reference:
|
||||||
## More on NgModules
|
## More on NgModules
|
||||||
|
|
||||||
You may also be interested in:
|
You may also be interested in:
|
||||||
* [Sharing NgModules](guide/singleton-services), which elaborates on the concepts covered on this page.
|
* [Sharing Modules](guide/sharing-ngmodules), which elaborates on the concepts covered on this page.
|
||||||
* [Lazy Loading Modules](guide/lazy-loading-ngmodules).
|
* [Lazy Loading Modules](guide/lazy-loading-ngmodules).
|
||||||
* [NgModule FAQ](guide/ngmodule-faq).
|
* [NgModule FAQ](guide/ngmodule-faq).
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -240,8 +240,8 @@
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"url": "guide/module-types",
|
"url": "guide/module-types",
|
||||||
"title": "Types of NgModules",
|
"title": "Types of Feature Modules",
|
||||||
"tooltip": "Description of the different types of feature module."
|
"tooltip": "Description of the different types of feature modules."
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"url": "guide/entry-components",
|
"url": "guide/entry-components",
|
||||||
|
|
Loading…
Reference in New Issue