diff --git a/CHANGES.md b/CHANGES.md
index d657d5dceab..83b360d3bde 100644
--- a/CHANGES.md
+++ b/CHANGES.md
@@ -31,6 +31,7 @@ Fixed Issues:
* [#1057](https://github.com/ckeditor/ckeditor-dev/issues/1057): Fixed: The [Notification](https://ckeditor.com/addon/notification) plugin overwrites Web Notifications API due to leakage to the global scope.
* [#1068](https://github.com/ckeditor/ckeditor-dev/issues/1068): Fixed: Upload widget paste listener ignores changes to the [`uploadWidgetDefinition`](https://docs.ckeditor.com/ckeditor4/docs/#!/api/CKEDITOR.fileTools.uploadWidgetDefinition).
* [#921](https://github.com/ckeditor/ckeditor-dev/issues/921): Fixed: [Edge] CKEditor erroneously perceives internal copy and paste as type "external".
+* [#1213](https://github.com/ckeditor/ckeditor-dev/issues/1213): Fixed: Multiple images uploaded using [Upload Image](https://ckeditor.com/cke4/addon/uploadimage) plugin are randomly duplicated or mangled.
API Changes:
diff --git a/plugins/uploadimage/plugin.js b/plugins/uploadimage/plugin.js
index be43955b4cc..0571382542b 100644
--- a/plugins/uploadimage/plugin.js
+++ b/plugins/uploadimage/plugin.js
@@ -6,6 +6,29 @@
'use strict';
( function() {
+ var uniqueNameCounter = 0,
+ // Black rectangle which is shown before image is loaded.
+ loadingImage = '';
+
+ // Returns number as a string. If a number has 1 digit only it returns it prefixed with an extra 0.
+ function padNumber( input ) {
+ if ( input <= 9 ) {
+ input = '0' + input;
+ }
+
+ return String( input );
+ }
+
+ // Returns an unique image file name.
+ function getUniqueImageFileName( type ) {
+ var date = new Date(),
+ dateParts = [ date.getFullYear(), date.getMonth() + 1, date.getDate(), date.getHours(), date.getMinutes(), date.getSeconds() ];
+
+ uniqueNameCounter += 1;
+
+ return 'image-' + CKEDITOR.tools.array.map( dateParts, padNumber ).join( '' ) + '-' + uniqueNameCounter + '.' + type;
+ }
+
CKEDITOR.plugins.add( 'uploadimage', {
requires: 'uploadwidget',
@@ -89,13 +112,22 @@
for ( i = 0; i < imgs.count(); i++ ) {
img = imgs.getItem( i );
- // Image have to contain src=data:...
- var isDataInSrc = img.getAttribute( 'src' ) && img.getAttribute( 'src' ).substring( 0, 5 ) == 'data:',
+ // Assign src once, as it might be a big string, so there's no point in duplicating it all over the place.
+ var imgSrc = img.getAttribute( 'src' ),
+ // Image have to contain src=data:...
+ isDataInSrc = imgSrc && imgSrc.substring( 0, 5 ) == 'data:',
isRealObject = img.data( 'cke-realelement' ) === null;
// We are not uploading images in non-editable blocs and fake objects (https://dev.ckeditor.com/ticket/13003).
if ( isDataInSrc && isRealObject && !img.data( 'cke-upload-id' ) && !img.isReadOnly( 1 ) ) {
- var loader = editor.uploadRepository.create( img.getAttribute( 'src' ) );
+ // Note normally we'd extract this logic into a separate function, but we should not duplicate this string, as it might
+ // be large.
+ var imgFormat = imgSrc.match( /image\/([a-z]+?);/i ),
+ loader;
+
+ imgFormat = ( imgFormat && imgFormat[ 1 ] ) || 'jpg';
+
+ loader = editor.uploadRepository.create( imgSrc, getUniqueImageFileName( imgFormat ) );
loader.upload( uploadUrl );
fileTools.markElement( img, 'uploadimage', loader.id );
@@ -109,11 +141,6 @@
}
} );
- // jscs:disable maximumLineLength
- // Black rectangle which is shown before image is loaded.
- var loadingImage = '';
- // jscs:enable maximumLineLength
-
/**
* The URL where images should be uploaded.
*
diff --git a/tests/plugins/uploadimage/uploadimage.js b/tests/plugins/uploadimage/uploadimage.js
index 1033fbbeb5b..c487ea05203 100644
--- a/tests/plugins/uploadimage/uploadimage.js
+++ b/tests/plugins/uploadimage/uploadimage.js
@@ -538,7 +538,7 @@
'test prevent upload fake elements (https://dev.ckeditor.com/ticket/13003)': function() {
var editor = this.editors.inline,
- createspy = sinon.spy( editor.uploadRepository, 'create' );
+ createSpy = sinon.spy( editor.uploadRepository, 'create' );
editor.fire( 'paste', {
dataValue: ''
@@ -546,7 +546,33 @@
editor.once( 'afterPaste', function() {
resume( function() {
- assert.isTrue( createspy.notCalled );
+ createSpy.restore();
+ assert.isTrue( createSpy.notCalled );
+ } );
+ } );
+
+ wait();
+ },
+
+ 'test uploads generate unique names (#1213)': function() {
+ var editor = this.editors.inline,
+ createSpy = sinon.spy( editor.uploadRepository, 'create' );
+
+ editor.fire( 'paste', {
+ dataValue: '' +
+ '' +
+ ''
+ } );
+
+ editor.once( 'afterPaste', function() {
+ resume( function() {
+ createSpy.restore();
+ assert.areSame( 3, createSpy.callCount, 'create call count' );
+
+ assert.isMatching( /image-\d+-\d+\.gif/, createSpy.args[ 0 ][ 1 ], 'file name passed to first call' );
+ assert.isMatching( /image-\d+-\d+\.gif/, createSpy.args[ 1 ][ 1 ], 'file name passed to second call' );
+ assert.areNotSame( createSpy.args[ 0 ][ 1 ], createSpy.args[ 1 ][ 1 ], 'first and second call names are different' );
+ assert.isMatching( /image-\d+-\d+\.png/, createSpy.args[ 2 ][ 1 ], 'png type is recognized' );
} );
} );