IMPORTANT: The Angular team recommends using standalone components instead of NgModule
for all new code. Use this guide to understand existing code built with @NgModule
.
An NgModule is a class marked by the @NgModule
decorator. This decorator accepts metadata that tells Angular how to compile component templates and configure dependency injection.
import {NgModule} from '@angular/core';@NgModule({ // Metadata goes here})export class CustomMenuModule { }
An NgModule has two main responsibilities:
- Declaring components, directives, and pipes that belong to the NgModule
- Add providers to the injector for components, directives, and pipes that import the NgModule
Declarations
The declarations
property of the @NgModule
metadata declares the components, directives, and pipes that belong to the NgModule.
@NgModule({ /* ... */ // CustomMenu and CustomMenuItem are components. declarations: [CustomMenu, CustomMenuItem],})export class CustomMenuModule { }
In the example above, the components CustomMenu
and CustomMenuItem
belong to CustomMenuModule
.
The declarations
property additionally accepts arrays of components, directives, and pipes. These arrays, in turn, may also contain other arrays.
const MENU_COMPONENTS = [CustomMenu, CustomMenuItem];const WIDGETS = [MENU_COMPONENTS, CustomSlider];@NgModule({ /* ... */ // This NgModule declares all of CustomMenu, CustomMenuItem, // CustomSlider, and CustomCheckbox. declarations: [WIDGETS, CustomCheckbox],})export class CustomMenuModule { }
If Angular discovers any components, directives, or pipes declared in more than one NgModule, it reports an error.
Any components, directives, or pipes must be explicitly marked as standalone: false
in order to be declared in an NgModule.
@Component({ // Mark this component as `standalone: false` so that it can be declared in an NgModule. standalone: false, /* ... */})export class CustomMenu { /* ... */ }
imports
Components declared in an NgModule may depend on other components, directives, and pipes. Add these dependencies to the imports
property of the @NgModule
metadata.
@NgModule({ /* ... */ // CustomMenu and CustomMenuItem depend on the PopupTrigger and SelectorIndicator components. imports: [PopupTrigger, SelectionIndicator], declarations: [CustomMenu, CustomMenuItem],})export class CustomMenuModule { }
The imports
array accepts other NgModules, as well as standalone components, directives, and pipes.
exports
An NgModule can export its declared components, directives, and pipes such that they're available to other components and NgModules.
@NgModule({ imports: [PopupTrigger, SelectionIndicator], declarations: [CustomMenu, CustomMenuItem], // Make CustomMenu and CustomMenuItem available to // components and NgModules that import CustomMenuModule. exports: [CustomMenu, CustomMenuItem],})export class CustomMenuModule { }
The exports
property is not limited to declarations, however. An NgModule can also export any other components, directives, pipes, and NgModules that it imports.
@NgModule({ imports: [PopupTrigger, SelectionIndicator], declarations: [CustomMenu, CustomMenuItem], // Also make PopupTrigger available to any component or NgModule that imports CustomMenuModule. exports: [CustomMenu, CustomMenuItem, PopupTrigger],})export class CustomMenuModule { }
NgModule
providers
Tip: See the Dependency Injection guide for information on dependency injection and providers.
An NgModule
can specify providers
for injected dependencies. These providers are available to:
- Any standalone component, directive, or pipe that imports the NgModule, and
- The
declarations
andproviders
of any other NgModule that imports the NgModule.
@NgModule({ imports: [PopupTrigger, SelectionIndicator], declarations: [CustomMenu, CustomMenuItem], // Provide the OverlayManager service providers: [OverlayManager], /* ... */})export class CustomMenuModule { }@NgModule({ imports: [CustomMenuModule], declarations: [UserProfile], providers: [UserDataClient],})export class UserProfileModule { }
In the example above:
- The
CustomMenuModule
providesOverlayManager
. - The
CustomMenu
andCustomMenuItem
components can injectOverlayManager
because they're declared inCustomMenuModule
. UserProfile
can injectOverlayManager
because its NgModule importsCustomMenuModule
.UserDataClient
can injectOverlayManager
because its NgModule importsCustomMenuModule
.
The forRoot
and forChild
pattern
Some NgModules define a static forRoot
method that accepts some configuration and returns an array of providers. The name "forRoot
" is a convention that indicates that these providers are intended to be added exclusively to the root of your application during bootstrap.
Any providers included in this way are eagerly loaded, increasing the JavaScript bundle size of your initial page load.
boorstrapApplication(MyApplicationRoot, { providers: [ CustomMenuModule.forRoot(/* some config */), ],});
Similarly, some NgModules may define a static forChild
that indicates the providers are intended to be added to components within your application hierarchy.
@Component({ /* ... */ providers: [ CustomMenuModule.forChild(/* some config */), ],})export class UserProfile { /* ... */ }
Bootstrapping an application
IMPORTANT: The Angular team recommends using bootstrapApplication instead of bootstrapModule
for all new code. Use this guide to understand existing applications bootstrapped with @NgModule
.
The @NgModule
decorator accepts an optional bootstrap
array that may contain one or more components.
You can use the bootstrapModule
method from either platformBrowser
or platformServer
to start an Angular application. When run, this function locates any elements on the page with a CSS selector that matches the listed componet(s) and renders those components on the page.
import {platformBrowser} from '@angular/platform-browser';@NgModule({ bootstrap: [MyApplication],})export class MyApplicationModule { }platformBrowser().bootstrapModule(MyApplicationModule);
Components listed in bootstrap
are automatically included in the NgModule's declarations.
When you bootstrap an application from an NgModule, the collected providers
of this module and all of the providers
of its imports
are eagerly loaded and available to inject for the entire application.