Table des matières

Création de directives

Quels intérêts ?

Directive simple & template

var app = angular.module('testDirectivesApp', [])
.controller('Controller', ['$scope', function($scope) {
  $scope.client = {
    nom: 'SMITH',
    prenom: 'John'
  };
}]);

app.directive('myClient', function() {
  return {
    template: 'Nom : {{client.nom}} Prénom : {{client.prenom}}'
  };
});

Utilisations possibles :

<html data-ng-app="testDirectivesApp" data-ng-controller="Controller">
...
	<my-client></my-client>
	<div my-client></div>

Résultat :

templateUrl

Excepté pour les petits templates, il est préférable d'utiliser un fichier template séparé en affectant la propriété templateUrl (notamment pour éviter les problèmes de quote/guillemets ) :

app.directive('myClient', function() {
  return {
    templateUrl: 'my-client.html'
  };
});

Nom : {{client.nom}} Prénom : {{client.prenom}}

La propriété templateUrl peut-être définie par une fonction :

app.directive('myClient', function() {
  return {
    templateUrl: function(elem, attr){
      return 'client-'+attr.type+'.html';
    }
  };
});

Templates :

Nom : {{client.nom}} Prénom : {{client.prenom}}

Adresse: {{client.adresse}}<br>
CP: {{client.cp}} {{client.ville}}

Utilisation :

<html data-ng-app="testDirectivesApp" data-ng-controller="Controller">
...
	<div myclient type="nom"></div>
	<div my-client type="adresse"></div>

restrict

La propriété restrict permet de définir le type de directive à créer ; restrict peut prendre les valeurs suivantes :

Si la directive est définie sur 1 élément :

app.directive('myClient', function() {
  return {
    restrict: 'E',
    templateUrl: function(elem, attr){
      return 'client-'+attr.type+'.html';
    }
  };
});

Seule la première ligne sera compilée par Angular, la seconde sera ignorée :

<html data-ng-app="testDirectivesApp" data-ng-controller="Controller">
...
	<my-client type="nom"></my-client>
	<div my-client type="adresse"></div>

scope

Le scope défini la portée d'une directive. Sans précision, le scope d'une directive est le scope de son contrôleur parent. Ce qui explique que notre directive ait pu accéder à la variable client de Controller.

Isolation

La directive précédemment créée a un défaut : elle dépend du contrôleur dans laquelle elle a été définie, et si nous souhaitons afficher un autre client, il faut définir un autre contrôleur…
La définition de la variable scope de la directive permet de résoudre ce problème :

scope:{} Défini à {}, la directive a un scope qui lui est propre, différent de celui du controller auquel elle appartient :

app.directive('myClient', function() {
  return {
    restrict: 'AE',
    scope:{},
    templateUrl: function(elem, attr){
      return 'client-'+attr.type+'.html';
    }
  };
});

Avec ce scope isolé, la directive ne peut plus accéder à la variable client définie dans le scope du controller : tester la page tests.html

Nous allons ajouter une propriété permettant d'initialiser la directive, et lui affecter la personne à afficher :

<html data-ng-app="testDirectivesApp" data-ng-controller="Controller">
...
	<my-client type="nom" personne="client"></my-client>
	<my-client type="adresse" personne="client"></my-client>

app.directive('myClient', function() {
	return{
		restrict:'EA',
		scope:{client:"=personne"},
		templateUrl: function(elem, attr){
			return 'client-'+attr.type+'.html';
		}
	}
});

L'attribut personne “bind” vers la variable client utilisée dans le scope de la directive.

Lorsqu'un attribut a le même nom que la variable du scope auquel il est associé, on peut utiliser la notation :

scope : {attributeName: "="}

Pour résumer (rapidement…) :

Valeurs de scope Résultats
false ou non définie pas de scope
{} scope isolé
{attributeName:“=”} scope isolé, avec passage d'un attribut de type expression valeur du même nom
{attributeNameInDirectiveScope:“=attributeName”} scope isolé, avec passage d'un attribut de type expression valeur de nom différent
{attributeName:“@”} scope isolé, avec passage d'un attribut de type texte du même nom
{attributeNameInDirectiveScope:“@attributeName”} scope isolé, avec passage d'un attribut de type texte de nom différent
{attributeName:“&”} scope isolé, avec passage d'un attribut de type expression action du même nom
{attributeNameInDirectiveScope:“&attributeName”} scope isolé, avec passage d'un attribut de type expression action de nom différent
true scope non isolé

Exemple de passage d'une expression action (ng-click) :

	<my-client type="nom" personne="client" ng-click="client.adresse=''"></my-client>
	<div my-client type="adresse" personne="client"></div>

Sur click du nom, l'adresse doit être vidée.

En l'état actuel du code, la directive ng-click n'est pas opérationnelle, il faut la passer à notre directive :

app.directive('myClient', function() {
	return{
		restrict:'EA',
		scope:{client:"=personne",onClick:"&ngClick"},
		templateUrl: function(elem, attr){
			return 'client-'+attr.type+'.html';
		}
	}
});

et la déclencher à nouveau dans le template :

<p ng-click="onClick">Nom : {{client.nom}} Prénom : {{client.prenom}}</p>

Link/compile

Lorsqu'il analyse une directive appelée dans une page et la traduit en un ensemble d'éléments DOM, AngularJS compile la directive en faisant appel aux fonctions suivantes, dans un ordre déterminé :

  1. compile
  2. controller
  3. pre-link
  4. post-link

Si une directive doit modifier ou créer le DOM d'un template défini dans une directive, elle doit donc définir certaines de ces fonctions.

fonction compile

compile : function(tElement, tAttrs){

}
Utilisée pour la manipulation du DOM (manipulation de tElement = template element), elle permet des manipulations qui s'appliqueront à tous les éléments DOM clonés du template associé à la directive. S'il est nécessaire de définir également la fonction link (ou pre ou post link), et que la fonction compile est définie, la fonction compile doit renvoyer le(s) fonction(s) link, étant donné que link est ignoré si compile existe.

fonction controller

controller : function($scope, $element, $attrs, $transclude, otherInjectables){

}
Elle doit être utilisée pour permettre l'interaction avec d'autres directives.

link : function(scope, iElement, iAttrs, controller){

}
Elle est habituellement utilisée pour enregistrer des listeners DOM (i.e., $watch expressions sur le scope) ou pour mettre à jour le DOM (i.e., manipulation sur iElement = instance individuelle de element). Elle est exécutée après que le template ait été cloné – voir <li ng-repeat…>, ou la fonction link est exécutée après que chaque <li> template (tElement) ait été cloné (en iElement)–. $watch permet à la directive d'être notifiée du changement de l'une des propriétés du scope (un scope est associé à chaque instance).

Exemple

Soit la directive SoldeDir, dont les fonctionnalités sont les suivantes :

app.directive('soldeDir', function() {
	return {
		restrict : 'EA',
		scope:{montant:"="},
		compile : function(tElem, tAttrs) {
			if (tAttrs.bold == "true")
				tElem.css("font-weight","bold");
			else
				tElem.css("font-weight","normal");
            var linkFunction = function($scope, tElem, tAttrs) {
            	var update=function(){
	            	if($scope.montant<0){
	            		tElem.html("Solde négatif : " + $scope.montant);
	            		tElem.css("color", "red");
	            	}else{
	            		tElem.html($scope.montant);
	            		tElem.css("color", "green");
	            	}
            	}
            	update();
            }
            return linkFunction;
		}
	}
});

	<div ng-init="valeur=-5">
	<input type="text" ng-model="valeur">
	<solde-dir bold="true" montant="valeur"></solde-dir>

La directive fonctionne pour l'instant à l'initialisation, mais ne reprend pas les éventuelles modifications du montant.
Pour remédier à ce problème, il faut ajouter dans la fonction link un appel au service $watch, pour observer et reporter les modifications du montant :

app.directive('soldeDir', function() {
	return {
		restrict : 'EA',
		scope:{montant:"="},
		compile : function(tElem, tAttrs) {
			if (tAttrs.bold == "true")
				tElem.css("font-weight","bold");
			else
				tElem.css("font-weight","normal");
            var linkFunction = function($scope, tElem, tAttrs) {
            	var update=function(){
	            	if($scope.montant<0){
	            		tElem.html("Solde négatif : " + $scope.montant);
	            		tElem.css("color", "red");
	            	}else{
	            		tElem.html($scope.montant);
	            		tElem.css("color", "green");
	            	}
            	}
            	update();
            }
                $scope.$watch('montant', function(oldVal, newVal) {
                    update();
                });
            return linkFunction;
		}
	}
});

Pour observer les changements d'un attribut de type texte (@) contenant une expression, on utilisera la méthode $observe sur l'attribut :

Soit la directive :

<dir attrib="test : {{valeur}}"></dir>

app.directive('dir', function() {
	return {
		restrict : 'E',
		scope:{attrib:"@"},
		link: function($scope, tElm, tAttrs,ctrl,transclude) {
			tAttrs.$observe('attrib', function() {
				$scope.attrib = $scope.$eval(tAttrs.attrib);
			});
		}
	};
});