2015-07-15 00:06:03 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2015-10-06 02:19:16 +00:00
|
|
|
jest.dontMock('ReactNativeAttributePayload');
|
2015-10-06 22:19:59 +00:00
|
|
|
jest.dontMock('StyleSheetRegistry');
|
2015-07-15 00:06:03 +00:00
|
|
|
jest.dontMock('deepDiffer');
|
2015-10-06 16:23:40 +00:00
|
|
|
jest.dontMock('flattenStyle');
|
2015-10-06 22:19:59 +00:00
|
|
|
jest.dontMock('styleDiffer');
|
|
|
|
|
2015-10-06 02:19:16 +00:00
|
|
|
var ReactNativeAttributePayload = require('ReactNativeAttributePayload');
|
2015-10-06 22:19:59 +00:00
|
|
|
var StyleSheetRegistry = require('StyleSheetRegistry');
|
2015-07-15 00:06:03 +00:00
|
|
|
|
2015-10-06 02:19:16 +00:00
|
|
|
var diff = ReactNativeAttributePayload.diff;
|
|
|
|
|
|
|
|
describe('ReactNativeAttributePayload', function() {
|
2015-07-15 00:06:03 +00:00
|
|
|
|
|
|
|
it('should work with simple example', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: 1, c: 3},
|
|
|
|
{b: 2, c: 3},
|
|
|
|
{a: true, b: true}
|
|
|
|
)).toEqual({a: null, b: 2});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should skip fields that are equal', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: 1, b: 'two', c: true, d: false, e: undefined, f: 0},
|
|
|
|
{a: 1, b: 'two', c: true, d: false, e: undefined, f: 0},
|
|
|
|
{a: true, b: true, c: true, d: true, e: true, f: true}
|
|
|
|
)).toEqual(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should remove fields', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: 1},
|
|
|
|
{},
|
|
|
|
{a: true}
|
|
|
|
)).toEqual({a: null});
|
|
|
|
});
|
|
|
|
|
2015-10-06 03:21:48 +00:00
|
|
|
it('should remove fields that are set to undefined', () => {
|
|
|
|
expect(diff(
|
|
|
|
{a: 1},
|
|
|
|
{a: undefined},
|
|
|
|
{a: true}
|
|
|
|
)).toEqual({a: null});
|
|
|
|
});
|
|
|
|
|
2015-07-15 00:06:03 +00:00
|
|
|
it('should ignore invalid fields', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: 1},
|
|
|
|
{b: 2},
|
|
|
|
{}
|
|
|
|
)).toEqual(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should use the diff attribute', () => {
|
|
|
|
var diffA = jest.genMockFunction().mockImpl((a, b) => true);
|
|
|
|
var diffB = jest.genMockFunction().mockImpl((a, b) => false);
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: [1], b: [3]},
|
|
|
|
{a: [2], b: [4]},
|
|
|
|
{a: {diff: diffA}, b: {diff: diffB}}
|
|
|
|
)).toEqual({a: [2]});
|
|
|
|
expect(diffA).toBeCalledWith([1], [2]);
|
|
|
|
expect(diffB).toBeCalledWith([3], [4]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not use the diff attribute on addition/removal', () => {
|
|
|
|
var diffA = jest.genMockFunction();
|
|
|
|
var diffB = jest.genMockFunction();
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: [1]},
|
|
|
|
{b: [2]},
|
|
|
|
{a: {diff: diffA}, b: {diff: diffB}}
|
|
|
|
)).toEqual({a: null, b: [2]});
|
|
|
|
expect(diffA).not.toBeCalled();
|
|
|
|
expect(diffB).not.toBeCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should do deep diffs of Objects by default', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: [1], b: {k: [3,4]}, c: {k: [4,4]} },
|
|
|
|
{a: [2], b: {k: [3,4]}, c: {k: [4,5]} },
|
|
|
|
{a: true, b: true, c: true}
|
|
|
|
)).toEqual({a: [2], c: {k: [4,5]}});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with undefined styles', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-10-06 03:21:48 +00:00
|
|
|
{ style: { a: '#ffffff', b: 1 } },
|
2015-10-06 02:19:16 +00:00
|
|
|
{ style: undefined },
|
2015-10-06 03:21:48 +00:00
|
|
|
{ style: { b: true } }
|
|
|
|
)).toEqual({ b: null });
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
|
|
|
{ style: undefined },
|
2015-10-06 03:21:48 +00:00
|
|
|
{ style: { a: '#ffffff', b: 1 } },
|
|
|
|
{ style: { b: true } }
|
|
|
|
)).toEqual({ b: 1 });
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
|
|
|
{ style: undefined },
|
|
|
|
{ style: undefined },
|
2015-10-06 03:21:48 +00:00
|
|
|
{ style: { b: true } }
|
2015-07-15 00:06:03 +00:00
|
|
|
)).toEqual(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with empty styles', () => {
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: 1, c: 3},
|
|
|
|
{},
|
|
|
|
{a: true, b: true}
|
|
|
|
)).toEqual({a: null});
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{},
|
|
|
|
{a: 1, c: 3},
|
|
|
|
{a: true, b: true}
|
|
|
|
)).toEqual({a: 1});
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{},
|
|
|
|
{},
|
|
|
|
{a: true, b: true}
|
|
|
|
)).toEqual(null);
|
|
|
|
});
|
|
|
|
|
2015-10-06 22:19:59 +00:00
|
|
|
it('should flatten nested styles and predefined styles', () => {
|
|
|
|
var validStyleAttribute = { someStyle: { foo: true, bar: true } };
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{},
|
|
|
|
{ someStyle: [{ foo: 1 }, { bar: 2 }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: 1, bar: 2 });
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 1 }, { bar: 2 }]},
|
|
|
|
{},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: null, bar: null });
|
|
|
|
|
|
|
|
var barStyle = StyleSheetRegistry.registerStyle({
|
|
|
|
bar: 3,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{},
|
|
|
|
{ someStyle: [[{ foo: 1 }, { foo: 2 }], barStyle]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: 2, bar: 3 });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should reset a value to a previous if it is removed', () => {
|
|
|
|
var validStyleAttribute = { someStyle: { foo: true, bar: true } };
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 1 }, { foo: 3 }]},
|
|
|
|
{ someStyle: [{ foo: 1 }, { bar: 2 }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: 1, bar: 2 });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not clear removed props if they are still in another slot', () => {
|
|
|
|
var validStyleAttribute = { someStyle: { foo: true, bar: true } };
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{}, { foo: 3, bar: 2 }]},
|
|
|
|
{ someStyle: [{ foo: 3 }, { bar: 2 }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual(null);
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{}, { foo: 3, bar: 2 }]},
|
|
|
|
{ someStyle: [{ foo: 1, bar: 1 }, { bar: 2 }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: 1 });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should clear a prop if a later style is explicit null/undefined', () => {
|
|
|
|
var validStyleAttribute = { someStyle: { foo: true, bar: true } };
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{}, { foo: 3, bar: 2 }]},
|
|
|
|
{ someStyle: [{ foo: 1 }, { bar: 2, foo: null }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: null });
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 3 }, { foo: null, bar: 2 }]},
|
|
|
|
{ someStyle: [{ foo: null }, { bar: 2 }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual(null);
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 1 }, { foo: null }]},
|
|
|
|
{ someStyle: [{ foo: 2 }, { foo: null }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual(null);
|
|
|
|
|
|
|
|
// Test the same case with object equality because an early bailout doesn't
|
|
|
|
// work in this case.
|
|
|
|
var fooObj = { foo: 3 };
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 1 }, fooObj]},
|
|
|
|
{ someStyle: [{ foo: 2 }, fooObj]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual(null);
|
|
|
|
|
|
|
|
expect(diff(
|
|
|
|
{ someStyle: [{ foo: 1 }, { foo: 3 }]},
|
|
|
|
{ someStyle: [{ foo: 2 }, { foo: undefined }]},
|
|
|
|
validStyleAttribute
|
|
|
|
)).toEqual({ foo: null });
|
|
|
|
});
|
|
|
|
|
2015-07-15 00:06:03 +00:00
|
|
|
// Function properties are just markers to native that events should be sent.
|
|
|
|
it('should convert functions to booleans', () => {
|
|
|
|
// Note that if the property changes from one function to another, we don't
|
|
|
|
// need to send an update.
|
2015-10-06 02:19:16 +00:00
|
|
|
expect(diff(
|
2015-07-15 00:06:03 +00:00
|
|
|
{a: function() { return 1; }, b: function() { return 2; }, c: 3},
|
|
|
|
{b: function() { return 9; }, c: function() { return 3; }, },
|
|
|
|
{a: true, b: true, c: true}
|
|
|
|
)).toEqual({a: null, c: true});
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|