diff options
| author | Igor Minar | 2011-08-02 16:33:20 -0700 |
|---|---|---|
| committer | Igor Minar | 2011-08-24 15:01:50 -0700 |
| commit | 08a33e7bb377a4d47917dbf5fabbe59b562f1e04 (patch) | |
| tree | 136537a62b5da24bc40ebde75d0862a501247303 /test/ScopeSpec.js | |
| parent | 30753cb1310893841fdb0b17c075b6a72e8c8d8a (diff) | |
| download | angular.js-08a33e7bb377a4d47917dbf5fabbe59b562f1e04.tar.bz2 | |
feat(scope): support for events
- register listeners with $on
- remove listeners with $removeListener
- fire event that bubbles to root with $emit
- fire event that propagates to all child scopes with $broadcast
Diffstat (limited to 'test/ScopeSpec.js')
| -rw-r--r-- | test/ScopeSpec.js | 353 |
1 files changed, 279 insertions, 74 deletions
diff --git a/test/ScopeSpec.js b/test/ScopeSpec.js index efc8c98c..b3bdf935 100644 --- a/test/ScopeSpec.js +++ b/test/ScopeSpec.js @@ -1,9 +1,9 @@ 'use strict'; -describe('Scope', function(){ +describe('Scope', function() { var root, mockHandler; - beforeEach(function(){ + beforeEach(function() { root = createScope(angular.service, { '$exceptionHandler': $exceptionHandlerMockFactory() }); @@ -11,14 +11,14 @@ describe('Scope', function(){ }); - describe('$root', function(){ - it('should point to itself', function(){ + describe('$root', function() { + it('should point to itself', function() { expect(root.$root).toEqual(root); expect(root.hasOwnProperty('$root')).toBeTruthy(); }); - it('should not have $root on children, but should inherit', function(){ + it('should not have $root on children, but should inherit', function() { var child = root.$new(); expect(child.$root).toEqual(root); expect(child.hasOwnProperty('$root')).toBeFalsy(); @@ -27,13 +27,13 @@ describe('Scope', function(){ }); - describe('$parent', function(){ - it('should point to itself in root', function(){ + describe('$parent', function() { + it('should point to itself in root', function() { expect(root.$root).toEqual(root); }); - it('should point to parent', function(){ + it('should point to parent', function() { var child = root.$new(); expect(root.$parent).toEqual(null); expect(child.$parent).toEqual(root); @@ -42,29 +42,29 @@ describe('Scope', function(){ }); - describe('$id', function(){ - it('should have a unique id', function(){ + describe('$id', function() { + it('should have a unique id', function() { expect(root.$id < root.$new().$id).toBeTruthy(); }); }); - describe('this', function(){ - it('should have a \'this\'', function(){ + describe('this', function() { + it('should have a \'this\'', function() { expect(root['this']).toEqual(root); }); }); - describe('$new()', function(){ - it('should create a child scope', function(){ + describe('$new()', function() { + it('should create a child scope', function() { var child = root.$new(); root.a = 123; expect(child.a).toEqual(123); }); - it('should instantiate controller and bind functions', function(){ + it('should instantiate controller and bind functions', function() { function Cntl($browser, name){ this.$browser = $browser; this.callCount = 0; @@ -73,7 +73,7 @@ describe('Scope', function(){ Cntl.$inject = ['$browser']; Cntl.prototype = { - myFn: function(){ + myFn: function() { expect(this).toEqual(cntl); this.callCount++; } @@ -94,15 +94,15 @@ describe('Scope', function(){ }); - describe('$service', function(){ - it('should have it on root', function(){ + describe('$service', function() { + it('should have it on root', function() { expect(root.hasOwnProperty('$service')).toBeTruthy(); }); }); - describe('$watch/$digest', function(){ - it('should watch and fire on simple property change', function(){ + describe('$watch/$digest', function() { + it('should watch and fire on simple property change', function() { var spy = jasmine.createSpy(); root.$watch('name', spy); root.$digest(); @@ -117,7 +117,7 @@ describe('Scope', function(){ }); - it('should watch and fire on expression change', function(){ + it('should watch and fire on expression change', function() { var spy = jasmine.createSpy(); root.$watch('name.first', spy); root.$digest(); @@ -132,8 +132,8 @@ describe('Scope', function(){ expect(spy).wasCalled(); }); - it('should delegate exceptions', function(){ - root.$watch('a', function(){throw new Error('abc');}); + it('should delegate exceptions', function() { + root.$watch('a', function() {throw new Error('abc');}); root.a = 1; root.$digest(); expect(mockHandler.errors[0].message).toEqual('abc'); @@ -141,34 +141,34 @@ describe('Scope', function(){ }); - it('should fire watches in order of addition', function(){ + it('should fire watches in order of addition', function() { // this is not an external guarantee, just our own sanity var log = ''; - root.$watch('a', function(){ log += 'a'; }); - root.$watch('b', function(){ log += 'b'; }); - root.$watch('c', function(){ log += 'c'; }); + root.$watch('a', function() { log += 'a'; }); + root.$watch('b', function() { log += 'b'; }); + root.$watch('c', function() { log += 'c'; }); root.a = root.b = root.c = 1; root.$digest(); expect(log).toEqual('abc'); }); - it('should delegate $digest to children in addition order', function(){ + it('should delegate $digest to children in addition order', function() { // this is not an external guarantee, just our own sanity var log = ''; var childA = root.$new(); var childB = root.$new(); var childC = root.$new(); - childA.$watch('a', function(){ log += 'a'; }); - childB.$watch('b', function(){ log += 'b'; }); - childC.$watch('c', function(){ log += 'c'; }); + childA.$watch('a', function() { log += 'a'; }); + childB.$watch('b', function() { log += 'b'; }); + childC.$watch('c', function() { log += 'c'; }); childA.a = childB.b = childC.c = 1; root.$digest(); expect(log).toEqual('abc'); }); - it('should repeat watch cycle while model changes are identified', function(){ + it('should repeat watch cycle while model changes are identified', function() { var log = ''; root.$watch('c', function(self, v){self.d = v; log+='c'; }); root.$watch('b', function(self, v){self.c = v; log+='b'; }); @@ -183,32 +183,32 @@ describe('Scope', function(){ expect(log).toEqual('abc'); }); - it('should repeat watch cycle from the root elemnt', function(){ + it('should repeat watch cycle from the root elemnt', function() { var log = ''; var child = root.$new(); - root.$watch(function(){ log += 'a'; }); - child.$watch(function(){ log += 'b'; }); + root.$watch(function() { log += 'a'; }); + child.$watch(function() { log += 'b'; }); root.$digest(); expect(log).toEqual('abab'); }); - it('should prevent infinite recursion', function(){ + it('should prevent infinite recursion', function() { root.$watch('a', function(self, v){self.b++;}); root.$watch('b', function(self, v){self.a++;}); root.a = root.b = 0; - expect(function(){ + expect(function() { root.$digest(); }).toThrow('100 $digest() iterations reached. Aborting!'); }); - it('should not fire upon $watch registration on initial $digest', function(){ + it('should not fire upon $watch registration on initial $digest', function() { var log = ''; root.a = 1; - root.$watch('a', function(){ log += 'a'; }); - root.$watch('b', function(){ log += 'b'; }); + root.$watch('a', function() { log += 'a'; }); + root.$watch('b', function() { log += 'b'; }); root.$digest(); log = ''; root.$digest(); @@ -216,12 +216,12 @@ describe('Scope', function(){ }); - it('should watch objects', function(){ + it('should watch objects', function() { var log = ''; root.a = []; root.b = {}; - root.$watch('a', function(){ log +='.';}); - root.$watch('b', function(){ log +='!';}); + root.$watch('a', function() { log +='.';}); + root.$watch('b', function() { log +='!';}); root.$digest(); log = ''; @@ -233,10 +233,10 @@ describe('Scope', function(){ }); - it('should prevent recursion', function(){ + it('should prevent recursion', function() { var callCount = 0; - root.$watch('name', function(){ - expect(function(){ + root.$watch('name', function() { + expect(function() { root.$digest(); }).toThrow('$digest already in progress'); callCount++; @@ -248,56 +248,66 @@ describe('Scope', function(){ }); - describe('$destroy', function(){ + describe('$destroy', function() { var first, middle, last, log; - beforeEach(function(){ + beforeEach(function() { log = ''; first = root.$new(); middle = root.$new(); last = root.$new(); - first.$watch(function(){ log += '1';}); - middle.$watch(function(){ log += '2';}); - last.$watch(function(){ log += '3';}); + first.$watch(function() { log += '1';}); + middle.$watch(function() { log += '2';}); + last.$watch(function() { log += '3';}); root.$digest(); log = ''; }); - it('should ignore remove on root', function(){ + it('should ignore remove on root', function() { root.$destroy(); root.$digest(); expect(log).toEqual('123'); }); - it('should remove first', function(){ + it('should remove first', function() { first.$destroy(); root.$digest(); expect(log).toEqual('23'); }); - it('should remove middle', function(){ + it('should remove middle', function() { middle.$destroy(); root.$digest(); expect(log).toEqual('13'); }); - it('should remove last', function(){ + it('should remove last', function() { last.$destroy(); root.$digest(); expect(log).toEqual('12'); }); + + it('should fire a $destroy event', function() { + var destructedScopes = []; + middle.$on('$destroy', function(event) { + destructedScopes.push(event.currentScope); + }); + middle.$destroy(); + expect(destructedScopes).toEqual([middle]); + }); + }); - describe('$eval', function(){ - it('should eval an expression', function(){ + describe('$eval', function() { + it('should eval an expression', function() { expect(root.$eval('a=1')).toEqual(1); expect(root.a).toEqual(1); @@ -306,24 +316,24 @@ describe('Scope', function(){ }); }); - describe('$evalAsync', function(){ + describe('$evalAsync', function() { - it('should run callback before $watch', function(){ + it('should run callback before $watch', function() { var log = ''; var child = root.$new(); root.$evalAsync(function(scope){ log += 'parent.async;'; }); - root.$watch('value', function(){ log += 'parent.$digest;'; }); + root.$watch('value', function() { log += 'parent.$digest;'; }); child.$evalAsync(function(scope){ log += 'child.async;'; }); - child.$watch('value', function(){ log += 'child.$digest;'; }); + child.$watch('value', function() { log += 'child.$digest;'; }); root.$digest(); expect(log).toEqual('parent.async;parent.$digest;child.async;child.$digest;'); }); - it('should cause a $digest rerun', function(){ + it('should cause a $digest rerun', function() { root.log = ''; root.value = 0; root.$watch('value', 'log = log + ".";'); - root.$watch('init', function(){ + root.$watch('init', function() { root.$evalAsync('value = 123; log = log + "=" '); expect(root.value).toEqual(0); }); @@ -331,7 +341,7 @@ describe('Scope', function(){ expect(root.log).toEqual('.=.'); }); - it('should run async in the same order as added', function(){ + it('should run async in the same order as added', function() { root.log = ''; root.$evalAsync("log = log + 1"); root.$evalAsync("log = log + 2"); @@ -342,8 +352,8 @@ describe('Scope', function(){ }); - describe('$apply', function(){ - it('should apply expression with full lifecycle', function(){ + describe('$apply', function() { + it('should apply expression with full lifecycle', function() { var log = ''; var child = root.$new(); root.$watch('a', function(scope, a){ log += '1'; }); @@ -352,33 +362,33 @@ describe('Scope', function(){ }); - it('should catch exceptions', function(){ + it('should catch exceptions', function() { var log = ''; var child = root.$new(); root.$watch('a', function(scope, a){ log += '1'; }); root.a = 0; - child.$apply(function(){ throw new Error('MyError'); }); + child.$apply(function() { throw new Error('MyError'); }); expect(log).toEqual('1'); expect(mockHandler.errors[0].message).toEqual('MyError'); $logMock.error.logs.shift(); }); - describe('exceptions', function(){ + describe('exceptions', function() { var $exceptionHandler, log; - beforeEach(function(){ + beforeEach(function() { log = ''; $exceptionHandler = jasmine.createSpy('$exceptionHandler'); root.$service = function(name) { return {$exceptionHandler:$exceptionHandler}[name]; }; - root.$watch(function(){ log += '$digest;'; }); + root.$watch(function() { log += '$digest;'; }); root.$digest(); log = ''; }); - it('should execute and return value and update', function(){ + it('should execute and return value and update', function() { root.name = 'abc'; expect(root.$apply(function(scope){ return scope.name; @@ -388,12 +398,207 @@ describe('Scope', function(){ }); - it('should catch exception and update', function(){ + it('should catch exception and update', function() { var error = new Error('MyError'); - root.$apply(function(){ throw error; }); + root.$apply(function() { throw error; }); expect(log).toEqual('$digest;'); expect($exceptionHandler).wasCalledWith(error); }); }); }); + + + describe('events', function() { + + describe('$emit', function() { + var log, child, grandChild, greatGrandChild; + + function logger(event) { + log += event.currentScope.id + '>'; + } + + beforeEach(function() { + log = ''; + child = root.$new(); + grandChild = child.$new(); + greatGrandChild = grandChild.$new(); + + root.id = 0; + child.id = 1; + grandChild.id = 2; + greatGrandChild.id = 3; + + root.$on('myEvent', logger); + child.$on('myEvent', logger); + grandChild.$on('myEvent', logger); + greatGrandChild.$on('myEvent', logger); + }); + + + it('should bubble event up to the root scope', function() { + grandChild.$emit('myEvent'); + expect(log).toEqual('2>1>0>'); + }); + + + it('should dispatch exceptions to the $exceptionHandler', function() { + child.$on('myEvent', function() { throw 'bubbleException'; }); + grandChild.$emit('myEvent'); + expect(log).toEqual('2>1>0>'); + expect(mockHandler.errors).toEqual(['bubbleException']); + }); + + + it('should allow cancelation of event propagation', function() { + child.$on('myEvent', function(event){ event.cancel(); }); + grandChild.$emit('myEvent'); + expect(log).toEqual('2>1>'); + }); + + + it('should remove event listener', function() { + function eventFn(){ + log += 'abc;'; + } + + child.$on('abc', eventFn); + child.$emit('abc'); + expect(log).toEqual('abc;'); + log = ''; + child.$removeListener('abc', eventFn); + child.$emit('abc'); + expect(log).toEqual(''); + }); + + + it('should forward method arguments', function() { + child.$on('abc', function(event, arg1, arg2){ + expect(event.name).toBe('abc'); + expect(arg1).toBe('arg1'); + expect(arg2).toBe('arg2'); + }); + child.$emit('abc', 'arg1', 'arg2'); + }); + + describe('event object', function() { + it('should have methods/properties', function() { + var event; + child.$on('myEvent', function(e){ + expect(e.targetScope).toBe(grandChild); + expect(e.currentScope).toBe(child); + expect(e.name).toBe('myEvent'); + event = e; + }); + grandChild.$emit('myEvent'); + expect(event).toBeDefined(); + }); + }); + }); + + + describe('$broadcast', function() { + describe('event propagation', function() { + var log, child1, child2, child3, grandChild11, grandChild21, grandChild22, + greatGrandChild211; + + function logger(event) { + log += event.currentScope.id + '>'; + } + + beforeEach(function() { + log = ''; + child1 = root.$new(); + child2 = root.$new(); + child3 = root.$new(); + grandChild11 = child1.$new(); + grandChild21 = child2.$new(); + grandChild22 = child2.$new(); + greatGrandChild211 = grandChild21.$new(); + + root.id = 0; + child1.id = 1; + child2.id = 2; + child3.id = 3; + grandChild11.id = 11; + grandChild21.id = 21; + grandChild22.id = 22; + greatGrandChild211.id = 211; + + root.$on('myEvent', logger); + child1.$on('myEvent', logger); + child2.$on('myEvent', logger); + child3.$on('myEvent', logger); + grandChild11.$on('myEvent', logger); + grandChild21.$on('myEvent', logger); + grandChild22.$on('myEvent', logger); + greatGrandChild211.$on('myEvent', logger); + + // R + // / | \ + // 1 2 3 + // / / \ + // 11 21 22 + // | + // 211 + }); + + + it('should broadcast an event from the root scope', function() { + root.$broadcast('myEvent'); + expect(log).toBe('0>1>11>2>21>211>22>3>'); + }); + + + it('should broadcast an event from a child scope', function() { + child2.$broadcast('myEvent'); + expect(log).toBe('2>21>211>22>'); + }); + + + it('should broadcast an event from a leaf scope', function() { + grandChild22.$broadcast('myEvent'); + expect(log).toBe('22>'); + }); + + + it('should not not fire any listeners for other events', function() { + root.$broadcast('fooEvent'); + expect(log).toBe(''); + }); + }); + + + describe('listener', function() { + it('should receive event object', function() { + var scope = angular.scope(), + child = scope.$new(), + event; + + child.$on('fooEvent', function(e) { + event = e; + }); + scope.$broadcast('fooEvent'); + + expect(event.name).toBe('fooEvent'); + expect(event.targetScope).toBe(scope); + expect(event.currentScope).toBe(child); + }); + + + it('should support passing messages as varargs', function() { + var scope = angular.scope(), + child = scope.$new(), + args; + + child.$on('fooEvent', function() { + args = arguments; + }); + scope.$broadcast('fooEvent', 'do', 're', 'me', 'fa'); + + expect(args.length).toBe(5); + expect([].splice.call(args, 1)).toEqual(['do', 're', 'me', 'fa']); + }); + }); + }); + }); }); |
