2017-05-08 23:47:34 -04:00
# NgModules
2017-03-31 19:57:13 -04:00
2017-03-06 05:43:33 -05:00
**NgModules** help organize an application into cohesive blocks of functionality.
<!-- CF: "app" and "application" are used interchangeably throughout this page.
I'm not sure what's appropriate, so I left them as is for now. -->
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
An NgModule is a class adorned with the ** @NgModule ** decorator function.
`@NgModule` takes a metadata object that tells Angular how to compile and your code.
2017-03-06 05:43:33 -05:00
It identifies the module's own components, directives, and pipes,
2017-02-22 13:09:39 -05:00
making some of them public so external components can use them.
2017-03-06 05:43:33 -05:00
`@NgModule` may add service providers to the application dependency injectors.
2017-02-22 13:09:39 -05:00
And there are many more options covered here.
2017-10-17 20:36:22 -04:00
{@a bootstrap}
For a quick overview of NgModules, consider reading the
[Bootstrapping ](guide/bootstrapping ) guide, which introduces NgModules and the essentials
2017-03-06 05:43:33 -05:00
of creating and maintaining a single root `AppModule` for the entire application.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
_This_ page covers NgModules in greater depth.
2017-02-22 13:09:39 -05:00
2017-05-08 23:47:34 -04:00
#### Live examples
2017-10-17 20:36:22 -04:00
This page explains NgModules through a progression of improvements to a sample with a "Heroes" theme.
2017-03-06 05:43:33 -05:00
Here's an index to live examples at key moments in the evolution of the sample:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* < live-example plnkr = "minimal.0" > The initial app</ live-example >
2017-02-22 13:09:39 -05:00
* < live-example plnkr = "contact.1b" > The first contact module</ live-example >
* < live-example plnkr = "contact.2" > The revised contact module</ live-example >
2017-03-31 19:57:13 -04:00
* < live-example plnkr = "pre-shared.3" > Just before adding SharedModule</ live-example >
2017-02-22 13:09:39 -05:00
* < live-example > The final version</ live-example >
2017-05-08 23:47:34 -04:00
#### Frequently asked questions (FAQs)
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
This page covers NgModule concepts in a tutorial fashion.
2017-02-22 13:09:39 -05:00
2017-04-30 16:10:32 -04:00
The companion [NgModule FAQs ](guide/ngmodule-faq "NgModule FAQs" ) guide
2017-03-06 05:43:33 -05:00
offers answers to specific design and implementation questions.
Read this page before reading those FAQs.
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< hr / >
2017-02-22 13:09:39 -05:00
{@a angular-modularity}
2017-03-06 05:43:33 -05:00
## Angular modularity
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
NgModules are a great way to organize an application and extend it with capabilities from external libraries.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Many Angular libraries are NgModules (such as `FormsModule` , `HttpModule` , and `RouterModule` ).
2017-03-06 05:43:33 -05:00
Many third-party libraries are available as NgModules (such as
2017-04-24 14:23:45 -04:00
< a href = "https://material.angular.io/" > Material Design< / a > ,
< a href = "http://ionicframework.com/" > Ionic< / a > ,
< a href = "https://github.com/angular/angularfire2" > AngularFire2< / a > ).
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
NgModules consolidate components, directives, and pipes into
2017-02-22 13:09:39 -05:00
cohesive blocks of functionality, each focused on a
feature area, application business domain, workflow, or common collection of utilities.
2017-10-17 20:36:22 -04:00
NgModules can also add services to the application.
2017-03-06 05:43:33 -05:00
Such services might be internally developed, such as the application logger.
Services can come from outside sources, such as the Angular router and Http client.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
NgModules can be loaded eagerly when the application starts.
They can also be _lazy-loaded_ asynchronously by the router.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
An NgModule is a class decorated with `@NgModule` metadata.
By setting metadata properties you tell Angular how your application parts fit together.
For example, you can do the following:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* _Declare_ which components, directives, and pipes belong to the NgModule.
* _Export_ some of those classes so that other component templates can use them.
* _Import_ other NgModules with the components, directives, and pipes needed by the components in _this_ NgModule.
* _Provide_ services at the application level that any application component can use.
* _Bootstrap_ the app with one or more top-level, _root_ components.
2017-02-22 13:09:39 -05:00
{@a root-module}
2017-05-08 23:47:34 -04:00
## The root _AppModule_
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Every Angular app has at least one NgModule class, the _root module_ .
You bootstrap _that_ NgModule to launch the application.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
By convention, the *root module* class is called `AppModule` and it exists in a file named `app.module.ts` .
The [**Angular CLI** ](https://cli.angular.io/ ) generates the initial `AppModule` for you when you create a project.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng new quickstart
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The root `AppModule` is all you need in a simple application with a few components.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
As the app grows, you may refactor the root `AppModule` into [*feature modules* ](#feature-modules )
that represent collections of related functionality.
For now, stick with the root `AppModule` created by the CLI.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "cli-quickstart/src/app/app.module.ts" title = "src/app/app.module.ts (CLI generated)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The initial `declarations` array identifies the application's only component, `AppComponent` ,
the _root component_ at the top of the app's component tree.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Soon you'll declare more [components ](#declare-component )
(and [directives ](#declare-directive ) and [pipes ](#declare-pipe ) too).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `@NgModule` metadata `imports` a single helper module, `BrowserModule` , which every browser app must import.
`BrowserModule` registers critical application service providers.
It also includes common directives like `NgIf` and `NgFor` , which become immediately visible and usable
in any of this NgModule's component templates.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `providers` array registers services with the top-level [_dependency injector_ ](guide/dependency-injection ).
There are no services to register ... yet.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Lastly, the `bootstrap` list identifies the `AppComponent` as the _bootstrap component_ .
When Angular launches the app, it renders the `AppComponent`
inside the `<app-root>` element tag of the `index.html` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Learn about that in the [bootstrapping ](guide/bootstrapping ) guide.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The CLI-generated `AppComponent` in this guide's sample has been simplified and consolidated into a single `app.component.ts` file like this:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.component.0.ts" title = "src/app/app.component.ts" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Run the app and follow along with the steps in this guide:
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng serve
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< hr / >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
{@a declarations}
{@a declare-directive}
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
## Declare directives
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a declarables}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
As the app evolves, you'll add directives, components, and pipes (the _declarables_ ).
You must declare each of these classes in an NgModule.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
As an exercise, begin by adding a `highlight.directive.ts` to the `src/app/` folder _by hand_ .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/highlight.directive.ts" title = "src/app/highlight.directive.ts" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `HighlightDirective` is an [attribute directive ](guide/attribute-directives )
that sets the background color of its host element.
Update the `AppComponent` template to attach this directive to the `<h1>` title element:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.component.1.ts" region = "template" title = "src/app/app.component.ts (template)" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The screen of the running app has not changed.
The `<h1>` is not highlighted.
Angular does not yet recognize the `highlight` attribute and is ignoring it.
You must declare the `HighlightDirective` in `AppModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Edit the `app.module.ts` file, import the `HighlightDirective` ,
and add it to the `AppModule` _declarations_ like this:
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1.ts" region = "directive" title = "src/app/app.module.ts (declarations)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The Angular CLI would have done all of this for you
if you'd created the `HighlightDirective` with the CLI command like this:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate directive highlight
< / code-example >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
But you didn't.
You created the file by hand so you must declare the directive by hand.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a declare-component}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
## Declare components
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Now add a `TitleComponent` to the app and this time create it with the CLI.
2017-03-30 15:04:18 -04:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate component title --flat --no-spec --inline-style
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `--flat` flag tells the CLI to generate all files to the `src/app/` folder.< br >
The `--no-spec` flag skips the test (`.spec`) file.< br >
The `--inline-style` flag prevents generation of the `.css` file (which you won't need).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "alert is-important" >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
To see which files would be created or changed by any `ng generate` command, append the `--dryRun` flag (`-d` for short).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Open the `AppModule` and look at the `declarations` where you will see that the CLI added
the `TitleComponent` for you.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1.ts" region = "component" title = "src/app/app.module.ts (declarations)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Now rewrite the `title.component.html` like this.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/title.component.html" region = "v1" title = "src/app/title.component.html" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
And move the `title` property from `app.component.ts` into the `title.component.ts` , which looks as follows after a little cleanup.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/title.component.ts" region = "v1" title = "src/app/title.component.ts" linenums = "false" >
< / code-example >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Rewrite `AppComponent` to display the new `TitleComponent` in the `<app-title>` element and get rid of the `title` property.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.component.1.ts" title = "src/app/app.component.ts (v1)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
#### Error if component not declared
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
There was no visible clue when you neglected to declare the `HighlightDirective` attribute directive.
The Angular compiler doesn't recognize `highlight` as an `<h1>` attribute but it doesn't complain either.
You'd discover it was undeclared only if you were looking for its effect.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Now try removing the declaration of the `TitleComponent` from `AppModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The Angular compiler behaves differently when it encounters an unrecognized HTML element.
The app ceases to display the page and
the browser console logs the following error
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
Uncaught Error: Template parse errors:
'app-title' is not a known element:
1. If 'app-title' is an Angular component, then verify that it is part of this NgModule.
2. If 'app-title' is a Web Component then add 'CUSTOM_ELEMENTS_SCHEMA' to the '@NgModule.schemas' of this component to suppress this message.
< / code-example >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
If you don't get that error, you might get this one:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
Uncaught Error: Component TitleComponent is not part of any NgModule or the module has not been imported into your module.
< / code-example >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< / div >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
**Always declare your [components ](#declare-component ), [directives ](#declare-directive ), and [pipes ](#declare-pipe )**.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a providers}
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
## Service providers
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The [Dependency Injection ](guide/dependency-injection ) page describes
the Angular hierarchical dependency-injection system and how to configure that system
with [providers ](guide/dependency-injection#providers ).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### NgModule providers
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
An NgModule can provide services.
A single instance of each provided service becomes available for injection into every class created with that NgModule's injector (or one of its descendant injectors).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
When Angular boots the application,
it creates the root `AppModule` with a root dependency injector.
Angular configures the root injector with the providers specified in the module's `@NgModule.providers` .
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
Later, when Angular creates a new instance of a class— be it a component, directive, service, or module— that new class can be injected with an instance of a service provided to the root injector by the `AppModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Angular also configures the root injector with the providers specified by [imported NgModules ](#imports ).
An NgModule's own providers are registered _after_ imported NgModule providers.
When there are multiple providers for the same injection token, the last registration wins.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Compared to Component providers
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Providing a service in `@Component.providers` metadata means that a new service instance will be created for each new instance of _that_ component and will be available for injection into _all of that component instance's descendant sub-components_ .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The service instance won't be injected into any other component instances.
Other instances of the same component class cannot see it.
Sibling and ancestor component instances cannot see it.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Component providers always supersede NgModule providers.
A component provider for injection token `X` creates a new service instance that "shadows" an NgModule provider for injection token `X` .
When the component or any of its sub-components inject `X` , they get the _component_ service instance, not the _NgModule_ service instance.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Should you provide a service in an _NgModule_ or a _component_ ?
The answer depends on how you want to scope the service.
If the service should be widely available, provide it in an NgModule.
If it should be visible only within a component tree, provide it in the component at the root of that tree.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### NgModule provider example
2017-02-22 13:09:39 -05:00
Many applications capture information about the currently logged-in user and make that information
accessible through a user service.
2017-10-17 20:36:22 -04:00
Use the CLI to create a `UserService` and provide it in the root `AppModule` .
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate service user --module=app
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
This command creates a skeleton `UserService` in `src/app/user.service.ts` and a companion test file, `src/app/user.service.spec.ts` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `--module=app` flag tells the CLI to provide the service class in the NgModule defined in the `src/app/app.module.ts` file.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
If you omit the `--module` flag, the CLI still creates the service but _does not provide it_ anywhere.
You have to do that yourself.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Confirm that the `--module=app` flag did provide the service in the root `AppModule` by inspecting the `@NgModule.providers` array in `src/app/app.module.ts`
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1.ts" region = "providers" title = "src/app/app.module.ts (providers)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Replace the generated contents of `src/app/user.service.ts` with the following dummy implementation.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/user.service.ts" title = "src/app/user.service.ts" linenums = "false" >
< / code-example >
2017-03-31 19:57:13 -04:00
2017-02-22 13:09:39 -05:00
Update the `TitleComponent` class with a constructor that injects the `UserService`
and sets the component's `user` property from the service.
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/title.component.ts" title = "src/app/title.component.ts" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Update the `TitleComponent` template to show the welcome message below the application title.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/title.component.html" title = "src/app/title.component.html" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
{@a imports}
2017-10-17 20:36:22 -04:00
## NgModule imports
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
In the revised `TitleComponent` , an `*ngIf` directive guards the message.
2017-02-22 13:09:39 -05:00
There is no message if there is no user.
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/title.component.html" region = "ngIf" title = "src/app/title.component.html (ngIf)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Although `AppModule` doesn't declare the `NgIf` directive, the application still compiles and runs.
2017-02-22 13:09:39 -05:00
How can that be? The Angular compiler should either ignore or complain about unrecognized HTML.
2017-10-17 20:36:22 -04:00
### Importing _BrowserModule_
Angular does recognize `NgIf` because the `AppModule` imports it indirectly
when it imports `BrowserModule` .
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/app.module.0.ts" region = "imports" title = "src/app/app.module.ts (imports)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Importing `BrowserModule` made all of its public components, directives, and pipes visible
2017-10-17 20:36:22 -04:00
to the templates of components declared in `AppModule` , which include `TitleComponent` .
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
{@a reexport}
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### Re-exported NgModules
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `NgIf` directive isn't declared in `BrowserModule` .
It's declared in `CommonModule` from `@angular/common` .
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
`CommonModule` contributes many of the common directives that applications need, including `ngIf` and `ngFor` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
`AppModule` doesn't import `CommonModule` directly.
But it benefits from the fact that `BrowserModule` imports `CommonModule`
**and [re-exports ](guide/ngmodule-faq#q-re-export ) it**.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The net effect is that an importer of `BrowserModule` gets `CommonModule` directives automatically as if it had declared them itself.
2017-03-31 19:57:13 -04:00
2017-03-27 11:08:53 -04:00
Many familiar Angular directives don't belong to `CommonModule` .
2017-10-17 20:36:22 -04:00
For example, `NgModel` and `RouterLink` belong to Angular's `FormsModule` and `RouterModule` respectively.
You must import those NgModules before you can use their directives.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
To illustrate this point, you'll extend the sample app with _contact editor_ whose `ContactComponent` is a form component.
You'll have to import form support from the Angular `FormsModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
{@a add-contact-editor}
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### Add a _contact editor_
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Imagine that you added the following _contact editor_ files
to the project by hand _without the help of the CLI_ .
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-tabs >
< code-pane title = "src/app/contact/contact.component.html" path = "ngmodule/src/app/contact/contact.component.html" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/contact/contact.component.ts" path = "ngmodule/src/app/contact/contact.component.3.ts" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/contact/contact.component.css" path = "ngmodule/src/app/contact/contact.component.css" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/contact/contact.service.ts" path = "ngmodule/src/app/contact/contact.service.ts" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/contact/awesome.pipe.ts" path = "ngmodule/src/app/contact/awesome.pipe.ts" >
< / code-pane >
2017-10-17 20:36:22 -04:00
< code-pane title = "src/app/contact/contact-highlight.directive.ts" path = "ngmodule/src/app/contact/contact-highlight.directive.ts" >
2017-03-27 11:08:53 -04:00
< / code-pane >
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Form components are often complex and this is one is no exception.
To make it manageable, all contact-related files are in an `src/app/contact` folder.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `ContactComponent` implementation is spread over three constituent HTML, TypeScript, and css files.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
There's a [custom pipe ](guide/pipes#custom-pipes ) (called `Awesome` ),
a `ContactHighlightDirective` , and a `ContactService` for fetching contacts.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactService` was added to the `AppModule` providers.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1b.ts" region = "providers" title = "src/app/app.module.ts (providers)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Now any class can inject the application-wide instances of the `ContactService` and `UserService` .
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Import supporting _FormsModule_
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactComponent` is written with Angular forms in the [template-driven ](guide/forms#template-driven ) style.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Notice the `[(ngModel)]` binding in the middle of the component template, `contact.component.html` .
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/contact/contact.component.html" region = "ngModel" title = "app/contact/contact.component.html (ngModel)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Two-way data binding `[(ngModel)]` is typical of the _template-driven_ style.
The `ngModel` is the selector for the `NgModel` directive.
Although `NgModel` is an Angular directive,
the _Angular compiler_ won't recognize it for two reasons:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
1. `AppModule` doesn't declare `NgModel` (and shouldn't).
2. `NgModel` wasn't imported via `BrowserModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
`ContactComponent` wouldn't behave like an Angular form anyway because
form features such as validation aren't part of the Angular core.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
To correct these problems, the `AppModule` must import _both_ the `BrowserModule`
_and_ the **FormsModule from '@angular/forms'** like this.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1.ts" region = "imports" title = "src/app/app.module.ts" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
You can write Angular form components in template-driven or [reactive ](guide/reactive-forms ) style.
NgModules with components written in the _reactive_ style
import the `ReactiveFormsModule` .
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
Now `[(ngModel)]` binding will work and the user input will be validated by Angular forms,
once you [declare the new component, pipe, and directive ](#declare-pipe ).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### Never re-declare
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Importing the `FormsModule` makes the `NgModelDirective` (and all of the other `FORMS_DIRECTIVES` ) available to components declared in `AppModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
*Do not also* add these directives to the `AppModule` metadata's declarations.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "alert is-critical" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
**Never re-declare classes that belong to another NgModule.**
Components, directives, and pipes should be declared in _exactly one NgModule_ .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / div >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
{@a declare-pipe}
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
## Declare pipes
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The revised application still won't compile until you declare the contact component, directive, and pipe.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Components and directives are *declarables* . So are **pipes** .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
You [learned earlier ](#declarations ) to generate and declare both components and directives with the CLI `ng generate` commands.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
There's also a CLI command to generate and declare the `AwesomePipe` :
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate pipe awesome
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
However, if you write these class files by hand or opt-out of declaration with the `--skip-import` flag, you'll have to add the declarations yourself.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
[You were told ](#add-contact-editor ) to add the _contact editor_ files by hand, so
you must manually update the `declarations` in the `AppModule` :
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.1.ts" region = "declarations" title = "src/app/app.module.ts (declarations)" linenums = "false" >
< / code-example >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< h3 class = "no-toc" > Display the < i > ContactComponent< / i > < / h3 >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
Update the `AppComponent` template to display the `ContactComponent` by placing an
element with its selector (`< app-contact > `) just below the title.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.component.1b.ts" region = "template" title = "src/app/app.component.ts (template)" linenums = "false" >
< / code-example >
2017-03-31 19:57:13 -04:00
2017-05-08 23:47:34 -04:00
< h3 class = "no-toc" > Run the app< / h3 >
2017-03-06 05:43:33 -05:00
Everything is in place to run the application with its contact editor.
2017-02-22 13:09:39 -05:00
Try the example:
2017-05-09 18:53:32 -04:00
2017-04-26 08:12:22 -04:00
< live-example embedded plnkr = "contact.1b" img = "guide/ngmodule/contact-1b-plunker.png" > < / live-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
## Selector conflicts
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Look closely at the screen.
Notice that the background of the application title text is _blue_ .
It should be _gold_ (see `src/app/app.component.html` ).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Only the contact name should be blue (see `src/app/contact/contact.component.html` ).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
What went wrong?
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
This application defines two highlight directives that
set the background color of their host elements with a different color (gold and blue).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
One is defined at the root level (`src/app/highlight.directive.ts`);
the other is in the contact editor folder (`src/app/contact/contact-highlight.directive.ts`).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Their class names are different (`HighlightDirective` and `ContactHighlightDirective` )
but their selectors both match any HTML element with a `highlight` attribute.
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-tabs >
< code-pane title = "src/app/highlight.directive.ts" path = "ngmodule/src/app/highlight.directive.ts" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-pane title = "src/app/contact/contact-highlight.directive.ts" path = "ngmodule/src/app/contact/contact-highlight.directive.ts" >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-pane >
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Both directives are declared in the same `AppModule` so both directives are active
for all components declared in `AppModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
There's nothing intrinsically wrong with multiple directives selecting the same element.
Each could modify the element in a different, non-conflicting way.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
In _this case_ , both directives compete to set the background color of the same element.
The directive that's declared later (`ContactHighlightDirective`) always wins because its DOM changes overwrite the changes by the earlier `HighlightDirective` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactHighlightDirective` will make the application title text blue
when it should be gold.
Only the contact name should be blue (see `src/app/contact/contact.component.html` ).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
If you cannot rename the selectors, you can resolve the conflicts by creating [feature modules ](#feature-modules )
that insulate the declarations in one NgModule from the declarations in another.
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
While it is legal to declare two _directives_ with the same selector in the same NgModule,
the compiler will not let you declare two _components_ with the same selector in the same NgModule because it **cannot insert multiple components in the same DOM location** .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Nor can you _import_ an NgModule that declares the same selector as another component in this NgModule.
The reason is the same: an HTML element may be controlled by at most one Angular component.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Either rename the selectors or use [feature modules ](#feature-modules ) to eliminate the conflict.
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
## Feature modules
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
This tiny app is already experiencing structural issues.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
* The root `AppModule` grows larger with each new application class.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
* There are conflicting directives.
The `ContactHighlightDirective` in the contact re-colors the work done by the `HighlightDirective` declared in `AppModule` and colors the application title text when it should color only the `ContactComponent` .
2017-02-22 13:09:39 -05:00
* The app lacks clear boundaries between contact functionality and other application features.
That lack of clarity makes it harder to assign development responsibilities to different teams.
2017-10-17 20:36:22 -04:00
_Feature modules_ can help resolve these issues.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Architecturally, a feature module is an NgModule class that is dedicated to an application feature or workflow.
Technically, it's another class adorned by the `@NgModule` decorator, just like a root `AppModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Feature module metadata have the same properties as root module metadata.
When loaded together, the root module and the feature module share the same dependency injector,
which means the services provided in a feature module are available to all.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
These two module types have the following significant technical differences:
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
* You _boot_ the root module to _launch_ the app;
you _import_ a feature module to _extend_ the app.
2017-10-17 20:36:22 -04:00
* A feature module can expose or hide its [declarables ](#declarables ) from other NgModules.
2017-02-22 13:09:39 -05:00
Otherwise, a feature module is distinguished primarily by its intent.
A feature module delivers a cohesive set of functionality
2017-03-06 05:43:33 -05:00
focused on an application business domain, user workflow, facility (forms, http, routing),
or collection of related utilities.
2017-10-17 20:36:22 -04:00
Feature modules help you partition the app into areas of specific interest and purpose.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
A feature module collaborates with the root module and with other NgModules
2017-02-22 13:09:39 -05:00
through the services it provides and
2017-03-06 05:43:33 -05:00
the components, directives, and pipes that it shares.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a contact-module-v1}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< h3 class = "no-toc" > Make < i > contact editor< / i > a feature< / h3 >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
In this section, you refactor the _contact editor_ functionality out of the root `AppModule`
and into a dedicated feature module by following these steps.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
1. Create the `ContactModule` feature module in its own folder.
1. Copy the _contact editor_ declarations and providers from `AppModule` to `ContactModule` .
1. Export the `ContactComponent` .
1. Import the `ContactModule` into the `AppModule` .
1. Cleanup the `AppModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
You'll create one new `ContactModule` class and change one existing `AppModule` class.
All other files are untouched.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Create the feature module
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Generate the _ContactModule_ and its folder with an Angular CLI command.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate module contact
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Here's the generated `ContactModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/contact/contact.module.0.ts" title = "src/app/contact/contact.module.ts" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
After modifying the initial `ContactsModule` as outlined above, it looks like this.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/contact/contact.module.2.ts" title = "src/app/contact/contact.module.ts" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The following sections discuss the important changes.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### Import _CommonModule_
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Notice that `ContactModule` imports `CommonModule` , not `BrowserModule` .
The CLI module generation took care of this for you.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Feature module components need the common Angular directives but
not the services and bootstrapping logic in `BrowserModule` .
See the [NgModule FAQs ](guide/ngmodule-faq#q-browser-vs-common-module )
for more details.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Import _FormsModule_
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactModule` imports the `FormsModule`
because its `ContactComponent` uses `NgModel` ,
one of the `FormsModule` directives.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< div class = "alert is-helpful" >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
NgModules don't inherit access to the declarations of the root `AppModule` or any other NgModule.
Each NgModule must import what it needs.
Because `ContactComponent` needs the form directives,
its `ContactModule` must import `FormsModule` .
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< h3 class = "no-toc" > Copy declarations< / h3 >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `ContactModule` declares the _contact editor_ components, directives and pipes.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
The app fails to compile at this point, in part because
`ContactComponent` is currently declared in both the `AppModule` and the `ContactModule` .
A component may only be declared in one NgModule.
You'll fix this problem shortly.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a root-scoped-providers}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Providers are root-scoped
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `ContactModule` provides the `ContactService` and the `AppModule` will stop providing it [after refactoring ](#refactor-appmodule ).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Architecturally, the `ContactService` belongs to the _contact editor_ domain.
Classes in the rest of the app do not need the `ContactService` and shouldn't inject it.
So it makes sense for the `ContactModule` to provide the `ContactService` as it does.
You might expect that the `ContactService` would only be injectable in classes declared or provided in the `ContactModule` .
That's not the case.
_Any_ class _anywhere_ can inject the `ContactService`
because `ContactModule` providers are _root_ -scoped.
< div class = "alert is-helpful" >
To be precise, all _eagerly loaded_ modules— modules loaded when the application starts — are root-scoped.
This `ContactModule` is eagerly loaded.
You will learn that services provided in [_lazy-loaded_ modules ](#lazy-loaded-modules )
have their own scope.
< / div >
Angular does not have _module_ -scoping mechanism.
Unlike components, NgModule instances do not have their own injectors
so they can't have their own provider scopes.
`ContactService` remains an _application_ -scoped service because Angular
registers all NgModule `providers` with the application's *root injector* .
This is true whether the service is provided directly in the root `AppModule`
or in an imported feature module like `ContactModule` .
In practice, service scoping is rarely an issue.
Components don't accidentally inject a service.
To inject the `ContactService` , you'd have to import its _type_
and explicitly inject the service into a class constructor.
Only _contact editor_ components should import the `ContactService` type.
If it's really important to you to restrict the scope of a service,
provide it in the feature's top-level component (`ContactComponent` in this case).
For more on this topic, see "[How do I restrict service scope to a module?](guide/ngmodule-faq#q-component-scoped-providers)"
in the [NgModule FAQs ](guide/ngmodule-faq ).
### Export public-facing components
The `ContactModule` makes the `ContactComponent` _public_ by _exporting_ it.
< code-example path = "ngmodule/src/app/contact/contact.module.2.ts" region = "exports" >
< / code-example >
Declared classes are _private_ by default.
Private [declarables ](#declarables ) may only appear in the templates of components declared by the _same_ NgModule.
They are invisible to components in _other_ NgModules.
That's a problem for the `AppComponent` .
Both components _used to be_ declared in `AppModule` so Angular could
display the `ContactComponent` within the `AppComponent` .
Now that the `ContactComponent` is declared in its own feature module.
The `AppComponent` cannot see it unless it is public.
The first step toward a solution is to _export_ the `ContactComponent` .
The second step is to _import_ the `ContactModule` in the `AppModule` ,
which you'll do when you [refactor the _AppModule_ ](#refactor-appmodule ).
The `AwesomePipe` and `ContactHighlightDirective` remain private and are hidden from the rest of the application.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactHighlightDirective` , being private, no longer overrides the `HighlightDirective` in the `AppComponent` . The background of the title text is gold as intended.
2017-03-06 05:43:33 -05:00
2017-10-17 20:36:22 -04:00
{@a refactor-appmodule}
### Refactor the _AppModule_
Return to the `AppModule` and remove everything specific to the _contact editor_ feature set.
2017-02-22 13:09:39 -05:00
Leave only the classes required at the application root level.
2017-10-17 20:36:22 -04:00
* Delete the _contact editor_ import statements.
* Delete the _contact editor_ declarations and providers.
* Delete the `FormsModule` from the `imports` list (the `AppComponent` doesn't need it).
* Import the `ContactModule` so the app can continue to display the exported `ContactComponent` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Here's the refactored `AppModule` , presented side-by-side with the previous version.
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-tabs >
2017-10-17 20:36:22 -04:00
< code-pane title = "app.module.ts (v2)" path = "ngmodule/src/app/app.module.2.ts" >
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-pane title = "app.module.ts (v1)" path = "ngmodule/src/app/app.module.1b.ts" >
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
### Improvements
There's a lot to like in the revised `AppModule` .
2017-02-22 13:09:39 -05:00
* It does not change as the _Contact_ domain grows.
2017-10-17 20:36:22 -04:00
* It only changes when you add new NgModules.
2017-02-22 13:09:39 -05:00
* It's simpler:
2017-03-31 19:57:13 -04:00
2017-03-06 05:43:33 -05:00
* Fewer import statements.
* No `FormsModule` import.
2017-10-17 20:36:22 -04:00
* No _contact editor_ declarations.
2017-03-06 05:43:33 -05:00
* No `ContactService` provider.
2017-10-17 20:36:22 -04:00
* No _highlight directive_ conflicts.
2017-02-22 13:09:39 -05:00
Try this `ContactModule` version of the sample.
2017-04-26 08:12:22 -04:00
< live-example embedded plnkr = "contact.2" img = "guide/ngmodule/contact-2-plunker.png" > Try the live example.< / live-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
{@a routing-modules}
{@a lazy-loaded-modules}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
## Routing modules
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Navigating the app with the [Angular Router ](guide/router ) reveals
new dimensions of the NgModule.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
In this segment, you'll learn to write _routing modules_ that configure the router.
You'll discover the implications of _lazy loading_ a feature module with the router's `loadChildren` method.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Imagine that the sample app has evolved substantially along the lines of the
[Tour of Heroes tutorial ](tutorial ).
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
* The app has three feature modules: Contact, Hero (new), and Crisis (new).
* The [Angular router ](guide/router ) helps users navigate among these modules.
* The `ContactComponent` is the default destination when the app starts.
* The `ContactModule` continues to be _eagerly loaded_ when the application starts.
* `HeroModule` and the `CrisisModule` are _lazy-loaded_ .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
There's too much code behind this sample app to review every line.
Instead, the guide explores just those parts necessary to understand new aspects of NgModules.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
You can examine the complete source for this version of the app in
2017-04-26 08:12:22 -04:00
the < live-example plnkr = "pre-shared.3" img = "guide/ngmodule/v3-plunker.png" > live example.< / live-example >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
{@a app-component-template}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< h3 class = "no-toc" > The root < i > AppComponent< / i > < / h3 >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The revised `AppComponent` template has
2017-02-22 13:09:39 -05:00
a title, three links, and a `<router-outlet>` .
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/app.component.3.ts" region = "template" title = "src/app/app.component.ts (v3 - Template)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `<app-contact>` element that displayed the `ContactComponent` is gone; you're routing to the _Contact_ page now.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< h3 class = "no-toc" > The root < i > AppModule< / i > < / h3 >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `AppModule` is slimmer now.
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/app.module.3.ts" title = "src/app/app.module.ts (v3)" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `AppModule` is no longer aware of the application domains such as contacts, heroes, and crises.
Those concerns are pushed down to `ContactModule` , `HeroesModule` , and `CrisisModule` respectively
and only the routing configuration knows about them.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The significant change from version 2 is the addition of the *AppRoutingModule* to the NgModule `imports` .
The `AppRoutingModule` is a [routing module ](guide/router#routing-module )
that handles the app's routing concerns.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
### _AppRoutingModule_
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
The router is the subject of the [Routing & Navigation ](guide/router ) guide, so this section skips many routing details and
concentrates on the _intersection_ of NgModules and routing.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
You can specify router configuration directly within the root `AppModule` or within a feature module.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The _Router guide_ recommends instead that you locate router configuration in separate, dedicated NgModules, called _routing modules_ .
You then import those routing modules into their corresponding root or feature modules.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The goal is to separate the normal declarative concerns of an NgModule from the often complex router configuration logic.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
By convention, a routing module's name ends in `...RoutingModule` .
The top-level root module is `AppModule` and it imports its companion _routing module_ called `AppRoutingModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app.module.3.ts" region = "imports" title = "src/app/app.module.ts (imports)" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Here is this app's `AppRoutingModule` , followed by a discussion.
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" title = "src/app/app-routing.module.ts" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `AppRoutingModule` defines three routes:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" title = "src/app/app-routing.module.ts (routes)" region = "routes" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
The first route redirects the empty URL (such as `http://host.com/` )
to another route whose path is `contact` (such as `http://host.com/contact` ).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `contact` route isn't defined within the `AppRoutingModule` .
It's defined in the _Contact_ feature's _own_ routing module, `ContactRoutingModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "l-sub-section" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
It's standard practice for feature modules with routing components to define their own routes.
You'll get to [`ContactRoutingModule` ](#contact-routing-module ) in a moment.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The remaining two routes use lazy loading syntax to tell the router where to find the modules for the hero and crisis features:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" region = "lazy-routes" linenums = "false" >
< / code-example >
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
A lazy-loaded NgModule location is a _string_ , not a _type_ .
In this app, the string identifies both the NgModule _file_ and the NgModule _class_ ,
2017-02-22 13:09:39 -05:00
the latter separated from the former by a `#` .
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
### Routing module imports
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
A _routing module_ typically imports the Angular `RouterModule` so it can register routes.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
It may also import a _feature module_ which registers routes (either directly or through its companion _routing module_ ).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
This `AppRoutingModule` does both.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" region = "imports" title = "src/app/app-routing.module.ts (imports)" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
It first imports the `ContactModule` , which [as you'll see ](#contact-routing-module ),
imports its own `ContactRoutingModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
**Import order matters!**
Because "contacts" is the first defined route and the default route for the app,
you must import it _before_ all other routing-related modules.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The second import registers the routes defined in this module
by calling the `RouterModule.forRoot` class method.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" region = "forRoot" >
< / code-example >
The `forRoot` method does two things:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
1. Configures the router with the supplied _routes_ .
1. Initializes the Angular router itself.
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< div class = "alert is-critical" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Call `RouterModule.forRoot` exactly once for the entire app.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Calling it in the `AppRoutingModule` , the companion to the root `AppModule` ,
is a good way to ensure that this method is called exactly once.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Never call `RouterModule.forRoot` in a feature's _routing module_ .
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Re-export _RouterModule_
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
All _routing modules_ should re-export the `RouterModule` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/app-routing.module.ts" region = "exports" title = "src/app/app-routing.module.ts (exports)" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Re-exporting `RouterModule` makes the router directives
available to the companion module that imports it.
This is a considerable convenience for the importing module.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
For example, the `AppComponent` template relies on the
[`routerLink` ](guide/router#router-links ) directive
to turn the user's clicks into navigations.
The Angular compiler only recognizes `routerLink` because
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
- `AppComponent` is declared by `AppModule` ,
- `AppModule` imports `AppRoutingModule` ,
- `AppRoutingModule` exports `RouterModule` , and
- `RouterModule` exports the `RouterLink` directive.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
If `AppRoutingModule` didn't re-export `RouterModule` , the `AppModule` would have to import the `RouterModule` itself.
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
{@a contact-routing-module}
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Routing to a feature module
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The three feature modules (`ContactModule`, `HeroModule` , `CrisisModule` )
have corresponding routing modules (`ContactRoutingModule`, `HeroRoutingModule` , `CrisisRoutingModule` ).
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
They follow the same pattern as the `AppRoutingModule` .
* define routes
* register the routes with Angular's `RouterModule`
* export the `RouterModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The `ContactRoutingModule` is the simplest of the three.
It defines and registers a single route to the `ContactComponent` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/contact/contact-routing.module.ts" region = "routing" title = "src/app/contact/contact-routing.module.ts (routing)" linenums = "false" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
There is **one critical difference** from `AppRoutingModule` :
you pass the routes to `RouterModule.forChild` , not `forRoot` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< div class = "alert is-important" >
2017-03-31 19:57:13 -04:00
2017-03-06 05:43:33 -05:00
Always call `RouterModule.forChild` in a feature-routing module.
2017-10-17 20:36:22 -04:00
Never call `RouterModule.forRoot` .
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
#### _ContactModule_ changes
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Because the app navigates to the `ContactComponent` instead of
simply displaying it in the `AppComponent` template,
the `ContactModule` has changed.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* It imports the `ContactRoutingModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* It no longer exports `ContactComponent` .
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `ContactComponent` is only displayed by the router,
No template references its `<app-contact>` selector.
There's no reason to make it public via the `exports` array.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
Here is the latest version, side-by-side with the previous version.
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-tabs >
2017-10-17 20:36:22 -04:00
< code-pane title = "contact.module.ts (v3)" path = "ngmodule/src/app/contact/contact.module.3.ts" region = "class" >
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-pane title = "contact.module.ts (v2)" path = "ngmodule/src/app/contact/contact.module.2.ts" region = "class" >
2017-03-27 11:08:53 -04:00
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
2017-02-22 13:09:39 -05:00
{@a hero-module}
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
### Lazy-loaded routing
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The `HeroModule` and `CrisisModule` have corresponding _routing modules_ , `HeroRoutingModule` and `CrisisRoutingModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The app _lazy loads_ the `HeroModule` and the `CrisisModule` .
That means the `HeroModule` and the `CrisisModule` are not loaded into the browser until the user navigates to their components.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< div class = "alert is-important" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Do not import the `HeroModule` or `CrisisModule` or any of their classes outside of their respective file folders.
If you do, you will unintentionally load those modules and all of their code
when the application starts, defeating the purpose of lazy loading.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
For example, if you import the `HeroService` in `AppModule` ,
the `HeroService` class and all related hero classes will be loaded when the application starts.
2017-02-22 13:09:39 -05:00
2017-04-11 16:44:52 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Lazy loading can improve the app's perceived performance because the browser doesn't have to process lazy-loaded code when the app starts.
It may _never_ process that code.
You cannot tell that these modules are lazy-loaded by looking at their _routing modules_ .
They happen to be a little more complex than `ContactRoutingModule` .
For example, The `HeroRoutingModule` has [child routes ](guide/router#child-routing-component ).
But the added complexity springs from intrinsic hero and crisis functionality, not from lazy loading.
Fundamentally, these _routing modules_ are just like `ContactRoutingModule` and you write them the same way.
{@a lazy-load-DI}
### Lazy-loaded NgModule providers
There is a **runtime difference** that can be significant.
Services provided by lazy-loaded NgModules are only available to classes instantiated within the lazy-loaded context. The reason has to do with dependency injection.
When an NgModule is _eagerly loaded_ as the application starts,
its providers are added to the application's _root injector_ .
Any class in the application can inject a service from the _root injector_ .
When the router _lazy loads_ an NgModule, Angular instantiates the module
with a _child injector_ (a descendant of the _root injector_ )
and adds the module's providers to this _child injector_ .
Classes created with the _child injector_ can inject one of its provided services.
Classes created with _root injector_ cannot.
Each of the three feature modules has its own data access service.
Because the `ContactModule` is _eagerly loaded_ when the application starts,
its `ContactService` is provided by the application's _root dependency injector_ .
That means the `ContactService` can be injected into any application class, including hero and crisis components.
Because `CrisisModule` is _lazy-loaded_ ,
its `CrisisService` is provided by the `CrisisModule` _child injector_ .
It can only be injected into one of the crisis components.
No other kind of component can inject the `CrisisService` because no other kind of component can be reached along a route that lazy loads the `CrisisModule` .
### Lazy-loaded NgModule lifetime
Both eager and lazy-loaded NgModules are created _once_ and never destroyed.
This means that their provided service instances are created _once_ and never destroyed.
As you navigate among the application components, the router creates and destroys instances of the contact, hero, and crisis components.
When these components inject data services provided by their modules,
they get the same data service instance each time.
If the `HeroService` kept a cache of unsaved changes and the user navigated to the `ContactComponent` or the `CrisisListComponent` , the pending hero changes would remain in the one `HeroService` instance, waiting to be saved.
But if you provided the `HeroService` in the `HeroComponent` instead of the `HeroModule` , new `HeroService` instances would be created each time
the user navigated to a hero component. Previously pending hero changes would be lost.
To illustrate this point, the sample app provides the `HeroService` in the `HeroComponent` rather than the `HeroModule` .
Run the app, open the browser development tools, and look at the console as you navigate among the feature pages.
< code-example language = "sh" class = "code-shell" linenums = "false" >
// App starts
ContactService instance created.
...
// Navigate to Crisis Center
CrisisService instance created.
...
// Navigate to Heroes
HeroService instance created.
...
// Navigate to Contact
HeroService instance destroyed.
...
// Navigate back to Heroes
HeroService instance created.
< / code-example >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
The console log shows the `HeroService` repeatedly created and destroyed.
The `ContactService` and `CrisisService` are created but never destroyed, no matter where you navigate.
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
#### Run it
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
Try this routed version of the sample.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< live-example embedded plnkr = "pre-shared.3" img = "guide/ngmodule/v3-plunker.png" > Try the live example.< / live-example >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
{@a shared-module}
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
## Shared modules
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
The app is shaping up.
But there are a few annoying problems.
There are three unnecessarily different _highlight directives_
and the many files cluttering the app folder level could be better organized.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
You can eliminate the duplication and tidy-up by writing a `SharedModule`
to hold the common components, directives, and pipes.
Then share this NgModule with the other NgModules that need these declarables.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Use the CLI to create the `SharedModule` class in its `src/app/shared` folder.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example language = "sh" class = "code-shell" >
ng generate module shared
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Now refactor as follows:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
- Move the `AwesomePipe` from `src/app/contact` to `src/app/shared` .
- Move the `HighlightDirective` from `src/app/hero` to `src/app/shared` .
- Delete the _highlight directive_ classes from `src/app/` and `src/app/contact` .
- Update the `SharedModule` as follows:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/shared/shared.module.ts" title = "src/app/shared/shared.module.ts" >
< / code-example >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Note the following:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* It declares and exports the shared pipe and directive.
* It imports and re-exports the `CommonModule` and `FormsModule`
* It can re-export `FormsModule` without importing it.
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
### Re-exporting NgModules
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Technically, there is no need for `SharedModule` to import `CommonModule` or `FormsModule` .
`SharedModule` doesn't declare anything that needs material from `CommonModule` or `FormsModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
But NgModules that would like to import `SharedModule` for its pipe and highlight directive happen also to declare components that need `NgIf` and `NgFor` from `CommonModule`
and do two-way binding with `[(ngModel)]` from the `FormsModule` .
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
Normally, they'd have to import `CommonModule` and `FormsModule` as well as `SharedModule` .
Now they can just import `SharedModule` .
By exporting `CommonModule` and `FormsModule` ,
`SharedModule` makes them available to its importers _for free_ .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
#### A trimmer _ContactModule_
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
See how `ContactModule` became more concise, compared to its previous version:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-tabs >
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
< code-pane title = "src/app/contact/contact.module.ts (v4)" path = "ngmodule/src/app/contact/contact.module.ts" >
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-pane title = "src/app/contact/contact.module.ts (v3)" path = "ngmodule/src/app/contact/contact.module.3.ts" >
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Notice the following:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
* The `AwesomePipe` and `ContactHighlightDirective` are gone.
* The imports include `SharedModule` instead of `CommonModule` and `FormsModule` .
* The new version is leaner and cleaner.
2017-02-22 13:09:39 -05:00
### Why _TitleComponent_ isn't shared
2017-03-06 05:43:33 -05:00
`SharedModule` exists to make commonly used components, directives, and pipes available
2017-10-17 20:36:22 -04:00
for use in the templates of components in many other NgModules.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
The `TitleComponent` is used only once by the `AppComponent` .
2017-02-22 13:09:39 -05:00
There's no point in sharing it.
2017-03-31 19:57:13 -04:00
{@a no-shared-module-providers}
2017-02-22 13:09:39 -05:00
### Why _UserService_ isn't shared
2017-03-06 05:43:33 -05:00
While many components share the same service instances,
2017-10-17 20:36:22 -04:00
they rely on Angular dependency injection to do this kind of sharing, not the NgModule system.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Several components of the sample inject the `UserService` .
There should be only one instance of the `UserService` in the entire application
and only one provider of it.
2017-02-22 13:09:39 -05:00
`UserService` is an application-wide singleton.
2017-10-17 20:36:22 -04:00
You don't want each NgModule to have its own separate instance.
2017-04-30 16:10:32 -04:00
Yet there is [a real danger ](guide/ngmodule-faq#q-why-bad ) of that happening
2017-02-22 13:09:39 -05:00
if the `SharedModule` provides the `UserService` .
2017-04-10 11:51:13 -04:00
< div class = "alert is-critical" >
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Do *not* specify app-wide singleton `providers` in a shared module.
2017-10-17 20:36:22 -04:00
A lazy-loaded NgModule that imports that shared module makes its own copy of the service.
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
{@a core-module}
## The Core module
2017-10-17 20:36:22 -04:00
2017-03-06 05:43:33 -05:00
At the moment, the root folder is cluttered with the `UserService`
and `TitleComponent` that only appear in the root `AppComponent` .
You didn't include them in the `SharedModule` for reasons just explained.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Instead, gather them in a single `CoreModule` that you import once when the app starts
2017-03-31 07:23:16 -04:00
and never import anywhere else.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Perform the following steps:
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
1. Create a `CoreModule` class in an `src/app/core` folder.
1. Move the `TitleComponent` and `UserService` from `src/app/` to `src/app/core` .
1. Declare and export the `TitleComponent` .
1. Provide the `UserService` .
1. Update the root `AppModule` to import `CoreModule` .
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Most of this work is familiar. The interesting part is the `CoreModule` .
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< code-example path = "ngmodule/src/app/core/core.module.ts" region = "v4" title = "src/app/core/core.module.ts" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-04-10 11:51:13 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-03-06 05:43:33 -05:00
You're importing some extra symbols from the Angular core library that you're not using yet.
2017-03-27 11:08:53 -04:00
They'll become relevant later in this page.
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
The `@NgModule` metadata should be familiar.
2017-10-17 20:36:22 -04:00
You declare the `TitleComponent` because this NgModule owns it.
You export it because `AppComponent` (which is in `AppModule` ) displays the title in its template.
2017-03-06 05:43:33 -05:00
`TitleComponent` needs the Angular `NgIf` directive that you import from `CommonModule` .
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
`CoreModule` provides the `UserService` . Angular registers that provider with the app root injector,
2017-02-22 13:09:39 -05:00
making a singleton instance of the `UserService` available to any component that needs it,
whether that component is eagerly or lazily loaded.
2017-04-10 11:51:13 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-05-08 23:47:34 -04:00
< h3 class = "no-toc" > Why bother?< / h3 >
2017-10-17 20:36:22 -04:00
2017-02-22 13:09:39 -05:00
This scenario is clearly contrived.
The app is too small to worry about a single service file and a tiny, one-time component.
A `TitleComponent` sitting in the root folder isn't bothering anyone.
The root `AppModule` can register the `UserService` itself,
2017-03-06 05:43:33 -05:00
as it does currently, even if you decide to relocate the `UserService` file to the `src/app/core` folder.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Real-world apps have more to worry about.
They can have several single-use components (such as spinners, message toasts, and modal dialogs)
2017-02-22 13:09:39 -05:00
that appear only in the `AppComponent` template.
2017-03-06 05:43:33 -05:00
You don't import them elsewhere so they're not shared in that sense.
2017-02-22 13:09:39 -05:00
Yet they're too big and messy to leave loose in the root folder.
Apps often have many singleton services like this sample's `UserService` .
2017-03-06 05:43:33 -05:00
Each must be registered exactly once, in the app root injector, when the application starts.
2017-02-22 13:09:39 -05:00
2017-03-31 07:23:16 -04:00
While many components inject such services in their constructors— and
therefore require JavaScript `import` statements to import their symbols— no
2017-10-17 20:36:22 -04:00
other component or NgModule should define or re-create the services themselves.
2017-03-06 05:43:33 -05:00
Their _providers_ aren't shared.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
We recommend collecting such single-use classes and hiding their details inside a `CoreModule` .
2017-02-22 13:09:39 -05:00
A simplified root `AppModule` imports `CoreModule` in its capacity as orchestrator of the application as a whole.
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
2017-10-17 20:36:22 -04:00
#### A trimmer _AppModule_
2017-02-22 13:09:39 -05:00
Here is the updated `AppModule` paired with version 3 for comparison:
2017-03-27 11:08:53 -04:00
< code-tabs >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/app.module.ts (v4)" path = "ngmodule/src/app/app.module.ts" region = "v4" >
< / code-pane >
2017-02-22 13:09:39 -05:00
2017-03-27 11:08:53 -04:00
< code-pane title = "src/app/app.module.ts (v3)" path = "ngmodule/src/app/app.module.3.ts" >
< / code-pane >
< / code-tabs >
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
`AppModule` now has the following qualities:
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
* A little smaller because many `src/app/root` classes have moved to other NgModules.
* Stable because you'll add future components and providers to other NgModules, not this one.
* Delegated to imported NgModules rather than doing work.
2017-03-06 05:43:33 -05:00
* Focused on its main task, orchestrating the app as a whole.
2017-02-22 13:09:39 -05:00
{@a core-for-root}
2017-10-17 20:36:22 -04:00
### Configure core services with _CoreModule.forRoot_
2017-03-31 19:57:13 -04:00
2017-10-17 20:36:22 -04:00
An NgModule that adds providers to the application can offer a facility for configuring those providers as well.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
By convention, the `forRoot` static method both provides and configures services at the same time.
2017-02-22 13:09:39 -05:00
It takes a service configuration object and returns a
2017-04-30 16:10:32 -04:00
[ModuleWithProviders ](api/core/ModuleWithProviders ), which is
2017-03-06 05:43:33 -05:00
a simple object with the following properties:
2017-03-31 19:57:13 -04:00
2017-03-06 05:43:33 -05:00
* `ngModule` : the `CoreModule` class
* `providers` : the configured providers
2017-02-22 13:09:39 -05:00
The root `AppModule` imports the `CoreModule` and adds the `providers` to the `AppModule` providers.
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< div class = "l-sub-section" >
2017-03-27 11:08:53 -04:00
2017-03-06 05:43:33 -05:00
More precisely, Angular accumulates all imported providers before appending the items listed in `@NgModule.providers` .
This sequence ensures that whatever you add explicitly to the `AppModule` providers takes precedence
2017-10-17 20:36:22 -04:00
over the providers of imported NgModules.
2017-03-27 11:08:53 -04:00
2017-04-10 11:51:13 -04:00
< / div >
2017-03-27 11:08:53 -04:00
Add a `CoreModule.forRoot` method that configures the core `UserService` .
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
You've extended the core `UserService` with an optional, injected `UserServiceConfig` .
2017-02-22 13:09:39 -05:00
If a `UserServiceConfig` exists, the `UserService` sets the user name from that config.
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/core/user.service.ts" region = "ctor" title = "src/app/core/user.service.ts (constructor)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
Here's `CoreModule.forRoot` that takes a `UserServiceConfig` object:
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/core/core.module.ts" region = "for-root" title = "src/app/core/core.module.ts (forRoot)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Lastly, call it within the `imports` list of the `AppModule` .
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/app.module.ts" region = "import-for-root" title = "src/app//app.module.ts (imports)" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
The app displays "Miss Marple" as the user instead of the default "Sherlock Holmes".
2017-04-10 11:51:13 -04:00
< div class = "alert is-important" >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Call `forRoot` only in the root module, `AppModule` .
Calling it in any other NgModule, particularly in a lazy-loaded NgModule,
2017-03-06 05:43:33 -05:00
is contrary to the intent and can produce a runtime error.
2017-02-22 13:09:39 -05:00
Remember to _import_ the result; don't add it to any other `@NgModule` list.
2017-04-10 11:51:13 -04:00
< / div >
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
< hr >
2017-02-22 13:09:39 -05:00
{@a prevent-reimport}
2017-10-17 20:36:22 -04:00
### Prevent reimport of the _CoreModule_
2017-02-22 13:09:39 -05:00
Only the root `AppModule` should import the `CoreModule` .
2017-10-17 20:36:22 -04:00
[Bad things happen ](guide/ngmodule-faq#q-why-bad ) if a lazy-loaded NgModule imports it.
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
You could hope that no developer makes that mistake.
Or you can guard against it and fail fast by adding the following `CoreModule` constructor.
2017-02-22 13:09:39 -05:00
2017-03-31 19:57:13 -04:00
< code-example path = "ngmodule/src/app/core/core.module.ts" region = "ctor" title = "src/app/core/core.module.ts" linenums = "false" >
2017-03-27 11:08:53 -04:00
< / code-example >
2017-02-22 13:09:39 -05:00
The constructor tells Angular to inject the `CoreModule` into itself.
That seems dangerously circular.
2017-03-06 05:43:33 -05:00
The injection would be circular if Angular looked for `CoreModule` in the _current_ injector.
The `@SkipSelf` decorator means "look for `CoreModule` in an ancestor injector, above me in the injector hierarchy."
2017-02-22 13:09:39 -05:00
If the constructor executes as intended in the `AppModule` ,
there is no ancestor injector that could provide an instance of `CoreModule` .
The injector should give up.
2017-03-06 05:43:33 -05:00
By default, the injector throws an error when it can't find a requested provider.
2017-02-22 13:09:39 -05:00
The `@Optional` decorator means not finding the service is OK.
The injector returns `null` , the `parentModule` parameter is null,
and the constructor concludes uneventfully.
2017-10-17 20:36:22 -04:00
It's a different story if you improperly import `CoreModule` into a lazy-loaded NgModule such as `HeroModule` (try it).
2017-02-22 13:09:39 -05:00
2017-10-17 20:36:22 -04:00
Angular creates a lazy-loaded NgModule with its own injector, a _child_ of the root injector.
2017-03-06 05:43:33 -05:00
`@SkipSelf` causes Angular to look for a `CoreModule` in the parent injector, which this time is the root injector.
2017-02-22 13:09:39 -05:00
Of course it finds the instance imported by the root `AppModule` .
2017-03-31 19:57:13 -04:00
Now `parentModule` exists and the constructor throws the error.
2017-05-08 23:47:34 -04:00
## Conclusion
2017-02-22 13:09:39 -05:00
You made it! You can examine and download the complete source for this final version from the live example.
2017-04-26 08:12:22 -04:00
< live-example embedded img = "guide/ngmodule/final-plunker.png" > < / live-example >
2017-02-22 13:09:39 -05:00
2017-05-08 23:47:34 -04:00
## Frequently asked questions
2017-02-22 13:09:39 -05:00
2017-03-06 05:43:33 -05:00
Now that you understand NgModules, you may be interested
2017-04-30 16:10:32 -04:00
in the companion [NgModule FAQs ](guide/ngmodule-faq "NgModule FAQs" ) page
2017-04-24 14:23:45 -04:00
with its ready answers to specific design and implementation questions.