This is a preview-quality chapter of our continuously deployed eBook on AngularJS for .NET developers. You can read more about the project at http://henriquat.re. You can also follow us on twitter (Project:@henriquatreJS, Authors: @ingorammer and @christianweyer)
A common question when looking at AngularJS for the first time is about how to combine it with existing JavaScript code and components. For jQuery UI in particular, the team around AngularJS has created the AngularUI project (http://angular-ui.github.com/) which provides an easy way to use jQuery UI with Angular.). For other frameworks - or especially for your custom DOM-manipulating code - the necessary steps might not be as obvious at the first glance.
As you have seen in the previous chapters, AngularJS puts a very strong focus on separation of concerns:
Most existing components (especially in the jQuery-world) usually work in a different way. Without Angular, when you wanted to achieve a certain behavior, like displaying a date-picker, you would usually go with a jQuery UI extension (like 'datepicker') and use call similar to $("#someinput").datepicker({...})
. This would extend a standard <input>
element with an ID of someinput
to be turned into a datepicker.
In this command's options, you would have usually specified a callback to be invoked whenever the user selects/changes the date in the input box. But you wouldn't do this just for a single datepicker -- no, this kind of jQuery UI-extension would be littered throughout your business code for nearly every input control in your HTML. After all, your datepickers need access to the underlying model values to restrict the input to valid ranges and to perform additional validation if necessary. This mix of business code and DOM-manipulating code is sometimes the reason for maintenance headaches of JavaScript developers.
With this in mind, you can see the conundrum: how do you take classic JavaScript code (with this mix of DOM interactions and event handlers with direct manipulation of underlying business objects) and put it into the well defined structure of AngularJS?
So let's just see how we're going to take a jQuery-based fragment and move it forward to a reusable AngularJS directive. In the following example, you'll see a simple text entry box which is treated as a jQuery UI datepicker. When changing the date, the underlying business object will be updated and the value will be shown underneath the input box.
This demonstration consists mainly of two parts. The first is the rather straightforward and easy to understand HTML markup:
<div> Date Of Birth: <input type="text" id="dateOfBirth"> <br> Current user's date of birth: <span id="dateOfBirthDisplay"></span> </div>
So far, so good. Now let's have a look at the jQuery-based code behind this HTML:
$(function () { var user = { dateOfBirth: new Date(1970, 0, 1) }; var displayValue = function () { $("#dateOfBirthDisplay").text(new Date(user.dateOfBirth).toDateString()); }; var processChange = function() { user.dateOfBirth = $("#dateOfBirth").datepicker("getDate"); displayValue(); }; $("#dateOfBirth").datepicker({ inline: true, onClose: processChange, onSelect: processChange } ); displayValue(); // initial display of value in input-box $("#dateOfBirth").datepicker("setDate", user.dateOfBirth); });
Please note that this fragment, even though it already mixes code from different areas of responsibility (DOM manipulation, data conversion, business object population) is hardly complete: in practice, you'd quite likely also have to add validation code to check for date formats and ranges. But let's keep this simple for now.
Now, if code like this would only occur once throughout your application, its use could be quite acceptable. Unfortunately, without a framework like AngularJS, code like this would be written in multiple throughout your application. In fact, whenever you'd have a date-entry box, you would quite likely see code like this. As you can imagine: maintenance of a codebase like this might be quite a bit harder than it should be.
As you've seen before, AngularJS allows you to separate the concerns if your code into three different parts: the model, the controller and directives which perform DOM manipulation. So let's look at this particular example and how we can transform this into a more suitable - and maintainable - form.
At first, we will change the HTML markup to tell AngularJS which internal application name should be used (as we will register the directives with this application). In addition, we'll already add the necessary data-binding information and add the directive my-datepicker
instead of the <input>
-element which we've used before. (Please note: this particular directive does not yet exist in AngularJS, but we'll build it throughout the remainder of this chapter.)
<div ng-app="demo" ng-controller="DemoController"> Date Of Birth: <my-datepicker type="text" ng-model="user.dateOfBirth" /> <br> Current user's date of birth: <span id="dateOfBirthDisplay">{{user.dateOfBirth}}</span> </div>
The matching controller simply exposes the user
property via its $scope
and we have removed all interactions with jQuery UI from this business logic code:
function DemoController($scope) { $scope.user = { dateOfBirth: new Date(1970, 0, 1) } }
To create the link to jQuery UI's datepicker-addin, we are introducing the following directive. (Worry not: we'll discuss this directive line-by-line just in a few seconds.)
angular.module("demo", []).directive('myDatepicker', function ($parse) { return { restrict: "E", replace: true, transclude: false, compile: function (element, attrs) { var modelAccessor = $parse(attrs.ngModel); var html = "<input type='text' id='" + attrs.id + "' >" + "</input>"; var newElem = $(html); element.replaceWith(newElem); return function (scope, element, attrs, controller) { var processChange = function () { var date = new Date(element.datepicker("getDate")); scope.$apply(function (scope) { // Change bound variable modelAccessor.assign(scope, date); }); }; element.datepicker({ inline: true, onClose: processChange, onSelect: processChange }); scope.$watch(modelAccessor, function (val) { var date = new Date(val); element.datepicker("setDate", date); }); }; } }; });
Contrary to the example which you've seen in Introduction to Directives, we're not simply returning alink
function. Instead, we're returning a so called compile
function for this directive.
The reason for this is based on the internal logic in which directives are applied by Angular: in the first phase, the compile phase, you can modify the HTML-element which will be added to the DOM at the location of your directive. You can for example emit a completely different HTML element. In the second phase, during linking, you can change the content and behavior of the element after the result of the compilation has been added to the DOM.
The important thing to note is that, if your directive uses a compile
function, it is required to return thelink
function which should be called at a later time.
So let's look at the individual elements of our directive, step by step.
First, we're defining a module demo (which we will later reference from the HTML) and we're adding a directive called myDatepicker to it. We require a reference to $parse
to be injected in the code so that we can later use this to parse model-binding expressions.
angular.module("demo", []) .directive('myDatepicker', function ($parse) {
We then indicate that our directive will be used as an HTML-elements (and not as an attribute or CSS-class).
restrict: "E"
We then tell AngularJS that we want to replace the element with the result of our directive but don't require automatic transclusion of the content. (You can read more about transclusion at Introduction to Directives).
replace: true, transclude: false,
The remainder of the directive is the compile function. Let's first look at the compilation inside this function. At the beginning of this method, we use $parse
to parse the ng-model
attribute which has been specified in the HTML markup (it contains the target field for data binding).
$parse
converts (or compiles) an AngularJS expression into a function. If we for example specify "user.dateOfBirth" as an expression, then $parse
will return a function which allows us to retrieve and set this value from the underlying scope. (Please note: as always with AngularJS, the naming convention islowercase-with-dashes in HTML and camelCase in JS, so that the JavaScript field attrs.ngModel
will contain the string-contents of the HTML-attribute ng-model
).
compile: function (element, attrs) { var modelAccessor = $parse(attrs.ngModel);
After this, we use jQueryOnly (or the AngularJS-supplied minimum jQuery equivalent) to create an<input>
-element and then replace the existing directive element (which temporarily exists in the HTML) with this new <input>
. (And while doing this, we're preserving the HTML element's ID from the HTML element which defined this directive.)
var html = "<input type='text' id='" + attrs.id + "' >" + "</input>"; var newElem = $(html); element.replaceWith(newElem);
Up to this point, we've looked only at the compilation part. What's next is that we're returning the link
function from this compile
function:
return function (scope, element, attrs, controller) { /* ... */ };
In this returned link
-function, we're doing three things:
onClose
and onSelect
callbacks.datepicker()
extension to add the desired behavior to the underyling <input>
-elementLet's first have a look at the function which will later be used for the datepicker's callbacks.
var processChange = function () { var date = new Date(element.datepicker("getDate")); scope.$apply(function (scope) { modelAccessor.assign(scope, date); }); };
The important parts here happen in the call to scope.$apply
. This method is used to change model values in a way which allows AngularJS' change tracking to update the underlying HTML elements. (Or more correctly, it triggers the watches
which have been registered for the model).
The modelAccessor
we're using in this method is the return value of the earlier call to $parse(...)
: its assign
method is a preprocessed accessor function which sets the configured (according to the ng-model
attribute on the HTML directive) field of the corresponding object to the specified value.
To recap: whenever this processChange
-function is called, it will extract the current value from the HTML element which underpins the datepicker by using its getDate
access-method. It will then apply the changes to the underlying model by using the assign()
method of a pre-parsed accessor within an $apply
-block.
In the next step, we simply use jQuery UI's datepicker extension to get the desired client-side behavior. We pass the processChange
- function which we've defined above as the callbacks for onClose
and onSelect
. In this way processChanges
will be called whenever the user changes the datepicker's value.
element.datepicker({ inline: true, onClose: processChange, onSelect: processChange });
And as the final step, we watch the model for changes so that we can update the UI as required. To do this, we pass the modelAccessor
(which we've received as the return value from $parse
) to Angular's $watch method, which is part of the current scope
.
The callback to $watch
will be triggered whenever the model value has been changed. (By using $apply
or by updating a bound UI control).
scope.$watch(modelAccessor, function (val) { var date = new Date(val); element.datepicker("setDate", date); });
In this way, we have now defined a re-usable directive which works independent from any GUI. In fact, this component could easily be re-used in a completely different AngularJS application.
If you run this initial version of the AngularJS application, you can see the following behavior:
At this point, the data-binding works as expected, we have a very clear separation of concerns (there is absolutely no direct binding between your business code in the Controller and jQuery UI's datepicker.)
This was a preview-quality chapter of our continuously deployed eBook on AngularJS for .NET developers. If you enjoyed this chapter, you can read more about the project at http://henriquat.re. You can also follow us on twitter (Project: @henriquatreJS, Authors: @ingorammer and @christianweyer)
(source : http://henriquat.re/directives/advanced-directives-combining-angular-with-existing-components-and-jquery/angularAndJquery.html)