You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1979 lines
74 KiB

/* globals document */
var toStr = Object.prototype.toString;
var canDistinguishSparseFromUndefined = 0 in [undefined]; // IE 6 - 8 have a bug where this returns false.
var ifHasDenseUndefinedsIt = canDistinguishSparseFromUndefined ? it : xit;
var undefinedIfNoSparseBug = canDistinguishSparseFromUndefined ? undefined : { valueOf: function () { return 0; } };
var hasStrictMode = (function () {
'use strict';
return !this;
}());
var ifHasStrictIt = hasStrictMode ? it : xit;
describe('Array', function () {
var testSubject;
beforeEach(function () {
testSubject = [2, 3, undefinedIfNoSparseBug, true, 'hej', null, false, 0];
delete testSubject[1];
});
var createArrayLikeFromArray = function createArrayLikeFromArray(arr) {
var o = {};
Array.prototype.forEach.call(arr, function (e, i) {
o[i] = e;
});
o.length = arr.length;
return o;
};
describe('#forEach()', function () {
var expected, actual;
beforeEach(function () {
expected = { 0: 2, 2: undefinedIfNoSparseBug, 3: true, 4: 'hej', 5: null, 6: false, 7: 0 };
actual = {};
});
it('should pass the right parameters', function () {
var callback = jasmine.createSpy('callback');
var array = ['1'];
array.forEach(callback);
expect(callback).toHaveBeenCalledWith('1', 0, array);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.forEach(function (a) {
i += 1;
arr.push(a + 3);
});
expect(arr).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should set the right context when given none', function () {
var context;
[1].forEach(function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should iterate all', function () {
testSubject.forEach(function (obj, index) {
actual[index] = obj;
});
expect(actual).toExactlyMatch(expected);
});
it('should iterate all using a context', function () {
var o = { a: actual };
testSubject.forEach(function (obj, index) {
this.a[index] = obj;
}, o);
expect(actual).toExactlyMatch(expected);
});
it('should iterate all in an array-like object', function () {
var ts = createArrayLikeFromArray(testSubject);
Array.prototype.forEach.call(ts, function (obj, index) {
actual[index] = obj;
});
expect(actual).toExactlyMatch(expected);
});
it('should iterate all in an array-like object using a context', function () {
var ts = createArrayLikeFromArray(testSubject);
var o = { a: actual };
Array.prototype.forEach.call(ts, function (obj, index) {
this.a[index] = obj;
}, o);
expect(actual).toExactlyMatch(expected);
});
describe('strings', function () {
var str = 'Hello, World!';
it('should iterate all in a string', function () {
actual = [];
Array.prototype.forEach.call(str, function (item, index) {
actual[index] = item;
});
expect(actual).toExactlyMatch(str.split(''));
});
it('should iterate all in a string using a context', function () {
actual = [];
var o = { a: actual };
Array.prototype.forEach.call(str, function (item, index) {
this.a[index] = item;
}, o);
expect(actual).toExactlyMatch(str.split(''));
});
});
it('should have a boxed object as list argument of callback', function () {
var listArg;
Array.prototype.forEach.call('foo', function (item, index, list) {
listArg = list;
});
expect(typeof listArg).toBe('object');
expect(toStr.call(listArg)).toBe('[object String]');
});
ifHasStrictIt('does not autobox the content in strict mode', function () {
var context;
[1].forEach(function () {
'use strict';
context = this;
}, 'x');
expect(typeof context).toBe('string');
});
});
describe('#some()', function () {
var actual, expected, numberOfRuns;
beforeEach(function () {
expected = { 0: 2, 2: undefinedIfNoSparseBug, 3: true };
actual = {};
numberOfRuns = 0;
});
it('should pass the correct values along to the callback', function () {
var callback = jasmine.createSpy('callback');
var array = ['1'];
array.some(callback);
expect(callback).toHaveBeenCalledWith('1', 0, array);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.some(function (a) {
i += 1;
arr.push(a + 3);
return i > 3;
});
expect(arr).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should set the right context when given none', function () {
/* eslint-disable array-callback-return */
var context;
[1].some(function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should return false if it runs to the end', function () {
/* eslint-disable array-callback-return */
actual = testSubject.some(function () {});
expect(actual).toBeFalsy();
});
it('should return true if it is stopped somewhere', function () {
actual = testSubject.some(function () { return true; });
expect(actual).toBeTruthy();
});
it('should return false if there are no elements', function () {
actual = [].some(function () { return true; });
expect(actual).toBeFalsy();
});
it('should stop after 3 elements', function () {
testSubject.some(function (obj, index) {
actual[index] = obj;
numberOfRuns += 1;
return numberOfRuns === 3;
});
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements using a context', function () {
var o = { a: actual };
testSubject.some(function (obj, index) {
this.a[index] = obj;
numberOfRuns += 1;
return numberOfRuns === 3;
}, o);
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements in an array-like object', function () {
var ts = createArrayLikeFromArray(testSubject);
Array.prototype.some.call(ts, function (obj, index) {
actual[index] = obj;
numberOfRuns += 1;
return numberOfRuns === 3;
});
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements in an array-like object using a context', function () {
var ts = createArrayLikeFromArray(testSubject);
var o = { a: actual };
Array.prototype.some.call(ts, function (obj, index) {
this.a[index] = obj;
numberOfRuns += 1;
return numberOfRuns === 3;
}, o);
expect(actual).toExactlyMatch(expected);
});
it('should have a boxed object as list argument of callback', function () {
var listArg;
Array.prototype.some.call('foo', function (item, index, list) {
listArg = list;
});
expect(typeof listArg).toBe('object');
expect(toStr.call(listArg)).toBe('[object String]');
});
});
describe('#every()', function () {
var actual, expected, numberOfRuns;
beforeEach(function () {
expected = { 0: 2, 2: undefinedIfNoSparseBug, 3: true };
actual = {};
numberOfRuns = 0;
});
it('should pass the correct values along to the callback', function () {
var callback = jasmine.createSpy('callback');
var array = ['1'];
array.every(callback);
expect(callback).toHaveBeenCalledWith('1', 0, array);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.every(function (a) {
i += 1;
arr.push(a + 3);
return i <= 3;
});
expect(arr).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should set the right context when given none', function () {
/* eslint-disable array-callback-return */
var context;
[1].every(function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should return true if the array is empty', function () {
actual = [].every(function () { return true; });
expect(actual).toBeTruthy();
actual = [].every(function () { return false; });
expect(actual).toBeTruthy();
});
it('should return true if it runs to the end', function () {
actual = [1, 2, 3].every(function () { return true; });
expect(actual).toBeTruthy();
});
it('should return false if it is stopped before the end', function () {
actual = [1, 2, 3].every(function () { return false; });
expect(actual).toBeFalsy();
});
it('should return after 3 elements', function () {
testSubject.every(function (obj, index) {
actual[index] = obj;
numberOfRuns += 1;
return numberOfRuns !== 3;
});
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements using a context', function () {
var o = { a: actual };
testSubject.every(function (obj, index) {
this.a[index] = obj;
numberOfRuns += 1;
return numberOfRuns !== 3;
}, o);
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements in an array-like object', function () {
var ts = createArrayLikeFromArray(testSubject);
Array.prototype.every.call(ts, function (obj, index) {
actual[index] = obj;
numberOfRuns += 1;
return numberOfRuns !== 3;
});
expect(actual).toExactlyMatch(expected);
});
it('should stop after 3 elements in an array-like object using a context', function () {
var ts = createArrayLikeFromArray(testSubject);
var o = { a: actual };
Array.prototype.every.call(ts, function (obj, index) {
this.a[index] = obj;
numberOfRuns += 1;
return numberOfRuns !== 3;
}, o);
expect(actual).toExactlyMatch(expected);
});
it('should have a boxed object as list argument of callback', function () {
var listArg;
Array.prototype.every.call('foo', function (item, index, list) {
listArg = list;
});
expect(typeof listArg).toBe('object');
expect(toStr.call(listArg)).toBe('[object String]');
});
});
describe('#indexOf()', function () {
'use strict';
var actual, expected;
beforeEach(function () {
testSubject = [2, 3, undefinedIfNoSparseBug, true, 'hej', null, 2, false, 0];
delete testSubject[1];
});
it('should find the element', function () {
expected = 4;
actual = testSubject.indexOf('hej');
expect(actual).toBe(expected);
});
it('should not find the element', function () {
expected = -1;
actual = testSubject.indexOf('mus');
expect(actual).toBe(expected);
});
ifHasDenseUndefinedsIt('should find undefined as well', function () {
expected = -1;
actual = testSubject.indexOf(undefined);
expect(actual).not.toBe(expected);
});
ifHasDenseUndefinedsIt('should skip unset indexes', function () {
expected = 2;
actual = testSubject.indexOf(undefined);
expect(actual).toBe(expected);
});
it('should use a strict test', function () {
actual = testSubject.indexOf(null);
expect(actual).toBe(5);
actual = testSubject.indexOf('2');
expect(actual).toBe(-1);
});
it('should skip the first if fromIndex is set', function () {
expect(testSubject.indexOf(2, 2)).toBe(6);
expect(testSubject.indexOf(2, 0)).toBe(0);
expect(testSubject.indexOf(2, 6)).toBe(6);
});
it('should work with negative fromIndex', function () {
expect(testSubject.indexOf(2, -3)).toBe(6);
expect(testSubject.indexOf(2, -9)).toBe(0);
});
it('should work with fromIndex being greater than the length', function () {
expect(testSubject.indexOf(0, 20)).toBe(-1);
});
it('should work with fromIndex being negative and greater than the length', function () {
expect(testSubject.indexOf('hej', -20)).toBe(4);
});
describe('Array-like', function ArrayLike() {
var indexOf = Array.prototype.indexOf;
var testAL;
beforeEach(function beforeEach() {
testAL = {};
testSubject = [2, 3, undefinedIfNoSparseBug, true, 'hej', null, 2, false, 0];
testSubject.forEach(function (o, i) {
testAL[i] = o;
});
testAL.length = testSubject.length;
});
it('should find the element (array-like)', function () {
expected = 4;
actual = indexOf.call(testAL, 'hej');
expect(actual).toBe(expected);
});
it('should not find the element (array-like)', function () {
expected = -1;
actual = indexOf.call(testAL, 'mus');
expect(actual).toBe(expected);
});
ifHasDenseUndefinedsIt('should find undefined as well (array-like)', function () {
expected = -1;
actual = indexOf.call(testAL, undefined);
expect(actual).not.toBe(expected);
});
ifHasDenseUndefinedsIt('should skip unset indexes (array-like)', function () {
expected = 2;
actual = indexOf.call(testAL, undefined);
expect(actual).toBe(expected);
});
it('should use a strict test (array-like)', function () {
actual = Array.prototype.indexOf.call(testAL, null);
expect(actual).toBe(5);
actual = Array.prototype.indexOf.call(testAL, '2');
expect(actual).toBe(-1);
});
it('should skip the first if fromIndex is set (array-like)', function () {
expect(indexOf.call(testAL, 2, 2)).toBe(6);
expect(indexOf.call(testAL, 2, 0)).toBe(0);
expect(indexOf.call(testAL, 2, 6)).toBe(6);
});
it('should work with negative fromIndex (array-like)', function () {
expect(indexOf.call(testAL, 2, -3)).toBe(6);
expect(indexOf.call(testAL, 2, -9)).toBe(0);
});
it('should work with fromIndex being greater than the length (array-like)', function () {
expect(indexOf.call(testAL, 0, 20)).toBe(-1);
});
it('should work with fromIndex being negative and greater than the length (array-like)', function () {
expect(indexOf.call(testAL, 'hej', -20)).toBe(4);
});
});
});
describe('#lastIndexOf()', function () {
'use strict';
var actual, expected;
beforeEach(function () {
testSubject = [2, 3, undefinedIfNoSparseBug, true, 'hej', null, 2, 3, false, 0];
delete testSubject[1];
delete testSubject[7];
});
describe('Array', function () {
it('should find the element', function () {
expected = 4;
actual = testSubject.lastIndexOf('hej');
expect(actual).toBe(expected);
});
it('should not find the element', function () {
expected = -1;
actual = testSubject.lastIndexOf('mus');
expect(actual).toBe(expected);
});
ifHasDenseUndefinedsIt('should find undefined as well', function () {
expected = -1;
actual = testSubject.lastIndexOf(undefined);
expect(actual).not.toBe(expected);
});
ifHasDenseUndefinedsIt('should skip unset indexes', function () {
expected = 2;
actual = testSubject.lastIndexOf(undefined);
expect(actual).toBe(expected);
});
it('should use a strict test', function () {
actual = testSubject.lastIndexOf(null);
expect(actual).toBe(5);
actual = testSubject.lastIndexOf('2');
expect(actual).toBe(-1);
});
it('should skip the first if fromIndex is set', function () {
expect(testSubject.lastIndexOf(2, 2)).toBe(0);
expect(testSubject.lastIndexOf(2, 0)).toBe(0);
expect(testSubject.lastIndexOf(2, 6)).toBe(6);
});
it('should work with negative fromIndex', function () {
expect(testSubject.lastIndexOf(2, -3)).toBe(6);
expect(testSubject.lastIndexOf(2, -9)).toBe(0);
});
it('should work with fromIndex being greater than the length', function () {
expect(testSubject.lastIndexOf(2, 20)).toBe(6);
});
it('should work with fromIndex being negative and greater than the length', function () {
expect(testSubject.lastIndexOf(2, -20)).toBe(-1);
});
});
describe('Array like', function () {
var lastIndexOf = Array.prototype.lastIndexOf;
var testAL;
beforeEach(function () {
testAL = {};
testSubject.forEach(function (o, i) {
testAL[i] = o;
});
testAL.length = testSubject.length;
});
it('should find the element (array-like)', function () {
expected = 4;
actual = lastIndexOf.call(testAL, 'hej');
expect(actual).toBe(expected);
});
it('should not find the element (array-like)', function () {
expected = -1;
actual = lastIndexOf.call(testAL, 'mus');
expect(actual).toBe(expected);
});
ifHasDenseUndefinedsIt('should find undefined as well (array-like)', function () {
expected = -1;
actual = lastIndexOf.call(testAL, undefined);
expect(actual).not.toBe(expected);
});
ifHasDenseUndefinedsIt('should skip unset indexes (array-like)', function () {
expected = 2;
actual = lastIndexOf.call(testAL, undefined);
expect(actual).toBe(expected);
});
it('should use a strict test (array-like)', function () {
actual = lastIndexOf.call(testAL, null);
expect(actual).toBe(5);
actual = lastIndexOf.call(testAL, '2');
expect(actual).toBe(-1);
});
it('should skip the first if fromIndex is set', function () {
expect(lastIndexOf.call(testAL, 2, 2)).toBe(0);
expect(lastIndexOf.call(testAL, 2, 0)).toBe(0);
expect(lastIndexOf.call(testAL, 2, 6)).toBe(6);
});
it('should work with negative fromIndex', function () {
expect(lastIndexOf.call(testAL, 2, -3)).toBe(6);
expect(lastIndexOf.call(testAL, 2, -9)).toBe(0);
});
it('should work with fromIndex being greater than the length', function () {
expect(lastIndexOf.call(testAL, 2, 20)).toBe(6);
});
it('should work with fromIndex being negative and greater than the length', function () {
expect(lastIndexOf.call(testAL, 2, -20)).toBe(-1);
});
});
});
describe('#filter()', function () {
var filteredArray;
var callback = function callback(o, i) {
return i !== 3 && i !== 5;
};
beforeEach(function () {
testSubject = [2, 3, undefinedIfNoSparseBug, true, 'hej', 3, null, false, 0];
delete testSubject[1];
filteredArray = [2, undefinedIfNoSparseBug, 'hej', null, false, 0];
});
describe('Array object', function () {
it('should call the callback with the proper arguments', function () {
var predicate = jasmine.createSpy('predicate');
var arr = ['1'];
arr.filter(predicate);
expect(predicate).toHaveBeenCalledWith('1', 0, arr);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.filter(function (a) {
i += 1;
if (i <= 4) {
arr.push(a + 3);
}
return true;
});
expect(arr).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
ifHasDenseUndefinedsIt('should skip unset values', function () {
var passedValues = {};
testSubject = [1, 2, 3, 4];
delete testSubject[1];
testSubject.filter(function (o, i) {
passedValues[i] = o;
return true;
});
expect(passedValues).toExactlyMatch(testSubject);
});
it('should pass the right context to the filter', function () {
var passedValues = {};
testSubject = [1, 2, 3, 4];
delete testSubject[1];
testSubject.filter(function (o, i) {
this[i] = o;
return true;
}, passedValues);
expect(passedValues).toExactlyMatch(testSubject);
});
it('should set the right context when given none', function () {
/* eslint-disable array-callback-return */
var context;
[1].filter(function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should remove only the values for which the callback returns false', function () {
var result = testSubject.filter(callback);
expect(result).toExactlyMatch(filteredArray);
});
it('should leave the original array untouched', function () {
var copy = testSubject.slice();
testSubject.filter(callback);
expect(testSubject).toExactlyMatch(copy);
});
it('should not be affected by same-index mutation', function () {
var results = [1, 2, 3].filter(function (value, index, array) {
array[index] = 'a'; // eslint-disable-line no-param-reassign
return true;
});
expect(results).toEqual([1, 2, 3]);
});
});
describe('Array like', function () {
beforeEach(function () {
testSubject = createArrayLikeFromArray(testSubject);
});
it('should call the predicate with the proper arguments', function () {
var predicate = jasmine.createSpy('predicate');
var arr = createArrayLikeFromArray(['1']);
Array.prototype.filter.call(arr, predicate);
expect(predicate).toHaveBeenCalledWith('1', 0, arr);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = createArrayLikeFromArray([1, 2, 3]);
var i = 0;
Array.prototype.filter.call(arr, function (a) {
i += 1;
if (i <= 4) {
arr[i + 2] = a + 3;
arr.length += 1;
}
return true;
});
expect(Array.prototype.slice.call(arr)).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should skip non-set values', function () {
var passedValues = createArrayLikeFromArray([]);
testSubject = createArrayLikeFromArray([1, 2, 3, 4]);
delete testSubject[1];
Array.prototype.filter.call(testSubject, function (o, i) {
passedValues[i] = o;
passedValues.length = i + 1;
return true;
});
expect(passedValues).toEqual(testSubject);
});
it('should set the right context when given none', function () {
var context;
Array.prototype.filter.call(createArrayLikeFromArray([1]), function () { context = this; }, undefined);
expect(context).toBe(function () { return this; }.call());
});
it('should pass the right context to the filter', function () {
var passedValues = {};
testSubject = createArrayLikeFromArray([1, 2, 3, 4]);
delete testSubject[1];
Array.prototype.filter.call(testSubject, function (o, i) {
this[i] = o;
this.length = i + 1;
return true;
}, passedValues);
expect(passedValues).toEqual(testSubject);
});
it('should remove only the values for which the callback returns false', function () {
var result = Array.prototype.filter.call(testSubject, callback);
expect(result).toExactlyMatch(filteredArray);
});
it('should leave the original array untouched', function () {
var copy = createArrayLikeFromArray(testSubject);
Array.prototype.filter.call(testSubject, callback);
expect(testSubject).toExactlyMatch(copy);
});
});
it('should have a boxed object as list argument of callback', function () {
var actual;
Array.prototype.filter.call('foo', function (item, index, list) {
actual = list;
});
expect(typeof actual).toBe('object');
expect(toStr.call(actual)).toBe('[object String]');
});
});
describe('#map()', function () {
var callback;
beforeEach(function () {
var i = 0;
callback = function () {
return i++;
};
});
describe('Array object', function () {
it('should call mapper with the right parameters', function () {
var mapper = jasmine.createSpy('mapper');
var array = [1];
array.map(mapper);
expect(mapper).toHaveBeenCalledWith(1, 0, array);
});
it('should set the context correctly', function () {
var context = {};
testSubject.map(function (o, i) {
this[i] = o;
}, context);
expect(context).toExactlyMatch(testSubject);
});
it('should set the right context when given none', function () {
var context;
[1].map(function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should not change the array it is called on', function () {
var copy = testSubject.slice();
testSubject.map(callback);
expect(testSubject).toExactlyMatch(copy);
});
it('should only run for the number of objects in the array when it started', function () {
var arr = [1, 2, 3];
var i = 0;
arr.map(function (o) {
arr.push(o + 3);
i += 1;
return o;
});
expect(arr).toExactlyMatch([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should properly translate the values as according to the callback', function () {
var result = testSubject.map(callback);
var expected = [0, 0, 1, 2, 3, 4, 5, 6];
delete expected[1];
expect(result).toExactlyMatch(expected);
});
it('should skip non-existing values', function () {
var array = [1, 2, 3, 4];
var i = 0;
delete array[2];
array.map(function () {
i += 1;
});
expect(i).toBe(3);
});
});
describe('Array-like', function () {
beforeEach(function () {
testSubject = createArrayLikeFromArray(testSubject);
});
it('should call mapper with the right parameters', function () {
var mapper = jasmine.createSpy('mapper');
var array = createArrayLikeFromArray([1]);
Array.prototype.map.call(array, mapper);
expect(mapper).toHaveBeenCalledWith(1, 0, array);
});
it('should set the context correctly', function () {
var context = {};
Array.prototype.map.call(testSubject, function (o, i) {
this[i] = o;
this.length = i + 1;
}, context);
expect(context).toEqual(testSubject);
});
it('should set the right context when given none', function () {
var context;
Array.prototype.map.call(createArrayLikeFromArray([1]), function () { context = this; });
expect(context).toBe(function () { return this; }.call());
});
it('should not change the array it is called on', function () {
var copy = createArrayLikeFromArray(testSubject);
Array.prototype.map.call(testSubject, callback);
expect(testSubject).toExactlyMatch(copy);
});
it('should only run for the number of objects in the array when it started', function () {
var arr = createArrayLikeFromArray([1, 2, 3]);
var i = 0;
Array.prototype.map.call(arr, function (o) {
Array.prototype.push.call(arr, o + 3);
i += 1;
return o;
});
expect(Array.prototype.slice.call(arr)).toEqual([1, 2, 3, 4, 5, 6]);
expect(i).toBe(3);
});
it('should properly translate the values as according to the callback', function () {
var result = Array.prototype.map.call(testSubject, callback);
var expected = [0, 0, 1, 2, 3, 4, 5, 6];
delete expected[1];
expect(result).toExactlyMatch(expected);
});
it('should skip non-existing values', function () {
var array = createArrayLikeFromArray([1, 2, 3, 4]);
var i = 0;
delete array[2];
Array.prototype.map.call(array, function () {
i += 1;
});
expect(i).toBe(3);
});
});
it('should have a boxed object as list argument of callback', function () {
var actual;
Array.prototype.map.call('foo', function (item, index, list) {
actual = list;
});
expect(typeof actual).toBe('object');
expect(toStr.call(actual)).toBe('[object String]');
});
});
describe('#reduce()', function () {
beforeEach(function () {
testSubject = [1, 2, 3];
});
describe('Array', function () {
it('should pass the correct arguments to the callback', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduce(spy);
expect(spy.calls[0].args).toExactlyMatch([1, 2, 1, testSubject]);
});
it('should start with the right initialValue', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduce(spy, 0);
expect(spy.calls[0].args).toExactlyMatch([0, 1, 0, testSubject]);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.reduce(function (a, b) {
i += 1;
if (i <= 4) {
arr.push(a + 3);
}
return b;
});
expect(arr).toEqual([1, 2, 3, 4, 5]);
expect(i).toBe(2);
});
it('should work as expected for empty arrays', function () {
var spy = jasmine.createSpy();
expect(function () {
[].reduce(spy);
}).toThrow();
expect(spy).not.toHaveBeenCalled();
});
it('should throw correctly if no callback is given', function () {
expect(function () {
testSubject.reduce();
}).toThrow();
});
it('should return the expected result', function () {
expect(testSubject.reduce(function (a, b) {
return String(a || '') + String(b || '');
})).toBe(testSubject.join(''));
});
it('should not directly affect the passed array', function () {
var copy = testSubject.slice();
testSubject.reduce(function (a, b) {
return a + b;
});
expect(testSubject).toEqual(copy);
});
it('should skip non-set values', function () {
delete testSubject[1];
var visited = {};
testSubject.reduce(function (a, b) {
if (a) { visited[a] = true; }
if (b) { visited[b] = true; }
return 0;
});
expect(visited).toEqual({ 1: true, 3: true });
});
it('should have the right length', function () {
expect(testSubject.reduce.length).toBe(1);
});
});
describe('Array-like objects', function () {
beforeEach(function () {
testSubject = createArrayLikeFromArray(testSubject);
testSubject.reduce = Array.prototype.reduce;
});
it('should pass the correct arguments to the callback', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduce(spy);
expect(spy.calls[0].args).toExactlyMatch([1, 2, 1, testSubject]);
});
it('should start with the right initialValue', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduce(spy, 0);
expect(spy.calls[0].args).toExactlyMatch([0, 1, 0, testSubject]);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = createArrayLikeFromArray([1, 2, 3]);
var i = 0;
Array.prototype.reduce.call(arr, function (a, b) {
i += 1;
if (i <= 4) {
arr[i + 2] = a + 3;
}
return b;
});
expect(arr).toEqual({
0: 1,
1: 2,
2: 3,
3: 4,
4: 5,
length: 3
});
expect(i).toBe(2);
});
it('should work as expected for empty arrays', function () {
var spy = jasmine.createSpy();
expect(function () {
Array.prototype.reduce.call({ length: 0 }, spy);
}).toThrow();
expect(spy).not.toHaveBeenCalled();
});
it('should throw correctly if no callback is given', function () {
expect(function () {
testSubject.reduce();
}).toThrow();
});
it('should return the expected result', function () {
expect(testSubject.reduce(function (a, b) {
return String(a || '') + String(b || '');
})).toBe('123');
});
it('should not directly affect the passed array', function () {
var copy = createArrayLikeFromArray(testSubject);
testSubject.reduce(function (a, b) {
return a + b;
});
delete testSubject.reduce;
expect(testSubject).toEqual(copy);
});
it('should skip non-set values', function () {
delete testSubject[1];
var visited = {};
testSubject.reduce(function (a, b) {
if (a) { visited[a] = true; }
if (b) { visited[b] = true; }
return 0;
});
expect(visited).toEqual({ 1: true, 3: true });
});
it('should have the right length', function () {
expect(testSubject.reduce.length).toBe(1);
});
});
it('should have a boxed object as list argument of callback', function () {
var actual;
Array.prototype.reduce.call('foo', function (accumulator, item, index, list) {
actual = list;
});
expect(typeof actual).toBe('object');
expect(toStr.call(actual)).toBe('[object String]');
});
});
describe('#reduceRight()', function () {
beforeEach(function () {
testSubject = [1, 2, 3];
});
describe('Array', function () {
it('should pass the correct arguments to the callback', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduceRight(spy);
expect(spy.calls[0].args).toExactlyMatch([3, 2, 1, testSubject]);
});
it('should start with the right initialValue', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduceRight(spy, 0);
expect(spy.calls[0].args).toExactlyMatch([0, 3, 2, testSubject]);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = [1, 2, 3];
var i = 0;
arr.reduceRight(function (a, b) {
i += 1;
if (i <= 4) {
arr.push(a + 3);
}
return b;
});
expect(arr).toEqual([1, 2, 3, 6, 5]);
expect(i).toBe(2);
});
it('should work as expected for empty arrays', function () {
var spy = jasmine.createSpy();
expect(function () {
[].reduceRight(spy);
}).toThrow();
expect(spy).not.toHaveBeenCalled();
});
it('should work as expected for empty arrays with an initial value', function () {
var spy = jasmine.createSpy();
var result;
result = [].reduceRight(spy, '');
expect(spy).not.toHaveBeenCalled();
expect(result).toBe('');
});
it('should throw correctly if no callback is given', function () {
expect(function () {
testSubject.reduceRight();
}).toThrow();
});
it('should return the expected result', function () {
expect(testSubject.reduceRight(function (a, b) {
return String(a || '') + String(b || '');
})).toBe('321');
});
it('should not directly affect the passed array', function () {
var copy = testSubject.slice();
testSubject.reduceRight(function (a, b) {
return a + b;
});
expect(testSubject).toEqual(copy);
});
it('should skip non-set values', function () {
delete testSubject[1];
var visited = {};
testSubject.reduceRight(function (a, b) {
if (a) { visited[a] = true; }
if (b) { visited[b] = true; }
return 0;
});
expect(visited).toEqual({ 1: true, 3: true });
});
it('should have the right length', function () {
expect(testSubject.reduceRight.length).toBe(1);
});
});
describe('Array-like objects', function () {
beforeEach(function () {
testSubject = createArrayLikeFromArray(testSubject);
testSubject.reduceRight = Array.prototype.reduceRight;
});
it('should pass the correct arguments to the callback', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduceRight(spy);
expect(spy.calls[0].args).toExactlyMatch([3, 2, 1, testSubject]);
});
it('should start with the right initialValue', function () {
var spy = jasmine.createSpy().andReturn(0);
testSubject.reduceRight(spy, 0);
expect(spy.calls[0].args).toExactlyMatch([0, 3, 2, testSubject]);
});
it('should not affect elements added to the array after it has begun', function () {
var arr = createArrayLikeFromArray([1, 2, 3]);
var i = 0;
Array.prototype.reduceRight.call(arr, function (a, b) {
i += 1;
if (i <= 4) {
arr[i + 2] = a + 3;
}
return b;
});
expect(arr).toEqual({
0: 1,
1: 2,
2: 3,
3: 6,
4: 5,
length: 3 // does not get updated on property assignment
});
expect(i).toBe(2);
});
it('should work as expected for empty arrays', function () {
var spy = jasmine.createSpy();
expect(function () {
Array.prototype.reduceRight.call({ length: 0 }, spy);
}).toThrow();
expect(spy).not.toHaveBeenCalled();
});
it('should throw correctly if no callback is given', function () {
expect(function () {
testSubject.reduceRight();
}).toThrow();
});
it('should return the expected result', function () {
expect(testSubject.reduceRight(function (a, b) {
return String(a || '') + String(b || '');
})).toBe('321');
});
it('should not directly affect the passed array', function () {
var copy = createArrayLikeFromArray(testSubject);
testSubject.reduceRight(function (a, b) {
return a + b;
});
delete testSubject.reduceRight;
expect(testSubject).toEqual(copy);
});
it('should skip non-set values', function () {
delete testSubject[1];
var visited = {};
testSubject.reduceRight(function (a, b) {
if (a) { visited[a] = true; }
if (b) { visited[b] = true; }
return 0;
});
expect(visited).toEqual({ 1: true, 3: true });
});
it('should have the right length', function () {
expect(testSubject.reduceRight.length).toBe(1);
});
});
it('should have a boxed object as list argument of callback', function () {
var actual;
Array.prototype.reduceRight.call('foo', function (accumulator, item, index, list) {
actual = list;
});
expect(typeof actual).toBe('object');
expect(toStr.call(actual)).toBe('[object String]');
});
});
describe('.isArray()', function () {
it('should be true for Array', function () {
expect(Array.isArray([])).toBe(true);
});
it('should be false for primitives', function () {
var primitives = [
'foo',
true,
false,
42,
0,
-0,
NaN,
Infinity,
-Infinity
];
primitives.forEach(function (v) {
expect(Array.isArray(v)).toBe(false);
});
});
it('should fail for other objects', function () {
var objects = [
{},
/foo/,
arguments
];
if (Object.create) {
objects.push(Object.create(null));
}
objects.forEach(function (v) {
expect(Array.isArray(v)).toBe(false);
});
});
if (typeof document !== 'undefined') {
it('should be false for an HTML element', function () {
var el = document.getElementsByTagName('div');
expect(Array.isArray(el)).toBe(false);
});
}
});
describe('#shift()', function () {
it('works on arrays', function () {
var arr = [1, 2];
var result = arr.shift();
expect(result).toBe(1);
expect(arr.length).toBe(1);
expect(Object.prototype.hasOwnProperty.call(arr, 0)).toBe(true);
expect(Object.prototype.hasOwnProperty.call(arr, 1)).toBe(false);
expect(arr[0]).toBe(2);
expect(arr[1]).toBeUndefined();
});
it('is generic', function () {
var obj = { 0: 1, 1: 2, length: 2 };
var result = Array.prototype.shift.call(obj);
expect(result).toBe(1);
expect(obj.length).toBe(1);
expect(Object.prototype.hasOwnProperty.call(obj, 0)).toBe(true);
expect(Object.prototype.hasOwnProperty.call(obj, 1)).toBe(false);
expect(obj[0]).toBe(2);
expect(obj[1]).toBeUndefined();
});
});
describe('#unshift()', function () {
it('should return length', function () {
expect([].unshift(0)).toBe(1);
});
it('works on arrays', function () {
var arr = [1];
var result = arr.unshift(undefined);
expect(result).toBe(2);
expect(arr.length).toBe(2);
expect(Object.prototype.hasOwnProperty.call(arr, 0)).toBe(true);
expect(Object.prototype.hasOwnProperty.call(arr, 1)).toBe(true);
expect(arr[0]).toBeUndefined();
expect(arr[1]).toBe(1);
});
it('is generic', function () {
var obj = { 0: 1, length: 1 };
var result = Array.prototype.unshift.call(obj, undefined);
expect(result).toBe(2);
expect(obj.length).toBe(2);
expect(Object.prototype.hasOwnProperty.call(obj, 0)).toBe(true);
expect(Object.prototype.hasOwnProperty.call(obj, 1)).toBe(true);
expect(obj[0]).toBeUndefined();
expect(obj[1]).toBe(1);
});
});
describe('#splice()', function () {
var b = ['b'];
var a = [1, 'a', b];
var test;
var makeArray = function (l, givenPrefix) {
var prefix = givenPrefix || '';
var length = l;
var arr = [];
while (length--) {
arr.unshift(prefix + Array(length + 1).join(' ') + length);
}
return arr;
};
beforeEach(function () {
test = a.slice(0);
});
it('has the right length', function () {
expect(Array.prototype.splice.length).toBe(2);
});
/* This test is disabled, because ES6 normalizes actual
* browser behavior, contradicting ES5.
*/
xit('treats undefined deleteCount as 0', function () {
expect(test.splice(0).length).toBe(0);
expect(test.splice(0)).toEqual(test.splice(0, 0));
});
// ES6 introduced a proper default value
it('defaults deleteCount to length - start if there is only 1 argument', function () {
expect([0, 1, 2].splice(0).length).toBe(3);
expect([0, 1, 2].splice(1).length).toBe(2);
});
it('basic implementation test 1', function () {
expect(test.splice(0, 0)).toEqual([]);
});
it('basic implementation test 2', function () {
test.splice(0, 2);
expect(test).toEqual([b]);
});
it('should return right result 1', function () {
var array = [];
array.splice(0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20);
array.splice(1, 0, 'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26');
array.splice(5, 0, 'XXX');
expect(array).toEqual([1, 'F1', 'F2', 'F3', 'F4', 'XXX', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20]);
});
it('should return right result 2', function () {
var array = makeArray(6);
array.splice(array.length - 1, 1, '');
array.splice(0, 1, 1, 2, 3, 4);
array.splice(0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45);
array.splice(4, 0, '99999999999999');
expect(array).toEqual([1, 2, 3, 4, '99999999999999', 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 1, 2, 3, 4, ' 1', ' 2', ' 3', ' 4', '']);
});
it('should return right result 3', function () {
var array = [1, 2, 3];
array.splice(0, array.length);
array.splice(0, 1, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
array.splice(1, 1, 'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26');
array.splice(5, 1, 'YYY', 'XXX');
array.splice(0, 1);
array.splice(0, 2);
array.pop();
array.push.apply(array, makeArray(10, '-'));
array.splice(array.length - 2, 10);
array.splice();
array.splice(1, 1, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 1, 2, 3, 4, 5, 6, 7, 8, 9);
array.splice(1, 1, 'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 1, 23, 4, 5, 6, 7, 8);
array.splice(30, 10);
array.splice(30, 1);
array.splice(30, 0);
array.splice(2, 5, 1, 2, 3, 'P', 'LLL', 'CCC', 'YYY', 'XXX');
array.push(1, 2, 3, 4, 5, 6);
array.splice(1, 6, 1, 2, 3, 4, 5, 6, 7, 8, 9, 4, 5, 6, 7, 8, 9);
array.splice(3, 7);
array.unshift(7, 8, 9, 10, 11);
array.pop();
array.splice(5, 2);
array.pop();
array.unshift.apply(array, makeArray(8, '~'));
array.pop();
array.splice(3, 1, 'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 1, 23, 4, 5, 6, 7, 8);
array.splice(4, 5, 'P', 'LLL', 'CCC', 'YYY', 'XXX');
expect(array).toEqual(['~0', '~ 1', '~ 2', 'F1', 'P', 'LLL', 'CCC', 'YYY', 'XXX', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 1, 23, 4, 5, 6, 7, 8, '~ 4', '~ 5', '~ 6', '~ 7', 7, 8, 9, 10, 11, 2, 4, 5, 6, 7, 8, 9, 'CCC', 'YYY', 'XXX', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 1, 23, 4, 9, 10, 1, 2, 3, 4, 5, 6, 7, 8, 9, 'YYY', 'XXX', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12', 'F13', 'F14', 'F15', 'F16', 'F17', 'F18', 'F19', 'F20', 'F21', 'F22', 'F23', 'F24', 'F25', 'F26', 3, 4, 5, 6, 7, 8, 9, '-0', '- 1', '- 2', '- 3', '- 4', '- 5', '- 6', '- 7', 1, 2, 3]);
});
it('should do nothing if method called with no arguments', function () {
expect(test.splice()).toEqual([]);
expect(test).toEqual(a);
});
it('should set first argument to 0 if first argument is set but undefined', function () {
var test2 = test.slice(0);
expect(test.splice(void 0, 2)).toEqual(test2.splice(0, 2));
expect(test).toEqual(test2);
});
it('should work with objects - adding 1', function () {
var obj = {};
Array.prototype.splice.call(obj, 0, 0, 1, 2, 3);
expect(obj.length).toBe(3);
});
it('should work with objects - adding 2', function () {
var obj = { 0: 1, length: 1 };
Array.prototype.splice.call(obj, 1, 0, 2, 3);
expect(obj.length).toBe(3);
});
it('should work with objects - removing', function () {
var obj = { 0: 1, 1: 2, 2: 3, length: 3 };
Array.prototype.splice.call(obj, 0, 3);
expect(obj.length).toBe(0);
});
it('should work with objects - replacing', function () {
var obj = { 0: 99, length: 1 };
Array.prototype.splice.call(obj, 0, 1, 1, 2, 3);
expect(obj.length).toBe(3);
expect(obj[0]).toBe(1);
});
ifHasDenseUndefinedsIt('should not break on sparse arrays in Opera', function () {
// test from https://github.com/wikimedia/VisualEditor/blob/d468b00311e69c2095b9da360c5745153342a5c3/src/ve.utils.js#L182-L196
var n = 256;
var arr = [];
arr[n] = 'a';
arr.splice(n + 1, 0, 'b');
expect(arr[n]).toBe('a');
});
ifHasDenseUndefinedsIt('should not break on sparse arrays in Safari 7/8', function () {
// test from https://github.com/wikimedia/VisualEditor/blob/d468b00311e69c2095b9da360c5745153342a5c3/src/ve.utils.js#L182-L196
var justFine = new Array(1e5 - 1);
justFine[10] = 'x';
var tooBig = new Array(1e5);
tooBig[8] = 'x';
justFine.splice(1, 1);
expect(8 in justFine).toBe(false);
expect(justFine.indexOf('x')).toBe(9);
tooBig.splice(1, 1);
expect(6 in tooBig).toBe(false);
expect(tooBig.indexOf('x')).toBe(7);
});
});
describe('#join()', function () {
it('defaults to a comma separator when none is provided', function () {
expect([1, 2].join()).toBe('1,2');
});
it('defaults to a comma separator when undefined is provided', function () {
expect([1, 2].join(undefined)).toBe('1,2');
});
it('works, extended', function () {
expect([].join()).toBe('');
expect([undefined].join()).toBe('');
expect([undefined, undefined].join()).toBe(',');
expect([null, null].join()).toBe(',');
expect([undefined, undefined].join('|')).toBe('|');
expect([null, null].join('|')).toBe('|');
expect([1, 2, 3].join('|')).toBe('1|2|3');
expect([1, 2, 3].join(null)).toBe('1null2null3');
expect([1, 2, 3].join({})).toBe('1[object Object]2[object Object]3');
expect([1, 2, 3].join('')).toBe('123');
});
it('is generic', function () {
var obj = { 0: 1, 1: 2, 2: 3, 3: 4, length: 3 };
expect(Array.prototype.join.call(obj, ',')).toBe('1,2,3');
});
it('works with a string literal', function () {
var str = '123';
expect(Array.prototype.join.call(str, ',')).toBe('1,2,3');
});
it('works with `arguments`', function () {
var args = (function () { return arguments; }(1, 2, 3));
expect(Array.prototype.join.call(args, ',')).toBe('1,2,3');
});
});
describe('#push()', function () {
it('works on arrays', function () {
var arr = [];
var result = arr.push(undefined);
expect(result).toBe(1);
expect(arr.length).toBe(1);
expect(Object.prototype.hasOwnProperty.call(arr, 0)).toBe(true);
expect(arr[0]).toBeUndefined();
});
it('is generic', function () {
var obj = {};
var result = Array.prototype.push.call(obj, undefined);
expect(result).toBe(1);
expect(obj.length).toBe(1);
expect(Object.prototype.hasOwnProperty.call(obj, 0)).toBe(true);
expect(obj[0]).toBeUndefined();
});
});
describe('#pop()', function () {
it('works on arrays', function () {
var arr = [1, 2, 3];
var result = arr.pop();
expect(result).toBe(3);
expect(arr.length).toBe(2);
expect(Object.prototype.hasOwnProperty.call(arr, 2)).toBe(false);
expect(arr[2]).toBeUndefined();
});
it('is generic', function () {
var obj = { 0: 1, 1: 2, 2: 3, length: 3 };
var result = Array.prototype.pop.call(obj);
expect(result).toBe(3);
expect(obj.length).toBe(2);
expect(Object.prototype.hasOwnProperty.call(obj, 2)).toBe(false);
expect(obj[2]).toBeUndefined();
});
});
describe('#slice()', function () {
it('works on arrays', function () {
var arr = [1, 2, 3, 4];
var result = arr.slice(1, 3);
expect(result).toEqual([2, 3]);
});
it('is generic', function () {
var obj = { 0: 1, 1: 2, 2: 3, 3: 4, length: 4 };
var result = Array.prototype.slice.call(obj, 1, 3);
expect(result).toEqual([2, 3]);
});
it('works with `arguments`', function () {
var obj = (function () {
return arguments;
}(1, 2, 3, 4));
var result = Array.prototype.slice.call(obj, 1, 3);
expect(result).toEqual([2, 3]);
});
it('boxed string access', function () {
var obj = '1234';
var result = Array.prototype.slice.call(obj, 1, 3);
expect(result).toEqual(['2', '3']);
});
if (typeof document !== 'undefined') {
it('should be able to slice a NodeList', function () {
var nodes = document.getElementsByTagName('div');
expect(Array.isArray(Array.prototype.slice.call(nodes))).toBe(true);
});
}
});
describe('#sort()', function () {
describe('usage', function () {
it('requires a function or undefined as first argument', function () {
var actual = [1, 2];
expect(actual.sort()).toBe(actual);
expect(actual.sort(undefined)).toBe(actual);
expect(actual.sort(function () { return 0; })).toBe(actual);
});
it('requires a non-function or non-undefined to throw a `TypeError`', function () {
expect(function () { [1, 2].sort(null); }).toThrow();
expect(function () { [1, 2].sort(1); }).toThrow();
expect(function () { [1, 2].sort(''); }).toThrow();
expect(function () { [1, 2].sort(true); }).toThrow();
expect(function () { [1, 2].sort({}); }).toThrow();
expect(function () { [1, 2].sort([]); }).toThrow();
expect(function () { [1, 2].sort(new Date()); }).toThrow();
expect(function () { [1, 2].sort(/pattern/); }).toThrow();
});
});
describe('ascending', function () {
it('[5,2,4,6,1,3] should result in [1,2,3,4,5,6]', function () {
var actual = [5, 2, 4, 6, 1, 3];
var expected = [1, 2, 3, 4, 5, 6];
actual.sort();
expect(actual).toEqual(expected);
});
it('[5,2,2,6,1,3] should result in [1,2,2,3,5,6]', function () {
var actual = [5, 2, 2, 6, 1, 3];
var expected = [1, 2, 2, 3, 5, 6];
actual.sort();
expect(actual).toEqual(expected);
});
it('[0,0,0,0,0,1] should result in [0,0,0,0,0,1]', function () {
var actual = [0, 0, 0, 0, 0, 1];
var expected = [0, 0, 0, 0, 0, 1];
actual.sort();
expect(actual).toEqual(expected);
});
it('[0,0,0,0,0,-1] should result in [-1,0,0,0,0,0]', function () {
var actual = [0, 0, 0, 0, 0, -1];
var expected = [-1, 0, 0, 0, 0, 0];
actual.sort();
expect(actual).toEqual(expected);
});
it('[f,e,d,a,c,b] should result in [a,b,c,d,e,f]', function () {
var actual = ['f', 'e', 'd', 'a', 'c', 'b'];
var expected = ['a', 'b', 'c', 'd', 'e', 'f'];
actual.sort();
expect(actual).toEqual(expected);
});
it('[f,e,d,,,,a,c,b] should result in [a,b,c,d,e,f,,,]', function () {
var actual = ['f', 'e', 'd', 1, 2, 'a', 'c', 'b'];
delete actual[3];
delete actual[4];
var expected = ['a', 'b', 'c', 'd', 'e', 'f'];
expected.length = 8;
actual.sort();
expect(actual).toEqual(expected);
});
it('[f,e,d,,null,,a,c,b] should result in [a,b,c,d,e,f,null,,,]', function () {
var actual = ['f', 'e', 'd', 1, null, 2, 'a', 'c', 'b'];
delete actual[3];
delete actual[5];
var expected = ['a', 'b', 'c', 'd', 'e', 'f', null];
expected.length = 9;
actual.sort();
expect(actual).toEqual(expected);
});
it('[f,e,d,,null,undefined,a,c,b] should result in [a,b,c,d,e,f,null,undefined,,]', function () {
var actual = ['f', 'e', 'd', 1, null, undefined, 'a', 'c', 'b'];
delete actual[3];
var expected = ['a', 'b', 'c', 'd', 'e', 'f', null, undefined];
expected.length = 9;
actual.sort();
expect(actual).toEqual(expected);
});
it('[] should result in []', function () {
var actual = [];
var expected = [];
actual.sort();
expect(actual).toEqual(expected);
});
it('[1] should result in [1]', function () {
var actual = [1];
var expected = [1];
actual.sort();
expect(actual).toEqual(expected);
});
it('result should find only greater or equal values', function () {
var actual = [];
var i;
for (i = 0; i < 100; i += 1) {
actual.push(('00' + (Math.random() * 100).toFixed(0)).slice(-3));
}
actual.sort();
for (i = 0; i < actual.length - 1; i += 1) {
expect(actual[i] <= actual[i + 1]).toBe(true);
}
});
});
describe('descending', function () {
var descending = function (left, right) {
var leftS = String(left);
var rightS = String(right);
if (leftS === rightS) {
return +0;
}
if (leftS < rightS) {
return 1;
}
return -1;
};
it('[5,2,4,6,1,3] should result in [6,5,4,3,2,1]', function () {
var actual = [5, 2, 4, 6, 1, 3];
var expected = [6, 5, 4, 3, 2, 1];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[5,2,2,6,1,3] should result in [6,5,4,2,2,1]', function () {
var actual = [5, 2, 2, 6, 1, 3];
var expected = [6, 5, 3, 2, 2, 1];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[0,0,0,0,0,1] should result in [1,0,0,0,0,0]', function () {
var actual = [0, 0, 0, 0, 0, 1];
var expected = [1, 0, 0, 0, 0, 0];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[0,0,0,0,0,-1] should result in [0,0,0,0,0,-1]', function () {
var actual = [0, 0, 0, 0, 0, -1];
var expected = [0, 0, 0, 0, 0, -1];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[f,e,d,a,c,b] should result in [f,e,d,c,b,a]', function () {
var actual = ['f', 'e', 'd', 'a', 'c', 'b'];
var expected = ['f', 'e', 'd', 'c', 'b', 'a'];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[f,e,d,,,a,c,b] should result in [f,e,d,c,b,a,,,]', function () {
var actual = ['f', 'e', 'd', 1, 2, 'a', 'c', 'b'];
delete actual[3];
delete actual[4];
var expected = ['f', 'e', 'd', 'c', 'b', 'a'];
expected.length = 8;
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[f,e,d,,null,,a,c,b] should result in [null,f,e,d,c,b,a,,,]', function () {
var actual = ['f', 'e', 'd', 1, null, 2, 'a', 'c', 'b'];
delete actual[3];
delete actual[5];
var expected = [null, 'f', 'e', 'd', 'c', 'b', 'a'];
expected.length = 9;
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[f,e,d,undefined,null,,a,c,b] should result in [null,f,e,d,c,b,a,undefined,,]', function () {
var actual = ['f', 'e', 'd', undefined, null, 2, 'a', 'c', 'b'];
delete actual[5];
var expected = [null, 'f', 'e', 'd', 'c', 'b', 'a', undefined];
expected.length = 9;
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[] should result in []', function () {
var actual = [];
var expected = [];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('[1] should result in [1]', function () {
var actual = [1];
var expected = [1];
actual.sort(descending);
expect(actual).toEqual(expected);
});
it('result should find only lesser or equal values', function () {
var actual = [];
var i;
for (i = 0; i < 100; i += 1) {
actual.push(('00' + (Math.random() * 100).toFixed(0)).slice(-3));
}
actual.sort(descending);
for (i = 0; i < actual.length - 1; i += 1) {
expect(actual[i] >= actual[i + 1]).toBe(true);
}
});
});
describe('returned value', function () {
it('should be the source object', function () {
var actual = [1, 3, 2];
expect(actual.sort()).toBe(actual);
});
});
describe('when used generically', function () {
var descending = function (left, right) {
var leftS = String(left);
var rightS = String(right);
if (leftS === rightS) {
return +0;
}
if (leftS < rightS) {
return 1;
}
return -1;
};
var args = function () {
return arguments;
};
it('should not sort objects without length', function () {
var actual = {
0: 5,
1: 2,
2: 4,
3: 6,
4: 1,
5: 3
};
var expected = {
0: 5,
1: 2,
2: 4,
3: 6,
4: 1,
5: 3
};
Array.prototype.sort.call(actual);
expect(actual).toEqual(expected);
Array.prototype.sort.call(actual, descending);
expect(actual).toEqual(expected);
});
it('should sort objects ascending with length', function () {
var actual = {
0: 5,
1: 2,
2: 4,
3: 6,
4: 1,
5: 3,
length: 6
};
var expected = {
0: 1,
1: 2,
2: 3,
3: 4,
4: 5,
5: 6,
length: 6
};
Array.prototype.sort.call(actual);
expect(actual).toEqual(expected);
});
it('should sort objects descending with length', function () {
var actual = {
0: 5,
1: 2,
2: 4,
3: 6,
4: 1,
5: 3,
length: 6
};
var expected = {
0: 6,
1: 5,
2: 4,
3: 3,
4: 2,
5: 1,
length: 6
};
Array.prototype.sort.call(actual, descending);
expect(actual).toEqual(expected);
});
it('should sort objects descending with mixed content types and with length', function () {
var actual = {
0: 5,
1: 2,
2: 4,
4: null,
6: 1,
7: 3,
length: 8
};
var expected = {
0: null,
1: 5,
2: 4,
3: 3,
4: 2,
5: 1,
length: 8
};
Array.prototype.sort.call(actual, descending);
expect(actual).toEqual(expected);
});
it('should sort `arguments` object ascending', function () {
var actual = args(5, 2, 4, 6, 1, 3);
var expected = args(1, 2, 3, 4, 5, 6);
Array.prototype.sort.call(actual);
expect(actual).toEqual(expected);
});
it('should sort `arguments` object ascending with mixed content types', function () {
var actual = args(5, 2, 4, null, 1, 3);
var expected = args(1, 2, 3, 4, 5, null);
Array.prototype.sort.call(actual);
expect(actual).toEqual(expected);
});
it('should sort `arguments` object descending', function () {
var actual = args(5, 2, 4, 6, 1, 3);
var expected = args(6, 5, 4, 3, 2, 1);
Array.prototype.sort.call(actual, descending);
expect(actual).toEqual(expected);
});
it('should sort `arguments` object descending with mixed content types', function () {
var actual = args(5, 2, 4, null, 1, 3);
var expected = args(null, 5, 4, 3, 2, 1);
Array.prototype.sort.call(actual, descending);
expect(actual).toEqual(expected);
});
});
});
});