From e11ca0ad49fa19c86b7e3474fdf31652fc4280df Mon Sep 17 00:00:00 2001 From: Kamil Piechaczek Date: Thu, 27 Sep 2018 14:14:02 +0200 Subject: [PATCH 1/4] model#deleteContent() will proper merge elements inside limit element. --- src/model/utils/deletecontent.js | 6 ++++++ tests/model/utils/deletecontent.js | 26 ++++++++++++++++++++++++++ 2 files changed, 32 insertions(+) diff --git a/src/model/utils/deletecontent.js b/src/model/utils/deletecontent.js index d3a627027..e367ff7ff 100644 --- a/src/model/utils/deletecontent.js +++ b/src/model/utils/deletecontent.js @@ -139,11 +139,17 @@ function mergeBranches( writer, startPos, endPos ) { startPos = Position.createAfter( startParent ); endPos = Position.createBefore( endParent ); + // One more check if both positions ended up in the same parent. See: https://github.com/ckeditor/ckeditor5/issues/1265. + if ( endParent == startPos.parent ) { + return; + } + if ( !endPos.isEqual( startPos ) ) { // In this case, before we merge, we need to move `endParent` to the `startPos`: // x[]{}y // becomes: // x[]y{} + writer.insert( endParent, startPos ); } diff --git a/tests/model/utils/deletecontent.js b/tests/model/utils/deletecontent.js index befef6e58..e6813a2e6 100644 --- a/tests/model/utils/deletecontent.js +++ b/tests/model/utils/deletecontent.js @@ -772,6 +772,10 @@ describe( 'DataController utils', () => { schema.extend( '$block', { allowIn: 'blockLimit' } ); schema.register( 'paragraph', { inheritAllFrom: '$block' } ); + schema.register( 'blockQuote', { + allowWhere: '$block', + allowContentOf: '$root' + } ); } ); test( @@ -804,6 +808,28 @@ describe( 'DataController utils', () => { 'foo [barbaz] qux', 'foo [] qux' ); + + // See: https://github.com/ckeditor/ckeditor5/issues/1265. + it( 'should proper merge two elements which are inside limit element', () => { + setData( model, + '' + + '
' + + 'Foo' + + '
' + + '[]Bar' + + '
' + ); + + model.modifySelection( doc.selection, { direction: 'backward' } ); + deleteContent( model, doc.selection ); + + expect( getData( model ) ).to.equal( + '' + + '
' + + 'Foo[]Bar' + + '
' + + '
' ); + } ); } ); describe( 'should leave a paragraph if the entire content was selected', () => { From 6bdb715bef44d06329ebd2493d608c81d377e604 Mon Sep 17 00:00:00 2001 From: Kamil Piechaczek Date: Thu, 27 Sep 2018 14:16:44 +0200 Subject: [PATCH 2/4] Removed invalid blank line. --- src/model/utils/deletecontent.js | 1 - 1 file changed, 1 deletion(-) diff --git a/src/model/utils/deletecontent.js b/src/model/utils/deletecontent.js index e367ff7ff..2a8996d50 100644 --- a/src/model/utils/deletecontent.js +++ b/src/model/utils/deletecontent.js @@ -149,7 +149,6 @@ function mergeBranches( writer, startPos, endPos ) { // x[]{}y // becomes: // x[]y{} - writer.insert( endParent, startPos ); } From a391790c722519d7ffa192cc8f2e29dbcadbf8fe Mon Sep 17 00:00:00 2001 From: Kamil Piechaczek Date: Wed, 3 Oct 2018 10:56:42 +0200 Subject: [PATCH 3/4] Changed a way how to resolve the problem. --- src/model/utils/deletecontent.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/model/utils/deletecontent.js b/src/model/utils/deletecontent.js index 2a8996d50..b794e69c8 100644 --- a/src/model/utils/deletecontent.js +++ b/src/model/utils/deletecontent.js @@ -139,11 +139,6 @@ function mergeBranches( writer, startPos, endPos ) { startPos = Position.createAfter( startParent ); endPos = Position.createBefore( endParent ); - // One more check if both positions ended up in the same parent. See: https://github.com/ckeditor/ckeditor5/issues/1265. - if ( endParent == startPos.parent ) { - return; - } - if ( !endPos.isEqual( startPos ) ) { // In this case, before we merge, we need to move `endParent` to the `startPos`: // x[]{}y @@ -184,19 +179,24 @@ function shouldAutoparagraph( schema, position ) { // Check if parents of two positions can be merged by checking if there are no limit/object // boundaries between those two positions. // +// Merge also will not be allowed if only single element occurs between specified positions (`leftPos` and `rightPos`). +// // E.g. in

x[]

{} // we'll check

, , and . // Usually, widget and caption are marked as objects/limits in the schema, so in this case merging will be blocked. function checkCanBeMerged( leftPos, rightPos, schema ) { const rangeToCheck = new Range( leftPos, rightPos ); + let countElement = 0; for ( const value of rangeToCheck.getWalker() ) { + countElement += 1; + if ( schema.isLimit( value.item ) ) { return false; } } - return true; + return countElement > 1; } function insertParagraph( writer, position, selection ) { From 8d41bbd76b6a08048e3d0e76c6c48ca559bd57bb Mon Sep 17 00:00:00 2001 From: Kamil Piechaczek Date: Fri, 5 Oct 2018 08:28:42 +0200 Subject: [PATCH 4/4] Fixed a bug in a proper way an added one more test. --- src/model/utils/deletecontent.js | 13 +++--------- tests/model/utils/deletecontent.js | 32 ++++++++++++++++++++++++++++++ 2 files changed, 35 insertions(+), 10 deletions(-) diff --git a/src/model/utils/deletecontent.js b/src/model/utils/deletecontent.js index b794e69c8..7dfa28579 100644 --- a/src/model/utils/deletecontent.js +++ b/src/model/utils/deletecontent.js @@ -118,10 +118,8 @@ function mergeBranches( writer, startPos, endPos ) { return; } - // If one of the positions is a root, then there's nothing more to merge (at least in the current state of implementation). - // Theoretically in this case we could unwrap the

: <$root>x[]

{}y

, but we don't need to support it yet - // so let's just abort. - if ( !startParent.parent || !endParent.parent ) { + // If one of the positions is a limit element, then there's nothing to merge because we don't want to cross the limit boundaries. + if ( writer.model.schema.isLimit( startParent ) || writer.model.schema.isLimit( endParent ) ) { return; } @@ -179,24 +177,19 @@ function shouldAutoparagraph( schema, position ) { // Check if parents of two positions can be merged by checking if there are no limit/object // boundaries between those two positions. // -// Merge also will not be allowed if only single element occurs between specified positions (`leftPos` and `rightPos`). -// // E.g. in

x[]

{} // we'll check

, , and . // Usually, widget and caption are marked as objects/limits in the schema, so in this case merging will be blocked. function checkCanBeMerged( leftPos, rightPos, schema ) { const rangeToCheck = new Range( leftPos, rightPos ); - let countElement = 0; for ( const value of rangeToCheck.getWalker() ) { - countElement += 1; - if ( schema.isLimit( value.item ) ) { return false; } } - return countElement > 1; + return true; } function insertParagraph( writer, position, selection ) { diff --git a/tests/model/utils/deletecontent.js b/tests/model/utils/deletecontent.js index e6813a2e6..463d93880 100644 --- a/tests/model/utils/deletecontent.js +++ b/tests/model/utils/deletecontent.js @@ -830,6 +830,38 @@ describe( 'DataController utils', () => { '' + '' ); } ); + + it( 'should proper merge elements which are inside limit element (nested elements)', () => { + setData( model, + '

' + + '' + + '
' + + 'Foo.' + + '
' + + 'Foo' + + '
' + + '
' + + '[]Bar' + + '
' + + '
' + ); + + model.modifySelection( doc.selection, { direction: 'backward' } ); + deleteContent( model, doc.selection ); + + expect( getData( model ) ).to.equal( + '
' + + '' + + '
' + + 'Foo.' + + '
' + + 'Foo[]Bar' + + '
' + + '
' + + '
' + + '
' + ); + } ); } ); describe( 'should leave a paragraph if the entire content was selected', () => {