[ReactNative] deepDiff by default

This commit is contained in:
Spencer Ahrens 2015-07-14 17:06:03 -07:00
parent 7a23651ee1
commit 5d4140c513
3 changed files with 166 additions and 20 deletions

View File

@ -0,0 +1,150 @@
/**
* Copyright 2004-present Facebook. All Rights Reserved.
*/
'use strict';
jest.dontMock('diffRawProperties');
jest.dontMock('deepDiffer');
var diffRawProperties = require('diffRawProperties');
describe('diffRawProperties', function() {
it('should work with simple example', () => {
expect(diffRawProperties(
null,
{a: 1, c: 3},
{b: 2, c: 3},
{a: true, b: true}
)).toEqual({a: null, b: 2});
});
it('should skip fields that are equal', () => {
expect(diffRawProperties(
null,
{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', () => {
expect(diffRawProperties(
null,
{a: 1},
{},
{a: true}
)).toEqual({a: null});
});
it('should ignore invalid fields', () => {
expect(diffRawProperties(
null,
{a: 1},
{b: 2},
{}
)).toEqual(null);
});
it('should override the updatePayload argument', () => {
var updatePayload = {a: 1};
var result = diffRawProperties(
updatePayload,
{},
{b: 2},
{b: true}
);
expect(result).toBe(updatePayload);
expect(result).toEqual({a: 1, b: 2});
});
it('should use the diff attribute', () => {
var diffA = jest.genMockFunction().mockImpl((a, b) => true);
var diffB = jest.genMockFunction().mockImpl((a, b) => false);
expect(diffRawProperties(
null,
{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();
expect(diffRawProperties(
null,
{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', () => {
expect(diffRawProperties(
null,
{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', () => {
expect(diffRawProperties(
null,
{a: 1, c: 3},
undefined,
{a: true, b: true}
)).toEqual({a: null});
expect(diffRawProperties(
null,
undefined,
{a: 1, c: 3},
{a: true, b: true}
)).toEqual({a: 1});
expect(diffRawProperties(
null,
undefined,
undefined,
{a: true, b: true}
)).toEqual(null);
});
it('should work with empty styles', () => {
expect(diffRawProperties(
null,
{a: 1, c: 3},
{},
{a: true, b: true}
)).toEqual({a: null});
expect(diffRawProperties(
null,
{},
{a: 1, c: 3},
{a: true, b: true}
)).toEqual({a: 1});
expect(diffRawProperties(
null,
{},
{},
{a: true, b: true}
)).toEqual(null);
});
// 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.
expect(diffRawProperties(
null,
{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});
});
});

View File

@ -11,6 +11,8 @@
*/ */
'use strict'; 'use strict';
var deepDiffer = require('deepDiffer');
/** /**
* diffRawProperties takes two sets of props and a set of valid attributes * diffRawProperties takes two sets of props and a set of valid attributes
* and write to updatePayload the values that changed or were deleted * and write to updatePayload the values that changed or were deleted
@ -33,6 +35,7 @@ function diffRawProperties(
var prevProp; var prevProp;
var isScalar; var isScalar;
var shouldUpdate; var shouldUpdate;
var differ;
if (nextProps) { if (nextProps) {
for (var propKey in nextProps) { for (var propKey in nextProps) {
@ -53,16 +56,11 @@ function diffRawProperties(
} }
if (prevProp !== nextProp) { if (prevProp !== nextProp) {
// If you want a property's diff to be detected, you must configure it // Scalars and new props are always updated. Objects use deepDiffer by
// to be so - *or* it must be a scalar property. For now, we'll allow // default, but can be optimized with custom differs.
// creation with any attribute that is not scalar, but we should differ = validAttributeConfig.diff || deepDiffer;
// eventually even reject those unless they are properly configured.
isScalar = typeof nextProp !== 'object' || nextProp === null; isScalar = typeof nextProp !== 'object' || nextProp === null;
shouldUpdate = isScalar || shouldUpdate = isScalar || !prevProp || differ(prevProp, nextProp);
!prevProp ||
validAttributeConfig.diff &&
validAttributeConfig.diff(prevProp, nextProp);
if (shouldUpdate) { if (shouldUpdate) {
updatePayload = updatePayload || {}; updatePayload = updatePayload || {};
updatePayload[propKey] = nextProp; updatePayload[propKey] = nextProp;
@ -99,14 +97,14 @@ function diffRawProperties(
if (nextProp === undefined) { if (nextProp === undefined) {
nextProp = null; // null is a sentinel we explicitly send to native nextProp = null; // null is a sentinel we explicitly send to native
} }
// If you want a property's diff to be detected, you must configure it // Scalars and new props are always updated. Objects use deepDiffer by
// to be so - *or* it must be a scalar property. For now, we'll allow // default, but can be optimized with custom differs.
// creation with any attribute that is not scalar, but we should differ = validAttributeConfig.diff || deepDiffer;
// eventually even reject those unless they are properly configured.
isScalar = typeof nextProp !== 'object' || nextProp === null; isScalar = typeof nextProp !== 'object' || nextProp === null;
shouldUpdate = isScalar && prevProp !== nextProp || shouldUpdate =
validAttributeConfig.diff && isScalar &&
validAttributeConfig.diff(prevProp, nextProp); prevProp !== nextProp ||
differ(prevProp, nextProp);
if (shouldUpdate) { if (shouldUpdate) {
updatePayload = updatePayload || {}; updatePayload = updatePayload || {};
updatePayload[propKey] = nextProp; updatePayload[propKey] = nextProp;

View File

@ -15,7 +15,6 @@ var RCTUIManager = require('NativeModules').UIManager;
var UnimplementedView = require('UnimplementedView'); var UnimplementedView = require('UnimplementedView');
var createReactNativeComponentClass = require('createReactNativeComponentClass'); var createReactNativeComponentClass = require('createReactNativeComponentClass');
var deepDiffer = require('deepDiffer');
var insetsDiffer = require('insetsDiffer'); var insetsDiffer = require('insetsDiffer');
var pointsDiffer = require('pointsDiffer'); var pointsDiffer = require('pointsDiffer');
var matricesDiffer = require('matricesDiffer'); var matricesDiffer = require('matricesDiffer');
@ -54,9 +53,8 @@ function requireNativeComponent(
viewConfig.uiViewClassName = viewName; viewConfig.uiViewClassName = viewName;
viewConfig.validAttributes = {}; viewConfig.validAttributes = {};
for (var key in nativeProps) { for (var key in nativeProps) {
// TODO: deep diff by default in diffRawProperties instead of setting it here var differ = TypeToDifferMap[nativeProps[key]];
var differ = TypeToDifferMap[nativeProps[key]] || deepDiffer; viewConfig.validAttributes[key] = differ ? {diff: differ} : true;
viewConfig.validAttributes[key] = {diff: differ};
} }
if (__DEV__) { if (__DEV__) {
wrapperComponent && verifyPropTypes(wrapperComponent, viewConfig); wrapperComponent && verifyPropTypes(wrapperComponent, viewConfig);