From 1a20e15a615ecb344cfb0dd8b614982910240909 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Est=C3=AAv=C3=A3o?= Date: Thu, 23 May 2019 10:31:39 +0100 Subject: [PATCH] Implement support for nested lists on GB-mobile. (#15566) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Add indent and outdent commands. * Add code for handling delete on nested lists. * Handle case when full removal of the content happens. * Update record creation and change propagation. * If already processed don’t go and delete anything more. * Remove unused/invalid method. * Update documentation. * No longer needed dedicate list-edit for native. --- .../src/components/rich-text/index.native.js | 94 ++++++++++++++++--- .../components/rich-text/list-edit.native.js | 55 ----------- 2 files changed, 79 insertions(+), 70 deletions(-) delete mode 100644 packages/block-editor/src/components/rich-text/list-edit.native.js diff --git a/packages/block-editor/src/components/rich-text/index.native.js b/packages/block-editor/src/components/rich-text/index.native.js index d47ace5c39144b..7431817743a6e9 100644 --- a/packages/block-editor/src/components/rich-text/index.native.js +++ b/packages/block-editor/src/components/rich-text/index.native.js @@ -26,9 +26,11 @@ import { split, toHTMLString, insert, + __UNSTABLE_LINE_SEPARATOR as LINE_SEPARATOR, __unstableInsertLineSeparator as insertLineSeparator, __unstableIsEmptyLine as isEmptyLine, isCollapsed, + remove, } from '@wordpress/rich-text'; import { decodeEntities } from '@wordpress/html-entities'; import { BACKSPACE } from '@wordpress/keycodes'; @@ -104,7 +106,6 @@ export class RichText extends Component { this.onBlur = this.onBlur.bind( this ); this.onTextUpdate = this.onTextUpdate.bind( this ); this.onContentSizeChange = this.onContentSizeChange.bind( this ); - this.onFormatChangeForceChild = this.onFormatChangeForceChild.bind( this ); this.onFormatChange = this.onFormatChange.bind( this ); this.formatToValue = memize( this.formatToValue.bind( this ), @@ -157,14 +158,9 @@ export class RichText extends Component { } /** - * Creates a RichText value "record" from native content and selection + * Creates a RichText value "record" from the current content and selection * information * - * @param {string} currentContent The content (usually an HTML string) from - * the native component. - * @param {number} selectionStart The start of the selection. - * @param {number} selectionEnd The end of the selection (same as start if - * cursor instead of selection). * * @return {Object} A RichText value with formats and selection. */ @@ -256,10 +252,6 @@ export class RichText extends Component { } ).map( ( name ) => gutenbergFormatNamesToAztec[ name ] ).filter( Boolean ); } - onFormatChangeForceChild( record ) { - this.onFormatChange( record, true ); - } - onFormatChange( record ) { const { start, end, activeFormats = [] } = record; const changeHandlers = pickBy( this.props, ( v, key ) => @@ -362,18 +354,18 @@ export class RichText extends Component { if ( event.shiftKey ) { this.needsSelectionUpdate = true; const insertedLineBreak = { ...insert( currentRecord, '\n' ) }; - this.onFormatChangeForceChild( insertedLineBreak ); + this.onFormatChange( insertedLineBreak ); } else if ( this.onSplit && isEmptyLine( currentRecord ) ) { this.onSplit( currentRecord ); } else { this.needsSelectionUpdate = true; const insertedLineSeparator = { ...insertLineSeparator( currentRecord ) }; - this.onFormatChange( insertedLineSeparator, ! this.firedAfterTextChanged ); + this.onFormatChange( insertedLineSeparator ); } } else if ( event.shiftKey || ! this.onSplit ) { this.needsSelectionUpdate = true; const insertedLineBreak = { ...insert( currentRecord, '\n' ) }; - this.onFormatChangeForceChild( insertedLineBreak ); + this.onFormatChange( insertedLineBreak ); } else { this.onSplit( currentRecord ); } @@ -390,6 +382,78 @@ export class RichText extends Component { const keyCode = BACKSPACE; // TODO : should we differentiate BACKSPACE and DELETE? const isReverse = keyCode === BACKSPACE; + this.lastEventCount = event.nativeEvent.eventCount; + this.comesFromAztec = true; + this.firedAfterTextChanged = event.nativeEvent.firedAfterTextChanged; + const value = this.createRecord(); + const { replacements, text, start, end } = value; + let newValue; + + // Always handle full content deletion ourselves. + if ( start === 0 && end !== 0 && end >= value.text.length ) { + newValue = remove( value, start, end ); + this.props.onChange( newValue ); + this.forceSelectionUpdate( 0, 0 ); + return; + } + + if ( this.multilineTag ) { + if ( keyCode === BACKSPACE ) { + const index = start - 1; + + if ( text[ index ] === LINE_SEPARATOR ) { + const collapsed = isCollapsed( value ); + + // If the line separator that is about te be removed + // contains wrappers, remove the wrappers first. + if ( collapsed && replacements[ index ] && replacements[ index ].length ) { + const newReplacements = replacements.slice(); + + newReplacements[ index ] = replacements[ index ].slice( 0, -1 ); + newValue = { + ...value, + replacements: newReplacements, + }; + } else { + newValue = remove( + value, + // Only remove the line if the selection is + // collapsed, otherwise remove the selection. + collapsed ? start - 1 : start, + end + ); + } + } + } else if ( text[ end ] === LINE_SEPARATOR ) { + const collapsed = isCollapsed( value ); + + // If the line separator that is about te be removed + // contains wrappers, remove the wrappers first. + if ( collapsed && replacements[ end ] && replacements[ end ].length ) { + const newReplacements = replacements.slice(); + + newReplacements[ end ] = replacements[ end ].slice( 0, -1 ); + newValue = { + ...value, + replacements: newReplacements, + }; + } else { + newValue = remove( + value, + start, + // Only remove the line if the selection is + // collapsed, otherwise remove the selection. + collapsed ? end + 1 : end, + ); + } + } + + if ( newValue ) { + this.onFormatChange( newValue ); + return; + } + } + const empty = this.isEmpty(); if ( onMerge ) { @@ -764,7 +828,7 @@ export class RichText extends Component { onTagNameChange={ onTagNameChange } tagName={ tagName } value={ record } - onChange={ this.onFormatChangeForceChild } + onChange={ this.onFormatChange } /> ) } { isSelected && ( diff --git a/packages/block-editor/src/components/rich-text/list-edit.native.js b/packages/block-editor/src/components/rich-text/list-edit.native.js deleted file mode 100644 index 3f0cdf0fa6e349..00000000000000 --- a/packages/block-editor/src/components/rich-text/list-edit.native.js +++ /dev/null @@ -1,55 +0,0 @@ -/** - * WordPress dependencies - */ - -import { Toolbar } from '@wordpress/components'; -import { __ } from '@wordpress/i18n'; -import { - __unstableChangeListType as changeListType, - __unstableIsListRootSelected as isListRootSelected, - __unstableIsActiveListType as isActiveListType, -} from '@wordpress/rich-text'; - -/** - * Internal dependencies - */ - -import BlockFormatControls from '../block-format-controls'; - -export const ListEdit = ( { - onTagNameChange, - tagName, - value, - onChange, -} ) => ( - - - -);