Skip to content
11675 lines (9211 sloc) 642 KB

1.5.0 ennoblement-facilitation (2016-02-05)

Bug Fixes

Features

Breaking Changes

  • ngAria: due to d06431e5, Where appropriate, ngAria now applies ARIA to custom controls only, not native inputs. Because of this, support for aria-multiline on textareas has been removed.

1.5.0-rc.2 controller-requisition (2016-01-28)

Deprecation Warning

  • The ngTouch module's ngClick directive has been deprecated and disabled by default. See the breaking changes section for more information

Bug Fixes

Features

  • $compile:
    • allow required controllers to be bound to the directive controller (56c3666f, #6040, #5893, #13763)
    • allow directive definition property require to be an object (cd21216f, #8401, #13763)
    • call $onInit on directive controllers after all sibling controllers have been constructed (3ffdf380, #13763)
  • $locale: include original locale ID in $locale (63492a02, #13390)
  • $resource: add support for timeout in cancellable actions (d641901b, #13824)

Performance Improvements

  • $compile: avoid needless overhead when wrapping text nodes (92e4801d)
  • ngAnimate: speed up areAnimationsAllowed check (683bd92f)

Breaking Changes

The $sanitize service will now remove instances of the usemap attribute from any elements passed to it.

This attribute is used to reference another element by name or id. Since the name and id attributes are already blacklisted, a sanitized usemap attribute could only reference unsanitized content, which is a security risk.

The ngClick override directive from the ngTouch module is deprecated and disabled by default. This means that on touch-based devices, users might now experience a 300ms delay before a click event is fired.

If you rely on this directive, you can still enable it with the $touchProvider.ngClickOverrideEnabled()method:

angular.module('myApp').config(function($touchProvider) {
  $touchProvider.ngClickOverrideEnabled(true);
});

Going forward, we recommend using FastClick or perhaps one of the Angular 3rd party touch-related modules that provide similar functionality.

Also note that modern browsers already remove the 300ms delay under some circumstances:

  • Chrome and Firefox for Android remove the 300ms delay when the well-known <meta name="viewport" content="width=device-width"> is set
  • Internet Explorer removes the delay when touch-action css property is set to none or manipulation
  • Since iOs 8, Safari removes the delay on so-called "slow taps"

See this article by Telerik for more info on the topic.

Note that this change does not affect the ngSwipe directive.

1.4.9 implicit-superannuation (2016-01-21)

Bug Fixes

Minor Features

Performance Improvements

  • ngAnimate: speed up areAnimationsAllowed check (2d3303dd)

Breaking Changes

While we do not deem the following to be a real breaking change we are highlighting it here in the changelog to ensure that it does not surprise anyone.

Possible breaking change for users who updated their code to provide a timeout promise for a $resource request in version v1.4.8.

Up to v1.4.7 (included), using a promise as a timeout in $resource, would silently fail (i.e. have no effect).

In v1.4.8, using a promise as timeout would have the (buggy) behaviour described in https://github.com/angular/angular.js/pull/12657#issuecomment-152108887. (I.e. it will work as expected for the first time you resolve the promise and will cancel all subsequent requests after that - one has to re-create the resource class. This was not documented.)

With this change, using a promise as timeout in v1.4.9 onwards is not allowed. It will log a warning and ignore the timeout value.

If you need support for cancellable $resource actions, you should upgrade to version 1.5 or higher.

1.5.0-rc.1 quantum-fermentation (2016-01-15)

Features

Bug Fixes

Breaking Changes

  • $component: These breaking changes affect only applications updating from previous 1.5 beta / rc versions

    • Due to d91cf167, the default controllerAs value for components is now $ctrl (previously the name of the component was used). To migrate, either set controllerAs to the component name, or change the property name in your templates to $ctrl

    • Due to 25bc5318, it is no longer possible to set the restrict option on directives created via the module.component() helper. All components are now element directives (restrict: 'E'). If you need a directive that is not an element then you must use the module.directive() helper instead.

    • Due to f31c5a39, components are now always created with scope: {} (isolate scope). Previously, it was also possible to create components with scope: true or scope: false. If your components rely on this scope configuration, you will have to create a regular directive instead.

    • Due to 6a47c0d7, the transclude property is now false by default (previously true). If you created components that expected transclusion then you must change your code to specify transclude: true.

  • linky: due to 98c2db7f,

Before this change, the filter assumed that the input (if not undefined/null) was of type 'string' and that certain methods (such as .match()) would be available on it. Passing a non-string value would most likely result in a not-very-useful error being thrown (trying to call a method that does not exist) or in unexpected behavior (if the input happened to have the assumed methods).

After this change, a proper (informative) error will be thrown. If you want to pass non-string values through linky, you need to explicitly convert them to strings first. Since input values could be initialized asynchronously, undefined or null will still be returned unchanged (without throwing an error).

1.5.0-rc.0 oblong-panoptikum (2015-12-09)

This is the first Release Candidate for AngularJS 1.5.0. Please try upgrading your applications and report any regressions or other issues you find as soon as possible.

Features

Bug Fixes

Breaking Changes

This is only a breaking change to a feature that was added in beta 2. If you have not started using multi-slot transclusion then this will not affect you.

The keys and values for the transclude map of the directive definition have been swapped around to be more consistent with the other maps, such as scope and bindToController.

Now the key is the slot name and the value is a normalized element selector.

Using a promise as timeout is no longer supported and will log a warning. It never worked the way it was supposed to anyway.

Before:

var deferred = $q.defer();
var User = $resource('/api/user/:id', {id: '@id'}, {
  get: {method: 'GET', timeout: deferred.promise}
});

var user = User.get({id: 1});   // sends a request
deferred.resolve();             // aborts the request

// Now, we need to re-define `User` passing a new promise as `timeout`
// or else all subsequent requests from `someAction` will be aborted
User = $resource(...);
user = User.get({id: 2});

After:

var User = $resource('/api/user/:id', {id: '@id'}, {
  get: {method: 'GET', cancellable: true}
});

var user = User.get({id: 1});   // sends a request
user.$cancelRequest();      // aborts the request

user = User.get({id: 2});

The $sanitize service will now remove instances of the <use> tag from the content passed to it.

This element is used to import external SVG resources, which is a security risk as the $sanitize service does not have access to the resource in order to sanitize it.

A new property to access route resolves is now available on the scope of the route. The default name for this property is $resolve. If your scope already contains a property with this name then it will be hidden or overwritten.

In this case, you should choose a custom name for this property, that does not collide with other properties on the scope, by specifying the resolveAs property on the route.

A new property to access all the locals for an expression is now available on the scope. This property is $locals.

  • If scope.$locals already exists, the way to reference this property is now this.$locals.
  • If the locals themselves include a property $locals then the way to reference that is now $locals.$locals.

1.4.8 ice-manipulation (2015-11-19)

Bug Fixes

Performance Improvements

  • $compile: use static jquery data method to avoid creating new instances (55ad192e)
  • copy:
    • avoid regex in isTypedArray (19fab4a1)
    • only validate/clear if the user specifies a destination (d1293540, #12068)
  • merge: remove unnecessary wrapping of jqLite element (ce6a96b0, #13236)

Breaking Changes

1.5.0-beta.2 effective-delegation (2015-11-17)

Bug Fixes

Features

Performance Improvements

  • $compile: use static jquery data method to avoid creating new instances (9b90c32f)
  • $interpolate: provide a simplified result for constant expressions (cf83b4f4)
  • copy:
    • avoid regex in isTypedArray (c8768d12)
    • only validate/clear if the user specifies a destination (33c67ce7, #12068)
  • merge: remove unnecessary wrapping of jqLite element (4daafd3d, #13236)

Breaking Changes

ngMessage is now compiled with a priority of 1, which means directives on the same element as ngMessage with a priority lower than 1 will be applied when ngMessage calls the $transclude function. Previously, they were applied during the initial compile phase and were passed the comment element created by the transclusion of ngMessage. To restore this behavior, custom directives need to have their priority increased to at least "1".

Previously, an non array-like input would pass through the orderBy filter unchanged. Now, an error is thrown. This can be worked around by converting an object to an array, either manually or using a filter such as https://github.com/petebacondarwin/angular-toArrayFilter. (null and undefined still pass through without an error, in order to support asynchronous loading of resources.)

1.5.0-beta.1 dense-dispersion (2015-09-29)

Bug Fixes

Features

Performance Improvements

  • $compile: Lazily compile the transclude function (652b83eb)

Breaking Changes

  • $sanitize: due to 181fc567, The svg support in is now an opt-in option

Applications that depend on this option can use to turn the option back on, but while doing so, please read the warning provided in the documentation for information on preventing click-hijacking attacks when this option is turned on.

If your data contains falsy values ('', 0, false and null) for option groups, then these options will now be placed into option groups. Previously all of these falsy values were treated as the option not being a member of a group.

Only option groups that are undefined will result in the option being put in no group. If you have data that contains falsy values that should not be used as groups then you must filter the values before passing them to ngOptions converting falsy values to undefined.

ngOptions will now throw if ngModel is not present on the select element. Previously, having no ngModel let ngOptions silently fail, which could lead to hard to debug errors. The change should therefore not affect any applications, as it simply makes the requirement more strict and alerts the developer explicitly.

1.4.7 dark-luminescence (2015-09-29)

Bug Fixes

  • $compile: use createMap() for $$observe listeners when initialized from attr interpolation (5a98e806, #10446)
  • $parse:
    • block assigning to fields of a constructor (a7f3761e, #12860)
    • do not convert to string computed properties multiple times (698af191)
  • filters: ensure formatNumber observes i18n decimal separators (4994acd2, #10342, #12850)
  • jqLite: properly handle dash-delimited node names in jqLiteBuildFragment (cdd1227a, #10617, #12759)
  • ngAnimate:
    • ensure anchoring uses body as a container when needed (9d3704ca, #12872)
    • callback detection should only use RAF when necessary (fa8c399f)
  • ngMessages: prevent race condition with ngAnimate (7295c60f, #12856, #12903)
  • ngOptions:

Features

Breaking Changes

1.3.20 shallow-translucence (2015-09-29)

Bug Fixes

  • $parse: do not convert to string computed properties multiple times (d434f3db)

Breaking Changes

1.2.29 ultimate-deprecation (2015-09-29)

Bug Fixes

Breaking Changes

1.4.6 multiplicative-elevation (2015-09-17)

Bug Fixes

Performance Improvements

  • Angular: only create new collection in getBlockNodes if the block has changed (0202663e, #9899)

1.3.19 glutinous-shriek (2015-09-15)

Bug Fixes

  • $http: propagate status -1 for timed out requests (f13055a0, #4491, #8756)
  • $location: don't crash if navigating outside the app base (623ce1ad, #11667)
  • $parse: throw error when accessing a restricted property indirectly (ec98c94c, #12833)
  • ngModel: validate pattern against the viewValue (274e9353, #12344)

Features

  • ngAnimate: introduce $animate.flush for unit testing (f98e0384)

Possible Breaking Changes

The ngPattern and pattern directives will validate the regex against the viewValue of ngModel, i.e. the value of the model before the $parsers are applied. Previously, the modelValue (the result of the $parsers) was validated.

This fixes issues where input[date] and input[number] cannot be validated because the viewValue string is parsed into Date and Number respectively (starting with Angular 1.3). It also brings the directives in line with HTML5 constraint validation, which validates against the input value.

This change is unlikely to cause applications to fail, because even in Angular 1.2, the value that was validated by pattern could have been manipulated by the $parsers, as all validation was done inside this pipeline.

If you rely on the pattern being validated against the modelValue, you must create your own validator directive that overwrites the built-in pattern validator:

.directive('patternModelOverwrite', function patternModelOverwriteDirective() {
  return {
    restrict: 'A',
    require: '?ngModel',
    priority: 1,
    compile: function() {
      var regexp, patternExp;

      return {
        pre: function(scope, elm, attr, ctrl) {
          if (!ctrl) return;

          attr.$observe('pattern', function(regex) {
            /**
             * The built-in directive will call our overwritten validator
             * (see below). We just need to update the regex.
             * The preLink fn guaranetees our observer is called first.
             */
            if (isString(regex) && regex.length > 0) {
              regex = new RegExp('^' + regex + '$');
            }

            if (regex && !regex.test) {
              //The built-in validator will throw at this point
              return;
            }

            regexp = regex || undefined;
          });

        },
        post: function(scope, elm, attr, ctrl) {
          if (!ctrl) return;

          regexp, patternExp = attr.ngPattern || attr.pattern;

          //The postLink fn guarantees we overwrite the built-in pattern validator
          ctrl.$validators.pattern = function(value) {
            return ctrl.$isEmpty(value) ||
              isUndefined(regexp) ||
              regexp.test(value);
          };
        }
      };
    }
  };
});

1.4.5 permanent-internship (2015-08-28)

Bug Fixes

Breaking Changes

The ngPattern and pattern directives will validate the regex against the viewValue of ngModel, i.e. the value of the model before the $parsers are applied. Previously, the modelValue (the result of the $parsers) was validated.

This fixes issues where input[date] and input[number] cannot be validated because the viewValue string is parsed into Date and Number respectively (starting with Angular 1.3). It also brings the directives in line with HTML5 constraint validation, which validates against the input value.

This change is unlikely to cause applications to fail, because even in Angular 1.2, the value that was validated by pattern could have been manipulated by the $parsers, as all validation was done inside this pipeline.

If you rely on the pattern being validated against the modelValue, you must create your own validator directive that overwrites the built-in pattern validator:

.directive('patternModelOverwrite', function patternModelOverwriteDirective() {
  return {
    restrict: 'A',
    require: '?ngModel',
    priority: 1,
    compile: function() {
      var regexp, patternExp;

      return {
        pre: function(scope, elm, attr, ctrl) {
          if (!ctrl) return;

          attr.$observe('pattern', function(regex) {
            /**
             * The built-in directive will call our overwritten validator
             * (see below). We just need to update the regex.
             * The preLink fn guaranetees our observer is called first.
             */
            if (isString(regex) && regex.length > 0) {
              regex = new RegExp('^' + regex + '$');
            }

            if (regex && !regex.test) {
              //The built-in validator will throw at this point
              return;
            }

            regexp = regex || undefined;
          });

        },
        post: function(scope, elm, attr, ctrl) {
          if (!ctrl) return;

          regexp, patternExp = attr.ngPattern || attr.pattern;

          //The postLink fn guarantees we overwrite the built-in pattern validator
          ctrl.$validators.pattern = function(value) {
            return ctrl.$isEmpty(value) ||
              isUndefined(regexp) ||
              regexp.test(value);
          };
        }
      };
    }
  };
});

1.3.18 collective-penmanship (2015-08-18)

Bug Fixes

1.4.4 pylon-requirement (2015-08-13)

Bug Fixes

Features

Performance Improvements

Breaking Changes

  • ngAnimate: due to 32d3cbb3, CSS classes added/removed by ngAnimate are now applied synchronously once the first digest has passed.

The previous behavior involved ngAnimate having to wait for one requestAnimationFrame before CSS classes were added/removed. The CSS classes are now applied directly after the first digest that is triggered after $animate.addClass, $animate.removeClass or $animate.setClass is called. If any of your code relies on waiting for one frame before checking for CSS classes on the element then please change this behavior. If a parent class-based animation, however, is run through a JavaScript animation which triggers an animation for beforeAddClass and/or beforeRemoveClass then the CSS classes will not be applied in time for the children (and the parent class-based animation will not be cancelled by any child animations).

  • $q due to 6838c979, When writing tests, there is no need to call $timeout.flush() to resolve a call to $q.when with a value.

The previous behavior involved creating an extra promise that needed to be resolved. This is no longer needed when $q.when is called with a value. In the case that the test is not aware if $q.when is called with a value or another promise, it is possible to replace $timeout.flush(); with $timeout.flush(0);.

describe('$q.when', function() {
  it('should not need a call to $timeout.flush() to resolve already resolved promises',
      inject(function($q, $timeout) {
    $q.when('foo');
    // In Angular 1.4.3 a call to `$timeout.flush();` was needed
    $timeout.verifyNoPendingTasks();
  }));

  it('should accept $timeout.flush(0) when not sure if $q.when was called with a value or a promise',
      inject(function($q, $timeout) {
    $q.when('foo');
    $timeout.flush(0);
    $timeout.verifyNoPendingTasks();
  }));

  it('should need a call to $timeout.flush() to resolve $q.when when called with a promise',
        inject(function($q, $timeout) {
    $q.when($q.when('foo'));
    $timeout.flush();
    $timeout.verifyNoPendingTasks();
  }));
});

1.4.3 foam-acceleration (2015-07-15)

Bug Fixes

  • $animateCss: ensure animations execute if only a keyframeStyle is provided (97d79eec, #12124, #12340)
  • loader: define isFunction (9ea52d81)
  • ngAnimate: ensure that orphaned elements do not throw errors when animated (e4aeae0c, #11975, #12338)

1.4.2 nebular-readjustment (2015-07-06)

Bug Fixes

Features

1.3.17 tsktskskly-euouae (2015-07-06)

Bug Fixes

1.4.1 hyperionic-illumination (2015-06-16)

Bug Fixes

Features

Performance Improvements

  • $compile: avoid jquery data calls when there is no data (9efb0d5e)

1.3.16 cookie-oatmealification (2015-06-05)

Bug Fixes

  • $compile: throw error on invalid directive name (634e4671, #11281, #11109)
  • $cookies: update $cookies to prevent duplicate cookie writes and play nice with external code (706a93ab, #11490, #11515)
  • $http: throw error if success and error methods do not receive a function (731e1f65, #11330, #11333)
  • core: ensure that multiple requests to requestAnimationFrame are buffered (0adc0364, #11791)
  • filterFilter: fix matching against null/undefined (9dd0fe35, #11573, #11617)
  • jqLite:
    • check for "length" in obj in isArrayLike to prevent iOS8 JIT bug from surfacing (647f3f55, #11508)
    • attr should ignore comment, text and attribute nodes (181e5ebc)
  • ngAnimate:
    • ensure that minified repaint code isn't removed (d5c99ea4, #9936)
  • ngAria: handle elements with role="checkbox/menuitemcheckbox" (1c282af5, #11317, #11321)
  • ngModel: allow setting model to NaN when asyncValidator is present (b64519fe, #11315, #11411)
  • ngTouch:
  • select: prevent unknown option being added to select when bound to null property (9e3f82bb, #11872, #11875)

Features

1.4.0 jaracimrman-existence (2015-05-26)

Bug Fixes

  • $animate:
  • $animateCss: ensure that custom durations do not confuse the gcs cache (e0e1b520, #11723, #11852)
  • $http: do not modify the config object passed into $http short methods (f7a4b481)
  • ngAnimate:
    • close follow-up class-based animations when the same class is added/removed when removed/added (db246eb7, #11717)
    • ensure nested class-based animations are spaced out with a RAF (213c2a70, #11812)
    • class-based animations must not set addClass/removeClass CSS classes on the element (3a3db690, #11810)
    • ensure that repeated structural calls during pre-digest function (2327f5a0, #11867)
    • ensure that cancelled class-based animations are properly cleaned up (718ff844, #11652)
    • throw an error if a callback is passed to animate methods (9bb4d6cc, #11826, #11713)
    • ensure anchored animations remove the leave element at correct time (64c66d0e, #11850)
  • select: prevent unknown option being added to select when bound to null property (4090491c, #11872, #11875)

Features

1.4.0-rc.2 rocket-zambonimation (2015-05-12)

Bug Fixes

  • $compile: ensure directive names have no leading or trailing whitespace (bab474aa, #11397, #11772)
  • $httpParamSerializerJQLike: follow jQuery logic for nested params (2420a0a7, #11551, #11635)
  • jqLite: check for "length" in obj in isArrayLike to prevent iOS8 JIT bug from surfacing (426a5ac0, #11508)
  • ngAnimate:
    • ensure that multiple requests to requestAnimationFrame are buffered (db20b830, #11791)
    • ensure that an object is always returned even when no animation is set to run (d5683d21)
    • force use of ng-anchor instead of a suffixed -anchor CSS class when triggering anchor animations (df24410c)
    • rename ng-animate-anchor to ng-anchor (e6d053de)
    • ensure that shared CSS classes between anchor nodes are retained (e0014002, #11681)
    • prohibit usage of the ng-animate class with classNameFilter (1002b80a, #11431, #11807)
    • ensure that the temporary CSS classes are applied before detection (f7e9ff1a, #11769, #11804)
    • ensure that all jqLite elements are deconstructed properly (64d05180, #11658)
    • ensure animations are not attempted on text nodes (2aacc2d6, #11703)
    • ensure JS animations recognize $animateCss directly (0681a540)
  • ngClass: add/remove classes which are properties of Object.prototype (f7b99970, #11813, #11814)
  • ngOptions:
  • ngTouch:
    • check undefined tagName for SVG event target (74eb17d7)
    • don't prevent click event after a touchmove (95521876, #10985)

Features

  • $resource: include request context in error message (266bc652, #11363)

Breaking Changes

ngAnimate

  • $animateCss: due to d5683d21, The $animateCss service will now always return an object even if the animation is not set to run. If your code is using $animateCss then please consider the following code change:
// before
var animator = $animateCss(element, { ... });
if (!animator) {
  continueApp();
  return;
}
var runner = animator.start();
runner.done(continueApp);
runner.then(continueApp);

// now
var animator = $animateCss(element, { ... });
var runner = animator.start();
runner.done(continueApp);
runner.then(continueApp);
  • due to df24410c, Prior to this fix there were to ways to apply CSS animation code to an anchor animation. With this fix, the suffixed CSS -anchor classes are now not used anymore for CSS anchor animations.

Instead just use the ng-anchor CSS class like so:

<div class="container-animation" ng-if="on">
   <div ng-animate-ref="1" class="my-anchor-element"></div>
</div>

<div class="container-animation" ng-if="!on">
   <div ng-animate-ref="1" class="my-anchor-element"></div>
</div>

before:

/* before (notice the container-animation CSS class) */
.container-animation-anchor {
  transition:0.5s linear all;
}

now:

/* now (just use `ng-anchor` on a class that both the
   elements that contain `ng-animate-ref` share) */
.my-anchor-element.ng-anchor {
  transition:0.5s linear all;
}
  • due to e6d053de, if your CSS code made use of the ng-animate-anchor CSS class for referencing the anchored animation element then your code must now use ng-anchor instead.

  • due to 1002b80a, partially or fully using a regex value containing ng-animate as a token is not allowed anymore. Doing so will trigger a minErr exception to be thrown.

So don't do this:

// only animate elements that contain the `ng-animate` CSS class
$animateProvider.classNameFilter(/ng-animate/);

// or partially contain it
$animateProvider.classNameFilter(/some-class ng-animate another-class/);

but this is OK:

$animateProvider.classNameFilter(/ng-animate-special/);

ngOptions

Although it is unlikely that anyone is using it in this way, this change does change the behavior of ngOptions in the following case:

  • you are iterating over an array-like object, using the array form of the ngOptions syntax (item.label for item in items) and that object contains non-numeric property keys.

In this case these properties with non-numeric keys will be ignored.

** Here array-like is defined by the result of a call to this internal function: https://github.com/angular/angular.js/blob/v1.4.0-rc.1/src/Angular.js#L198-L211 **

To get the desired behavior you need to iterate using the object form of the ngOptions syntax (value.label for (key, value) in items)`).

v1.4.0-rc.1 sartorial-chronography (2015-04-24)

Bug Fixes

  • $animate:
    • ensure that from styles are applied for class-based animations (8f819d2c)
    • make sure the JS animation lookup is an object lookup (103a39ca, #11619)
  • $animateCss: ensure that rAF waiting loop doesn't ignore pending items during a flush (90e424b2)
  • $http: stop coercing falsy HTTP request bodies to null / empty body (e04a887c, #11552, #11593)
  • ngAnimate:
    • close parent animations only when there are classes to resolve (1459be17)
    • ensure ngClass-based classes are always resolved for CSS-enabled animations (89f081e4)
    • do not abort animation if only ng-anchor-in is used (3333a5c3)
    • ensure that a filtered-out leave animation always runs its DOM operation (6dd64ab5, #11555)
    • ensure that animations work when the app is bootstrapped on the document node (bee14ed1, #11574)
    • ensure SVG classes are properly removed (fa0bbded)
  • ngAria: change accessibility keypress event to use event.which if it is provided (249f9b81, #11340)
  • ngMessageFormat:
  • select: allow empty option to be added dynamically by ng-repeat (abf59c28, #11470, #11512)

Features

  • $animate: provide support for animations on elements outside of $rootElement (e41faaa2)

v1.4.0-rc.0 smooth-unwinding (2015-04-10)

Bug Fixes

Features

Performance Improvements

Breaking Changes

  • $animate: due to c8700f04, JavaScript and CSS animations can no longer be run in parallel. With earlier versions of ngAnimate, both CSS and JS animations would be run together when multiple animations were detected. This feature has now been removed, however, the same effect, with even more possibilities, can be achieved by injecting $animateCss into a JavaScript-defined animation and creating custom CSS-based animations from there. Read the ngAnimate docs for more info.

  • $animate: due to c8700f04, The function params for $animate.enabled() when an element is used are now flipped. This fix allows the function to act as a getter when a single element param is provided.

// < 1.4
$animate.enabled(false, element);

// 1.4+
$animate.enabled(element, false);
  • $animate: due to c8700f04, In addition to disabling the children of the element, $animate.enabled(element, false) will now also disable animations on the element itself.

  • $animate: due to c8700f04, Animation-related callbacks are now fired on $animate.on instead of directly being on the element.

// < 1.4
element.on('$animate:before', function(e, data) {
  if (data.event === 'enter') { ... }
});
element.off('$animate:before', fn);

// 1.4+
$animate.on(element, 'enter', function(data) {
  //...
});
$animate.off(element, 'enter', fn);
  • $animate: due to c8700f04, There is no need to call $scope.$apply or $scope.$digest inside of an animation promise callback anymore since the promise is resolved within a digest automatically (but a digest is not run unless the promise is chained).
// < 1.4
$animate.enter(element).then(function() {
  $scope.$apply(function() {
    $scope.explode = true;
  });
});

// 1.4+
$animate.enter(element).then(function() {
  $scope.explode = true;
});
  • $animate: due to c8700f04, When an enter, leave or move animation is triggered then it will always end any pending or active parent class based animations (animations triggered via ngClass) in order to ensure that any CSS styles are resolved in time.

1.4.0-beta.6 cookie-liberation (2015-03-17)

Bug Fixes

  • $animate: call applyStyles from options on leave (4374f892, #10068)
  • $browser: don't crash if history.state access causes error in IE (3b8163b7, #10367, #10369)
  • $sanitize: disallow unsafe svg animation tags (67688d5c, #11290)
  • Angular: properly compare RegExp with other objects for equality (f22e1fc9, #11204, #11205)
  • date filter: display localized era for G format codes (2b4dfa9e, #10503, #11266)
  • filterFilter:
  • form: allow dynamic form names which initially evaluate to blank (410f7c68)
  • jqLite: attr should ignore comment, text and attribute nodes (bb5bf7f8)
  • ng/$locale: add ERA info in generic locale (4acb0af2)
  • ngJq: don't rely on existence of jqlite (342e5f3c, #11044)
  • ngMessages: ensure that multi-level transclusion works with ngMessagesInclude (d7ec5f39, #11196)
  • ngOptions: fix model<->option interaction when using track by (6a03ca27, #10869, #10893)
  • rootScope: prevent memory leak when destroying scopes (fb7db4a0, #11173, #11169)

Features

Breaking Changes

$cookies no longer exposes properties that represent the current browser cookie values. Now you must explicitly the methods described above to access the cookie values. This also means that you can no longer watch the $cookies properties for changes to the browser's cookies.

This feature is generally only needed if a 3rd party library was programmatically changing the cookies at runtime. If you rely on this then you must either write code that can react to the 3rd party library making the changes to cookies or implement your own polling mechanism.

1.3.15 locality-filtration (2015-03-17)

Bug Fixes

Features

1.4.0-beta.5 karmic-stabilization (2015-02-24)

Bug Fixes

Features

  • CommonJS: - angular modules are now packaged for npm with helpful exports

  • limitTo: extend the filter to take a beginning index argument (aaae3cc4, #5355, #10899)

  • ngMessages: provide support for dynamic message resolution (c9a4421f, #10036, #9338)
  • ngOptions: add support for disabling an option (da9eac86, #638, #11017)

Performance Improvements

  • $compile:
    • replace forEach(controller) with plain loops (5b522867, #11084)
    • avoid .data when fetching required controllers (fa0aa839)
  • ngOptions: only watch labels if a display expression is specified (51faaffd)

Breaking Changes

The ngMessagesInclude attribute is now its own directive and that must be placed as a child element within the element with the ngMessages directive. (Keep in mind that the former behavior of the ngMessageInclude attribute was that all included ngMessage template code was placed at the bottom of the element containing the ngMessages directive; therefore to make this behave in the same way, place the element containing the ngMessagesInclude directive at the end of the container containing the ngMessages directive).

<!-- AngularJS 1.3.x -->
<div ng-messages="model.$error" ng-messages-include="remote.html">
  <div ng-message="required">Your message is required</div>
</div>

<!-- AngularJS 1.4.x -->
<div ng-messages="model.$error">
  <div ng-message="required">Your message is required</div>
  <div ng-messages-include="remote.html"></div>
</div>

it is no longer possible to use interpolation inside the ngMessages attribute expression. This technique is generally not recommended, and can easily break when a directive implementation changes. In cases where a simple expression is not possible, you can delegate accessing the object to a function:

<div ng-messages="ctrl.form['field_{{$index}}'].$error">...</div>

would become

<div ng-messages="ctrl.getMessages($index)">...</div>

where ctrl.getMessages()

ctrl.getMessages = function($index) {
  return ctrl.form['field_' + $index].$error;
}

transformRequest functions can no longer modify request headers.

Before this commit transformRequest could modify request headers, ex.:

function requestTransform(data, headers) {
    headers = angular.extend(headers(), {
      'X-MY_HEADER': 'abcd'
    });
  }
  return angular.toJson(data);
}

This behavior was unintended and undocumented, so the change should affect very few applications. If one needs to dynamically add / remove headers it should be done in a header function, for example:

$http.get(url, {
  headers: {
    'X-MY_HEADER': function(config) {
      return 'abcd'; //you've got access to a request config object to specify header value dynamically
    }
  }
})

1.3.14 instantaneous-browserification (2015-02-24)

Features

  • CommonJS: - angular modules are now packaged for npm with helpful exports

Bug Fixes

1.4.0-beta.4 overlyexplosive-poprocks (2015-02-09)

Bug Fixes

Features

  • ng-jq: adds the ability to force jqLite or a specific jQuery version (09ee82d8)

1.3.13 meticulous-riffleshuffle (2015-02-09)

Bug Fixes

1.4.0-beta.3 substance-mimicry (2015-02-02)

Bug Fixes

  • $compile:
    • do not initialize optional '&' binding if attribute not specified (6a38dbfd, #6404, #9216)
    • respect return value from controller constructor (62d514b0)
  • $controller: throw better error when controller expression is bad (dda65e99, #10875, #10910)
  • $parse:
    • handle null targets at assign (2e5a7e52)
    • remove references to last arguments to a fn call (e61eae1b, #10894)
  • a: don't reload if there is only a name attribute (d729fcf0, #6273, #10880)
  • angular.copy: support copying TypedArrays (aa0f6449, #10745)
  • filter: format timezone correctly in the case that UTC timezone is used (8c469191, #9359)
  • ngRoute: don't duplicate optional params into query (27bf2ce4, #10689)
  • ngScenario: allow ngScenario to handle lazy-loaded and manually bootstrapped applications (c69caa7b, #10723)
  • validators: maxlength should use viewValue for $isEmpty (bfcf9946, #10898)

Features

  • $compile: allow using bindToController as object, support both new/isolate scopes (35498d70, #10420, #10467)
  • filter: support conversion to timezone other than UTC (c6d8512a, #10858)
  • ngMocks: cleanup $inject annotations after each test (0baa17a3)

Performance Improvements

  • $scope: Add a property $$watchersCount to scope (c1500ea7)
  • $parse new and more performant parser (0d42426)

Breaking Changes

  • $compile: due to 6a38dbfd, Previously, '&' expressions would always set up a function in the isolate scope. Now, if the binding is marked as optional and the attribute is not specified, no function will be added to the isolate scope.

1.3.12 outlandish-knitting (2015-02-02)

Bug Fixes

  • $controller: throw better error when controller expression is bad (632b2ddd, #10875, #10910)
  • $parse: remove references to last arguments to a fn call (7caad220, #10894)
  • ngRoute: don't duplicate optional params into query (f41ca4a5, #10689)
  • ngScenario: Allow ngScenario to handle lazy-loaded and manually bootstrapped applications (0bcd0872, #10723)
  • validators: maxlength should use viewValue for $isEmpty (abd8e2a9, #10898)

Features

  • ngMocks: cleanup $inject annotations after each test (6ec59460)

1.4.0-beta.2 holographic-rooster (2015-01-26)

Bug Fixes

Breaking Changes

Closes #9992 Closes #10352

1.3.11 spiffy-manatee (2015-01-26)

Bug Fixes

  • $location: don't rewrite when link is shift-clicked (939ca37c, #9904, #9906)
  • htmlAnchorDirective:
    • remove "element !== target element" check (779e3f6b, #10866)
    • don't add event listener if replaced, ignore event if target is different element (837a0775, #4262, #10849)

1.4.0-beta.1 trepidatious-salamander (2015-01-20)

Bug Fixes

Features

Breaking Changes

Previously, the order of items when using ngRepeat to iterate over object properties was guaranteed to be consistent by sorting the keys into alphabetic order.

Now, the order of the items is browser dependent based on the order returned from iterating over the object using the for key in obj syntax.

It seems that browsers generally follow the strategy of providing keys in the order in which they were defined, although there are exceptions when keys are deleted and reinstated. See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/delete#Cross-browser_issues

The best approach is to convert Objects into Arrays by a filter such as https://github.com/petebacondarwin/angular-toArrayFilter or some other mechanism, and then sort them manually in the order you need.

Closes #6210 Closes #10538

1.3.10 heliotropic-sundial (2015-01-20)

Bug Fixes

1.4.0-beta.0 photonic-umbrakinesis (2015-01-13)

Bug Fixes

Features

Performance Improvements

Breaking Changes

  • limitTo: due to a3c3bf33, limitTo changed behavior when limit value is invalid. Instead of returning empty object/array it returns unchanged input.

  • ngOptions: due to 7fda214c,

When using ngOptions: the directive applies a surrogate key as the value of the <option> element. This commit changes the actual string used as the surrogate key. We now store a string that is computed by calling hashKey on the item in the options collection; previously it was the index or key of the item in the collection.

(This is in keeping with the way that the unknown option value is represented in the select directive.)

Before you might have seen:

<select ng-model="x" ng-option="i in items">
  <option value="1">a</option>
  <option value="2">b</option>
  <option value="3">c</option>
  <option value="4">d</option>
</select>

Now it will be something like:

<select ng-model="x" ng-option="i in items">
  <option value="string:a">a</option>
  <option value="string:b">b</option>
  <option value="string:c">c</option>
  <option value="string:d">d</option>
</select>

If your application code relied on this value, which it shouldn't, then you will need to modify your application to accommodate this. You may find that you can use the track by feature of ngOptions as this provides the ability to specify the key that is stored.

When iterating over an object's properties using the (key, value) in obj syntax the order of the elements used to be sorted alphabetically. This was an artificial attempt to create a deterministic ordering since browsers don't guarantee the order. But in practice this is not what people want and so this change iterates over properties in the order they are returned by Object.keys(obj), which is almost always the order in which the properties were defined.

setting the ngOptions attribute expression after the element is compiled, will no longer trigger the ngOptions behavior. This worked previously because the ngOptions logic was part of the select directive, while it is now implemented in the ngOptions directive itself.

the select directive will now use strict comparison of the ngModel scope value against option values to determine which option is selected. This means Number scope values will not be matched against numeric option strings. In Angular 1.3.x, setting scope.x = 200 would select the option with the value 200 in the following select:

<select ng-model="x">
  <option value="100">100</option>
  <option value="200">200</option>
</select>

In Angular 1.4.x, the 'unknown option' will be selected. To remedy this, you can simply initialize the model as a string: scope.x = '200', or if you want to keep the model as a Number, you can do the conversion via $formatters and $parsers on ngModel:

ngModelCtrl.$parsers.push(function(value) {
  return parseInt(value, 10); // Convert option value to number
});

ngModelCtrl.$formatters.push(function(value) {
  return value.toString(); // Convert scope value to string
});

1.3.9 multidimensional-awareness (2015-01-13)

Bug Fixes

  • $parse: allow use of locals in assignments (86900814)
  • filterFilter: use isArray() to determine array type (d4b60ada, #10621)

Features

Performance Improvements

1.3.8 prophetic-narwhal (2014-12-19)

Bug Fixes

Performance Improvements

  • limitTo: replace for loop with slice (cd77c089)

1.3.7 leaky-obstruction (2014-12-15)

Bug Fixes

  • $compile: use createMap() for $$observe listeners when initialized from attr interpolation (8e28bb4c)
  • $http:
  • $parse: a chain of field accessors should use a single getterFn (c90ad968)
  • ngRepeat: allow extra whitespaces in (key,value) part of micro-syntax (ef640cbc, #6827, #6833)
  • orderBy: do not try to call valueOf/toString on null (a097aa95, #10385, #10386)

Features

Performance Improvements

  • $compile: only re-$interpolate attribute values at link time if changed since compile (9ae0c01c)

Breaking Changes

Previously, if either value being compared in the orderBy comparator was null or undefined, the order would, incorrectly, not change. Now, this order behaves more like Array.prototype.sort, which by default pushes null behind objects, due to n occurring after [ (the first characters of their stringified forms) in ASCII / Unicode. If toString is customized, or does not exist, the behavior is undefined.

1.2.28 finnish-disembarkation (2014-12-15)

Bug Fixes

1.3.6 robofunky-danceblaster (2014-12-08)

Bug Fixes

  • $browser: prevent infinite digests when clearing the hash of a url (10ac5948, #9629, #9635, #10228, #10308)
  • $http: preserve config object when resolving from cache (facfec98, #9004, #9030)
  • $location:
  • $parse:
    • fix operators associativity (ed1243ff)
    • follow JavaScript context for unbound functions (429938da)
  • filterFilter:
    • don't match primitive sub-expressions against any prop (a75537d4)
    • ignore function properties and account for inherited properties (5ced914c, #9984)
    • correctly handle deep expression objects (f7cf8460, #7323, #9698, #9757)
  • inputs: ignoring input events in IE caused by placeholder changes or focus/blur on inputs with placeholders (55d9db56, #9265)
  • linky: make urls starting with www. links, like markdown (915a891a, #10290)
  • ngAnimate: do not use jQuery class API (40a537c2, #10024, #10329)
  • ngMock: allow numeric timeouts in $httpBackend mock (acb066e8, #4891)
  • ngModel:
    • always use the most recent viewValue for validation (2d6a0a1d, #10126, #10299)
    • fixing many keys incorrectly marking inputs as dirty (d21dff21)
  • ngSanitize: exclude smart quotes at the end of the link (7c6be43e, #7307)
  • numberFilter: numbers rounding to zero shouldn't be negative (96c61fe7, #10278)
  • orderBy:
    • make object-to-primtiive behavior work for objects with null prototype (3aa57528)
    • maintain order in array of objects when predicate is not provided (8bfeddb5, #9566, #9747, #10311)

Features

  • $$jqLite: export jqLite as a private service (f2e7f875)
  • $injector: print caller name in "unknown provider" errors (when available) (013b522c, #8135, #9721)
  • jsonFilter: add optional arg to define custom indentation (1191edba, #9771)
  • ngAria: bind keypress on ng-click w/ option (5481e2cf, #10288)

Breaking Changes

We no longer throw an ihshprfx error if the URL after the base path contains only a hash fragment. Previously, if the base URL was http://abc.com/base/ and the hashPrefix is ! then trying to parse http://abc.com/base/#some-fragment would have thrown an error. Now we simply assume it is a normal fragment and that the path is empty, resulting $location.absUrl() === "http://abc.com/base/#!/#some-fragment".

This should not break any applications, but you can no longer rely on receiving the ihshprfx error for paths that have the syntax above. It is actually more similar to what currently happens for invalid extra paths anyway: If the base URL and hashPrfix are set up as above, then http://abc.com/base/other/path does not throw an error but just ignores the extra path: http://abc.com/base.

  • filterFilter: due to a75537d4,

    Named properties in the expression object will only match against properties on the same level. Previously, named string properties would match against properties on the same level or deeper.

    Before:

      arr = filterFilter([{level1: {level2: 'test'}}], {level1: 'test'});   // arr.length -> 1

    After:

      arr = filterFilter([{level1: {level2: 'test'}}], {level1: 'test'});   // arr.length -> 0

    In order to match deeper nested properties, you have to either match the depth level of the property or use the special $ key (which still matches properties on the same level or deeper). E.g.:

      // Both examples below have `arr.length === 1`
      arr = filterFilter([{level1: {level2: 'test'}}], {level1: {level2: 'test'}});
      arr = filterFilter([{level1: {level2: 'test'}}], {$: 'test'});

1.3.5 cybernetic-mercantilism (2014-12-01)

Bug Fixes

1.3.4 highfalutin-petroglyph (2014-11-24)

Bug Fixes

Features

Performance Improvements

  • use Object.create instead of creating temporary constructors (bf6a79c3, #10058)

Breaking Changes

  • ngModelOptions: due to bb4d3b73, previously, ngModel invoked getter/setters in the global context.

For example:

<input ng-model="model.value" ng-model-options="{ getterSetter: true }">

would previously invoke model.value() in the global context.

Now, ngModel invokes value with model as the context.

It's unlikely that real apps relied on this behavior. If they did they can use .bind to explicitly bind a getter/getter to the global context, or just reference globals normally without this.

1.2.27 prime-factorization (2014-11-20)

Bug Fixes

  • $animate: clear the GCS cache even when no animation is detected (f619d032, #8813)
  • $browser:
  • $http: add missing shortcut methods and missing docs (ec4fe1bc, #9180, #9321)
  • $location:
    • revert erroneous logic and backport refactorings from master (1ee9b4ef, #8492)
    • allow 0 in path() and hash() (f807d7ab)
  • $parse: add quick check for Function constructor in fast path (756640f5)
  • $parse, events: prevent accidental misuse of properties on $event (4d0614fd, #9969)
  • ngMock: $httpBackend should match data containing Date objects correctly (1426b029, #5127)
  • orderBy: sort by identity if no predicate is given (45b896a1, #5847, #4579, #9403)
  • select: ensure the label attribute is updated in Internet Explorer (16833d0f, #9621, #10042)

Performance Improvements

  • orderBy: copy array with slice instead of for loop (409bcb38, #9942)

1.3.3 undersea-arithmetic (2014-11-17)

Bug Fixes

  • $http: don't parse single space responses as JSON (6f19a6fd, #9907)
  • minErr: stringify non-JSON compatible objects in error messages (cf43ccdf, #10085)
  • $rootScope: handle cyclic references in scopes when creating error messages (e80053d9, #10085)
  • ngRepeat: support cyclic object references in error messages (fa12c3c8, #9838, #10065, #10085)
  • ngMock: call $interval callbacks even when invokeApply is false (d81ff888, #10032)
  • ngPattern: match behavior of native HTML pattern attribute (85eb9660, #9881, #9888)
  • select: ensure the label attribute is updated in Internet Explorer (6604c236, #9621, #10042)

Features

  • $location: allow to location to be changed during $locationChangeStart (a9352c19, #9607, #9678)
  • $routeProvider: allow setting caseInsensitiveMatch on the provider (0db573b7, #6477, #9873)

Performance Improvements

  • orderBy: copy array with slice instead of for loop (8eabc546, #9942)

Breaking Changes

  • $parse: due to fbad2805, you can't use characters that have special meaning in AngularJS expressions (ex.: . or -) as part of filter's name. Before this commit custom filters could contain special characters (like a dot) in their name but this wasn't intentional.

1.3.2 cardiovasculatory-magnification (2014-11-07)

Bug Fixes

  • $compile: do not rebind parent bound transclude functions (841c0907, #9413)
  • $parse:
    • stateful interceptors override an undefined expression (ed99821e, #9821, #9825)
    • add quick check for Function constructor in fast path (e676d642)
  • $parse, events: prevent accidental misuse of properties on $event (e057a9aa)
  • ngRoute: allow proto inherited properties in route params object (b4770582, #8181, #9731)
  • select: use strict comparison for isSelected with selectAs (9e305948, #9639, #9949)

Features

  • ngAria: announce ngMessages with aria-live (187e4318, #9834)
  • ngMock: decorator that adds Scope#$countChildScopes and Scope#$countWatchers (74981c9f, #9926, #9871)

Security Note

This release also contains security fixes for expression sandbox bypasses.

These issues affect only applications with known server-side XSS holes that are also using CSP to secure their client-side code. If your application falls into this rare category, we recommend updating your version of Angular.

We'd like to thank security researches Sebastian Lekies, Jann Horn, and Gábor Molnár for reporting these issues to us.

We also added a documentation page focused on security, which contains some of the best practices, DOs and DON'Ts. Please check out https://docs.angularjs.org/guide/security.

1.3.1 spectral-lobster (2014-10-31)

Bug Fixes

  • $compile: returning null when an optional controller is not found (2cd5b4ec, #9404, #9392)
  • $observe: check if the attribute is undefined (531a8de7, #9707, #9720)
  • $parse: support dirty-checking objects with null prototype (28661d1a, #9568)
  • $sce: use msie instead of $document[0].documentMode (45252c3a, #9661)
  • $templateRequest: ignore JSON Content-Type header and content (1bd473eb, #5756, #9619)
  • i18n: rename datetimeSymbols to be camelCase (94f5a285)
  • loader: fix double spaces (8b2f1a47, #9630)
  • ngMock: $httpBackend should match data containing Date objects correctly (1025f6eb, #5127)
  • ngSanitize: attribute name: xmlns:href -> xlink:href (4cccf0f2, #9769)
  • select: assign result of track exp to element value (4b4098bf, #9718, #9592)
  • templateRequest: allow empty html template (52ceec22, #9581)
  • testability: escape regex chars in findBindings if using exactMatch (02aa4f4b, #9595, #9600)

Features

  • $compile: allow $watchCollection to be used in bi-directional bindings (40bbc981, #9725)
  • ngSanitize: accept SVG elements and attributes (a54b25d7, #9578, #9751)

Breaking Changes

  • $observe: Due to 531a8de7, observers no longer register on undefined attributes. For example, if you were using $observe on an absent optional attribute to set a default value, the following would not work anymore:
<my-dir></my-dir>
// link function for directive myDir
link: function(scope, element, attr) {
  attr.$observe('myAttr', function(newVal) {
    scope.myValue = newVal ? newVal : 'myDefaultValue';
  })
}

Instead, check if the attribute is set before registering the observer:

link: function(scope, element, attr) {
  if (attr.myAttr) {
    // register the observer
  } else {
    // set the default
  }
}

1.3.0 superluminal-nudge (2014-10-13)

Bug Fixes

Features

Breaking Changes

  • $animate: due to e5f4d7b1, staggering animations that use transitions will now always block the transition from starting (via transition: 0s none) up until the stagger step kicks in. The former behavior was that the block was removed as soon as the pending class was added. This fix allows for styles to be applied in the pending class without causing an animation to trigger prematurely.

1.3.0-rc.5 impossible-choreography (2014-10-08)

Bug Fixes

Features

Performance Improvements

  • $animate:
    • access DOM less in resolveElementClasses (22358cf9)
    • don't join classes before it's necessary in resolveElementClasses (003c44ec)
  • ngBind: set textContent rather than using element.text() (074a146d, #9369, #9396)

Breaking Changes

If a template contains directives within comment nodes, and there is more than a single node in the template, those comment nodes are removed. The impact of this breaking change is expected to be quite low.

Closes #9212 Closes #9215

The $animate CSS class API will always defer changes until the end of the next digest. This allows ngAnimate to coalesce class changes which occur over a short period of time into 1 or 2 DOM writes, rather than many. This prevents jank in browsers such as IE, and is generally a good thing.

If you find that your classes are not being immediately applied, be sure to invoke $digest().

Closes #8234 Closes #9263

ngOptions will now throw an error when the comprehension expressions contains both a select as and track by expression.

These expressions are fundamentally incompatible because it is not possible to reliably and consistently determine the parent object of a model, since select as can assign any child of a value as the model value.

Prior to refactorings in this release, neither of these expressions worked correctly independently, and did not work at all when combined.

See #6564

Order of events has changed. Previously: $locationChangeStart -> $locationChangeSuccess -> $routeChangeStart -> $routeChangeSuccess

Now: $locationChangeStart -> $routeChangeStart -> $locationChangeSuccess -> -> $routeChangeSuccess

Fixes #5581 Closes #5714 Closes #9502

1.3.0-rc.4 unicorn-hydrafication (2014-10-01)

Bug Fixes

Features

Performance Improvements

  • $rootScope: moving internal queues out of the Scope instances (b1192518, #9071)
  • benchmark: add ngBindOnce benchmarks to largetable-bp (2c8b4648)
  • ngForm,ngModel: move initial addClass to the compile phase (b1ee5386, #8268)

Breaking Changes

$transclude functions no longer attach $destroy event handlers to the transcluded content, and so the associated transclude scope will not automatically be destroyed if you remove a transcluded element from the DOM using direct DOM manipulation such as the jquery remove() method.

If you want to explicitly remove DOM elements inside your directive that have been compiled, and so potentially contain child (and transcluded) scopes, then it is your responsibility to get hold of the scope and destroy it at the same time.

The suggested approach is to create a new child scope of your own around any DOM elements that you wish to manipulate in this way and destroy those scopes if you remove their contents - any child scopes will then be destroyed and cleaned up automatically.

Note that all the built-in directives that manipulate the DOM (ngIf, ngRepeat, ngSwitch, etc) already follow this best practice, so if you only use these for manipulating the DOM then you do not have to worry about this change.

Closes #9095 Closes #9281

  • $parse: due to 5572b40b,

  • $scope['this'] no longer exits on the $scope object

  • $parse-ed expressions no longer allow chaining 'this' such as this['this'] or $parent['this']
  • 'this' in $parse-ed expressions can no longer be overridden, if a variable named 'this' is put on the scope it must be accessed using this['this']

Closes #9105

(Note: this change landed in 1.3.0-rc.3, but was not considered a breaking change at the time).

For text based inputs (text, email, url), the $viewValue will now always be converted to a string, regardless of what type the value is on the model.

To migrate, any code or expressions that expect the $viewValue to be anything other than string should be updated to expect a string.

  • input: due to a0bfdd0d60882125f614a91c321f12f730735e7b (see #8949),

Similar to input[number] Angular will now throw if the model value for a input[date] is not a Date object. Previously, Angular only showed an empty string instead. Angular does not set validation errors on the <input> in this case as those errors are shown to the user, but the erroneous state was caused by incorrect application logic and not by the user.

1.2.26 captivating-disinterest (2014-10-01)

Bug Fixes

  • select: make ctrl.hasOption method consistent (11d2242d, #8761)

1.3.0-rc.3 aggressive-pacification (2014-09-23)

Bug Fixes

  • ngModel: support milliseconds in time and datetime (4b83f6ca, #8874)

Features

  • $location: add ability to opt-out of <base> tag requirement in html5Mode (dc3de7fb, #8934)
  • formController: add $setUntouched to propagate untouched state (fd899755, #9050)
  • input: support dynamic element validation (729c238e, #4791, #1404)
  • ngAria: add an ngAria module to make a11y easier (d1434c99, #5486)

Performance Improvements

  • map: use Array.prototype.map (a591e8b8)

Breaking Changes

  • $location: due to dc3de7fb, The $location.html5Mode API has changed to allow enabling html5Mode by passing an object (as well as still supporting passing a boolean). Symmetrically, the method now returns an object instead of a boolean value.

    To migrate, follow the code example below:

    Before:

    var mode = $locationProvider.html5Mode();

    After:

    var mode = $locationProvider.html5Mode().enabled;

Fixes #8934

1.2.25 hypnotic-gesticulation (2014-09-16)

Bug Fixes

  • i18n: fix typo at i18n generation code (1b6d74cc)
  • ngLocale: Regenerate Locale Files (06c76694)
  • select: update option labels when model changes (d89d59f4, #9025)

1.3.0-rc.2 tactile-perception (2014-09-16)

Bug Fixes

Features

Performance Improvements

  • $compile: move $$isolateBinding creation to directive factory instead of on each link (56f09f0b)
  • $parse:
    • execute watched expressions only when the inputs change (fca6be71, #9006, #9082)
    • remove binaryFn and valueFn wrappers from filter expressions (67919c80)

Breaking Changes

  • $parse: due to fca6be71, all filters are assumed to be stateless functions

Previously it was just a good practice to make all filters stateless. Now it's a requirement in order for the model change-observation to pick up all changes.

If an existing filter is stateful, it can be flagged as such but keep in mind that this will result in a significant performance-penalty (or rather lost opportunity to benefit from a major perf improvement) that will affect the $digest duration.

To flag a filter as stateful do the following:

myApp.filter('myFilter', function() {
  function myFilter(input) { ... };
  myFilter.$stateful = true;
  return myFilter;
});

1.3.0-rc.1 backyard-atomicity (2014-09-09)

Bug Fixes

  • $location:
  • $parse: disallow passing Function to Array.sort (bd8ad0fb)
  • input: check scope.$$phase only on $rootScope (bf59d727)
  • ngAnimate: support removing classes from SVG elements when using jQuery (b3b67213, #8872, #8893)
  • ngEventDirs: check scope.$$phase only on $rootScope (203ea10f, #8891)
  • ngForm: don't clear validity of whole form when removing control (953ee22f, #8863)
  • ngInclude: correctly add svg-namespaced template content (6639ca9d, #7538, #8981, #8997)
  • ngModel:
    • update model value with async validators correctly (64c3b745)
    • render immediately also with async validators (f94d5515)
    • properly parse min/max date values as strings for date inputs (088545c1, #6755)
    • revalidate the model when min/max expression values change for date inputs (b3502835, #6755)
    • consider ngMin/ngMax values when validating number input types (25541c1f)
    • revalidate the model when min/max expression values change for number inputs (7b273a2c, #2404)
  • ngModelOptions: do not trigger digest on setViewValue if debouncing (e322cd9b, #8814, #8850, #8911)
  • ngRepeat: preserve original position of elements that are being animated away (ed637330, #8918, #8994)
  • ngSwitch: ensure correct iterator is passed to async function (712299c2, #8833)
  • numberFilter: format numbers that round to zero as nonnegative (ae952fbf, #8489)
  • orderBy: allow arrayLike objects to be ordered (cbdaabfb, #8944)

Features

  • angular.forEach: add the array/object as the 3rd param like the native array forEach (df9e60c8, #7902)
  • ngModelOptions: add allowInvalid option (3c538c1d, #8290, #8313)

Performance Improvements

  • $parse:
    • remove getterFn wrapper for internal use (b3b476db, #8901)
    • removing references to Parser/Lexer from parsed expressions (43c67ccd)
    • calculate array lengths once at start of loop (907b8c16)
  • extend: remove use of forEach to remove calls/closures/passing arguments (9bedeb33, #8898)
  • jQuery: only trigger $destroy if a handler exists (f6aa1c55, #8859)

Breaking Changes

  • ngModelController,formController: due to 6046e14b,

  • ctrl.$error no longer contains entries for validators that were successful.

  • ctrl.$setValidity now differentiates between true, false, undefined and null, instead of previously only truthy vs falsy.

Closes #8941- ngSwitch: due to 0f806d96,

Ever since 0df93fd, tagged in v1.0.0rc1, the ngSwitch directive has had an undocumented change attribute, used for evaluating a scope expression when the switch value changes.

While it's unlikely, applications which may be using this feature should work around the removal by adding a custom directive which will perform the eval instead. Directive controllers are re-instantiated when being transcluded, so by putting the attribute on each item that you want to be notified of a change to, you can more or less emulate the old behavior.

Example:

angular.module("switchChangeWorkaround", []).
  directive("onSwitchChanged", function() {
    return {
      link: function($scope, $element, $attrs) {
        $scope.$parent.$eval($attrs.onSwitchChanged);
      }
    };
  });
<div ng-switch="switcher">
  <div ng-switch-when="a" on-switch-changed="doSomethingInParentScope()"></div>
  <div ng-switch-when="b" on-switch-changed="doSomethingInParentScope()"></div>
</div>

Closes #8858 Closes #8822

1.2.24 static-levitation (2014-09-09)

Bug Fixes

  • $browser: detect changes to the browser url that happened in sync (2ece4d03, #6976)
  • $compile:
  • $location:
    • don't call toString on null values (c12e8d46)
    • remove an unused parameter of $location.url (c65796d4)
    • allow numeric location setter arguments (68a09ba7, #7054)
  • $parse: disallow passing Function to Array.sort (b39e1d47)
  • form: ensure concurrent animations use setClass (d7548fdf, #8166)
  • input: check scope.$$phase only on $rootScope (36e6de1d)
  • ngEventDirs:
  • ngRepeat: improve errors for duplicate items (1812af58)
  • numberFilter: format numbers that round to zero as nonnegative (7e02fa07, #8489)
  • orderBy: allow arrayLike objects to be ordered (94b0f2d3, #8944)

Breaking Changes

The blur and focus event fire synchronously, also during DOM operations that remove elements. This lead to errors as the Angular model was not in a consistent state. See this fiddle for a demo.

This change executes the expression of those events using scope.$evalAsync if an $apply is in progress, otherwise keeps the old behavior.

Fixes #4979 Fixes #5945 Closes #8803 Closes #6910 Closes #5402

1.3.0-RC.0 sonic-boltification (2014-08-29)

Bug Fixes

  • $animate:
  • $browser: detect changes to the browser url that happened in sync (3be00df4, #6976)
  • $compile: use the correct namespace for transcluded svg elements (cb73a37c, #8808, #8816)
  • $location: always resolve relative links in html5mode to <base> url (22948807, #8492, #8172)
  • $parse: properly handle dots at the end of identifiers (8ac90357, #4613, #4912, #8559)
  • Angular: remove duplicate nodeName_ references (a4520a74)
  • currencyFilter: pass through null and undefined values (c2aaddbe, #8605)
  • docs: don't throw exception on the 404 page (550ba01b, #8518)
  • input:
    • validate minlength/maxlength for non-string values (77ce5b89, #7967, #8811)
    • allow to use seconds in input[time] and input[datetime-local] (5f90340a)
    • use year 1970 instead of 1900 for input[time] (29f0b568)
  • ngBindHtml: throw error if interpolation is used in expression (cd21602d, #8824)
  • ngEventDirs: execute blur and focus expression using scope.$evalAsync (719c747c, #4979, #5945, #8803, #6910, #5402)
  • ngModel:
    • always format the viewValue as a string for text, url and email types (1eda1836)
    • allow non-assignable binding when getterSetter is used (ab878a6c, #8704)
    • treat undefined parse responses as parse errors (db044c40)
  • ngRepeat: improve errors for duplicate items (0604bb7b)
  • ngSwitch: avoid removing DOM nodes twice within watch operation (c9b0bfec, #8662)
  • numberFilter: pass through null and undefined values (2ae10f67, #8605, #8842)

Features

  • core:
    • add angular.reloadWithDebugInfo() (41c1b88)
  • $animate:
    • use promises instead of callbacks for animations (bf0f5502)
    • coalesce concurrent class-based animations within a digest loop (2f4437b3)
  • $compile:
  • $compile/ngBind: allow disabling binding info (3660fd09)
  • $http: implement mechanism for coalescing calls to $apply in $http (ea6fc6e6, #8736, #7634, #5297)
  • $rootScope: implement $applyAsync to support combining calls to $apply into a single digest. (e94d454b)
  • $templateRequest: introduce the $templateRequest service (a70e2833)
  • filter: allow to define the timezone for formatting dates (4739b1d9)
  • filterFilter: pass index to function predicate (46343c60, #654)
  • input: allow to define the timezone for parsing dates (cc6fc199, #8447)
  • minErr: allow specifying ErrorConstructor in minErr constructor (a6bd4bc8)
  • ngModel: provide validation API functions for sync and async validations (2ae4f40b)
  • ngRoute: alias string as redirectTo property in .otherwise() (3b5d75c0, #7794)
  • testability: add $$testability service (85880a64)

Performance Improvements

  • $compile:
    • add debug classes in compile phase (e0489abd)
    • only iterate over elements with link functions (fdf9989f, #8741)
  • nodeName_: simplify the code and reduce the number of DOM calls (5a1a0c96)
  • select: execute render after $digest cycle (6f7018d5, #8825)

Breaking Changes

since 1.2.0 and 1.3.0-beta.1

Angular now requires a <base> tag when html5 mode of $location is enabled. Reasoning: Using html5 mode without a <base href="..."> tag makes relative links for images, links, ... relative to the current url if the browser supports the history API. However, if the browser does not support the history API Angular falls back to using the #, and then all those relative links would be broken.

The <base> tag is also needed when a deep url is loaded from the server, e.g. http://server/some/page/url. In that case, Angular needs to decide which part of the url is the base of the application, and which part is path inside of the application.

To summarize: Now all relative links are always relative to the <base> tag.

Exception (also a breaking change): Link tags whose href attribute starts with a # will only change the hash of the url, but nothing else (e.g. <a href="#someAnchor">). This is to make it easy to scroll to anchors inside a document.

Related to #6162 Closes #8492

since 1.2.17 and 1.3.0-beta.10

In html5 mode without a <base> tag on older browser that don't support the history API relative paths were adding up. E.g. clicking on <a href="page1"> and then on <a href="page2"> would produce $location.path()==='/page1/page2'. The code that introduced this behavior was removed and Angular now also requires a <base> tag to be present when using html5 mode.

Closes #8172, #8233

  • ngInclude, ngMessage, ngView and directives that load templates: due to a70e2833

Angular will now throw a $compile minErr each a template fails to download for ngView, directives and ngMessage template requests. This changes the former behavior of silently ignoring failed HTTP requests--or when the template itself is empty. Please ensure that all directive, ngView and ngMessage code now properly addresses this scenario. NgInclude is unaffected from this change.

If any stagger code consisted of having BOTH transition staggers and delay staggers together then that will not work the same way. Angular will now instead choose the highest stagger delay value and set the timeout to wait for that before applying the active CSS class.

Both the API for the cancelation method and the done callback for $animate animations is different. Instead of using a callback function for each of the $animate animation methods, a promise is used instead.

//before
$animate.enter(element, container, null, callbackFn);

//after
$animate.enter(element, container).then(callbackFn);

The animation can now be cancelled via $animate.cancel(promise).

//before
var cancelFn = $animate.enter(element, container);
cancelFn(); //cancels the animation

//after
var promise = $animate.enter(element, container);
$animate.cancel(promise); //cancels the animation

keep in mind that you will still need to run $scope.$apply inside of the then callback to trigger a digest.

$animate.addClass, $animate.removeClass and $animate.setClass will no longer start the animation right after being called in the directive code. The animation will only commence once a digest has passed. This means that all animation-related testing code requires an extra digest to kick off the animation.

//before this fix
$animate.addClass(element, 'super');
expect(element).toHaveClass('super');

//now
$animate.addClass(element, 'super');
$rootScope.$digest();
expect(element).toHaveClass('super');

$animate will also tally the amount of times classes are added and removed and only animate the left over classes once the digest kicks in. This means that for any directive code that adds and removes the same CSS class on the same element then this may result in no animation being triggered at all.

$animate.addClass(element, 'klass');
$animate.removeClass(element, 'klass');

$rootScope.$digest();

//nothing happens...

The value of $binding data property on an element is always an array now and the expressions do not include the curly braces {{ ... }}.

  • currencyFilter: due to c2aaddbe, previously the currency filter would convert null and undefined values into empty string, after this change these values will be passed through.

Only cases when the currency filter is chained with another filter that doesn't expect null/undefined will be affected. This should be very rare.

This change will not change the visual output of the filter because the interpolation will convert the null/undefined to an empty string.

Closes #8605

  • numberFilter: due to 2ae10f67, previously the number filter would convert null and undefined values into empty string, after this change these values will be passed through.

Only cases when the number filter is chained with another filter that doesn't expect null/undefined will be affected. This should be very rare.

This change will not change the visual output of the filter because the interpolation will convert the null/undefined to an empty string.

Closes #8605 Closes #8842

NgModel.viewValue will always be used when rendering validations for minlength and maxlength.

Closes #7967 Closes #8811

According to the HTML5 spec input[time] should create dates based on the year 1970 (used to be based on the year 1900).

Related to #8447.

Any parser code from before that returned an undefined value (or nothing at all) will now cause a parser failure. When this occurs none of the validators present in $validators will run until the parser error is gone. The error will be stored on ngModel.$error.

The blur and focus event fire synchronously, also during DOM operations that remove elements. This lead to errors as the Angular model was not in a consistent state. See this fiddle for a demo.

This change executes the expression of those events using scope.$evalAsync if an $apply is in progress, otherwise keeps the old behavior.

Fixes #4979 Fixes #5945 Closes #8803 Closes #6910 Closes #5402

The returned value from directive controller constructors are now ignored, and only the constructed instance itself will be attached to the node's expando. This change is necessary in order to ensure that it's possible to bind properties to the controller's instance before the actual constructor is invoked, as a convenience to developers.

In the past, the following would have worked:

angular.module("myApp", []).
    directive("myDirective", function() {
        return {
            controller: function($scope) {
                return {
                    doAThing: function() { $scope.thingDone = true; },
                    undoAThing: function() { $scope.thingDone = false; }
                };
            },
            link: function(scope, element, attrs, ctrl) {
                ctrl.doAThing();
            }
        };
    });

However now, the reference to doAThing() will be undefined, because the return value of the controller's constructor is ignored. In order to work around this, one can opt for several strategies, including the use of _.extend() or merge() like routines, like so:

angular.module("myApp", []).
    directive("myDirective", function() {
        return {
            controller: function($scope) {
                _.extend(this, {
                    doAThing: function() { $scope.thingDone = true; },
                    undoAThing: function() { $scope.thingDone = false; }
                });
            },
            link: function(scope, element, attrs, ctrl) {
                ctrl.doAThing();
            }
        };
    });

1.2.23 superficial-malady (2014-08-22)

Bug Fixes

  • $location:
    • rewrite relative URI correctly if path==='/' in legacy html5Mode (c6e4defc, #8684)
    • don't call indexOf() of undefined href attribute (74a7afcb, #7721, #8681)
  • $sanitize: sanitize javascript urls with comments (4f387050, #8274)
  • Angular: make Date comparison in equals() NaN-aware (98f60372, #8650, #8715)
  • copy: clear array destinations correctly for non-array sources (888b0f54, #8610, #8702)
  • input:
    • use lowercase method to account for undefined type (456026ef)
    • by default, do not trim input[type=password] values (ebece0bc, #8250, #8230)
  • linky: handle quotes around email addresses (effc98fd, #8520)
  • minErr: encode btstrpd error input to strip angle brackets (aaf9c5e5, #8683)
  • ngHref: remove attribute when empty value instead of ignoring (ed56872b, #2755)

Breaking Changes

Previously, input[type=password] would trim values by default, and would require an explicit ng-trim="false" to disable the trimming behavior. After this CL, ng-trim no longer affects input[type=password], and will never trim the password value.

Closes #8250 Closes #8230

1.3.0-beta.19 rafter-ascension (2014-08-22)

Bug Fixes

  • $compile:
    • use the correct namespace for transcluded SVG elements (ffbd276d, #8716)
    • update the jQuery .context when an element is replaced by replace:true directive (f02f7d9c, #8253, #7900)
  • $location:
    • rewrite relative URI correctly if path==='/' in legacy html5Mode (d18b2819, #8684)
    • don't call indexOf() of undefined href attribute (5b77e30c, #7721, #8681)
  • $parse: remove unused variable declaration in generated getters (6acea115)
  • $sanitize: sanitize javascript urls with comments (b7e82a33, #8274)
  • $watchGroup: call listener once when the watchExpressions array is empty (bf0e8373)
  • Angular: make Date comparison in equals() NaN-aware (693e846a, #8650, #8715)
  • Scope: don't clear the phase when an exception is thrown from asyncQueue or watch (bf1a57ad)
  • copy: clear array destinations correctly for non-array sources (a603e202, #8610, #8702)
  • forEach: match behavior of Array.prototype.forEach (ignore missing properties) (36230194, #8510, #8522, #8525)
  • input:
    • use lowercase method to account for undefined type (066c0499)
    • by default, do not trim input[type=password] values (a7fb357f, #8250, #8230)
  • jQuery: cooperate with other libraries monkey-patching jQuery.cleanData (b9389b26, #8471)
  • jqLite:
  • linky: handle quotes around email addresses (a9d22712, #8520)
  • minErr: encode btstrpd error input to strip angle brackets (0872388a, #8683)
  • ngRepeat:
    • allow aliasAs identifiers which contain but do not match reserved words (d713ad1b, #8729)
    • make allowed aliasAs expressions more strict (09b29870, #8438, #8440)

Features

  • $compile:
  • benchpress: configure benchpress grunt task (6bdaa4bc)
  • jqLite: implement the detach method (1a05daf5, #5461)
  • ngRoute: add method for changing url params (77a1acc7)

Performance Improvements

  • $compile:
    • don't register $destroy callbacks on element-transcluded nodes (b5f7970b)
    • refactor publicLinkFn to simplify the code and use 'for in' loop (645625cf)
    • clone the nodeList during linking only if necessary (3e0a2e1f)
    • delay object initialization in nodeLinkFn (31ed0af7)
    • optimize nodeLinkFn (35134a0e)
    • optimize publicLinkFn (274e9c4d)
  • $interpolate: do not keep empty separators (94b5c9f0)
  • $parse:
    • don't bind filters to a context (8863b9d0)
    • optimize filter implementation (ece6ef47)
    • speed up fn invocation for no args case (a17578ad)
    • speed up fn invocation by optimizing arg collection (fecfc5b0)
    • use no-proto maps as caches and avoid hasOwnProperty checks (d302ea0c)
    • trim expression only if string (a1341223)
  • $rootScope: do not use Function::call when not needed (7eae29e5)
  • Scope:
    • optimize $watchCollection when used for watching objects (e822e906)
    • don't use forEach in (301463a2)
    • watchCollection optimization (7d96ab0d)
    • exit $broadcast early if nobody is listening for the given event (a09fa356)
    • use remove the need for the extra watch in $watchGroup (3f0e642e, #8396)
  • benchpress: add benchpress node module and port over large table test (1229334f)
  • isObject: use strict comparison (d208ba25)
  • jqLite:
    • simplify jqLiteDealoc (f8f7a1df)
    • optimize event handler (d05f27e2)
    • only take str.split() path when needed (187b1b8e, #8648)
    • optimize off() (abb17cce)
    • refactor jqLiteExpandoStore to minimize access to expensive element.ng339 expando property (1e8698b3)
    • microoptimization in chaining fn (fafbd494)
    • don't use String#split in on() unless we need it (bda673f8)
    • don't check isString many times in constructor (443b521e)
    • optimize jqLiteAcceptsData method (b493c62f)
    • optimize append() and after() (8d933bf9)
    • don't register DOM listener for $destroy event (6251751a)
    • optimize event listener registration (566f1015)
    • improve createEventHandler method by switching from forEach to for loop (e9cd6dc0)
    • don't use forEach in off() (960a8410)
    • don't recreate the Node.contains polyfill (d1536e7c)
    • speed up shallowCopy and special case Attributes cloning (54fa16e4)
  • ngBind: bypass jquery/jqlite when setting text (0a738ce1)
  • ngRepeat:
    • simplify code and remove duplicate array.length access (08eb0558)
    • optimize marking of nodes that are being removed via an animation (36e35b2c)
    • use no-proto objects for blockMaps (13d113c5)
    • move work to compile fn (bdd853cb)
    • move updateScope fn to factory and reuse it for all repeaters (e58d65a5)
    • clone boundary comment nodes (fbd48845)

Breaking Changes

Now, ng-attr-* will never add the attribute to the DOM if any of the interpolated expressions evaluate to undefined.

To work around this, initialize values which are intended to be the empty string with the empty string:

For example, given the following markup:

<div ng-attr-style="border-radius: {{value}}{{units}}"></div>

If $scope.value is 4, and $scope.units is undefined, the resulting markup is unchanged:

<div ng-attr-style="border-radius: {{value}}{{units}}"></div>

However, if $scope.units is "", then the resulting markup is updated:

<div ng-attr-style="border-radius: {{value}}{{units}}" style="border-radius: 4"></div>

Closes #8376 Closes #8399

  • due to 0d608d04, element-transcluded directives now have an extra comment automatically appended to their cloned DOM

This comment is usually needed to keep track the end boundary in the event child directives modify the root node(s). If not used for this purpose it can be safely ignored.

  • due to 75c4cbf8, directive.type was renamed to directive.templateNamespace

This change is breaking only within 1.3.0-beta releases: directive.type was renamed to directive.templateNamespace

The property name type was too general.

  • $parse: due to 8863b9d0, this in filters is now undefined and no longer the scope

It's a bad practice for filters to have hidden dependencies, so pulling stuff from scope directly is not a good idea. Scope being the filter context was never documented as public API, so we don't expect that any significant code depends on this behavior.

If an existing filter has a dependency on the scope instance, the scope reference can be passed into the filter as a filter argument (this is highly discouraged for new code):

Before: {{ user.name | customFilter }} After: {{ user.name | customFilter:this }}

  • Scope: due to 0554c1aa, deregisterNotifier callback for $watch is no longer available

This API was available only in the last few 1.3 beta versions and is not very useful for applications, so we don't expect that anyone will be affected by this change.

  • input: due to a7fb357f, by default, do not trim input[type=password] values.

Previously, input[type=password] would trim values by default, and would require an explicit ng-trim="false" to disable the trimming behavior. After this change, ng-trim no longer affects input[type=password], and will never trim the password value.

Closes #8250 Closes #8230

1.3.0-beta.18 spontaneous-combustion (2014-08-12)

Bug Fixes

  • $compile: make '='-bindings NaN-aware (5038bf79, #8553, #8554)
  • $location: add semicolon to whitelist of delimiters to unencode (36258033, #5019)
  • $parse:
    • one-time binding for literal expressions works as expected (c024f282, #8209)
    • correctly assign expressions who's path is undefined and that use brackets notation (c03ad249, #8039)
  • Scope: add deregisterNotifier to oneTimeLiteralWatch signature (a001a417)
  • jqLite:
    • allow triggerHandler() to accept custom event (01d81cda, #8469)
    • fix regression where mutating the dom tree on a event breaks jqLite.remove (a00c9bca, #8359)
  • ngSanitize: ensure html is a string in htmlParser() (34781f18, #8417, #8416)
  • select:
    • ensure that at least one option has the selected attribute set (25a476ea, #8366, #8429)
    • do not update selected property of an option element on digest with no change event (cdc7db3f, #8221, #7715)

Features

Performance Improvements

  • $q: move Deferred and Promise methods to prototypes (23bc92b1, #8300)
  • input: prevent additional $digest when input is already touched (dd2a803f, #8450)

Breaking Changes

  • jQuery: due to 9e7cb3c3, Angular no longer supports jQuery versions below 2.1.1.
  • $q: due to 23bc92b1, Promises methods are no longer enumerated when using for-loops with hasOwnProperty check. E.g. angular.extends

1.2.22 finicky-pleasure (2014-08-12)

Bug Fixes

  • $compile: make '='-bindings NaN-aware (0b0acb03, #8553, #8554)
  • $parse: correctly assign expressions who's path is undefined and that use brackets notation (60366c8d, #8039)
  • jqLite: allow triggerHandler() to accept custom event (d262378b, #8469, #8505)
  • ngSanitize: ensure html is a string in htmlParser() (9ee07551, #8417, #8416)
  • select:
    • ensure that at least one option has the selected attribute set (79538afd, #8366, #8429)
    • do not update selected property of an option element on digest with no change event (c2860944, #8221, #7715)

Features

1.3.0-beta.17 turing-autocompletion (2014-07-25)

Bug Fixes

  • angular.copy: clone regexp flags correctly (86340a59, #5781, #8337)
  • docs: change plnkr form to open in same window (925b2080)
  • jqLite: triggerHandler support unbind self (8a27abae, #5984)
  • ngHref: remove attribute when empty value instead of ignoring (469ea338, #2755)

Features

  • $compile: change directive's restrict setting to default to EA (element/attribute) (11f5aeee, #8321)
  • $q: add streamlined ES6-style interface for using $q (f3a763fd, #8311, #6427)
  • ngRepeat: provide support for aliasing filtered repeater results as a scope member (e0adb9c4, #5919, #8046, #8282)

Performance Improvements

  • $parse: don't use reflective calls in generated functions (c54228fb)

Breaking Changes

  • $compile: due to 11f5aeee, directives now match elements by default unless specific restriction rules are set via restrict property.

This means that if a directive 'myFoo' previously didn't specify matching restriction, it will now match both the attribute and element form.

Before:

<div my-foo></div> <---- my-foo attribute matched the directive

<my-foo></my-foo> <---- no match

After:

<div my-foo></div> <---- my-foo attribute matched the directive

<my-foo></my-foo> <---- my-foo element matched the directive

It is not expected that this will be a problem in practice because of widespread use of prefixes that make <my-foo> like elements unlikely.

Closes #8321

1.2.21 wizard-props (2014-07-25)

Bug Fixes

Performance Improvements

  • $compile: only create jqLite object when necessary (71eb1901)
  • $parse: don't use reflective calls in generated functions (cbdf0c2a)
  • forEach: use native for loop instead of forEach for Arrays (492b0cdf)
  • jqLite: expose the low-level jqLite.data/removeData calls (3c46c943)
  • ngBindHtml: move addClass to the compile phase (8eede099, #8261)

1.3.0-beta.16 pizza-transubstantiation (2014-07-18)

Bug Fixes

Features

Performance Improvements

  • $compile: only create jqLite object when necessary (a160f76f)
  • bindOnce more performant interpolation and lazy one-time binding (86d55c1d)
  • jqLite: expose the low-level jqLite.data/removeData calls (e4ba8943)
  • ngBindHtml: move addClass to the compile phase (903e7352, #8261)

Breaking Changes

  • $compile: due to e8066c4b, Directives which previously depended on the implicit grouping between directive-start and directive-end attributes must be refactored in order to see this same behavior.

Before:

<div data-fancy-directive-start>{{start}}</div>
  <p>Grouped content</p>
<div data-fancy-directive-end>{{end}}</div>
.directive('fancyDirective', function() {
  return {
    link: angular.noop
  };
})

After:

<div data-fancy-directive-start>{{start}}</div>
  <p>Grouped content</p>
<div data-fancy-directive-end>{{end}}</div>
.directive('fancyDirective', function() {
  return {
    multiElement: true, // Explicitly mark as a multi-element directive.
    link: angular.noop
  };
})

Closes #5372 Closes #6574 Closes #5370 Closes #8044 Closes #7336

Previously, it was possible for an action passed to $watch to be a string, interpreted as an angular expresison. This is no longer supported. The action now has to be a function. Passing an action to $watch is still optional.

Before:

$scope.$watch('state', ' name="" ');

After:

$scope.$watch('state', function () {
  $scope.name = "";
});

Closes #8190

If using any of the mechanisms specified above, then migrate by specifying the attribute ng-app to the root element. E.g.

<div ng-app="module">...</div>

Closes #8147

The ngList directive no longer supports splitting the view value via a regular expression. We need to be able to re-join list items back together and doing this when you can split with regular expressions can lead to inconsistent behavior and would be much more complex to support.

If your application relies upon ngList splitting with a regular expression then you should either try to convert the separator to a simple string or you can implement your own version of this directive for you application.

Closes #4008 Closes #2561 Closes #4344

** Directive Priority Changed ** - this commit changes the priority of ngSwitchWhen and ngSwitchDefault from 800 to 1200. This makes their priority higher than ngRepeat, which allows items to be repeated on the switch case element reliably.

In general your directives should have a lower priority than these directives if you want them to exist inside the case elements. If you relied on the priority of these directives then you should check that your code still operates correctly.

Closes #8235

1.3.0-beta.15 unbelievable-advancement (2014-07-11)

Bug Fixes

  • $animate:
  • $http:
    • don't remove content-type header if data is set by request transform (c7c363cf, #7910)
    • add ability to remove default headers (172a4093, #5784)
  • $location: remove query args when passed in object (2c7d0857, #6565)
  • input:
  • jqLite:
    • correctly dealoc svg elements in IE (012ab1f8)
    • remove exposed dealoc method (9c5b407f)
  • ngModel: test & update correct model when running $validate (f3cb2741, #7836, #7837)
  • parseKeyValue: ignore properties in prototype chain (cb42766a, #8070, #8068)
  • select: auto-select new option that is marked as selected (b8ae73e1, #6828)

Features

  • $animate: allow directives to cancel animation events (ca752790, #7722)
  • $controller: disable using global controller constructors (3f2232b5)
  • FormController: add $rollbackViewValue to rollback all controls (85b77314, #7595)
  • input: support constant expressions for ngTrueValue/ngFalseValue (c90cefe1, #8041, #5346, #1199)
  • ngAnimate: conditionally allow child animations to run in parallel with parent animations (8252b8be, #7946)
  • ngModel: bind to getters/setters (b9fcf017, #768)

Performance Improvements

  • $compile: no longer need nodeType filter when setting $scope data (b0ca5195, #7887)

Breaking Changes

$controller will no longer look for controllers on window. The old behavior of looking on window for controllers was originally intended for use in examples, demos, and toy apps. We found that allowing global controller functions encouraged poor practices, so we resolved to disable this behavior by default.

To migrate, register your controllers with modules rather than exposing them as globals:

Before:

function MyController() {
  // ...
}

After:

angular.module('myApp', []).controller('MyController', [function() {
  // ...
}]);

Although it's not recommended, you can re-enable the old behavior like this:

angular.module('myModule').config(['$controllerProvider', function($controllerProvider) {
  // this option might be handy for migrating old apps, but please don't use it
  // in new ones!
  $controllerProvider.allowGlobals();
}]);

Previously, these attributes would always be treated as strings. However, they are now parsed as expressions, and will throw if an expression is non-constant.

To convert non-constant strings into constant expressions, simply wrap them in an extra pair of quotes, like so:

<input type="checkbox" ng-model="..." ng-true-value="'truthyValue'">

Closes #8041 Closes #5346 Closes #1199

1.2.20 accidental-beautification (2014-07-11)

Bug Fixes

  • $http:
    • don't remove content-type header if data is set by request transform (7027844d, #7910)
    • add ability to remove default headers (172a4093, #5784)
  • $location: remove query args when passed in object (a26acb64, #6565)
  • input:
  • parseKeyValue: ignore properties in prototype chain (873acf8f, #8070, #8068)

Features

  • ngAnimate: conditionally allow child animations to run in parallel with parent animations (931789ec, #7946)

1.3.0-beta.14 harmonious-cacophonies (2014-06-30)

This release contains security fixes for $parse that prevent arbitrary code execution via Angular expressions under some very specific conditions. The only applications affected by these vulnerabilities are those that match all of the following conditions:

  • application mixes server-side and client-side templating
  • the server-side templating contains XSS vulnerabilities
  • the vulnerabilities in the server-side templating are being guarded by server-side XSS filters or on the client-side via CSP
  • the server-side XSS vulnerabilities can be used to augment the client-side template processed by Angular

Applications not meeting all of the conditions are not vulnerable.

This fix is in both 1.3.0-beta.14 and 1.2.19 release.

The Angular team would like to thank Jann Horn for reporting these vulnerabilities via [security@angularjs.org].

Bug Fixes

  • $compile: bind ng-attr-* even if unbound attribute follows ng-attr-* (8b0258d8, #7739)
  • $http:
    • should not read statusText on IE<10 when request is aborted (31ae3e71)
    • add the PATCH shortcut back (b28b5caa, #5894)
  • $injector: check if a fn is an array explicitly (b1a6baac, #7904, #2653)
  • $interval: when canceling, use clearInterval from $window instead of global scope. (a4904c0f)
  • $parse:
    • prevent invocation of Function's bind, call and apply (77ada4c8)
    • forbid proto properties in angular expressions (6081f207)
    • forbid {define,lookup}{Getter,Setter} properties (48fa3aad)
    • forbid referencing Object in angular expressions (528be29d)
    • handle constants as one-time binding expressions (d9763f1b, #7970)
  • $timeout/$interval: if invokeApply is false, do not use evalAsync (19b6b343, #7999, #7103)
  • Angular: nodeName should always be lowercase (dafb8a3c, #3987)
  • Angular.copy: preserve prototype chain when copying objects (b59b04f9, #5063, #3767, #4996)
  • core: drop the toBoolean function (bdfc9c02, #3969, #4277, #7960)
  • injector: allow multiple loading of function modules (2f0a4488, #7255)
  • input:
  • jqLite:
    • never add to the cache for non-element/document nodes (91754a76, #7966)
    • don't attach event handlers to comments or text nodes (462dbb20, #7913, #7942)
    • convert NodeList to an Array to make PhantomJS 1.x happy (ceaea861, #7851)
  • numberFilter: correctly round fractions despite floating-point arithmetics issues in JS (189cd064, #7870, #7878)
  • testabilityPatch: fix invocations of angular.mock.dump (e8e07502)

Features

  • NgModel:
    • port the email input type to use the validators pipeline (67379242)
    • port the URL input type to use the validators pipeline (3ee65730)
  • jqLite: support isDefaultPrevented for triggerHandler dummies (7e71acd1, #8008)

Performance Improvements

  • forEach: use native for loop instead of forEach for Arrays (36625de0)

Breaking Changes

You can no longer invoke .bind, .call or .apply on a function in angular expressions. This is to disallow changing the behavior of existing functions in an unforeseen fashion.

The (deprecated) proto property does not work inside angular expressions anymore.

This prevents the use of {define,lookup}{Getter,Setter} inside angular expressions. If you really need them for some reason, please wrap/bind them to make them less dangerous, then make them available through the scope object.

This prevents the use of Object inside angular expressions. If you need Object.keys, make it accessible in the scope.

This changes angular.copy so that it applies the prototype of the original object to the copied object. Previously, angular.copy would copy properties of the original object's prototype chain directly onto the copied object.

This means that if you iterate over only the copied object's hasOwnProperty properties, it will no longer contain the properties from the prototype. This is actually much more reasonable behavior and it is unlikely that applications are actually relying on this.

If this behavior is relied upon, in an app, then one should simply iterate over all the properties on the object (and its inherited properties) and not filter them with hasOwnProperty.

Be aware that this change also uses a feature that is not compatible with IE8. If you need this to work on IE8 then you would need to provide a polyfill for Object.create and Object.getPrototypeOf.

  • core: due to bdfc9c02, values 'f', '0', 'false', 'no', 'n', '[]' are no longer treated as falsy. Only JavaScript falsy values are now treated as falsy by the expression parser; there are six of them: false, null, undefined, NaN, 0 and "".

Closes #3969 Closes #4277 Closes #7960

Previously, even if invokeApply was set to false, a $rootScope digest would occur during promise resolution. This is no longer the case, as promises returned from $timeout and $interval will no longer trigger $evalAsync (which in turn causes a $digest) if invokeApply is false.

Workarounds include manually triggering $scope.$apply(), or returning $q.defer().promise from a promise callback, and resolving or rejecting it when appropriate.

var interval = $interval(function() {
  if (someRequirementFulfilled) {
    $interval.cancel(interval);
    $scope.$apply();
  }
}, 100, 0, false);

or:

var interval = $interval(function (idx) {
  // make the magic happen
}, 1000, 10, false);
interval.then(function(idx) {
  var deferred = $q.defer();
  // do the asynchronous magic --- $evalAsync will cause a digest and cause
  // bindings to update.
  return deferred.promise;
});

1.2.19 precognitive-flashbacks (2014-06-30)

Bug Fixes

  • $compile: bind ng-attr-* even if unbound attribute follows ng-attr-* (ed59370d)
  • $http: should not read statusText on IE<10 when request is aborted (0c80df21)
  • $injector: check if a fn is an array explicitly (67c11b9a, #7904, #2653)
  • $interval: when canceling, use clearInterval from $window instead of global scope. (f780ccfa)
  • $parse:
    • make the window check in ensureSafeObject IE8 friendly (ba62e975)
    • prevent invocation of Function's bind, call and apply (07fa87a8)
    • forbid proto properties in angular expressions (cb713e60)
    • forbid {define,lookup}{Getter,Setter} properties (89ca8597)
    • forbid referencing Object in angular expressions (bc6fb7cc)
  • injector: allow multiple loading of function modules (d71f16e7, #7255)
  • input:
  • jqLite: change expando property to a more unique name (74e1cc68)
  • numberFilter: correctly round fractions despite floating-point arithmetics issues in JS (e5f454c8, #7870, #7878)
  • testabilityPatch: fix invocations of angular.mock.dump (5e944a1c)

Performance Improvements

  • jqLite: don't use reflection to access expandoId (a4faa5cd)

Breaking Changes

You can no longer invoke .bind, .call or .apply on a function in angular expressions. This is to disallow changing the behavior of existing functions in an unforeseen fashion.

The (deprecated) proto property does not work inside angular expressions anymore.

This prevents the use of {define,lookup}{Getter,Setter} inside angular expressions. If you really need them for some reason, please wrap/bind them to make them less dangerous, then make them available through the scope object.

This prevents the use of Object inside angular expressions. If you need Object.keys, make it accessible in the scope.

1.3.0-beta.13 idiosyncratic-numerification (2014-06-16)

Bug Fixes

  • jqLite: change expando property to a more unique name (20c3c9e2)

1.3.0-beta.12 ephemeral-acceleration (2014-06-13)

Bug Fixes

  • $compile:
  • $injector: report circularity in circular dependency error message (545d22b4, #7500)
  • $parse: Handle one-time to null (600a41a7, #7743, #7787)
  • NgModel:
    • ensure pattern and ngPattern use the same validator (1be9bb9d)
    • make ngMinlength and ngMaxlength as standalone directives (26d91b65, #6750)
    • make sure the ngMinlength and ngMaxlength validators use the $validators pipeline (5b8e7ecf, #6304)
    • make sure the pattern validator uses the $validators pipeline (e63d4253)
    • make sure the required validator uses the $validators pipeline (e53554a0, #5164)
  • jqLite: data should store data only on Element and Document nodes (a196c8bc)
  • ngResource: don't convert literal values into Resource objects when isArray is true (16dfcb61, #6314, #7741)

Features

  • NgModel: introduce the $validators pipeline (a8c7cb81)
  • attrs: trigger observers for specific ng-attributes (d9b90d7c, #7758)
  • input: add $touched and $untouched states (adcc5a00)
  • ngInclude: emit $includeContentError when HTTP request fails (e4419daf, #5803)

Performance Improvements

  • $compile: move ng-binding class stamping for interpolation into compile phase (35358fdd)
  • $http: move xsrf cookie check to after cache check in $http (dd1d189e, #7717)
  • Scope: change Scope#id to be a simple number (8c6a8171)
  • forEach: cache array length (55991e33)
  • isArray: use native Array.isArray (751ebc17, #7735)
  • isWindow optimize internal isWindow call (b68ac4cb)
  • jqLite:
    • cache collection length for all methods that work on a single element (41d2eba5)
    • improve performance of jqLite#text (92489886)
    • optimize adding nodes to a jqLite collection (31faeaa7)
    • optimize element deallocation (e35abc9d)
    • don't use reflection to access expandoId (ea9a130a)
  • ngBind: set the ng-binding class during compilation instead of linking (fd5f3896)
  • shallowCopy: use Object.keys to improve performance (04468db4)

Breaking Changes

Requesting isolate scope and any other scope on a single element is an error. Before this change, the compiler let two directives request a child scope and an isolate scope if the compiler applied them in the order of non-isolate scope directive followed by isolate scope directive.

Now the compiler will error regardless of the order.

If you find that your code is now throwing a $compile:multidir error, check that you do not have directives on the same element that are trying to request both an isolate and a non-isolate scope and fix your code.

Closes #4402 Closes #4421

If an expression is used on ng-pattern (such as ng-pattern="exp") or on the pattern attribute (something like on pattern="{{ exp }}") and the expression itself evaluates to a string then the validator will not parse the string as a literal regular expression object (a value like /abc/i). Instead, the entire string will be created as the regular expression to test against. This means that any expression flags will not be placed on the RegExp object. To get around this limitation, use a regular expression object as the value for the expression.

//before
$scope.exp = '/abc/i';

//after
$scope.exp = /abc/i;
  • Scope: due to 8c6a8171, Scope#$id is now of time number rather than string. Since the id is primarily being used for debugging purposes this change should not affect anyone.
  • forEach: due to 55991e33, forEach will iterate only over the initial number of items in the array. So if items are added to the array during the iteration, these won't be iterated over during the initial forEach call.

This change also makes our forEach behave more like Array#forEach.

  • jqLite: due to a196c8bc, previously it was possible to set jqLite data on Text/Comment nodes, but now that is allowed only on Element and Document nodes just like in jQuery. We don't expect that app code actually depends on this accidental feature.

1.2.18 ear-extendability (2014-06-13)

Bug Fixes

  • $compile:
    • ensure transclude works at root of templateUrl (fd420c40, #7183, #7772)
    • bound transclusion to correct scope (1382d4e8)
    • don't pass transcludes to non-transclude templateUrl directives (b9ddef2a)
    • don't pass transclude to template of non-transclude directive (eafba9e2)
    • fix nested isolated transclude directives (bb931097, #1809, #7499)
    • pass transcludeFn down to nested transclude directives (8df5f325, #7240, #7387)
  • $injector: report circularity in circular dependency error message (14e797c1, #7500)
  • ngResource: don't convert literal values into Resource objects when isArray is true (f0904cf1, #6314, #7741)

Performance Improvements

  • $compile: move ng-binding class stamping for interpolation into compile phase (81b7e5ab)
  • $http: move xsrf cookie check to after cache check in $http (8b86d363, #7717)
  • isArray: use native Array.isArray (6c14fb1e)
  • jqLite: cache collection length for all methods that work on a single element (6d418ef5)
  • ngBind: set the ng-binding class during compilation instead of linking (1b189027)

1.3.0-beta.11 transclusion-deforestation (2014-06-06)

Bug Fixes

  • $animate: remove the need to add display:block !important for ngShow/ngHide (7c011e79, #3813)
  • $compile:
    • bound transclusion to correct scope (56c60218)
    • set the iteration state before linking (0c8a2cd2)
    • don't pass transcludes to non-transclude templateUrl directives (2ee29c5d)
    • don't pass transclude to template of non-transclude directive (19af0397)
    • fix nested isolated transclude directives (d414b787, #1809, #7499)
    • pass transcludeFn down to nested transclude directives (1fef5fe8, #7240, #7387)
  • $parse: fix parsing error with leading space and one time bind (24c844df, #7640)
  • angular.copy: support circular references in the value being copied (083f496d, #7618)
  • angular.toJson: only strip properties beginning with $$, not $ (c054288c)
  • ngAnimate:
    • $animate methods should accept native DOM elements (222d4737)
    • fix property name that is used to calculate cache key (9f5c4370, #7566)
  • ngClass: support multiple classes in key (7eaaca8e)
  • ngIf: ensure that the correct (transcluded) scope is used (d71df9f8)
  • ngLocale: fix i18n code-generation to support get_vf_, decimals_, and get_wt_ (cbab51ca)
  • ngRepeat: ensure that the correct (transcluded) scope is used (b87e5fc0)
  • ngShow: ensure that the display property is never set to block (1d90744f, #7707)

Features

  • $resource: allow props beginning with $ to be used on resources (d3c50c84)

Breaking Changes

The isolated scope of a component directive no longer leaks into the template that contains the instance of the directive. This means that you can no longer access the isolated scope from attributes on the element where the isolated directive is defined.

See https://github.com/angular/angular.js/issues/10236 for an example.

  • $resource: due to d3c50c84,

    If you expected $resource to strip these types of properties before, you will have to manually do this yourself now.

  • angular.toJson: due to c054288c,

    toJson() will no longer strip properties starting with a single $. If you relied on toJson()'s stripping these types of properties before, you will have to do it manually now. It will still strip properties starting with $$ though.

1.2.17 - quantum disentanglement (2014-06-06)

Bug Fixes

  • $animate:
    • remove the need to add display:block!important for ngShow/ngHide (55b2f0e8, #3813)
    • retain inline styles for property-specific transitions (ad08638c, #7503)
    • ensure class-based animations always perform a DOM operation if skipped (34d07403, #6957)
  • $compile:
    • do not merge attrs that are the same for replace directives (b635903e, #7463)
    • pass transcludeFn down to nested transclude directives (11385060, #7240, #7387)
    • set $isolateScope correctly for sync template directives (5319621a, #6942)
    • reference correct directive name in ctreq error (6bea0591, #7062, #7067)
    • fix regression which affected old jQuery releases (a97a172e)
  • $httpBackend: don't error when JSONP callback is called with no parameter (a7ccb753, #7031)
  • $location:
  • angular.copy: support circular references in the value being copied (5c997209, #7618)
  • grunt-utils: ensure special inline CSS works when angular is not a global (d4231171, #7176)
  • input:
    • fix ReferenceError in event listener (2d7cb14a)
    • don't dirty model when input event is triggered due to a placeholder change (109e5d1d, #2614, #5960)
  • jqLite: use jQuery only if jQuery.fn.on is present (fafcd628)
  • limitTo: do not convert Infinity to NaN (fcdac65a, #6771, #7118)
  • ngAnimate: $animate methods should accept native DOM elements (9227a5db)
  • ngClass:
    • support multiple classes in key (85ce5d0d)
    • handle index changes when an item is unshifted (a4cc9e19, #7256)
  • ngLocale: fix i18n code-generation to support get_vf_, decimals_, and get_wt_ (96a31476)
  • ngSanitize: encode surrogate pair properly (3d0b49c0, #5088, #6911)
  • ngSwitch: properly support case labels with different numbers of transclude fns (32aa4915)
  • numberFilter: fix rounding error edge case (0388eed7, #7453, #7478)

Features

Performance Improvements

  • $interpolate: optimize value stringification (9d4fa33e, #7501)
  • scope: 10x. Share the child scope class. (9ab9bf6b)

1.3.0-beta.10 excessive-clarification (2014-05-23)

Bug Fixes

  • $animate: retain inline styles for property-specific transitions (98b9d68e, #7503)
  • $compile: do not merge attrs that are the same for replace directives (1ab6e908, #7463)
  • $parse: remove deprecated promise unwrapping (fa6e411d)
  • Scope: $broadcast and $emit should set event.currentScope to null (82f45aee, #7445, #7523)
  • ngModel: do not dirty the input on $commitViewValue if nothing was changed (facd904a, #7457, #7495)

Features

Performance Improvements

Breaking Changes

  • $compile: due to eec6394a, The replace flag for defining directives that replace the element that they are on will be removed in the next major angular version. This feature has difficult semantics (e.g. how attributes are merged) and leads to more problems compared to what it solves. Also, with Web Components it is normal to have custom elements in the DOM.

  • $parse: due to fa6e411d, promise unwrapping has been removed. It has been deprecated since 1.2.0-rc.3. It can no longer be turned on. Two methods have been removed:

    • $parseProvider.unwrapPromises
    • $parseProvider.logPromiseWarnings
  • Scope: due to 82f45aee, #7445, #7523 $broadcast and $emit will now reset the currentScope property of the event to null once the event finished propagating. If any code depends on asynchronously accessing their currentScope property, it should be migrated to use targetScope instead. All of these cases should be considered programming bugs.

1.3.0-beta.9 release-naming (2014-05-16)

Bug Fixes

  • $compile: pass transcludeFn down to nested transclude directives (4f03dc5a, #7240, #7387)
  • jqLite: use jQuery only if jQuery.fn.on present (e9bc51cb)
  • ngClass: handle index changes when an item is unshifted (5fbd618c, #7256)
  • ngMessages: annotate ngMessages controller for minification (0282ca97)
  • numberFilter: fix rounding error edge case (81d427b5, #7453, #7478)

Features

  • ngTouch: add optional ngSwipeDisableMouse attribute to ngSwipe directives to ignore mouse events. (5a568b4f, #6627, #6626)

Breaking Changes

  • jqLite: due to d71dbb1a, the jQuery detach() method does not trigger the $destroy event. If you want to destroy Angular data attached to the element, use remove().

1.3.0-beta.8 accidental-haiku (2014-05-09)

Bug Fixes

  • $compile: set $isolateScope correctly for sync template directives (562c4e42, #6942)
  • $httpBackend: Add missing expectHEAD() method (e1d61784, #7320)
  • $interpolate: don't ReferenceError when context is undefined (924ee6db, #7230, #7237)
  • grunt-utils: ensure special inline CSS works when angular is not a global (af72f40a, #7176)
  • injector: invoke config blocks for module after all providers (c0b4e2db, #7139, #7147)
  • ngModelOptions:
  • ngSanitize: encode surrogate pair properly (627b0354, #5088, #6911)
  • ngSrc, ngSrcset: only interpolate if all expressions are defined (8d180383, #6984)
  • ngSwitch: properly support case labels with different numbers of transclude fns (ac37915e, #7372, #7373)

Features

  • $compile: allow SVG and MathML templates via special type property (f0e12ea7, #7265)
  • $interpolate: add optional allOrNothing param (c2362e3f)
  • FormController: commit $viewValue of all child controls when form is submitted (a0ae07bd, #7017)
  • NgMessages: introduce the NgMessages module and directives (0f4016c8)

Breaking Changes

Previously, it was possible to register a response interceptor like so:

// register the interceptor as a service
$provide.factory('myHttpInterceptor', function($q, dependency1, dependency2) {
  return function(promise) {
    return promise.then(function(response) {
      // do something on success
      return response;
    }, function(response) {
      // do something on error
      if (canRecover(response)) {
        return responseOrNewPromise
      }
      return $q.reject(response);
    });
  }
});

$httpProvider.responseInterceptors.push('myHttpInterceptor');

Now, one must use the newer API introduced in v1.1.4 (4ae46814), like so:

$provide.factory('myHttpInterceptor', function($q) {
  return {
    response: function(response) {
      // do something on success
      return response;
    },
    responseError: function(response) {
      // do something on error
      if (canRecover(response)) {
        return responseOrNewPromise
      }
      return $q.reject(response);
    }
  };
});

$httpProvider.interceptors.push('myHttpInterceptor');

More details on the new interceptors API (which has been around as of v1.1.4) can be found at https://docs.angularjs.org/api/ng/service/$http#interceptors

Previously, config blocks would be able to control behavior of provider registration, due to being invoked prior to provider registration. Now, provider registration always occurs prior to configuration for a given module, and therefore config blocks are not able to have any control over a providers registration.

Example:

Previously, the following:

angular.module('foo', [])
  .provider('$rootProvider', function() {
    this.$get = function() { ... }
  })
  .config(function($rootProvider) {
    $rootProvider.dependentMode = "B";
  })
  .provider('$dependentProvider', function($rootProvider) {
     if ($rootProvider.dependentMode === "A") {
       this.$get = function() {
        // Special mode!
       }
     } else {
       this.$get = function() {
         // something else
       }
    }
  });

would have "worked", meaning behavior of the config block between the registration of "$rootProvider" and "$dependentProvider" would have actually accomplished something and changed the behavior of the app. This is no longer possible within a single module.

This commit changes the API on NgModelController, both semantically and in terms of adding and renaming methods.

  • $setViewValue(value) - This method still changes the $viewValue but does not immediately commit this change through to the $modelValue as it did previously. Now the value is committed only when a trigger specified in an associated ngModelOptions directive occurs. If ngModelOptions also has a debounce delay specified for the trigger then the change will also be debounced before being committed. In most cases this should not have a significant impact on how NgModelController is used: If updateOn includes default then $setViewValue will trigger a (potentially debounced) commit immediately.
  • $cancelUpdate() - is renamed to $rollbackViewValue() and has the same meaning, which is to revert the current $viewValue back to the $lastCommittedViewValue, to cancel any pending debounced updates and to re-render the input.

To migrate code that used $cancelUpdate() follow the example below:

Before:

$scope.resetWithCancel = function (e) {
  if (e.keyCode == 27) {
    $scope.myForm.myInput1.$cancelUpdate();
    $scope.myValue = '';
  }
};

After:

$scope.resetWithCancel = function (e) {
  if (e.keyCode == 27) {
    $scope.myForm.myInput1.$rollbackViewValue();
    $scope.myValue = '';
  }
}

v1.3.0-beta.7 proper-attribution (2014-04-25)

Bug Fixes

  • $location: don't clobber path during parsing of path (498835a1, #7199)

Performance Improvements

  • scope: ~10x speedup from sharing the child scope class. (8377e818)

v1.3.0-beta.6 expedient-caffeination (2014-04-21)

Bug Fixes

  • $animate: ensure class-based animations always perform a domOperation if skipped (708f2ba9, #6957)
  • $compile:
    • reference correct directive name in ctreq error (1192531e, #7062, #7067)
    • fix regression which affected old jQuery releases (ef64169d)
  • $location:
  • input: don't dirty model when input event triggered due to placeholder change (ff428e72, #2614, #5960)
  • limitTo: do not convert Infinity to NaN (5dee9e4a, #6771, #7118)
  • ngModelController: introduce $cancelUpdate to cancel pending updates (940fcb40, #6994, #7014)

Features

  • $resource: Make stripping of trailing slashes configurable. (3878be52)
  • Scope: add $watchGroup method for observing a set of expressions (21f93163)
  • injector: "strict-DI" mode which disables "automatic" function annotation (4b1695ec, #6719, #6717, #4504, #6069, #3611)
  • ngModelOptions: custom triggers and debounce of ngModel updates (dbe381f2, #1285)

Performance Improvements

  • $compile: watch interpolated expressions individually (0ebfa0d1)
  • $interpolate: speed up interpolation by recreating watchGroup approach (546cb429)

Breaking Changes

  • $interpolate: due to 88c2193c, the function returned by $interpolate no longer has a .parts array set on it.

    Instead it has two arrays:

    • .expressions, an array of the expressions in the interpolated text. The expressions are parsed with $parse, with an extra layer converting them to strings when computed
    • .separators, an array of strings representing the separations between interpolations in the text. This array is always 1 item longer than the .expressions array for easy merging with it

1.3.0-beta.5 chimeric-glitterfication (2014-04-03)

Bug Fixes

  • $animate:
    • insert elements at the start of the parent container instead of at the end (1cb8584e, #4934, #6275)
    • ensure the CSS driver properly works with SVG elements (c67bd69c, #6030)
  • $parse: mark constant unary minus expressions as constant (7914d346, #6932)
  • Scope:
    • revert the __proto__ cleanup as that could cause regressions (71c11e96)
    • more scope clean up on $destroy to minimize leaks (d64d41ed, #6794, #6856, #6968)
  • ngClass: handle ngClassOdd/Even affecting the same classes (c9677920, #5271)

Breaking Changes

  • $animate: due to 1cb8584e, $animate will no longer default the after parameter to the last element of the parent container. Instead, when after is not specified, the new element will be inserted as the first child of the parent container.

To update existing code, change all instances of $animate.enter() or $animate.move() from:

$animate.enter(element, parent);

to:

$animate.enter(element, parent, angular.element(parent[0].lastChild));

1.2.16 badger-enumeration (2014-04-03)

Bug Fixes

  • $animate:
    • ensure the CSS driver properly works with SVG elements (38ea5426, #6030)
    • prevent cancellation timestamp from being too far in the future (35d635cb, #6748)
    • run CSS animations before JS animations to avoid style inheritance (0e5106ec, #6675)
  • $parse: mark constant unary minus expressions as constant (6e420ff2, #6932)
  • Scope:
  • filter.ngdoc: Check if "input" variable is defined (a275d539, #6819)
  • input: don't perform HTML5 validation on updated model-value (b2363e31, #6796, #6806)
  • ngClass: handle ngClassOdd/Even affecting the same classes (55fe6d63, #5271)

Features

1.3.0-beta.4 inconspicuous-deception (2014-03-28)

Bug Fixes

  • $animate:
    • prevent cancellation timestamp from being too far in the future (ff5cf736, #6748)
    • make CSS blocking optional for class-based animations (1bebe36a, #6674, #6739)
    • run CSS animations before JS animations to avoid style inheritance (2317af68, #6675)
  • Scope: aggressively clean up scope on $destroy to minimize leaks (f552f251, #6794, #6856)
  • doc-gen: Run Gulp on Windows too (47ba6014, #6346)
  • filter.ngdoc: Check if "input" variable is defined (4a6d4de5, #6819)
  • input: don't perform HTML5 validation on updated model-value (b472d027, #6796, #6806)

Features

Breaking Changes

  • $animate: due to 1bebe36a,

    Any class-based animation code that makes use of transitions and uses the setup CSS classes (such as class-add and class-remove) must now provide a empty transition value to ensure that its styling is applied right away. In other words if your animation code is expecting any styling to be applied that is defined in the setup class then it will not be applied "instantly" unless a transition:0s none value is present in the styling for that CSS class. This situation is only the case if a transition is already present on the base CSS class once the animation kicks off.

Before:

.animated.my-class-add {
  opacity:0;
  transition:0.5s linear all;
}
.animated.my-class-add.my-class-add-active {
  opacity:1;
}

After:

.animated.my-class-add {
  transition:0s linear all;
  opacity:0;
}
.animated.my-class-add.my-class-add-active {
  transition:0.5s linear all;
  opacity:1;
}

Please view the documentation for ngAnimate for more info.

1.3.0-beta.3 emotional-waffles (2014-03-21)

Bug Fixes

  • ngAnimate: support webkitCancelRequestAnimationFrame in addition to webkitCancelAnimationFrame (c839f78b, #6526)
  • $http: allow sending Blob data using $http (b8cc71d4, #5012)
  • $httpBackend: don't error when JSONP callback is called with no parameter (6680b7b9, #4987, #6735)
  • $rootScope: ng-repeat can't handle NaN values. #4605 (fb6062fb, #4605)
  • $rootScope: $watchCollection should call listener with old value (78057a94, #2621, #5661, #5688, #6736)
  • angular.bootstrap: allow angular to load only once (748a6c8d, #5863, #5587)
  • jqLite: inheritedData() now traverses Shadow DOM boundaries via the host property of DocumentFragment (8a96f317, #6637)
  • ngCookie: convert non-string values to string (36528310, #6151, #6220)
  • ngTouch: update workaround for Webkit quirk (bc42950b, #6302)
  • orderBy: support string predicates containing non-ident characters (37bc5ef4, #6143, #6144)
  • select: avoid checking option element's selected property in render (f40f54c6, #2448, #5994)

Features

  • $compile: add support for $observer deregistration (299b220f, #5609)
  • ngMock.$httpBackend: added support for function as URL matcher (d6cfcace, #4580)

Breaking Changes

  • $compile: due to 299b220f, calling attr.$observe no longer returns the observer function, but a deregistration function instead. To migrate the code follow the example below:

Before:

directive('directiveName', function() {
  return {
    link: function(scope, elm, attr) {
      var observer = attr.$observe('someAttr', function(value) {
        console.log(value);
      });
    }
  };
});

After:

directive('directiveName', function() {
  return {
    link: function(scope, elm, attr) {
      var observer = function(value) {
        console.log(value);
      };

      attr.$observe('someAttr', observer);
    }
  };
});
  • $httpBackend: due to 6680b7b9, the JSONP behavior for erroneous and empty responses changed: Previously, a JSONP response was regarded as erroneous if it was empty. Now Angular is listening to the correct events to detect errors, i.e. even empty responses can be successful.

v1.2.15 beer-underestimating (2014-03-21)

Bug Fixes

  • $$RAFProvider: check for webkitCancelRequestAnimationFrame (e84da228, #6526)
  • $$rAF: always fallback to a $timeout incase native rAF isn't supported (ee8e4a94, #6654)
  • $compile: support templates with thead and tfoot root elements (ca0ac649, #6289)
  • $http:
  • $rootScope:
  • angular.bootstrap: only allow angular to load once (0d60f8d3, #5863, #5587)
  • jqLite: traverse host property for DocumentFragment in inheritedData() (98d825e1, #6637)
  • ngAnimate: setting classNameFilter disables animation inside ng-if (a41a2a1d, #6539)
  • ngCookie: convert non-string values to string (93d1c95c, #6151, #6220)
  • ngTouch: update workaround for desktop Webkit quirk (01a34f51, #6302)
  • orderBy: support string predicates containing non-ident characters (10d3e1e4, #6143, #6144)
  • select: avoid checking option element selected properties in render (dc149de9, #2448, #5994, #6769)

1.3.0-beta.2 silent-ventriloquism (2014-03-14)

Bug Fixes

  • $$rAF: always fallback to a $timeout in case native rAF isn't supported (7b5e0199, #6654)
  • $http: don't convert 0 status codes to 404 for non-file protocols (56e73ea3, #6074, #6155)
  • ngAnimate: setting classNameFilter disables animation inside ng-if (129e2e02, #6539)

Features

  • whitelist blob urls for sanitization of data-bound image urls (47ab8df4, #4623)

1.3.0-beta.1 retractable-eyebrow (2014-03-07)

Bug Fixes

Features

  • input: support types date, time, datetime-local, month, week (46bd6dc8, #5864)

Breaking Changes

  • build: due to eaa1d00b, As communicated before, IE8 is no longer supported.
  • input: types date, time, datetime-local, month, week now always require a Date object as model (46bd6dc8, #5864)

For more info: http://blog.angularjs.org/2013/12/angularjs-13-new-release-approaches.html

1.2.14 feisty-cryokinesis (2014-03-01)

Bug Fixes

  • $animate:
    • delegate down to addClass/removeClass if setClass is not found (18c41af0, #6463)
    • ensure all comment nodes are removed during a leave animation (f4f1f43d, #6403)
    • only block keyframes if a stagger is set to occur (e71e7b6c, #4225)
    • ensure that animatable directives cancel expired leave animations (e9881991, #5886)
    • ensure all animated elements are taken care of during the closing timeout (99720fb5, #6395)
    • fix for TypeError Cannot call method 'querySelectorAll' in cancelChildAnimations (c914cd99, #6205)
  • $http:
  • $parse: reduce false-positives in isElement tests (5fe1f39f, #4805, #5675)
  • input: use ValidityState to determine validity (c2d447e3, #4293, #2144, #4857, #5120, #4945, #5500, #5944)
  • isElement: reduce false-positives in isElement tests (75515852)
  • jqLite:
  • numberFilter: convert all non-finite/non-numbers/non-numeric strings to the empty string (cceb455f, #6188, #6261)
  • $parse: support trailing commas in object & array literals (6b049c74)
  • ngHref: bind ng-href to xlink:href for SVGAElement (2bce71e9, #5904)

Features

  • $animate: animate dirty, pristine, valid, invalid for form/fields (33443966, #5378)

Performance Improvements

  • $animate: use rAF instead of timeouts to issue animation callbacks (4c4537e6)
  • $cacheFactory: skip LRU bookkeeping for caches with unbound capacity (a4078fca, #6193, #6226)

1.2.13 romantic-transclusion (2014-02-14)

Bug Fixes

Features

  • filterFilter: support deeply nested predicate objects (b4eed8ad, #6215)

Breaking Changes

  • $animate:

    • due to 4f84f6b3, ngClass and {{ class }} will now call the setClass animation callback instead of addClass / removeClass when both a addClass/removeClass operation is being executed on the element during the animation.

      Please include the setClass animation callback as well as addClass and removeClass within your JS animations to work with ngClass and {{ class }} directives.

    • due to cf5e463a, Both the $animate:before and $animate:after DOM events must be now registered prior to the $animate operation taking place. The $animate:close event can be registered anytime afterwards.

      DOM callbacks used to fired for each and every animation operation that occurs within the $animate service provided in the ngAnimate module. This may end up slowing down an application if 100s of elements are being inserted into the page. Therefore after this change callbacks are only fired if registered on the element being animated.

  • input:

    • due to a9fcb0d0, input[type=file] will no longer support ngModel. Due to browser support being spotty among target browsers, file inputs cannot be cleanly supported, and even features which technically do work (such as ng-change) work in an inconsistent way depending on the attributes of the form control.

      As a workaround, one can manually listen for change events on file inputs and handle them manually.

1.2.12 cauliflower-eradication (2014-02-07)

Bug Fixes

  • $compile: retain CSS classes added in cloneAttachFn on asynchronous directives (5ed721b9, #5439, #5617)
  • $http:
  • $locale: minor grammar amends for the locale locale_lt (95be253f, #6164)
  • $q: make $q.reject support finally and catch (074b0675, #6048, #6076)
  • docs: clarify doc for "args" in $broadcast and $emit (caed2dfe, #6047)
  • filterFilter: don't interpret dots in predicate object fields as paths (339a1658, #6005, #6009)
  • http: make jshint happy (6609e3da)
  • jqLite: trim HTML string in jqLite constructor (36d37c0e, #6053)
  • mocks:
    • rename mock.animate to ngAnimateMock and ensure it contains all test helper code for ngAnimate (4224cd51, #5822, #5917)
    • remove usage of $animate.flushNext in favor of queuing (906fdad0)
    • always call functions injected with inject with this set to the current spec (3bf43903, #6102)
    • refactor currentSpec to work w/ Jasmine 2 (95f0bf9b, #5662)
  • ngMock: return false from mock $interval.cancel() when no argument is supplied (dd24c783, #6103)
  • ngResource:

Breaking Changes

The animation mock module has been renamed from mock.animate to ngAnimateMock. In addition to the rename, animations will not block within test code even when ngAnimateMock is used. However, all function calls to $animate will be recorded into $animate.queue and are available within test code to assert animation calls. In addition, $animate.triggerReflow() is now only available when ngAnimateMock is used.

1.2.11 cryptocurrency-hyperdeflation (2014-02-03)

Bug Fixes

  • $compile: retain CSS classes added in cloneAttachFn on asynchronous directives (5ed721b9, #5439, #5617)
  • $http: update httpBackend to use ActiveXObject on IE8 if necessary (ef210e5e, #5677, #5679)
  • $q: make $q.reject support finally and catch (074b0675, #6048, #6076)
  • filterFilter: don't interpret dots in predicate object fields as paths (339a1658, #6005, #6009)
  • mocks: refactor currentSpec to work w/ Jasmine 2 (95f0bf9b, #5662)
  • ngResource: don't append number to '$' in url param value when encoding URI (ce1f1f97, #6003, #6004)

1.2.10 augmented-serendipity (2014-01-24)

Bug Fixes

  • $parse: do not use locals to resolve object properties (f09b6aa5, #5838, #5862)
  • a: don't call preventDefault on click when a SVGAElement has an xlink:href attribute (e0209169, #5896, #5897)
  • input: use Chromium's email validation regexp (79e519fe, #5899, #5924)
  • ngRoute: pipe preceding route param no longer masks ? or * operator (fd6bac7d, #5920)

Features

1.2.9 enchanted-articulacy (2014-01-15)

Bug Fixes

  • $animate:
    • ensure the final closing timeout respects staggering animations (ed53100a)
    • prevent race conditions for class-based animations when animating on the same CSS class (4aa9df7a, #5588)
    • correctly detect and handle CSS transition changes during class addition and removal (7d5d62da)
    • avoid accidentally matching substrings when resolving the presence of className tokens (524650a4)
  • $http: ensure default headers PUT and POST are different objects (e1cfb195, #5742, #5747, #5764)
  • $rootScope: prevent infinite $digest by checking if asyncQueue is empty when decrementing ttl (2cd09c9f, #2622)

Features

  • $animate:
    • provide support for DOM callbacks (dde1b294)
    • use requestAnimationFrame instead of a timeout to issue a reflow (4ae3184c, #4278, #4225)

Breaking Changes

  • $http: due to e1cfb195, it is now necessary to separately specify default HTTP headers for PUT, POST and PATCH requests, as these no longer share a single object.

    To migrate your code, follow the example below:

    Before:

    // Will apply to POST, PUT and PATCH methods
    $httpProvider.defaults.headers.post = {
      "X-MY-CSRF-HEADER": "..."
    };
    

    After:

    // POST, PUT and PATCH default headers must be specified seperately,
    // as they do not share data.
    $httpProvider.defaults.headers.post =
      $httpProvider.defaults.headers.put =
      $httpProviders.defaults.headers.patch = {
        "X-MY-CSRF-HEADER": "..."
      };
    

1.2.8 interdimensional-cartography (2014-01-10)

Bug Fixes

  • $http:
  • $location: return '/' for root path in hashbang mode (63cd873f, #5650, #5712)
  • $parse: fix CSP nested property evaluation, and issue that prevented its tests from failing (3b1a4fe0, #5591, #5592)
  • closure: add Closure externs for angular.$q.Promise.finally (caeb7402, #4757)
  • ngMock window.inject: Remove Error 'stack' property changes (7e916455)

Features

1.2.7 emoji-clairvoyance (2014-01-03)

Bug Fixes

  • $animate:
    • ensue class-based animations are always skipped before structural post-digest tasks are run (bc492c0f, #5582)
    • remove trailing s from computed transition duration styles (50bf0296)
  • $http: (3d38fff8)
    • use ActiveX XHR when making PATCH requests on IE8 (6c17d02b, #2518, #5043)
    • fix 'type mismatch' error on IE8 after each request (fd9a03e1)
    • Ignore multiple calls to onreadystatechange with readyState=4 (4f572366, #5426)
  • $injector: remove the INSTANTIATING flag properly when done (186a5912, #4361, #5577)
  • $location:
  • $resource: prevent URL template from collapsing into an empty string (131e4014, #5455, #5493)
  • $sanitize: consider the size attribute as a valid/allowed attribute (056c8493, #5522)
  • Scope: don't let watch deregistration mess up the dirty-checking digest loop (884ef0db, #5525)
  • input:
    • use apply on the change event only when one isn't already in progress (a80049fd, #5293)
    • prevent double $digest when using jQuery trigger. (1147f219, #5293)
  • ngRepeat: allow for more flexible coding style in ngRepeat expression (c9705b75, #5537, #5598)
  • ngRoute: instantiate controller when template is empty (498365f2, #5550)
  • ngShow/ngHide, ngIf: functions with zero args should be truthy (01c5be46, #5414)

Performance Improvements

  • Scope: limit propagation of $broadcast to scopes that have listeners for the event (80e7a455, #5341, #5371)

1.2.6 taco-salsafication (2013-12-19)

Bug Fixes

  • $animate: use a scheduled timeout in favor of a fallback property to close transitions (54637a33, #5255, #5241, #5405)
  • $compile: remove invalid IE exceptional case for href (c7a1d1ab, #5479)
  • $location: parse xlink:href for SVGAElements (bc3ff2ce, #5472, #5198, #5199, #4098, #1420)
  • $log: should work in IE8 (4f5758e6, #5400)
  • $parse: return undefined if an intermediate property's value is null (26d43cac, #5480)
  • closure: add type definition for Scope#$watchCollection (8f329ffb, #5475)
  • forEach: allow looping over result of querySelectorAll in IE8 (274a6734)
  • input: do not hold input for composition on Android (3dc18037, #5308)
  • jqLite: support unbind self within handler (2f91cfd0)
  • ngRepeat: allow multiline expressions (cbb3ce2c, #5000)
  • select: invalidate when multiple, required, and model is [] (5c97731a, #5337)

Features

  • jqLite: provide support for element.one() (937caab6)
  • ngAnimate: provide configuration support to match specific className values to trigger animations (cef084ad, #5357, #5283)

Performance Improvements

  • compile: add class 'ng-scope' before cloning and other micro-optimizations (f3a796e5, #5471)
  • $parse: use a faster path when the number of path parts is low (f4462319)
  • use faster check for $$ prefix (06c5cfc7)

1.2.5 singularity-expansion (2013-12-13)

Bug Fixes

  • $compile: allow literals in isolate scope references (43072e38, #5296)
  • angular-mocks: use copy of mock data in $httpBackend (f69dc162)
  • closure: add missing FormController extern definitions (1d5e18b0, #5303)
  • ngInclude: add template to DOM before linking other directives (30a8b7d0, #5247)
  • ngView: add template to DOM before linking other directives (f8944efe)

Performance Improvements

  • $injector: remove invoke optimization that doesn't work (05e4fd34, #5388)
  • $resource: use shallow copy instead of angular.copy (fcd2a813, #5300)
  • a: do not link when href or name exists in template (f3de5b6e, #5362)
  • jqLite: implement and use the empty method in place of html(‘’) (3410f65e, #4457)

Breaking Changes

  • angular-mocks: due to f69dc162, some tests that rely on identity comparison rather than equality comparison in checking mock http responses will be broken, since now each mock response is a copy of the original response. This is usually fixable by changing a .toBe() comparison to toEqual() inside of tests.

1.2.4 wormhole-blaster (2013-12-06)

Bug Fixes

  • $animate:
  • $compile:
    • update cloned elements if the template arrives after the cloning (b0972a2e)
    • ensure the isolated local watch lastValue is always in sync (2d0f6ccb, #5182)
  • $rootScope:
    • ensure that when the $destroy event is broadcast on $rootScope that it does something (d802ed1b, #5169)
    • ensure the phase is cleared within a digest if an exception is raised by a watcher (d3c486dd)
  • $sanitize: don't rely on YARR regex engine executing immediately in order to prevent object mutation (81b81856, #5193, #5192)
  • closure: closure compiler shouldn't rename .defaults.transformRequest (f01087f8)
  • input: ensure ngModelWatch() triggers second digest pass when appropriate (b6d54393, #5258, #5282)
  • isElement: return boolean value rather than truthy value. (2dbb6f9a, #4519, #4534)
  • jqLite: ignore incompatible nodes on find() (1169b544, #4120)
  • ngInit: evaluate ngInit before ngInclude (0e50810c, #5167, #5208)
  • ngSanitize: prefer textContent to innerText to avoid layout trashing (bf1972dc)

Performance Improvements

  • $parse: micro-optimization for ensureSafeObject function (689dfb16, #5246)
  • Scope: short-circuit after dirty-checking last dirty watcher (d070450c, #5272, #5287)

1.2.3 unicorn-zapper (2013-11-27)

Bug Fixes

  • $animate:
  • ensure blocked keyframe animations are unblocked before the DOM operation (2efe8230, #5106)
  • ensure animations are disabled during bootstrap to prevent unwanted structural animations (eed23332, #5130)
  • $sanitize: use the same whitelist mechanism as $compile does (33352348, #3748)
  • input: react to form auto completion, through the change event, on modern browsers (a090400f, #1460)
  • $attrs: add $attrs.$attr to externs so that it isn't renamed on js minification (bcca8054)

Features

No new features in this release

Breaking Changes

There are no breaking changes in this release (promise!)

1.2.2 consciousness-inertia (2013-11-22)

Bug Fixes

  • $animate:
    • ensure keyframe animations are blocked around the reflow (6760d7a3, #5018)
    • ensure transition animations are unblocked before the dom operation occurs (062fbed8, #5014, #4265)
    • ensure addClass/removeClass animations do not snap during reflow (76e4db6f, #4892)
    • ensure the DOM operation isn't run twice (7067a8fb, #4949)
  • $compile:
    • secure form[action] & iframesrcdoc
    • ensure CSS classes are added and removed only when necessary (0cd7e8f2)
  • $httpBackend: only IE8 and below can't use script.onload for JSONP (a3172a28, #4523, #4527, #4922)
  • $parse: allow for new lines in expr when promise unwrapping is on (40647b17, #4718)
  • $resource: Always return a resource instance when calling class methods on resources. (f6ecf9a3, #4545, #5061)
  • httpBackend: should not read response data when request is aborted (6f1050df, #4913, #4940)
  • loader: expose $$minErr to modules such asngResource (9e89a31b, #5050)
  • ngAnimate:
    • correctly retain and restore existing styles during and after animation (c42d0a04, #4869)
    • use a fallback CSS property that doesn't break existing styles (1d50663b, #4902, #5030)
  • ngClass: ensure that ngClass only adds/removes the changed classes (6b8bbe4d, #4960, #4944)
  • ngController: fix issue with ngInclude on the same element (6288cf5c, #4431)
  • ngInclude:
    • Don't throw when the ngInclude element contains content with directives. (0a7cbb33)
    • allow ngInclude to load scripts when jQuery is included (c47abd0d, #3756)
  • ngMock: fixes httpBackend expectation with body object (4d16472b, #4956)
  • ngView: Don't throw when the ngView element contains content with directives. (e6521e74, #5069)
  • tests: Correct tests for IE11 (57924234, #5046)
  • input: hold listener during text composition (a4e6d962, #4684)

1.2.1 underscore-empathy (2013-11-14)

Bug Fixes

  • $compile:
    • accessing controllers of transcluded directives from children (90f87072, #4935)
    • correctly handle interpolated style in replace templates (e1254b26, #4882)
  • $resource: don't use $parse for @dotted.member (9577702e)
  • bootstrap: make IE8 happy (a61b65d0)
  • loader: don't rely on internal APIs (8425e9fe, #4437, #4874)
  • minErr: remove references to internal APIs (94764ee0)
  • ngIf: don't create multiple elements when changing from a truthy value to another thruthy value (4612705e, #4852)
  • urlUtils:

Features

1.2.0 timely-delivery (2013-11-08)

Features

  • animations:
    • ensure CSS transitions can work with inherited CSS class definitions (9d69a0a7)
    • provide support for staggering animations with CSS (74848307)
  • $parse: secure expressions by hiding "private" properties (3d6a89e8)
  • docs:
    • provide index pages for each angular module (a7e12b79)
    • add forward slash shortcut key for search bar (74912802)
  • jqLite: expose isolateScope() getter similar to scope() (27e9340b)
  • misc: add externs file for Closure Compiler (9d0a6977)

Bug Fixes

  • $animate:
    • don't force animations to be enabled (98adc9e0)
    • only apply the fallback property if any transition animations are detected (94700807)
    • avoid hanging animations if the active CSS transition class is missing (b89584db, #4732, #4490)
    • ensure staggering animations understand multiple delay values (41a2d5b3)
    • ensure the active class is not applied if cancelled during reflow (e53ff431, #4699)
    • use direct DOM comparison when checking for $rootElement (d434eabe, #4679)
    • ensure former nodes are fully cleaned up when a follow-up structural animation takes place (7f0767ac, #4435)
    • ensure enable/disable animations work when the document node is used (6818542c, #4669)
    • skip unnecessary addClass/removeClass animations (76b628bc, #4401, #2332)
    • ensure animations work properly when the $rootElement is being animated (2623de14, #4397, #4231)
    • only cancel class-based animations if the follow-up class contains CSS transition/keyframe animation code (f5289fe8, #4463, #3784)
  • $compile:
    • don't leak isolate scope state when replaced directive is used multiple times (b5af198f)
    • correct isolate scope distribution to controllers (3fe4491a)
    • replaced element has isolate scope (97c7a4e3)
    • only pass isolate scope to children that belong to the isolate directive (d0efd5ee)
    • make isolate scope truly isolate (909cabd3, #1924, #2500)
    • don't instantiate controllers twice for element transclude directives (18ae985c, #4654)
    • attribute bindings should not break due to terminal directives (79223eae, #4525, #4528, #4649)
    • instantiate controllers when re-entering compilation (faf5b980, #4434, #4616)
  • $injector: allow a constructor function to return a function (c22adbf1)
  • $parse: check function call context to be safe (6d324c76, #4417)
  • angular-mocks: add inline dependency annotation (6d23591c, #4448)
  • animateSpec: run digest to enable animations before tests (aea76f0d)
  • bootstrap-prettify: share $animate and $$postDigestQueue with demo apps (1df3da36)
  • csp:
  • docModuleComponents: implement anchor scroll when content added (eb51b024, #4703)
  • input: keep track of min/max attars on-the-fly (4b653aea)
  • ngAnimate: fix cancelChildAnimations throwing exception (b9557b0a, #4548)
  • ngClassSpec: clear animation enable fn from postDigestQueue (ffa9d0a6)
  • ngEventDirectives: parse expression only once during compile phase. (9a828738)
  • ngIf:
    • destroy child scope when destroying DOM (9483373c)
    • ngIf removes elements dynamically added to it (e19067c9)
  • ngInclude: only run anchorScroll after animation is done (d378f550, #4723)
  • ngMock: throw more descriptive errors for $animate.flushNext() (6fb19157)
  • ngModel: deregister from the form on scope not DOM destruction (8f989d65, #4226, #4779)
  • ngScenario: correctly disable animations for end 2 end tests (9d004585)
  • ngView:
    • only run anchorScroll after animation is done (da344daa)
    • ensure the new view element is placed after the old view element (3f568b22, #4362)
  • ngdocs:
    • create mock Doc objects correctly (d4493fda)
    • shortDescription() should not error if no description (4c8fa353)
    • remove the side search bar (6c20ec19)

Breaking Changes

  • $compile:

    • due to d0efd5ee, Child elements that are defined either in the application template or in some other directives template do not get the isolate scope. In theory, nobody should rely on this behavior, as it is very rare - in most cases the isolate directive has a template.

    • due to 909cabd3, Directives without isolate scope do not get the isolate scope from an isolate directive on the same element. If your code depends on this behavior (non-isolate directive needs to access state from within the isolate scope), change the isolate directive to use scope locals to pass these explicitly.

    Before

    <input ng-model="$parent.value" ng-isolate>
    
    .directive('ngIsolate', function() {
      return {
        scope: {},
        template: '{{value}}'
      };
    });
    

    After

    <input ng-model="value" ng-isolate>
    
    .directive('ngIsolate', function() {
      return {
        scope: {value: '=ngModel'},
        template: '{{value}}
      };
    });
    

    Closes #1924 and #2500

    Previously, the interpolation priority was -100 in 1.2.0-rc.2, and 100 before 1.2.0-rc.2. Before this change the binding was setup in the post-linking phase.

    Now the attribute interpolation (binding) executes as a directive with priority 100 and the binding is set up in the pre-linking phase.

    Closes #4525, #4528, and #4649

  • $parse: due to 3d6a89e8,

    This commit introduces the notion of "private" properties (properties whose names begin and/or end with an underscore) on the scope chain. These properties will not be available to Angular expressions (i.e. {{ }} interpolation in templates and strings passed to $parse) They are freely available to JavaScript code (as before).

    Motivation

    Angular expressions execute in a limited context. They do not have direct access to the global scope, window, document or the Function constructor. However, they have direct access to names/properties on the scope chain. It has been a long standing best practice to keep sensitive APIs outside of the scope chain (in a closure or your controller.) That's easier said that done for two reasons:

    1. JavaScript does not have a notion of private properties so if you need someone on the scope chain for JavaScript use, you also expose it to Angular expressions
    2. the new "controller as" syntax that's now in increased usage exposes the entire controller on the scope chain greatly increasing the exposed surface.

    Though Angular expressions are written and controlled by the developer, they:

    1. Typically deal with user input
    2. Don't get the kind of test coverage that JavaScript code would

    This commit provides a way, via a naming convention, to allow publishing/restricting properties from controllers/scopes to Angular expressions enabling one to only expose those properties that are actually needed by the expressions.

  • csp: due to 08f376f2, triggering ngCsp directive via ng:csp attribute is not supported any more. Please use data-ng-csp instead.

  • jqLite: due to 27e9340b, jqLite.scope() (commonly used through angular.element(node).scope()) does not return the isolate scope on the element that triggered directive with isolate scope. Use jqLite.isolateScope() instead.

1.2.0-rc.3 ferocious-twitch (2013-10-14)

Features

  • $interval: add a service wrapping setInterval (2b5ce84f)
  • $sce: simpler patterns for $sceDelegateProviders white/blacklists (93ce5923, #4006)
  • $filter: allow map of filters to be registered (4033cf28, #4036, #4091)
  • $compile: support tel: links in a[href] (e7730297)

  • Directives:

    • ngRepeat: support repeating over ngInclude and other directives that replace repeated nodes (9efa46ae, #3104)
    • event directives: add ngCopy, ngCut, and ngPaste (147c6929, #4172)
  • Misc:

    • jQuery 1.10.x support (e0c134b8, #3764)
    • minErr: linkify error messages on minErr docs pages (6aaae062)
    • tutorial: add step 12 on animations to the phonecat tutorial (ad525645)

Bug Fixes

  • $compile:
    • abort compilation when duplicate element transclusion (63c5334c, #3893, #4217, #3307)
    • make order directives w/ same priority deterministic (4357da85)
    • fix (reverse) directive postLink fn execution order (31f190d4, #3558)
    • don't terminate compilation for regular transclusion directives (fe214501)
    • ng-attr to support dash separated attribute names (8e6e3eba)
    • allow interpolations for non-event handlers attrs (8e1276c0)
    • link parents before traversing (742271ff, #3792, #3923, #3935, #3927)
    • collect ranges on multiple directives on one element (6a8edc1d, #4002)
  • $parse:
    • deprecate promise unwrapping and make it an opt-in (5dc35b52, #4158, #4270)
    • disallow access to window and dom in expressions (be0b4856)
  • $httpBackend:
  • $location:
  • $log: prevent logging undefined for $log in IE (4ff1a650, #1705)
  • Scope:

    • $evalAsync executes on the right scope (10cc1a42)
    • make stopPropagation only stop its own event (47f7bd70, #4204)
  • Filters:

    • date: allow negative millisecond value strings (025c9219)
  • Directives:

    • correct priority of structural directives (ngRepeat, ngSwitchWhen, ngIf, ngInclude, ngView) (b7af76b4)
    • input: false is no longer an empty value by default (b56b21a8, #3490)
    • ngBindHtml: watch string value instead of wrapper (e2068ad4, #3932)
    • ngOptions: ignore object properties which start with $ (aa3c54c7)
    • ngRepeat: correctly track elements even when the collection is initially undefined (31c56f54, #4145, #3964)
    • ngTransclude: detect ngTranslude usage without a transclusion directive (5a1a6b86, #3759)
  • jqLite:

    • ignore class methods on comment elements (64fd2c42)
    • use get/setAttribute so that jqLite works on SVG nodes (c785267e, #3858)
  • Misc:

    • isArrayLike: correctly handle string primitives (5b8c7884, #3356)
    • protect calls to hasOwnProperty in public API (7a586e5c, #3331)
  • ngRoute:

    • ngView: IE8 regression due to expando on non-element nodes (255e8c13, #3971)
    • $route: parametrized routes do not match against locations that would not valorize each parameters. (0ff86c32)
  • ngResource:

  • ngSanitize:

    • sanitize DOCTYPE declarations correctly (e66c23fe, #3931)
    • sanitizer should not accept <!--> as a valid comment (21e9e8cf)
  • ngTouch:

  • ngAnimate:

    • ensure that a timeStamp is created if not provided by the browser event (cd216c4c, #3053)
    • perform internal caching on getComputedStyle to boost the performance of CSS3 transitions/animations (b1e604e3, #4011, #4124)
    • ensure structural animations skip all child animations even if no animation is present during compile (cc584607, #3215)
    • cancel any ongoing child animations during move and leave animations (3f31a7c7)
    • ensure elapsedTime always considers delay values (079dd939)
    • ensure transition-property is not changed when only keyframe animations are in use (2df3c9f5, #3933)
    • avoid completing the animation asynchronously unless CSS transtiions/animations are present (2a63dfa6, #4023, #3940)
    • ensure that delays are always considered before an animation closes (0a63adce, #4028)
    • check elapsedTime on current event (d50ed6bf)
    • support addClass/removeClass animations on SVG nodes (c785267e, #3858)
  • ngScenario:

Breaking Changes

  • $compile: due to 31f190d4, the order of postLink fn is now mirror opposite of the order in which corresponding preLinking and compile functions execute.

    Previously the compile/link fns executed in this order controlled via priority:

    • CompilePriorityHigh, CompilePriorityMedium, CompilePriorityLow
    • compile child nodes
    • PreLinkPriorityHigh, PreLinkPriorityMedium, PreLinkPriorityLow
    • link child nodes
    • PostLinkPriorityHigh, PostLinkPriorityMedium, PostLinkPriorityLow

    This was changed to:

    • CompilePriorityHigh, CompilePriorityMedium, CompilePriorityLow
    • compile child nodes
    • PreLinkPriorityHigh, PreLinkPriorityMedium, PreLinkPriorityLow
    • link child nodes
    • PostLinkPriorityLow, PostLinkPriorityMedium , PostLinkPriorityHigh

    Very few directives in practice rely on order of postLinking function (unlike on the order of compile functions), so in the rare case of this change affecting an existing directive, it might be necessary to convert it to a preLinking function or give it negative priority (look at the diff of this commit to see how an internal attribute interpolation directive was adjusted).

  • $parse:

    • due to 5dc35b52, $parse and templates in general will no longer automatically unwrap promises. This feature has been deprecated and if absolutely needed, it can be reenabled during transitional period via $parseProvider.unwrapPromises(true) api.
    • due to b6a37d11, feature added in rc.2 that unwraps return values from functions if the values are promises (if promise unwrapping is enabled - see previous point), was reverted due to breaking a popular usage pattern.
  • directives: due to b7af76b4, the priority of ngRepeat, ngSwitchWhen, ngIf, ngInclude and ngView has changed. This could affect directives that explicitly specify their priority.

    In order to make ngRepeat, ngSwitchWhen, ngIf, ngInclude and ngView work together in all common scenarios their directives are being adjusted to achieve the following precedence:

    Directive        | Old Priority | New Priority
    =============================================
    ngRepeat         | 1000         | 1000
    ---------------------------------------------
    ngSwitchWhen     | 500          | 800
    ---------------------------------------------
    ngIf             | 1000         | 600
    ---------------------------------------------
    ngInclude/ngView | 1000         | 400
    
  • form/ngForm due to 7a586e5c, Inputs with name equal to "hasOwnProperty" are not allowed inside form or ngForm directives.

    Before, inputs whose name was "hasOwnProperty" were quietly ignored and not added to the scope. Now a badname exception is thrown.

    Using "hasOwnProperty" for an input name would be very unusual and bad practice.

    Either do not include such an input in a form or ngForm directive or change the name of the input.

  • ngScenario: due to 28f56a38, browserTrigger now uses an eventData object instead of direct parameters for mouse events. To migrate, place the keys,x and y parameters inside of an object and place that as the third parameter for the browserTrigger function.

1.2.0-rc.2 barehand-atomsplitting (2013-09-04)

Features

  • Scope: asynchronously auto-flush $evalAsync queue when outside of $digest cycle (6b91aa0a, #3539, #2438)
  • minErr: log minerr doc url in development builds (37123cd2, #3566)
  • ngMock:
    • allow passing an object literal as shorthand to module (f737c97d)
    • add support for creating dynamic style sheets within test code (fb3a7db0)

Bug Fixes

  • $http: allow empty responses to be cached (8e48c4ff, #3809)
  • $injector: don't parse fns with no args (44b6b72e)
  • $parse: handle promises returned from parsed function calls (3a658220, #3503)
  • $q:
    • reject should catch & forward exceptions thrown in error callbacks (5d9f4205)
    • fix forwarding resolution when callbacks aren't functions (7d188d63, #3535)
  • $location: fix history problems on Boxee box (eefcdad0)
  • $timeout: clean deferreds immediately after callback exec/cancel (920a3804)

  • Directives:

    • ngTransclude:
      • clear the transclusion point before transcluding (eed299a3)
      • make the transclusion available to parent post-link function (bf79bd41)
    • ngView: ensure ngClass works with together with ngView's transclusion behavior (40c0220c)
  • Filters:

  • Misc:

  • i18n: remove obsolete locale files (6382e21f)

  • ngAnimate:

    • ensure that ngClass is always compiled before enter, leave and move animations are applied (36ad40b1, #3727, #3603)
    • cut down on extra $timeout calls (4382df03)
    • skip ngAnimate animations if the provided element already has transitions applied to it (7c605ddf, #3587)
    • only apply a timeout when transitions or keyframe animations are used (ee2f3d21, #3613)
    • ensure older versions of webkit work for animations (b1a43cd0)
  • ngMocks: $logProvider should not use internal APIs (baaa73ee, #3612)

Breaking Changes

  • i18n: due to 6382e21f, some uncommon region-specific local files were removed.

1.0.8 bubble-burst (2013-08-22)

Contains only these fixes cherry-picked from v1.2.0rc1.

Bug Fixes

  • $compile:
  • $http: ensure case-insensitive header overriding (25d9f5a8)
  • $location:
    • default to / for the url base if no base[href] (cbe31d8d, #2762)
    • prevent infinite digest error due to IE bug (97abb124, #2802)
    • don't crash on invalid query parameters (b9dcb35e)
  • $parse: move global getter out of parse.js (099138fb)
  • $q: call reject() even if $exceptionHandler rethrows (d59027c4)
  • $timeout: clean deferreds immediately after callback exec/cancel (ac69392c)
  • $sanitize: match URI schemes case-insensitively (fcd761b9, #3210)
  • Scope: watches can be safely unregistered inside watch handlers (a4ec2979, #2915)

  • ngMock

    • $timeout should forward delay argument (a5fb372e)
  • jqLite:

    • return array from multi select in val() (01cd3495)
    • forgive unregistration of a non-registered handler (ac5b9055)
    • prepend array in correct order (63414b96)
    • correctly monkey-patch core jQuery methods (815053e4)
  • Directives:

    • form: pick the right attribute name for ngForm (dc1e55ce, #2997)
    • input: fix the email regex to accept TLDs up to 6 characters long (ad76e77f)
    • ngCloak: hide element even when CSS 'display' is set (06b0930b)
    • ngSubmit: expose $event to ngSubmit callback (b0d5f062)
    • ngValue: made ngValue to write value attribute to element (3b898664)
  • Filters:

    • number: always convert scientific notation to decimal (408e8682)
    • orderBy: remove redundant if statement (ec1cece2)
  • i18n: Do not transform arrays into objects (751c77f8)

  • jqLite:

    • return array from multi select in val() (01cd3495)
    • forgive unregistration of a non-registered handler (ac5b9055)
    • prepend array in correct order (63414b96)
    • correctly monkey-patch core jQuery methods (815053e4)
  • Misc:

    • angular.copy: change angular.copy to correctly clone RegExp (5cca077e, #3473, #3474)
    • angular.equals:
      • add support for regular expressions (a357649d, #2685)
      • {} and [] should not be considered equivalent (da1f7c76)
    • angular.toJson: skip JSON.stringify for undefined (332a3c79)

1.2.0rc1 spooky-giraffe (2013-08-13)

Full Commit Log

Features

  • ngAnimate: complete rewrite of animations (81923f1e)

  • $sce: new $sce service for Strict Contextual Escaping and lots of other security enhancements (bea9422e)

  • minErr: add error message minification and better error messages (c8fcf3b3, 09fa0656, b8ea7f6a)

  • $compile:

    • support animation hooks bindings to class attributes (f2dfa891)
    • support multi-element directive (e46100f7)
    • support "Controller as" instance syntax for directives (b3777f27)
  • $http: accept function as headers value (a7150f12)

  • $q:

    • add .catch() as shorthand for defining promise error handlers (a207665d, #2048, #3476)
    • added support for promise notification (2a5c3555)
  • $resource:

    • support an unescaped URL port in the url template (b94ca12f, #2778)
    • expose promise as $promise instead of only $then (05772e15)
  • $route: express style route matching (support for optional params and new wildcard syntax) (04cebcc1)

  • jqLite: switch bind/unbind to more recent jQuery on/off (f1b94b4b)

  • Misc:

  • Directives:

    • add ngFocus and ngBlur directives (2bb27d49, #1277)

    • ngRepeat: add $even and $odd props to iterator (52b8211f)

    • ngForm: supports namespaces in form names (8ea802a1)

    • ngBindHtml: combine ng-bind-html and ng-bind-html-unsafe (dae69473)

    • ngPluralize: add alternative mapping using attributes (a170fc1a, #2454)

  • ngMobile/ngTouch:

    • emit swipeleft and swiperight events (ab189142)
    • refactor swipe logic from ngSwipe directive to $swipe service. (f4c6b2c7)
  • ngMock:

    • $timeout.flushNext can expect specific timeout delays (462ed033)
    • support delay limit for $timeout.flush (b7fdabc4)
    • support a matching function for data param (08daa779)
  • scenario: expose jQuery for usage outside of angular scenario (3fdbe81a)

  • ngDocs:

    • provide support for user to jump between different versions of the angularjs doc (46dfb92a)
    • add links to source for API (52d6a599)
    • support popover, foldouts and foldover annotations (ef229688)
    • provide documentation for the new ngRepeat repeater syntax (b3650457)
    • provide support for inline variable hinting (21c70729)

Bug Fixes

  • $compile:

    • correct controller instantiation for async directives (c173ca41, #3493, #3482, #3537, #3540)
    • always instantiate controllers before pre-link fns run (5c560117, #3493, #3482, #3514)
    • always instantiate controllers in parent->child order (45f9f623, #2738)
    • don't check attr.specified on non-ie7 (f9ea69f6, #3231, #2160)
    • allow data: image URIs in img[src] bindings (3e39ac7e)
    • empty normalized href url should pass sanitation check (fc8c9baa, #2219)
    • prevent infinite loop w/ replace+transclude directives (69f42b76, #2155)
    • reject multi-expression interpolations for src attribute (38deedd6)
    • disallow interpolations for DOM event handlers (39841f2e)
    • sanitize values bound to img[src] (1adf29af)
    • support multi-element group over text nodes (b28f9694)
    • correct component transclusion on compilation root. (15e1a29c)
  • $http:

    • allow interceptors to completely override headers (514dc0eb, #2770)
    • treat headers as case-insensitive when overriding defaults (53359d54)
  • $location:

    • don't initialize url hash in hashbang mode unnecessarily (d4d34aba)
    • prevent infinite digest error due to IE bug (dca23173, #2802)
    • in html5 mode, default to / for the url base if no base[href] (aef09800, #2762)
    • fix parameter handling on search() (705c9d95)
  • $parse:

    • unwrap promise when setting a field (61906d35, #1827)
    • disallow access to Function constructor (5349b200)
  • $q: call reject() even if $exceptionHandler rethrows (664526d6)

  • $resource: check whether response matches action.isArray (a644ca7b, #2255)

  • $sanitize: match URI schemes case-insensitively (7fef06fe, #3210)

  • Scope:

    • ensure that isolate scopes use the main evalAsync queue (3967f5f7)
    • watches can now be safely unregistered inside watch handlers (8bd6619b, #2915)
  • jqLite:

    • properly detect unsupported calls for on()/off() (3824e400, 4f5dfbc3, #3501)
    • return array from multi select in val() (306a6134)
    • forgive unregistration of a non-registered handler (ab59cc6c)
    • support space-separated events in off (bdd4e982, #3256)
    • prepend array in correct order (fd87eb0c)
    • allow override of jqLite.triggerHandler event object (0cac8729)
    • added optional name arg in removeData (e1a050e6)
    • correctly monkey-patch core jQuery methods (da5f537c)
  • i18n: Do not transform arrays into objects (b3d7a038)

  • ngMobile/ngTouch:

  • ngMock:

    • keep withCredentials on passThrough (3079a6f4)
    • keep mock.$log the api in sync with $log (f274c0a6, #2343)
  • ngScenario: select().option(val) should prefer exact value match (22a9b1ac, #2856)

  • Directives:

    • ngRepeat:

    • ngShowHide: change the .ng-hide CSS class to use an !important flag (246c1439)

    • ngSubmit: expose $event to ngSubmit callback (3371fc25)

    • ngValue: made ngValue to write value attribute to element (09a1e7af)

    • ngView: ensure ngView is terminal and uses its own manual transclusion system (87405e25)

    • ngCloak: hide ngCloak-ed element even when CSS 'display' is set (3ffddad1)

    • input[email]: fix the email regex to accept TLDs up to 6 characters long (af731354)

    • form: pick the right attribute name for ngForm (0fcd1e3b, #2997)

    • select: don't support binding to select[multiple] (d87fa004, #3230)

  • Filters:

    • numberFilter: always convert scientific notation to decimal (a13c01a8)
  • Misc:

    • detect transition/animation on older Android browsers (ef5bc6c7)
    • handle duplicate params in parseKeyValue/toKeyValue (80739409)
    • don't crash on invalid query parameters (8264d080)
    • change angular.copy to correctly clone RegExp (f80730f4, #3473, #3474)
    • angular.equals now supports for regular expressions (724819e3, #2685)
    • angular.equals should not match keys defined in the prototype chain (7829c50f)
    • angular.equals should not consider {} and [] to be equivalent (1dcafd18)
    • angular.bootstrap should throw an error when bootstrapping a bootstrapped element (3ee744cc)
    • angular.toJson should skip JSON.stringify for undefined (5a294c86)
    • change css wrapping in grunt to prepend styles to the top of the head tag (fbad068a)

Breaking Changes

  • ngAnimate: due to 81923f1e, too many things changed, we'll write up a separate doc with migration instructions and will publish it at http://yearofmoo.com. Please check out the ngAnimate module docs and $animate api docs in the meantime.

  • $compile:

    • due to 1adf29af and 3e39ac7e, img[src] URLs are now being sanitized and a whitelist configured via $compileProvider can be used to configure what safe urls look like.

      By default all common protocol prefixes are whitelisted including data: URIs with mime types image/*. Therefore this change is expected to have no impact on apps that don't contain malicious image links.

    • due to 38deedd6, binding more than a single expression to *[src] or *[ng-src] with the exception of <a> and <img> elements is not supported.

      Concatenating expressions makes it hard to understand whether some combination of concatenated values are unsafe to use and potentially subject to XSS vulnerabilities. To simplify the task of auditing for XSS issues, we now require that a single expression be used for *[src/ng-src] bindings such as bindings for iframe[src], object[src], etc. (but not img[src/ng-src] since that value is sanitized).

    This change ensures that the possible pool of values that are used for data-binding is easier to trace down.

    To migrate your code, follow the example below:

    Before:
        JS:
            scope.baseUrl = 'page';
            scope.a = 1;
            scope.b = 2;
        HTML:
            <!-- Are a and b properly escaped here? Is baseUrl
                 controlled by user? -->
            <iframe src="{{baseUrl}}?a={{a}&b={{b}}">
    
    After:
        JS:
            var baseUrl = "page";
            scope.getIframeSrc = function() {
              // There are obviously better ways to do this.  The
              // key point is that one will think about this and do
              // it the right way.
              var qs = ["a", "b"].map(function(value, name) {
                  return encodeURIComponent(name) + "=" +
                         encodeURIComponent(value);
                }).join("&");
              // baseUrl isn't on scope so it isn't bound to a user
              // controlled value.
              return baseUrl + "?" + qs;
            }
        HTML: <iframe src="{{getIframeSrc()}}">
    
    • due to 39841f2e, Interpolations inside DOM event handlers are disallowed.

      DOM event handlers execute arbitrary Javascript code. Using an interpolation for such handlers means that the interpolated value is a JS string that is evaluated. Storing or generating such strings is error prone and leads to XSS vulnerabilities. On the other hand, ngClick and other Angular specific event handlers evaluate Angular expressions in non-window (Scope) context which makes them much safer.

      To migrate the code follow the example below:

      Before:

      JS:   scope.foo = 'alert(1)';
      HTML: <div onclick="{{foo}}">
      

      After:

      JS:   scope.foo = function() { alert(1); }
      HTML: <div ng-click="foo()">
      
    • due to e46100f7, existing directives with name ending with "-start" or "-end" will stop working.

      This change was necessary to enable multi-element directives. The best fix is to rename existing directives, so that they don't end with these suffixes.

  • $q: due to f078762d, the always method is now exposed as finally.

    The reason for this change is to align $q with the Q promise library, despite the fact that this makes it a bit more difficult to use with non-ES5 browsers, like IE8.

    finally also goes well together with catch api that was added to $q recently and is part of the DOM promises standard.

    To migrate the code follow the example below:

    Before:

      $http.get('/foo').always(doSomething);
    

    After:

      $http.get('/foo').finally(doSomething);
    

    or for IE8 compatible code:

      $http.get('/foo')['finally'](doSomething);
    
  • $resource:

    • due to 05772e15, resource instance does not have a $then function anymore. Use the $promise.then instead.

      Before:

      Resource.query().$then(callback);
      

      After:

      Resource.query().$promise.then(callback);
      
    • due to 05772e15, instance methods return the promise rather than the instance itself.

      Before:

      resource.$save().chaining = true;
      

      After:

      resource.$save();
      resource.chaining = true;
      
    • due to 05772e15, on success, the resource promise is resolved with the resource instance rather than http response object.

      Use interceptor api to access the http response object.

      Before:

      Resource.query().$then(function(response) {...});
      

      After:

      var Resource = $resource('/url', {}, {
        get: {
          method: 'get',
          interceptor: {
            response: function(response) {
              // expose response
              return response;
            }
          }
        }
      });
      
  • $route:

    • due to 04cebcc1, the syntax for named wildcard parameters in routes has changed from *wildcard to :wildcard*

      To migrate the code, follow the example below. Here, *highlight becomes :highlight*:

      Before:

      $routeProvider.when('/Book1/:book/Chapter/:chapter/*highlight/edit',
                {controller: noop, templateUrl: 'Chapter.html'});
      

      After:

      $routeProvider.when('/Book1/:book/Chapter/:chapter/:highlight*/edit',
              {controller: noop, templateUrl: 'Chapter.html'});
      
    • due to 5599b55b, applications that use $route will now need to load an angular-route.js file and define a dependency on the ngRoute module.

      Before:

      ...
      <script src="angular.js"></script>
      ...
      var myApp = angular.module('myApp', ['someOtherModule']);
      ...
      

      After:

      ...
      <script src="angular.js"></script>
      <script src="angular-route.js"></script>
      ...
      var myApp = angular.module('myApp', ['ngRoute', 'someOtherModule']);
      ...
      
  • $location: due to 80739409, $location.search now supports multiple keys with the same value provided that the values are stored in an array in $location.search.

    Before this change:

    • parseKeyValue only took the last key overwriting all the previous keys;
    • toKeyValue joined the keys together in a comma delimited string.

      This was deemed buggy behavior. If your server relied on this behavior then either the server should be fixed, or a simple serialization of the array should be done on the client before passing it to $location.

  • ngBindHtml, sce: due to dae69473,

    ngHtmlBindUnsafe has been removed and replaced by ngHtmlBind (which has been moved from ngSanitize module to the core ng module). ngBindHtml provides ngHtmlBindUnsafe like behavior (evaluate an expression and innerHTML the result into the DOM) when bound to the result of $sce.trustAsHtml(string). When bound to a plain string, the string is sanitized via $sanitize before being innerHTML'd. If the $sanitize service isn't available (ngSanitize module is not loaded) and the bound expression evaluates to a value that is not trusted an exception is thrown.

  • ngForm: due to 8ea802a1,

    If you have form names that will evaluate as an expression:

    <form name="ctrl.form">
    

    And if you are accessing the form from your controller:

    Before:

    function($scope) {
      $scope['ctrl.form'] // form controller instance
    }
    

    After:

    function($scope) {
      $scope.ctrl.form // form controller instance
    }
    

    This makes it possible to access a form from a controller using the new "controller as" syntax. Supporting the previous behavior offers no benefit.

  • ngView: due to 7d69d52a, previously ngView only updated its content, after this change ngView will recreate itself every time a new content is included. This ensures that a single rootElement for all the included contents always exists, which makes definition of css styles for animations much easier.

  • ngInclude: due to aa2133ad, previously ngInclude only updated its content, after this change ngInclude will recreate itself every time a new content is included. This ensures that a single rootElement for all the included contents always exists, which makes definition of css styles for animations much easier.

  • select: due to d87fa004, binding to select[multiple] directly or via ngMultiple (ng-multiple) directive is not supported. This feature never worked with two-way data-binding, so it's not expected that anybody actually depends on it.

  • ngMobile: due to 94ec84e7, since all the code in the ngMobile module is touch related, we are renaming the module to ngTouch.

    To migrate, please replace all references to "ngMobile" with "ngTouch" and "angular-mobile.js" to "angular-touch.js".

1.1.5 triangle-squarification (2013-05-22)

Note: 1.1.x releases are considered unstable. They pass all tests but we reserve the right to change new features/apis in between minor releases. Check them out and please give us feedback.

Note: This release also contains all bug fixes available in 1.0.7.

Features

  • $animator:
    • provide support for custom animation events (c53d4c94)
    • allow to globally disable and enable animations (5476cb6e)
  • $http:

    • add support for aborting via timeout promises (9f4f5937, #1159)
    • add a default content type header for PATCH requests (f9b897de)
    • add timeout support for JSONP requests (cda7b711)
  • $parse: add support for ternary operators to parser (6798fec4)

  • $q: add $q.always() method (6605adf6)

  • $controller: support "Controller as" syntax (cd38cbf9, 400f9360)

  • $injector: add has method for querying (80341cb9, #2556)

  • Directives:

    • ngAnimate:
      • add support for CSS3 Animations with working delays and multiple durations (14757874)
      • cancel previous incomplete animations when new animations take place (4acc28a3)
    • ngSrcset: add new ngSrcset directive (d551d729, #2601)
    • ngIf: add directive to remove and recreate DOM elements (2f96fbd1)
    • select: match options by expression other than object identity (c32a859b)
    • ngInclude: $includeContentRequested event (af0eaa30)
  • Mobile:

    • ngClick: Add a CSS class while the element is held down via a tap (52a55ec6)
    • ngSwipe: Add ngSwipeRight/Left directives to ngMobile (5e0f876c)
  • docs:

    • Add FullText search to replace Google search in docs (3a49b7ee)
    • external links to github, plunkr and jsfiddle available for code examples (c8197b44)
    • add variable type hinting with colors (404c9a65)
    • support for HTML table generation from docs code (b3a62b2e)
  • scenario runner: adds mousedown and mouseup event triggers to scenario (629fb373)

    Bug Fixes

    • $animator: remove dependency on window.setTimeout (021bdf39)

    • $controller: allow dots in a controller name (de2cdb06)

    • $location:

      • prevent navigation when event isDefaultPrevented (2c69a673)
      • compare against actual instead of current URL (a348e90a)
      • prevent navigation if already on the URL (4bd7bedf)
      • fix URL interception in hash-bang mode (58ef3230, #1051)
      • correctly rewrite Html5 urls (77ff1085)
    • $resource:

      • null default param results in TypeError (cefbcd47)
      • collapse empty suffix parameters correctly (53061363)
    • $rootScope: ensure $watchCollection correctly handles arrayLike objects (6452707d)

    • date filter: correctly format dates with more than 3 sub-second digits (4f2e3606)

    • jqLite: pass a dummy event into triggerHandler (0401a7f5)

    • Directives:

      • ngAnimate:
        • eval ng-animate expression on each animation (fd21c750)
        • prevent animation on initial page load (570463a4)
        • skip animation on first render (1351ba26)
      • ngPattern: allow modifiers on inline ng-pattern (12b6deb1, #1437)
      • ngRepeat:
      • ngView: accidentally compiling leaving content (9956baed)
    • scenario runner: correct bootstrap issue on IE (ab755a25)

Breaking Changes

  • $animator/ngAnimate: due to 11f712bc, css transition classes changed from foo-setup/foo-start to foo/foo-active

    The CSS transition classes have changed suffixes. To migrate rename

      .foo-setup {...} to .foo {...}
      .foo-start {...} to .foo-active {...}
    

    or for type: enter, leave, move, show, hide

      .foo-type-setup {...} to .foo-type {...}
      .foo-type-start {...} to .foo-type-active {...}
    
  • $resource: due to 53061363, a / followed by a ., in the last segment of the URL template is now collapsed into a single . delimiter.

    For example: users/.json will become users.json. If your server relied upon this sequence then it will no longer work. In this case you can now escape the /. sequence with /\.

1.0.7 monochromatic-rainbow (2013-05-22)

Bug Fixes

  • $browser: should use first value for a cookie. (3952d35a, #2635)

  • $cookieStore: $cookieStore.get now parses blank string as blank string (cf4729fa)

  • $location: back-button should fire $locationChangeStart (dc9a5806, #2109)

  • $parse: Fix context access and double function call (7812ae75, #2496)

  • dateFilter: correctly format ISODates on Android<=2.1 (f046f6f7, #2277)

  • jqLite: correct implementation of mouseenter/mouseleave event (06f2b2a8, #2131)

  • angular.copy/angular.extend: do not copy $$hashKey in copy/extend functions. (6d0b325f, #1875)

  • i18n: escape all chars above \u007f in locale files (695c54c1, #2417)

  • Directives:

    • ngPluralize: handle the empty string as a valid override (67a4a25b, #2575)
    • select: ensure empty option is not lost in IE9 (4622af3f, #2150)
    • ngModel: use paste/cut events in IE to support context menu (363e4cbf, #1462)
    • ngClass: should remove classes when object is the same but property has changed (0ac969a5)
  • PhoneCat Tutorial: renamed Testacular to Karma (angular-phonecat)

1.1.4 quantum-manipulation (2013-04-03)

Note: 1.1.x releases are considered unstable. They pass all tests but we reserve the right to change new features/apis in between minor releases. Check them out and please give us feedback.

Note: This release also contains all bug fixes available in 1.0.6.

Features

  • $compile:

  • $q: $q.all() now accepts hash (e27bb6eb)

  • $resource: ability to override url in resource actions (60f1f099)

  • $route: add caseInsensitiveMatch option for url matching (5e18a15f)

  • http:

    • support request/response promise chaining (4ae46814)
    • set custom default cache in $http.defaults.cache (99f3b70b)
  • JQLite: ready() now supports document.readyState=='complete' (753fc9e5)

  • Scenario: autodisable animations when running e2e tests (fec4ef38)

  • Scope: add $watchCollection method for observing collections (5eb96855)

  • angular.bootstrap: support deferred bootstrap (mainly useful for tools like test runners and Batarang) (603fe0d1)

  • ngMobile: add ngMobile module with mobile-specific ngClick (707c65d5)

  • Directives:

    • ngKeypress: add ngKeypress directive for handling keypress event (f20646bc)
    • ngSwitch: Preserve the order of the elements not in the ng-switch (e88d6179, #1074)
    • ngAnimate: add support for animation (0b6f1ce5)
    • ngRepeat: add support for custom tracking of items (61f2767c)

Breaking Changes

  • $route: due to 6f71e809, in $routeChangeStart event, nextRoute.$route property is gone. Use the nextRoute object itself instead of nextRoute.$route.

  • ngRepeat: due to 61f2767c, it is now considered an error to have two identical items (identified by the new "track by" expression) in a collection that is fed into the repeater. This behavior was previously tolerated.

  • ngSwitch: due to e88d6179, elements not in the ng-switch were rendered after the ng-switch elements. Now they are rendered in-place.

    Templates with ngSwitch directives and nested non-ngSwitchWhen elements should be updated to preserve render order.

    For example: The following was previously rendered with <li>1</li> after <li>2</li>:

      <ul ng-switch="select">
        <li>1</li>
        <li ng-switch-when="option">2</li>
      </ul>
    

    To keep the old behavior, use:

      <ul ng-switch="select">
        <li ng-switch-when="1">2</li>
        <li>1</li>
      </ul>
    

1.0.6 universal-irreversibility (2013-04-04)

Bug Fixes

  • $compile:
    • compile replace directives in external template (398691be, #1859)
    • whitelist file:// in url sanitization (7b236b29)
    • handle elements with no childNodes property (bec614fd)
  • $http: don't encode URL query substring "null" to "+" (86d191ed)
  • $httpBackend: prevent DOM err due to dereferencing .responseText (509ec745, #1922)
  • $location:
    • parse FirefoxOS packaged app urls (3a81dd8b, #2112)
    • correctly rewrite html5 url to hashbang url (9befe370)
  • $route: make nextRoute.$route private (6f71e809, #1907)
  • mocks: prevent NPE when module definition outside of it. (5c735eb4)
  • dateFilter: correct timezone date filter for 1/2 hour offsets (1c1cd4fd)

1.1.3 radioactive-gargle (2013-02-20)

Note: 1.1.x releases are considered unstable. They pass all tests but we reserve the right to change new features/apis in between minor releases. Check them out and please give us feedback.

Note: This release also contains all bug fixes available in 1.0.5.

Bug Fixes

  • $compile:
    • initialize interpolated attributes before directive linking (bb8448c0)
    • interpolate @ locals before the link function runs (2ed53087)
  • $http:
    • do not encode special characters @$:, in params (288b69a3)
  • $resource:
    • params should expand array values properly (2a212344)

Features

  • $http: allow overriding the XSRF header and cookie name (8155c3a2)
  • $parse: added constant and literal properties (1ed63858)
  • $resource: expose promise based api via $then and $resolved (dba6bc73)
  • $routeProvider: add support to catch-all parameters in routes (7eafbb98)
  • Scope:
    • expose transcluded and isolate scope info for batarang (649b8922)
    • only evaluate constant $watch expressions once (1d7a95df)
  • angular.noConflict: added api to restore previous angular namespace reference (12ba6cec)
  • Directives:
    • ngSwitch: support multiple matches on ngSwitchWhen and ngSwitchDefault (0af17204, #1074)
  • Filters:
    • date: add [.,]sss formatter for milliseconds (df744f3a)
    • filter: add comparison function to filter (ace54ff0)

Breaking Changes

  • $http: due to 288b69a3, $http now follows RFC3986 and does not encode special characters like $@,: in params. If your application needs to encode these characters, encode them manually, before sending the request.
  • $resource: due to 2a212344, if the server relied on the buggy behavior of serializing arrays as http query arguments then either the backend should be fixed or a simple serialization of the array should be done on the client before calling the resource service.

1.0.5 flatulent-propulsion (2013-02-20)

Bug Fixes

  • $compile:
    • sanitize values bound to a[href] (9532234b)
    • rename $compileNote to compileNode (92ca7efa, #1941)
    • should not leak memory when there are top level empty text nodes (791804bd)
    • allow startingTag method to handle text / comment nodes (755beb2b)
  • $cookies: set cookies on Safari&IE when base[href] is undefined (70909245, #1190)
  • $http:
    • patch for Firefox bug w/ CORS and response headers (e19b04c9, #1468)
  • $resource:
    • update RegExp to allow urlParams with out leading slash (b7e1fb05)
  • Directives:
    • a: workaround IE bug affecting mailto urls (37e8b122, #1949)
    • ngClass: keep track of old ngClass value manually (5f5d4fea, #1637)
    • ngSwitch: make ngSwitch compatible with controller backwards-compatibility module (9b7c1d0f)
  • Filters:
    • date: invert timezone sign and always display sign (b001c8ec, #1261)
    • number: fix formatting when "0" passed as fractionSize (f5835963)
  • scenario runner: include error messages in XML output (d46fe3c2)
  • Misc:
    • don't use instanceof to detect arrays (3c2aee01, #1966)
    • angular.forEach should correctly iterate over objects with length prop (ec54712f, #1840)

1.1.2 tofu-animation (2013-01-22)

Note: 1.1.x releases are considered unstable. They pass all tests but we reserve the right to change new features/apis in between minor releases. Check them out and please give us feedback.

Note: This release also contains all bug fixes available in 1.0.4.

Features

  • $compile: support modifying the DOM structure in postlink fn (cdf6fb19)
  • $log: add $log.debug() (9e991ddb, #1592)
  • $parse: allow strict equality in angular expressions (a179a9a9, #908)
  • $resource:
    • allow dynamic default parameters (cc42c99b)
    • support all $http.config actions (af89daf4)
  • $route: allow using functions as template params in 'when' (faf02f0c)
  • $timeout-mock: add verifyNoPendingTasks method (f0c6ebc0, #1245)
  • directive:
  • limitTo filter: limitTo filter accepts strings (9e96d983, #653)
  • scenario:
    • add mouseover method to the ngScenario dsl (2f437e89)
    • fail when an option to select does not exist (15183f3e)

Breaking Changes

  • date: due to cc821502, string input without timezone info is now parsed as local time/date

1.0.4 bewildering-hair (2013-01-22)

Bug Fixes

  • $compile:
  • $injector:
  • $resource:
    • HTTP method should be case-insensitive (8991680d, #1403)
    • correct leading slash removal in resource URLs (b2f46251)
  • $route:
    • support route params not separated with slashes. (c6392616)
    • correctly extract $routeParams from urls (30a9da5d)
  • Scope: ensure that a scope is destroyed only once (d6da505f, #1627)
  • angular.equals:
  • date filter: parse string input as local time unless TZ is specified (cc821502, #847)
  • jqLite:
    • children() should only return elements (febb4c1c)
    • make next() ignore non-element nodes (76a6047a)
  • scenario: don't trigger input events on IE9 (8b9e6c35)
  • Directives:
    • ngRepeat: correctly apply $last if repeating over object (7e746015, #1789)
    • ngSwitch: don't leak when destroyed while not attached (a26234f7, #1621)
    • select: support optgroup + select[multiple] combo (26adeb11, #1553)

Features

  • $compile: support modifying the DOM structure in postlink fn (cdf6fb19)