GitXplorerGitXplorer
j

angularjs-handbook

public
31 stars
4 forks
0 issues

Commits

List of commits on branch master.
Unverified
bfe286074da54dcf3d2af3811b7bab7e02963883

Updating

jjedrichards committed 11 years ago
Unverified
7763a9adee52fed3c0f7e6f2717f921b8858e348

Updating

jjedrichards committed 11 years ago
Unverified
d104845fd4299a9e3196d957fe9158ce770b0748

Updating

jjedrichards committed 11 years ago
Unverified
3b00ef22091cb0e63f21b52b1e8811043f302798

Updating

jjedrichards committed 11 years ago
Unverified
d30ef9e925f78f012a8fd7d623d2e694f2f451a8

Updating

jjedrichards committed 11 years ago
Unverified
5bc8f63c7afe7f2ac7b76dc63dc3d7fc9a75c156

Updating

jjedrichards committed 11 years ago

README

The README file for this repository.

AngularJS Handbook

Table of contents

  1. Modules
  2. Components
  3. Scopes
  4. Views
  5. Directives
  6. Controllers
  7. Filters
  8. Promises
  9. HTTP
  10. Routing
  11. Forms
  12. Testing

Modules

A module is a logical collection of application components.

Modules provide a mechanism for declaring how groups of app components should be configured and then bootstrapped. They also provide a way for modules to declare dependencies on other modules.

Modules do not attempt to solve the problem of script lazy loading or inter-script dependencies.

Module lifecycle

Modules have two distinct phases in their lifecycle:

  1. The configuration phase. Constants are set and module components are registered and configured.
  2. The run phase. Module components are instantiated and injected with dependencies as needed.

Module creation

Modules are created by defining the module name and an array of dependencies on other modules:

angular.module("products",["resources.products","security.auth"]);

Module retrieval

Once created, modules can be retrieved later on by just passing in a single parameter, the module name:

var mod = angular.module("products");

Module dependencies

A module dependency implies that the configuration of the depended upon module should occur before that of the parent module, nothing more. The load/definition order of Angular modules is generally not important since at load time the component definition code is merely registered, not invoked.

App module structure

Some apps will declare a single "app" module namespace and lump all the components together. Other apps will be slightly more granular and attempt to group components into a limited set of modules organised by type, i.e. a "services" module.

It is generally preferable however to create a highly granular set of modules - one module per site section, or even one module per file, and in each case explicitly define each module's limited set of dependencies. In this way the inter-dependency of modules is clear and the code is more testable.

Whichever approach is taken, it is common to declare a main "app" module upon which top level dependencies are declared and any configuration work unique to the live site (i.e. not needed in test scenarios) can be completed.

Components

These are the various decoupled components that are registered on modules. They are variously: services, controllers, filters, directives, constants and simple values, either originating from the Angular core library or custom components.

Module components are registered during the configuration phase and instantiated and injected with dependencies later on during the run phase, therefore they can be registered in any order.

Each component is defined together with a unique name which is used to resolve dependency injections. Therefore module component names must be unique across the entire app.

A module component can be registered as follows:

angular.module("app")
    .service("name",ConstructorFunction);

As you can see a reference to a constructor function for the component is attached to the module but it is not invoked.

The registration syntax also supports method chaining. This is used when defining multiple components in one file. However it's generally a good idea to organise components into folders and individual files.

angular.module("header")
    .service("navService",ConstructorFunction)
    .controller("HeaderCtrl",ConstructorFunction);

Component dependency injection

A key feature of Angular is the way that module components are only instantiated as needed with their dependencies injected dynamically during the run phase. This makes writing testable code easier as modules and components can be created in isolation at will and injected with mock/stub dependencies during a test.

All dependency names are combined into one app wide namespace. Therefore components registered in one module are available for injection in any other module.

DI occurs when the Angular injector is able map a dependency to a previously registered component. There are two main types of DI syntax.

Inferred syntax

This is where the framework infers the dependencies from the function parameter names. This is a terse syntax but it will fail once the code has been minified since this will typically rename function parameters and destroy the mappings:

angular.module("app")
    .service("foo",function ($http,API,API_KEY) {
        // The AngularJS Core $http component and the custom constants
        // API and API_KEY will be injected here and available
        // for use in "foo" service.
        return function () {};
    });

Inline annotation syntax

In this syntax dependency names are hardcoded as strings into an ordered array along with the component function itself as the last element. This syntax is functionally equivalent to the inferred syntax above and resilient to minification although arguably more verbose and harder to maintain.

angular.module("app")
    .service("foo",["$http","API","API_KEY",function ($http,API,API_KEY) {
        return function () {};
    }]);

Constant components

Module constants can be set via the module constant method. These define immutable values that are then available for DI in both the configuration and run module phases.

angular.module("error-messages")
    .constant("API","/api/v1/")
    .constant("API_KEY","940e12e5eaeb4e1baf964e45fc946498")
    .constant("ERROR_MESSAGES",{
        "500": "Oh no, something has broken",
        "403": "You need to be logged in to do that"
    });

Config components

This is code to be run only during the configuration phase. Components can be injected into a config block and configured at will. Constants (and "provider" services, more on these later) are the only sort of component that may be injected during config:

angular.module("app")
    .config(function (API,API_KEY,apiService) {
        // Here the apiService provider is configured
        // with some constant values during the config phase
        apiService.base = API;
        apiService.key = API_KEY;
    );

Run components

This is code to be run after the configuration phase but before the module at large kicks into gear. This is where you can bootstrap your app if required. It is conceptually similar to an application's main function or primary entry point - the difference being that they are run on a per module basis and there can be multiple run blocks.

For example to expose the time at which the app started-up on the root view scope:

angular.module("app")
    .run(function ($rootScope) {
        $rootScope.appStartTime = new Date();
    });

Service components

AngularJS services are singleton components. Only one instance is created by the framework and then supplied for every DI request. Service uses are typically:

  • Persist and share data between components
  • Provide an interface for loading and accessing that data sync/aysnc
  • Containers for reusable chunks of app logic and functionality

Services can receive DI just like any other component and can be registered on a module in a number of ways.

module.service(name,fn)

This is the simplest pattern (although naming one of a set of module service creation methods service itself was a confusing naming mistake). It allows registration of a service via a constructor function:

angular.module("services")
    .service("myService",function () {
        this.foo = function () {
        };
        this.bar = function () {
        };
    });

module.factory(name,fn)

This is a slightly more flexible pattern. Register an arbitrary object as a service. Object creation logic can be more complex and private fields can be simulated:

angular.module("services")
    .factory("myServiceFactory",function () {
        var privateVal = "baz";
        return {
            foo: function () {
            },
            bar: function () {
            }
        }
    });

module.provider(name,fn)

The most complex pattern. It allows an arbitrary object to be registered just like the factory pattern but also allows that object to be configured during the configuration phase before it's used for DI. Usually overkill for most services, and most useful when a service needs to be re-used across applications with configurable changes to behaviour.

To register a provider service:

angular.module("services")
    .provider("myServiceProvider",function () {
        var configurableVal = "foo";
        this.setConfigurableVal = function (val) {
            configurableVal = val;
        }
        this.$get = function () {
            return {
                // Provider service body is defined here as the return
                // value of the special $get method and has access to
                // any configured values
            }
        }
    });

And to configure it:

angular.module("app")
    .config(function (myServiceProvider) {
        myServiceProvider.setConfigurableVal("foo");
    });

Note: Providers injected during the configuration phase have not been fully instantiated via their $get method yet, only the configuration API is exposed to the config code.

Scopes

Scopes are the application view model, that is, the model as it pertains to a section of the DOM.

Each scope is an instance of the Scope class. The Scope class doesn't contain any special data model related functionality beyond some framework utility methods, and therefore such data models can be thought of as POJOs (plain old JavaScript objects), at least in terms of data access and assignment.

A new scope instance is created every time Angular encounters a "scope creating directive" (such as ng-controller).

Any value assigned to a $scope instance will become available for evaluation in a view and will be automatically watched for changes - this is the live two way data binding between view and model that Angular is well known for.

Example of automatic scope creation and injection in a controller:

<div ng-controller="MyController">
    <p>{{foo}}</p>
</div>
angular.module("controllers")
    .controller("MyController",function ($scope) {
        // A new Scope instance has been instantiated and injected
        // ready for use with this controller and its view
        $scope.foo = "bar";
    });

Scope inheritance

All $scope objects inherit from the application's $rootScope instance via normal prototypical inheritance. Therefore properties defined in a parent scope are visible in all child scopes (unless a child scope redefines an identically named property). Since new scopes are created by directives/controllers they will tend to create a hierarchical, inheriting tree-like structure that partially reflects the DOM structure.

Note: that inherited primitive properties such as the string and number types will prototypically propagate into child scopes by initial value only. That is, if their value is changed in the child scope then that change won't be reflected back into any parent scopes. Therefore if it is required that scope properties are fully shared throughout the scope inheritance hierarchy then define them as properties on an object reference, e.g. obj:{prop:value}.

Scope events

Scopes can broadcast events down to child scopes, and emit events back up to parent scopes.

$scope.$broadcast("event",data);
$scope.$emit("event",data);
$scope.$on("event",handlerFunction);

Note: In general scope events should be used sparingly, the auto two-way data binding between scope and view can go a long way towards coordinating state changes across an application without the need for events. Events are best suited for notifying app actors of global async state changes that don't relate to state held in view scopes - for example routing changes.

Advanced scope methods

$scope.$new(isolate)

Creates a new child scope:

var childScope = $scope.$new();

$scope.$digest()

Immediately invokes all of a scope's watchers and dirty checking code in order to update any two-way data bindings. Will propagate checks into child scopes. Usually only used in testing code when there's a need to instantly simulate scope life cycles.

$scope.$apply(expression||fn)

By registering a piece of code via a scope's $apply method we indicate to Angular that it is likely to result in changes to a scope's data a model and therefore will require a future call to $digest.

Whenever we interact with Angular's core services and directives the framework will be wrapping our activities in $apply behind the scenes when needed, so we rarely have to call it explicitly. The only time we need to call it explicitly is when we make changes to a scope's model and Angular can't be reasonably expected to notice right away:

setTimeout(function () {
    $scope.$apply(function () {
        $scope.counter = $scope.counter + 1;
    });
},2000);

$scope.$watch(expression||fn,onChange)

Watches a scope value for changes. Fires the onChange listener when a change is detected. The watched value can either be defined by a string expression or a function that returns the actual value to be watched.

$scope.name = "Jack";
$scope.$watch("name",function () {
    // Name has changed
});

Views

Views are live portions of the DOM, defined by valid HTML templates and bound to Angular components via special attributes and expressions in the markup.

Views are glued to their respective controllers and directives via their shared $scope instance, the two-way data binding that Angular sets up between the view's DOM and the model values.

Note: Importantly views are declarative. They are focussed on describing what should happen rather how it should happen. The data model manipulation and mutation logic for UI behaviour should live in the view's controller, and any DOM manipulation code should live in directives.

View expressions

Expressions are simple JavaScript-like code snippets inside view templates that operate within the data context provided by the current scope. They can access scope values, call scope methods or perform simple operations.

Expressions are automatically re-evaluated and re-rendered when the related scope value changes. Angular uses an efficient "dirty checking" algorithm to achieve this.

Render a scope value in a template:

<span>{{user.firstname}}</span>

Render the result of a scope method call in a template:

<span>Remaining issues: {{remainingIssues()}}

Perform a simple operation:

<span>{{2+2}}</span>

Managing template files

Templates can be referenced in a number of ways:

  1. Via the ng-include directive in another template
  2. Mapped to url changes via the $routeProvider service
  3. Via the templateUrl option in a custom directive

A template can either be in the form of a <script> tag (which must be placed somewhere in the DOM as a child of ng-app):

<script type="text/ng-template" id="/templates/foo.html">
    Template contents.
</script>

Or simply an external .html file that is lazy loaded on demand. In either case once a template <script> tag has been parsed, or an external file has been loaded, the template is thereafter cached (in the $templateCache service) and available instantly.

In a large app there may be many tens or hundreds of external template files. There are Grunt tasks that help with concatenating such sets of files into the main app JavaScript bundle.

Unprocessed template display

Under some conditions views could briefly show their unprocessed {{}} expressions to the user while data is being gathered.

The ng-cloak directive can be used to hide portions of the DOM while Angular is waiting for scope data to be processed and then automatically shown when ready:

<div ng-controller="MyCtrl" ng-cloak>
       <h1>Hello, {{name}}!</h1>
</div>

An alternative approach is to place the expression in an ng-bind directive instead:

<div ng-controller="MyCtrl">
       <h1>Hello, <span ng-bind="name"></span>!</h1>
</div>

Directives

Directives are reusable blocks of Angular code/behaviour that can be attached to the live DOM via markers in the HTML. When Angular "compiles" the template HTML it will attempt to match directives with DOM elements based on these markers.

Directives are commonly used to attach specific behaviour to a DOM element or transform a DOM element's (and potentially its children's) structure.

Angular provides are large library of useful directives as part of the core framework, in addition you can define more application specific custom directives yourself.

Note: Directive markers can be placed in HTML using a number of methods (comments, tag names, classes and attributes). The recommended approach however is to use attributes in a dash delimited format.

Simple core directive examples

ngClick

Evaluates an expression when an element is clicked.

<button ng-click="onBtnClicked()">Click me</button>

ngRepeat

Repeats the current element based on an expression.

<li ng-repeat="user in users">
    {{user.name}}
</li>

ngClass

Adds class names to an element if an expression in an object hash of potential class names evaluates to true.

<span ng-class="{'warning':shouldWarn()}"></span>

Custom directives

Custom directives can be defined by returning a directive definition object from a factory function. The options object instructs the Angular compiler how to link a HTML template or DOM portion together with a scope in order to create a functioning directive.

angular.module("app")
    .directive("myDirective",function () {
        return {
            // Directive definition options
        };
    });

The directive definition object can take many options which are detailed here, but the below is a brief discussion about some of the more important ones.

scope

This value defines how directive scope data (that is, attribute values set in the HTML template and/or values in the matched $scope) are exposed to the directive.

The default value is false/null, which means the directive shares the current scope.

A value of true creates a new scope for the directive that prototypically inherits from the current scope.

An object hash {} can define various patterns for setting up data binding between the current scope and an otherwise new/isolated directive scope. Useful when creating reusable directives that should be encapsulated from the surrounding data context.

Patterns available in the scope object hash:

@ Attribute string expression binder

Binds a local property to the string result of an expression in an attribute the outer scope. Live changes are propagated into local scope. Example:

<div my-directive my-attr="{{username}}"></div>
scope: {
    localName: "@myAttr"
}
= Two way scope property binder

Two way binding between a local scope property and an outer scope property as defined by an attribute. Any change to the property in one scope will be reflected in the other.

<div my-directive my-attr="username"></div>
scope: {
    localUsername: "=myAttr"
}
& Expression invoker

Provides a mechanism for executing expressions in the context of the outer scope. Arguments can be passed via an object hash.

<div my-directive my-attr="increment(count)"></div>
scope: {
    increment: "&myAttr"
}
// Then later on in the directive:
$scope.increment({count:1});
? Optional flag

Adding the optional ? flag will avoid errors if the property doesn't exist in the parent scope, for example =?myAttr.

Note: by omitting the attribute name in the scope option object hash value string we indicate that the target attribute name is the same as the local isolate scope property name, e.g. scope: { prop: "=" } would bind to an attribute prop in the markup, such as <div my-directive prop="username"></div>.

restrict

Defines what sort of declaration style when encountered in the markup should be used to invoke the directive. Takes the form of a string containing a subset of the letters EACM - the presence of a letter allows the associated style. The default value is A.

For example, a directive named myDirective could be declared in the markup using the following styles:

  • Attribute <div my-directive></div> (A)
  • Element <my-directive></my-directive> (E)
  • Class <div class="my-directive"></div> !deprecated (C)
  • Comment <!-- directive:my-directive --> !deprecated (M)

Note: Favour the default A attribute style of declaration, especially when decorating an existing element with extra behaviour. The E element name style can be considered when creating custom elements unique to your app.

template/templateUrl

Replaces, or appends to, the current element with the HTML in the template, either specified as a HTML string in the case of template or as a path to an external template in the case of templateUrl. All attributes and classes are migrated from the original element onto the new one.

link

Define a function with a signature fn (scope,elem,attrs,ctrl). Used to attach DOM event listeners and manipulate the DOM based on UI interaction and changes in the scope model. Most directive logic goes here.

controller

Define a controller constructor function. The controller can be shared amoung directives thereby facilitating inter-directive communication.

Custom directive examples

Template expansion

A simple use for custom directives can be template "expansion. Conceptually similar to partials this sort of directive can be used to avoid repeating often used chunks of HTML.

For example, a directive to consistently render a block of HTML related to customer details. Directive definition:

angular.module("app")
    .directive("customer-details",function () {
        return {
            templateUrl: "/templates/customer-details.html"
        }
    });

The HTML:

<div customer-details></div>

The template:

<p>Name: {{customer.name}}</p>
<p>Address: {{customer.address}}</p>

Scope isolation

It's often good practice to isolate directive scopes and reduce their coupling to the surrounding HTML structure and data context. We need to map properties in the outer scope onto local properties in a isolated scope inside the directive.

angular.module("app")
    .controller("MyCtrl",function ($scope) {
           $scope.personA = "Jack";
           $scope.personB = "Jill";
    })
    .directive("myDirective",function () {
        return {
            template: "<p>{{localPerson}}</p>",
            scope: {
                localPerson: "="
            }
        }
    })

The HTML:

<div my-ctrl>
    <div my-directive local-person="personA"></div>
    <div my-directive local-person="personB"></div>
</div>

Here we are mapping value of the local-person attribute to the localPerson value in the directive's isolated scope. The value of the attribute is a property name expression that resolves to a value in the parent controller's scope.

In this way two way data binding can be set up between a parent scope context and an encapsulated isolated directive scope. Note that the actual detail of the mapping is defined declaratively in the template.

DOM interaction

Typically when manipulating the directive's DOM and attaching DOM event listeners we use the link function option. The link function takes the following parameters:

  • scope The Angular scope instance for the directive
  • element The element matched by the directive, wrapped in jqLite
  • attrs Object hash of normalised element attribute names and values

The link function is executed after Angular has processed the template and wired up the DOM, so its safe to transform the DOM and work with the scope.

angular.module("app")
    .directive("myDirective",function () {
        return {
            link: function ($scope,element,attrs) {
                // Manipulate the DOM:
                element.text("foobar");
                // Watch an attribute expression for changes:
                $scope.$watch(attrs.fooAttribute,function (value) {});
                // Listen to a DOM event:
                element.on("mousedown",function (e) {});
                // Respond to the element being destroyed:
                element.on("$destroy",function () {});
            }
        }
    });

Transcluding directives

"In computer science transclusion is the insertion of a document into another document at a particular point." - Wikipedia

In terms of Angular directives, transclusion describes the process of a directive wrapping the contents of the matched element with the result of its own compiled template. In other words, the matched element's contents is extracted and inserted into the directive's template at a particular point.

Consider a use case where arbitrary content should be wrapped in a HTML structure designed to turn it into an alert-style popup.

The original markup:

<div alert-popup>
    <p>Alert! Something happened you should know about.</p>
</div>

The directive:

angular.module("app")
    .directive("alertPopup",function () {
        return {
            transclude: true,
            template: "<div class='alert' ng-transclude></div>"
        }
    });

The resulting processed DOM:

<div alert-popup>
    <div class="alert" ng-transclude>
        <p>Alert! Something happened you should know about.</p>
    </div>
</div>

Note: the ng-transclude directive in the template defines the position at which the transclusion should be inserted.

Inter-directive cooperation

Directives can communicate with other directives in the same scope. When directives depend on one another we can use the require option to enforce their presence. An example of a cooperating set of directives would be a tab-based navigation managing a set of content panes.

Controllers

Controllers are constructor functions for JavaScript classes that are used to augment a view's scope with behaviour related to its particular data model.

Controllers are implemented by first preparing them for DI as a module component:

angular.module("app")
    .controller("MyCtrl",function ($scope) {
        $scope.salutation = "Hello";
        $scope.salutationTarget = "world";
        $scope.num = 2;
        $scope.double = function (value) {
            return value*2;
        }
    });

And then referencing them in a view via the ng-controller directive:

<div ng-controller="MyCtrl">
    <p>{{salutation}} {{salutationTarget}}</p>
    <p>{{num}} doubled is {{double(num)}}</p>
</div>

Controller dos and don'ts

  • DO make them slim. They should only contain simple, testable logic related only to the model data context for a single view.
  • DO inject controllers with services if they need access to any shared state or functionality.
  • DON'T manipulate the DOM in a controller, use the automatic data binding or core Angular directives library instead, or if you need to go further custom directives.
  • DON'T filter output directly, use Angular filters.

Filters

Create data transformation "pipelines" either in template expressions or other app components. A set of basic filters are provided by the framework but additional custom ones can be defined.

In views

For example in a template expression to limit a string to 80 chars and then transform it to all lowercase:

<p>{{myString | limitTo:80 | lowercase}}</p>

To format an object as JSON and print it out:

<pre>{{obj | json}}</pre>

Filters can also be applied to arrays, for example when reducing the elements outputted by the ng-repeat directive. In the code below only model items that contain an active property equal to true will be rendered.

<li ng-repeat="item in items | filter{active:true}">
    {{item.firstName}} {{item.lastName}}
</li>

In controllers

When using a filter directly in a controller then its DI name is formed from the pattern <filter-name>Filter, for example to inject a filter called bar:

angular.module("foo")
    .controller("FooCtrl",function ($scope,$barFilter) {
        $scope.filteredValue = $barFilter(valueToFilter);
    });

As a rule its better to define complex filtering logic in a controller since this can be tested in isolation away from the DOM.

Custom filters

Custom filters can also be defined via the filter method. The below is a simple custom filter for paginating an array:

angular.module("app")
    .filter("paginate",function () {
        return function (array,page,length) {
            var start = page*length;
            return array.slice(start,start+length);
        };
});

Filter dos and don'ts

  • DO make them fast and efficient, they are typically "hot" code paths
  • DO make them idempotent. Calling them multiple times shouldn't have side effects and they shouldn't modify passed-in data directly
  • DON'T return HTML markup from filters, they should just operate abstractly on data

Promises

Angular exposes a lightweight Promise/Deferred implementation via its core $q service.

// Generate an instance of Deferred:
var deferred = $q.defer();

// Access the contained Promise instance:
var promise = deferred.promise;

// Manipulate the Deferred instance to resolve
// or reject the Promise:
deferred.resolve(value);
// or deferred.reject(reason);

// Use the promise instance to asynchronously
// respond to the resolution/rejection:
promise.then(onResolve,onReject,onNotify);

Promise.then

Attaches callbacks for promise resolution, rejection and notifications.

Multiple then calls can attach multiple sets of callbacks that are invoked in order.

The return value of a then call is a new promise (enables promise chaining). The resolution callback should return the resolution value of the new promise, or in the case of a failure the rejection callback should return a new rejected promise with a reason.

function onResolve (res) {
    return res;
}

function onReject (reason) {
    if ( recoverable ) {
        return newPromiseOrValue;
    } else {
        return $.reject(reason);
    }
}

promise.then(onResolve,onReject);

HTTP

The $http service is the core Angular utility for communicating with a backend.

By default all $http request data objects are transformed into JSON and all responses are transformed into objects, if possible.

Requests

Requests can be made with the HTTP method shortcuts:

$http.get("/api/v1/endpoint");

Or by passing in a more detailed configuration object:

$http({
    method: "GET",
    url: "/api/v1/endpoint",
    params: {key:"value"},
    data: {}
});

Responses

Responses are exposed by Angular's $q promise implementation:

$http.get("/api/v1/endpoint")
    .then(onSuccess,onError);

The $http service also provides the success and error methods on top of the above promise API:

$http.get("/api/v1/endpoint")
    .success(function (data,status,headers,config) {
    })
    .error(function (data,status,headers,config) {
    });

Interceptors

Any global error handling, authentication, pre or post-processing of requests or responses can be done via interceptors.

Interceptors slot functionality into the promise chaining structure of the $http service.

An example to globally detect a 403 response:

angular.module("http.interceptors.auth")
    .factory("http403Interceptor",function ($q) {
        return {
            "responseError": function (res) {
                if ( res.status === 403 ) {
                    // Display error?
                    // Redirect to login?
                    // Etc
                }
                $q.reject(res);
            }
        }
    });
angular.module("http.interceptors.auth")
    .config(function ($httpProvider) {
        $httpProvider.interceptors.push("http403Interceptor")
    });

HTTP resources

When working with RESTful APIs we can abstract some of our HTTP layer using the $resource service. At the expense of some flexibility this enables us to quickly define a clientside representation of a backend resource and then interact with it with a basic CRUD API.

For example, to define a service for interacting with an API endpoint representing app users:

angular.module("resources.users")
    .factory("Users",function ($resource) {
        var Users = $resource("/users/:id");
        return Users;
    });

And then to use it:

angular.modules("userlist")
    .controller("UserListCrtl",function ($scope,Users) {
        // Attach an array of all users to the scope, the view
        // will auto update when the returned array is populated:
        $scope.allUsers = Users.query();
        // Get a particular user, will be an instance of Users:
        var user = Users.$get({id:12345});
    });

If additional flexibility is required beyond that provided by $resource then totally custom REST services can be quite easily built up using the lower level $http service.

Routing

The vast majority of app routing can be handled by Angular's core ngRoute service.

Configuring routes

Routing can be configured via $routeProvider:

angular.module("router")
    .config(function ($route,$routeProvider,$locationProvider) {
        $locationProvider.html5Mode = true;
        $routeProvider
            .when("/",{
                templateUrl: "/templates/home.html",
                controller: "HomeCtrl"
            })
            .when("/users/:id",{
                templateUrl: "/templates/user-detail.html",
                controller: "UserDetailCtrl",
                resolve: {
                    userData: function (UserService) {
                        return UserService.getUserData($route.current.params.id);
                    }
                }
            })
            .when("/bad-path",{
                redirectTo: "/good-path"
            })
            .when("/good-path",{
                templateUrl: "/templates/good-path.html"
            })
            .when("/404",{
                templateUrl: "/templates/404.html"
            })
            .otherwise({
                redirectTo("/404")
            })
    });

Everything should be self explanatory in the above code example, except perhaps the resolve option. This defines an object hash of dependencies for the route controller which return promises that should be resolved before rendering the route - for example promises that load essential API data for the route.

The controller/template combinations defined in the route configuration will be rendered into the DOM whenever the route changes at the location of the ng-view directive, of which there should be only one in your markup. For example:

<body>
    <nav></nav>
    <div class="main-container" ng-view></div>
    <footer></footer>
</body>

The UserCtrl for the above routes configuration might look something like:

app.module("user-detail")
    .controller("UserCtrl",function ($scope,userData) {
        $scope.userData = userData;
    });

Forms

To facilitate working with forms Angular actually defines a set of element name directives that match HTML's form input elements <input>, <select>, <textarea> etc. What this means is that when we use a standard form input element with Angular the framework is overlaying additional directive-based functionality behind the scenes.

The goal of extending form elements with directives is to de-couple the form data as displayed to the user from the form data as held in the attached scope's data model yet still retain the flexibility of Angular's two-way data binding.

The ng-model directive links a form element's value to a scope model value in a two-way binding. In the example below the model value $scope.user.firstname is bound to the text input's value - a user initiated change to the input's value in the DOM will change the scope value, and vice versa:

<label>First name</label>
<input type="text" ng-model="user.firstname">

In this way we can maintain an accurate representation of the form's data state in the scope model and easily pass it into our app, for example to use a HTTP service to save it to a server.

Accessing form state

By assigning name attribute's to both the form element and its child elements we can access information about their state in the current scope.

<div ng-controller="UserFormCtrl">
    <form name="userForm">
        <input type="text" name="firstname" ng-model="user.firstname">
    </form>
</div>

With the form element name attributes thusly populated we expose values such as $scope.userForm and $scope.userForm.firstname on the scope. These objects are actually references to controllers created by the framework and contain values that indicate an element's state:

  • $valid Is the element in a valid state
  • $invalid Is the element in an invalid state
  • $pristine Is the element in an unedited state
  • $dirty Is the element in an edited state
  • $errors An object hash of boolean values indicating specific validation errors, e.g. $errors.email indicates email validation failure

Note: Angular will also attach appropriate CSS classes indicating these states to the elements, for example .ng-valid, .ng-dirty etc.

Form data model transformations

It is often necessary to transform form model values. For example, a server might expect a date to be sent as a Unix timecode but such data is more appropriately displayed to the user in a DD/MM/YYYY format.

Such transformations happen in two directions. As the data model is formatted for display in the DOM, and DOM values are parsed for setting in the data model.

The ng-model directive exposes its controller to the scope to facilitate such transformations. Specifically the controller's $parsers and $formatters arrays, which can contain transformation pipelines.

These pipelines are built up by pushing functions into each array in a custom directive.

angular.module("user-form")
    .directive("date",function ($filter) {
        return {
            restrict: "A",
            require: "?ngModel",
            link: function (scope,element,attrs,ngModel) {
                if ( !ngModel ) return;
                ngModel.$formatters.push(function (value) {
                    return $filter("date")(value,"dd/mm/yyyy");
                });
                ngModel.$parsers.push(function (value) {
                    var parts = value.split("/");
                    var date;
                    // some code to convert parts array to Date etc.
                    return date.getTime()/1000;
                });
            }
        }
    });
<label>Start date</label>
<input type="text" name="startDate" ng-model="startDate" date>

Validation

Angular provides an array of additional directives that we can use to define validation rules on form elements. In fact, some of these directives are named the same as their HTML relatives - for example, adding type="email" to an <input> element will cause Angular's email validation directive to kick in.

It is for this reason that it is recommended to add the novalidate attribute to Angular forms to prevent HTML5's in-built form validation to kick in and potentially cause conflicts.

By using these directives and the way they expose their state onto the scope we can build up fairly richly interactive forms simply using view expressions.

<div ng-controller="UserFormCtrl">
    <form name="userForm" ng-submit="submit()" novalidate>
        <input type="email" name="email" ng-model="user.email">
        <span ng-show="userForm.email.$errors.email">Email is invalid</span>
        <input type="submit" ng-disabled="userForm.$invalid">
    </form>
</div>

Here we can see that a boolean value has been exposed at $scope.userForm.email.$errors.email indicating the validity of the input as an email address. If we had applied an ng-required directive we could expect to find that piece of validation state at $scope.userForm.email.$errors.required.

Available validation directives include:

  • ng-required
  • ng-minlength
  • ng-maxlength
  • ng-pattern
  • ng-min
  • ng-max

Custom validation

Custom validation is achieved in a similar way to data transformations. That is, by accessing the API exposed by ngModel's controller via a custom directive.

The example directive below will validate an input as an integer. Validation results are exposed on the $scope.formName.elementName.$errors object in the same way as the core validation directives.

angular.module("user-form")
    .directive("integer",function ($filter) {
        return {
            restrict: "A",
            require: "?ngModel",
            link: function (scope,element,attrs,ngModel) {
                if ( !ngModel ) return;
                ngModel.$parsers.push(function (value) {
                    // Pseudo code to test value as integer
                    if ( value === integer ) {
                        ngModel.$setValidity("integer",false);
                        return value;
                    } else {
                        ngModel.$setValidity("integer",false);
                        return undefined;
                    }
                });
            }
        }
    });
<label>Age</label>
<input type="text" name="userAge" ng-model="userAge" integer>

Nested forms

Nested <form> tags are not valid HTML, however Angular allows nested forms via the <ng-form> element directive. In this way reusable sub-forms can be created (for example, a password and confirm password input field set) and nested into parent forms. By applying name attributes to the sub forms we can access their properties from the scope.

Testing

Unit tests

Unit testing means testing individual components in isolation. Angular's dependency injection system strongly encourages writing highly encapsulated, decoupled components with very limited concerns that are easy to test.

It is common to write Angular unit tests in Jasmine and run them via the Karma test runner. Karma will dynamically generate HTML test suite pages based on your tests and run them via web sockets on local browsers and/or PhantomJS.

Although Angular is well suited to testing, there are varied approaches to writing unit tests for the various different types of components. What's more, writing tests will often involve using some features of the core framework that are rarely used in your standard application code - for example, the $injector and $compile services and Scope.$digest etc. That said, there's no "right" way to write an Angular unit test - as long as the test runs and tests something sensible then it's doing its job.

But be careful writing unit tests - don't drink the TDD kool aid. It's very easy to spend time crafting self-referential/tautological tests that don't add any value. For example, you have a function that sets x to 5, and then write a test confirming that the function sets x to 5. This unit test will likely never fail in the life time of the app, and therefore is just fluff. Try to write tests that test complex behaviours and algorithms.

Anatomy of a Jasmine test

Jasmine tests are composed from suites, specs and expectations.

// The 'describe' function creates a suite:
describe("A suite",function () {

    // The 'it' function creates a spec:
    it("a spec",function () {
        // The 'expect' function works with Jasmine matchers to
        // create an expectation:
        expect(true).toBeTruthy();
    });

    // Suites can be nested:
    describe("A nested suite",function () {
        // Etc.
    });
});

We can use the beforeEach and afterEach functions to define code that should be run before and after every spec in a suite:

describe("A suite",function () {

    // 'beforeEach' will be run before every spec. It shares its
    // 'this' context with the spec. The 'this' context will be
    // destroyed and recreated each time 'beforeEach' runs.
    beforeEach(function () {
        this.foo = true;
    });

    // `beforeEach` can also be used to initialise values in the
    // outer scope for use in each spec.
    var bar;
    beforeEach(function () {
        bar = false;
    });

    it("a spec",function () {
        expect(this.foo).toBeTruthy();
        expect(bar).toBeFalsy();
    });
});

Some further examples of expectations:

  • toBe Compares with ===
  • toEqual Compares with ==
  • toBeDefined Compares with undefined
  • toBeUndefined Compares with undefined
  • toBeNull Compares with null
  • toBeTruthy
  • toBeFalsy
  • toBeLessThan Compares numerical magnitude
  • toBeGreaterThan Compares numerical magnitude
  • toContain Checks if an element is present in an array
  • tothrow Checks if a function throws an error
  • not Inverts an expectation, expect(true).not.toBeFalsy

ngMock

Karma dynamically builds the test suite HTML page and includes functionality provided by the ngMock service. This exposes a number of useful test helpers:

window.module()

Loads modules in preparation for a test. The following code will load the projects-list and projects-service modules and their dependencies in preparation for a test.

beforeEach(function () {
    module("projects-list");
    module("projects-service");
});

When testing a component in isolation sometimes it's necessary to mock up one of its dependencies. That is, create a fake version of a dependency with stub methods and inject it into the component under test. This would enable us to be sure that any test failures are the fault of the component under test rather than a dependency.

We can use window.module() to register such a mock. The following code would ensure that whenever the projectsService component is requested for dependency injection during the test the mock object would be delivered instead:

beforeEach(function () {
    module("projects-list");
    module("projects-service");
    module({
        "projectsService": {
            getUsers: function () {
                return [
                    {user:"Jack",id:1},
                    {user:"Jill",id:2}
                ];
            }
        }
    });
});
window.inject()

Creates an instance of the $injector service to use in each test. The $injector will be prepped with the modules defined in an earlier call to window.module(). Use it to inject your test subjects into the tests.

describe("projectsService",function () {

    beforeEach(function () {
        module("project-service");
    });

    it("getUsers should return an array",inject(function (projectsService) {
        expect(typeof projectsService.getUsers()).toEqual("array");
    }));
});

Alternatively, instead of injecting the dependencies into every test spec we can move the injection to the beforeEach function. Useful if we have many it test specs that are all interested in the same test subject.

describe("projectsService",function () {

    var projectsService;

    beforeEach(function () {
        module("project-service");
        inject(function ($injector) {
            projectsService = $injector.get("projectsService");
        });
    });

    it("getUsers should return an array",function () {
        expect(typeof projectsService.getUsers()).toEqual("array");
    });
});
Mock $httpBackend

When ngMock is present the normal version of $httpBackend will be replaced with a special mock version that allows us to simulate fake async responses from a live API in our tests. When our test subjects make calls such as $http.get("/api/v1/users") we can intercept them and return dummy data. From the point of view of the test subject nothing has changed in the implementation.

$httpBackend.when(method,url[,data,headers])

Mocks an API end point along with a response.

$httpBackend.when("GET","/api/v1/users").respond([
    200,
    [
        {user:"Jack",id:1},
        {user:"Jill",id:2}
    ],
    {"Content-Type":"application/json; charset=UTF-8"}
]);
$httpBackend.expect(method,url[,data,headers])

Similar to $httpBackend.when although it implies an expectation that the call should be made sometime during the test. Use the verifyNoOutstandingExpectation and verifyNoOutstandingRequest methods in a afterEach block to ascertain if the expected calls were made.

$httpBackend.flush([count])

Upon calling flush the $httpBackend mock will immediately send all the pending fake responses. In this way async API calls can be tested synchronously.

Testing controllers

Well written controllers should end up being pretty slim, and confine themselves to just initialising a view's Scope and providing a simple API for its manipulatinon. This means they should be easily testable in isolation away from the DOM. When testing a controller it needs to be manually created along with a new Scope instance.

describe("The suite",function () {

    var createController;

    beforeEach(inject(function ($rootScope,$controller) {
        module("some-ctrl")
        createController = function () {
            return $controller("SomeCtrl",{
                $scope: $rootScope.$new()
            });
        }
    }));

    it("the spec",function () {
        var ctrl = createController();
        expect(ctrl.$scope.foo).toEqual("bar");
        ctrl.$scope.setFooToBaz();
        expect(ctrl.$scope.foo).toEqual("baz");
    });
});

Testing HTTP services

Testing HTTP services involves making use of the $httpBackend mock discussed above. Here's an example:

describe("projectsService",function () {

    var projectsService;
    var $httpBackend;

    beforeEach(function () {
        module("projects-service");
        inject(function ($injector) {
            projectsService = $injector.get("projectsService");
            $httpBackend = $injector.get("$httpBackend");
        });
        $httpBackend.when("GET","/api/v1/users").respond([
            200,
            [
                {user:"Jack",id:1},
                {user:"Jill",id:2}
            ],
            {"Content-Type":"application/json; charset=UTF-8"}
        ]);
    });

    it("getUsers should return an array",function () {
        var users;
        projectsService.getUsers().then(function (res) {
            users = res;
        });
        $httpBackend.flush();
        expect(users).toEqual("array");
    });
});

Testing directives

Testing directives can be a bit more complex since they rely on the DOM.

The basic flow for setting up such a test is:

  1. Compile a piece of HTML that references the directive.
  2. Bind the compiled HTML to a scope to create an isolated view in the test.
  3. Run $scope.$digest() on the view to force Angular to process the directive and any expressions/watchers.
  4. Run tests on the resulting HTML, i.e. see if the content has been correctly transformed or the correct CSS classes have been applied etc.
describe("someDirective",function () {

    var $compile;
    var $rootScope;
    var scope;
    var view;

    beforeEach(function () {
        module("some-directive");
        inject(function ($injector) {
            $compile = $injector.get("$compile");
            $rootScope = $injector.get("$rootScope");
        });
        scope = $rootScope.$new();
        view = $compile("<div some-directive></div>")(scope);
        scope.$digest();
    });

    it("add the correct contents",function () {
        expect(view.html()).toContain("foobar");
        expect(view.hasClass("baz")).toEqual(true);
    });
});

Note: If the directive uses its own template via templateUrl then we also need to make sure that the $templateCache service has been primed with the template in question. This can be achieved by modularising your templates with a Grunt task such as html2js.

End-to-end ("e2e") tests

The purpose of e2e tests are to test whether your controllers, services, directives and templates are all working together as expected to create a functional site when deployed on a live server with a working backend.

Often the flow of such tests will simulate a typical user journey. For example:

  1. Navigate to "/login"
  2. Enter username "Jack" and password "12345" into the form
  3. Click the log in button
  4. Has the user been redirected to "/"?
  5. Has the backend set a valid session cookie?

By running test scenarios like this automatically against a range of different browsers the need for manual testing can often be dramatically reduced down to a simple visual inspection.

Note: the term "e2e" testing is used somewhat synonymously with the terms "integration" test and "systems" test. The distinctions don't seem to be well defined and shift between different languages and development tasks.

Angular e2e tests are usually written in Jasmine and use the Protractor test runner. Protractor uses WebDriverJs to remote control browsers running on Selenium.

Selenium is a browser automation server. You can run it locally, or on a remote server. It can automate a variety of different browser versions and brands. SauceLabs provide a hosted Selenium service.

Protractor tests are written in Jasmine, with some extra global objects on window to facilitate driving the remote browser.

Protractor syntax

window.browser

Drives the main browser window, exposes methods such as get for navigating. Wraps the WebDriverJs driver object.

window.element()

A helper function for interacting with elements.

window.by()

A helper function for locating elements on the page, for example by CSS selector or id. Also enables locating via Angular specific syntax, such as ng-model attribute values etc.

window.protractor

The main Protractor namespace, contains helpful static variables and classes.

Jasmine/Protractor example

describe("Login",function () {

    var username = element(by.name("username"));
    var password = element(by.name("password"));
    var loginBtn = element(by.css(".login-btn"));

    beforeEach(function () {
        browser.manage().deleteAllCookies();
        browser.get("/login");
        email.sendKeys("Jack");
        password.sendKeys("12345");
        loginBtn.click();
    });

    it("should redirect to home",function () {
        expect(browser.getCurrentUrl()).toEqual("/");
    });

    it("should set the auth cookie",function () {
        expect(browser.manage().getCookie("auth_txt")).toBeDefined();
    });
});

Note: although the syntax looks synchronous WebDriverJs is actually adding each line to a queue of promises and executing them asynchronously.