Skip to content

Commit

Permalink
Propagate undo to context for empty TinyMCE history
Browse files Browse the repository at this point in the history
  • Loading branch information
aduth committed Jul 28, 2017
1 parent bce86b2 commit 3c68e0f
Show file tree
Hide file tree
Showing 4 changed files with 79 additions and 3 deletions.
35 changes: 34 additions & 1 deletion blocks/editable/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,17 @@
*/
import tinymce from 'tinymce';
import classnames from 'classnames';
import { last, isEqual, omitBy, forEach, merge, identity, find } from 'lodash';
import {
last,
isEqual,
omitBy,
forEach,
merge,
identity,
find,
defer,
noop,
} from 'lodash';
import { nodeListToReact } from 'dom-react';
import { Fill } from 'react-slot-fill';
import 'element-closest';
Expand Down Expand Up @@ -54,6 +64,7 @@ export default class Editable extends Component {
this.onKeyUp = this.onKeyUp.bind( this );
this.changeFormats = this.changeFormats.bind( this );
this.onSelectionChange = this.onSelectionChange.bind( this );
this.maybePropagateUndo = this.maybePropagateUndo.bind( this );
this.onPastePostProcess = this.onPastePostProcess.bind( this );

this.state = {
Expand All @@ -80,6 +91,7 @@ export default class Editable extends Component {
editor.on( 'keydown', this.onKeyDown );
editor.on( 'keyup', this.onKeyUp );
editor.on( 'selectionChange', this.onSelectionChange );
editor.on( 'BeforeExecCommand', this.maybePropagateUndo );
editor.on( 'PastePostProcess', this.onPastePostProcess );

patterns.apply( this, [ editor ] );
Expand Down Expand Up @@ -129,6 +141,23 @@ export default class Editable extends Component {
}
}

maybePropagateUndo( event ) {
const { onUndo } = this.context;
if ( onUndo && event.command === 'Undo' && ! this.editor.undoManager.hasUndo() ) {
// When user attempts Undo when empty Undo stack, propagate undo
// action to context handler. The compromise here is that: TinyMCE
// handles Undo until change, at which point `editor.save` resets
// history. If no history exists, let context handler have a turn.
// Defer in case an immediate undo causes TinyMCE to be destroyed,
// if other undo behaviors test presence of an input field.
defer( onUndo );

// We could return false here to stop other TinyMCE event handlers
// from running, but we assume TinyMCE won't do anything on an
// empty undo stack anyways.
}
}

onPastePostProcess( event ) {
const childNodes = Array.from( event.node.childNodes );
const isBlockDelimiter = ( node ) =>
Expand Down Expand Up @@ -514,3 +543,7 @@ export default class Editable extends Component {
);
}
}

Editable.contextTypes = {
onUndo: noop,
};
34 changes: 34 additions & 0 deletions blocks/editable/provider.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
/**
* External dependencies
*/
import { pick, noop } from 'lodash';

/**
* WordPress dependencies
*/
import { Component } from 'element';

/**
* The Editable Provider allows a rendering context to define global behaviors
* without requiring intermediate props to be passed through to the Editable.
* The provider accepts as props its `childContextTypes` which are passed to
* any Editable instance.
*/
class EditableProvider extends Component {
getChildContext() {
return pick(
this.props,
Object.keys( this.constructor.childContextTypes )
);
}

render() {
return this.props.children;
}
}

EditableProvider.childContextTypes = {
onUndo: noop,
};

export default EditableProvider;
1 change: 1 addition & 0 deletions blocks/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,5 +18,6 @@ export { default as BlockControls } from './block-controls';
export { default as BlockDescription } from './block-description';
export { default as BlockIcon } from './block-icon';
export { default as Editable } from './editable';
export { default as EditableProvider } from './editable/provider';
export { default as InspectorControls } from './inspector-controls';
export { default as MediaUploadButton } from './media-upload-button';
12 changes: 10 additions & 2 deletions editor/index.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
/**
* External dependencies
*/
import { bindActionCreators } from 'redux';
import { Provider as ReduxProvider } from 'react-redux';
import { Provider as SlotFillProvider } from 'react-slot-fill';
import moment from 'moment-timezone';
Expand All @@ -9,7 +10,7 @@ import 'moment-timezone/moment-timezone-utils';
/**
* WordPress dependencies
*/
import { parse } from 'blocks';
import { EditableProvider, parse } from 'blocks';
import { render } from 'element';
import { settings } from 'date';

Expand All @@ -19,6 +20,7 @@ import { settings } from 'date';
import './assets/stylesheets/main.scss';
import Layout from './layout';
import { createReduxStore } from './state';
import { undo } from './actions';

// Configure moment globally
moment.locale( settings.l10n.locale );
Expand Down Expand Up @@ -86,7 +88,13 @@ export function createEditorInstance( id, post ) {
render(
<ReduxProvider store={ store }>
<SlotFillProvider>
<Layout />
<EditableProvider {
...bindActionCreators( {
onUndo: undo,
}, store.dispatch ) }
>
<Layout />
</EditableProvider>
</SlotFillProvider>
</ReduxProvider>,
document.getElementById( id )
Expand Down

0 comments on commit 3c68e0f

Please sign in to comment.