-
Notifications
You must be signed in to change notification settings - Fork 2.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for Word's images pasting to Safari #4339
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Works pretty nice! I see that this code had already some partial review in original PR, so there is not much left. For now, I've tested it with jpg, png, and gif. Please, see the review comments.
I also see that you marked this PR as a feature, although the original issue has been reported as a bug. I suppose that we could change it to feature as image support for OS X Word edition.
plugins/pastetools/filter/image.js
Outdated
var blobUrls = removeDuplicates( CKEDITOR.tools.array.filter( imgTags, function( imgTag ) { | ||
return imgTag.match( /^blob:/i ); | ||
} ) ), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I suppose we could extract removeDuplicates
function to array tools as something like CKEDITOR.tools.array.unique
?
plugins/pastetools/filter/image.js
Outdated
return imgTag.match( /^blob:/i ); | ||
} ) ), | ||
promises = CKEDITOR.tools.array.map( blobUrls, function( blobUrl ) { | ||
return convertBlobUrlToBase64( blobUrl ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
convertToBlobUrlToBase64
could be probably used as higher-order function?
plugins/pastetools/filter/image.js
Outdated
fileType = 'image/png'; | ||
break; | ||
case '47494638': | ||
fileType = 'image/gif'; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wow, I didn't know that Word supports GIFs 😅 I was pretty sure they are converted to PNGs.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually they are converted to PNG… So it works because most browsers support APNG.
Safari in such a case use GIF format. It's strange that it isn't rendered correctly as we have GIF recognition 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding GIF mime type to array of supported image formats seems to do the trick.
plugins/pastetools/filter/image.js
Outdated
header += bytesHeader[ i ].toString( 16 ); | ||
} | ||
|
||
switch ( header ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's some dark magic 😄 I'm not sure if it should be named header
, probably File signature is more accurate here. It would be good to have real, existing images with these signatures, to confirm that the feature covers all expected image types. These can be checked e.g. here: https://www.filesignatures.net/index.php?search=FFD8FFE0&mode=SIG although I suppose it may be hard to get all spf
, jfif
etc extension examples and I'm not sure if Word is not converting them also into a more unified format.
plugins/ajax/plugin.js
Outdated
@@ -128,12 +136,16 @@ | |||
* @param {String} url The URL from which the data is loaded. | |||
* @param {Function} [callback] A callback function to be called on | |||
* data load. If not provided, the data will be loaded | |||
* synchronously. | |||
* @returns {String} The loaded data. For asynchronous requests, an | |||
* synchronously. Please notice that only text data might be loaded synchrnously. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
* synchronously. Please notice that only text data might be loaded synchrnously. | |
* synchronously. Please notice that only text data might be loaded synchronously. |
}, | ||
|
||
// (#1134) | ||
'test load async arraybuffer': function() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We are also missing text
and xml
types. I guess they are probably tested already as we are using them indirectly in ajax logic, but it would be good to check if API works correctly.
|
||
function simulatePasteBlob( editor, assertion, options ) { | ||
// jscs:disable maximumLineLength | ||
var imgBase64 = '', |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you think it would be possible to use real images instead with more test cases?
if ( !CKEDITOR.env.safari ) { | ||
bender.ignore(); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe it makes sense to keep this test also for different browsers to make sure we didn't introduce regression?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
All other browsers are covered by unit tests for Paste Tools, Paste from Word and Paste from Libre Office, so I don't think it would be necessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We still need to perform manual tests for images during the testing phase to make sure that we didn't introduce any regression. Enabling these tests on browsers which should support images significantly improves odds that we will actually test it, as generic tests may not have enough test samples.
Please, also ignore this test on mobile.
@bender-ckeditor-plugins: wysiwygarea, toolbar, undo, basicstyles, pastefromword, elementspath, image, format, sourcearea | ||
|
||
1. Open [attached Word document](_assets/Image_safari.docx). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It would be good to add a more complicated document with more image sources, especially gif and different flavors of jpg.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How can I produce different flavors of JPG? 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually we probably don't need to test different flavors of JPG, as all of its signatures starts with the same sequence: FF D8 FF
. Checking if the image file starts with it should be enough for our case.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How can I produce different flavors of JPG?
By using some conversion tool. But indeed making signature-less strict should work. I've checked File Signature Database and it doesn't seem that a less specific signature will catch any other filetype 🤞🏻
Hmm, that can be a little problematic, as |
It's been a while since we last heard from you. We are marking this pull request as stale due to inactivity. Please provide the requested feedback or the pull request will be closed after next 7 days. |
Ok,
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, that can be a little problematic, as Uint8Array is correct… Seems like jsduck is old enough to not recognize Typed Arrays 🤔
I see that Blob
is correctly recognized, so it's indeed weird that jsduck is not able to read this type. Maybe there is some workaround for custom type? If not, I suppose we will have to rename it into Object
to avoid building warnings? Although, it's pity going some weird workarounds because the tooling doesn't work correctly.
Currently, ajax methods are a bit mixed up. Previously, we had:
load
- loading withtext
responseloadXml
- loading withxml
response
now we have:
load
- by default text, but also supportsxml
andarraybuffer
loadXml
- still supporting onlyxml
Maybe, we should add explicit loadBinaryData/loadData
? It probably doesn't really make sense deprecating loadXml
as explicitness of this method is actually nice. Most readable seems to have load
(without default type), loadText
, loadXml
, loadBinaryData
but it's a bit too late for that.
* arrayWithoutDuplicates = CKEDITOR.tools.array.unique( array ); | ||
* console.log( arrayWithoutDuplicates ); // [ 1, 2, 3 ] | ||
* ``` | ||
* |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Missing @since
tag and @member CKEDITOR.tools.array
.
plugins/pastetools/filter/image.js
Outdated
@@ -11,7 +11,9 @@ | |||
'use strict'; | |||
|
|||
/** | |||
* Filter handling pasting images from RTF content. | |||
* Filter handling pasting images. In Safari images are extracted |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we have any strict check if it's actually Safari? I suppose it should rather tell that in case of missing RTF it will be using this method?
|
||
## Unexpected | ||
|
||
* Iage is not rendered correctly. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Typo.
if ( !CKEDITOR.env.safari ) { | ||
bender.ignore(); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We still need to perform manual tests for images during the testing phase to make sure that we didn't introduce any regression. Enabling these tests on browsers which should support images significantly improves odds that we will actually test it, as generic tests may not have enough test samples.
Please, also ignore this test on mobile.
@bender-ckeditor-plugins: wysiwygarea, toolbar, undo, basicstyles, pastefromword, elementspath, image, format, sourcearea | ||
|
||
1. Open [attached Word document](_assets/Image_safari.docx). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How can I produce different flavors of JPG?
By using some conversion tool. But indeed making signature-less strict should work. I've checked File Signature Database and it doesn't seem that a less specific signature will catch any other filetype 🤞🏻
function convertBlobUrlToBase64( blobUrlSrc ) { | ||
return new CKEDITOR.tools.promise( function( resolve ) { | ||
CKEDITOR.ajax.load( blobUrlSrc, function( arrayBuffer ) { | ||
var data = new Uint8Array( arrayBuffer ), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm curious about old IE support here (IE8-9) as Uint8Array
is not supported there. However, nothing breaks - seems like we are not supporting image pasting there at all.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, in old IEs this code is just ignored.
@@ -0,0 +1,16 @@ | |||
@bender-tags: feature, 4.16.0, 1134, 2800, pastefromword | |||
@bender-ui: collapsed | |||
@bender-ckeditor-plugins: wysiwygarea, toolbar, undo, pastefromword, image, pagebreak, autogrow |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It would be good to add sourcearea test steps similar to pasteimagessafari
manual test to make sure that the image is transformed to base64.
plugins/pastetools/filter/image.js
Outdated
@@ -179,6 +164,111 @@ | |||
} | |||
]; | |||
|
|||
/** | |||
* Array of all recognizable image file signatrues with their respective types. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
* Array of all recognizable image file signatrues with their respective types. | |
* Recognizable image file signatures with their respective types. |
var image = options && options.image || SAMPLE_PNG, | ||
imageDataUrl = options && options.imageDataUrl || SAMPLE_PNG, | ||
url = getObjectUrl( image, options && options.imageType ), | ||
template = options && options.template || '<p{%CLASS%}>Hello<img style="height:200px; width:200px" src="{%URL%}" />world.</p>', | ||
input = template.replace( /{%URL%}/g, url ).replace( /{%CLASS%}/g, ' class="MsoNormal"' ), | ||
expected = template.replace( /{%URL%}/g, imageDataUrl ).replace( /{%CLASS%}/g, '' ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The code is a bit unreadable with all default option values. Maybe we could simplify it with some merging logic?
CKEDITOR.tools.object.merge( defaultOptions, options || {} );
plugins/pastetools/filter/image.js
Outdated
return new CKEDITOR.tools.promise( function( resolve ) { | ||
CKEDITOR.ajax.load( blobUrlSrc, function( arrayBuffer ) { | ||
var data = new Uint8Array( arrayBuffer ), | ||
imageType = getImageTypeFromSignature( data.subarray( 0, 4 ) ), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't we pass here the whole arrayBuffer
? It seems like taking the first 4 characters is already done in getImageTypeFromSignature
function? In that case, this should be probably renamed again to just getImageType
.
Could you also rebase PR onto the latest major? Seems like there are some conficts. |
Ok, I've addressed issues found during the review.
TBH I don't find it's too late. We can always have additional methods and I'll raise an issue for it. |
…nversion of TypedArray is no longer needed.
a869c01
to
8e6392c
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good job!
What is the purpose of this pull request?
New feature
Does your PR contain necessary tests?
All patches that change the editor code must include tests. You can always read more
on PR testing,
how to set the testing environment and
how to create tests
in the official CKEditor documentation.
This PR contains
Did you follow the CKEditor 4 code style guide?
Your code should follow the guidelines from the CKEditor 4 code style guide which helps keep the entire codebase consistent.
What is the proposed changelog entry for this pull request?
What changes did you make?
I've basically taken the contents from @msamsel's #1976 and integrated it with our
CKEDITOR.plugins.pastetools.filters.image
filter.Which issues does your PR resolve?
Closes #1134.