diff --git a/packages/ckeditor5-image/src/imageresize/imageresizehandles.js b/packages/ckeditor5-image/src/imageresize/imageresizehandles.js
index c5e1725c60f..be553d0d68f 100644
--- a/packages/ckeditor5-image/src/imageresize/imageresizehandles.js
+++ b/packages/ckeditor5-image/src/imageresize/imageresizehandles.js
@@ -9,6 +9,7 @@
import Plugin from '@ckeditor/ckeditor5-core/src/plugin';
import WidgetResize from '@ckeditor/ckeditor5-widget/src/widgetresize';
+import ImageLoadObserver from '../image/imageloadobserver';
/**
* The image resize by handles feature.
@@ -37,21 +38,46 @@ export default class ImageResizeHandles extends Plugin {
* @inheritDoc
*/
init() {
+ const command = this.editor.commands.get( 'imageResize' );
+ this.bind( 'isEnabled' ).to( command );
+
+ this._setupResizerCreator();
+ }
+
+ /**
+ * Attaches the listeners responsible for creating a resizer for each image.
+ *
+ * @private
+ */
+ _setupResizerCreator() {
const editor = this.editor;
- const command = editor.commands.get( 'imageResize' );
+ const editingView = editor.editing.view;
- this.bind( 'isEnabled' ).to( command );
+ editingView.addObserver( ImageLoadObserver );
+
+ this.listenTo( editingView.document, 'imageLoaded', ( evt, domEvent ) => {
+ const imageView = editor.editing.view.domConverter.domToView( domEvent.target );
+ const widgetView = imageView.findAncestor( 'figure' );
+ let resizer = this.editor.plugins.get( WidgetResize ).getResizerByViewElement( widgetView );
+
+ if ( resizer ) {
+ // There are rare cases when image will be triggered multiple times for the same widget, e.g. when
+ // image's src was changed after upload (https://github.com/ckeditor/ckeditor5/pull/8108#issuecomment-708302992).
+ resizer.redraw();
+
+ return;
+ }
- editor.editing.downcastDispatcher.on( 'insert:image', ( evt, data, conversionApi ) => {
- const widget = conversionApi.mapper.toViewElement( data.item );
+ const mapper = editor.editing.mapper;
+ const imageModel = mapper.toModelElement( widgetView );
- const resizer = editor.plugins
+ resizer = editor.plugins
.get( WidgetResize )
.attachTo( {
unit: editor.config.get( 'image.resizeUnit' ),
- modelElement: data.item,
- viewElement: widget,
+ modelElement: imageModel,
+ viewElement: widgetView,
editor,
getHandleHost( domWidgetElement ) {
@@ -62,7 +88,7 @@ export default class ImageResizeHandles extends Plugin {
},
// TODO consider other positions.
isCentered() {
- const imageStyle = data.item.getAttribute( 'imageStyle' );
+ const imageStyle = imageModel.getAttribute( 'imageStyle' );
return !imageStyle || imageStyle == 'full' || imageStyle == 'alignCenter';
},
@@ -73,14 +99,14 @@ export default class ImageResizeHandles extends Plugin {
} );
resizer.on( 'updateSize', () => {
- if ( !widget.hasClass( 'image_resized' ) ) {
- editor.editing.view.change( writer => {
- writer.addClass( 'image_resized', widget );
+ if ( !widgetView.hasClass( 'image_resized' ) ) {
+ editingView.change( writer => {
+ writer.addClass( 'image_resized', widgetView );
} );
}
} );
resizer.bind( 'isEnabled' ).to( this );
- }, { priority: 'low' } );
+ } );
}
}
diff --git a/packages/ckeditor5-image/tests/imageresize/_utils/utils.js b/packages/ckeditor5-image/tests/imageresize/_utils/utils.js
new file mode 100644
index 00000000000..f4344aac024
--- /dev/null
+++ b/packages/ckeditor5-image/tests/imageresize/_utils/utils.js
@@ -0,0 +1,37 @@
+/**
+ * @license Copyright (c) 2003-2020, CKSource - Frederico Knabben. All rights reserved.
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+ */
+
+import ImageLoadObserver from '../../../src/image/imageloadobserver';
+
+// A 100x50 black png image
+export const IMAGE_SRC_FIXTURE = '' +
+ 'aM3g14MGNJMXKiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiJysRFNMgH0RpujAAAAAElFTkSuQmCC';
+
+export async function waitForAllImagesLoaded( editor ) {
+ // Returns a promise that waits for all the images in editor to be loaded.
+ // This is needed because resizers are created only after images are loaded.
+ const root = editor.model.document.getRoot();
+ const editingView = editor.editing.view;
+ const images = new Set();
+
+ for ( const curModel of root.getChildren() ) {
+ if ( curModel.is( 'element', 'image' ) ) {
+ const imageView = editor.editing.mapper.toViewElement( curModel );
+ images.add( editingView.domConverter.viewToDom( imageView ).querySelector( 'img' ) );
+ }
+ }
+
+ editingView.addObserver( ImageLoadObserver );
+
+ return new Promise( resolve => {
+ editingView.document.on( 'imageLoaded', ( evt, domEvent ) => {
+ images.delete( domEvent.target );
+
+ if ( images.size === 0 ) {
+ resolve();
+ }
+ } );
+ } );
+}
diff --git a/packages/ckeditor5-image/tests/imageresize/imageresizeediting.js b/packages/ckeditor5-image/tests/imageresize/imageresizeediting.js
index 963782b9f2d..649f9bc0299 100644
--- a/packages/ckeditor5-image/tests/imageresize/imageresizeediting.js
+++ b/packages/ckeditor5-image/tests/imageresize/imageresizeediting.js
@@ -18,12 +18,9 @@ import ImageStyle from '../../src/imagestyle';
import { setData } from '@ckeditor/ckeditor5-engine/src/dev-utils/model';
import { focusEditor } from '@ckeditor/ckeditor5-widget/tests/widgetresize/_utils/utils';
+import { IMAGE_SRC_FIXTURE } from './_utils/utils';
describe( 'ImageResizeEditing', () => {
- // 100x50 black png image
- const IMAGE_SRC_FIXTURE = '' +
- 'aM3g14MGNJMXKiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiJysRFNMgH0RpujAAAAAElFTkSuQmCC';
-
let editor, editorElement;
beforeEach( () => {
diff --git a/packages/ckeditor5-image/tests/imageresize/imageresizehandles.js b/packages/ckeditor5-image/tests/imageresize/imageresizehandles.js
index 833c8f120fe..b99b62bcd40 100644
--- a/packages/ckeditor5-image/tests/imageresize/imageresizehandles.js
+++ b/packages/ckeditor5-image/tests/imageresize/imageresizehandles.js
@@ -30,12 +30,9 @@ import {
} from '@ckeditor/ckeditor5-widget/tests/widgetresize/_utils/utils';
import WidgetResize from '@ckeditor/ckeditor5-widget/src/widgetresize';
+import { IMAGE_SRC_FIXTURE, waitForAllImagesLoaded } from './_utils/utils';
describe( 'ImageResizeHandles', () => {
- // 100x50 black png image
- const IMAGE_SRC_FIXTURE = '' +
- 'aM3g14MGNJMXKiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiJysRFNMgH0RpujAAAAAElFTkSuQmCC';
-
let widget, editor, view, viewDocument, editorElement;
beforeEach( () => {
@@ -62,7 +59,7 @@ describe( 'ImageResizeHandles', () => {
const attachToSpy = sinon.spy( localEditor.plugins.get( WidgetResize ), 'attachTo' );
- setData( localEditor.model, `[]` );
+ await setModelAndWaitForImages( localEditor, `[]` );
expect( attachToSpy.args[ 0 ][ 0 ] ).to.have.a.property( 'unit', '%' );
@@ -79,7 +76,7 @@ describe( 'ImageResizeHandles', () => {
it( 'uses the command on commit', async () => {
const spy = sinon.spy( editor.commands.get( 'imageResize' ), 'execute' );
- setData( editor.model, `foo[]` );
+ await setModelAndWaitForImages( editor, `foo[]` );
widget = viewDocument.getRoot().getChild( 1 );
const domParts = getWidgetDomParts( editor, widget, 'bottom-left' );
@@ -92,7 +89,7 @@ describe( 'ImageResizeHandles', () => {
} );
it( 'disables the resizer if the command is disabled', async () => {
- setData( editor.model, `foo[]` );
+ await setModelAndWaitForImages( editor, `foo[]` );
const resizer = getSelectedImageResizer( editor );
@@ -124,6 +121,8 @@ describe( 'ImageResizeHandles', () => {
editor.model.insertContent( writer.createElement( 'image', { src: IMAGE_SRC_FIXTURE } ) );
} );
+ await waitForAllImagesLoaded( editor );
+
const resizer = getSelectedImageResizer( editor );
const resizerWrapper = editor.ui.getEditableElement().querySelector( '.ck-widget__resizer' );
@@ -136,7 +135,8 @@ describe( 'ImageResizeHandles', () => {
beforeEach( async () => {
editor = await createEditor();
- setData( editor.model, `foo[]` );
+ await setModelAndWaitForImages( editor,
+ `foo[]` );
widget = viewDocument.getRoot().getChild( 1 );
} );
@@ -178,7 +178,7 @@ describe( 'ImageResizeHandles', () => {
beforeEach( async () => {
editor = await createEditor();
- setData( editor.model, `foo[]` );
+ await setModelAndWaitForImages( editor, `foo[]` );
widget = viewDocument.getRoot().getChild( 1 );
} );
@@ -200,9 +200,9 @@ describe( 'ImageResizeHandles', () => {
// Toggle _visibleResizer to force synchronous redraw. Otherwise you'd need to wait ~200ms for
// throttled redraw to take place, making tests slower.
- const visibleResizer = plugin._visibleResizer;
- plugin._visibleResizer = null;
- plugin._visibleResizer = visibleResizer;
+ for ( const [ , resizer ] of plugin._resizers.entries() ) {
+ resizer.redraw();
+ }
const resizerWrapper = document.querySelector( '.ck-widget__resizer' );
const shadowBoundingRect = resizerWrapper.getBoundingClientRect();
@@ -210,13 +210,37 @@ describe( 'ImageResizeHandles', () => {
expect( shadowBoundingRect.width ).to.equal( 100 );
expect( shadowBoundingRect.height ).to.equal( 50 );
} );
+
+ it( 'doesn\'t show resizers when undoing to multiple images', async () => {
+ // Based on https://github.com/ckeditor/ckeditor5/pull/8108#issuecomment-695949745.
+ await setModelAndWaitForImages( editor,
+ `[]` );
+
+ const paragraph = editor.model.change( writer => {
+ return writer.createElement( 'paragraph' );
+ } );
+ editor.model.insertContent( paragraph );
+
+ // Undo to go back to two, selected images.
+ editor.commands.get( 'undo' ).execute();
+
+ for ( let i = 0; i < 2; i++ ) {
+ widget = viewDocument.getRoot().getChild( i );
+ const domImage = getWidgetDomParts( editor, widget, 'bottom-right' ).widget.querySelector( 'img' );
+ viewDocument.fire( 'imageLoaded', { target: domImage } );
+
+ const domResizeWrapper = getWidgetDomParts( editor, widget, 'bottom-left' ).resizeWrapper;
+
+ expect( domResizeWrapper.getBoundingClientRect().height ).to.equal( 0 );
+ }
+ } );
} );
describe( 'table integration', () => {
it( 'works when resizing in a table', async () => {
editor = await createEditor();
- setData( editor.model,
+ await setModelAndWaitForImages( editor,
'
'
@@ -238,6 +262,36 @@ describe( 'ImageResizeHandles', () => {
} );
} );
+ it( 'doesn\'t create multiple resizers for a single image widget', async () => {
+ // https://github.com/ckeditor/ckeditor5/pull/8108#issuecomment-708302992
+ editor = await createEditor();
+ await setModelAndWaitForImages( editor, `[]` );
+ widget = viewDocument.getRoot().getChild( 0 );
+
+ const domParts = getWidgetDomParts( editor, widget, 'bottom-right' );
+ const alternativeImageFixture =
+ '';
+
+ // Change the image so that load event triggers for the same img element again.
+ domParts.widget.querySelector( 'img' ).src = alternativeImageFixture;
+ await waitForAllImagesLoaded( editor );
+
+ expect( domParts.widget.querySelectorAll( '.ck-widget__resizer' ).length ).to.equal( 1 );
+ } );
+
+ it( 'only creates a resizer after the image is loaded', async () => {
+ // https://github.com/ckeditor/ckeditor5/issues/8088
+ editor = await createEditor();
+ setData( editor.model, `[]` );
+ widget = viewDocument.getRoot().getChild( 0 );
+ const domParts = getWidgetDomParts( editor, widget, 'bottom-right' );
+
+ expect( domParts.widget.querySelectorAll( '.ck-widget__resizer' ).length ).to.equal( 0 );
+
+ await waitForAllImagesLoaded( editor );
+ expect( domParts.widget.querySelectorAll( '.ck-widget__resizer' ).length ).to.equal( 1 );
+ } );
+
describe( 'srcset integration', () => {
// The image is 96x96 pixels.
const imageBaseUrl = '/assets/sample.png';
@@ -272,11 +326,13 @@ describe( 'ImageResizeHandles', () => {
`
);
+ await waitForAllImagesLoaded( editor );
+
widget = viewDocument.getRoot().getChild( 0 );
model = editor.model.document.getRoot().getChild( 0 );
} );
- it( 'works with images containing srcset', () => {
+ it( 'works with images containing srcset', async () => {
const domParts = getWidgetDomParts( editor, widget, 'bottom-right' );
const initialPosition = getHandleCenterPoint( domParts.widget, 'bottom-right' );
const finalPointerPosition = initialPosition.clone().moveBy( -20, -20 );
@@ -289,7 +345,7 @@ describe( 'ImageResizeHandles', () => {
expect( model.getAttribute( 'width' ) ).to.equal( '76px' );
} );
- it( 'retains width after removing srcset', () => {
+ it( 'retains width after removing srcset', async () => {
const domParts = getWidgetDomParts( editor, widget, 'bottom-right' );
const initialPosition = getHandleCenterPoint( domParts.widget, 'bottom-right' );
const finalPointerPosition = initialPosition.clone().moveBy( -16, -16 );
@@ -339,7 +395,7 @@ describe( 'ImageResizeHandles', () => {
}
} );
- setData( editor.model, `foo[]` );
+ await setModelAndWaitForImages( editor, `foo[]` );
widget = viewDocument.getRoot().getChild( 1 );
@@ -395,7 +451,7 @@ describe( 'ImageResizeHandles', () => {
}
function getSelectedImageResizer( editor ) {
- return editor.plugins.get( 'WidgetResize' )._getResizerByViewElement(
+ return editor.plugins.get( 'WidgetResize' ).getResizerByViewElement(
editor.editing.view.document.selection.getSelectedElement()
);
}
@@ -415,4 +471,9 @@ describe( 'ImageResizeHandles', () => {
return newEditor;
} );
}
+
+ async function setModelAndWaitForImages( editor, data ) {
+ setData( editor.model, data );
+ return waitForAllImagesLoaded( editor );
+ }
} );
diff --git a/packages/ckeditor5-widget/src/widgetresize.js b/packages/ckeditor5-widget/src/widgetresize.js
index 4e9b4e9fd1e..1094f299e23 100644
--- a/packages/ckeditor5-widget/src/widgetresize.js
+++ b/packages/ckeditor5-widget/src/widgetresize.js
@@ -41,11 +41,10 @@ export default class WidgetResize extends Plugin {
/**
* The currently visible resizer.
*
- * @protected
* @observable
- * @member {module:widget/widgetresize/resizer~Resizer|null} #_visibleResizer
+ * @member {module:widget/widgetresize/resizer~Resizer|null} #visibleResizer
*/
- this.set( '_visibleResizer', null );
+ this.set( 'visibleResizer', null );
/**
* References an active resizer.
@@ -82,8 +81,8 @@ export default class WidgetResize extends Plugin {
this._observer.listenTo( domDocument, 'mouseup', this._mouseUpListener.bind( this ) );
const redrawFocusedResizer = () => {
- if ( this._visibleResizer ) {
- this._visibleResizer.redraw();
+ if ( this.visibleResizer ) {
+ this.visibleResizer.redraw();
}
};
@@ -91,7 +90,7 @@ export default class WidgetResize extends Plugin {
// Redraws occurring upon a change of visible resizer must not be throttled, as it is crucial for the initial
// render. Without it the resizer frame would be misaligned with resizing host for a fraction of second.
- this.on( 'change:_visibleResizer', redrawFocusedResizer );
+ this.on( 'change:visibleResizer', redrawFocusedResizer );
// Redrawing on any change of the UI of the editor (including content changes).
this.editor.ui.on( 'update', redrawFocusedResizerThrottled );
@@ -104,7 +103,7 @@ export default class WidgetResize extends Plugin {
viewSelection.on( 'change', () => {
const selectedElement = viewSelection.getSelectedElement();
- this._visibleResizer = this._getResizerByViewElement( selectedElement ) || null;
+ this.visibleResizer = this.getResizerByViewElement( selectedElement ) || null;
} );
}
@@ -149,9 +148,28 @@ export default class WidgetResize extends Plugin {
this._resizers.set( options.viewElement, resizer );
+ const viewSelection = this.editor.editing.view.document.selection;
+ const selectedElement = viewSelection.getSelectedElement();
+
+ // It could be that the element the resizer is created for is currently focused. In that
+ // case it should become visible.
+ if ( this.getResizerByViewElement( selectedElement ) == resizer ) {
+ this.visibleResizer = resizer;
+ }
+
return resizer;
}
+ /**
+ * Returns a resizer created for a given view element (widget element).
+ *
+ * @param {module:engine/view/containerelement~ContainerElement} viewElement View element associated with the resizer.
+ * @returns {module:widget/widgetresize/resizer~Resizer|undefined}
+ */
+ getResizerByViewElement( viewElement ) {
+ return this._resizers.get( viewElement );
+ }
+
/**
* Returns a resizer that contains a given resize handle.
*
@@ -167,17 +185,6 @@ export default class WidgetResize extends Plugin {
}
}
- /**
- * Returns a resizer created for a given view element (widget element).
- *
- * @protected
- * @param {module:engine/view/containerelement~ContainerElement} viewElement
- * @returns {module:widget/widgetresize/resizer~Resizer}
- */
- _getResizerByViewElement( viewElement ) {
- return this._resizers.get( viewElement );
- }
-
/**
* @protected
* @param {module:utils/eventinfo~EventInfo} event
diff --git a/packages/ckeditor5-widget/tests/widgetresize-integration.js b/packages/ckeditor5-widget/tests/widgetresize-integration.js
index 6b52d006411..353d9a2d810 100644
--- a/packages/ckeditor5-widget/tests/widgetresize-integration.js
+++ b/packages/ckeditor5-widget/tests/widgetresize-integration.js
@@ -11,6 +11,7 @@ import { setData as setModelData } from '@ckeditor/ckeditor5-engine/src/dev-util
import testUtils from '@ckeditor/ckeditor5-core/tests/_utils/utils';
import Image from '@ckeditor/ckeditor5-image/src/image';
import ImageResize from '@ckeditor/ckeditor5-image/src/imageresize';
+import { waitForAllImagesLoaded } from '@ckeditor/ckeditor5-image/tests/imageresize/_utils/utils';
describe( 'WidgetResize - integration', () => {
let editor, model, view, viewDocument, editorElement;
@@ -36,11 +37,13 @@ describe( 'WidgetResize - integration', () => {
return editor.destroy();
} );
- it( 'should not fire viewDocument#mousedown events after starting resizing', () => {
+ it( 'should not fire viewDocument#mousedown events after starting resizing', async () => {
const eventSpy = sinon.spy().named( 'ViewDocument#mousedown' );
setModelData( model, '[]' );
+ await waitForAllImagesLoaded( editor );
+
const resizeSquareUI = [ ...viewDocument.getRoot().getChild( 0 ).getChildren() ]
.find( element => element.hasClass( 'ck-widget__resizer' ) );
diff --git a/packages/ckeditor5-widget/tests/widgetresize.js b/packages/ckeditor5-widget/tests/widgetresize.js
index f1020cd4e7f..7d12c7870c6 100644
--- a/packages/ckeditor5-widget/tests/widgetresize.js
+++ b/packages/ckeditor5-widget/tests/widgetresize.js
@@ -468,20 +468,43 @@ describe( 'WidgetResize', () => {
} );
describe( 'attachTo()', () => {
- it( 'works without WidgetToolbarRepository plugin', async () => {
- const localEditorElement = createEditorElement();
- const localEditor = await ClassicEditor.create( localEditorElement, {
+ let localEditorElement, localEditor;
+
+ beforeEach( async () => {
+ localEditorElement = createEditorElement();
+ localEditor = await ClassicEditor.create( localEditorElement, {
plugins: [
WidgetResize, simpleWidgetPlugin
]
} );
+ } );
+
+ afterEach( () => {
+ localEditorElement.remove();
+ return localEditor.destroy();
+ } );
+
+ it( 'works without WidgetToolbarRepository plugin', async () => {
+ setModelData( localEditor.model, '[]' );
+
+ localEditor.plugins.get( WidgetResize ).attachTo( gerResizerOptions( localEditor ) );
+ // Nothing should be thrown.
+ } );
+ it( 'sets the visible resizer if associated widget is already focused', async () => {
setModelData( localEditor.model, '[]' );
- const resizerOptions = {
- modelElement: localEditor.model.document.getRoot().getChild( 0 ),
- viewElement: localEditor.editing.view.document.getRoot().getChild( 0 ),
- editor: localEditor,
+ const widgetResizePlugin = localEditor.plugins.get( WidgetResize );
+ const resizer = widgetResizePlugin.attachTo( gerResizerOptions( localEditor ) );
+
+ expect( widgetResizePlugin.visibleResizer ).to.eql( resizer );
+ } );
+
+ function gerResizerOptions( editor ) {
+ return {
+ modelElement: editor.model.document.getRoot().getChild( 0 ),
+ viewElement: editor.editing.view.document.getRoot().getChild( 0 ),
+ editor,
isCentered: () => false,
getHandleHost( domWidgetElement ) {
@@ -493,13 +516,7 @@ describe( 'WidgetResize', () => {
onCommit: commitStub
};
-
- localEditor.plugins.get( WidgetResize ).attachTo( resizerOptions );
- // Nothing should be thrown.
- // And clean up.
- localEditorElement.remove();
- return localEditor.destroy();
- } );
+ }
} );
describe( 'init()', () => {
diff --git a/packages/ckeditor5-widget/tests/widgetresize/_utils/utils.js b/packages/ckeditor5-widget/tests/widgetresize/_utils/utils.js
index c8759f1532c..222cc9383d9 100644
--- a/packages/ckeditor5-widget/tests/widgetresize/_utils/utils.js
+++ b/packages/ckeditor5-widget/tests/widgetresize/_utils/utils.js
@@ -152,6 +152,5 @@ export function getHandleCenterPoint( domWrapper, handlePosition ) {
}
export function focusEditor( editor ) {
- editor.focus();
editor.ui.focusTracker.isFocused = true;
}