import {describe, it, iit, xit, expect, beforeEach, afterEach} from 'angular2/test_lib'; import {IterableChanges} from 'angular2/src/change_detection/pipes/iterable_changes'; import {NumberWrapper} from 'angular2/src/facade/lang'; import {ListWrapper, MapWrapper} from 'angular2/src/facade/collection'; import {TestIterable} from '../iterable'; import {iterableChangesAsString} from '../util'; // todo(vicb): UnmodifiableListView / frozen object when implemented export function main() { describe('collection_changes', function() { describe('CollectionChanges', function() { var changes; beforeEach(() => { changes = new IterableChanges(); }); afterEach(() => { changes = null; }); it('should support list and iterables', () => { expect(IterableChanges.supportsObj([])).toBeTruthy(); expect(IterableChanges.supportsObj(new TestIterable())).toBeTruthy(); expect(IterableChanges.supportsObj(new Map())).toBeFalsy(); expect(IterableChanges.supportsObj(null)).toBeFalsy(); }); it('should support iterables', () => { let l = new TestIterable(); changes.check(l); expect(changes.toString()).toEqual(iterableChangesAsString({collection: []})); l.list = [1]; changes.check(l); expect(changes.toString()) .toEqual( iterableChangesAsString({collection: ['1[null->0]'], additions: ['1[null->0]']})); l.list = [2, 1]; changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['2[null->0]', '1[0->1]'], previous: ['1[0->1]'], additions: ['2[null->0]'], moves: ['1[0->1]'] })); }); it('should detect additions', () => { let l = []; changes.check(l); expect(changes.toString()).toEqual(iterableChangesAsString({collection: []})); l.push('a'); changes.check(l); expect(changes.toString()) .toEqual( iterableChangesAsString({collection: ['a[null->0]'], additions: ['a[null->0]']})); l.push('b'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString( {collection: ['a', 'b[null->1]'], previous: ['a'], additions: ['b[null->1]']})); }); it('should support changing the reference', () => { let l = [0]; changes.check(l); l = [1, 0]; changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['1[null->0]', '0[0->1]'], previous: ['0[0->1]'], additions: ['1[null->0]'], moves: ['0[0->1]'] })); l = [2, 1, 0]; changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['2[null->0]', '1[0->1]', '0[1->2]'], previous: ['1[0->1]', '0[1->2]'], additions: ['2[null->0]'], moves: ['1[0->1]', '0[1->2]'] })); }); it('should handle swapping element', () => { let l = [1, 2]; changes.check(l); ListWrapper.clear(l); l.push(2); l.push(1); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['2[1->0]', '1[0->1]'], previous: ['1[0->1]', '2[1->0]'], moves: ['2[1->0]', '1[0->1]'] })); }); it('should handle swapping element', () => { let l = ['a', 'b', 'c']; changes.check(l); ListWrapper.removeAt(l, 1); ListWrapper.insert(l, 0, 'b'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['b[1->0]', 'a[0->1]', 'c'], previous: ['a[0->1]', 'b[1->0]', 'c'], moves: ['b[1->0]', 'a[0->1]'] })); ListWrapper.removeAt(l, 1); l.push('a'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['b', 'c[2->1]', 'a[1->2]'], previous: ['b', 'a[1->2]', 'c[2->1]'], moves: ['c[2->1]', 'a[1->2]'] })); }); it('should detect changes in list', () => { let l = []; changes.check(l); l.push('a'); changes.check(l); expect(changes.toString()) .toEqual( iterableChangesAsString({collection: ['a[null->0]'], additions: ['a[null->0]']})); l.push('b'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString( {collection: ['a', 'b[null->1]'], previous: ['a'], additions: ['b[null->1]']})); l.push('c'); l.push('d'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['a', 'b', 'c[null->2]', 'd[null->3]'], previous: ['a', 'b'], additions: ['c[null->2]', 'd[null->3]'] })); ListWrapper.removeAt(l, 2); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['a', 'b', 'd[3->2]'], previous: ['a', 'b', 'c[2->null]', 'd[3->2]'], moves: ['d[3->2]'], removals: ['c[2->null]'] })); ListWrapper.clear(l); l.push('d'); l.push('c'); l.push('b'); l.push('a'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['d[2->0]', 'c[null->1]', 'b[1->2]', 'a[0->3]'], previous: ['a[0->3]', 'b[1->2]', 'd[2->0]'], additions: ['c[null->1]'], moves: ['d[2->0]', 'b[1->2]', 'a[0->3]'] })); }); it('should test string by value rather than by reference (Dart)', () => { let l = ['a', 'boo']; changes.check(l); var b = 'b'; var oo = 'oo'; ListWrapper.set(l, 1, b + oo); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({collection: ['a', 'boo'], previous: ['a', 'boo']})); }); it('should ignore [NaN] != [NaN] (JS)', () => { let l = [NumberWrapper.NaN]; changes.check(l); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString( {collection: [NumberWrapper.NaN], previous: [NumberWrapper.NaN]})); }); it('should detect [NaN] moves', () => { let l = [NumberWrapper.NaN, NumberWrapper.NaN]; changes.check(l); ListWrapper.insert(l, 0, 'foo'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['foo[null->0]', 'NaN[0->1]', 'NaN[1->2]'], previous: ['NaN[0->1]', 'NaN[1->2]'], additions: ['foo[null->0]'], moves: ['NaN[0->1]', 'NaN[1->2]'] })); }); it('should remove and add same item', () => { let l = ['a', 'b', 'c']; changes.check(l); ListWrapper.removeAt(l, 1); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['a', 'c[2->1]'], previous: ['a', 'b[1->null]', 'c[2->1]'], moves: ['c[2->1]'], removals: ['b[1->null]'] })); ListWrapper.insert(l, 1, 'b'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['a', 'b[null->1]', 'c[1->2]'], previous: ['a', 'c[1->2]'], additions: ['b[null->1]'], moves: ['c[1->2]'] })); }); it('should support duplicates', () => { let l = ['a', 'a', 'a', 'b', 'b']; changes.check(l); ListWrapper.removeAt(l, 0); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['a', 'a', 'b[3->2]', 'b[4->3]'], previous: ['a', 'a', 'a[2->null]', 'b[3->2]', 'b[4->3]'], moves: ['b[3->2]', 'b[4->3]'], removals: ['a[2->null]'] })); }); it('should support insertions/moves', () => { let l = ['a', 'a', 'b', 'b']; changes.check(l); ListWrapper.insert(l, 0, 'b'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['b[2->0]', 'a[0->1]', 'a[1->2]', 'b', 'b[null->4]'], previous: ['a[0->1]', 'a[1->2]', 'b[2->0]', 'b'], additions: ['b[null->4]'], moves: ['b[2->0]', 'a[0->1]', 'a[1->2]'] })); }); it('should not report unnecessary moves', () => { let l = ['a', 'b', 'c']; changes.check(l); ListWrapper.clear(l); l.push('b'); l.push('a'); l.push('c'); changes.check(l); expect(changes.toString()) .toEqual(iterableChangesAsString({ collection: ['b[1->0]', 'a[0->1]', 'c'], previous: ['a[0->1]', 'b[1->0]', 'c'], moves: ['b[1->0]', 'a[0->1]'] })); }); }); }); }