Issue
I need to perform some operations on scope and the template. It seems that I can do that in either the link
function or the controller
function (since both have access to the scope).
When is it the case when I have to use link
function and not the controller?
angular.module('myApp').directive('abc', function($timeout) {
return {
restrict: 'EA',
replace: true,
transclude: true,
scope: true,
link: function(scope, elem, attr) { /* link function */ },
controller: function($scope, $element) { /* controller function */ }
};
}
Also, I understand that link
is the non-angular world. So, I can use $watch
, $digest
and $apply
.
What is the significance of the link
function, when we already had controller?
Solution
After my initial struggle with the link
and controller
functions and reading quite a lot about them, I think now I have the answer.
First let's understand,
How do AngularJS directives work in a nutshell:
We begin with a template (as a string or loaded to a string)
var templateString = '<div my-directive>{{5 + 10}}</div>';
Now, this
templateString
is wrapped as an angular elementvar el = angular.element(templateString);
With
el
, now we compile it with$compile
to get back the link function.var l = $compile(el)
Here is what happens:
$compile
walks through the whole template and collects all the directives that it recognizes.- All the directives that are discovered are compiled recursively and their
link
functions are collected. - Then, all the
link
functions are wrapped in a newlink
function and returned asl
.
Finally, we provide
scope
function to thisl
(link) function which further executes the wrapped link functions with thisscope
and their corresponding elements.l(scope)
This adds the
template
as a new node to theDOM
and invokescontroller
which adds its watches to the scope which is shared with the template in DOM.
Comparing compile vs link vs controller :
Every directive is compiled only once and link function is retained for re-use. Therefore, if there's something applicable to all instances of a directive should be performed inside directive's
compile
function.Now, after compilation we have
link
function which is executed while attaching the template to the DOM. So, therefore we perform everything that is specific to every instance of the directive. For eg: attaching events, mutating the template based on scope, etc.Finally, the controller is meant to be available to be live and reactive while the directive works on the
DOM
(after getting attached). Therefore:(1) After setting up the view[V] (i.e. template) with link.
$scope
is our [M] and$controller
is our [C] in M V C(2) Take advantage the 2-way binding with $scope by setting up watches.
(3)
$scope
watches are expected to be added in the controller since this is what is watching the template during run-time.(4) Finally,
controller
is also used to be able to communicate among related directives. (LikemyTabs
example in https://docs.angularjs.org/guide/directive)(5) It's true that we could've done all this in the
link
function as well but its about separation of concerns.
Therefore, finally we have the following which fits all the pieces perfectly:
Answered By - Yugal Jindle
0 comments:
Post a Comment
Note: Only a member of this blog may post a comment.