From 1398753469d0212d4b35fdc6ab275efd82f7e4ec Mon Sep 17 00:00:00 2001 From: Riad Benguella Date: Tue, 28 Nov 2017 10:05:04 +0100 Subject: [PATCH] Framework: Adding the possibility to lock the editor --- blocks/library/heading/index.js | 18 +++-- blocks/library/list/index.js | 34 ++++---- blocks/library/paragraph/index.js | 17 ++-- editor/components/block-list/block.js | 34 +++++--- editor/components/block-mover/index.js | 77 +++++++++++-------- editor/components/block-mover/test/index.js | 5 ++ .../block-delete-button.js | 31 +++++--- editor/components/block-switcher/index.js | 41 ++++++---- .../editor-global-keyboard-shortcuts/index.js | 43 +++++++---- editor/components/inserter/index.js | 6 +- .../edit-post/modes/visual-editor/inserter.js | 30 ++++++-- lib/client-assets.php | 3 +- 12 files changed, 214 insertions(+), 125 deletions(-) diff --git a/blocks/library/heading/index.js b/blocks/library/heading/index.js index 972638871d5929..9fbca6f3c4bdae 100644 --- a/blocks/library/heading/index.js +++ b/blocks/library/heading/index.js @@ -151,13 +151,17 @@ registerBlockType( 'core/heading', { onFocus={ setFocus } onChange={ ( value ) => setAttributes( { content: value } ) } onMerge={ mergeBlocks } - onSplit={ ( before, after, ...blocks ) => { - setAttributes( { content: before } ); - insertBlocksAfter( [ - ...blocks, - createBlock( 'core/paragraph', { content: after } ), - ] ); - } } + onSplit={ + insertBlocksAfter ? + ( before, after, ...blocks ) => { + setAttributes( { content: before } ); + insertBlocksAfter( [ + ...blocks, + createBlock( 'core/paragraph', { content: after } ), + ] ); + } : + undefined + } style={ { textAlign: align } } placeholder={ placeholder || __( 'Write heading…' ) } />, diff --git a/blocks/library/list/index.js b/blocks/library/list/index.js index ead16de6ce29c6..724a43cf6d29e5 100644 --- a/blocks/library/list/index.js +++ b/blocks/library/list/index.js @@ -332,21 +332,25 @@ registerBlockType( 'core/list', { wrapperClassName="blocks-list" placeholder={ __( 'Write list…' ) } onMerge={ mergeBlocks } - onSplit={ ( before, after, ...blocks ) => { - if ( ! blocks.length ) { - blocks.push( createBlock( 'core/paragraph' ) ); - } - - if ( after.length ) { - blocks.push( createBlock( 'core/list', { - nodeName, - values: after, - } ) ); - } - - setAttributes( { values: before } ); - insertBlocksAfter( blocks ); - } } + onSplit={ + insertBlocksAfter ? + ( before, after, ...blocks ) => { + if ( ! blocks.length ) { + blocks.push( createBlock( 'core/paragraph' ) ); + } + + if ( after.length ) { + blocks.push( createBlock( 'core/list', { + nodeName, + values: after, + } ) ); + } + + setAttributes( { values: before } ); + insertBlocksAfter( blocks ); + } : + undefined + } />, ]; } diff --git a/blocks/library/paragraph/index.js b/blocks/library/paragraph/index.js index 88fc473f0c6a86..983661ce524466 100644 --- a/blocks/library/paragraph/index.js +++ b/blocks/library/paragraph/index.js @@ -170,13 +170,16 @@ class ParagraphBlock extends Component { } } focus={ focus } onFocus={ setFocus } - onSplit={ ( before, after, ...blocks ) => { - setAttributes( { content: before } ); - insertBlocksAfter( [ - ...blocks, - createBlock( 'core/paragraph', { content: after } ), - ] ); - } } + onSplit={ insertBlocksAfter ? + ( before, after, ...blocks ) => { + setAttributes( { content: before } ); + insertBlocksAfter( [ + ...blocks, + createBlock( 'core/paragraph', { content: after } ), + ] ); + } : + undefined + } onMerge={ mergeBlocks } onReplace={ onReplace } placeholder={ placeholder || __( 'Add text or type / to insert content' ) } diff --git a/editor/components/block-list/block.js b/editor/components/block-list/block.js index 9117bd3212677d..51ee229845e33d 100644 --- a/editor/components/block-list/block.js +++ b/editor/components/block-list/block.js @@ -3,7 +3,7 @@ */ import { connect } from 'react-redux'; import classnames from 'classnames'; -import { get, partial, reduce, size } from 'lodash'; +import { get, partial, reduce, size, flow } from 'lodash'; /** * WordPress dependencies @@ -13,6 +13,7 @@ import { keycodes } from '@wordpress/utils'; import { getBlockType, BlockEdit, getBlockDefaultClassname, createBlock, hasBlockSupport } from '@wordpress/blocks'; import { withFilters } from '@wordpress/components'; import { __, sprintf } from '@wordpress/i18n'; +import { withContext } from '@wordpress/components'; /** * Internal dependencies @@ -294,7 +295,7 @@ class BlockListBlock extends Component { case ENTER: // Insert default block after current block if enter and event // not already handled by descendant. - if ( target === this.node ) { + if ( target === this.node && ! this.props.isLocked ) { event.preventDefault(); this.props.onInsertBlocks( [ @@ -316,12 +317,14 @@ class BlockListBlock extends Component { case DELETE: // Remove block on backspace. if ( target === this.node ) { + const { uid, onRemove, previousBlock, onFocus, isLocked } = this.props; event.preventDefault(); - const { uid, onRemove, previousBlock, onFocus } = this.props; - onRemove( uid ); + if ( ! isLocked ) { + onRemove( uid ); - if ( previousBlock ) { - onFocus( previousBlock.uid, { offset: -1 } ); + if ( previousBlock ) { + onFocus( previousBlock.uid, { offset: -1 } ); + } } } break; @@ -338,7 +341,7 @@ class BlockListBlock extends Component { } render() { - const { block, order, mode, showContextualToolbar } = this.props; + const { block, order, mode, showContextualToolbar, isLocked } = this.props; const { name: blockName, isValid } = block; const blockType = getBlockType( blockName ); // translators: %s: Type of block (i.e. Text, Image etc) @@ -408,10 +411,10 @@ class BlockListBlock extends Component { focus={ focus } attributes={ block.attributes } setAttributes={ this.setAttributes } - insertBlocksAfter={ this.insertBlocksAfter } - onReplace={ onReplace } + insertBlocksAfter={ isLocked ? undefined : this.insertBlocksAfter } + onReplace={ isLocked ? undefined : onReplace } setFocus={ partial( onFocus, block.uid ) } - mergeBlocks={ this.mergeBlocks } + mergeBlocks={ isLocked ? undefined : this.mergeBlocks } className={ className } id={ block.uid } /> @@ -514,7 +517,14 @@ const mapDispatchToProps = ( dispatch, ownProps ) => ( { }, } ); -export default compose( +export default flow( withFilters( 'Editor.BlockItem' ), - connect( mapStateToProps, mapDispatchToProps ) + connect( mapStateToProps, mapDispatchToProps ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + + return { + isLocked: true === templateLock, + }; + } ), )( BlockListBlock ); diff --git a/editor/components/block-mover/index.js b/editor/components/block-mover/index.js index 0d3430f26b4376..55248e0de6ba39 100644 --- a/editor/components/block-mover/index.js +++ b/editor/components/block-mover/index.js @@ -2,13 +2,13 @@ * External dependencies */ import { connect } from 'react-redux'; -import { first, last } from 'lodash'; +import { first, last, flow } from 'lodash'; /** * WordPress dependencies */ import { __ } from '@wordpress/i18n'; -import { IconButton } from '@wordpress/components'; +import { IconButton, withContext } from '@wordpress/components'; import { getBlockType } from '@wordpress/blocks'; /** @@ -19,7 +19,11 @@ import { getBlockMoverLabel } from './mover-label'; import { isFirstBlock, isLastBlock, getBlockIndex, getBlock } from '../../selectors'; import { selectBlock } from '../../actions'; -export function BlockMover( { onMoveUp, onMoveDown, isFirst, isLast, uids, blockType, firstIndex } ) { +export function BlockMover( { onMoveUp, onMoveDown, isFirst, isLast, uids, blockType, firstIndex, isLocked } ) { + if ( isLocked ) { + return null; + } + // We emulate a disabled state because forcefully applying the `disabled` // attribute on the button while it has focus causes the screen to change // to an unfocused state (body as active element) without firing blur on, @@ -60,37 +64,46 @@ export function BlockMover( { onMoveUp, onMoveDown, isFirst, isLast, uids, block ); } -export default connect( - ( state, ownProps ) => { - const block = getBlock( state, first( ownProps.uids ) ); - - return ( { - isFirst: isFirstBlock( state, first( ownProps.uids ) ), - isLast: isLastBlock( state, last( ownProps.uids ) ), - firstIndex: getBlockIndex( state, first( ownProps.uids ) ), - blockType: block ? getBlockType( block.name ) : null, - } ); - }, - ( dispatch, ownProps ) => ( { - onMoveDown() { - if ( ownProps.uids.length === 1 ) { - dispatch( selectBlock( first( ownProps.uids ) ) ); - } +export default flow( + connect( + ( state, ownProps ) => { + const block = getBlock( state, first( ownProps.uids ) ); - dispatch( { - type: 'MOVE_BLOCKS_DOWN', - uids: ownProps.uids, + return ( { + isFirst: isFirstBlock( state, first( ownProps.uids ) ), + isLast: isLastBlock( state, last( ownProps.uids ) ), + firstIndex: getBlockIndex( state, first( ownProps.uids ) ), + blockType: block ? getBlockType( block.name ) : null, } ); }, - onMoveUp() { - if ( ownProps.uids.length === 1 ) { - dispatch( selectBlock( first( ownProps.uids ) ) ); - } + ( dispatch, ownProps ) => ( { + onMoveDown() { + if ( ownProps.uids.length === 1 ) { + dispatch( selectBlock( first( ownProps.uids ) ) ); + } - dispatch( { - type: 'MOVE_BLOCKS_UP', - uids: ownProps.uids, - } ); - }, - } ) + dispatch( { + type: 'MOVE_BLOCKS_DOWN', + uids: ownProps.uids, + } ); + }, + onMoveUp() { + if ( ownProps.uids.length === 1 ) { + dispatch( selectBlock( first( ownProps.uids ) ) ); + } + + dispatch( { + type: 'MOVE_BLOCKS_UP', + uids: ownProps.uids, + } ); + }, + } ) + ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + + return { + isLocked: true === templateLock, + }; + } ), )( BlockMover ); diff --git a/editor/components/block-mover/test/index.js b/editor/components/block-mover/test/index.js index 32d0c221624f30..f6f785b24b7e14 100644 --- a/editor/components/block-mover/test/index.js +++ b/editor/components/block-mover/test/index.js @@ -16,6 +16,11 @@ describe( 'BlockMover', () => { title: 'yolo-block', }; + it( 'should not render if the editor is locked', () => { + const wrapper = shallow( ); + expect( wrapper.type() ).toBe( null ); + } ); + it( 'should render two IconButton components with the following props', () => { const blockMover = shallow( ); expect( blockMover.hasClass( 'editor-block-mover' ) ).toBe( true ); diff --git a/editor/components/block-settings-menu/block-delete-button.js b/editor/components/block-settings-menu/block-delete-button.js index 57ed7e0d3f8a30..29f301850d52c9 100644 --- a/editor/components/block-settings-menu/block-delete-button.js +++ b/editor/components/block-settings-menu/block-delete-button.js @@ -8,14 +8,18 @@ import { flow, noop } from 'lodash'; * WordPress dependencies */ import { __ } from '@wordpress/i18n'; -import { IconButton } from '@wordpress/components'; +import { IconButton, withContext } from '@wordpress/components'; /** * Internal dependencies */ import { removeBlocks } from '../../actions'; -export function BlockDeleteButton( { onDelete, onClick = noop, small = false } ) { +export function BlockDeleteButton( { onDelete, onClick = noop, isLocked, small = false } ) { + if ( isLocked ) { + return null; + } + const label = __( 'Delete' ); return ( @@ -30,11 +34,20 @@ export function BlockDeleteButton( { onDelete, onClick = noop, small = false } ) ); } -export default connect( - undefined, - ( dispatch, ownProps ) => ( { - onDelete() { - dispatch( removeBlocks( ownProps.uids ) ); - }, - } ) +export default flow( + connect( + undefined, + ( dispatch, ownProps ) => ( { + onDelete() { + dispatch( removeBlocks( ownProps.uids ) ); + }, + } ) + ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + + return { + isLocked: true === templateLock, + }; + } ), )( BlockDeleteButton ); diff --git a/editor/components/block-switcher/index.js b/editor/components/block-switcher/index.js index 54be221ea1e9ec..9ece4a356c369b 100644 --- a/editor/components/block-switcher/index.js +++ b/editor/components/block-switcher/index.js @@ -2,13 +2,13 @@ * External dependencies */ import { connect } from 'react-redux'; -import { every, uniq, get, reduce, find } from 'lodash'; +import { every, uniq, get, reduce, find, flow } from 'lodash'; /** * WordPress dependencies */ import { __ } from '@wordpress/i18n'; -import { Dropdown, Dashicon, IconButton, Toolbar, NavigableMenu } from '@wordpress/components'; +import { Dropdown, Dashicon, IconButton, Toolbar, NavigableMenu, withContext } from '@wordpress/components'; import { getBlockType, getBlockTypes, switchToBlockType, BlockIcon } from '@wordpress/blocks'; import { keycodes } from '@wordpress/utils'; @@ -24,8 +24,8 @@ import { getBlock } from '../../selectors'; */ const { DOWN } = keycodes; -function BlockSwitcher( { blocks, onTransform } ) { - if ( ! blocks || ! blocks[ 0 ] ) { +function BlockSwitcher( { blocks, onTransform, isLocked } ) { + if ( ! blocks || ! blocks[ 0 ] || isLocked ) { return null; } const isMultiBlock = blocks.length > 1; @@ -126,18 +126,27 @@ function BlockSwitcher( { blocks, onTransform } ) { ); } -export default connect( - ( state, ownProps ) => { +export default flow( + connect( + ( state, ownProps ) => { + return { + blocks: ownProps.uids.map( ( uid ) => getBlock( state, uid ) ), + }; + }, + ( dispatch, ownProps ) => ( { + onTransform( blocks, name ) { + dispatch( replaceBlocks( + ownProps.uids, + switchToBlockType( blocks, name ) + ) ); + }, + } ) + ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + return { - blocks: ownProps.uids.map( ( uid ) => getBlock( state, uid ) ), + isLocked: true === templateLock, }; - }, - ( dispatch, ownProps ) => ( { - onTransform( blocks, name ) { - dispatch( replaceBlocks( - ownProps.uids, - switchToBlockType( blocks, name ) - ) ); - }, - } ) + } ), )( BlockSwitcher ); diff --git a/editor/components/editor-global-keyboard-shortcuts/index.js b/editor/components/editor-global-keyboard-shortcuts/index.js index a2c7c728cac748..aa543b21462f52 100644 --- a/editor/components/editor-global-keyboard-shortcuts/index.js +++ b/editor/components/editor-global-keyboard-shortcuts/index.js @@ -2,13 +2,13 @@ * External dependencies */ import { connect } from 'react-redux'; -import { first, last } from 'lodash'; +import { first, last, flow } from 'lodash'; /** * WordPress dependencies */ import { Component } from '@wordpress/element'; -import { KeyboardShortcuts } from '@wordpress/components'; +import { KeyboardShortcuts, withContext } from '@wordpress/components'; /** * Internal dependencies @@ -42,10 +42,12 @@ class EditorGlobalKeyboardShortcuts extends Component { } deleteSelectedBlocks( event ) { - const { multiSelectedBlockUids, onRemove } = this.props; + const { multiSelectedBlockUids, onRemove, isLocked } = this.props; if ( multiSelectedBlockUids.length ) { event.preventDefault(); - onRemove( multiSelectedBlockUids ); + if ( ! isLocked ) { + onRemove( multiSelectedBlockUids ); + } } } @@ -63,18 +65,27 @@ class EditorGlobalKeyboardShortcuts extends Component { } } -export default connect( - ( state ) => { +export default flow( + connect( + ( state ) => { + return { + uids: getBlockUids( state ), + multiSelectedBlockUids: getMultiSelectedBlockUids( state ), + }; + }, + { + clearSelectedBlock, + onMultiSelect: multiSelect, + onRedo: redo, + onUndo: undo, + onRemove: removeBlocks, + } + ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + return { - uids: getBlockUids( state ), - multiSelectedBlockUids: getMultiSelectedBlockUids( state ), + isLocked: true === templateLock, }; - }, - { - clearSelectedBlock, - onMultiSelect: multiSelect, - onRedo: redo, - onUndo: undo, - onRemove: removeBlocks, - } + } ), )( EditorGlobalKeyboardShortcuts ); diff --git a/editor/components/inserter/index.js b/editor/components/inserter/index.js index 1c5ecd4b82d6be..ebc55ac04f8a92 100644 --- a/editor/components/inserter/index.js +++ b/editor/components/inserter/index.js @@ -63,9 +63,10 @@ class Inserter extends Component { onInsertBlock, insertionPoint, hasSupportedBlocks, + isLocked, } = this.props; - if ( ! hasSupportedBlocks ) { + if ( ! hasSupportedBlocks || isLocked ) { return null; } @@ -127,10 +128,11 @@ export default flowRight( [ } ) ), withContext( 'editor' )( ( settings ) => { - const { blockTypes } = settings; + const { blockTypes, templateLock } = settings; return { hasSupportedBlocks: true === blockTypes || ! isEmpty( blockTypes ), + isLocked: true === templateLock, }; } ), ] )( Inserter ); diff --git a/editor/edit-post/modes/visual-editor/inserter.js b/editor/edit-post/modes/visual-editor/inserter.js index ad2ae85eeacb28..7fdab6538ef11d 100644 --- a/editor/edit-post/modes/visual-editor/inserter.js +++ b/editor/edit-post/modes/visual-editor/inserter.js @@ -3,12 +3,13 @@ */ import { connect } from 'react-redux'; import classnames from 'classnames'; +import { flow } from 'lodash'; /** * WordPress dependencies */ import { __, sprintf } from '@wordpress/i18n'; -import { IconButton } from '@wordpress/components'; +import { IconButton, withContext } from '@wordpress/components'; import { Component } from '@wordpress/element'; import { createBlock, BlockIcon } from '@wordpress/blocks'; @@ -41,13 +42,17 @@ export class VisualEditorInserter extends Component { } render() { - const { blockCount } = this.props; + const { blockCount, isLocked } = this.props; const { isShowingControls } = this.state; const { mostFrequentlyUsedBlocks } = this.props; const classes = classnames( 'editor-visual-editor__inserter', { 'is-showing-controls': isShowingControls, } ); + if ( isLocked ) { + return null; + } + return (
{ +export default flow( + connect( + ( state ) => { + return { + mostFrequentlyUsedBlocks: getMostFrequentlyUsedBlocks( state ), + blockCount: getBlockCount( state ), + }; + }, + { onInsertBlock: insertBlock }, + ), + withContext( 'editor' )( ( settings ) => { + const { templateLock } = settings; + return { - mostFrequentlyUsedBlocks: getMostFrequentlyUsedBlocks( state ), - blockCount: getBlockCount( state ), + isLocked: true === templateLock, }; - }, - { onInsertBlock: insertBlock }, + } ), )( VisualEditorInserter ); diff --git a/lib/client-assets.php b/lib/client-assets.php index ded267509348b2..20726eb0cf4f21 100644 --- a/lib/client-assets.php +++ b/lib/client-assets.php @@ -790,7 +790,8 @@ function gutenberg_editor_scripts_and_styles( $hook ) { $post_type_object = get_post_type_object( $post_to_edit['type'] ); if ( ! empty( $post_type_object->template ) ) { - $editor_settings['template'] = $post_type_object->template; + $editor_settings['template'] = $post_type_object->template; + $editor_settings['templateLock'] = ! empty( $post_type_object->template_lock ) && $post_type_object->template_lock; } $script = '( function() {';