Multiple Components 6.0
The AppComponent
is doing everything at the moment.
In the beginning, it showed details of a single hero.
Then it became a master/detail form with both a list of heroes and the hero detail.
Soon there will be new requirements and capabilities.
You can’t keep piling features on top of features in one component; that’s not maintainable.
You’ll need to break it up into sub-components, each focused on a specific task or workflow.
Eventually, the AppComponent
could become a simple shell that hosts those sub-components.
In this page, you’ll take the first step in that direction by carving out the hero details into a separate, reusable component. When you’re done, the app should look like this live example (view source).
Where you left off
Before getting started on this page, verify that you have the following structure from earlier in the Tour of Heroes. If not, go back to the previous pages.
- angular_tour_of_heroes
- lib
- app_component.{css,dart,html}
- src
- hero.dart
- mock_heroes.dart
- test
- …
- web
- index.html
- main.dart
- styles.css
- analysis_options.yaml
- pubspec.yaml
- lib
If the app isn’t running already, launch the app. As you make changes, keep it running by reloading the browser window.
Make a hero detail component
Create a file named hero_component.dart
.
This file will hold the new HeroComponent
.
Angular conventions:
-
The component class name should be written in upper camel case and end in the word “Component”. The hero detail component class is
HeroComponent
. -
The component file name should be in snake case —lowercase with underscore separation—and end in
_component.dart
. TheHeroComponent
class goes in thehero_component.dart
file. -
Internal implementation files should be placed under
lib/src
. See the pub package layout conventions for details.
Start writing the HeroComponent
as follows:
lib/src/hero_component.dart (initial version)
import 'package:ngdart/angular.dart';
import 'package:ngforms/ngforms.dart';
@Component(
selector: 'my-hero',
directives: [coreDirectives, formDirectives],
)
class HeroComponent {
}
To define a component, you always import the main Angular library.
The @Component
annotation provides the Angular metadata for the component.
The CSS selector name, my-hero
, will match the element tag
that identifies this component within a parent component’s template.
Near the end of this tutorial page,
you’ll add a <my-hero>
element to the AppComponent
template.
Hero detail template
To move the hero detail view to the HeroComponent
,
cut the hero detail content from the bottom of the AppComponent
template
and paste it into a new template
argument of the @Component
annotation.
The HeroComponent
has a hero, not a selected hero.
Replace selected
by hero
, everywhere in the template.
When you’re done, the new template should look like this:
lib/src/hero_component.dart (template)
template: '''
<div *ngIf="hero != null">
<h2>{{hero!.name}}</h2>
<div><label>id: </label>{{hero!.id}}</div>
<div>
<label>name: </label>
<input [(ngModel)]="hero!.name" placeholder="name">
</div>
</div>''',
In case you’re wondering, the !
symbol after hero
is a null-check operator
for Dart, one of the language features that help you write more error-proof
code. For details, checkout the null-safety introduction page.
You also might be confused as to why we still need to declare hero
as non-null
even though it’s already wrapped inside the ngIf
condition that checks whether
it’s null or not (try removing one of the !
and recompile!). The answer is
that hero
might be changed outside of the ngIf
condition. This is very
unlikely to occur (and definitely impossible to occur in this situation), but
when it does, the little !
will save you a ton of time debugging.
Add the hero property
The HeroComponent
template binds to the component’s hero
property.
Add that property, along with the requisite import,
to the HeroComponent
class.
lib/src/hero_component.dart (hero)
import 'hero.dart';
class HeroComponent {
Hero? hero;
}
The hero property is an input property
Later in this page,
the parent AppComponent
will tell the child HeroComponent
which hero to display
by binding its selected
to the hero
property of the HeroComponent
.
The binding will look like this:
<my-hero [hero]="selected"></my-hero>
Putting square brackets around the hero
property, to the left of the equal sign (=),
makes it the target of a property binding expression.
You must declare a target binding property to be an input property.
Otherwise, Angular rejects the binding and throws an error.
Declare that hero
is an input property by annotating it with @Input()
:
lib/src/hero_component.dart (Input annotation)
@Input()
Hero? hero;
Read more about input properties in the Attribute Directives page.
That’s it. The hero
property is the only thing in the HeroComponent
class.
All it does is receive a hero object through its hero
input property and then bind to that property with its template.
Here’s the complete HeroComponent
.
lib/src/hero_component.dart
import 'package:ngdart/angular.dart';
import 'package:ngforms/ngforms.dart';
import 'hero.dart';
@Component(
selector: 'my-hero',
template: '''
<div *ngIf="hero != null">
<h2>{{hero!.name}}</h2>
<div><label>id: </label>{{hero!.id}}</div>
<div>
<label>name: </label>
<input [(ngModel)]="hero!.name" placeholder="name">
</div>
</div>''',
directives: [coreDirectives, formDirectives],
)
class HeroComponent {
@Input()
Hero? hero;
}
Add HeroComponent to the AppComponent
The AppComponent
is still a master/detail view.
It used to display the hero details on its own, before you cut out that portion of the template.
Now it will delegate to the HeroComponent
.
Start by importing the HeroComponent
so AppComponent
can refer to it.
import 'src/hero_component.dart';
Recall that my-hero
is the CSS selector
in the HeroComponent
metadata.
That’s the tag name of the element that represents the HeroComponent
.
Add a <my-hero>
element near the bottom of the AppComponent
template,
where the hero detail view used to be.
Coordinate the master AppComponent
with the HeroComponent
by binding the selected
property of the AppComponent
to the hero
property of the HeroComponent
.
<my-hero [hero]="selected"></my-hero>
Now every time the selected
changes, the HeroComponent
gets a new hero to display.
The revised AppComponent
template should look like this:
lib/app_component.html
<h1>{{title}}</h1>
<h2>Heroes</h2>
<ul class="heroes">
<li *ngFor="let hero of heroes"
[class.selected]="hero == selected"
(click)="onSelect(hero)">
<span class="badge">{{hero.id}}</span> {{hero.name}}
</li>
</ul>
<my-hero [hero]="selected"></my-hero>
The detail should update every time the user picks a new hero. It’s not happening yet! Click a hero. No details. If you look for an error in the console of the browser development tools. No error.
It is as if Angular were ignoring the new tag. That’s because it is ignoring the new tag.
The directives list
A browser ignores HTML tags and attributes that it doesn’t recognize. So does Angular.
You’ve imported HeroComponent
, and you’ve used <my-hero>
in
the template, but you haven’t told Angular about it.
Just as you’ve done for the built-in Angular directives, tell Angular
about the hero detail component by listing it in the metadata directives
list. You don’t need formDirectives
anymore, so delete it and the
angular_forms
import at the top of the file:
lib/app_component.dart (directives)
directives: [coreDirectives, HeroComponent],
open_in_browser Refresh the browser. The app works!
App design changes
As before, whenever a user clicks on a hero name,
the hero detail appears below the hero list.
But now the HeroComponent
is presenting those details.
Refactoring the original AppComponent
into two components yields benefits, both now and in the future:
- You simplified the
AppComponent
by reducing its responsibilities. - You can evolve the
HeroComponent
into a rich hero editor without touching the parentAppComponent
. - You can evolve the
AppComponent
without touching the hero detail view. - You can reuse the
HeroComponent
in the template of some future parent component.
Review the app structure
Verify that you have the following structure:
- angular_tour_of_heroes
- lib
- app_component.{css,dart,html}
- src
- hero.dart
- hero_component.dart
- mock_heroes.dart
- test
- …
- web
- index.html
- main.dart
- styles.css
- analysis_options.yaml
- pubspec.yaml
- lib
Here are the code files discussed in this page.
The road you’ve travelled
Here’s what you achieved in this page:
- You created a reusable component.
- You learned how to make a component accept input.
- You learned to declare the app directives in a
directives
list. - You learned to bind a parent component to a child component.
Your app should look like this live example (view source).
The road ahead
The Tour of Heroes app is more reusable with shared components,
but its (mock) data is still hard coded within the AppComponent
.
That’s not sustainable.
Data access should be refactored to a separate service
and shared among the components that need data.
You’ll learn to create services in the next tutorial page.