From 849de71e1b9f7240c75c6975ab13b668a1cded3b Mon Sep 17 00:00:00 2001 From: Enrique Piqueras Date: Tue, 6 Aug 2019 09:27:18 -0400 Subject: [PATCH] Core Data: Add support for autosaving entities. (#16903) * Core Data: Make current offset selector private. * Core Data: Add support for autosaving entities. * Core Data: Clarify why and how autosave payload data is gathered, with a comment. --- .../developers/data/data-core.md | 35 ++++---- packages/core-data/README.md | 35 ++++---- packages/core-data/src/actions.js | 88 +++++++++++++++---- packages/core-data/src/reducer.js | 1 + packages/core-data/src/selectors.js | 30 +++++-- 5 files changed, 133 insertions(+), 56 deletions(-) diff --git a/docs/designers-developers/developers/data/data-core.md b/docs/designers-developers/developers/data/data-core.md index ceb9daef907720..bdfbe68438a392 100644 --- a/docs/designers-developers/developers/data/data-core.md +++ b/docs/designers-developers/developers/data/data-core.md @@ -71,22 +71,6 @@ _Returns_ - `?Array`: An array of autosaves for the post, or undefined if there is none. -# **getCurrentUndoOffset** - -Returns the current undo offset for the -entity records edits history. The offset -represents how many items from the end -of the history stack we are at. 0 is the -last edit, -1 is the second last, and so on. - -_Parameters_ - -- _state_ `Object`: State tree. - -_Returns_ - -- `number`: The current undo offset. - # **getCurrentUser** Returns the current user. @@ -359,6 +343,21 @@ _Returns_ - `boolean`: Whether or not the user can upload media. Defaults to `true` if the OPTIONS request is being made. +# **isAutosavingEntityRecord** + +Returns true if the specified entity record is autosaving, and false otherwise. + +_Parameters_ + +- _state_ `Object`: State tree. +- _kind_ `string`: Entity kind. +- _name_ `string`: Entity name. +- _recordId_ `number`: Record ID. + +_Returns_ + +- `boolean`: Whether the entity record is autosaving or not. + # **isPreviewEmbedFallback** Determines if the returned preview is an oEmbed link fallback. @@ -403,7 +402,7 @@ _Parameters_ _Returns_ -- `?Object`: Whether the entity record is saving or not. +- `boolean`: Whether the entity record is saving or not. @@ -561,6 +560,7 @@ _Parameters_ - _kind_ `string`: Kind of the entity. - _name_ `string`: Name of the entity. - _recordId_ `Object`: ID of the record. +- _options_ `Object`: Saving options. # **saveEntityRecord** @@ -571,6 +571,7 @@ _Parameters_ - _kind_ `string`: Kind of the received entity. - _name_ `string`: Name of the received entity. - _record_ `Object`: Record to be saved. +- _options_ `Object`: Saving options. # **undo** diff --git a/packages/core-data/README.md b/packages/core-data/README.md index cf9f0a9989b28f..cc89ec095f5117 100644 --- a/packages/core-data/README.md +++ b/packages/core-data/README.md @@ -191,6 +191,7 @@ _Parameters_ - _kind_ `string`: Kind of the entity. - _name_ `string`: Name of the entity. - _recordId_ `Object`: ID of the record. +- _options_ `Object`: Saving options. # **saveEntityRecord** @@ -201,6 +202,7 @@ _Parameters_ - _kind_ `string`: Kind of the received entity. - _name_ `string`: Name of the received entity. - _record_ `Object`: Record to be saved. +- _options_ `Object`: Saving options. # **undo** @@ -280,22 +282,6 @@ _Returns_ - `?Array`: An array of autosaves for the post, or undefined if there is none. -# **getCurrentUndoOffset** - -Returns the current undo offset for the -entity records edits history. The offset -represents how many items from the end -of the history stack we are at. 0 is the -last edit, -1 is the second last, and so on. - -_Parameters_ - -- _state_ `Object`: State tree. - -_Returns_ - -- `number`: The current undo offset. - # **getCurrentUser** Returns the current user. @@ -568,6 +554,21 @@ _Returns_ - `boolean`: Whether or not the user can upload media. Defaults to `true` if the OPTIONS request is being made. +# **isAutosavingEntityRecord** + +Returns true if the specified entity record is autosaving, and false otherwise. + +_Parameters_ + +- _state_ `Object`: State tree. +- _kind_ `string`: Entity kind. +- _name_ `string`: Entity name. +- _recordId_ `number`: Record ID. + +_Returns_ + +- `boolean`: Whether the entity record is autosaving or not. + # **isPreviewEmbedFallback** Determines if the returned preview is an oEmbed link fallback. @@ -612,7 +613,7 @@ _Parameters_ _Returns_ -- `?Object`: Whether the entity record is saving or not. +- `boolean`: Whether the entity record is saving or not. diff --git a/packages/core-data/src/actions.js b/packages/core-data/src/actions.js index 8e78957bbb6756..43861be226ca5d 100644 --- a/packages/core-data/src/actions.js +++ b/packages/core-data/src/actions.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { castArray, merge, isEqual, find } from 'lodash'; +import { castArray, get, merge, isEqual, find } from 'lodash'; /** * Internal dependencies @@ -127,7 +127,11 @@ export function receiveEmbedPreview( url, preview ) { * @return {Object} Action object. */ export function* editEntityRecord( kind, name, recordId, edits ) { - const { transientEdits = {}, mergedEdits = {} } = yield select( 'getEntity', kind, name ); + const { transientEdits = {}, mergedEdits = {} } = yield select( + 'getEntity', + kind, + name + ); const record = yield select( 'getEntityRecord', kind, name, recordId ); const editedRecord = yield select( 'getEditedEntityRecord', @@ -143,10 +147,11 @@ export function* editEntityRecord( kind, name, recordId, edits ) { // Clear edits when they are equal to their persisted counterparts // so that the property is not considered dirty. edits: Object.keys( edits ).reduce( ( acc, key ) => { + const recordValue = get( record[ key ], 'raw', record[ key ] ); const value = mergedEdits[ key ] ? - merge( record[ key ], edits[ key ] ) : + merge( recordValue, edits[ key ] ) : edits[ key ]; - acc[ key ] = isEqual( record[ key ], value ) ? undefined : value; + acc[ key ] = isEqual( recordValue, value ) ? undefined : value; return acc; }, {} ), transientEdits, @@ -209,29 +214,77 @@ export function* redo() { * @param {string} kind Kind of the received entity. * @param {string} name Name of the received entity. * @param {Object} record Record to be saved. + * @param {Object} options Saving options. */ -export function* saveEntityRecord( kind, name, record ) { +export function* saveEntityRecord( + kind, + name, + record, + { isAutosave = false } = { isAutosave: false } +) { const entities = yield getKindEntities( kind ); const entity = find( entities, { kind, name } ); if ( ! entity ) { return; } - const key = entity.key || DEFAULT_ENTITY_KEY; - const recordId = record[ key ]; + const entityIdKey = entity.key || DEFAULT_ENTITY_KEY; + const recordId = record[ entityIdKey ]; - yield { type: 'SAVE_ENTITY_RECORD_START', kind, name, recordId }; + yield { type: 'SAVE_ENTITY_RECORD_START', kind, name, recordId, isAutosave }; let error; try { - const updatedRecord = yield apiFetch( { - path: `${ entity.baseURL }${ recordId ? '/' + recordId : '' }`, - method: recordId ? 'PUT' : 'POST', - data: record, - } ); - yield receiveEntityRecords( kind, name, updatedRecord, undefined, true ); + const path = `${ entity.baseURL }${ recordId ? '/' + recordId : '' }`; + if ( isAutosave ) { + const persistedRecord = yield select( + 'getEntityRecord', + kind, + name, + recordId + ); + const currentUser = yield select( 'getCurrentUser' ); + const currentUserId = currentUser ? currentUser.id : undefined; + const autosavePost = yield select( + 'getAutosave', + persistedRecord.type, + persistedRecord.id, + currentUserId + ); + // Autosaves need all expected fields to be present. + // So we fallback to the previous autosave and then + // to the actual persisted entity if the edits don't + // have a value. + let data = { ...persistedRecord, ...autosavePost, ...record }; + data = Object.keys( data ).reduce( ( acc, key ) => { + if ( key in [ 'title', 'excerpt', 'content' ] ) { + acc[ key ] = get( data[ key ], 'raw', data[ key ] ); + } + return acc; + }, {} ); + const autosave = yield apiFetch( { + path: `${ path }/autosaves`, + method: 'POST', + data, + } ); + yield receiveAutosaves( persistedRecord.id, autosave ); + } else { + const updatedRecord = yield apiFetch( { + path, + method: recordId ? 'PUT' : 'POST', + data: record, + } ); + yield receiveEntityRecords( kind, name, updatedRecord, undefined, true ); + } } catch ( _error ) { error = _error; } - yield { type: 'SAVE_ENTITY_RECORD_FINISH', kind, name, recordId, error }; + yield { + type: 'SAVE_ENTITY_RECORD_FINISH', + kind, + name, + recordId, + error, + isAutosave, + }; } /** @@ -240,8 +293,9 @@ export function* saveEntityRecord( kind, name, record ) { * @param {string} kind Kind of the entity. * @param {string} name Name of the entity. * @param {Object} recordId ID of the record. + * @param {Object} options Saving options. */ -export function* saveEditedEntityRecord( kind, name, recordId ) { +export function* saveEditedEntityRecord( kind, name, recordId, options ) { if ( ! ( yield select( 'hasEditsForEntityRecord', kind, name, recordId ) ) ) { return; } @@ -252,7 +306,7 @@ export function* saveEditedEntityRecord( kind, name, recordId ) { recordId ); const record = { id: recordId, ...edits }; - yield* saveEntityRecord( kind, name, record ); + yield* saveEntityRecord( kind, name, record, options ); } /** diff --git a/packages/core-data/src/reducer.js b/packages/core-data/src/reducer.js index 96ed12e5ef78c7..7f0972e19e481b 100644 --- a/packages/core-data/src/reducer.js +++ b/packages/core-data/src/reducer.js @@ -213,6 +213,7 @@ function entity( entityConfig ) { [ action.recordId ]: { pending: action.type === 'SAVE_ENTITY_RECORD_START', error: action.error, + isAutosave: action.isAutosave, }, }; } diff --git a/packages/core-data/src/selectors.js b/packages/core-data/src/selectors.js index e0ac2aa856ae1c..c8aec286a1107c 100644 --- a/packages/core-data/src/selectors.js +++ b/packages/core-data/src/selectors.js @@ -207,6 +207,25 @@ export const getEditedEntityRecord = createSelector( ( state ) => [ state.entities.data ] ); +/** + * Returns true if the specified entity record is autosaving, and false otherwise. + * + * @param {Object} state State tree. + * @param {string} kind Entity kind. + * @param {string} name Entity name. + * @param {number} recordId Record ID. + * + * @return {boolean} Whether the entity record is autosaving or not. + */ +export function isAutosavingEntityRecord( state, kind, name, recordId ) { + const { pending, isAutosave } = get( + state.entities.data, + [ kind, name, 'saving', recordId ], + {} + ); + return Boolean( pending && isAutosave ); +} + /** * Returns true if the specified entity record is saving, and false otherwise. * @@ -215,14 +234,15 @@ export const getEditedEntityRecord = createSelector( * @param {string} name Entity name. * @param {number} recordId Record ID. * - * @return {Object?} Whether the entity record is saving or not. + * @return {boolean} Whether the entity record is saving or not. */ export function isSavingEntityRecord( state, kind, name, recordId ) { - return get( + const { pending, isAutosave } = get( state.entities.data, - [ kind, name, 'saving', recordId, 'pending' ], - false + [ kind, name, 'saving', recordId ], + {} ); + return Boolean( pending && ! isAutosave ); } /** @@ -250,7 +270,7 @@ export function getLastEntitySaveError( state, kind, name, recordId ) { * * @return {number} The current undo offset. */ -export function getCurrentUndoOffset( state ) { +function getCurrentUndoOffset( state ) { return state.undo.offset; }