Skip to content

Commit

Permalink
Editor: Deeply diff/merge edited property edits/updates
Browse files Browse the repository at this point in the history
  • Loading branch information
aduth committed Oct 23, 2018
1 parent a3cd835 commit edfa989
Show file tree
Hide file tree
Showing 4 changed files with 263 additions and 36 deletions.
88 changes: 88 additions & 0 deletions packages/editor/src/store/object.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
/**
* External dependencies
*/
import { isPlainObject } from 'lodash';

/**
* Returns an object against which it is safe to perform mutating operations,
* given the original object and its current working copy.
*
* @param {Object} original Original object.
* @param {Object} working Working object.
*
* @return {Object} Mutation-safe object.
*/
function getMutateSafeObject( original, working ) {
if ( original === working ) {
return { ...original };
}

return working;
}

/**
* Given two objects, returns the minimal object shape of the difference of
* values contained in RHS and not LHS, recursively. Returns RHS reference if
* result of difference would be the same as the RHS object.
*
* @param {Object} lhs Original object to compare against.
* @param {Object} rhs New object from which to generate difference.
*
* @return {Object} Minimal difference.
*/
export function diff( lhs, rhs ) {
let diffed = rhs;
for ( const key in rhs ) {
let value = rhs[ key ];
if ( isPlainObject( value ) && isPlainObject( lhs[ key ] ) ) {
// Recurse to generate diff of child values.
value = diff( lhs[ key ], value );

// If a diff of the child value is non-empty, it's inferred to be
// non-equal and should replace the value in the returned diff.
// Otherwise, if equal, fall through to delete from diff.
if ( Object.keys( value ).length ) {
diffed = getMutateSafeObject( rhs, diffed );
diffed[ key ] = value;
continue;
}
} else if ( value !== lhs[ key ] ) {
// To preserve reference, invert the iteration to _keep_ values
// which are different, and mutate only to delete equal values.
continue;
}

diffed = getMutateSafeObject( rhs, diffed );
delete diffed[ key ];
}

return diffed;
}

/**
* Given objects, returns the combined object shape of the merging of their
* values. Returns the reference of the first object if all objects are the
* same.
*
* @param {...Object} objects Objects to merge.
*
* @return {Object} Merged object.
*/
export function merge( ...objects ) {
const firstObject = objects[ 0 ];
return objects.reduce( ( merged, object ) => {
for ( const key in object ) {
let value = object[ key ];
if ( isPlainObject( merged[ key ] ) && isPlainObject( value ) ) {
value = merge( merged[ key ], value );
}

if ( merged[ key ] !== value ) {
merged = getMutateSafeObject( firstObject, merged );
merged[ key ] = value;
}
}

return merged;
}, firstObject );
}
45 changes: 9 additions & 36 deletions packages/editor/src/store/reducer.js
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ import withHistory from '../utils/with-history';
import withChangeDetection from '../utils/with-change-detection';
import { PREFERENCES_DEFAULTS, EDITOR_SETTINGS_DEFAULTS } from './defaults';
import { insertAt, moveTo } from './array';
import { merge, diff } from './object';

/**
* Returns a post attribute value, flattening nested rendered content using its
Expand Down Expand Up @@ -231,27 +232,7 @@ export const editor = flow( [
switch ( action.type ) {
case 'EDIT_POST':
case 'SETUP_EDITOR_STATE':
const recurseEdits = ( edits, newState ) => {
return reduce( edits, ( result, value, key ) => {
// Only assign into result if not already same value
if ( value !== newState[ key ] ) {
// Avoid mutating original state by creating shallow
// clone. Should only occur once per reduce.
if ( result === newState ) {
result = { ...newState };
}

if ( typeof newState[ key ] === 'object' && ! Array.isArray( newState[ key ] ) ) {
result[ key ] = recurseEdits( value, newState[ key ] );
} else {
result[ key ] = value;
}
}
return result;
}, newState );
};

return recurseEdits( action.edits, state );
return merge( state, action.edits );

case 'RESET_BLOCKS':
if ( 'content' in state ) {
Expand All @@ -265,22 +246,14 @@ export const editor = flow( [

case 'UPDATE_POST':
case 'RESET_POST':
const getCanonicalValue = action.type === 'UPDATE_POST' ?
( key ) => action.edits[ key ] :
( key ) => getPostRawValue( action.post[ key ] );

return reduce( state, ( result, value, key ) => {
if ( value !== getCanonicalValue( key ) ) {
return result;
}

if ( state === result ) {
result = { ...state };
}
let updates;
if ( action.type === 'UPDATE_POST' ) {
updates = action.edits;
} else {
updates = mapValues( action.post, getPostRawValue );
}

delete result[ key ];
return result;
}, state );
return diff( updates, state );
}

return state;
Expand Down
53 changes: 53 additions & 0 deletions packages/editor/src/store/test/object.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
/**
* Internal dependencies
*/
import { diff, merge } from '../object';

describe( 'object', () => {
describe( 'diff', () => {
it( 'should return RHS reference if result would be same as RHS', () => {
const lhs = {};
const rhs = { a: { b: { c: 1 } } };
const result = diff( lhs, rhs );

expect( result ).toBe( rhs );
} );

it( 'should omit deeply equal property values', () => {
const lhs = { a: { b: { c: 1 } } };
const rhs = { a: { b: { c: 1 } } };
const result = diff( lhs, rhs );

expect( result ).not.toBe( rhs );
expect( result ).toEqual( {} );
} );

it( 'should return minimal object difference', () => {
const lhs = { a: { b: { c: 1 } } };
const rhs = { a: { b: { c: 1, d: 2 } } };
const result = diff( lhs, rhs );

expect( result ).not.toBe( rhs );
expect( result ).toEqual( { a: { b: { d: 2 } } } );
} );
} );

describe( 'merge', () => {
it( 'should return equal reference if same, deeply', () => {
const obj1 = { a: { b: { c: 1 } } };
const obj2 = { a: { b: { c: 1 } } };
const result = merge( obj1, obj2 );

expect( result ).toBe( obj1 );
} );

it( 'should deeply merged object', () => {
const obj1 = { a: { b: { c: 1 } } };
const obj2 = { a: { b: { d: 2 } } };
const result = merge( obj1, obj2 );

expect( result ).not.toBe( obj1 );
expect( result ).toEqual( { a: { b: { c: 1, d: 2 } } } );
} );
} );
} );
113 changes: 113 additions & 0 deletions packages/editor/src/store/test/reducer.js
Original file line number Diff line number Diff line change
Expand Up @@ -1037,6 +1037,119 @@ describe( 'state', () => {
} );
} );

it( 'should merge object values', () => {
const original = editor( undefined, {
type: 'EDIT_POST',
edits: {
meta: {
a: 1,
},
},
} );

const state = editor( original, {
type: 'EDIT_POST',
edits: {
meta: {
b: 2,
},
},
} );

expect( state.present.edits ).toEqual( {
meta: {
a: 1,
b: 2,
},
} );
} );

it( 'return state by reference on unchanging update', () => {
const original = editor( undefined, {} );

const state = editor( original, {
type: 'UPDATE_POST',
edits: {},
} );

expect( state.present.edits ).toBe( original.present.edits );
} );

it( 'unset updated post values which match', () => {
const original = editor( undefined, {
type: 'EDIT_POST',
edits: {
title: 'modified title',
meta: {
a: 1,
b: 2,
},
},
} );

const state = editor( original, {
type: 'UPDATE_POST',
edits: {
title: 'modified title',
meta: {
a: 1,
},
},
} );

expect( state.present.edits ).toEqual( {
meta: {
b: 2,
},
} );
} );

it( 'unset reset post values which match by canonical value', () => {
const original = editor( undefined, {
type: 'EDIT_POST',
edits: {
title: 'modified title',
},
} );

const state = editor( original, {
type: 'RESET_POST',
post: {
title: {
raw: 'modified title',
},
},
} );

expect( state.present.edits ).toEqual( {} );
} );

it( 'unset top-level key of empty object value', () => {
const original = editor( undefined, {
type: 'EDIT_POST',
edits: {
title: 'modified title',
meta: {
a: 1,
b: 2,
},
},
} );

const state = editor( original, {
type: 'UPDATE_POST',
edits: {
title: 'modified title',
meta: {
a: 1,
b: 2,
},
},
} );

expect( state.present.edits ).toEqual( {} );
} );

it( 'should save initial post state', () => {
const state = editor( undefined, {
type: 'SETUP_EDITOR_STATE',
Expand Down

0 comments on commit edfa989

Please sign in to comment.