This is a walkthrough is to help guide you through the construction of a custom directive in Angular ###The Set Up The initial set up for this is very easy. We do not have to install anything through npm, as there is not going to be a backend. We are simply going to be creating a file structure, the files we will need and adding the proper script tags to the index.html document.
###The fie structure In our root directory we are going to have five directories and our index.html. In our 'components' subdirectory well have another directory called 'myDirective', which will have three javascript files, 'custom-directive.js', 'directive-services.js' and 'directive.js'. In our 'controllers' directory we will have one javascript file, 'home.js'. In our 'js' directory we will have our 'app.js'. In our 'templates' subdirectory we will have 'home.html'. In our 'css' directory we will have 'style.css'.
- root
- components
- custom-directive.js
- directive-services.js
- directive.js
- controllers
- homeController.js
- js
- app.js
- templates
- home.html
- custom-directive.html
- css
- style.css
- index.html
###index.html Inside of our index's head tags we are going to include script tags for Bootstrap js and jQuery, we will have link tags for Bootstrap css and style.css.
<link rel="stylesheet" href="http://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css">
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.12.4/jquery.min.js"></script>
<script src="http://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js"></script>
<link rel="stylesheet" href="/css/style.css" media="screen" charset="utf-8">
Before the closing body tag, we will have our script tags for Angular 1.5.5, angular-ui-router, app.js, homeController.js, custom-directive.js, directive.js and directive-services.js.
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.5.5/angular.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular-ui-router/0.2.8/angular-ui-router.min.js"></script>
<script type="text/javascript" src="/js/app.js"></script>
<script type="text/javascript" src="controllers/homeController.js">
</script>
<!--
aboutMe -->
<script type="text/javascript" src="/components/myDirective/directive.js"></script>
<script type="text/javascript" src="/components/myDirective/custom-directive.js"></script>
<script type="text/javascript" src="/components/myDirective/directive-services.js"></script>
##App.js
We start by creating an IIFE (Immediately Invoked Function Expression), inside of which we set our angular.module()
, we pass in the name of our app and the directives we will be using, in this case it is 'ui.router' and 'custom.directive'.
Then, still inside of the IIFE, we do a '.config', which we pass '$stateProvider', '$urlRouterProvider' and '$locationProvider' as arguments. These are all components of ui-router.
The first line inside of the '.config' will be our $urlRouterProvider.otherwise()
, which will pass the value of root, '/', which will default our app to the root route when an non-existent route is selected.
We will then set up our other route, which in this case will be the home route. We start by adding '$stateProvider', underneath we do '.state', where will pass in the name of the route, 'home', the url, '/', the templateUrl, '/templates/home.html' and the controller, 'homeController'.
Your final app.js should look like this:
(function(){
angular
.module('custom',[
'ui.router',
'custom.directive'
])
.config(function($stateProvider, $urlRouterProvider, $locationProvider){
$urlRouterProvider.otherwise('/');
$stateProvider
.state('home',{
url: '/',
templateUrl: '/templates/home.html',
controller: 'homeController'
});
});
})();
##Setting up the controller
We will now set up 'home.js'. We start by adding a IIFE (we love IIFEs). We will place inside of it 'angular.module' statement, which we again pass the name of our app, which is still 'custom'. Beneath the '.module' we will place a '.controller', where we will pass in the name of 'homeController' as a string and after as a variable. We will then inject '$scope' and '$location' into 'homeController'. We have a test function at the bottom, just to ensure everything is working properly, it is not necessary.
Your final 'home.js' should look like this.
(function(){
angular
.module('custom')
.controller('homeController', homeController);
homeController.$inject= ['$scope', '$location'];
function homeController($scope, $location){
console.log('homeController');
}
})();
##Setting up custom-directive.js and directive.js
We are going to start making the main feature of this walkthrough, the directive. Now you may be asking yourself, "what is a custom directive?"
At a high level, directives are markers on a DOM element (such as an attribute, element name, comment or CSS class) that tell AngularJS's HTML compiler ($compile) to attach a specified behavior to that DOM element (e.g. via event listeners), or even to transform the DOM element and its children.
Source
Essentially, a custom directive is a directive we make to manipulate the DOM in some form.
To start, we are going to add a IIFE in our 'custom-directive.js' file, which you should be used to by now. We are going to do our 'angular.module()' statement. Next we will do our '.directive()' statement, where well pass in the name 'customDirective' and a variable by the same name. We will then create a function that will return our directive. Inside the function we will declare a variable called 'directive' and we will set it equal to an object. Inside of the object we will have "restrict: 'E'", which sets the restriction to DOM elements only. The next set of key-value pairs will be "templateUrl: '/templates/custom-directive.html'". Following that, our next pair will be 'scope', set to an empty object, which creates an isolate scope for the directive. Then we will have "controller: directiveController", which allows us to pass around scope. Finally, we will have "controllerAs: 'directiveController'", this will keep the controller intact if/when we minify.
Now that our directive object is completed, we will return the variable 'directive' and close the function.
We will, underneath the closing bracket of the function inject and empty array to 'directiveController'. There is a test function, that calls 'directiveController'.
Your final product will look like:
//make iife
(function(){
//set up module for angular
angular
.module('custom.MyDirective.custom-directive', [])
//make the directive name
.directive('customDirective', customDirective);
//make fuction to return the directive
function customDirective(){
var directive = {
//restrict to a dom element
restrict: 'E',
//html that the directive will use
templateUrl: '/templates/custom-directive.html',
//makeing isolate scope for the directive
scope: {},
//adds a controller for the directive to use to pass scope around
controller: directiveController,
//pass the controller as for when the file is minify
controllerAs: 'directiveController'
};
//return the directive that we made
return directive;
}
directiveController.$inject = [];
function directiveController() {
console.log('hit');
}
})();
##Setting up directive.js
We will declare at the top of the file 'use strict'.
We will do our final 'angular.module()', which we will pass 'custom.directive' and inside of square brackets we will pass 'custom.MyDirective.custom-directive'.
Your final 'directive.js' file should look thusly:
"use strict";
angular
.module('custom.directive',[
'custom.MyDirective.custom-directive'
// 'custom.MyDirective.directive-services'
]);
##Setting up directive-services.js and style.css
They are both empty, bam, you're done. Style it if you want, but for the purposes of this exercise, you don't have to.