Commit 7ed580b2 authored by gergely-ujvari's avatar gergely-ujvari

Merge pull request #1058 from hypothesis/angular-upgrade-1

Angular upgrade v1.2.13
parents 076e87d6 bfbb58c3
...@@ -9,29 +9,26 @@ annotation = ['$filter', 'annotator', ($filter, annotator) -> ...@@ -9,29 +9,26 @@ annotation = ['$filter', 'annotator', ($filter, annotator) ->
scope.save(e) scope.save(e)
# Watch for changes # Watch for changes
scope.$watch 'model.$modelValue.id', (id) -> scope.$watch 'model', (model) ->
scope.thread = annotator.threading.idTable[id] scope.thread = annotator.threading.idTable[model.id]
scope.auth = {} scope.auth = {}
scope.auth.delete = scope.auth.delete =
if scope.model.$modelValue? and annotator.plugins?.Permissions? if model? and annotator.plugins?.Permissions?
annotator.plugins.Permissions.authorize 'delete', scope.model.$modelValue annotator.plugins.Permissions.authorize 'delete', model
else else
true true
scope.auth.update = scope.auth.update =
if scope.model.$modelValue? and annotator.plugins?.Permissions? if scope.model? and annotator.plugins?.Permissions?
annotator.plugins.Permissions.authorize 'update', scope.model.$modelValue annotator.plugins.Permissions.authorize 'update', model
else else
true true
# Publish the controller
scope.model = controller
controller: 'AnnotationController' controller: 'AnnotationController'
priority: 100 # Must run before ngModel
require: '?ngModel' require: '?ngModel'
restrict: 'C' restrict: 'C'
scope: scope:
model: '=ngModel'
mode: '@' mode: '@'
replies: '@' replies: '@'
templateUrl: 'annotation.html' templateUrl: 'annotation.html'
......
...@@ -540,20 +540,20 @@ class Annotation ...@@ -540,20 +540,20 @@ class Annotation
$scope.cancel = ($event) -> $scope.cancel = ($event) ->
$event?.stopPropagation() $event?.stopPropagation()
$scope.editing = false $scope.editing = false
drafts.remove $scope.model.$modelValue drafts.remove $scope.model
annotator.enableAnnotating drafts.isEmpty() annotator.enableAnnotating drafts.isEmpty()
switch $scope.action switch $scope.action
when 'create' when 'create'
annotator.deleteAnnotation $scope.model.$modelValue annotator.deleteAnnotation $scope.model
else else
$scope.model.$modelValue.text = $scope.origText $scope.model.text = $scope.origText
$scope.model.$modelValue.tags = $scope.origTags $scope.model.tags = $scope.origTags
$scope.action = 'create' $scope.action = 'create'
$scope.save = ($event) -> $scope.save = ($event) ->
$event?.stopPropagation() $event?.stopPropagation()
annotation = $scope.model.$modelValue annotation = $scope.model
# Forbid saving comments without a body (text or tags) # Forbid saving comments without a body (text or tags)
if annotator.isComment(annotation) and not annotation.text and if annotator.isComment(annotation) and not annotation.text and
...@@ -610,14 +610,13 @@ class Annotation ...@@ -610,14 +610,13 @@ class Annotation
$event?.stopPropagation() $event?.stopPropagation()
$scope.action = 'edit' $scope.action = 'edit'
$scope.editing = true $scope.editing = true
$scope.origText = $scope.model.$modelValue.text $scope.origText = $scope.model.text
$scope.origTags = $scope.model.$modelValue.tags $scope.origTags = $scope.model.tags
drafts.add $scope.model.$modelValue, -> $scope.cancel() drafts.add $scope.model, -> $scope.cancel()
annotator.disableAnnotating() annotator.disableAnnotating()
$scope.delete = ($event) -> $scope.delete = ($event) ->
$event?.stopPropagation() $event?.stopPropagation()
annotation = $scope.model.$modelValue
replies = $scope.thread.children?.length or 0 replies = $scope.thread.children?.length or 0
# We can delete the annotation if it hasn't got any replies or it is # We can delete the annotation if it hasn't got any replies or it is
...@@ -632,27 +631,26 @@ class Annotation ...@@ -632,27 +631,26 @@ class Annotation
annotator.plugins.Permissions.authorize 'delete', reply annotator.plugins.Permissions.authorize 'delete', reply
annotator.deleteAnnotation reply annotator.deleteAnnotation reply
annotator.deleteAnnotation annotation annotator.deleteAnnotation $scope.model
else else
$scope.action = 'delete' $scope.action = 'delete'
$scope.editing = true $scope.editing = true
$scope.origText = $scope.model.$modelValue.text $scope.origText = $scope.model.text
$scope.origTags = $scope.model.$modelValue.tags $scope.origTags = $scope.model.tags
$scope.model.$modelValue.text = '' $scope.model.text = ''
$scope.model.$modelValue.tags = '' $scope.model.tags = ''
$scope.$watch 'editing', -> $scope.$emit 'toggleEditing' $scope.$watch 'editing', -> $scope.$emit 'toggleEditing'
$scope.$watch 'model.$modelValue.id', (id) -> $scope.$watch 'model.id', (id) ->
if id? if id?
annotation = $scope.model.$modelValue $scope.thread = $scope.model.thread
$scope.thread = annotation.thread
# Check if this is a brand new annotation # Check if this is a brand new annotation
if annotation? and drafts.contains annotation if drafts.contains $scope.model
$scope.editing = true $scope.editing = true
$scope.$watch 'model.$modelValue.target', (targets) -> $scope.$watch 'model.target', (targets) ->
return unless targets return unless targets
for target in targets for target in targets
if target.diffHTML? if target.diffHTML?
...@@ -674,19 +672,19 @@ class Annotation ...@@ -674,19 +672,19 @@ class Annotation
# is not a pre-determined route map. One possibility would be to # is not a pre-determined route map. One possibility would be to
# unify everything so that it's relative to the app URL. # unify everything so that it's relative to the app URL.
prefix = $scope.$parent.baseUrl.replace /\/\w+\/$/, '' prefix = $scope.$parent.baseUrl.replace /\/\w+\/$/, ''
$scope.shared_link = prefix + '/a/' + $scope.model.$modelValue.id $scope.shared_link = prefix + '/a/' + $scope.model.id
$scope.shared = false $scope.shared = false
return
$scope.$watchCollection 'model.$modelValue.thread.children', (newValue=[]) -> $scope.$watchCollection 'model.thread.children', (newValue=[]) ->
annotation = $scope.model.$modelValue return unless $scope.model
return unless annotation
replies = (r.message for r in newValue) replies = (r.message for r in newValue)
replies = replies.sort(annotator.sortAnnotations).reverse() replies = replies.sort(annotator.sortAnnotations).reverse()
annotation.reply_list = replies $scope.model.reply_list = replies
$scope.toggle = -> $scope.toggle = ->
$element.find('.share-dialog').slideToggle() $element.find('.share-dialog').slideToggle()
return
$scope.share = ($event) -> $scope.share = ($event) ->
$event.stopPropagation() $event.stopPropagation()
...@@ -696,10 +694,10 @@ class Annotation ...@@ -696,10 +694,10 @@ class Annotation
$scope.rebuildHighlightText = -> $scope.rebuildHighlightText = ->
if annotator.text_regexp? if annotator.text_regexp?
$scope.model.$modelValue.highlightText = $scope.model.$modelValue.text $scope.model.highlightText = $scope.model.text
for regexp in annotator.text_regexp for regexp in annotator.text_regexp
$scope.model.$modelValue.highlightText = $scope.model.highlightText =
$scope.model.$modelValue.highlightText.replace regexp, annotator.highlighter $scope.model.highlightText.replace regexp, annotator.highlighter
class Editor class Editor
...@@ -845,11 +843,11 @@ class Search ...@@ -845,11 +843,11 @@ class Search
$scope.openDetails = (annotation) -> $scope.openDetails = (annotation) ->
# Temporary workaround, until search result annotation card # Temporary workaround, until search result annotation card
# scopes get their 'annotation' fields, too. # scopes get their 'annotation' fields, too.
return unless annotation return unless annotation
for p in providers for p in providers
p.channel.notify p.channel.notify
method: 'scrollTo' method: 'scrollTo'
params: annotation.$$tag params: annotation.$$tag
refresh = => refresh = =>
$scope.search_filter = $routeParams.matched $scope.search_filter = $routeParams.matched
......
...@@ -6,19 +6,19 @@ authentication = -> ...@@ -6,19 +6,19 @@ authentication = ->
code: null code: null
link: (scope, elem, attr, ctrl) -> link: (scope, elem, attr, ctrl) ->
angular.extend scope, base angular.copy base, scope.model
controller: [ controller: [
'$scope', 'authentication', '$scope', 'authentication',
($scope, authentication) -> ($scope, authentication) ->
$scope.$on '$reset', => angular.extend $scope.model, base $scope.$on '$reset', => angular.copy base, $scope.model
$scope.submit = (form) -> $scope.submit = (form) ->
angular.extend authentication, $scope.model
return unless form.$valid return unless form.$valid
authentication["$#{form.$name}"] -> authentication["$#{form.$name}"] ->
$scope.$emit 'success', form.$name $scope.$emit 'success', form.$name
] ]
scope: restrict: 'ACE'
model: '=authentication'
markdown = ['$filter', '$timeout', ($filter, $timeout) -> markdown = ['$filter', '$timeout', ($filter, $timeout) ->
...@@ -84,11 +84,16 @@ privacy = -> ...@@ -84,11 +84,16 @@ privacy = ->
permissions permissions
scope.model = controller controller.$render = ->
scope.level = controller.$viewValue
scope.levels = levels scope.levels = levels
scope.setLevel = (level) ->
controller.$setViewValue level
controller.$render()
require: '?ngModel' require: '?ngModel'
restrict: 'E' restrict: 'E'
scope: true scope: {}
templateUrl: 'privacy.html' templateUrl: 'privacy.html'
...@@ -282,6 +287,7 @@ repeatAnim = -> ...@@ -282,6 +287,7 @@ repeatAnim = ->
itemElm itemElm
.css({ 'margin-left': itemElm.width() }) .css({ 'margin-left': itemElm.width() })
.animate({ 'margin-left': '0px' }, 1500) .animate({ 'margin-left': '0px' }, 1500)
return
# Directive to edit/display a tag list. # Directive to edit/display a tag list.
tags = ['$window', ($window) -> tags = ['$window', ($window) ->
...@@ -330,16 +336,13 @@ tags = ['$window', ($window) -> ...@@ -330,16 +336,13 @@ tags = ['$window', ($window) ->
] ]
notification = ['$filter', ($filter) -> notification = ['$filter', ($filter) ->
link: (scope, elem, attrs, controller) ->
return unless controller?
# Publish the controller
scope.model = controller
controller: 'NotificationController' controller: 'NotificationController'
priority: 100 # Must run before ngModel
require: '?ngModel' require: '?ngModel'
restrict: 'C' restrict: 'C'
scope: {} scope:
model: '=ngModel'
click: '&onClick'
close: '&onClose'
templateUrl: 'notification.html' templateUrl: 'notification.html'
] ]
......
...@@ -530,6 +530,7 @@ class Hypothesis extends Annotator ...@@ -530,6 +530,7 @@ class Hypothesis extends Annotator
discardDrafts: -> discardDrafts: ->
return @element.injector().get('drafts').discard() return @element.injector().get('drafts').discard()
class AuthenticationProvider class AuthenticationProvider
constructor: -> constructor: ->
@actions = @actions =
......
/** /**
* @license AngularJS v1.2.0-rc.2 * @license AngularJS v1.2.13
* (c) 2010-2012 Google, Inc. http://angularjs.org * (c) 2010-2014 Google, Inc. http://angularjs.org
* License: MIT * License: MIT
*/ */
(function(window, angular, undefined) {'use strict'; (function(window, angular, undefined) {'use strict';
var $resourceMinErr = angular.$$minErr('$resource'); var $resourceMinErr = angular.$$minErr('$resource');
// Helper functions and regex to lookup a dotted path on an object
// stopping at undefined/null. The path must be composed of ASCII
// identifiers (just like $parse)
var MEMBER_NAME_REGEX = /^(\.[a-zA-Z_$][0-9a-zA-Z_$]*)+$/;
function isValidDottedPath(path) {
return (path != null && path !== '' && path !== 'hasOwnProperty' &&
MEMBER_NAME_REGEX.test('.' + path));
}
function lookupDottedPath(obj, path) {
if (!isValidDottedPath(path)) {
throw $resourceMinErr('badmember', 'Dotted member path "@{0}" is invalid.', path);
}
var keys = path.split('.');
for (var i = 0, ii = keys.length; i < ii && obj !== undefined; i++) {
var key = keys[i];
obj = (obj !== null) ? obj[key] : undefined;
}
return obj;
}
/**
* Create a shallow copy of an object and clear other fields from the destination
*/
function shallowClearAndCopy(src, dst) {
dst = dst || {};
angular.forEach(dst, function(value, key){
delete dst[key];
});
for (var key in src) {
if (src.hasOwnProperty(key) && !(key.charAt(0) === '$' && key.charAt(1) === '$')) {
dst[key] = src[key];
}
}
return dst;
}
/** /**
* @ngdoc overview * @ngdoc overview
* @name ngResource * @name ngResource
...@@ -14,12 +55,13 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -14,12 +55,13 @@ var $resourceMinErr = angular.$$minErr('$resource');
* *
* # ngResource * # ngResource
* *
* `ngResource` is the name of the optional Angular module that adds support for interacting with * The `ngResource` module provides interaction support with RESTful services
* [RESTful](http://en.wikipedia.org/wiki/Representational_State_Transfer) server-side data sources. * via the $resource service.
* `ngReource` provides the {@link ngResource.$resource `$resource`} serivce.
* *
* {@installModule resource} * {@installModule resource}
* *
* <div doc-module-components="ngResource"></div>
*
* See {@link ngResource.$resource `$resource`} for usage. * See {@link ngResource.$resource `$resource`} for usage.
*/ */
...@@ -63,7 +105,7 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -63,7 +105,7 @@ var $resourceMinErr = angular.$$minErr('$resource');
* *
* @param {Object.<Object>=} actions Hash with declaration of custom action that should extend the * @param {Object.<Object>=} actions Hash with declaration of custom action that should extend the
* default set of resource actions. The declaration should be created in the format of {@link * default set of resource actions. The declaration should be created in the format of {@link
* ng.$http#Parameters $http.config}: * ng.$http#usage_parameters $http.config}:
* *
* {action1: {method:?, params:?, isArray:?, headers:?, ...}, * {action1: {method:?, params:?, isArray:?, headers:?, ...},
* action2: {method:?, params:?, isArray:?, headers:?, ...}, * action2: {method:?, params:?, isArray:?, headers:?, ...},
...@@ -71,21 +113,23 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -71,21 +113,23 @@ var $resourceMinErr = angular.$$minErr('$resource');
* *
* Where: * Where:
* *
* - **`action`** – {string} – The name of action. This name becomes the name of the method on your * - **`action`** – {string} – The name of action. This name becomes the name of the method on
* resource object. * your resource object.
* - **`method`** – {string} – HTTP request method. Valid methods are: `GET`, `POST`, `PUT`, `DELETE`, * - **`method`** – {string} – HTTP request method. Valid methods are: `GET`, `POST`, `PUT`,
* and `JSONP`. * `DELETE`, and `JSONP`.
* - **`params`** – {Object=} – Optional set of pre-bound parameters for this action. If any of the * - **`params`** – {Object=} – Optional set of pre-bound parameters for this action. If any of
* parameter value is a function, it will be executed every time when a param value needs to be * the parameter value is a function, it will be executed every time when a param value needs to
* obtained for a request (unless the param was overridden). * be obtained for a request (unless the param was overridden).
* - **`url`** – {string} – action specific `url` override. The url templating is supported just like * - **`url`** – {string} – action specific `url` override. The url templating is supported just
* for the resource-level urls. * like for the resource-level urls.
* - **`isArray`** – {boolean=} – If true then the returned object for this action is an array, see * - **`isArray`** – {boolean=} – If true then the returned object for this action is an array,
* `returns` section. * see `returns` section.
* - **`transformRequest`** – `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` – * - **`transformRequest`** –
* `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` –
* transform function or an array of such functions. The transform function takes the http * transform function or an array of such functions. The transform function takes the http
* request body and headers and returns its transformed (typically serialized) version. * request body and headers and returns its transformed (typically serialized) version.
* - **`transformResponse`** – `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` – * - **`transformResponse`** –
* `{function(data, headersGetter)|Array.<function(data, headersGetter)>}` –
* transform function or an array of such functions. The transform function takes the http * transform function or an array of such functions. The transform function takes the http
* response body and headers and returns its transformed (typically deserialized) version. * response body and headers and returns its transformed (typically deserialized) version.
* - **`cache`** – `{boolean|Cache}` – If true, a default $http cache will be used to cache the * - **`cache`** – `{boolean|Cache}` – If true, a default $http cache will be used to cache the
...@@ -94,7 +138,7 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -94,7 +138,7 @@ var $resourceMinErr = angular.$$minErr('$resource');
* caching. * caching.
* - **`timeout`** – `{number|Promise}` – timeout in milliseconds, or {@link ng.$q promise} that * - **`timeout`** – `{number|Promise}` – timeout in milliseconds, or {@link ng.$q promise} that
* should abort the request when resolved. * should abort the request when resolved.
* - **`withCredentials`** - `{boolean}` - whether to to set the `withCredentials` flag on the * - **`withCredentials`** - `{boolean}` - whether to set the `withCredentials` flag on the
* XHR object. See {@link https://developer.mozilla.org/en/http_access_control#section_5 * XHR object. See {@link https://developer.mozilla.org/en/http_access_control#section_5
* requests with credentials} for more information. * requests with credentials} for more information.
* - **`responseType`** - `{string}` - see {@link * - **`responseType`** - `{string}` - see {@link
...@@ -131,7 +175,7 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -131,7 +175,7 @@ var $resourceMinErr = angular.$$minErr('$resource');
* usually the resource is assigned to a model which is then rendered by the view. Having an empty * usually the resource is assigned to a model which is then rendered by the view. Having an empty
* object results in no rendering, once the data arrives from the server then the object is * object results in no rendering, once the data arrives from the server then the object is
* populated with the data and the view automatically re-renders itself showing the new data. This * populated with the data and the view automatically re-renders itself showing the new data. This
* means that in most case one never has to write a callback function for the action methods. * means that in most cases one never has to write a callback function for the action methods.
* *
* The action methods on the class object or instance object can be invoked with the following * The action methods on the class object or instance object can be invoked with the following
* parameters: * parameters:
...@@ -153,14 +197,15 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -153,14 +197,15 @@ var $resourceMinErr = angular.$$minErr('$resource');
* *
* On success, the promise is resolved with the same resource instance or collection object, * On success, the promise is resolved with the same resource instance or collection object,
* updated with data from server. This makes it easy to use in * updated with data from server. This makes it easy to use in
* {@link ngRoute.$routeProvider resolve section of $routeProvider.when()} to defer view rendering * {@link ngRoute.$routeProvider resolve section of $routeProvider.when()} to defer view
* until the resource(s) are loaded. * rendering until the resource(s) are loaded.
* *
* On failure, the promise is resolved with the {@link ng.$http http response} object, * On failure, the promise is resolved with the {@link ng.$http http response} object, without
* without the `resource` property. * the `resource` property.
* *
* - `$resolved`: `true` after first server interaction is completed (either with success or rejection), * - `$resolved`: `true` after first server interaction is completed (either with success or
* `false` before that. Knowing if the Resource has been resolved is useful in data-binding. * rejection), `false` before that. Knowing if the Resource has been resolved is useful in
* data-binding.
* *
* @example * @example
* *
...@@ -197,14 +242,15 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -197,14 +242,15 @@ var $resourceMinErr = angular.$$minErr('$resource');
newCard.name = "Mike Smith"; newCard.name = "Mike Smith";
newCard.$save(); newCard.$save();
// POST: /user/123/card {number:'0123', name:'Mike Smith'} // POST: /user/123/card {number:'0123', name:'Mike Smith'}
// server returns: {id:789, number:'01234', name: 'Mike Smith'}; // server returns: {id:789, number:'0123', name: 'Mike Smith'};
expect(newCard.id).toEqual(789); expect(newCard.id).toEqual(789);
* </pre> * </pre>
* *
* The object returned from this function execution is a resource "class" which has "static" method * The object returned from this function execution is a resource "class" which has "static" method
* for each action in the definition. * for each action in the definition.
* *
* Calling these methods invoke `$http` on the `url` template with the given `method`, `params` and `headers`. * Calling these methods invoke `$http` on the `url` template with the given `method`, `params` and
* `headers`.
* When the data is returned from the server then the object is an instance of the resource type and * When the data is returned from the server then the object is an instance of the resource type and
* all of the non-GET methods are available with `$` prefix. This allows you to easily support CRUD * all of the non-GET methods are available with `$` prefix. This allows you to easily support CRUD
* operations (create, read, update, delete) on server-side data. * operations (create, read, update, delete) on server-side data.
...@@ -217,7 +263,7 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -217,7 +263,7 @@ var $resourceMinErr = angular.$$minErr('$resource');
}); });
</pre> </pre>
* *
* It's worth noting that the success callback for `get`, `query` and other method gets passed * It's worth noting that the success callback for `get`, `query` and other methods gets passed
* in the response that came from the server as well as $http header getter function, so one * in the response that came from the server as well as $http header getter function, so one
* could rewrite the above example and get access to http headers as: * could rewrite the above example and get access to http headers as:
* *
...@@ -232,56 +278,38 @@ var $resourceMinErr = angular.$$minErr('$resource'); ...@@ -232,56 +278,38 @@ var $resourceMinErr = angular.$$minErr('$resource');
}); });
</pre> </pre>
* # Buzz client * # Creating a custom 'PUT' request
* In this example we create a custom method on our resource to make a PUT request
Let's look at what a buzz client created with the `$resource` service looks like: * <pre>
<doc:example> * var app = angular.module('app', ['ngResource', 'ngRoute']);
<doc:source jsfiddle="false"> *
<script> * // Some APIs expect a PUT request in the format URL/object/ID
function BuzzController($resource) { * // Here we are creating an 'update' method
this.userId = 'googlebuzz'; * app.factory('Notes', ['$resource', function($resource) {
this.Activity = $resource( * return $resource('/notes/:id', null,
'https://www.googleapis.com/buzz/v1/activities/:userId/:visibility/:activityId/:comments', * {
{alt:'json', callback:'JSON_CALLBACK'}, * 'update': { method:'PUT' }
{get:{method:'JSONP', params:{visibility:'@self'}}, replies: {method:'JSONP', params:{visibility:'@self', comments:'@comments'}}} * });
); * }]);
} *
* // In our controller we get the ID from the URL using ngRoute and $routeParams
BuzzController.prototype = { * // We pass in $routeParams and our Notes factory along with $scope
fetch: function() { * app.controller('NotesCtrl', ['$scope', '$routeParams', 'Notes',
this.activities = this.Activity.get({userId:this.userId}); function($scope, $routeParams, Notes) {
}, * // First get a note object from the factory
expandReplies: function(activity) { * var note = Notes.get({ id:$routeParams.id });
activity.replies = this.Activity.replies({userId:this.userId, activityId:activity.id}); * $id = note.id;
} *
}; * // Now call update passing in the ID first then the object you are updating
BuzzController.$inject = ['$resource']; * Notes.update({ id:$id }, note);
</script> *
* // This will PUT /notes/ID with the note object in the request payload
<div ng-controller="BuzzController"> * }]);
<input ng-model="userId"/> * </pre>
<button ng-click="fetch()">fetch</button>
<hr/>
<div ng-repeat="item in activities.data.items">
<h1 style="font-size: 15px;">
<img src="{{item.actor.thumbnailUrl}}" style="max-height:30px;max-width:30px;"/>
<a href="{{item.actor.profileUrl}}">{{item.actor.name}}</a>
<a href ng-click="expandReplies(item)" style="float: right;">Expand replies: {{item.links.replies[0].count}}</a>
</h1>
{{item.object.content | html}}
<div ng-repeat="reply in item.replies.data.items" style="margin-left: 20px;">
<img src="{{reply.actor.thumbnailUrl}}" style="max-height:30px;max-width:30px;"/>
<a href="{{reply.actor.profileUrl}}">{{reply.actor.name}}</a>: {{reply.content | html}}
</div>
</div>
</div>
</doc:source>
<doc:scenario>
</doc:scenario>
</doc:example>
*/ */
angular.module('ngResource', ['ng']). angular.module('ngResource', ['ng']).
factory('$resource', ['$http', '$parse', '$q', function($http, $parse, $q) { factory('$resource', ['$http', '$q', function($http, $q) {
var DEFAULT_ACTIONS = { var DEFAULT_ACTIONS = {
'get': {method:'GET'}, 'get': {method:'GET'},
'save': {method:'POST'}, 'save': {method:'POST'},
...@@ -293,10 +321,7 @@ angular.module('ngResource', ['ng']). ...@@ -293,10 +321,7 @@ angular.module('ngResource', ['ng']).
forEach = angular.forEach, forEach = angular.forEach,
extend = angular.extend, extend = angular.extend,
copy = angular.copy, copy = angular.copy,
isFunction = angular.isFunction, isFunction = angular.isFunction;
getter = function(obj, path) {
return $parse(path)(obj);
};
/** /**
* We need our custom method because encodeURIComponent is too aggressive and doesn't follow * We need our custom method because encodeURIComponent is too aggressive and doesn't follow
...@@ -318,9 +343,9 @@ angular.module('ngResource', ['ng']). ...@@ -318,9 +343,9 @@ angular.module('ngResource', ['ng']).
/** /**
* This method is intended for encoding *key* or *value* parts of query component. We need a custom * This method is intended for encoding *key* or *value* parts of query component. We need a
* method because encodeURIComponent is too aggressive and encodes stuff that doesn't have to be * custom method because encodeURIComponent is too aggressive and encodes stuff that doesn't
* encoded per http://tools.ietf.org/html/rfc3986: * have to be encoded per http://tools.ietf.org/html/rfc3986:
* query = *( pchar / "/" / "?" ) * query = *( pchar / "/" / "?" )
* pchar = unreserved / pct-encoded / sub-delims / ":" / "@" * pchar = unreserved / pct-encoded / sub-delims / ":" / "@"
* unreserved = ALPHA / DIGIT / "-" / "." / "_" / "~" * unreserved = ALPHA / DIGIT / "-" / "." / "_" / "~"
...@@ -352,8 +377,12 @@ angular.module('ngResource', ['ng']). ...@@ -352,8 +377,12 @@ angular.module('ngResource', ['ng']).
var urlParams = self.urlParams = {}; var urlParams = self.urlParams = {};
forEach(url.split(/\W/), function(param){ forEach(url.split(/\W/), function(param){
if (!(new RegExp("^\\d+$").test(param)) && param && (new RegExp("(^|[^\\\\]):" + param + "(\\W|$)").test(url))) { if (param === 'hasOwnProperty') {
urlParams[param] = true; throw $resourceMinErr('badname', "hasOwnProperty is not a valid parameter name.");
}
if (!(new RegExp("^\\d+$").test(param)) && param &&
(new RegExp("(^|[^\\\\]):" + param + "(\\W|$)").test(url))) {
urlParams[param] = true;
} }
}); });
url = url.replace(/\\:/g, ':'); url = url.replace(/\\:/g, ':');
...@@ -363,7 +392,9 @@ angular.module('ngResource', ['ng']). ...@@ -363,7 +392,9 @@ angular.module('ngResource', ['ng']).
val = params.hasOwnProperty(urlParam) ? params[urlParam] : self.defaults[urlParam]; val = params.hasOwnProperty(urlParam) ? params[urlParam] : self.defaults[urlParam];
if (angular.isDefined(val) && val !== null) { if (angular.isDefined(val) && val !== null) {
encodedVal = encodeUriSegment(val); encodedVal = encodeUriSegment(val);
url = url.replace(new RegExp(":" + urlParam + "(\\W|$)", "g"), encodedVal + "$1"); url = url.replace(new RegExp(":" + urlParam + "(\\W|$)", "g"), function(match, p1) {
return encodedVal + p1;
});
} else { } else {
url = url.replace(new RegExp("(\/?):" + urlParam + "(\\W|$)", "g"), function(match, url = url.replace(new RegExp("(\/?):" + urlParam + "(\\W|$)", "g"), function(match,
leadingSlashes, tail) { leadingSlashes, tail) {
...@@ -377,7 +408,7 @@ angular.module('ngResource', ['ng']). ...@@ -377,7 +408,7 @@ angular.module('ngResource', ['ng']).
}); });
// strip trailing slashes and set the url // strip trailing slashes and set the url
url = url.replace(/\/+$/, ''); url = url.replace(/\/+$/, '') || '/';
// then replace collapse `/.` if found in the last URL path segment before the query // then replace collapse `/.` if found in the last URL path segment before the query
// E.g. `http://url.com/id./format?q=x` becomes `http://url.com/id.format?q=x` // E.g. `http://url.com/id./format?q=x` becomes `http://url.com/id.format?q=x`
url = url.replace(/\/\.(?=\w+($|\?))/, '.'); url = url.replace(/\/\.(?=\w+($|\?))/, '.');
...@@ -396,7 +427,7 @@ angular.module('ngResource', ['ng']). ...@@ -396,7 +427,7 @@ angular.module('ngResource', ['ng']).
}; };
function ResourceFactory(url, paramDefaults, actions) { function resourceFactory(url, paramDefaults, actions) {
var route = new Route(url); var route = new Route(url);
actions = extend({}, DEFAULT_ACTIONS, actions); actions = extend({}, DEFAULT_ACTIONS, actions);
...@@ -406,7 +437,8 @@ angular.module('ngResource', ['ng']). ...@@ -406,7 +437,8 @@ angular.module('ngResource', ['ng']).
actionParams = extend({}, paramDefaults, actionParams); actionParams = extend({}, paramDefaults, actionParams);
forEach(actionParams, function(value, key){ forEach(actionParams, function(value, key){
if (isFunction(value)) { value = value(); } if (isFunction(value)) { value = value(); }
ids[key] = value && value.charAt && value.charAt(0) == '@' ? getter(data, value.substr(1)) : value; ids[key] = value && value.charAt && value.charAt(0) == '@' ?
lookupDottedPath(data, value.substr(1)) : value;
}); });
return ids; return ids;
} }
...@@ -416,7 +448,7 @@ angular.module('ngResource', ['ng']). ...@@ -416,7 +448,7 @@ angular.module('ngResource', ['ng']).
} }
function Resource(value){ function Resource(value){
copy(value || {}, this); shallowClearAndCopy(value || {}, this);
} }
forEach(actions, function(action, name) { forEach(actions, function(action, name) {
...@@ -425,6 +457,7 @@ angular.module('ngResource', ['ng']). ...@@ -425,6 +457,7 @@ angular.module('ngResource', ['ng']).
Resource[name] = function(a1, a2, a3, a4) { Resource[name] = function(a1, a2, a3, a4) {
var params = {}, data, success, error; var params = {}, data, success, error;
/* jshint -W086 */ /* (purposefully fall through case statements) */
switch(arguments.length) { switch(arguments.length) {
case 4: case 4:
error = a4; error = a4;
...@@ -456,14 +489,18 @@ angular.module('ngResource', ['ng']). ...@@ -456,14 +489,18 @@ angular.module('ngResource', ['ng']).
case 0: break; case 0: break;
default: default:
throw $resourceMinErr('badargs', throw $resourceMinErr('badargs',
"Expected up to 4 arguments [params, data, success, error], got {0} arguments", arguments.length); "Expected up to 4 arguments [params, data, success, error], got {0} arguments",
arguments.length);
} }
/* jshint +W086 */ /* (purposefully fall through case statements) */
var isInstanceCall = data instanceof Resource; var isInstanceCall = this instanceof Resource;
var value = isInstanceCall ? data : (action.isArray ? [] : new Resource(data)); var value = isInstanceCall ? data : (action.isArray ? [] : new Resource(data));
var httpConfig = {}; var httpConfig = {};
var responseInterceptor = action.interceptor && action.interceptor.response || defaultResponseInterceptor; var responseInterceptor = action.interceptor && action.interceptor.response ||
var responseErrorInterceptor = action.interceptor && action.interceptor.responseError || undefined; defaultResponseInterceptor;
var responseErrorInterceptor = action.interceptor && action.interceptor.responseError ||
undefined;
forEach(action, function(value, key) { forEach(action, function(value, key) {
if (key != 'params' && key != 'isArray' && key != 'interceptor') { if (key != 'params' && key != 'isArray' && key != 'interceptor') {
...@@ -471,34 +508,37 @@ angular.module('ngResource', ['ng']). ...@@ -471,34 +508,37 @@ angular.module('ngResource', ['ng']).
} }
}); });
httpConfig.data = data; if (hasBody) httpConfig.data = data;
route.setUrlParams(httpConfig, extend({}, extractParams(data, action.params || {}), params), action.url); route.setUrlParams(httpConfig,
extend({}, extractParams(data, action.params || {}), params),
action.url);
var promise = $http(httpConfig).then(function(response) { var promise = $http(httpConfig).then(function(response) {
var data = response.data, var data = response.data,
promise = value.$promise; promise = value.$promise;
if (data) { if (data) {
if ( angular.isArray(data) != !!action.isArray ) { // Need to convert action.isArray to boolean in case it is undefined
throw $resourceMinErr('badcfg', 'Error in resource configuration. Expected response' + // jshint -W018
' to contain an {0} but got an {1}', if (angular.isArray(data) !== (!!action.isArray)) {
throw $resourceMinErr('badcfg', 'Error in resource configuration. Expected ' +
'response to contain an {0} but got an {1}',
action.isArray?'array':'object', angular.isArray(data)?'array':'object'); action.isArray?'array':'object', angular.isArray(data)?'array':'object');
} }
// jshint +W018
if (action.isArray) { if (action.isArray) {
value.length = 0; value.length = 0;
forEach(data, function(item) { forEach(data, function(item) {
value.push(new Resource(item)); value.push(new Resource(item));
}); });
} else { } else {
copy(data, value); shallowClearAndCopy(data, value);
value.$promise = promise; value.$promise = promise;
} }
} }
value.$resolved = true; value.$resolved = true;
(success||noop)(value, response.headers);
response.resource = value; response.resource = value;
return response; return response;
...@@ -508,8 +548,15 @@ angular.module('ngResource', ['ng']). ...@@ -508,8 +548,15 @@ angular.module('ngResource', ['ng']).
(error||noop)(response); (error||noop)(response);
return $q.reject(response); return $q.reject(response);
}).then(responseInterceptor, responseErrorInterceptor); });
promise = promise.then(
function(response) {
var value = responseInterceptor(response);
(success||noop)(value, response.headers);
return value;
},
responseErrorInterceptor);
if (!isInstanceCall) { if (!isInstanceCall) {
// we are creating instance / collection // we are creating instance / collection
...@@ -530,19 +577,19 @@ angular.module('ngResource', ['ng']). ...@@ -530,19 +577,19 @@ angular.module('ngResource', ['ng']).
if (isFunction(params)) { if (isFunction(params)) {
error = success; success = params; params = {}; error = success; success = params; params = {};
} }
var result = Resource[name](params, this, success, error); var result = Resource[name].call(this, params, this, success, error);
return result.$promise || result; return result.$promise || result;
}; };
}); });
Resource.bind = function(additionalParamDefaults){ Resource.bind = function(additionalParamDefaults){
return ResourceFactory(url, extend({}, paramDefaults, additionalParamDefaults), actions); return resourceFactory(url, extend({}, paramDefaults, additionalParamDefaults), actions);
}; };
return Resource; return Resource;
} }
return ResourceFactory; return resourceFactory;
}]); }]);
......
/** /**
* @license AngularJS v1.2.0-rc.2 * @license AngularJS v1.2.13
* (c) 2010-2012 Google, Inc. http://angularjs.org * (c) 2010-2014 Google, Inc. http://angularjs.org
* License: MIT * License: MIT
*/ */
(function(window, angular, undefined) {'use strict'; (function(window, angular, undefined) {'use strict';
var copy = angular.copy,
equals = angular.equals,
extend = angular.extend,
forEach = angular.forEach,
isDefined = angular.isDefined,
isFunction = angular.isFunction,
isString = angular.isString,
jqLite = angular.element,
noop = angular.noop,
toJson = angular.toJson;
function inherit(parent, extra) {
return extend(new (extend(function() {}, {prototype:parent}))(), extra);
}
/** /**
* @ngdoc overview * @ngdoc overview
* @name ngRoute * @name ngRoute
...@@ -30,10 +14,14 @@ function inherit(parent, extra) { ...@@ -30,10 +14,14 @@ function inherit(parent, extra) {
* *
* The `ngRoute` module provides routing and deeplinking services and directives for angular apps. * The `ngRoute` module provides routing and deeplinking services and directives for angular apps.
* *
* ## Example
* See {@link ngRoute.$route#example $route} for an example of configuring and using `ngRoute`.
*
* {@installModule route} * {@installModule route}
* *
* <div doc-module-components="ngRoute"></div>
*/ */
/* global -ngRouteModule */
var ngRouteModule = angular.module('ngRoute', ['ng']). var ngRouteModule = angular.module('ngRoute', ['ng']).
provider('$route', $RouteProvider); provider('$route', $RouteProvider);
...@@ -44,11 +32,19 @@ var ngRouteModule = angular.module('ngRoute', ['ng']). ...@@ -44,11 +32,19 @@ var ngRouteModule = angular.module('ngRoute', ['ng']).
* *
* @description * @description
* *
* Used for configuring routes. See {@link ngRoute.$route $route} for an example. * Used for configuring routes.
*
* ## Example
* See {@link ngRoute.$route#example $route} for an example of configuring and using `ngRoute`.
* *
* ## Dependencies
* Requires the {@link ngRoute `ngRoute`} module to be installed. * Requires the {@link ngRoute `ngRoute`} module to be installed.
*/ */
function $RouteProvider(){ function $RouteProvider(){
function inherit(parent, extra) {
return angular.extend(new (angular.extend(function() {}, {prototype:parent}))(), extra);
}
var routes = {}; var routes = {};
/** /**
...@@ -61,13 +57,13 @@ function $RouteProvider(){ ...@@ -61,13 +57,13 @@ function $RouteProvider(){
* `$location.path` will be updated to add or drop the trailing slash to exactly match the * `$location.path` will be updated to add or drop the trailing slash to exactly match the
* route definition. * route definition.
* *
* * `path` can contain named groups starting with a colon (`:name`). All characters up * * `path` can contain named groups starting with a colon: e.g. `:name`. All characters up
* to the next slash are matched and stored in `$routeParams` under the given `name` * to the next slash are matched and stored in `$routeParams` under the given `name`
* when the route matches. * when the route matches.
* * `path` can contain named groups starting with a colon and ending with a star (`:name*`). * * `path` can contain named groups starting with a colon and ending with a star:
* All characters are eagerly stored in `$routeParams` under the given `name` * e.g.`:name*`. All characters are eagerly stored in `$routeParams` under the given `name`
* when the route matches. * when the route matches.
* * `path` can contain optional named groups with a question mark (`:name?`). * * `path` can contain optional named groups with a question mark: e.g.`:name?`.
* *
* For example, routes like `/color/:color/largecode/:largecode*\/edit` will match * For example, routes like `/color/:color/largecode/:largecode*\/edit` will match
* `/color/brown/largecode/code/with/slashs/edit` and extract: * `/color/brown/largecode/code/with/slashs/edit` and extract:
...@@ -81,9 +77,9 @@ function $RouteProvider(){ ...@@ -81,9 +77,9 @@ function $RouteProvider(){
* *
* Object properties: * Object properties:
* *
* - `controller` – `{(string|function()=}` – Controller fn that should be associated with newly * - `controller` – `{(string|function()=}` – Controller fn that should be associated with
* created scope or the name of a {@link angular.Module#controller registered controller} * newly created scope or the name of a {@link angular.Module#controller registered
* if passed as a string. * controller} if passed as a string.
* - `controllerAs` – `{string=}` – A controller alias name. If present the controller will be * - `controllerAs` – `{string=}` – A controller alias name. If present the controller will be
* published to scope under the `controllerAs` name. * published to scope under the `controllerAs` name.
* - `template` – `{string=|function()=}` – html template as a string or a function that * - `template` – `{string=|function()=}` – html template as a string or a function that
...@@ -105,17 +101,22 @@ function $RouteProvider(){ ...@@ -105,17 +101,22 @@ function $RouteProvider(){
* `$location.path()` by applying the current route * `$location.path()` by applying the current route
* *
* - `resolve` - `{Object.<string, function>=}` - An optional map of dependencies which should * - `resolve` - `{Object.<string, function>=}` - An optional map of dependencies which should
* be injected into the controller. If any of these dependencies are promises, they will be * be injected into the controller. If any of these dependencies are promises, the router
* resolved and converted to a value before the controller is instantiated and the * will wait for them all to be resolved or one to be rejected before the controller is
* `$routeChangeSuccess` event is fired. The map object is: * instantiated.
* If all the promises are resolved successfully, the values of the resolved promises are
* injected and {@link ngRoute.$route#$routeChangeSuccess $routeChangeSuccess} event is
* fired. If any of the promises are rejected the
* {@link ngRoute.$route#$routeChangeError $routeChangeError} event is fired. The map object
* is:
* *
* - `key` – `{string}`: a name of a dependency to be injected into the controller. * - `key` – `{string}`: a name of a dependency to be injected into the controller.
* - `factory` - `{string|function}`: If `string` then it is an alias for a service. * - `factory` - `{string|function}`: If `string` then it is an alias for a service.
* Otherwise if function, then it is {@link api/AUTO.$injector#invoke injected} * Otherwise if function, then it is {@link api/AUTO.$injector#invoke injected}
* and the return value is treated as the dependency. If the result is a promise, it is resolved * and the return value is treated as the dependency. If the result is a promise, it is
* before its value is injected into the controller. Be aware that `ngRoute.$routeParams` will * resolved before its value is injected into the controller. Be aware that
* still refer to the previous route within these resolve functions. Use `$route.current.params` * `ngRoute.$routeParams` will still refer to the previous route within these resolve
* to access the new route parameters, instead. * functions. Use `$route.current.params` to access the new route parameters, instead.
* *
* - `redirectTo` – {(string|function())=} – value to update * - `redirectTo` – {(string|function())=} – value to update
* {@link ng.$location $location} path with and trigger route redirection. * {@link ng.$location $location} path with and trigger route redirection.
...@@ -130,8 +131,8 @@ function $RouteProvider(){ ...@@ -130,8 +131,8 @@ function $RouteProvider(){
* The custom `redirectTo` function is expected to return a string which will be used * The custom `redirectTo` function is expected to return a string which will be used
* to update `$location.path()` and `$location.search()`. * to update `$location.path()` and `$location.search()`.
* *
* - `[reloadOnSearch=true]` - {boolean=} - reload route when only $location.search() * - `[reloadOnSearch=true]` - {boolean=} - reload route when only `$location.search()`
* changes. * or `$location.hash()` changes.
* *
* If the option is set to `false` and url in the browser changes, then * If the option is set to `false` and url in the browser changes, then
* `$routeUpdate` event is broadcasted on the root scope. * `$routeUpdate` event is broadcasted on the root scope.
...@@ -147,7 +148,7 @@ function $RouteProvider(){ ...@@ -147,7 +148,7 @@ function $RouteProvider(){
* Adds a new route definition to the `$route` service. * Adds a new route definition to the `$route` service.
*/ */
this.when = function(path, route) { this.when = function(path, route) {
routes[path] = extend( routes[path] = angular.extend(
{reloadOnSearch: true}, {reloadOnSearch: true},
route, route,
path && pathRegExp(path, route) path && pathRegExp(path, route)
...@@ -159,7 +160,7 @@ function $RouteProvider(){ ...@@ -159,7 +160,7 @@ function $RouteProvider(){
? path.substr(0, path.length-1) ? path.substr(0, path.length-1)
: path +'/'; : path +'/';
routes[redirectPath] = extend( routes[redirectPath] = angular.extend(
{redirectTo: path}, {redirectTo: path},
pathRegExp(redirectPath, route) pathRegExp(redirectPath, route)
); );
...@@ -189,7 +190,7 @@ function $RouteProvider(){ ...@@ -189,7 +190,7 @@ function $RouteProvider(){
path = path path = path
.replace(/([().])/g, '\\$1') .replace(/([().])/g, '\\$1')
.replace(/(\/)?:(\w+)([\?|\*])?/g, function(_, slash, key, option){ .replace(/(\/)?:(\w+)([\?\*])?/g, function(_, slash, key, option){
var optional = option === '?' ? option : null; var optional = option === '?' ? option : null;
var star = option === '*' ? option : null; var star = option === '*' ? option : null;
keys.push({ name: key, optional: !!optional }); keys.push({ name: key, optional: !!optional });
...@@ -198,7 +199,9 @@ function $RouteProvider(){ ...@@ -198,7 +199,9 @@ function $RouteProvider(){
+ (optional ? '' : slash) + (optional ? '' : slash)
+ '(?:' + '(?:'
+ (optional ? slash : '') + (optional ? slash : '')
+ (star && '(.+)?' || '([^/]+)?') + ')' + (star && '(.+?)' || '([^/]+)')
+ (optional || '')
+ ')'
+ (optional || ''); + (optional || '');
}) })
.replace(/([\/$\*])/g, '\\$1'); .replace(/([\/$\*])/g, '\\$1');
...@@ -225,8 +228,15 @@ function $RouteProvider(){ ...@@ -225,8 +228,15 @@ function $RouteProvider(){
}; };
this.$get = ['$rootScope', '$location', '$routeParams', '$q', '$injector', '$http', '$templateCache', '$sce', this.$get = ['$rootScope',
function( $rootScope, $location, $routeParams, $q, $injector, $http, $templateCache, $sce) { '$location',
'$routeParams',
'$q',
'$injector',
'$http',
'$templateCache',
'$sce',
function($rootScope, $location, $routeParams, $q, $injector, $http, $templateCache, $sce) {
/** /**
* @ngdoc object * @ngdoc object
...@@ -255,8 +265,9 @@ function $RouteProvider(){ ...@@ -255,8 +265,9 @@ function $RouteProvider(){
* *
* You can define routes through {@link ngRoute.$routeProvider $routeProvider}'s API. * You can define routes through {@link ngRoute.$routeProvider $routeProvider}'s API.
* *
* The `$route` service is typically used in conjunction with the {@link ngRoute.directive:ngView `ngView`} * The `$route` service is typically used in conjunction with the
* directive and the {@link ngRoute.$routeParams `$routeParams`} service. * {@link ngRoute.directive:ngView `ngView`} directive and the
* {@link ngRoute.$routeParams `$routeParams`} service.
* *
* @example * @example
This example shows how changing the URL hash causes the `$route` to match a route against the This example shows how changing the URL hash causes the `$route` to match a route against the
...@@ -265,7 +276,7 @@ function $RouteProvider(){ ...@@ -265,7 +276,7 @@ function $RouteProvider(){
Note that this example is using {@link ng.directive:script inlined templates} Note that this example is using {@link ng.directive:script inlined templates}
to get it working on jsfiddle as well. to get it working on jsfiddle as well.
<example module="ngView" deps="angular-route.js"> <example module="ngViewExample" deps="angular-route.js">
<file name="index.html"> <file name="index.html">
<div ng-controller="MainCntl"> <div ng-controller="MainCntl">
Choose: Choose:
...@@ -298,7 +309,9 @@ function $RouteProvider(){ ...@@ -298,7 +309,9 @@ function $RouteProvider(){
</file> </file>
<file name="script.js"> <file name="script.js">
angular.module('ngView', ['ngRoute']).config(function($routeProvider, $locationProvider) { angular.module('ngViewExample', ['ngRoute'])
.config(function($routeProvider, $locationProvider) {
$routeProvider.when('/Book/:bookId', { $routeProvider.when('/Book/:bookId', {
templateUrl: 'book.html', templateUrl: 'book.html',
controller: BookCntl, controller: BookCntl,
...@@ -337,17 +350,17 @@ function $RouteProvider(){ ...@@ -337,17 +350,17 @@ function $RouteProvider(){
} }
</file> </file>
<file name="scenario.js"> <file name="protractorTest.js">
it('should load and compile correct template', function() { it('should load and compile correct template', function() {
element('a:contains("Moby: Ch1")').click(); element(by.linkText('Moby: Ch1')).click();
var content = element('.doc-example-live [ng-view]').text(); var content = element(by.css('.doc-example-live [ng-view]')).getText();
expect(content).toMatch(/controller\: ChapterCntl/); expect(content).toMatch(/controller\: ChapterCntl/);
expect(content).toMatch(/Book Id\: Moby/); expect(content).toMatch(/Book Id\: Moby/);
expect(content).toMatch(/Chapter Id\: 1/); expect(content).toMatch(/Chapter Id\: 1/);
element('a:contains("Scarlet")').click(); element(by.partialLinkText('Scarlet')).click();
sleep(2); // promises are not part of scenario waiting
content = element('.doc-example-live [ng-view]').text(); content = element(by.css('.doc-example-live [ng-view]')).getText();
expect(content).toMatch(/controller\: BookCntl/); expect(content).toMatch(/controller\: BookCntl/);
expect(content).toMatch(/Book Id\: Scarlet/); expect(content).toMatch(/Book Id\: Scarlet/);
}); });
...@@ -362,11 +375,12 @@ function $RouteProvider(){ ...@@ -362,11 +375,12 @@ function $RouteProvider(){
* @eventType broadcast on root scope * @eventType broadcast on root scope
* @description * @description
* Broadcasted before a route change. At this point the route services starts * Broadcasted before a route change. At this point the route services starts
* resolving all of the dependencies needed for the route change to occurs. * resolving all of the dependencies needed for the route change to occur.
* Typically this involves fetching the view template as well as any dependencies * Typically this involves fetching the view template as well as any dependencies
* defined in `resolve` route property. Once all of the dependencies are resolved * defined in `resolve` route property. Once all of the dependencies are resolved
* `$routeChangeSuccess` is fired. * `$routeChangeSuccess` is fired.
* *
* @param {Object} angularEvent Synthetic event object.
* @param {Route} next Future route information. * @param {Route} next Future route information.
* @param {Route} current Current route information. * @param {Route} current Current route information.
*/ */
...@@ -383,7 +397,8 @@ function $RouteProvider(){ ...@@ -383,7 +397,8 @@ function $RouteProvider(){
* *
* @param {Object} angularEvent Synthetic event object. * @param {Object} angularEvent Synthetic event object.
* @param {Route} current Current route information. * @param {Route} current Current route information.
* @param {Route|Undefined} previous Previous route information, or undefined if current is first route entered. * @param {Route|Undefined} previous Previous route information, or undefined if current is
* first route entered.
*/ */
/** /**
...@@ -394,6 +409,7 @@ function $RouteProvider(){ ...@@ -394,6 +409,7 @@ function $RouteProvider(){
* @description * @description
* Broadcasted if any of the resolve promises are rejected. * Broadcasted if any of the resolve promises are rejected.
* *
* @param {Object} angularEvent Synthetic event object
* @param {Route} current Current route information. * @param {Route} current Current route information.
* @param {Route} previous Previous route information. * @param {Route} previous Previous route information.
* @param {Route} rejection Rejection of the promise. Usually the error of the failed promise. * @param {Route} rejection Rejection of the promise. Usually the error of the failed promise.
...@@ -477,9 +493,10 @@ function $RouteProvider(){ ...@@ -477,9 +493,10 @@ function $RouteProvider(){
last = $route.current; last = $route.current;
if (next && last && next.$$route === last.$$route if (next && last && next.$$route === last.$$route
&& equals(next.pathParams, last.pathParams) && !next.reloadOnSearch && !forceReload) { && angular.equals(next.pathParams, last.pathParams)
&& !next.reloadOnSearch && !forceReload) {
last.params = next.params; last.params = next.params;
copy(last.params, $routeParams); angular.copy(last.params, $routeParams);
$rootScope.$broadcast('$routeUpdate', last); $rootScope.$broadcast('$routeUpdate', last);
} else if (next || last) { } else if (next || last) {
forceReload = false; forceReload = false;
...@@ -487,7 +504,7 @@ function $RouteProvider(){ ...@@ -487,7 +504,7 @@ function $RouteProvider(){
$route.current = next; $route.current = next;
if (next) { if (next) {
if (next.redirectTo) { if (next.redirectTo) {
if (isString(next.redirectTo)) { if (angular.isString(next.redirectTo)) {
$location.path(interpolate(next.redirectTo, next.params)).search(next.params) $location.path(interpolate(next.redirectTo, next.params)).search(next.params)
.replace(); .replace();
} else { } else {
...@@ -500,29 +517,30 @@ function $RouteProvider(){ ...@@ -500,29 +517,30 @@ function $RouteProvider(){
$q.when(next). $q.when(next).
then(function() { then(function() {
if (next) { if (next) {
var locals = extend({}, next.resolve), var locals = angular.extend({}, next.resolve),
template, templateUrl; template, templateUrl;
forEach(locals, function(value, key) { angular.forEach(locals, function(value, key) {
locals[key] = isString(value) ? $injector.get(value) : $injector.invoke(value); locals[key] = angular.isString(value) ?
$injector.get(value) : $injector.invoke(value);
}); });
if (isDefined(template = next.template)) { if (angular.isDefined(template = next.template)) {
if (isFunction(template)) { if (angular.isFunction(template)) {
template = template(next.params); template = template(next.params);
} }
} else if (isDefined(templateUrl = next.templateUrl)) { } else if (angular.isDefined(templateUrl = next.templateUrl)) {
if (isFunction(templateUrl)) { if (angular.isFunction(templateUrl)) {
templateUrl = templateUrl(next.params); templateUrl = templateUrl(next.params);
} }
templateUrl = $sce.getTrustedResourceUrl(templateUrl); templateUrl = $sce.getTrustedResourceUrl(templateUrl);
if (isDefined(templateUrl)) { if (angular.isDefined(templateUrl)) {
next.loadedTemplateUrl = templateUrl; next.loadedTemplateUrl = templateUrl;
template = $http.get(templateUrl, {cache: $templateCache}). template = $http.get(templateUrl, {cache: $templateCache}).
then(function(response) { return response.data; }); then(function(response) { return response.data; });
} }
} }
if (isDefined(template)) { if (angular.isDefined(template)) {
locals['$template'] = template; locals['$template'] = template;
} }
return $q.all(locals); return $q.all(locals);
...@@ -533,7 +551,7 @@ function $RouteProvider(){ ...@@ -533,7 +551,7 @@ function $RouteProvider(){
if (next == $route.current) { if (next == $route.current) {
if (next) { if (next) {
next.locals = locals; next.locals = locals;
copy(next.params, $routeParams); angular.copy(next.params, $routeParams);
} }
$rootScope.$broadcast('$routeChangeSuccess', next, last); $rootScope.$broadcast('$routeChangeSuccess', next, last);
} }
...@@ -552,10 +570,10 @@ function $RouteProvider(){ ...@@ -552,10 +570,10 @@ function $RouteProvider(){
function parseRoute() { function parseRoute() {
// Match a route // Match a route
var params, match; var params, match;
forEach(routes, function(route, path) { angular.forEach(routes, function(route, path) {
if (!match && (params = switchRouteMatcher($location.path(), route))) { if (!match && (params = switchRouteMatcher($location.path(), route))) {
match = inherit(route, { match = inherit(route, {
params: extend({}, $location.search(), params), params: angular.extend({}, $location.search(), params),
pathParams: params}); pathParams: params});
match.$$route = route; match.$$route = route;
} }
...@@ -569,7 +587,7 @@ function $RouteProvider(){ ...@@ -569,7 +587,7 @@ function $RouteProvider(){
*/ */
function interpolate(string, params) { function interpolate(string, params) {
var result = []; var result = [];
forEach((string||'').split(':'), function(segment, i) { angular.forEach((string||'').split(':'), function(segment, i) {
if (i === 0) { if (i === 0) {
result.push(segment); result.push(segment);
} else { } else {
...@@ -599,7 +617,7 @@ ngRouteModule.provider('$routeParams', $RouteParamsProvider); ...@@ -599,7 +617,7 @@ ngRouteModule.provider('$routeParams', $RouteParamsProvider);
* Requires the {@link ngRoute `ngRoute`} module to be installed. * Requires the {@link ngRoute `ngRoute`} module to be installed.
* *
* The route parameters are a combination of {@link ng.$location `$location`}'s * The route parameters are a combination of {@link ng.$location `$location`}'s
* {@link ng.$location#search `search()`} and {@link ng.$location#path `path()`}. * {@link ng.$location#methods_search `search()`} and {@link ng.$location#methods_path `path()`}.
* The `path` parameters are extracted when the {@link ngRoute.$route `$route`} path is matched. * The `path` parameters are extracted when the {@link ngRoute.$route `$route`} path is matched.
* *
* In case of parameter name collision, `path` params take precedence over `search` params. * In case of parameter name collision, `path` params take precedence over `search` params.
...@@ -626,6 +644,8 @@ function $RouteParamsProvider() { ...@@ -626,6 +644,8 @@ function $RouteParamsProvider() {
} }
ngRouteModule.directive('ngView', ngViewFactory); ngRouteModule.directive('ngView', ngViewFactory);
ngRouteModule.directive('ngView', ngViewFillContentFactory);
/** /**
* @ngdoc directive * @ngdoc directive
...@@ -648,6 +668,16 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -648,6 +668,16 @@ ngRouteModule.directive('ngView', ngViewFactory);
* The enter and leave animation occur concurrently. * The enter and leave animation occur concurrently.
* *
* @scope * @scope
* @priority 400
* @param {string=} onload Expression to evaluate whenever the view updates.
*
* @param {string=} autoscroll Whether `ngView` should call {@link ng.$anchorScroll
* $anchorScroll} to scroll the viewport after the view is updated.
*
* - If the attribute is not set, disable scrolling.
* - If the attribute is set without value, enable scrolling.
* - Otherwise enable scrolling only if the `autoscroll` attribute value evaluated
* as an expression yields a truthy value.
* @example * @example
<example module="ngViewExample" deps="angular-route.js" animations="true"> <example module="ngViewExample" deps="angular-route.js" animations="true">
<file name="index.html"> <file name="index.html">
...@@ -659,8 +689,8 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -659,8 +689,8 @@ ngRouteModule.directive('ngView', ngViewFactory);
<a href="Book/Gatsby/ch/4?key=value">Gatsby: Ch4</a> | <a href="Book/Gatsby/ch/4?key=value">Gatsby: Ch4</a> |
<a href="Book/Scarlet">Scarlet Letter</a><br/> <a href="Book/Scarlet">Scarlet Letter</a><br/>
<div class="example-animate-container"> <div class="view-animate-container">
<div ng-view class="view-example"></div> <div ng-view class="view-animate"></div>
</div> </div>
<hr /> <hr />
...@@ -688,7 +718,9 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -688,7 +718,9 @@ ngRouteModule.directive('ngView', ngViewFactory);
</file> </file>
<file name="animations.css"> <file name="animations.css">
.example-animate-container { .view-animate-container {
position:relative;
height:100px!important;
position:relative; position:relative;
background:white; background:white;
border:1px solid black; border:1px solid black;
...@@ -696,14 +728,12 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -696,14 +728,12 @@ ngRouteModule.directive('ngView', ngViewFactory);
overflow:hidden; overflow:hidden;
} }
.example-animate-container > div { .view-animate {
padding:10px; padding:10px;
} }
.view-example.ng-enter, .view-example.ng-leave { .view-animate.ng-enter, .view-animate.ng-leave {
-webkit-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s; -webkit-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s;
-moz-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s;
-o-transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s;
transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s; transition:all cubic-bezier(0.250, 0.460, 0.450, 0.940) 1.5s;
display:block; display:block;
...@@ -718,39 +748,33 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -718,39 +748,33 @@ ngRouteModule.directive('ngView', ngViewFactory);
padding:10px; padding:10px;
} }
.example-animate-container { .view-animate.ng-enter {
position:relative;
height:100px;
}
.view-example.ng-enter {
left:100%; left:100%;
} }
.view-example.ng-enter.ng-enter-active { .view-animate.ng-enter.ng-enter-active {
left:0; left:0;
} }
.view-animate.ng-leave.ng-leave-active {
.view-example.ng-leave { }
.view-example.ng-leave.ng-leave-active {
left:-100%; left:-100%;
} }
</file> </file>
<file name="script.js"> <file name="script.js">
angular.module('ngViewExample', ['ngRoute', 'ngAnimate'], function($routeProvider, $locationProvider) { angular.module('ngViewExample', ['ngRoute', 'ngAnimate'],
$routeProvider.when('/Book/:bookId', { function($routeProvider, $locationProvider) {
templateUrl: 'book.html', $routeProvider.when('/Book/:bookId', {
controller: BookCntl, templateUrl: 'book.html',
controllerAs: 'book' controller: BookCntl,
}); controllerAs: 'book'
$routeProvider.when('/Book/:bookId/ch/:chapterId', { });
templateUrl: 'chapter.html', $routeProvider.when('/Book/:bookId/ch/:chapterId', {
controller: ChapterCntl, templateUrl: 'chapter.html',
controllerAs: 'chapter' controller: ChapterCntl,
}); controllerAs: 'chapter'
});
// configure html5 to get links working on jsfiddle // configure html5 to get links working on jsfiddle
$locationProvider.html5Mode(true); $locationProvider.html5Mode(true);
}); });
function MainCntl($route, $routeParams, $location) { function MainCntl($route, $routeParams, $location) {
...@@ -770,16 +794,17 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -770,16 +794,17 @@ ngRouteModule.directive('ngView', ngViewFactory);
} }
</file> </file>
<file name="scenario.js"> <file name="protractorTest.js">
it('should load and compile correct template', function() { it('should load and compile correct template', function() {
element('a:contains("Moby: Ch1")').click(); element(by.linkText('Moby: Ch1')).click();
var content = element('.doc-example-live [ng-view]').text(); var content = element(by.css('.doc-example-live [ng-view]')).getText();
expect(content).toMatch(/controller\: ChapterCntl/); expect(content).toMatch(/controller\: ChapterCntl/);
expect(content).toMatch(/Book Id\: Moby/); expect(content).toMatch(/Book Id\: Moby/);
expect(content).toMatch(/Chapter Id\: 1/); expect(content).toMatch(/Chapter Id\: 1/);
element('a:contains("Scarlet")').click(); element(by.partialLinkText('Scarlet')).click();
content = element('.doc-example-live [ng-view]').text();
content = element(by.css('.doc-example-live [ng-view]')).getText();
expect(content).toMatch(/controller\: BookCntl/); expect(content).toMatch(/controller\: BookCntl/);
expect(content).toMatch(/Book Id\: Scarlet/); expect(content).toMatch(/Book Id\: Scarlet/);
}); });
...@@ -796,17 +821,17 @@ ngRouteModule.directive('ngView', ngViewFactory); ...@@ -796,17 +821,17 @@ ngRouteModule.directive('ngView', ngViewFactory);
* @description * @description
* Emitted every time the ngView content is reloaded. * Emitted every time the ngView content is reloaded.
*/ */
ngViewFactory.$inject = ['$route', '$anchorScroll', '$compile', '$controller', '$animate']; ngViewFactory.$inject = ['$route', '$anchorScroll', '$animate'];
function ngViewFactory( $route, $anchorScroll, $compile, $controller, $animate) { function ngViewFactory( $route, $anchorScroll, $animate) {
return { return {
restrict: 'ECA', restrict: 'ECA',
terminal: true, terminal: true,
priority: 1000, priority: 400,
transclude: 'element', transclude: 'element',
compile: function(element, attr, linker) { link: function(scope, $element, attr, ctrl, $transclude) {
return function(scope, $element, attr) {
var currentScope, var currentScope,
currentElement, currentElement,
autoScrollExp = attr.autoscroll,
onloadExp = attr.onload || ''; onloadExp = attr.onload || '';
scope.$on('$routeChangeSuccess', update); scope.$on('$routeChangeSuccess', update);
...@@ -827,42 +852,67 @@ function ngViewFactory( $route, $anchorScroll, $compile, $controller, ...@@ -827,42 +852,67 @@ function ngViewFactory( $route, $anchorScroll, $compile, $controller,
var locals = $route.current && $route.current.locals, var locals = $route.current && $route.current.locals,
template = locals && locals.$template; template = locals && locals.$template;
if (template) { if (angular.isDefined(template)) {
var newScope = scope.$new(); var newScope = scope.$new();
linker(newScope, function(clone) { var current = $route.current;
// Note: This will also link all children of ng-view that were contained in the original
// html. If that content contains controllers, ... they could pollute/change the scope.
// However, using ng-view on an element with additional content does not make sense...
// Note: We can't remove them in the cloneAttchFn of $transclude as that
// function is called before linking the content, which would apply child
// directives to non existing elements.
var clone = $transclude(newScope, function(clone) {
$animate.enter(clone, null, currentElement || $element, function onNgViewEnter () {
if (angular.isDefined(autoScrollExp)
&& (!autoScrollExp || scope.$eval(autoScrollExp))) {
$anchorScroll();
}
});
cleanupLastView(); cleanupLastView();
});
clone.html(template); currentElement = clone;
$animate.enter(clone, null, $element); currentScope = current.scope = newScope;
currentScope.$emit('$viewContentLoaded');
var link = $compile(clone.contents()), currentScope.$eval(onloadExp);
current = $route.current; } else {
cleanupLastView();
}
}
}
};
}
currentScope = current.scope = newScope; // This directive is called during the $transclude call of the first `ngView` directive.
currentElement = clone; // It will replace and compile the content of the element with the loaded template.
// We need this directive so that the element content is already filled when
// the link function of another directive on the same element as ngView
// is called.
ngViewFillContentFactory.$inject = ['$compile', '$controller', '$route'];
function ngViewFillContentFactory($compile, $controller, $route) {
return {
restrict: 'ECA',
priority: -400,
link: function(scope, $element) {
var current = $route.current,
locals = current.locals;
if (current.controller) { $element.html(locals.$template);
locals.$scope = currentScope;
var controller = $controller(current.controller, locals);
if (current.controllerAs) {
currentScope[current.controllerAs] = controller;
}
clone.data('$ngControllerController', controller);
clone.contents().data('$ngControllerController', controller);
}
link(currentScope); var link = $compile($element.contents());
currentScope.$emit('$viewContentLoaded');
currentScope.$eval(onloadExp);
// $anchorScroll might listen on event... if (current.controller) {
$anchorScroll(); locals.$scope = scope;
}); var controller = $controller(current.controller, locals);
} else { if (current.controllerAs) {
cleanupLastView(); scope[current.controllerAs] = controller;
}
} }
$element.data('$ngControllerController', controller);
$element.children().data('$ngControllerController', controller);
} }
link(scope);
} }
}; };
} }
......
/** /**
* @license AngularJS v1.2.0-rc.2 * @license AngularJS v1.2.13
* (c) 2010-2012 Google, Inc. http://angularjs.org * (c) 2010-2014 Google, Inc. http://angularjs.org
* License: MIT * License: MIT
*/ */
(function(window, angular, undefined) {'use strict'; (function(window, angular, undefined) {'use strict';
...@@ -18,6 +18,8 @@ var $sanitizeMinErr = angular.$$minErr('$sanitize'); ...@@ -18,6 +18,8 @@ var $sanitizeMinErr = angular.$$minErr('$sanitize');
* *
* {@installModule sanitize} * {@installModule sanitize}
* *
* <div doc-module-components="ngSanitize"></div>
*
* See {@link ngSanitize.$sanitize `$sanitize`} for usage. * See {@link ngSanitize.$sanitize `$sanitize`} for usage.
*/ */
...@@ -49,100 +51,124 @@ var $sanitizeMinErr = angular.$$minErr('$sanitize'); ...@@ -49,100 +51,124 @@ var $sanitizeMinErr = angular.$$minErr('$sanitize');
* it into the returned string, however, since our parser is more strict than a typical browser * it into the returned string, however, since our parser is more strict than a typical browser
* parser, it's possible that some obscure input, which would be recognized as valid HTML by a * parser, it's possible that some obscure input, which would be recognized as valid HTML by a
* browser, won't make it through the sanitizer. * browser, won't make it through the sanitizer.
* The whitelist is configured using the functions `aHrefSanitizationWhitelist` and
* `imgSrcSanitizationWhitelist` of {@link ng.$compileProvider `$compileProvider`}.
* *
* @param {string} html Html input. * @param {string} html Html input.
* @returns {string} Sanitized html. * @returns {string} Sanitized html.
* *
* @example * @example
<doc:example module="ngSanitize"> <doc:example module="ngSanitize">
<doc:source> <doc:source>
<script> <script>
function Ctrl($scope, $sce) { function Ctrl($scope, $sce) {
$scope.snippet = $scope.snippet =
'<p style="color:blue">an html\n' + '<p style="color:blue">an html\n' +
'<em onmouseover="this.textContent=\'PWN3D!\'">click here</em>\n' + '<em onmouseover="this.textContent=\'PWN3D!\'">click here</em>\n' +
'snippet</p>'; 'snippet</p>';
$scope.deliberatelyTrustDangerousSnippet = function() { $scope.deliberatelyTrustDangerousSnippet = function() {
return $sce.trustAsHtml($scope.snippet); return $sce.trustAsHtml($scope.snippet);
}; };
} }
</script> </script>
<div ng-controller="Ctrl"> <div ng-controller="Ctrl">
Snippet: <textarea ng-model="snippet" cols="60" rows="3"></textarea> Snippet: <textarea ng-model="snippet" cols="60" rows="3"></textarea>
<table> <table>
<tr> <tr>
<td>Directive</td> <td>Directive</td>
<td>How</td> <td>How</td>
<td>Source</td> <td>Source</td>
<td>Rendered</td> <td>Rendered</td>
</tr> </tr>
<tr id="bind-html-with-sanitize"> <tr id="bind-html-with-sanitize">
<td>ng-bind-html</td> <td>ng-bind-html</td>
<td>Automatically uses $sanitize</td> <td>Automatically uses $sanitize</td>
<td><pre>&lt;div ng-bind-html="snippet"&gt;<br/>&lt;/div&gt;</pre></td> <td><pre>&lt;div ng-bind-html="snippet"&gt;<br/>&lt;/div&gt;</pre></td>
<td><div ng-bind-html="snippet"></div></td> <td><div ng-bind-html="snippet"></div></td>
</tr> </tr>
<tr id="bind-html-with-trust"> <tr id="bind-html-with-trust">
<td>ng-bind-html</td> <td>ng-bind-html</td>
<td>Bypass $sanitize by explicitly trusting the dangerous value</td> <td>Bypass $sanitize by explicitly trusting the dangerous value</td>
<td><pre>&lt;div ng-bind-html="deliberatelyTrustDangerousSnippet()"&gt;<br/>&lt;/div&gt;</pre></td> <td>
<td><div ng-bind-html="deliberatelyTrustDangerousSnippet()"></div></td> <pre>&lt;div ng-bind-html="deliberatelyTrustDangerousSnippet()"&gt;
</tr> &lt;/div&gt;</pre>
<tr id="bind-default"> </td>
<td>ng-bind</td> <td><div ng-bind-html="deliberatelyTrustDangerousSnippet()"></div></td>
<td>Automatically escapes</td> </tr>
<td><pre>&lt;div ng-bind="snippet"&gt;<br/>&lt;/div&gt;</pre></td> <tr id="bind-default">
<td><div ng-bind="snippet"></div></td> <td>ng-bind</td>
</tr> <td>Automatically escapes</td>
</table> <td><pre>&lt;div ng-bind="snippet"&gt;<br/>&lt;/div&gt;</pre></td>
</div> <td><div ng-bind="snippet"></div></td>
</doc:source> </tr>
<doc:scenario> </table>
it('should sanitize the html snippet by default', function() { </div>
expect(using('#bind-html-with-sanitize').element('div').html()). </doc:source>
toBe('<p>an html\n<em>click here</em>\nsnippet</p>'); <doc:protractor>
}); it('should sanitize the html snippet by default', function() {
expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()).
it('should inline raw snippet if bound to a trusted value', function() { toBe('<p>an html\n<em>click here</em>\nsnippet</p>');
expect(using('#bind-html-with-trust').element("div").html()). });
toBe("<p style=\"color:blue\">an html\n" +
"<em onmouseover=\"this.textContent='PWN3D!'\">click here</em>\n" + it('should inline raw snippet if bound to a trusted value', function() {
"snippet</p>"); expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).
}); toBe("<p style=\"color:blue\">an html\n" +
"<em onmouseover=\"this.textContent='PWN3D!'\">click here</em>\n" +
it('should escape snippet without any filter', function() { "snippet</p>");
expect(using('#bind-default').element('div').html()). });
toBe("&lt;p style=\"color:blue\"&gt;an html\n" +
"&lt;em onmouseover=\"this.textContent='PWN3D!'\"&gt;click here&lt;/em&gt;\n" + it('should escape snippet without any filter', function() {
"snippet&lt;/p&gt;"); expect(element(by.css('#bind-default div')).getInnerHtml()).
}); toBe("&lt;p style=\"color:blue\"&gt;an html\n" +
"&lt;em onmouseover=\"this.textContent='PWN3D!'\"&gt;click here&lt;/em&gt;\n" +
it('should update', function() { "snippet&lt;/p&gt;");
input('snippet').enter('new <b onclick="alert(1)">text</b>'); });
expect(using('#bind-html-with-sanitize').element('div').html()).toBe('new <b>text</b>');
expect(using('#bind-html-with-trust').element('div').html()).toBe('new <b onclick="alert(1)">text</b>'); it('should update', function() {
expect(using('#bind-default').element('div').html()).toBe("new &lt;b onclick=\"alert(1)\"&gt;text&lt;/b&gt;"); element(by.model('snippet')).clear();
}); element(by.model('snippet')).sendKeys('new <b onclick="alert(1)">text</b>');
</doc:scenario> expect(element(by.css('#bind-html-with-sanitize div')).getInnerHtml()).
toBe('new <b>text</b>');
expect(element(by.css('#bind-html-with-trust div')).getInnerHtml()).toBe(
'new <b onclick="alert(1)">text</b>');
expect(element(by.css('#bind-default div')).getInnerHtml()).toBe(
"new &lt;b onclick=\"alert(1)\"&gt;text&lt;/b&gt;");
});
</doc:protractor>
</doc:example> </doc:example>
*/ */
var $sanitize = function(html) { function $SanitizeProvider() {
this.$get = ['$$sanitizeUri', function($$sanitizeUri) {
return function(html) {
var buf = [];
htmlParser(html, htmlSanitizeWriter(buf, function(uri, isImage) {
return !/^unsafe/.test($$sanitizeUri(uri, isImage));
}));
return buf.join('');
};
}];
}
function sanitizeText(chars) {
var buf = []; var buf = [];
htmlParser(html, htmlSanitizeWriter(buf)); var writer = htmlSanitizeWriter(buf, angular.noop);
return buf.join(''); writer.chars(chars);
}; return buf.join('');
}
// Regular Expressions for parsing tags and attributes // Regular Expressions for parsing tags and attributes
var START_TAG_REGEXP = /^<\s*([\w:-]+)((?:\s+[\w:-]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)\s*>/, var START_TAG_REGEXP =
/^<\s*([\w:-]+)((?:\s+[\w:-]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)\s*>/,
END_TAG_REGEXP = /^<\s*\/\s*([\w:-]+)[^>]*>/, END_TAG_REGEXP = /^<\s*\/\s*([\w:-]+)[^>]*>/,
ATTR_REGEXP = /([\w:-]+)(?:\s*=\s*(?:(?:"((?:[^"])*)")|(?:'((?:[^'])*)')|([^>\s]+)))?/g, ATTR_REGEXP = /([\w:-]+)(?:\s*=\s*(?:(?:"((?:[^"])*)")|(?:'((?:[^'])*)')|([^>\s]+)))?/g,
BEGIN_TAG_REGEXP = /^</, BEGIN_TAG_REGEXP = /^</,
BEGING_END_TAGE_REGEXP = /^<\s*\//, BEGING_END_TAGE_REGEXP = /^<\s*\//,
COMMENT_REGEXP = /<!--(.*?)-->/g, COMMENT_REGEXP = /<!--(.*?)-->/g,
DOCTYPE_REGEXP = /<!DOCTYPE([^>]*?)>/i,
CDATA_REGEXP = /<!\[CDATA\[(.*?)]]>/g, CDATA_REGEXP = /<!\[CDATA\[(.*?)]]>/g,
URI_REGEXP = /^((ftp|https?):\/\/|mailto:|tel:|#)/i, // Match everything outside of normal chars and " (quote character)
NON_ALPHANUMERIC_REGEXP = /([^\#-~| |!])/g; // Match everything outside of normal chars and " (quote character) NON_ALPHANUMERIC_REGEXP = /([^\#-~| |!])/g;
// Good source of info about elements and attributes // Good source of info about elements and attributes
...@@ -157,23 +183,29 @@ var voidElements = makeMap("area,br,col,hr,img,wbr"); ...@@ -157,23 +183,29 @@ var voidElements = makeMap("area,br,col,hr,img,wbr");
// http://dev.w3.org/html5/spec/Overview.html#optional-tags // http://dev.w3.org/html5/spec/Overview.html#optional-tags
var optionalEndTagBlockElements = makeMap("colgroup,dd,dt,li,p,tbody,td,tfoot,th,thead,tr"), var optionalEndTagBlockElements = makeMap("colgroup,dd,dt,li,p,tbody,td,tfoot,th,thead,tr"),
optionalEndTagInlineElements = makeMap("rp,rt"), optionalEndTagInlineElements = makeMap("rp,rt"),
optionalEndTagElements = angular.extend({}, optionalEndTagInlineElements, optionalEndTagBlockElements); optionalEndTagElements = angular.extend({},
optionalEndTagInlineElements,
optionalEndTagBlockElements);
// Safe Block Elements - HTML5 // Safe Block Elements - HTML5
var blockElements = angular.extend({}, optionalEndTagBlockElements, makeMap("address,article,aside," + var blockElements = angular.extend({}, optionalEndTagBlockElements, makeMap("address,article," +
"blockquote,caption,center,del,dir,div,dl,figure,figcaption,footer,h1,h2,h3,h4,h5,h6," + "aside,blockquote,caption,center,del,dir,div,dl,figure,figcaption,footer,h1,h2,h3,h4,h5," +
"header,hgroup,hr,ins,map,menu,nav,ol,pre,script,section,table,ul")); "h6,header,hgroup,hr,ins,map,menu,nav,ol,pre,script,section,table,ul"));
// Inline Elements - HTML5 // Inline Elements - HTML5
var inlineElements = angular.extend({}, optionalEndTagInlineElements, makeMap("a,abbr,acronym,b,bdi,bdo," + var inlineElements = angular.extend({}, optionalEndTagInlineElements, makeMap("a,abbr,acronym,b," +
"big,br,cite,code,del,dfn,em,font,i,img,ins,kbd,label,map,mark,q,ruby,rp,rt,s,samp,small," + "bdi,bdo,big,br,cite,code,del,dfn,em,font,i,img,ins,kbd,label,map,mark,q,ruby,rp,rt,s," +
"span,strike,strong,sub,sup,time,tt,u,var")); "samp,small,span,strike,strong,sub,sup,time,tt,u,var"));
// Special Elements (can contain anything) // Special Elements (can contain anything)
var specialElements = makeMap("script,style"); var specialElements = makeMap("script,style");
var validElements = angular.extend({}, voidElements, blockElements, inlineElements, optionalEndTagElements); var validElements = angular.extend({},
voidElements,
blockElements,
inlineElements,
optionalEndTagElements);
//Attributes that have href and hence need to be sanitized //Attributes that have href and hence need to be sanitized
var uriAttrs = makeMap("background,cite,href,longdesc,src,usemap"); var uriAttrs = makeMap("background,cite,href,longdesc,src,usemap");
...@@ -181,7 +213,7 @@ var validAttrs = angular.extend({}, uriAttrs, makeMap( ...@@ -181,7 +213,7 @@ var validAttrs = angular.extend({}, uriAttrs, makeMap(
'abbr,align,alt,axis,bgcolor,border,cellpadding,cellspacing,class,clear,'+ 'abbr,align,alt,axis,bgcolor,border,cellpadding,cellspacing,class,clear,'+
'color,cols,colspan,compact,coords,dir,face,headers,height,hreflang,hspace,'+ 'color,cols,colspan,compact,coords,dir,face,headers,height,hreflang,hspace,'+
'ismap,lang,language,nohref,nowrap,rel,rev,rows,rowspan,rules,'+ 'ismap,lang,language,nohref,nowrap,rel,rev,rows,rowspan,rules,'+
'scope,scrolling,shape,span,start,summary,target,title,type,'+ 'scope,scrolling,shape,size,span,start,summary,target,title,type,'+
'valign,value,vspace,width')); 'valign,value,vspace,width'));
function makeMap(str) { function makeMap(str) {
...@@ -215,14 +247,22 @@ function htmlParser( html, handler ) { ...@@ -215,14 +247,22 @@ function htmlParser( html, handler ) {
// Comment // Comment
if ( html.indexOf("<!--") === 0 ) { if ( html.indexOf("<!--") === 0 ) {
index = html.indexOf("-->"); // comments containing -- are not allowed unless they terminate the comment
index = html.indexOf("--", 4);
if ( index >= 0 ) { if ( index >= 0 && html.lastIndexOf("-->", index) === index) {
if (handler.comment) handler.comment( html.substring( 4, index ) ); if (handler.comment) handler.comment( html.substring( 4, index ) );
html = html.substring( index + 3 ); html = html.substring( index + 3 );
chars = false; chars = false;
} }
// DOCTYPE
} else if ( DOCTYPE_REGEXP.test(html) ) {
match = html.match( DOCTYPE_REGEXP );
if ( match ) {
html = html.replace( match[0] , '');
chars = false;
}
// end tag // end tag
} else if ( BEGING_END_TAGE_REGEXP.test(html) ) { } else if ( BEGING_END_TAGE_REGEXP.test(html) ) {
match = html.match( END_TAG_REGEXP ); match = html.match( END_TAG_REGEXP );
...@@ -254,21 +294,21 @@ function htmlParser( html, handler ) { ...@@ -254,21 +294,21 @@ function htmlParser( html, handler ) {
} }
} else { } else {
html = html.replace(new RegExp("(.*)<\\s*\\/\\s*" + stack.last() + "[^>]*>", 'i'), function(all, text){ html = html.replace(new RegExp("(.*)<\\s*\\/\\s*" + stack.last() + "[^>]*>", 'i'),
text = text. function(all, text){
replace(COMMENT_REGEXP, "$1"). text = text.replace(COMMENT_REGEXP, "$1").replace(CDATA_REGEXP, "$1");
replace(CDATA_REGEXP, "$1");
if (handler.chars) handler.chars( decodeEntities(text) ); if (handler.chars) handler.chars( decodeEntities(text) );
return ""; return "";
}); });
parseEndTag( "", stack.last() ); parseEndTag( "", stack.last() );
} }
if ( html == last ) { if ( html == last ) {
throw $sanitizeMinErr('badparse', "The sanitizer was unable to parse the following block of html: {0}", html); throw $sanitizeMinErr('badparse', "The sanitizer was unable to parse the following block " +
"of html: {0}", html);
} }
last = html; last = html;
} }
...@@ -295,13 +335,14 @@ function htmlParser( html, handler ) { ...@@ -295,13 +335,14 @@ function htmlParser( html, handler ) {
var attrs = {}; var attrs = {};
rest.replace(ATTR_REGEXP, function(match, name, doubleQuotedValue, singleQuotedValue, unquotedValue) { rest.replace(ATTR_REGEXP,
var value = doubleQuotedValue function(match, name, doubleQuotedValue, singleQuotedValue, unquotedValue) {
|| singleQuotedValue var value = doubleQuotedValue
|| unquotedValue || singleQuotedValue
|| ''; || unquotedValue
|| '';
attrs[name] = decodeEntities(value); attrs[name] = decodeEntities(value);
}); });
if (handler.start) handler.start( tagName, attrs, unary ); if (handler.start) handler.start( tagName, attrs, unary );
} }
...@@ -326,15 +367,32 @@ function htmlParser( html, handler ) { ...@@ -326,15 +367,32 @@ function htmlParser( html, handler ) {
} }
} }
var hiddenPre=document.createElement("pre");
var spaceRe = /^(\s*)([\s\S]*?)(\s*)$/;
/** /**
* decodes all entities into regular string * decodes all entities into regular string
* @param value * @param value
* @returns {string} A string with decoded entities. * @returns {string} A string with decoded entities.
*/ */
var hiddenPre=document.createElement("pre");
function decodeEntities(value) { function decodeEntities(value) {
hiddenPre.innerHTML=value.replace(/</g,"&lt;"); if (!value) { return ''; }
return hiddenPre.innerText || hiddenPre.textContent || '';
// Note: IE8 does not preserve spaces at the start/end of innerHTML
// so we must capture them and reattach them afterward
var parts = spaceRe.exec(value);
var spaceBefore = parts[1];
var spaceAfter = parts[3];
var content = parts[2];
if (content) {
hiddenPre.innerHTML=content.replace(/</g,"&lt;");
// innerText depends on styling as it doesn't display hidden elements.
// Therefore, it's better to use textContent not to cause unnecessary
// reflows. However, IE<9 don't support textContent so the innerText
// fallback is necessary.
content = 'textContent' in hiddenPre ?
hiddenPre.textContent : hiddenPre.innerText;
}
return spaceBefore + content + spaceAfter;
} }
/** /**
...@@ -364,7 +422,7 @@ function encodeEntities(value) { ...@@ -364,7 +422,7 @@ function encodeEntities(value) {
* comment: function(text) {} * comment: function(text) {}
* } * }
*/ */
function htmlSanitizeWriter(buf){ function htmlSanitizeWriter(buf, uriValidator){
var ignore = false; var ignore = false;
var out = angular.bind(buf, buf.push); var out = angular.bind(buf, buf.push);
return { return {
...@@ -373,12 +431,14 @@ function htmlSanitizeWriter(buf){ ...@@ -373,12 +431,14 @@ function htmlSanitizeWriter(buf){
if (!ignore && specialElements[tag]) { if (!ignore && specialElements[tag]) {
ignore = tag; ignore = tag;
} }
if (!ignore && validElements[tag] == true) { if (!ignore && validElements[tag] === true) {
out('<'); out('<');
out(tag); out(tag);
angular.forEach(attrs, function(value, key){ angular.forEach(attrs, function(value, key){
var lkey=angular.lowercase(key); var lkey=angular.lowercase(key);
if (validAttrs[lkey]==true && (uriAttrs[lkey]!==true || value.match(URI_REGEXP))) { var isImage = (tag === 'img' && lkey === 'src') || (lkey === 'background');
if (validAttrs[lkey] === true &&
(uriAttrs[lkey] !== true || uriValidator(value, isImage))) {
out(' '); out(' ');
out(key); out(key);
out('="'); out('="');
...@@ -391,7 +451,7 @@ function htmlSanitizeWriter(buf){ ...@@ -391,7 +451,7 @@ function htmlSanitizeWriter(buf){
}, },
end: function(tag){ end: function(tag){
tag = angular.lowercase(tag); tag = angular.lowercase(tag);
if (!ignore && validElements[tag] == true) { if (!ignore && validElements[tag] === true) {
out('</'); out('</');
out(tag); out(tag);
out('>'); out('>');
...@@ -410,7 +470,9 @@ function htmlSanitizeWriter(buf){ ...@@ -410,7 +470,9 @@ function htmlSanitizeWriter(buf){
// define ngSanitize module and register $sanitize service // define ngSanitize module and register $sanitize service
angular.module('ngSanitize', []).value('$sanitize', $sanitize); angular.module('ngSanitize', []).provider('$sanitize', $SanitizeProvider);
/* global sanitizeText: false */
/** /**
* @ngdoc filter * @ngdoc filter
...@@ -477,41 +539,43 @@ angular.module('ngSanitize', []).value('$sanitize', $sanitize); ...@@ -477,41 +539,43 @@ angular.module('ngSanitize', []).value('$sanitize', $sanitize);
</tr> </tr>
</table> </table>
</doc:source> </doc:source>
<doc:scenario> <doc:protractor>
it('should linkify the snippet with urls', function() { it('should linkify the snippet with urls', function() {
expect(using('#linky-filter').binding('snippet | linky')). expect(element(by.id('linky-filter')).element(by.binding('snippet | linky')).getText()).
toBe('Pretty text with some links:&#10;' + toBe('Pretty text with some links: http://angularjs.org/, us@somewhere.org, ' +
'<a href="http://angularjs.org/">http://angularjs.org/</a>,&#10;' + 'another@somewhere.org, and one more: ftp://127.0.0.1/.');
'<a href="mailto:us@somewhere.org">us@somewhere.org</a>,&#10;' + expect(element.all(by.css('#linky-filter a')).count()).toEqual(4);
'<a href="mailto:another@somewhere.org">another@somewhere.org</a>,&#10;' +
'and one more: <a href="ftp://127.0.0.1/">ftp://127.0.0.1/</a>.');
}); });
it ('should not linkify snippet without the linky filter', function() { it('should not linkify snippet without the linky filter', function() {
expect(using('#escaped-html').binding('snippet')). expect(element(by.id('escaped-html')).element(by.binding('snippet')).getText()).
toBe("Pretty text with some links:\n" + toBe('Pretty text with some links: http://angularjs.org/, mailto:us@somewhere.org, ' +
"http://angularjs.org/,\n" + 'another@somewhere.org, and one more: ftp://127.0.0.1/.');
"mailto:us@somewhere.org,\n" + expect(element.all(by.css('#escaped-html a')).count()).toEqual(0);
"another@somewhere.org,\n" +
"and one more: ftp://127.0.0.1/.");
}); });
it('should update', function() { it('should update', function() {
input('snippet').enter('new http://link.'); element(by.model('snippet')).clear();
expect(using('#linky-filter').binding('snippet | linky')). element(by.model('snippet')).sendKeys('new http://link.');
toBe('new <a href="http://link">http://link</a>.'); expect(element(by.id('linky-filter')).element(by.binding('snippet | linky')).getText()).
expect(using('#escaped-html').binding('snippet')).toBe('new http://link.'); toBe('new http://link.');
expect(element.all(by.css('#linky-filter a')).count()).toEqual(1);
expect(element(by.id('escaped-html')).element(by.binding('snippet')).getText())
.toBe('new http://link.');
}); });
it('should work with the target property', function() { it('should work with the target property', function() {
expect(using('#linky-target').binding("snippetWithTarget | linky:'_blank'")). expect(element(by.id('linky-target')).
toBe('<a target="_blank" href="http://angularjs.org/">http://angularjs.org/</a>'); element(by.binding("snippetWithTarget | linky:'_blank'")).getText()).
toBe('http://angularjs.org/');
expect(element(by.css('#linky-target a')).getAttribute('target')).toEqual('_blank');
}); });
</doc:scenario> </doc:protractor>
</doc:example> </doc:example>
*/ */
angular.module('ngSanitize').filter('linky', function() { angular.module('ngSanitize').filter('linky', ['$sanitize', function($sanitize) {
var LINKY_URL_REGEXP = /((ftp|https?):\/\/|(mailto:)?[A-Za-z0-9._%+-]+@)\S*[^\s\.\;\,\(\)\{\}\<\>]/, var LINKY_URL_REGEXP =
/((ftp|https?):\/\/|(mailto:)?[A-Za-z0-9._%+-]+@)\S*[^\s.;,(){}<>]/,
MAILTO_REGEXP = /^mailto:/; MAILTO_REGEXP = /^mailto:/;
return function(text, target) { return function(text, target) {
...@@ -519,31 +583,43 @@ angular.module('ngSanitize').filter('linky', function() { ...@@ -519,31 +583,43 @@ angular.module('ngSanitize').filter('linky', function() {
var match; var match;
var raw = text; var raw = text;
var html = []; var html = [];
// TODO(vojta): use $sanitize instead
var writer = htmlSanitizeWriter(html);
var url; var url;
var i; var i;
var properties = {};
if (angular.isDefined(target)) {
properties.target = target;
}
while ((match = raw.match(LINKY_URL_REGEXP))) { while ((match = raw.match(LINKY_URL_REGEXP))) {
// We can not end in these as they are sometimes found at the end of the sentence // We can not end in these as they are sometimes found at the end of the sentence
url = match[0]; url = match[0];
// if we did not match ftp/http/mailto then assume mailto // if we did not match ftp/http/mailto then assume mailto
if (match[2] == match[3]) url = 'mailto:' + url; if (match[2] == match[3]) url = 'mailto:' + url;
i = match.index; i = match.index;
writer.chars(raw.substr(0, i)); addText(raw.substr(0, i));
properties.href = url; addLink(url, match[0].replace(MAILTO_REGEXP, ''));
writer.start('a', properties);
writer.chars(match[0].replace(MAILTO_REGEXP, ''));
writer.end('a');
raw = raw.substring(i + match[0].length); raw = raw.substring(i + match[0].length);
} }
writer.chars(raw); addText(raw);
return html.join(''); return $sanitize(html.join(''));
function addText(text) {
if (!text) {
return;
}
html.push(sanitizeText(text));
}
function addLink(url, text) {
html.push('<a ');
if (angular.isDefined(target)) {
html.push('target="');
html.push(target);
html.push('" ');
}
html.push('href="');
html.push(url);
html.push('">');
addText(text);
html.push('</a>');
}
}; };
}); }]);
})(window, window.angular); })(window, window.angular);
This source diff could not be displayed because it is too large. You can view the blob instead.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment