- Identify components on websites you commonly use
- Create a custom directive
- Explain why custom directives are valuable
Before this lesson, students should already be able to:
- Describe Angular directives
As you've seen by now, a huge amount of the code you work with in Angular are directives. Angular was designed to be an extension of HTML - a way to have custom-defined interactive tags of your own making.
While we've been getting good at using the directives that come with Angular, it's time to start seeing what we can do if we start building our own custom directives.
One of the most obvious uses of this is when you've got repetitive code to render some information or data - when you have a repeated component somewhere within your site. If you're using a single component (and all of its associated HTML) repetitively in your views, it's a simple DRY principle – gather that view and funcitonality so that it's all written together in one place.
By extracting a component to a custom directive, we can just reference that directive whenever we need to use the component and avoid repeating the code to render it.
As an example, we're going to mess around with duplicating something that's becoming a common pattern in interface design – the concept of a card. Applications like Twitter, Pinterest, Facebook, and a lot more are moving towards this design pattern.
Everyone's favorite CSS framework, Bootstrap, is even on board, where in version 4+ you're able to create a card with just some CSS classes:
Let's see if we can make something similar, wrapped up in our own custom HTML element.
But first:
Let's take a moment to look at our favorite websites and see if we can find any pieces of a webpage that seem like they could be captured as a component.
Take five minutes and inspect our starter code in this repo's starter-code/app/
directory. You'll see a pretty normal Angular app, and since we're repeating using those cards, and there's a few consistent tags we're repeating every time we render a card, we're going to experiment with making those cards a custom-defined directive.
Note: if you are going to run this code locally on Chrome, you'll need to run npm init
and then install http-server with npm install http-server -g
. To run locally, simply use the command http-server
.
Using our starter code, our goal is to take:
<div class='card'>
<h4 class="card-title">{{card.question}}</h4>
<h6>Cards Against Assembly</h6>
</div>
and end up with a reusable <card></card>
component, maybe something like:
<card question="{{card.question}}"></card>
Rather than just throw this wherever, let's add it to our existing app.js
Of course, it could be named anything, but it's sort of a view, and it's definitely a card, so cardView
felt right. Up to you.
Just like controllers, factories, anything else we've made in angular, the first line is a simple extension of angular
:
angular
.module('CardsAgainstAssembly', [])
.directive('card', CardViewDirective);
An important thing to point out: The first argument is the name of the directive and how you'll use it in your HTML; and remember, Angular converts camelCase
to snake-case
for us, so if you want to use <secret-garden></secret-garden>
in your HTML, name your directive .directive('secretGarden', myFunctionIHaventMadeYet)
.
Now, we obviously need a function named cardView
!
function CardViewDirective(){
var directive = {};
return directive;
}
Nothing fancy yet - we're just constructing an object. We'll put some specifics in there now, but that's simple so far.
You've got a couple interesting options when making your own directives. We'll go through them all, quickly, and you can play with them on your own in a bit.
restrict
replace
template/templateUrl
scope
While the name isn't obvious, the restrict
option lets us decide what kind of directive we want to make. It looks like this:
restrict = 'EACM';
E
is element. An HTML element, like<card></card>
A
is attribute. Like<div card="something"></div>
C
is class. Like<div class="card"></div>
M
is comment. Like<!-- directive: card -->
You can choose to have just one, all of the above, or any combination you like. You should steer towards elements & attributes as much as possible, though – classes can get messy with other CSS classes, and comments could just end up weird if there isn't a good reason for it.
For ours, let's play with just an element.
function CardViewDirective(){
var directive = {
restrict : 'E'
};
return directive;
}
Replace is pretty straightforward. Should this directive replace the HTML? Do you want it to get rid of what's in the template & swap it out with the template we're going to make? Or add to it, and not remove the original. For example, replacing would mean:
<div ng-repeat="card in cards.questions" >
<card></card>
</div>
Would actually render as:
<div ng-repeat="card in cards.questions" >
<div class='card'>
<h4 class="card-title">{{question}}</h4>
<h6>Cards Against Assembly</h6>
</div>
</div>
See, replaced. Let's say we like that for our example:
function CardViewDirective(){
var directive = {
restrict : 'E',
replace : true
};
return directive;
}
This is where our partial view comes in. Now, if it's a pretty tiny, self-contained directive, you can use template="Some javascript " + string + " concatenation";
But that easily starts getting ugly, so it's often better (even for small directives like this) to make a quick little partial HTML file and reference it with templateUrl=
instead.
Let's extract our existing card tags, and throw them in a partial. Cut out:
<div class='card'>
<h4 class="card-title">{{card.question}}</h4>
<h6>Cards Against Assembly</h6>
</div>
Quickly touch _cardView.html
or some similarly obvious-named partial, and paste it back in.
<!-- app/templates/_cardView.html -->
<div class='card'>
<h4 class="card-title">{{card.question}}</h4>
<h6>Cards Against Assembly</h6>
</div>
In js/app.js
, we can add our option:
function CardViewDirective (){
var directive = {
//'A' == attribute, 'E' == element, 'C' == class, 'M' == comment
restrict : 'E',
replace : true,
templateUrl : "_cardView.html"
};
return directive;
}
And lastly, in our index.html
, let's finally use our custom directive. So exciting. This is it. Here we go.
<!-- index.html -->
<div class='col-sm-6 col-md-6 col-lg-4' ng-repeat="card in cards.questions" >
<card></card>
</div>
TRY IT! So awesome! We've now got this much more readable index.html
, with a very semantic HTML tag describing exactly what we want rendered.
This is awesome. This is a great, reusable component. Except for one thing.
If you notice, our template uses {{card.question}}
inside it. This obviously works perfectly - we're geniuses. But what if we wanted to render a card somewhere outside of that ng-repeat
, where card in cards.questions
isn't a thing. What if we want to render a one-off card, reusing our awesome new directive elsewhere? Isn't that part of the point?
It sure is. We're lacking a precise scope.
Just like controllers, we want to define what our scope is. We want to be able to say "Render a card, with these details, in whatever context I need to render it in." A card shouldn't rely on a controller's data to know what information to render inside it. The controller should pass that data to our directive, so it's freestanding and not relying on anyone but itself.
That's where scope
comes in, and this lets us decide what attributes our element should have! For example, in our card example, maybe we want to render a card with just a string somewhere outside of this controller. We want to make our own card with our own hardcoded text.
Try this. In your index.html
, adjust our <card>
element to say:
<card question="{{card.question}}"></card>
In context, you'll see that the ng-repeat
is giving us the variable card
, and we're actually just rendering that out as a string. But we've decided we want to have an attribute called question
to pass data through. We made that up, it's appropriate to our example, but it can be anything.
There are only two other pieces we need to make this reality.
In our _cardView.html
partial, let's adjust to:
<div class='card'>
<h4 class="card-title">{{question}}</h4>
<h6>Cards Against Assembly</h6>
</div>
No longer reliant on a variable named card
, it's now just reliant on an element having the attribute of question
.
And finally, in js/app.js
:
angular
.module('CardsAgainstAssembly',[])
.directive('card', CardViewDirective);
function CardViewDirective(){
var directive = {
//'A' == attribute, 'E' == element, 'C' == class, 'M' == comment
restrict : 'E',
replace : true,
templateUrl : "_cardView.html",
scope : {
question: '@'
}
};
return directive;
}
In scope
, we just define an object. The key is whatever want the attribute on the element to be named. So if we want <card bagel=""></card>
, then we'd need a key named bagel
in our scope object.
The value is one of 3 options.
scope: {
ngModel: '=', // Bind the ngModel to the object given
onSend: '&', // Pass a reference to the method
fromName: '@' // Store the string associated by fromName
}
The corresponding options would look like:
<div scope-example ng-model="to" on-send="sendMail(email)" from-name="[email protected]" />
The =
is a mechanism for binding data that might change; the &
is for passing a reference to a function you might want to call; and the @
is simply storing a string & passing it through to the template.
Our last test is to see if we can make a card using just a hardcoded string. Then we'll know our card is really reusable.
Somewhere outside the context of the controller, let's say just above the footer in our index.html
, throw a handmade card in:
<!-- ... -->
</section>
<hr/>
<card question="Why is Angular so awesome?"></card>
<footer>
<!-- ... -->
Would you look at that? Our own custom directive - a reusable, semantic HTML component that we designed ourselves.
Now, while we dove pretty deep into explaining each part, you can see it's really just a combination of quickly defining a custom directive, what options you want, making a template, and then using it.
For practice, let's break up into pairs and make something extra.
If you didn't notice, there's some extra code included in your starter index.html
:
<header class='navbar'>
<h1 class='pull-left'>Cards Against Assembly</h1>
<scores></scores>
</header>
Our goal is to craft a custom directive to show off our players scores, like so:
As a pair, build a custom directive that makes use of the PlayersController
included in your starter code, listing out each player & their score, built as a custom <score></score>
directive.
You have 15 minutes! Go!
- Where can you imagine using custom directives?
- What four types of directives can you make?
- How do you pass information into a custom directive?