@ckeditor/ckeditor5-image 40.0.0 → 40.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (253) hide show
  1. package/CHANGELOG.md +25 -25
  2. package/LICENSE.md +3 -3
  3. package/build/image.js +2 -2
  4. package/build/translations/ar.js +1 -1
  5. package/build/translations/ast.js +1 -1
  6. package/build/translations/az.js +1 -1
  7. package/build/translations/bg.js +1 -1
  8. package/build/translations/bn.js +1 -1
  9. package/build/translations/bs.js +1 -1
  10. package/build/translations/ca.js +1 -1
  11. package/build/translations/cs.js +1 -1
  12. package/build/translations/da.js +1 -1
  13. package/build/translations/de-ch.js +1 -1
  14. package/build/translations/de.js +1 -1
  15. package/build/translations/el.js +1 -1
  16. package/build/translations/en-au.js +1 -1
  17. package/build/translations/en-gb.js +1 -1
  18. package/build/translations/eo.js +1 -1
  19. package/build/translations/es.js +1 -1
  20. package/build/translations/et.js +1 -1
  21. package/build/translations/eu.js +1 -1
  22. package/build/translations/fa.js +1 -1
  23. package/build/translations/fi.js +1 -1
  24. package/build/translations/fr.js +1 -1
  25. package/build/translations/gl.js +1 -1
  26. package/build/translations/he.js +1 -1
  27. package/build/translations/hi.js +1 -1
  28. package/build/translations/hr.js +1 -1
  29. package/build/translations/hu.js +1 -1
  30. package/build/translations/id.js +1 -1
  31. package/build/translations/it.js +1 -1
  32. package/build/translations/ja.js +1 -1
  33. package/build/translations/jv.js +1 -1
  34. package/build/translations/km.js +1 -1
  35. package/build/translations/kn.js +1 -1
  36. package/build/translations/ko.js +1 -1
  37. package/build/translations/ku.js +1 -1
  38. package/build/translations/lt.js +1 -1
  39. package/build/translations/lv.js +1 -1
  40. package/build/translations/ms.js +1 -1
  41. package/build/translations/nb.js +1 -1
  42. package/build/translations/ne.js +1 -1
  43. package/build/translations/nl.js +1 -1
  44. package/build/translations/no.js +1 -1
  45. package/build/translations/pl.js +1 -1
  46. package/build/translations/pt-br.js +1 -1
  47. package/build/translations/pt.js +1 -1
  48. package/build/translations/ro.js +1 -1
  49. package/build/translations/ru.js +1 -1
  50. package/build/translations/si.js +1 -1
  51. package/build/translations/sk.js +1 -1
  52. package/build/translations/sq.js +1 -1
  53. package/build/translations/sr-latn.js +1 -1
  54. package/build/translations/sr.js +1 -1
  55. package/build/translations/sv.js +1 -1
  56. package/build/translations/th.js +1 -1
  57. package/build/translations/tk.js +1 -1
  58. package/build/translations/tr.js +1 -1
  59. package/build/translations/tt.js +1 -1
  60. package/build/translations/ug.js +1 -1
  61. package/build/translations/uk.js +1 -1
  62. package/build/translations/ur.js +1 -1
  63. package/build/translations/uz.js +1 -1
  64. package/build/translations/vi.js +1 -1
  65. package/build/translations/zh-cn.js +1 -1
  66. package/build/translations/zh.js +1 -1
  67. package/ckeditor5-metadata.json +3 -3
  68. package/lang/contexts.json +5 -0
  69. package/lang/translations/ar.po +20 -0
  70. package/lang/translations/ast.po +20 -0
  71. package/lang/translations/az.po +20 -0
  72. package/lang/translations/bg.po +20 -0
  73. package/lang/translations/bn.po +20 -0
  74. package/lang/translations/bs.po +20 -0
  75. package/lang/translations/ca.po +20 -0
  76. package/lang/translations/cs.po +20 -0
  77. package/lang/translations/da.po +20 -0
  78. package/lang/translations/de-ch.po +20 -0
  79. package/lang/translations/de.po +20 -0
  80. package/lang/translations/el.po +20 -0
  81. package/lang/translations/en-au.po +20 -0
  82. package/lang/translations/en-gb.po +20 -0
  83. package/lang/translations/en.po +20 -0
  84. package/lang/translations/eo.po +20 -0
  85. package/lang/translations/es.po +20 -0
  86. package/lang/translations/et.po +20 -0
  87. package/lang/translations/eu.po +20 -0
  88. package/lang/translations/fa.po +20 -0
  89. package/lang/translations/fi.po +20 -0
  90. package/lang/translations/fr.po +20 -0
  91. package/lang/translations/gl.po +20 -0
  92. package/lang/translations/he.po +20 -0
  93. package/lang/translations/hi.po +20 -0
  94. package/lang/translations/hr.po +20 -0
  95. package/lang/translations/hu.po +20 -0
  96. package/lang/translations/id.po +20 -0
  97. package/lang/translations/it.po +20 -0
  98. package/lang/translations/ja.po +20 -0
  99. package/lang/translations/jv.po +20 -0
  100. package/lang/translations/km.po +20 -0
  101. package/lang/translations/kn.po +20 -0
  102. package/lang/translations/ko.po +20 -0
  103. package/lang/translations/ku.po +20 -0
  104. package/lang/translations/lt.po +20 -0
  105. package/lang/translations/lv.po +20 -0
  106. package/lang/translations/ms.po +20 -0
  107. package/lang/translations/nb.po +20 -0
  108. package/lang/translations/ne.po +20 -0
  109. package/lang/translations/nl.po +20 -0
  110. package/lang/translations/no.po +20 -0
  111. package/lang/translations/pl.po +20 -0
  112. package/lang/translations/pt-br.po +20 -0
  113. package/lang/translations/pt.po +20 -0
  114. package/lang/translations/ro.po +20 -0
  115. package/lang/translations/ru.po +20 -0
  116. package/lang/translations/si.po +20 -0
  117. package/lang/translations/sk.po +20 -0
  118. package/lang/translations/sq.po +20 -0
  119. package/lang/translations/sr-latn.po +20 -0
  120. package/lang/translations/sr.po +20 -0
  121. package/lang/translations/sv.po +20 -0
  122. package/lang/translations/th.po +20 -0
  123. package/lang/translations/tk.po +20 -0
  124. package/lang/translations/tr.po +20 -0
  125. package/lang/translations/tt.po +20 -0
  126. package/lang/translations/ug.po +21 -1
  127. package/lang/translations/uk.po +20 -0
  128. package/lang/translations/ur.po +20 -0
  129. package/lang/translations/uz.po +20 -0
  130. package/lang/translations/vi.po +20 -0
  131. package/lang/translations/zh-cn.po +20 -0
  132. package/lang/translations/zh.po +20 -0
  133. package/package.json +3 -3
  134. package/src/augmentation.d.ts +56 -56
  135. package/src/augmentation.js +5 -5
  136. package/src/autoimage.d.ts +52 -52
  137. package/src/autoimage.js +132 -132
  138. package/src/image/converters.d.ts +66 -66
  139. package/src/image/converters.js +232 -232
  140. package/src/image/imageblockediting.d.ts +59 -58
  141. package/src/image/imageblockediting.js +153 -152
  142. package/src/image/imageediting.d.ts +30 -30
  143. package/src/image/imageediting.js +63 -63
  144. package/src/image/imageinlineediting.d.ts +60 -59
  145. package/src/image/imageinlineediting.js +177 -176
  146. package/src/image/imageloadobserver.d.ts +48 -48
  147. package/src/image/imageloadobserver.js +52 -52
  148. package/src/image/imageplaceholder.d.ts +39 -0
  149. package/src/image/imageplaceholder.js +113 -0
  150. package/src/image/imagetypecommand.d.ts +44 -44
  151. package/src/image/imagetypecommand.js +80 -80
  152. package/src/image/insertimagecommand.d.ts +66 -66
  153. package/src/image/insertimagecommand.js +120 -120
  154. package/src/image/replaceimagesourcecommand.d.ts +51 -34
  155. package/src/image/replaceimagesourcecommand.js +75 -44
  156. package/src/image/ui/utils.d.ts +25 -25
  157. package/src/image/ui/utils.js +44 -44
  158. package/src/image/utils.d.ts +64 -64
  159. package/src/image/utils.js +121 -121
  160. package/src/image.d.ts +34 -34
  161. package/src/image.js +38 -38
  162. package/src/imageblock.d.ts +34 -33
  163. package/src/imageblock.js +38 -37
  164. package/src/imagecaption/imagecaptionediting.d.ts +89 -89
  165. package/src/imagecaption/imagecaptionediting.js +225 -225
  166. package/src/imagecaption/imagecaptionui.d.ts +26 -26
  167. package/src/imagecaption/imagecaptionui.js +61 -61
  168. package/src/imagecaption/imagecaptionutils.d.ts +38 -38
  169. package/src/imagecaption/imagecaptionutils.js +62 -62
  170. package/src/imagecaption/toggleimagecaptioncommand.d.ts +66 -66
  171. package/src/imagecaption/toggleimagecaptioncommand.js +138 -138
  172. package/src/imagecaption.d.ts +26 -26
  173. package/src/imagecaption.js +30 -30
  174. package/src/imageconfig.d.ts +712 -713
  175. package/src/imageconfig.js +5 -5
  176. package/src/imageinline.d.ts +34 -33
  177. package/src/imageinline.js +38 -37
  178. package/src/imageinsert/imageinsertui.d.ts +72 -44
  179. package/src/imageinsert/imageinsertui.js +174 -141
  180. package/src/imageinsert/imageinsertviaurlui.d.ts +44 -0
  181. package/src/imageinsert/imageinsertviaurlui.js +122 -0
  182. package/src/imageinsert/ui/imageinsertformview.d.ts +56 -0
  183. package/src/imageinsert/ui/imageinsertformview.js +112 -0
  184. package/src/imageinsert/ui/imageinserturlview.d.ts +107 -0
  185. package/src/imageinsert/ui/imageinserturlview.js +156 -0
  186. package/src/imageinsert.d.ts +33 -33
  187. package/src/imageinsert.js +37 -37
  188. package/src/imageinsertviaurl.d.ts +31 -30
  189. package/src/imageinsertviaurl.js +35 -34
  190. package/src/imageresize/imageresizebuttons.d.ts +67 -67
  191. package/src/imageresize/imageresizebuttons.js +217 -217
  192. package/src/imageresize/imageresizeediting.d.ts +37 -37
  193. package/src/imageresize/imageresizeediting.js +165 -165
  194. package/src/imageresize/imageresizehandles.d.ts +31 -31
  195. package/src/imageresize/imageresizehandles.js +114 -114
  196. package/src/imageresize/resizeimagecommand.d.ts +42 -42
  197. package/src/imageresize/resizeimagecommand.js +63 -63
  198. package/src/imageresize.d.ts +27 -27
  199. package/src/imageresize.js +31 -31
  200. package/src/imagesizeattributes.d.ts +34 -34
  201. package/src/imagesizeattributes.js +142 -143
  202. package/src/imagestyle/converters.d.ts +24 -24
  203. package/src/imagestyle/converters.js +79 -79
  204. package/src/imagestyle/imagestylecommand.d.ts +68 -68
  205. package/src/imagestyle/imagestylecommand.js +107 -107
  206. package/src/imagestyle/imagestyleediting.d.ts +50 -50
  207. package/src/imagestyle/imagestyleediting.js +108 -108
  208. package/src/imagestyle/imagestyleui.d.ts +56 -56
  209. package/src/imagestyle/imagestyleui.js +192 -192
  210. package/src/imagestyle/utils.d.ts +101 -101
  211. package/src/imagestyle/utils.js +329 -329
  212. package/src/imagestyle.d.ts +32 -32
  213. package/src/imagestyle.js +36 -36
  214. package/src/imagetextalternative/imagetextalternativecommand.d.ts +34 -34
  215. package/src/imagetextalternative/imagetextalternativecommand.js +44 -44
  216. package/src/imagetextalternative/imagetextalternativeediting.d.ts +28 -28
  217. package/src/imagetextalternative/imagetextalternativeediting.js +35 -35
  218. package/src/imagetextalternative/imagetextalternativeui.d.ts +68 -68
  219. package/src/imagetextalternative/imagetextalternativeui.js +173 -173
  220. package/src/imagetextalternative/ui/textalternativeformview.d.ts +90 -72
  221. package/src/imagetextalternative/ui/textalternativeformview.js +121 -121
  222. package/src/imagetextalternative.d.ts +29 -29
  223. package/src/imagetextalternative.js +33 -33
  224. package/src/imagetoolbar.d.ts +35 -35
  225. package/src/imagetoolbar.js +57 -57
  226. package/src/imageupload/imageuploadediting.d.ts +111 -111
  227. package/src/imageupload/imageuploadediting.js +337 -337
  228. package/src/imageupload/imageuploadprogress.d.ts +42 -42
  229. package/src/imageupload/imageuploadprogress.js +211 -211
  230. package/src/imageupload/imageuploadui.d.ts +23 -23
  231. package/src/imageupload/imageuploadui.js +81 -57
  232. package/src/imageupload/uploadimagecommand.d.ts +60 -60
  233. package/src/imageupload/uploadimagecommand.js +100 -100
  234. package/src/imageupload/utils.d.ts +33 -33
  235. package/src/imageupload/utils.js +112 -112
  236. package/src/imageupload.d.ts +32 -32
  237. package/src/imageupload.js +36 -36
  238. package/src/imageutils.d.ts +125 -125
  239. package/src/imageutils.js +306 -306
  240. package/src/index.d.ts +48 -48
  241. package/src/index.js +39 -39
  242. package/src/pictureediting.d.ts +88 -88
  243. package/src/pictureediting.js +130 -130
  244. package/theme/imageinsert.css +5 -17
  245. package/theme/imageplaceholder.css +10 -0
  246. package/build/image.js.map +0 -1
  247. package/src/imageinsert/ui/imageinsertformrowview.d.ts +0 -61
  248. package/src/imageinsert/ui/imageinsertformrowview.js +0 -54
  249. package/src/imageinsert/ui/imageinsertpanelview.d.ts +0 -106
  250. package/src/imageinsert/ui/imageinsertpanelview.js +0 -161
  251. package/src/imageinsert/utils.d.ts +0 -25
  252. package/src/imageinsert/utils.js +0 -58
  253. package/theme/imageinsertformrowview.css +0 -36
@@ -1,337 +1,337 @@
1
- /**
2
- * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
3
- * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
- */
5
- /**
6
- * @module image/imageupload/imageuploadediting
7
- */
8
- import { Plugin } from 'ckeditor5/src/core';
9
- import { UpcastWriter } from 'ckeditor5/src/engine';
10
- import { Notification } from 'ckeditor5/src/ui';
11
- import { ClipboardPipeline } from 'ckeditor5/src/clipboard';
12
- import { FileRepository } from 'ckeditor5/src/upload';
13
- import { env } from 'ckeditor5/src/utils';
14
- import ImageUtils from '../imageutils';
15
- import UploadImageCommand from './uploadimagecommand';
16
- import { fetchLocalImage, isLocalImage } from '../../src/imageupload/utils';
17
- import { createImageTypeRegExp } from './utils';
18
- /**
19
- * The editing part of the image upload feature. It registers the `'uploadImage'` command
20
- * and the `imageUpload` command as an aliased name.
21
- *
22
- * When an image is uploaded, it fires the {@link ~ImageUploadEditing#event:uploadComplete `uploadComplete`} event
23
- * that allows adding custom attributes to the {@link module:engine/model/element~Element image element}.
24
- */
25
- export default class ImageUploadEditing extends Plugin {
26
- /**
27
- * @inheritDoc
28
- */
29
- static get requires() {
30
- return [FileRepository, Notification, ClipboardPipeline, ImageUtils];
31
- }
32
- static get pluginName() {
33
- return 'ImageUploadEditing';
34
- }
35
- /**
36
- * @inheritDoc
37
- */
38
- constructor(editor) {
39
- super(editor);
40
- editor.config.define('image', {
41
- upload: {
42
- types: ['jpeg', 'png', 'gif', 'bmp', 'webp', 'tiff']
43
- }
44
- });
45
- this._uploadImageElements = new Map();
46
- }
47
- /**
48
- * @inheritDoc
49
- */
50
- init() {
51
- const editor = this.editor;
52
- const doc = editor.model.document;
53
- const conversion = editor.conversion;
54
- const fileRepository = editor.plugins.get(FileRepository);
55
- const imageUtils = editor.plugins.get('ImageUtils');
56
- const clipboardPipeline = editor.plugins.get('ClipboardPipeline');
57
- const imageTypes = createImageTypeRegExp(editor.config.get('image.upload.types'));
58
- const uploadImageCommand = new UploadImageCommand(editor);
59
- // Register `uploadImage` command and add `imageUpload` command as an alias for backward compatibility.
60
- editor.commands.add('uploadImage', uploadImageCommand);
61
- editor.commands.add('imageUpload', uploadImageCommand);
62
- // Register upcast converter for uploadId.
63
- conversion.for('upcast')
64
- .attributeToAttribute({
65
- view: {
66
- name: 'img',
67
- key: 'uploadId'
68
- },
69
- model: 'uploadId'
70
- });
71
- // Handle pasted images.
72
- // For every image file, a new file loader is created and a placeholder image is
73
- // inserted into the content. Then, those images are uploaded once they appear in the model
74
- // (see Document#change listener below).
75
- this.listenTo(editor.editing.view.document, 'clipboardInput', (evt, data) => {
76
- // Skip if non empty HTML data is included.
77
- // https://github.com/ckeditor/ckeditor5-upload/issues/68
78
- if (isHtmlIncluded(data.dataTransfer)) {
79
- return;
80
- }
81
- const images = Array.from(data.dataTransfer.files).filter(file => {
82
- // See https://github.com/ckeditor/ckeditor5-image/pull/254.
83
- if (!file) {
84
- return false;
85
- }
86
- return imageTypes.test(file.type);
87
- });
88
- if (!images.length) {
89
- return;
90
- }
91
- evt.stop();
92
- editor.model.change(writer => {
93
- // Set selection to paste target.
94
- if (data.targetRanges) {
95
- writer.setSelection(data.targetRanges.map(viewRange => editor.editing.mapper.toModelRange(viewRange)));
96
- }
97
- editor.execute('uploadImage', { file: images });
98
- });
99
- });
100
- // Handle HTML pasted with images with base64 or blob sources.
101
- // For every image file, a new file loader is created and a placeholder image is
102
- // inserted into the content. Then, those images are uploaded once they appear in the model
103
- // (see Document#change listener below).
104
- this.listenTo(clipboardPipeline, 'inputTransformation', (evt, data) => {
105
- const fetchableImages = Array.from(editor.editing.view.createRangeIn(data.content))
106
- .map(value => value.item)
107
- .filter(viewElement => isLocalImage(imageUtils, viewElement) &&
108
- !viewElement.getAttribute('uploadProcessed'))
109
- .map(viewElement => { return { promise: fetchLocalImage(viewElement), imageElement: viewElement }; });
110
- if (!fetchableImages.length) {
111
- return;
112
- }
113
- const writer = new UpcastWriter(editor.editing.view.document);
114
- for (const fetchableImage of fetchableImages) {
115
- // Set attribute marking that the image was processed already.
116
- writer.setAttribute('uploadProcessed', true, fetchableImage.imageElement);
117
- const loader = fileRepository.createLoader(fetchableImage.promise);
118
- if (loader) {
119
- writer.setAttribute('src', '', fetchableImage.imageElement);
120
- writer.setAttribute('uploadId', loader.id, fetchableImage.imageElement);
121
- }
122
- }
123
- });
124
- // Prevents from the browser redirecting to the dropped image.
125
- editor.editing.view.document.on('dragover', (evt, data) => {
126
- data.preventDefault();
127
- });
128
- // Upload placeholder images that appeared in the model.
129
- doc.on('change', () => {
130
- // Note: Reversing changes to start with insertions and only then handle removals. If it was the other way around,
131
- // loaders for **all** images that land in the $graveyard would abort while in fact only those that were **not** replaced
132
- // by other images should be aborted.
133
- const changes = doc.differ.getChanges({ includeChangesInGraveyard: true }).reverse();
134
- const insertedImagesIds = new Set();
135
- for (const entry of changes) {
136
- if (entry.type == 'insert' && entry.name != '$text') {
137
- const item = entry.position.nodeAfter;
138
- const isInsertedInGraveyard = entry.position.root.rootName == '$graveyard';
139
- for (const imageElement of getImagesFromChangeItem(editor, item)) {
140
- // Check if the image element still has upload id.
141
- const uploadId = imageElement.getAttribute('uploadId');
142
- if (!uploadId) {
143
- continue;
144
- }
145
- // Check if the image is loaded on this client.
146
- const loader = fileRepository.loaders.get(uploadId);
147
- if (!loader) {
148
- continue;
149
- }
150
- if (isInsertedInGraveyard) {
151
- // If the image was inserted to the graveyard for good (**not** replaced by another image),
152
- // only then abort the loading process.
153
- if (!insertedImagesIds.has(uploadId)) {
154
- loader.abort();
155
- }
156
- }
157
- else {
158
- // Remember the upload id of the inserted image. If it acted as a replacement for another
159
- // image (which landed in the $graveyard), the related loader will not be aborted because
160
- // this is still the same image upload.
161
- insertedImagesIds.add(uploadId);
162
- // Keep the mapping between the upload ID and the image model element so the upload
163
- // can later resolve in the context of the correct model element. The model element could
164
- // change for the same upload if one image was replaced by another (e.g. image type was changed),
165
- // so this may also replace an existing mapping.
166
- this._uploadImageElements.set(uploadId, imageElement);
167
- if (loader.status == 'idle') {
168
- // If the image was inserted into content and has not been loaded yet, start loading it.
169
- this._readAndUpload(loader);
170
- }
171
- }
172
- }
173
- }
174
- }
175
- });
176
- // Set the default handler for feeding the image element with `src` and `srcset` attributes.
177
- // Also set the natural `width` and `height` attributes (if not already set).
178
- this.on('uploadComplete', (evt, { imageElement, data }) => {
179
- const urls = data.urls ? data.urls : data;
180
- this.editor.model.change(writer => {
181
- writer.setAttribute('src', urls.default, imageElement);
182
- this._parseAndSetSrcsetAttributeOnImage(urls, imageElement, writer);
183
- imageUtils.setImageNaturalSizeAttributes(imageElement);
184
- });
185
- }, { priority: 'low' });
186
- }
187
- /**
188
- * @inheritDoc
189
- */
190
- afterInit() {
191
- const schema = this.editor.model.schema;
192
- // Setup schema to allow uploadId and uploadStatus for images.
193
- // Wait for ImageBlockEditing or ImageInlineEditing to register their elements first,
194
- // that's why doing this in afterInit() instead of init().
195
- if (this.editor.plugins.has('ImageBlockEditing')) {
196
- schema.extend('imageBlock', {
197
- allowAttributes: ['uploadId', 'uploadStatus']
198
- });
199
- }
200
- if (this.editor.plugins.has('ImageInlineEditing')) {
201
- schema.extend('imageInline', {
202
- allowAttributes: ['uploadId', 'uploadStatus']
203
- });
204
- }
205
- }
206
- /**
207
- * Reads and uploads an image.
208
- *
209
- * The image is read from the disk and as a Base64-encoded string it is set temporarily to
210
- * `image[src]`. When the image is successfully uploaded, the temporary data is replaced with the target
211
- * image's URL (the URL to the uploaded image on the server).
212
- */
213
- _readAndUpload(loader) {
214
- const editor = this.editor;
215
- const model = editor.model;
216
- const t = editor.locale.t;
217
- const fileRepository = editor.plugins.get(FileRepository);
218
- const notification = editor.plugins.get(Notification);
219
- const imageUtils = editor.plugins.get('ImageUtils');
220
- const imageUploadElements = this._uploadImageElements;
221
- model.enqueueChange({ isUndoable: false }, writer => {
222
- writer.setAttribute('uploadStatus', 'reading', imageUploadElements.get(loader.id));
223
- });
224
- return loader.read()
225
- .then(() => {
226
- const promise = loader.upload();
227
- const imageElement = imageUploadElements.get(loader.id);
228
- // Force re–paint in Safari. Without it, the image will display with a wrong size.
229
- // https://github.com/ckeditor/ckeditor5/issues/1975
230
- /* istanbul ignore next -- @preserve */
231
- if (env.isSafari) {
232
- const viewFigure = editor.editing.mapper.toViewElement(imageElement);
233
- const viewImg = imageUtils.findViewImgElement(viewFigure);
234
- editor.editing.view.once('render', () => {
235
- // Early returns just to be safe. There might be some code ran
236
- // in between the outer scope and this callback.
237
- if (!viewImg.parent) {
238
- return;
239
- }
240
- const domFigure = editor.editing.view.domConverter.mapViewToDom(viewImg.parent);
241
- if (!domFigure) {
242
- return;
243
- }
244
- const originalDisplay = domFigure.style.display;
245
- domFigure.style.display = 'none';
246
- // Make sure this line will never be removed during minification for having "no effect".
247
- domFigure._ckHack = domFigure.offsetHeight;
248
- domFigure.style.display = originalDisplay;
249
- });
250
- }
251
- model.enqueueChange({ isUndoable: false }, writer => {
252
- writer.setAttribute('uploadStatus', 'uploading', imageElement);
253
- });
254
- return promise;
255
- })
256
- .then(data => {
257
- model.enqueueChange({ isUndoable: false }, writer => {
258
- const imageElement = imageUploadElements.get(loader.id);
259
- writer.setAttribute('uploadStatus', 'complete', imageElement);
260
- this.fire('uploadComplete', { data, imageElement });
261
- });
262
- clean();
263
- })
264
- .catch(error => {
265
- // If status is not 'error' nor 'aborted' - throw error because it means that something else went wrong,
266
- // it might be generic error and it would be real pain to find what is going on.
267
- if (loader.status !== 'error' && loader.status !== 'aborted') {
268
- throw error;
269
- }
270
- // Might be 'aborted'.
271
- if (loader.status == 'error' && error) {
272
- notification.showWarning(error, {
273
- title: t('Upload failed'),
274
- namespace: 'upload'
275
- });
276
- }
277
- // Permanently remove image from insertion batch.
278
- model.enqueueChange({ isUndoable: false }, writer => {
279
- writer.remove(imageUploadElements.get(loader.id));
280
- });
281
- clean();
282
- });
283
- function clean() {
284
- model.enqueueChange({ isUndoable: false }, writer => {
285
- const imageElement = imageUploadElements.get(loader.id);
286
- writer.removeAttribute('uploadId', imageElement);
287
- writer.removeAttribute('uploadStatus', imageElement);
288
- imageUploadElements.delete(loader.id);
289
- });
290
- fileRepository.destroyLoader(loader);
291
- }
292
- }
293
- /**
294
- * Creates the `srcset` attribute based on a given file upload response and sets it as an attribute to a specific image element.
295
- *
296
- * @param data Data object from which `srcset` will be created.
297
- * @param image The image element on which the `srcset` attribute will be set.
298
- */
299
- _parseAndSetSrcsetAttributeOnImage(data, image, writer) {
300
- // Srcset attribute for responsive images support.
301
- let maxWidth = 0;
302
- const srcsetAttribute = Object.keys(data)
303
- // Filter out keys that are not integers.
304
- .filter(key => {
305
- const width = parseInt(key, 10);
306
- if (!isNaN(width)) {
307
- maxWidth = Math.max(maxWidth, width);
308
- return true;
309
- }
310
- })
311
- // Convert each key to srcset entry.
312
- .map(key => `${data[key]} ${key}w`)
313
- // Join all entries.
314
- .join(', ');
315
- if (srcsetAttribute != '') {
316
- const attributes = {
317
- srcset: srcsetAttribute
318
- };
319
- if (!image.hasAttribute('width') && !image.hasAttribute('height')) {
320
- attributes.width = maxWidth;
321
- }
322
- writer.setAttributes(attributes, image);
323
- }
324
- }
325
- }
326
- /**
327
- * Returns `true` if non-empty `text/html` is included in the data transfer.
328
- */
329
- export function isHtmlIncluded(dataTransfer) {
330
- return Array.from(dataTransfer.types).includes('text/html') && dataTransfer.getData('text/html') !== '';
331
- }
332
- function getImagesFromChangeItem(editor, item) {
333
- const imageUtils = editor.plugins.get('ImageUtils');
334
- return Array.from(editor.model.createRangeOn(item))
335
- .filter(value => imageUtils.isImage(value.item))
336
- .map(value => value.item);
337
- }
1
+ /**
2
+ * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
3
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
+ */
5
+ /**
6
+ * @module image/imageupload/imageuploadediting
7
+ */
8
+ import { Plugin } from 'ckeditor5/src/core';
9
+ import { UpcastWriter } from 'ckeditor5/src/engine';
10
+ import { Notification } from 'ckeditor5/src/ui';
11
+ import { ClipboardPipeline } from 'ckeditor5/src/clipboard';
12
+ import { FileRepository } from 'ckeditor5/src/upload';
13
+ import { env } from 'ckeditor5/src/utils';
14
+ import ImageUtils from '../imageutils';
15
+ import UploadImageCommand from './uploadimagecommand';
16
+ import { fetchLocalImage, isLocalImage } from '../../src/imageupload/utils';
17
+ import { createImageTypeRegExp } from './utils';
18
+ /**
19
+ * The editing part of the image upload feature. It registers the `'uploadImage'` command
20
+ * and the `imageUpload` command as an aliased name.
21
+ *
22
+ * When an image is uploaded, it fires the {@link ~ImageUploadEditing#event:uploadComplete `uploadComplete`} event
23
+ * that allows adding custom attributes to the {@link module:engine/model/element~Element image element}.
24
+ */
25
+ export default class ImageUploadEditing extends Plugin {
26
+ /**
27
+ * @inheritDoc
28
+ */
29
+ static get requires() {
30
+ return [FileRepository, Notification, ClipboardPipeline, ImageUtils];
31
+ }
32
+ static get pluginName() {
33
+ return 'ImageUploadEditing';
34
+ }
35
+ /**
36
+ * @inheritDoc
37
+ */
38
+ constructor(editor) {
39
+ super(editor);
40
+ editor.config.define('image', {
41
+ upload: {
42
+ types: ['jpeg', 'png', 'gif', 'bmp', 'webp', 'tiff']
43
+ }
44
+ });
45
+ this._uploadImageElements = new Map();
46
+ }
47
+ /**
48
+ * @inheritDoc
49
+ */
50
+ init() {
51
+ const editor = this.editor;
52
+ const doc = editor.model.document;
53
+ const conversion = editor.conversion;
54
+ const fileRepository = editor.plugins.get(FileRepository);
55
+ const imageUtils = editor.plugins.get('ImageUtils');
56
+ const clipboardPipeline = editor.plugins.get('ClipboardPipeline');
57
+ const imageTypes = createImageTypeRegExp(editor.config.get('image.upload.types'));
58
+ const uploadImageCommand = new UploadImageCommand(editor);
59
+ // Register `uploadImage` command and add `imageUpload` command as an alias for backward compatibility.
60
+ editor.commands.add('uploadImage', uploadImageCommand);
61
+ editor.commands.add('imageUpload', uploadImageCommand);
62
+ // Register upcast converter for uploadId.
63
+ conversion.for('upcast')
64
+ .attributeToAttribute({
65
+ view: {
66
+ name: 'img',
67
+ key: 'uploadId'
68
+ },
69
+ model: 'uploadId'
70
+ });
71
+ // Handle pasted images.
72
+ // For every image file, a new file loader is created and a placeholder image is
73
+ // inserted into the content. Then, those images are uploaded once they appear in the model
74
+ // (see Document#change listener below).
75
+ this.listenTo(editor.editing.view.document, 'clipboardInput', (evt, data) => {
76
+ // Skip if non empty HTML data is included.
77
+ // https://github.com/ckeditor/ckeditor5-upload/issues/68
78
+ if (isHtmlIncluded(data.dataTransfer)) {
79
+ return;
80
+ }
81
+ const images = Array.from(data.dataTransfer.files).filter(file => {
82
+ // See https://github.com/ckeditor/ckeditor5-image/pull/254.
83
+ if (!file) {
84
+ return false;
85
+ }
86
+ return imageTypes.test(file.type);
87
+ });
88
+ if (!images.length) {
89
+ return;
90
+ }
91
+ evt.stop();
92
+ editor.model.change(writer => {
93
+ // Set selection to paste target.
94
+ if (data.targetRanges) {
95
+ writer.setSelection(data.targetRanges.map(viewRange => editor.editing.mapper.toModelRange(viewRange)));
96
+ }
97
+ editor.execute('uploadImage', { file: images });
98
+ });
99
+ });
100
+ // Handle HTML pasted with images with base64 or blob sources.
101
+ // For every image file, a new file loader is created and a placeholder image is
102
+ // inserted into the content. Then, those images are uploaded once they appear in the model
103
+ // (see Document#change listener below).
104
+ this.listenTo(clipboardPipeline, 'inputTransformation', (evt, data) => {
105
+ const fetchableImages = Array.from(editor.editing.view.createRangeIn(data.content))
106
+ .map(value => value.item)
107
+ .filter(viewElement => isLocalImage(imageUtils, viewElement) &&
108
+ !viewElement.getAttribute('uploadProcessed'))
109
+ .map(viewElement => { return { promise: fetchLocalImage(viewElement), imageElement: viewElement }; });
110
+ if (!fetchableImages.length) {
111
+ return;
112
+ }
113
+ const writer = new UpcastWriter(editor.editing.view.document);
114
+ for (const fetchableImage of fetchableImages) {
115
+ // Set attribute marking that the image was processed already.
116
+ writer.setAttribute('uploadProcessed', true, fetchableImage.imageElement);
117
+ const loader = fileRepository.createLoader(fetchableImage.promise);
118
+ if (loader) {
119
+ writer.setAttribute('src', '', fetchableImage.imageElement);
120
+ writer.setAttribute('uploadId', loader.id, fetchableImage.imageElement);
121
+ }
122
+ }
123
+ });
124
+ // Prevents from the browser redirecting to the dropped image.
125
+ editor.editing.view.document.on('dragover', (evt, data) => {
126
+ data.preventDefault();
127
+ });
128
+ // Upload placeholder images that appeared in the model.
129
+ doc.on('change', () => {
130
+ // Note: Reversing changes to start with insertions and only then handle removals. If it was the other way around,
131
+ // loaders for **all** images that land in the $graveyard would abort while in fact only those that were **not** replaced
132
+ // by other images should be aborted.
133
+ const changes = doc.differ.getChanges({ includeChangesInGraveyard: true }).reverse();
134
+ const insertedImagesIds = new Set();
135
+ for (const entry of changes) {
136
+ if (entry.type == 'insert' && entry.name != '$text') {
137
+ const item = entry.position.nodeAfter;
138
+ const isInsertedInGraveyard = entry.position.root.rootName == '$graveyard';
139
+ for (const imageElement of getImagesFromChangeItem(editor, item)) {
140
+ // Check if the image element still has upload id.
141
+ const uploadId = imageElement.getAttribute('uploadId');
142
+ if (!uploadId) {
143
+ continue;
144
+ }
145
+ // Check if the image is loaded on this client.
146
+ const loader = fileRepository.loaders.get(uploadId);
147
+ if (!loader) {
148
+ continue;
149
+ }
150
+ if (isInsertedInGraveyard) {
151
+ // If the image was inserted to the graveyard for good (**not** replaced by another image),
152
+ // only then abort the loading process.
153
+ if (!insertedImagesIds.has(uploadId)) {
154
+ loader.abort();
155
+ }
156
+ }
157
+ else {
158
+ // Remember the upload id of the inserted image. If it acted as a replacement for another
159
+ // image (which landed in the $graveyard), the related loader will not be aborted because
160
+ // this is still the same image upload.
161
+ insertedImagesIds.add(uploadId);
162
+ // Keep the mapping between the upload ID and the image model element so the upload
163
+ // can later resolve in the context of the correct model element. The model element could
164
+ // change for the same upload if one image was replaced by another (e.g. image type was changed),
165
+ // so this may also replace an existing mapping.
166
+ this._uploadImageElements.set(uploadId, imageElement);
167
+ if (loader.status == 'idle') {
168
+ // If the image was inserted into content and has not been loaded yet, start loading it.
169
+ this._readAndUpload(loader);
170
+ }
171
+ }
172
+ }
173
+ }
174
+ }
175
+ });
176
+ // Set the default handler for feeding the image element with `src` and `srcset` attributes.
177
+ // Also set the natural `width` and `height` attributes (if not already set).
178
+ this.on('uploadComplete', (evt, { imageElement, data }) => {
179
+ const urls = data.urls ? data.urls : data;
180
+ this.editor.model.change(writer => {
181
+ writer.setAttribute('src', urls.default, imageElement);
182
+ this._parseAndSetSrcsetAttributeOnImage(urls, imageElement, writer);
183
+ imageUtils.setImageNaturalSizeAttributes(imageElement);
184
+ });
185
+ }, { priority: 'low' });
186
+ }
187
+ /**
188
+ * @inheritDoc
189
+ */
190
+ afterInit() {
191
+ const schema = this.editor.model.schema;
192
+ // Setup schema to allow uploadId and uploadStatus for images.
193
+ // Wait for ImageBlockEditing or ImageInlineEditing to register their elements first,
194
+ // that's why doing this in afterInit() instead of init().
195
+ if (this.editor.plugins.has('ImageBlockEditing')) {
196
+ schema.extend('imageBlock', {
197
+ allowAttributes: ['uploadId', 'uploadStatus']
198
+ });
199
+ }
200
+ if (this.editor.plugins.has('ImageInlineEditing')) {
201
+ schema.extend('imageInline', {
202
+ allowAttributes: ['uploadId', 'uploadStatus']
203
+ });
204
+ }
205
+ }
206
+ /**
207
+ * Reads and uploads an image.
208
+ *
209
+ * The image is read from the disk and as a Base64-encoded string it is set temporarily to
210
+ * `image[src]`. When the image is successfully uploaded, the temporary data is replaced with the target
211
+ * image's URL (the URL to the uploaded image on the server).
212
+ */
213
+ _readAndUpload(loader) {
214
+ const editor = this.editor;
215
+ const model = editor.model;
216
+ const t = editor.locale.t;
217
+ const fileRepository = editor.plugins.get(FileRepository);
218
+ const notification = editor.plugins.get(Notification);
219
+ const imageUtils = editor.plugins.get('ImageUtils');
220
+ const imageUploadElements = this._uploadImageElements;
221
+ model.enqueueChange({ isUndoable: false }, writer => {
222
+ writer.setAttribute('uploadStatus', 'reading', imageUploadElements.get(loader.id));
223
+ });
224
+ return loader.read()
225
+ .then(() => {
226
+ const promise = loader.upload();
227
+ const imageElement = imageUploadElements.get(loader.id);
228
+ // Force re–paint in Safari. Without it, the image will display with a wrong size.
229
+ // https://github.com/ckeditor/ckeditor5/issues/1975
230
+ /* istanbul ignore next -- @preserve */
231
+ if (env.isSafari) {
232
+ const viewFigure = editor.editing.mapper.toViewElement(imageElement);
233
+ const viewImg = imageUtils.findViewImgElement(viewFigure);
234
+ editor.editing.view.once('render', () => {
235
+ // Early returns just to be safe. There might be some code ran
236
+ // in between the outer scope and this callback.
237
+ if (!viewImg.parent) {
238
+ return;
239
+ }
240
+ const domFigure = editor.editing.view.domConverter.mapViewToDom(viewImg.parent);
241
+ if (!domFigure) {
242
+ return;
243
+ }
244
+ const originalDisplay = domFigure.style.display;
245
+ domFigure.style.display = 'none';
246
+ // Make sure this line will never be removed during minification for having "no effect".
247
+ domFigure._ckHack = domFigure.offsetHeight;
248
+ domFigure.style.display = originalDisplay;
249
+ });
250
+ }
251
+ model.enqueueChange({ isUndoable: false }, writer => {
252
+ writer.setAttribute('uploadStatus', 'uploading', imageElement);
253
+ });
254
+ return promise;
255
+ })
256
+ .then(data => {
257
+ model.enqueueChange({ isUndoable: false }, writer => {
258
+ const imageElement = imageUploadElements.get(loader.id);
259
+ writer.setAttribute('uploadStatus', 'complete', imageElement);
260
+ this.fire('uploadComplete', { data, imageElement });
261
+ });
262
+ clean();
263
+ })
264
+ .catch(error => {
265
+ // If status is not 'error' nor 'aborted' - throw error because it means that something else went wrong,
266
+ // it might be generic error and it would be real pain to find what is going on.
267
+ if (loader.status !== 'error' && loader.status !== 'aborted') {
268
+ throw error;
269
+ }
270
+ // Might be 'aborted'.
271
+ if (loader.status == 'error' && error) {
272
+ notification.showWarning(error, {
273
+ title: t('Upload failed'),
274
+ namespace: 'upload'
275
+ });
276
+ }
277
+ // Permanently remove image from insertion batch.
278
+ model.enqueueChange({ isUndoable: false }, writer => {
279
+ writer.remove(imageUploadElements.get(loader.id));
280
+ });
281
+ clean();
282
+ });
283
+ function clean() {
284
+ model.enqueueChange({ isUndoable: false }, writer => {
285
+ const imageElement = imageUploadElements.get(loader.id);
286
+ writer.removeAttribute('uploadId', imageElement);
287
+ writer.removeAttribute('uploadStatus', imageElement);
288
+ imageUploadElements.delete(loader.id);
289
+ });
290
+ fileRepository.destroyLoader(loader);
291
+ }
292
+ }
293
+ /**
294
+ * Creates the `srcset` attribute based on a given file upload response and sets it as an attribute to a specific image element.
295
+ *
296
+ * @param data Data object from which `srcset` will be created.
297
+ * @param image The image element on which the `srcset` attribute will be set.
298
+ */
299
+ _parseAndSetSrcsetAttributeOnImage(data, image, writer) {
300
+ // Srcset attribute for responsive images support.
301
+ let maxWidth = 0;
302
+ const srcsetAttribute = Object.keys(data)
303
+ // Filter out keys that are not integers.
304
+ .filter(key => {
305
+ const width = parseInt(key, 10);
306
+ if (!isNaN(width)) {
307
+ maxWidth = Math.max(maxWidth, width);
308
+ return true;
309
+ }
310
+ })
311
+ // Convert each key to srcset entry.
312
+ .map(key => `${data[key]} ${key}w`)
313
+ // Join all entries.
314
+ .join(', ');
315
+ if (srcsetAttribute != '') {
316
+ const attributes = {
317
+ srcset: srcsetAttribute
318
+ };
319
+ if (!image.hasAttribute('width') && !image.hasAttribute('height')) {
320
+ attributes.width = maxWidth;
321
+ }
322
+ writer.setAttributes(attributes, image);
323
+ }
324
+ }
325
+ }
326
+ /**
327
+ * Returns `true` if non-empty `text/html` is included in the data transfer.
328
+ */
329
+ export function isHtmlIncluded(dataTransfer) {
330
+ return Array.from(dataTransfer.types).includes('text/html') && dataTransfer.getData('text/html') !== '';
331
+ }
332
+ function getImagesFromChangeItem(editor, item) {
333
+ const imageUtils = editor.plugins.get('ImageUtils');
334
+ return Array.from(editor.model.createRangeOn(item))
335
+ .filter(value => imageUtils.isImage(value.item))
336
+ .map(value => value.item);
337
+ }