@anu3ev/fabric-image-editor 0.1.13 → 0.1.19

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.
package/dist/main.js DELETED
@@ -1,3290 +0,0 @@
1
- import { ActiveSelection as z, util as U, controlsUtils as Oe, InteractiveFabricObject as Te, loadSVGFromURL as Ee, FabricImage as de, Point as ke, Rect as xe, Circle as Be, Triangle as Ze, Group as ze, Pattern as Ue, Canvas as Pe } from "fabric";
2
- import { create as Ye } from "jsondiffpatch";
3
- var He = "useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict", I = function() {
4
- for (var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : 21, t = "", a = crypto.getRandomValues(new Uint8Array(e |= 0)); e--; )
5
- t += He[a[e] & 63];
6
- return t;
7
- };
8
- function le(o, e, t, a, r, n, i) {
9
- try {
10
- var s = o[n](i), c = s.value;
11
- } catch (u) {
12
- return void t(u);
13
- }
14
- s.done ? e(c) : Promise.resolve(c).then(a, r);
15
- }
16
- function We(o) {
17
- return function() {
18
- var e = this, t = arguments;
19
- return new Promise(function(a, r) {
20
- var n = o.apply(e, t);
21
- function i(c) {
22
- le(n, a, r, i, s, "next", c);
23
- }
24
- function s(c) {
25
- le(n, a, r, i, s, "throw", c);
26
- }
27
- i(void 0);
28
- });
29
- };
30
- }
31
- class Z {
32
- /**
33
- * Конструктор принимает редактор и опции.
34
- * @param {Object} params
35
- * @param {ImageEditor} params.editor – редактор, содержащий canvas
36
- * @param {Object} params.options — настройки редактора (см. defaults.js)
37
- * @param {Boolean} [params.options.canvasDragging] — включить перетаскивание канваса
38
- * @param {Boolean} [params.options.mouseWheelZooming] — включить зум колесом мыши
39
- * @param {Boolean} [params.options.bringToFrontOnSelection] — поднимать объект на передний план при выборе
40
- * @param {Boolean} [params.options.copyObjectsByHotkey] — копировать объекты по Ctrl+C
41
- * @param {Boolean} [params.options.pasteImageFromClipboard] — вставлять изображения и объекты из буфера обмена
42
- * @param {Boolean} [params.options.undoRedoByHotKeys] — отмена/повтор по Ctrl+Z/Ctrl+Y
43
- * @param {Boolean} [params.options.selectAllByHotkey] — выделение всех объектов по Ctrl+A
44
- * @param {Boolean} [params.options.deleteObjectsByHotkey] — удаление объектов по Delete
45
- * @param {Boolean} [params.options.resetObjectFitByDoubleClick] — сброс фита объекта по двойному клику
46
- */
47
- constructor(e) {
48
- var {
49
- editor: t,
50
- options: a = {}
51
- } = e;
52
- this.editor = t, this.canvas = t.canvas, this.options = a, this.isUndoRedoKeyPressed = !1, this.handleAdaptCanvasToContainerBound = Z.debounce(this.handleAdaptCanvasToContainer.bind(this), 500), this.handleCopyEventBound = this.handleCopyEvent.bind(this), this.handlePasteEventBound = this.handlePasteEvent.bind(this), this.handleUndoRedoEventBound = this.handleUndoRedoEvent.bind(this), this.handleUndoRedoKeyUpBound = this.handleUndoRedoKeyUp.bind(this), this.handleSelectAllEventBound = this.handleSelectAllEvent.bind(this), this.handleDeleteObjectsEventBound = this.handleDeleteObjectsEvent.bind(this), this.handleObjectModifiedHistoryBound = Z.debounce(this.handleObjectModifiedHistory.bind(this), 300), this.handleObjectRotatingHistoryBound = Z.debounce(this.handleObjectRotatingHistory.bind(this), 300), this.handleObjectAddedHistoryBound = this.handleObjectAddedHistory.bind(this), this.handleObjectRemovedHistoryBound = this.handleObjectRemovedHistory.bind(this), this.handleOverlayUpdateBound = this.handleOverlayUpdate.bind(this), this.handleCanvasDragStartBound = this.handleCanvasDragStart.bind(this), this.handleCanvasDraggingBound = this.handleCanvasDragging.bind(this), this.handleCanvasDragEndBound = this.handleCanvasDragEnd.bind(this), this.handleMouseWheelZoomBound = this.handleMouseWheelZoom.bind(this), this.handleBringToFrontBound = this.handleBringToFront.bind(this), this.handleResetObjectFitBound = this.handleResetObjectFit.bind(this), this.handleLockedSelectionBound = this._filterLockedSelection.bind(this), this.init();
53
- }
54
- /**
55
- * Инициализация всех обработчиков согласно опциям.
56
- */
57
- init() {
58
- var {
59
- adaptCanvasToContainer: e,
60
- canvasDragging: t,
61
- mouseWheelZooming: a,
62
- bringToFrontOnSelection: r,
63
- copyObjectsByHotkey: n,
64
- pasteImageFromClipboard: i,
65
- undoRedoByHotKeys: s,
66
- selectAllByHotkey: c,
67
- deleteObjectsByHotkey: u,
68
- resetObjectFitByDoubleClick: l
69
- } = this.options;
70
- t && (this.canvas.on("mouse:down", this.handleCanvasDragStartBound), this.canvas.on("mouse:move", this.handleCanvasDraggingBound), this.canvas.on("mouse:up", this.handleCanvasDragEndBound)), a && this.canvas.on("mouse:wheel", this.handleMouseWheelZoomBound), r && (this.canvas.on("selection:created", this.handleBringToFrontBound), this.canvas.on("selection:updated", this.handleBringToFrontBound)), l && this.canvas.on("mouse:dblclick", this.handleResetObjectFitBound), e && window.addEventListener("resize", this.handleAdaptCanvasToContainerBound, {
71
- capture: !0
72
- }), n && document.addEventListener("keydown", this.handleCopyEventBound, {
73
- capture: !0
74
- }), i && document.addEventListener("paste", this.handlePasteEventBound, {
75
- capture: !0
76
- }), s && (document.addEventListener("keydown", this.handleUndoRedoEventBound, {
77
- capture: !0
78
- }), document.addEventListener("keyup", this.handleUndoRedoKeyUpBound, {
79
- capture: !0
80
- })), c && document.addEventListener("keydown", this.handleSelectAllEventBound, {
81
- capture: !0
82
- }), u && document.addEventListener("keydown", this.handleDeleteObjectsEventBound, {
83
- capture: !0
84
- }), this.canvas.on("object:modified", this.handleObjectModifiedHistoryBound), this.canvas.on("object:rotating", this.handleObjectRotatingHistoryBound), this.canvas.on("object:added", this.handleObjectAddedHistoryBound), this.canvas.on("object:removed", this.handleObjectRemovedHistoryBound), this.canvas.on("object:added", this.handleOverlayUpdateBound), this.canvas.on("selection:created", this.handleOverlayUpdateBound), this.canvas.on("selection:created", this.handleLockedSelectionBound.bind(this)), this.canvas.on("selection:updated", this.handleLockedSelectionBound.bind(this));
85
- }
86
- /**
87
- * При массовом выделении объектов удаляем из него залоченные.
88
- */
89
- _filterLockedSelection(e) {
90
- var {
91
- selected: t,
92
- e: a
93
- } = e;
94
- if (!(!(t != null && t.length) || !(a instanceof MouseEvent)) && t.length !== 1) {
95
- var r = t.some((s) => s.locked);
96
- if (r) {
97
- var n = t.filter((s) => !s.locked);
98
- if (n.length === 0) {
99
- this.canvas.discardActiveObject();
100
- return;
101
- }
102
- if (n.length === 1) {
103
- this.canvas.setActiveObject(n[0]);
104
- return;
105
- }
106
- var i = new z(n, {
107
- canvas: this.canvas
108
- });
109
- this.canvas.setActiveObject(i), this.canvas.requestRenderAll();
110
- }
111
- }
112
- }
113
- /**
114
- * Обработчики для сохранения состояния редактора в истории.
115
- * Срабатывают при изменении объектов (перемещение, изменение размера и т.д.).
116
- */
117
- handleObjectModifiedHistory() {
118
- this.editor.historyManager.skipHistory || this.editor.historyManager.saveState();
119
- }
120
- handleObjectRotatingHistory() {
121
- this.editor.historyManager.skipHistory || this.editor.historyManager.saveState();
122
- }
123
- handleObjectAddedHistory() {
124
- this.editor.historyManager.skipHistory || this.editor.historyManager.saveState();
125
- }
126
- handleObjectRemovedHistory() {
127
- this.editor.historyManager.skipHistory || this.editor.historyManager.saveState();
128
- }
129
- /**
130
- * Обновление overlayMask при добавлении объектов или выделении.
131
- */
132
- handleOverlayUpdate() {
133
- var {
134
- interactionBlocker: e
135
- } = this.editor;
136
- !e.isBlocked || !e.overlayMask || this.editor.interactionBlocker.refresh();
137
- }
138
- // --- Глобальные DOM-обработчики ---
139
- /**
140
- * Обработчик изменения размеров окна браузера.
141
- * Адаптирует канвас к размерам контейнера.
142
- */
143
- handleAdaptCanvasToContainer() {
144
- var {
145
- width: e,
146
- height: t
147
- } = this.editor.montageArea;
148
- this.editor.canvasManager.setResolutionWidth(e, {
149
- adaptCanvasToContainer: !0,
150
- withoutSave: !0
151
- }), this.editor.canvasManager.setResolutionHeight(t, {
152
- adaptCanvasToContainer: !0,
153
- withoutSave: !0
154
- }), this.editor.canvasManager.centerMontageArea(), this.editor.selectionManager.selectAll(), this.editor.transformManager.fitObject({
155
- fitAsOneObject: !0,
156
- withoutSave: !0
157
- });
158
- }
159
- /**
160
- * Обработчик для Ctrl+C (копирование).
161
- * @param {Object} event — объект события
162
- * @param {Boolean} event.ctrlKey — зажата ли клавиша Ctrl
163
- * @param {Boolean} event.metaKey — зажата ли клавиша Cmd (для Mac)
164
- * @param {String} event.code — код клавиши
165
- */
166
- handleCopyEvent(e) {
167
- var {
168
- ctrlKey: t,
169
- metaKey: a,
170
- code: r
171
- } = e;
172
- !t && !a || r !== "KeyC" || (e.preventDefault(), this.editor.clipboardManager.copy());
173
- }
174
- /**
175
- * Обработчик вставки объекта или изображения из буфера обмена.
176
- * @param {Object} event — объект события
177
- * @param {Object} event.clipboardData — данные из буфера обмена
178
- * @param {Array} event.clipboardData.items — элементы буфера обмена
179
- *
180
- * TODO: Попробовать вынести методы в классы-менеджеры.
181
- */
182
- handlePasteEvent(e) {
183
- var t, {
184
- clipboardData: a
185
- } = e;
186
- if (a != null && (t = a.items) !== null && t !== void 0 && t.length) {
187
- var {
188
- imageManager: r
189
- } = this.editor, {
190
- items: n
191
- } = a, i = n[n.length - 1];
192
- if (i.type.indexOf("image") !== -1) {
193
- var s = i.getAsFile();
194
- if (!s) return;
195
- var c = new FileReader();
196
- c.onload = (g) => {
197
- this.editor.imageManager.importImage({
198
- source: g.target.result
199
- });
200
- }, c.readAsDataURL(s);
201
- return;
202
- }
203
- var u = a.getData("text/html");
204
- if (u) {
205
- var l = new DOMParser(), d = l.parseFromString(u, "text/html"), h = d.querySelector("img");
206
- if (h != null && h.src) {
207
- r.importImage({
208
- source: h.src
209
- });
210
- return;
211
- }
212
- }
213
- this.editor.clipboardManager.paste();
214
- }
215
- }
216
- /**
217
- * Обработчик для отмены/повтора (Ctrl+Z/Ctrl+Y).
218
- * @param {Object} event — объект события
219
- * @param {Boolean} event.ctrlKey — зажата ли клавиша Ctrl
220
- * @param {Boolean} event.metaKey — зажата ли клавиша Cmd (для Mac)
221
- * @param {String} event.code — код клавиши
222
- */
223
- handleUndoRedoEvent(e) {
224
- var t = this;
225
- return We(function* () {
226
- var {
227
- ctrlKey: a,
228
- metaKey: r,
229
- code: n,
230
- repeat: i
231
- } = e;
232
- !a && !r || i || t.isUndoRedoKeyPressed || (n === "KeyZ" ? (e.preventDefault(), t.isUndoRedoKeyPressed = !0, yield t.editor.historyManager.undo()) : n === "KeyY" && (e.preventDefault(), t.isUndoRedoKeyPressed = !0, yield t.editor.historyManager.redo()));
233
- })();
234
- }
235
- /**
236
- * Обработчик для отпускания клавиш Ctrl+Z/Ctrl+Y.
237
- * @param {Object} event — объект события
238
- * @param {String} event.code — код клавиши
239
- */
240
- handleUndoRedoKeyUp(e) {
241
- var {
242
- code: t
243
- } = e;
244
- (t === "KeyZ" || t === "KeyY") && (this.isUndoRedoKeyPressed = !1);
245
- }
246
- /**
247
- * Обработчик для выделения всех объектов (Ctrl+A).
248
- * @param {Object} event — объект события
249
- * @param {Boolean} event.ctrlKey — зажата ли клавиша Ctrl
250
- * @param {Boolean} event.metaKey — зажата ли клавиша Cmd (для Mac)
251
- * @param {String} event.code — код клавиши
252
- */
253
- handleSelectAllEvent(e) {
254
- var {
255
- ctrlKey: t,
256
- metaKey: a,
257
- code: r
258
- } = e;
259
- !t && !a || r !== "KeyA" || (e.preventDefault(), this.editor.selectionManager.selectAll());
260
- }
261
- /**
262
- * Обработчик для удаления объектов (Delete).
263
- * @param {Object} event — объект события
264
- * @param {String} event.code — код клавиши
265
- */
266
- handleDeleteObjectsEvent(e) {
267
- e.code === "Delete" && (e.preventDefault(), this.editor.deletionManager.deleteSelectedObjects());
268
- }
269
- // --- Обработчики для событий canvas (Fabric) ---
270
- /**
271
- * Начало перетаскивания канваса (срабатывает при mouse:down).
272
- * @param {Object} options
273
- * @param {Object} options.e — объект события
274
- */
275
- handleCanvasDragStart(e) {
276
- var {
277
- e: t
278
- } = e;
279
- t.altKey && (this.canvas.isDragging = !0, this.canvas.selection = !1, this.canvas.lastPosX = t.clientX, this.canvas.lastPosY = t.clientY);
280
- }
281
- /**
282
- * Перетаскивание канваса (mouse:move).
283
- * @param {Object} options
284
- * @param {Object} options.e — объект события
285
- *
286
- * TODO: Надо как-то ограничить область перетаскивания, чтобы канвас не уходил сильно далеко за пределы экрана
287
- */
288
- handleCanvasDragging(e) {
289
- var {
290
- e: t
291
- } = e;
292
- if (this.canvas.isDragging) {
293
- var a = this.canvas.viewportTransform;
294
- a[4] += t.clientX - this.canvas.lastPosX, a[5] += t.clientY - this.canvas.lastPosY, this.canvas.requestRenderAll(), this.canvas.lastPosX = t.clientX, this.canvas.lastPosY = t.clientY;
295
- }
296
- }
297
- /**
298
- * Завершение перетаскивания канваса (mouse:up).
299
- * Сохраняет новое положение канваса.
300
- */
301
- handleCanvasDragEnd() {
302
- this.canvas.setViewportTransform(this.canvas.viewportTransform), this.canvas.isDragging = !1, this.canvas.selection = !0;
303
- }
304
- /**
305
- * Обработчик зума колесиком мыши.
306
- * @param {Object} options
307
- * @param {Object} options.e — объект события
308
- */
309
- handleMouseWheelZoom(e) {
310
- var {
311
- e: t
312
- } = e, a = 1e-3, r = -t.deltaY * a;
313
- this.editor.transformManager.zoom(r), t.preventDefault(), t.stopPropagation();
314
- }
315
- /**
316
- * Обработчик, поднимающий выделенные объекты на передний план.
317
- * @param {Object} event - объект события Fabric
318
- * @param {Array} event.selected - массив выбранных объектов
319
- */
320
- handleBringToFront(e) {
321
- var {
322
- selected: t
323
- } = e;
324
- t != null && t.length && t.forEach((a) => {
325
- this.editor.layerManager.bringToFront(a);
326
- });
327
- }
328
- /**
329
- * Обработчик сброса объекта по двойному клику.
330
- */
331
- handleResetObjectFit(e) {
332
- var {
333
- target: t
334
- } = e;
335
- t && this.editor.transformManager.resetObject(t);
336
- }
337
- /**
338
- * Метод для удаления всех слушателей
339
- */
340
- destroy() {
341
- window.removeEventListener("resize", this.handleAdaptCanvasToContainerBound, {
342
- capture: !0
343
- }), document.removeEventListener("keydown", this.handleCopyEventBound, {
344
- capture: !0
345
- }), document.removeEventListener("paste", this.handlePasteEventBound, {
346
- capture: !0
347
- }), document.removeEventListener("keydown", this.handleUndoRedoEventBound, {
348
- capture: !0
349
- }), document.removeEventListener("keyup", this.handleUndoRedoKeyUpBound, {
350
- capture: !0
351
- }), document.removeEventListener("keydown", this.handleSelectAllEventBound, {
352
- capture: !0
353
- }), document.removeEventListener("keydown", this.handleDeleteObjectsEventBound, {
354
- capture: !0
355
- }), this.options.canvasDragging && (this.canvas.off("mouse:down", this.handleCanvasDragStartBound), this.canvas.off("mouse:move", this.handleCanvasDraggingBound), this.canvas.off("mouse:up", this.handleCanvasDragEndBound)), this.options.mouseWheelZooming && this.canvas.off("mouse:wheel", this.handleMouseWheelZoomBound), this.options.bringToFrontOnSelection && (this.canvas.off("selection:created", this.handleBringToFrontBound), this.canvas.off("selection:updated", this.handleBringToFrontBound)), this.options.resetObjectFitByDoubleClick && this.canvas.off("mouse:dblclick", this.handleResetObjectFitBound), this.canvas.off("object:modified", this.handleObjectModifiedHistoryBound), this.canvas.off("object:rotating", this.handleObjectRotatingHistoryBound), this.canvas.off("object:added", this.handleObjectAddedHistoryBound), this.canvas.off("object:removed", this.handleObjectRemovedHistoryBound), this.canvas.off("object:added", this.handleOverlayUpdateBound), this.canvas.off("selection:created", this.handleOverlayUpdateBound), this.canvas.off("selection:created", this.handleLockedSelectionBound.bind(this)), this.canvas.off("selection:updated", this.handleLockedSelectionBound.bind(this));
356
- }
357
- // Дебаунс для снижения частоты сохранения состояния
358
- static debounce(e, t) {
359
- var a = null;
360
- return function() {
361
- for (var r = arguments.length, n = new Array(r), i = 0; i < r; i++)
362
- n[i] = arguments[i];
363
- var s = this;
364
- clearTimeout(a), a = setTimeout(() => {
365
- e.apply(s, n);
366
- }, t);
367
- };
368
- }
369
- }
370
- class _e {
371
- /**
372
- * @description Класс для динамической загрузки внешних модулей.
373
- */
374
- constructor() {
375
- this.cache = /* @__PURE__ */ new Map(), this.loaders = {
376
- jspdf: () => import("jspdf")
377
- };
378
- }
379
- /**
380
- * Загружает модуль по имени и сохраняет промис в кеше.
381
- * @param {string} name — строковый литерал, например 'fabric' или '../helpers'
382
- * @returns {Promise<Module>} — namespace-объект ES-модуля
383
- */
384
- loadModule(e) {
385
- return this.loaders[e] ? (this.cache.has(e) || this.cache.set(e, this.loaders[e]()), this.cache.get(e)) : Promise.reject(new Error('Unknown module "'.concat(e, '"')));
386
- }
387
- }
388
- class Re {
389
- /**
390
- * @param {string|URL} [scriptUrl] — URL скрипта воркера.
391
- * По-умолчанию использует файл рядом с этим модулем
392
- */
393
- constructor() {
394
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : new URL("" + new URL("worker.js", import.meta.url).href, import.meta.url);
395
- this.worker = new Worker(e, {
396
- type: "module"
397
- }), this._callbacks = /* @__PURE__ */ new Map(), this.worker.onmessage = this._handleMessage.bind(this);
398
- }
399
- /**
400
- * Обработчик сообщений от воркера
401
- * @param {Object} data
402
- * @param {String} data.action - название действия
403
- * @param {String} data.requestId - уникальный идентификатор запроса
404
- * @param {Boolean} data.success - успешность выполнения действия
405
- * @param {Object} data.data - данные, которые вернул воркер
406
- * @param {String} data.error - ошибка, если она произошла
407
- * @returns {void}
408
- */
409
- _handleMessage(e) {
410
- var {
411
- data: t
412
- } = e, {
413
- requestId: a,
414
- success: r,
415
- data: n,
416
- error: i
417
- } = t, s = this._callbacks.get(a);
418
- s && (r ? s.resolve(n) : s.reject(new Error(i)), this._callbacks.delete(a));
419
- }
420
- /**
421
- * Универсальный метод отправки команды в воркер
422
- * @param {String} action
423
- * @param {Object} payload
424
- * @param {Array} [transferables] - массив объектов, которые нужно передать в воркер
425
- * @returns {Promise<any>}
426
- */
427
- post(e, t) {
428
- var a = arguments.length > 2 && arguments[2] !== void 0 ? arguments[2] : [], r = "".concat(e, ":").concat(I(8));
429
- return new Promise((n, i) => {
430
- this._callbacks.set(r, {
431
- resolve: n,
432
- reject: i
433
- }), this.worker.postMessage({
434
- action: e,
435
- payload: t,
436
- requestId: r
437
- }, a);
438
- });
439
- }
440
- /**
441
- * Завершает работу воркера
442
- */
443
- terminate() {
444
- this.worker.terminate();
445
- }
446
- }
447
- var p = 12, Ve = 2, X = 8, Q = 20, Fe = 100, J = 20, K = 8, Ge = 100, $ = 32, q = 1, Xe = "#2B2D33", ee = "#3D8BF4", te = "#FFFFFF";
448
- function Y(o, e, t, a, r) {
449
- var n = p, i = Ve;
450
- o.save(), o.translate(e, t), o.rotate(U.degreesToRadians(r.angle)), o.fillStyle = te, o.strokeStyle = ee, o.lineWidth = q, o.beginPath(), o.roundRect(-12 / 2, -12 / 2, n, n, i), o.fill(), o.stroke(), o.restore();
451
- }
452
- function he(o, e, t, a, r) {
453
- var n = X, i = Q, s = Fe;
454
- o.save(), o.translate(e, t), o.rotate(U.degreesToRadians(r.angle)), o.fillStyle = te, o.strokeStyle = ee, o.lineWidth = q, o.beginPath(), o.roundRect(-8 / 2, -20 / 2, n, i, s), o.fill(), o.stroke(), o.restore();
455
- }
456
- function ge(o, e, t, a, r) {
457
- var n = J, i = K, s = Ge;
458
- o.save(), o.translate(e, t), o.rotate(U.degreesToRadians(r.angle)), o.fillStyle = te, o.strokeStyle = ee, o.lineWidth = q, o.beginPath(), o.roundRect(-20 / 2, -8 / 2, n, i, s), o.fill(), o.stroke(), o.restore();
459
- }
460
- var Qe = "", Ae = new Image();
461
- Ae.src = Qe;
462
- function Je(o, e, t, a, r) {
463
- var n = $, i = n / 2;
464
- o.save(), o.translate(e, t), o.rotate(U.degreesToRadians(r.angle)), o.fillStyle = Xe, o.beginPath(), o.arc(0, 0, i, 0, 2 * Math.PI), o.fill(), o.drawImage(Ae, -16 / 2, -16 / 2, i, i), o.restore();
465
- }
466
- var Ke = {
467
- // Угловые точки
468
- tl: {
469
- render: Y,
470
- sizeX: p,
471
- sizeY: p,
472
- offsetX: 0,
473
- offsetY: 0
474
- },
475
- tr: {
476
- render: Y,
477
- sizeX: p,
478
- sizeY: p,
479
- offsetX: 0,
480
- offsetY: 0
481
- },
482
- bl: {
483
- render: Y,
484
- sizeX: p,
485
- sizeY: p,
486
- offsetX: 0,
487
- offsetY: 0
488
- },
489
- br: {
490
- render: Y,
491
- sizeX: p,
492
- sizeY: p,
493
- offsetX: 0,
494
- offsetY: 0
495
- },
496
- // Середина вертикалей
497
- ml: {
498
- render: he,
499
- sizeX: X,
500
- sizeY: Q,
501
- offsetX: 0,
502
- offsetY: 0
503
- },
504
- mr: {
505
- render: he,
506
- sizeX: X,
507
- sizeY: Q,
508
- offsetX: 0,
509
- offsetY: 0
510
- },
511
- // Середина горизонталей
512
- mt: {
513
- render: ge,
514
- sizeX: J,
515
- sizeY: K,
516
- offsetX: 0,
517
- offsetY: 0
518
- },
519
- mb: {
520
- render: ge,
521
- sizeX: J,
522
- sizeY: K,
523
- offsetX: 0,
524
- offsetY: 0
525
- },
526
- // Специальный «rotate» контрол
527
- mtr: {
528
- render: Je,
529
- sizeX: $,
530
- sizeY: $,
531
- offsetX: 0,
532
- offsetY: -32
533
- }
534
- };
535
- class $e {
536
- static apply() {
537
- var e = Oe.createObjectDefaultControls();
538
- Object.entries(Ke).forEach((t) => {
539
- var [a, r] = t;
540
- Object.assign(e[a], {
541
- render: r.render,
542
- sizeX: r.sizeX,
543
- sizeY: r.sizeY,
544
- offsetX: r.offsetX,
545
- offsetY: r.offsetY
546
- });
547
- }), Te.ownDefaults.controls = e;
548
- }
549
- }
550
- var qe = "", et = "", tt = "", at = "", rt = "", nt = "", it = "", ot = "";
551
- function ve(o, e, t, a, r, n, i) {
552
- try {
553
- var s = o[n](i), c = s.value;
554
- } catch (u) {
555
- return void t(u);
556
- }
557
- s.done ? e(c) : Promise.resolve(c).then(a, r);
558
- }
559
- function st(o) {
560
- return function() {
561
- var e = this, t = arguments;
562
- return new Promise(function(a, r) {
563
- var n = o.apply(e, t);
564
- function i(c) {
565
- ve(n, a, r, i, s, "next", c);
566
- }
567
- function s(c) {
568
- ve(n, a, r, i, s, "throw", c);
569
- }
570
- i(void 0);
571
- });
572
- };
573
- }
574
- const k = {
575
- style: {
576
- position: "absolute",
577
- display: "none",
578
- background: "#2B2D33",
579
- borderRadius: "8px",
580
- padding: "0 8px",
581
- height: "32px",
582
- gap: "10px",
583
- zIndex: 10,
584
- alignItems: "center"
585
- },
586
- btnStyle: {
587
- background: "transparent",
588
- border: "none",
589
- display: "flex",
590
- alignItems: "center",
591
- justifyContent: "center",
592
- height: "20px",
593
- width: "20px",
594
- cursor: "pointer",
595
- transition: "background-color 0.2s ease, transform 0.1s ease",
596
- transform: "scale(1)"
597
- },
598
- btnHover: {
599
- backgroundColor: "rgba(255,255,255,0.15)",
600
- borderRadius: "50%",
601
- transform: "scale(1.1)"
602
- },
603
- toolbarClass: "fabric-editor-toolbar",
604
- btnClass: "fabric-editor-toolbar-btn",
605
- lockedActions: [{
606
- name: "Разблокировать",
607
- handle: "unlock"
608
- }],
609
- actions: [{
610
- name: "Создать копию",
611
- handle: "copyPaste"
612
- }, {
613
- name: "Заблокировать",
614
- handle: "lock"
615
- }, {
616
- name: "На передний план",
617
- handle: "bringToFront"
618
- }, {
619
- name: "На задний план",
620
- handle: "sendToBack"
621
- }, {
622
- name: "На один уровень вверх",
623
- handle: "bringForward"
624
- }, {
625
- name: "На один уровень вниз",
626
- handle: "sendBackwards"
627
- }, {
628
- name: "Удалить",
629
- handle: "delete"
630
- }],
631
- offsetTop: 50,
632
- icons: {
633
- copyPaste: qe,
634
- delete: ot,
635
- lock: et,
636
- unlock: tt,
637
- bringToFront: nt,
638
- sendToBack: it,
639
- bringForward: at,
640
- sendBackwards: rt
641
- },
642
- handlers: {
643
- copyPaste: function() {
644
- var o = st(function* (t) {
645
- yield t.clipboardManager.copy(), yield t.clipboardManager.paste();
646
- });
647
- function e(t) {
648
- return o.apply(this, arguments);
649
- }
650
- return e;
651
- }(),
652
- delete: (o) => {
653
- o.deletionManager.deleteSelectedObjects();
654
- },
655
- lock: (o) => {
656
- o.objectLockManager.lockObject();
657
- },
658
- unlock: (o) => {
659
- o.objectLockManager.unlockObject();
660
- },
661
- bringForward: (o) => {
662
- o.layerManager.bringForward();
663
- },
664
- bringToFront: (o) => {
665
- o.layerManager.bringToFront();
666
- },
667
- sendToBack: (o) => {
668
- o.layerManager.sendToBack();
669
- },
670
- sendBackwards: (o) => {
671
- o.layerManager.sendBackwards();
672
- }
673
- }
674
- };
675
- function me(o, e) {
676
- var t = Object.keys(o);
677
- if (Object.getOwnPropertySymbols) {
678
- var a = Object.getOwnPropertySymbols(o);
679
- e && (a = a.filter(function(r) {
680
- return Object.getOwnPropertyDescriptor(o, r).enumerable;
681
- })), t.push.apply(t, a);
682
- }
683
- return t;
684
- }
685
- function j(o) {
686
- for (var e = 1; e < arguments.length; e++) {
687
- var t = arguments[e] != null ? arguments[e] : {};
688
- e % 2 ? me(Object(t), !0).forEach(function(a) {
689
- ct(o, a, t[a]);
690
- }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(o, Object.getOwnPropertyDescriptors(t)) : me(Object(t)).forEach(function(a) {
691
- Object.defineProperty(o, a, Object.getOwnPropertyDescriptor(t, a));
692
- });
693
- }
694
- return o;
695
- }
696
- function ct(o, e, t) {
697
- return (e = ut(e)) in o ? Object.defineProperty(o, e, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : o[e] = t, o;
698
- }
699
- function ut(o) {
700
- var e = dt(o, "string");
701
- return typeof e == "symbol" ? e : e + "";
702
- }
703
- function dt(o, e) {
704
- if (typeof o != "object" || !o) return o;
705
- var t = o[Symbol.toPrimitive];
706
- if (t !== void 0) {
707
- var a = t.call(o, e);
708
- if (typeof a != "object") return a;
709
- throw new TypeError("@@toPrimitive must return a primitive value.");
710
- }
711
- return (e === "string" ? String : Number)(o);
712
- }
713
- class lt {
714
- /**
715
- * @param {object} options
716
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
717
- */
718
- constructor(e) {
719
- var {
720
- editor: t
721
- } = e;
722
- if (this.options = t.options, !!this.options.showToolbar) {
723
- this.editor = t, this.canvas = t.canvas;
724
- var a = this.options.toolbar || {};
725
- this.config = j(j(j({}, k), a), {}, {
726
- style: j(j({}, k.style), a.style || {}),
727
- btnStyle: j(j({}, k.btnStyle), a.btnStyle || {}),
728
- icons: j(j({}, k.icons), a.icons || {}),
729
- handlers: j(j({}, k.handlers), a.handlers || {})
730
- }), this.currentTarget = null, this.currentLocked = null, this.isTransforming = !1, this._onMouseDown = this._handleMouseDown.bind(this), this._onObjectMoving = this._startTransform.bind(this), this._onObjectScaling = this._startTransform.bind(this), this._onObjectRotating = this._startTransform.bind(this), this._onMouseUp = this._endTransform.bind(this), this._onObjectModified = this._endTransform.bind(this), this._onSelectionChange = this._updateToolbar.bind(this), this._onSelectionClear = () => {
731
- this.el.style.display = "none";
732
- }, this._createDOM(), this._bindEvents();
733
- }
734
- }
735
- /**
736
- * Создаёт DOM элемент панели инструментов и добавляет его в canvas
737
- * @private
738
- */
739
- _createDOM() {
740
- var {
741
- style: e
742
- } = this.config;
743
- this.el = document.createElement("div"), Object.assign(this.el.style, e), this.canvas.wrapperEl.appendChild(this.el), this._onBtnOver = (t) => {
744
- var a = t.target.closest("button");
745
- a && Object.assign(a.style, this.config.btnHover);
746
- }, this._onBtnOut = (t) => {
747
- var a = t.target.closest("button");
748
- a && Object.assign(a.style, this.config.btnStyle);
749
- }, this.el.addEventListener("mouseover", this._onBtnOver), this.el.addEventListener("mouseout", this._onBtnOut);
750
- }
751
- /**
752
- * Отрисовывает кнопки панели инструментов
753
- * @private
754
- * @param {array} actions - массив действий для отрисовки
755
- * @param {string} actions[].name - название действия
756
- * @param {string} actions[].handle - название обработчика
757
- */
758
- _renderButtons(e) {
759
- var t = this;
760
- this.el.innerHTML = "";
761
- var a = function() {
762
- var {
763
- name: i,
764
- handle: s
765
- } = r, {
766
- icons: c,
767
- btnStyle: u,
768
- handlers: l
769
- } = t.config, d = document.createElement("button");
770
- d.innerHTML = c[s] ? '<img src="'.concat(c[s], '" title="').concat(i, '" />') : i, Object.assign(d.style, u), d.onclick = () => {
771
- var h;
772
- return (h = l[s]) === null || h === void 0 ? void 0 : h.call(l, t.editor);
773
- }, t.el.appendChild(d);
774
- };
775
- for (var r of e)
776
- a();
777
- }
778
- /**
779
- * Привязывает события к canvas
780
- * @private
781
- */
782
- _bindEvents() {
783
- this.canvas.on("mouse:down", this._onMouseDown), this.canvas.on("object:moving", this._onObjectMoving), this.canvas.on("object:scaling", this._onObjectScaling), this.canvas.on("object:rotating", this._onObjectRotating), this.canvas.on("mouse:up", this._onMouseUp), this.canvas.on("object:modified", this._onObjectModified), this.canvas.on("selection:created", this._onSelectionChange), this.canvas.on("selection:updated", this._onSelectionChange), this.canvas.on("selection:changed", this._onSelectionChange), this.canvas.on("after:render", this._onSelectionChange), this.canvas.on("selection:cleared", this._onSelectionClear);
784
- }
785
- /**
786
- * На время трансформации скрываем тулбар
787
- * @private
788
- * @param {Object} opt - объект события
789
- */
790
- _handleMouseDown(e) {
791
- var t;
792
- (t = e.transform) !== null && t !== void 0 && t.actionPerformed && this._startTransform();
793
- }
794
- /**
795
- * Начало трансформации объекта
796
- * @private
797
- */
798
- _startTransform() {
799
- this.isTransforming = !0, this.el.style.display = "none";
800
- }
801
- /**
802
- * Завершение трансформации объекта
803
- * @private
804
- */
805
- _endTransform() {
806
- this.isTransforming = !1, this._updatePos();
807
- }
808
- /**
809
- * Обновляет панель инструментов в зависимости от выделенного объекта и его состояния
810
- * @private
811
- */
812
- _updateToolbar() {
813
- if (!this.isTransforming) {
814
- var e = this.canvas.getActiveObject();
815
- if (!e) {
816
- this.el.style.display = "none", this.currentTarget = null;
817
- return;
818
- }
819
- var t = !!e.locked;
820
- if (e !== this.currentTarget || t !== this.currentLocked) {
821
- this.currentTarget = e, this.currentLocked = t;
822
- var a = t ? this.config.lockedActions : this.config.actions;
823
- this._renderButtons(a);
824
- }
825
- this._updatePos();
826
- }
827
- }
828
- /**
829
- * Обновляет позицию панели инструментов в зависимости от положения выделенного объекта
830
- * @private
831
- */
832
- _updatePos() {
833
- if (!this.isTransforming) {
834
- var e = this.canvas.getActiveObject();
835
- if (!e) {
836
- this.el.style.display = "none";
837
- return;
838
- }
839
- var {
840
- el: t,
841
- config: a,
842
- canvas: r
843
- } = this;
844
- e.setCoords();
845
- var n = r.getZoom(), [, , , , i, s] = r.viewportTransform, {
846
- x: c
847
- } = e.getCenterPoint(), {
848
- top: u,
849
- height: l
850
- } = e.getBoundingRect(!1, !0), d = c * n + i, h = d - t.offsetWidth / 2, g = (u + l) * n + s + a.offsetTop;
851
- Object.assign(t.style, {
852
- left: "".concat(h, "px"),
853
- top: "".concat(g, "px"),
854
- display: "flex"
855
- });
856
- }
857
- }
858
- /**
859
- * Удаляет слушатели событий и DOM элемент панели инструментов
860
- */
861
- destroy() {
862
- this.el.removeEventListener("mouseover", this._onBtnOver), this.el.removeEventListener("mouseout", this._onBtnOut), this.canvas.off("mouse:down", this._onMouseDown), this.canvas.off("object:moving", this._onObjectMoving), this.canvas.off("object:scaling", this._onObjectScaling), this.canvas.off("object:rotating", this._onObjectRotating), this.canvas.off("mouse:up", this._onMouseUp), this.canvas.off("object:modified", this._onObjectModified), this.canvas.off("selection:created", this._onSelectionChange), this.canvas.off("selection:updated", this._onSelectionChange), this.canvas.off("selection:changed", this._onSelectionChange), this.canvas.off("after:render", this._onSelectionChange), this.canvas.off("selection:cleared", this._onSelectionClear), this.el.remove();
863
- }
864
- }
865
- function fe(o, e, t, a, r, n, i) {
866
- try {
867
- var s = o[n](i), c = s.value;
868
- } catch (u) {
869
- return void t(u);
870
- }
871
- s.done ? e(c) : Promise.resolve(c).then(a, r);
872
- }
873
- function V(o) {
874
- return function() {
875
- var e = this, t = arguments;
876
- return new Promise(function(a, r) {
877
- var n = o.apply(e, t);
878
- function i(c) {
879
- fe(n, a, r, i, s, "next", c);
880
- }
881
- function s(c) {
882
- fe(n, a, r, i, s, "throw", c);
883
- }
884
- i(void 0);
885
- });
886
- };
887
- }
888
- class ht {
889
- /**
890
- * @param {object} options
891
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
892
- */
893
- constructor(e) {
894
- var {
895
- editor: t
896
- } = e;
897
- this.editor = t, this.canvas = t.canvas, this._historySuspendCount = 0, this.baseState = null, this.patches = [], this.currentIndex = 0, this.maxHistoryLength = t.options.maxHistoryLength, this.totalChangesCount = 0, this.baseStateChangesCount = 0, this._createDiffPatcher();
898
- }
899
- /** Проверка, нужно ли пропускать сохранение истории */
900
- get skipHistory() {
901
- return this._historySuspendCount > 0;
902
- }
903
- _createDiffPatcher() {
904
- this.diffPatcher = Ye({
905
- objectHash(e) {
906
- return [e.id, e.format, e.locked, e.left, e.top, e.width, e.height, e.flipX, e.flipY, e.scaleX, e.scaleY, e.angle, e.opacity].join("-");
907
- },
908
- arrays: {
909
- detectMove: !0,
910
- includeValueOnMove: !1
911
- },
912
- textDiff: {
913
- minLength: 60
914
- }
915
- });
916
- }
917
- /** Увеличить счётчик приостановки истории */
918
- suspendHistory() {
919
- this._historySuspendCount += 1;
920
- }
921
- /** Уменьшить счётчик приостановки истории */
922
- resumeHistory() {
923
- this._historySuspendCount = Math.max(0, this._historySuspendCount - 1);
924
- }
925
- /**
926
- * Проверяет, есть ли в редакторе несохранённые изменения
927
- * @returns {boolean}
928
- */
929
- hasUnsavedChanges() {
930
- return this.totalChangesCount > 0;
931
- }
932
- /**
933
- * Получает текущую позицию в общей истории изменений
934
- * @returns {number}
935
- */
936
- getCurrentChangePosition() {
937
- return this.baseStateChangesCount + this.currentIndex;
938
- }
939
- /**
940
- * Получаем полное состояние, применяя все диффы к базовому состоянию.
941
- */
942
- getFullState() {
943
- for (var {
944
- baseState: e,
945
- currentIndex: t,
946
- patches: a
947
- } = this, r = JSON.parse(JSON.stringify(e)), n = 0; n < t; n += 1)
948
- r = this.diffPatcher.patch(r, a[n]);
949
- return console.log("getFullState state", r), r;
950
- }
951
- /**
952
- * Сохраняем текущее состояние в виде диффа от последнего сохранённого полного состояния.
953
- */
954
- saveState() {
955
- if (console.log("saveState"), !this.skipHistory) {
956
- console.time("saveState");
957
- var e = this.canvas.toDatalessObject(["selectable", "evented", "id", "format", "width", "height", "locked", "lockMovementX", "lockMovementY", "lockRotation", "lockScalingX", "lockScalingY", "lockSkewingX", "lockSkewingY"]);
958
- if (console.timeEnd("saveState"), !this.baseState) {
959
- this.baseState = e, this.patches = [], this.currentIndex = 0, console.log("Базовое состояние сохранено.");
960
- return;
961
- }
962
- var t = this.getFullState(), a = this.diffPatcher.diff(t, e);
963
- if (!a) {
964
- console.log("Нет изменений для сохранения.");
965
- return;
966
- }
967
- console.log("baseState", this.baseState), this.currentIndex < this.patches.length && this.patches.splice(this.currentIndex), console.log("diff", a), this.totalChangesCount += 1, this.patches.push(a), this.currentIndex += 1, this.patches.length > this.maxHistoryLength && (this.baseState = this.diffPatcher.patch(this.baseState, this.patches[0]), this.patches.shift(), this.currentIndex -= 1, this.baseStateChangesCount += 1), console.log("Состояние сохранено. Текущий индекс истории:", this.currentIndex);
968
- }
969
- }
970
- /**
971
- * Функция загрузки состояния в канвас.
972
- * @param {object} fullState - полное состояние канваса
973
- * @fires editor:history-state-loaded
974
- */
975
- loadStateFromFullState(e) {
976
- var t = this;
977
- return V(function* () {
978
- if (e) {
979
- console.log("loadStateFromFullState fullState", e), yield t.canvas.loadFromJSON(e);
980
- var a = t.canvas.getObjects().find((n) => n.id === "montage-area");
981
- a && (t.editor.montageArea = a);
982
- var r = t.canvas.getObjects().find((n) => n.id === "overlay-mask");
983
- r && (t.editor.interactionBlocker.overlayMask = r, t.editor.interactionBlocker.overlayMask.visible = !1), t.canvas.renderAll(), t.canvas.fire("editor:history-state-loaded");
984
- }
985
- })();
986
- }
987
- /**
988
- * Undo – отмена последнего действия, восстанавливая состояние по накопленным диффам.
989
- * @fires editor:undo
990
- */
991
- undo() {
992
- var e = this;
993
- return V(function* () {
994
- if (!e.skipHistory) {
995
- if (e.currentIndex <= 0) {
996
- console.log("Нет предыдущих состояний для отмены.");
997
- return;
998
- }
999
- e.suspendHistory();
1000
- try {
1001
- e.currentIndex -= 1, e.totalChangesCount -= 1;
1002
- var t = e.getFullState();
1003
- yield e.loadStateFromFullState(t), console.log("Undo выполнен. Текущий индекс истории:", e.currentIndex), e.canvas.fire("editor:undo");
1004
- } catch (a) {
1005
- console.error("undo. Ошибка отмены действия: ", a), e.canvas.fire("editor:error", {
1006
- message: "Ошибка отмены действия: ".concat(a.message)
1007
- });
1008
- } finally {
1009
- e.resumeHistory();
1010
- }
1011
- }
1012
- })();
1013
- }
1014
- /**
1015
- * Redo – повтор ранее отменённого действия.
1016
- * @fires editor:redo
1017
- */
1018
- redo() {
1019
- var e = this;
1020
- return V(function* () {
1021
- if (!e.skipHistory) {
1022
- if (e.currentIndex >= e.patches.length) {
1023
- console.log("Нет состояний для повтора.");
1024
- return;
1025
- }
1026
- e.suspendHistory();
1027
- try {
1028
- e.currentIndex += 1, e.totalChangesCount += 1;
1029
- var t = e.getFullState();
1030
- console.log("fullState", t), yield e.loadStateFromFullState(t), console.log("Redo выполнен. Текущий индекс истории:", e.currentIndex), e.canvas.fire("editor:redo");
1031
- } catch (a) {
1032
- console.error("redo. Ошибка повтора действия: ", a), e.canvas.fire("editor:error", {
1033
- message: "Ошибка повтора действия: ".concat(a.message)
1034
- });
1035
- } finally {
1036
- e.resumeHistory();
1037
- }
1038
- }
1039
- })();
1040
- }
1041
- }
1042
- var gt = 0.1, vt = 2, mt = 0.1, ft = 90, x = 16, B = 16, D = 4096, N = 4096;
1043
- function Me(o, e, t, a, r, n, i) {
1044
- try {
1045
- var s = o[n](i), c = s.value;
1046
- } catch (u) {
1047
- return void t(u);
1048
- }
1049
- s.done ? e(c) : Promise.resolve(c).then(a, r);
1050
- }
1051
- function T(o) {
1052
- return function() {
1053
- var e = this, t = arguments;
1054
- return new Promise(function(a, r) {
1055
- var n = o.apply(e, t);
1056
- function i(c) {
1057
- Me(n, a, r, i, s, "next", c);
1058
- }
1059
- function s(c) {
1060
- Me(n, a, r, i, s, "throw", c);
1061
- }
1062
- i(void 0);
1063
- });
1064
- };
1065
- }
1066
- class C {
1067
- /**
1068
- * @param {object} options
1069
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
1070
- */
1071
- constructor(e) {
1072
- var {
1073
- editor: t
1074
- } = e;
1075
- this.editor = t, this.options = t.options, this._createdBlobUrls = [];
1076
- }
1077
- /**
1078
- * Импорт изображения
1079
- * @param {Object} options
1080
- * @param {File|string} [options.source] - URL изображения или объект File
1081
- * @param {String} [options.scale] - Если изображение не вписывается в допустимые размеры, то как масштабировать:
1082
- * 'image-contain' - скейлит картинку, чтобы она вписалась в монтажную область
1083
- * 'image-cover' - скейлит картинку, чтобы она вписалась в монтажную область
1084
- * 'scale-montage' - Обновляет backstore-резолюцию монтажной области (масштабирует
1085
- * экспортный размер канваса под размер изображения)
1086
- * @param {Boolean} [options.withoutSave] - Не сохранять в историю изменений
1087
- */
1088
- importImage(e) {
1089
- var t = this;
1090
- return T(function* () {
1091
- var {
1092
- source: a,
1093
- scale: r = "image-".concat(t.options.scaleType),
1094
- withoutSave: n = !1
1095
- } = e;
1096
- if (a) {
1097
- var i = yield t.getContentType(a);
1098
- if (!t.isAllowedContentType(i)) {
1099
- var s = "Неверный contentType для изображения: ".concat(i, ". Ожидается один из: ").concat(t.options.acceptContentTypes.join(", "), ".");
1100
- console.error("ImageManager. ".concat(s)), t.editor.canvas.fire("editor:error", {
1101
- message: s
1102
- });
1103
- return;
1104
- }
1105
- var {
1106
- canvas: c,
1107
- montageArea: u,
1108
- transformManager: l,
1109
- historyManager: d
1110
- } = t.editor;
1111
- d.suspendHistory();
1112
- try {
1113
- var h, g;
1114
- if (a instanceof File)
1115
- h = URL.createObjectURL(a);
1116
- else if (typeof a == "string") {
1117
- var v = yield fetch(a, {
1118
- mode: "cors"
1119
- }), f = yield v.blob({
1120
- type: i,
1121
- quality: 1
1122
- });
1123
- h = URL.createObjectURL(f);
1124
- } else
1125
- throw new Error("ImportImage. Неверный тип источника изображения. Ожидается URL или объект File.");
1126
- t._createdBlobUrls.push(h);
1127
- var y = C.getFormatFromContentType(i);
1128
- if (y === "svg") {
1129
- var m = yield Ee(h);
1130
- g = U.groupSVGElements(m.objects, m.options);
1131
- } else
1132
- g = yield de.fromURL(h, {
1133
- crossOrigin: "anonymous"
1134
- });
1135
- var {
1136
- width: b,
1137
- height: M
1138
- } = g;
1139
- if (M > N || b > D) {
1140
- var L = yield t.resizeImageToBoundaries(g._element.src, "max"), O = URL.createObjectURL(L);
1141
- t._createdBlobUrls.push(O), g = yield de.fromURL(O, {
1142
- crossOrigin: "anonymous"
1143
- });
1144
- }
1145
- if (g.set("id", "".concat(g.type, "-").concat(I())), g.set("format", y), r === "scale-montage")
1146
- t.editor.canvasManager.scaleMontageAreaToImage({
1147
- object: g,
1148
- withoutSave: !0
1149
- });
1150
- else {
1151
- var {
1152
- width: S,
1153
- height: A
1154
- } = u, P = t.calculateScaleFactor({
1155
- imageObject: g,
1156
- scaleType: r
1157
- });
1158
- r === "image-contain" && P < 1 ? l.fitObject({
1159
- object: g,
1160
- type: "contain",
1161
- withoutSave: !0
1162
- }) : r === "image-cover" && (b > S || M > A) && l.fitObject({
1163
- object: g,
1164
- type: "cover",
1165
- withoutSave: !0
1166
- });
1167
- }
1168
- c.add(g), c.centerObject(g), c.setActiveObject(g), c.renderAll(), d.resumeHistory(), n || d.saveState();
1169
- } catch (E) {
1170
- console.error("importImage. Ошибка импорта изображения: ", E), c.fire("editor:error", {
1171
- message: "Ошибка импорта изображения: ".concat(E.message)
1172
- }), d.resumeHistory();
1173
- }
1174
- }
1175
- })();
1176
- }
1177
- /**
1178
- * Функция для ресайза изображения до максимальных размеров,
1179
- * если оно их превышает. Сохраняет пропорции.
1180
- *
1181
- * @param {HTMLImageElement} imageEl - HTML элемент изображения
1182
- * @param {string} [size='max | min'] - максимальный или минимальный размер
1183
- * @returns {Promise<string>} - возвращает Promise с новым dataURL
1184
- */
1185
- resizeImageToBoundaries(e) {
1186
- var t = arguments, a = this;
1187
- return T(function* () {
1188
- var r = t.length > 1 && t[1] !== void 0 ? t[1] : "max", n = "Размер изображения больше максимального размера канваса, поэтому оно будет уменьшено до максимальных размеров: ".concat(D, "x").concat(N);
1189
- console.warn("importImage. ".concat(n)), a.editor.canvas.fire("editor:warning", {
1190
- message: n
1191
- });
1192
- var i = yield a.editor.workerManager.post("resizeImage", {
1193
- dataURL: e,
1194
- maxWidth: D,
1195
- maxHeight: N,
1196
- sizeType: r
1197
- });
1198
- return i;
1199
- })();
1200
- }
1201
- /**
1202
- * Экспорт изображения в файл – экспортируется содержимое монтажной области.
1203
- * Независимо от текущего зума, экспортируется монтажная область в исходном масштабе. Можно экспортировать как base64.
1204
- * @param {Object} options - опции
1205
- * @param {string} options.fileName - имя файла
1206
- * @param {string} options.contentType - тип контента
1207
- * @param {Boolean} options.exportAsBase64 - экспортировать как base64
1208
- * @param {Boolean} options.exportAsBlob - экспортировать как blob
1209
- * @returns {Promise<File> | String} - файл или base64
1210
- * @fires editor:canvas-exported
1211
- */
1212
- exportCanvasAsImageFile() {
1213
- var e = arguments, t = this;
1214
- return T(function* () {
1215
- var {
1216
- fileName: a = "image.png",
1217
- contentType: r = "image/png",
1218
- exportAsBase64: n = !1,
1219
- exportAsBlob: i = !1
1220
- } = e.length > 0 && e[0] !== void 0 ? e[0] : {}, {
1221
- canvas: s,
1222
- montageArea: c,
1223
- workerManager: u
1224
- } = t.editor, l = r === "application/pdf", d = l ? "image/jpg" : r, h = C.getFormatFromContentType(d);
1225
- c.setCoords();
1226
- var {
1227
- left: g,
1228
- top: v,
1229
- width: f,
1230
- height: y
1231
- } = c.getBoundingRect(), m = yield s.clone(["id", "format", "locked"]);
1232
- ["image/jpg", "image/jpeg"].includes(d) && (m.backgroundColor = "#ffffff");
1233
- var b = m.getObjects().find((w) => w.id === c.id);
1234
- b.visible = !1, m.viewportTransform = [1, 0, 0, 1, -g, -v], m.setDimensions({
1235
- width: f,
1236
- height: y
1237
- }, {
1238
- backstoreOnly: !0
1239
- }), m.renderAll();
1240
- var M = m.getObjects().filter((w) => w.format).every((w) => w.format === "svg");
1241
- if (h === "svg" && M) {
1242
- var L = m.toSVG();
1243
- m.dispose();
1244
- var O = C._exportSVGStringAsFile(L, {
1245
- exportAsBase64: n,
1246
- exportAsBlob: i,
1247
- fileName: a
1248
- }), S = {
1249
- image: O,
1250
- format: "svg",
1251
- contentType: "image/svg+xml",
1252
- fileName: a.replace(/\.[^/.]+$/, ".svg")
1253
- };
1254
- return s.fire("editor:canvas-exported", S), S;
1255
- }
1256
- var A = yield new Promise((w) => {
1257
- m.getElement().toBlob(w);
1258
- });
1259
- if (m.dispose(), i) {
1260
- var P = {
1261
- image: A,
1262
- format: h,
1263
- contentType: d,
1264
- fileName: a
1265
- };
1266
- return s.fire("editor:canvas-exported", P), P;
1267
- }
1268
- var E = yield createImageBitmap(A), re = yield u.post("toDataURL", {
1269
- format: h,
1270
- quality: 1,
1271
- bitmap: E
1272
- }, [E]);
1273
- if (l) {
1274
- var ne = 0.264583, W = f * ne, _ = y * ne, we = (yield t.editor.moduleLoader.loadModule("jspdf")).jsPDF, R = new we({
1275
- orientation: W > _ ? "landscape" : "portrait",
1276
- unit: "mm",
1277
- format: [W, _]
1278
- });
1279
- if (R.addImage(re, "JPG", 0, 0, W, _), n) {
1280
- var Ce = R.output("datauristring"), ie = {
1281
- image: Ce,
1282
- format: "pdf",
1283
- contentType: "application/pdf",
1284
- fileName: a
1285
- };
1286
- return s.fire("editor:canvas-exported", ie), ie;
1287
- }
1288
- var De = R.output("blob"), Ne = new File([De], a, {
1289
- type: "application/pdf"
1290
- }), oe = {
1291
- image: Ne,
1292
- format: "pdf",
1293
- contentType: "application/pdf",
1294
- fileName: a
1295
- };
1296
- return s.fire("editor:canvas-exported", oe), oe;
1297
- }
1298
- if (n) {
1299
- var se = {
1300
- image: re,
1301
- format: h,
1302
- contentType: d,
1303
- fileName: a
1304
- };
1305
- return s.fire("editor:canvas-exported", se), se;
1306
- }
1307
- var ce = h === "svg" && !M ? a.replace(/\.[^/.]+$/, ".png") : a, Le = new File([A], ce, {
1308
- type: d
1309
- }), ue = {
1310
- image: Le,
1311
- format: h,
1312
- contentType: d,
1313
- fileName: ce
1314
- };
1315
- return s.fire("editor:canvas-exported", ue), ue;
1316
- })();
1317
- }
1318
- /**
1319
- * Экспорт выбранного объекта в виде изображения или base64
1320
- * @param {Object} options - опции
1321
- * @param {fabric.Object} options.object - объект для экспорта
1322
- * @param {String} options.fileName - имя файла
1323
- * @param {String} options.contentType - тип контента
1324
- * @param {Boolean} options.exportAsBase64 - экспортировать как base64
1325
- * @param {Boolean} options.exportAsBlob - экспортировать как blob
1326
- * @returns {String} base64
1327
- * @fires editor:object-exported
1328
- */
1329
- exportObjectAsImageFile() {
1330
- var e = arguments, t = this;
1331
- return T(function* () {
1332
- var {
1333
- object: a,
1334
- fileName: r = "image.png",
1335
- contentType: n = "image/png",
1336
- exportAsBase64: i = !1,
1337
- exportAsBlob: s = !1
1338
- } = e.length > 0 && e[0] !== void 0 ? e[0] : {}, {
1339
- canvas: c,
1340
- workerManager: u
1341
- } = t.editor, l = a || c.getActiveObject();
1342
- if (!l)
1343
- return console.error("exportObjectAsDataURL. Не выбран объект"), c.fire("editor:error", {
1344
- message: "Не выбран объект для экспорта"
1345
- }), "";
1346
- var d = C.getFormatFromContentType(n);
1347
- if (d === "svg") {
1348
- var h = l.toSVG(), g = t._exportSVGStringAsFile(h, {
1349
- exportAsBase64: i,
1350
- exportAsBlob: s,
1351
- fileName: r
1352
- }), v = {
1353
- image: g,
1354
- format: d,
1355
- contentType: "image/svg+xml",
1356
- fileName: r.replace(/\.[^/.]+$/, ".svg")
1357
- };
1358
- return c.fire("editor:object-exported", v), v;
1359
- }
1360
- if (i) {
1361
- var f = yield createImageBitmap(l._element), y = yield u.post("toDataURL", {
1362
- format: d,
1363
- quality: 1,
1364
- bitmap: f
1365
- }, [f]), m = {
1366
- image: y,
1367
- format: d,
1368
- contentType: n,
1369
- fileName: r
1370
- };
1371
- return c.fire("editor:object-exported", m), m;
1372
- }
1373
- var b = l.toCanvasElement(), M = yield new Promise((A) => {
1374
- b.toBlob(A);
1375
- });
1376
- if (s) {
1377
- var L = {
1378
- image: M,
1379
- format: d,
1380
- contentType: n,
1381
- fileName: r
1382
- };
1383
- return c.fire("editor:object-exported", L), L;
1384
- }
1385
- var O = new File([M], r, {
1386
- type: n
1387
- }), S = {
1388
- image: O,
1389
- format: d,
1390
- contentType: n,
1391
- fileName: r
1392
- };
1393
- return c.fire("editor:object-exported", S), S;
1394
- })();
1395
- }
1396
- /**
1397
- * Удаляет все созданные blobURL
1398
- * @returns {void}
1399
- */
1400
- revokeBlobUrls() {
1401
- this._createdBlobUrls.forEach(URL.revokeObjectURL), this._createdBlobUrls = [];
1402
- }
1403
- /**
1404
- * Преобразует SVG-строку в Blob, файл, или base64
1405
- * @param {string} svgString - SVG-строка
1406
- * @param {Object} options - опции
1407
- * @param {Boolean} options.exportAsBase64 - экспортировать как base64
1408
- * @param {Boolean} options.exportAsBlob - экспортировать как blob
1409
- * @param {String} options.fileName - имя файла
1410
- * @returns {Blob|String|File} - Blob, base64 или файл
1411
- * @private
1412
- * @static
1413
- */
1414
- static _exportSVGStringAsFile(e) {
1415
- var {
1416
- exportAsBase64: t,
1417
- exportAsBlob: a,
1418
- fileName: r
1419
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};
1420
- return a ? new Blob([e], {
1421
- type: "image/svg+xml"
1422
- }) : t ? "data:image/svg+xml;base64,".concat(btoa(e)) : new File([e], r.replace(/\.[^/.]+$/, ".svg"), {
1423
- type: "image/svg+xml"
1424
- });
1425
- }
1426
- /**
1427
- * Извлекает чистый формат (subtype) из contentType,
1428
- * отбросив любую часть после «+» или «;»
1429
- * @param {string} contentType
1430
- * @returns {string} формат, например 'png', 'jpeg', 'svg'
1431
- * @static
1432
- */
1433
- static getFormatFromContentType() {
1434
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : "", t = e.match(/^[^/]+\/([^+;]+)/);
1435
- return t ? t[1] : "";
1436
- }
1437
- /**
1438
- * Проверяет, является ли contentType допустимым типом изображения.
1439
- * @param {string} contentType - тип контента
1440
- * @returns {boolean} true, если contentType допустим, иначе false
1441
- */
1442
- isAllowedContentType() {
1443
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : "", {
1444
- acceptContentTypes: t
1445
- } = this.editor.options;
1446
- return t.includes(e);
1447
- }
1448
- /**
1449
- * Проверяет, является ли формат допустимым.
1450
- * @param {string} format - формат изображения, например 'png', 'jpeg', 'svg'
1451
- * @returns {boolean} true, если формат допустим, иначе false
1452
- */
1453
- isAllowedFormat() {
1454
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : "", {
1455
- acceptContentTypes: t
1456
- } = this.editor.options, a = "image/".concat(e);
1457
- return t.includes(a);
1458
- }
1459
- /**
1460
- * Получает contentType изображения из источника
1461
- * @param {File|string} source - URL изображения или объект File
1462
- * @returns {Promise<string>|string} - MIME-тип изображения
1463
- * @public
1464
- */
1465
- getContentType(e) {
1466
- var t = this;
1467
- return T(function* () {
1468
- return typeof e == "string" ? t.getContentTypeFromUrl(e) : e.type || "application/octet-stream";
1469
- })();
1470
- }
1471
- /**
1472
- * Получает contentType изображения через HTTP HEAD запрос или анализ URL
1473
- * @param {string} src - URL изображения
1474
- * @returns {Promise<string>} - MIME-тип изображения
1475
- * @public
1476
- */
1477
- getContentTypeFromUrl(e) {
1478
- var t = this;
1479
- return T(function* () {
1480
- if (e.startsWith("data:")) {
1481
- var a = e.match(/^data:([^;]+)/);
1482
- return a ? a[1] : "application/octet-stream";
1483
- }
1484
- try {
1485
- var r = yield fetch(e, {
1486
- method: "HEAD"
1487
- }), n = r.headers.get("content-type");
1488
- if (n && n.startsWith("image/"))
1489
- return n.split(";")[0];
1490
- } catch (i) {
1491
- console.warn("HEAD запрос неудачен, определяем тип по расширению:", i);
1492
- }
1493
- return t.getContentTypeFromExtension(e);
1494
- })();
1495
- }
1496
- /**
1497
- * Определяет contentType по расширению файла в URL
1498
- * @param {string} url - URL файла
1499
- * @returns {string} - MIME-тип
1500
- * @public
1501
- */
1502
- getContentTypeFromExtension(e) {
1503
- try {
1504
- var t, {
1505
- acceptContentTypes: a
1506
- } = this.editor.options, r = new URL(e), n = (t = r.pathname.split(".").pop()) === null || t === void 0 ? void 0 : t.toLowerCase(), i = {};
1507
- return a.forEach((s) => {
1508
- var c = C.getFormatFromContentType(s);
1509
- c && (i[c] = s);
1510
- }), i[n] || "application/octet-stream";
1511
- } catch (s) {
1512
- return console.warn("Не удалось определить расширение из URL:", e, s), "application/octet-stream";
1513
- }
1514
- }
1515
- /**
1516
- * Рассчитывает коэффициент масштабирования изображения.
1517
- * @param {object} imageObject - объект изображения
1518
- * @param {string} scaleType - тип масштабирования ('contain' или 'cover')
1519
- * @returns {number} коэффициент масштабирования
1520
- */
1521
- calculateScaleFactor(e) {
1522
- var {
1523
- imageObject: t,
1524
- scaleType: a = "contain"
1525
- } = e, {
1526
- montageArea: r
1527
- } = this.editor;
1528
- if (!r || !t) return 1;
1529
- var n = r.width, i = r.height, {
1530
- width: s,
1531
- height: c
1532
- } = t;
1533
- return a === "contain" || a === "image-contain" ? Math.min(n / s, i / c) : a === "cover" || a === "image-cover" ? Math.max(n / s, i / c) : 1;
1534
- }
1535
- }
1536
- class Mt {
1537
- /**
1538
- * @param {object} options
1539
- * @param {ImageEditor} options.editor – экземпляр редактора
1540
- */
1541
- constructor(e) {
1542
- var {
1543
- editor: t
1544
- } = e;
1545
- this.editor = t;
1546
- }
1547
- /**
1548
- * Устанавливаем внутреннюю ширину канваса (для экспорта)
1549
- * @param {String} width
1550
- * @param {Object} options
1551
- * @param {Boolean} options.preserveProportional - Сохранить пропорции
1552
- * @param {Boolean} options.withoutSave - Не сохранять состояние
1553
- * @fires editor:resolution-width-changed
1554
- */
1555
- setResolutionWidth(e) {
1556
- var {
1557
- preserveProportional: t,
1558
- withoutSave: a,
1559
- adaptCanvasToContainer: r
1560
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};
1561
- if (e) {
1562
- var {
1563
- canvas: n,
1564
- montageArea: i,
1565
- options: {
1566
- canvasBackstoreWidth: s
1567
- }
1568
- } = this.editor, {
1569
- width: c,
1570
- height: u
1571
- } = i, l = Number(Math.max(Math.min(e, D), x));
1572
- if (!s || s === "auto" || r ? this.adaptCanvasToContainer() : s ? this.setCanvasBackstoreWidth(s) : this.setCanvasBackstoreWidth(l), i.set({
1573
- width: l
1574
- }), n.clipPath.set({
1575
- width: l
1576
- }), t) {
1577
- var d = l / c, h = u * d;
1578
- this.setResolutionHeight(h);
1579
- return;
1580
- }
1581
- var {
1582
- left: g,
1583
- top: v
1584
- } = this.getObjectDefaultCoords(i), f = n.getZoom();
1585
- n.setViewportTransform([f, 0, 0, f, g, v]), this.centerMontageArea(), a || this.editor.historyManager.saveState(), n == null || n.fire("editor:resolution-width-changed", {
1586
- width: e
1587
- });
1588
- }
1589
- }
1590
- /**
1591
- * Устанавливаем внутреннюю высоту канваса (для экспорта)
1592
- * @param {String} height
1593
- * @param {Object} options
1594
- * @param {Boolean} options.preserveProportional - Сохранить пропорции
1595
- * @param {Boolean} options.withoutSave - Не сохранять состояние
1596
- * @fires editor:resolution-height-changed
1597
- */
1598
- setResolutionHeight(e) {
1599
- var {
1600
- preserveProportional: t,
1601
- withoutSave: a,
1602
- adaptCanvasToContainer: r
1603
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};
1604
- if (e) {
1605
- var {
1606
- canvas: n,
1607
- montageArea: i,
1608
- options: {
1609
- canvasBackstoreHeight: s
1610
- }
1611
- } = this.editor, {
1612
- width: c,
1613
- height: u
1614
- } = i, l = Number(Math.max(Math.min(e, N), B));
1615
- if (!s || s === "auto" || r ? this.adaptCanvasToContainer() : s ? this.setCanvasBackstoreHeight(s) : this.setCanvasBackstoreHeight(l), i.set({
1616
- height: l
1617
- }), n.clipPath.set({
1618
- height: l
1619
- }), t) {
1620
- var d = l / u, h = c * d;
1621
- this.setResolutionWidth(h);
1622
- return;
1623
- }
1624
- var {
1625
- left: g,
1626
- top: v
1627
- } = this.getObjectDefaultCoords(i), f = n.getZoom();
1628
- n.setViewportTransform([f, 0, 0, f, g, v]), this.centerMontageArea(), a || this.editor.historyManager.saveState(), n == null || n.fire("editor:resolution-height-changed", {
1629
- height: e
1630
- });
1631
- }
1632
- }
1633
- /**
1634
- * Центрирует монтажную область и ClipPath точно по центру канваса
1635
- * и устанавливает правильный viewportTransform.
1636
- */
1637
- centerMontageArea() {
1638
- var {
1639
- canvas: e,
1640
- montageArea: t
1641
- } = this.editor, a = e.getWidth(), r = e.getHeight(), n = e.getZoom(), i = new ke(a / 2, r / 2);
1642
- t.set({
1643
- left: a / 2,
1644
- top: r / 2
1645
- }), e.clipPath.set({
1646
- left: a / 2,
1647
- top: r / 2
1648
- }), e.renderAll();
1649
- var s = e.viewportTransform;
1650
- s[4] = a / 2 - i.x * n, s[5] = r / 2 - i.y * n, e.setViewportTransform(s), e.renderAll();
1651
- }
1652
- /**
1653
- * Метод для получения координат объекта с учетом текущего зума
1654
- * @param {fabric.Object} object - объект, координаты которого нужно получить
1655
- * @returns {Object} координаты объекта
1656
- */
1657
- getObjectDefaultCoords(e) {
1658
- var {
1659
- canvas: t
1660
- } = this.editor, a = e || t.getActiveObject();
1661
- if (!a)
1662
- return console.error("getObjectDefaultCoords. Не выбран объект"), this.editor.canvas.fire("editor:error", {
1663
- message: "Не выбран объект для получения координат"
1664
- }), {};
1665
- var {
1666
- width: r,
1667
- height: n
1668
- } = a, i = t.getZoom(), s = (r - r * i) / 2, c = (n - n * i) / 2;
1669
- return {
1670
- left: s,
1671
- top: c
1672
- };
1673
- }
1674
- setCanvasBackstoreWidth(e) {
1675
- if (!(!e || typeof e != "number")) {
1676
- var t = Math.max(Math.min(e, D), x);
1677
- this.editor.canvas.setDimensions({
1678
- width: t
1679
- }, {
1680
- backstoreOnly: !0
1681
- });
1682
- }
1683
- }
1684
- setCanvasBackstoreHeight(e) {
1685
- if (!(!e || typeof e != "number")) {
1686
- var t = Math.max(Math.min(e, N), B);
1687
- this.editor.canvas.setDimensions({
1688
- height: t
1689
- }, {
1690
- backstoreOnly: !0
1691
- });
1692
- }
1693
- }
1694
- adaptCanvasToContainer() {
1695
- var {
1696
- canvas: e
1697
- } = this.editor, t = e.editorContainer, a = t.clientWidth, r = t.clientHeight, n = Math.max(Math.min(a, D), x), i = Math.max(Math.min(r, N), B);
1698
- console.log("adaptCanvasToContainer newWidth", n), console.log("adaptCanvasToContainer newHeight", i), e.setDimensions({
1699
- width: n,
1700
- height: i
1701
- }, {
1702
- backstoreOnly: !0
1703
- });
1704
- }
1705
- /**
1706
- * Устанавливаем CSS ширину канваса для отображения
1707
- * @param {string|number} width
1708
- * @fires editor:display-canvas-width-changed
1709
- */
1710
- setCanvasCSSWidth(e) {
1711
- this.setDisplayDimension({
1712
- element: "canvas",
1713
- dimension: "width",
1714
- value: e
1715
- });
1716
- }
1717
- /**
1718
- * Устанавливаем CSS высоту канваса для отображения
1719
- * @param {string|number} height
1720
- * @fires editor:display-canvas-height-changed
1721
- */
1722
- setCanvasCSSHeight(e) {
1723
- this.setDisplayDimension({
1724
- element: "canvas",
1725
- dimension: "height",
1726
- value: e
1727
- });
1728
- }
1729
- /**
1730
- * Устанавливаем CSS ширину обертки канваса для отображения
1731
- * @param {string|number} width
1732
- * @fires editor:display-wrapper-width-changed
1733
- */
1734
- setCanvasWrapperWidth(e) {
1735
- this.setDisplayDimension({
1736
- element: "wrapper",
1737
- dimension: "width",
1738
- value: e
1739
- });
1740
- }
1741
- /**
1742
- * Устанавливаем CSS высоту обертки канваса для отображения
1743
- * @param {string|number} height
1744
- * @fires editor:display-wrapper-height-changed
1745
- */
1746
- setCanvasWrapperHeight(e) {
1747
- this.setDisplayDimension({
1748
- element: "wrapper",
1749
- dimension: "height",
1750
- value: e
1751
- });
1752
- }
1753
- /**
1754
- * Устанавливаем CSS ширину контейнера редактора для отображения
1755
- * @param {string|number} width
1756
- * @fires editor:display-container-width-changed
1757
- */
1758
- setEditorContainerWidth(e) {
1759
- this.setDisplayDimension({
1760
- element: "container",
1761
- dimension: "width",
1762
- value: e
1763
- });
1764
- }
1765
- /**
1766
- * Устанавливаем CSS высоту контейнера редактора для отображения
1767
- * @param {string|number} height
1768
- * @fires editor:display-container-height-changed
1769
- */
1770
- setEditorContainerHeight(e) {
1771
- this.setDisplayDimension({
1772
- element: "container",
1773
- dimension: "height",
1774
- value: e
1775
- });
1776
- }
1777
- /**
1778
- * Устанавливаем CSS ширину или высоту канваса для отображения
1779
- * @param {Object} options
1780
- * @param {String} [options.element] - элемент, для которого устанавливаем размеры:
1781
- * canvas (upper & lower), wrapper, container
1782
- * @param {('width'|'height')} [options.dimension]
1783
- * @param {string|number} [options.value]
1784
- * @fires editor:display-{element}-{dimension}-changed
1785
- */
1786
- setDisplayDimension() {
1787
- var {
1788
- element: e,
1789
- dimension: t,
1790
- value: a
1791
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {};
1792
- if (a) {
1793
- var {
1794
- canvas: r,
1795
- options: {
1796
- editorContainer: n
1797
- }
1798
- } = this.editor, i = [];
1799
- switch (e) {
1800
- case "canvas":
1801
- i.push(r.lowerCanvasEl, r.upperCanvasEl);
1802
- break;
1803
- case "wrapper":
1804
- i.push(r.wrapperEl);
1805
- break;
1806
- case "container":
1807
- i.push(n);
1808
- break;
1809
- default:
1810
- i.push(r.lowerCanvasEl, r.upperCanvasEl);
1811
- }
1812
- var s = t === "width" ? "width" : "height";
1813
- if (typeof a == "string") {
1814
- i.forEach((l) => {
1815
- l.style[s] = a;
1816
- });
1817
- return;
1818
- }
1819
- var c = parseFloat(a);
1820
- if (!isNaN(c)) {
1821
- var u = "".concat(c, "px");
1822
- i.forEach((l) => {
1823
- l.style[s] = u;
1824
- }), r.fire("editor:display-".concat(e, "-").concat(s, "-changed"), {
1825
- element: e,
1826
- value: a
1827
- });
1828
- }
1829
- }
1830
- }
1831
- /**
1832
- * Если изображение вписывается в допустимые значения, то масштабируем под него канвас
1833
- * @param {Object} options
1834
- * @param {fabric.Object} [options.object] - Объект с изображением, которое нужно масштабировать
1835
- * @param {Boolean} [options.withoutSave] - Не сохранять состояние
1836
- * @param {Boolean} [options.preserveAspectRatio] - Сохранять изначальные пропорции монтажной области
1837
- * @fires editor:canvas-scaled
1838
- */
1839
- scaleMontageAreaToImage() {
1840
- var {
1841
- object: e,
1842
- preserveAspectRatio: t,
1843
- withoutSave: a
1844
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
1845
- canvas: r,
1846
- montageArea: n,
1847
- transformManager: i,
1848
- options: {
1849
- montageAreaWidth: s,
1850
- montageAreaHeight: c
1851
- }
1852
- } = this.editor, u = e || r.getActiveObject();
1853
- if (!(!u || u.type !== "image" && u.format !== "svg")) {
1854
- var {
1855
- width: l,
1856
- height: d
1857
- } = u;
1858
- if (l < x || d < B) {
1859
- var h = "Размер изображения меньше минимального размера канваса, поэтому оно будет растянуто до минимальных размеров: ".concat(x, "x").concat(B);
1860
- console.warn("scaleMontageAreaToImage. ".concat(h)), r.fire("editor:warning", {
1861
- message: h
1862
- });
1863
- }
1864
- var g = Math.min(l, D), v = Math.min(d, N);
1865
- if (t) {
1866
- var {
1867
- width: f,
1868
- height: y
1869
- } = n, m = l / f, b = d / y, M = Math.max(m, b);
1870
- g = f * M, v = y * M;
1871
- }
1872
- this.setResolutionWidth(g, {
1873
- withoutSave: !0
1874
- }), this.setResolutionHeight(v, {
1875
- withoutSave: !0
1876
- }), (l > s || d > c) && i.calculateAndApplyDefaultZoom(s, c), i.resetObject(u, {
1877
- withoutSave: !0
1878
- }), r.centerObject(u), r.renderAll(), a || this.editor.historyManager.saveState(), r.fire("editor:canvas-scaled", {
1879
- width: g,
1880
- height: v
1881
- });
1882
- }
1883
- }
1884
- /**
1885
- * Очистка холста
1886
- * @fires editor:cleared
1887
- */
1888
- clearCanvas() {
1889
- var {
1890
- canvas: e,
1891
- montageArea: t,
1892
- historyManager: a
1893
- } = this.editor;
1894
- a.suspendHistory(), e.clear(), e.add(t), e.renderAll(), a.resumeHistory(), a.saveState(), e == null || e.fire("editor:cleared");
1895
- }
1896
- /**
1897
- * Установка зума и масштаба для канваса и сброс трансформации всех объектов
1898
- * @param {Object} options
1899
- * @param {Boolean} options.withoutSave - Не сохранять состояние
1900
- * @fires editor:default-scale-set
1901
- */
1902
- setDefaultScale() {
1903
- var {
1904
- withoutSave: e
1905
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
1906
- canvas: t,
1907
- transformManager: a,
1908
- historyManager: r,
1909
- options: {
1910
- montageAreaWidth: n,
1911
- montageAreaHeight: i
1912
- }
1913
- } = this.editor;
1914
- a.resetZoom(), this.setResolutionWidth(n, {
1915
- withoutSave: !0
1916
- }), this.setResolutionHeight(i, {
1917
- withoutSave: !0
1918
- }), t.renderAll(), a.resetObjects(), e || r.saveState(), t.fire("editor:default-scale-set");
1919
- }
1920
- /**
1921
- * Получение всех объектов внутри монтажной области редактора
1922
- * @returns {Array} массив объектов
1923
- */
1924
- getObjects() {
1925
- var {
1926
- canvas: e,
1927
- montageArea: t,
1928
- interactionBlocker: {
1929
- overlayMask: a
1930
- }
1931
- } = this.editor, r = e.getObjects();
1932
- return r.filter((n) => n.id !== t.id && n.id !== a.id);
1933
- }
1934
- }
1935
- class jt {
1936
- /**
1937
- * @param {object} options
1938
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
1939
- */
1940
- constructor(e) {
1941
- var {
1942
- editor: t
1943
- } = e;
1944
- this.editor = t, this.options = t.options;
1945
- }
1946
- /**
1947
- * Метод рассчитывает дефолтный, максимальный, и минимальный зум таким образом,
1948
- * чтобы монтажная область визуально занимала переданные размеры.
1949
- * Если размеры не переданы, то используются дефолтные размеры монтажной области переданные в options.
1950
- * @param {number} [targetWidth] — желаемая видимая ширина (px)
1951
- * @param {number} [targetHeight] — желаемая видимая высота (px)
1952
- */
1953
- calculateAndApplyDefaultZoom() {
1954
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : this.options.montageAreaWidth, t = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : this.options.montageAreaHeight, {
1955
- width: a,
1956
- height: r
1957
- } = this.editor.montageArea, n = e / a, i = t / r, s = Math.min(n, i), {
1958
- minZoom: c,
1959
- maxZoom: u,
1960
- maxZoomFactor: l
1961
- } = this.options;
1962
- this.minZoom = Math.min(s / l, c), this.maxZoom = Math.max(s * l, u), this.defaultZoom = s, this.setZoom(s);
1963
- }
1964
- /**
1965
- * Увеличение/уменьшение масштаба
1966
- * @param {Number} scale - Шаг зума
1967
- * @param {Object} options - Координаты зума (по умолчанию центр канваса)
1968
- * @param {Number} options.pointX - Координата X точки зума
1969
- * @param {Number} options.pointY - Координата Y точки зума
1970
- * @fires editor:zoom-changed
1971
- * Если передавать координаты курсора, то нужно быть аккуратнее, так как юзер может выйти за пределы рабочей области
1972
- */
1973
- zoom() {
1974
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : mt, t = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};
1975
- if (e) {
1976
- var {
1977
- canvas: a,
1978
- minZoom: r,
1979
- maxZoom: n
1980
- } = this.editor, i = a.getZoom(), {
1981
- x: s = t.pointX,
1982
- y: c = t.pointY
1983
- } = a.getCenterPoint(), u = Number((i + Number(e)).toFixed(2));
1984
- u > n && (u = n), u < r && (u = r), a.zoomToPoint({
1985
- x: Number(s),
1986
- y: Number(c)
1987
- }, u), a.fire("editor:zoom-changed", {
1988
- currentZoom: a.getZoom(),
1989
- zoom: u,
1990
- pointX: s,
1991
- pointY: c
1992
- });
1993
- }
1994
- }
1995
- /**
1996
- * Установка зума
1997
- * @param {Number} zoom - Зум
1998
- * @fires editor:zoom-changed
1999
- */
2000
- setZoom() {
2001
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : this.defaultZoom, {
2002
- canvas: t,
2003
- minZoom: a,
2004
- maxZoom: r
2005
- } = this.editor, {
2006
- x: n,
2007
- y: i
2008
- } = t.getCenterPoint(), s = e;
2009
- e > r && (s = r), e < a && (s = a), t.zoomToPoint({
2010
- x: Number(n),
2011
- y: Number(i)
2012
- }, s), t.fire("editor:zoom-changed", {
2013
- currentZoom: t.getZoom(),
2014
- zoom: s,
2015
- pointX: n,
2016
- pointY: i
2017
- });
2018
- }
2019
- /**
2020
- * Сброс зума
2021
- * @fires editor:zoom-changed
2022
- */
2023
- resetZoom() {
2024
- var {
2025
- canvas: e,
2026
- defaultZoom: t
2027
- } = this.editor, {
2028
- x: a,
2029
- y: r
2030
- } = e.getCenterPoint();
2031
- e.zoomToPoint({
2032
- x: Number(a),
2033
- y: Number(r)
2034
- }, t), this.editor.canvas.fire("editor:zoom-changed", {
2035
- currentZoom: e.getZoom()
2036
- });
2037
- }
2038
- /**
2039
- * Поворот объекта на заданный угол
2040
- * @param {number} angle
2041
- * @param {Object} options
2042
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2043
- * @fires editor:object-rotated
2044
- */
2045
- rotate() {
2046
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : ft, {
2047
- withoutSave: t
2048
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2049
- canvas: a,
2050
- historyManager: r
2051
- } = this.editor, n = a.getActiveObject();
2052
- if (n) {
2053
- var i = n.angle + e;
2054
- n.rotate(i), n.setCoords(), a.renderAll(), t || r.saveState(), a.fire("editor:object-rotated", {
2055
- angle: i
2056
- });
2057
- }
2058
- }
2059
- /**
2060
- * Отразить по горизонтали
2061
- * @param {Object} options
2062
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2063
- * @fires editor:object-flipped-x
2064
- */
2065
- flipX() {
2066
- var {
2067
- withoutSave: e
2068
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2069
- canvas: t,
2070
- historyManager: a
2071
- } = this.editor, r = t.getActiveObject();
2072
- r && (r.flipX = !r.flipX, t.renderAll(), e || a.saveState(), t.fire("editor:object-flipped-x"));
2073
- }
2074
- /**
2075
- * Отразить по вертикали
2076
- * @param {Object} options
2077
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2078
- * @fires editor:object-flipped-y
2079
- */
2080
- flipY() {
2081
- var {
2082
- withoutSave: e
2083
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2084
- canvas: t,
2085
- historyManager: a
2086
- } = this.editor, r = t.getActiveObject();
2087
- r && (r.flipY = !r.flipY, t.renderAll(), e || a.saveState(), t.fire("editor:object-flipped-y"));
2088
- }
2089
- /**
2090
- * Установка прозрачности объекта
2091
- * @param {Number} opacity - Прозрачность от 0 до 1
2092
- * @fires editor:object-opacity-changed
2093
- */
2094
- setActiveObjectOpacity() {
2095
- var {
2096
- object: e,
2097
- opacity: t = 1,
2098
- withoutSave: a
2099
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2100
- canvas: r,
2101
- historyManager: n
2102
- } = this.editor, i = e || r.getActiveObject();
2103
- i && (i.type === "activeselection" ? i.getObjects().forEach((s) => {
2104
- s.set("opacity", t);
2105
- }) : i.set("opacity", t), r.renderAll(), a || n.saveState(), r.fire("editor:object-opacity-changed", t));
2106
- }
2107
- /**
2108
- * Масштабирование изображения
2109
- * @param {Object} options
2110
- * @param {fabric.Object} [options.object] - Объект с изображением, которое нужно масштабировать
2111
- * @param {String} [options.type] - Тип масштабирования
2112
- * 'contain' - скейлит картинку, чтобы она вмещалась
2113
- * 'cover' - скейлит картинку, чтобы она вписалась в размер канвас
2114
- * @param {Boolean} [options.withoutSave] - Не сохранять состояние
2115
- * @param {Boolean} [options.fitAsOneObject] - Масштабировать все объекты в активной группе как один объект
2116
- * @fires editor:image-fitted
2117
- */
2118
- fitObject() {
2119
- var {
2120
- object: e,
2121
- type: t = this.options.scaleType,
2122
- withoutSave: a,
2123
- fitAsOneObject: r
2124
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2125
- canvas: n,
2126
- imageManager: i,
2127
- historyManager: s
2128
- } = this.editor, c = e || n.getActiveObject();
2129
- if (c) {
2130
- if (["activeselection"].includes(c.type) && !r) {
2131
- var u = c.getObjects();
2132
- n.discardActiveObject(), u.forEach((h) => {
2133
- var g = i.calculateScaleFactor({
2134
- imageObject: h,
2135
- scaleType: t
2136
- });
2137
- h.scale(g), n.centerObject(h);
2138
- });
2139
- var l = new z(u, {
2140
- canvas: n
2141
- });
2142
- n.setActiveObject(l);
2143
- } else {
2144
- var d = i.calculateScaleFactor({
2145
- imageObject: c,
2146
- scaleType: t
2147
- });
2148
- c.scale(d), n.centerObject(c);
2149
- }
2150
- n.renderAll(), a || s.saveState(), n.fire("editor:image-fitted", {
2151
- type: t
2152
- });
2153
- }
2154
- }
2155
- /**
2156
- * Установка дефолтного масштаба для всех объектов внутри монтажной области редактора
2157
- */
2158
- resetObjects() {
2159
- this.editor.canvasManager.getObjects().forEach((e) => {
2160
- this.resetObject(e);
2161
- });
2162
- }
2163
- /**
2164
- * Сброс масштаба объекта до дефолтного
2165
- * @param {fabric.Object} object
2166
- * @param {Boolean} [fitOnlyBigImage] - растягивать только большие изображения
2167
- * @returns
2168
- * @fires editor:object-reset
2169
- */
2170
- resetObject(e) {
2171
- var {
2172
- alwaysFitObject: t = !1,
2173
- withoutSave: a = !1
2174
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2175
- canvas: r,
2176
- montageArea: n,
2177
- imageManager: i,
2178
- historyManager: s,
2179
- options: {
2180
- scaleType: c
2181
- }
2182
- } = this.editor, u = e || r.getActiveObject();
2183
- if (!(!u || u.locked)) {
2184
- if (s.suspendHistory(), u.type !== "image" && u.format !== "svg" && (u.set({
2185
- scaleX: 1,
2186
- scaleY: 1,
2187
- flipX: !1,
2188
- flipY: !1,
2189
- angle: 0
2190
- }), r.centerObject(u), r.renderAll()), t)
2191
- this.fitObject({
2192
- object: u,
2193
- withoutSave: !0
2194
- });
2195
- else {
2196
- var {
2197
- width: l,
2198
- height: d
2199
- } = n, {
2200
- width: h,
2201
- height: g
2202
- } = u, v = i.calculateScaleFactor({
2203
- imageObject: u,
2204
- scaleType: c
2205
- });
2206
- c === "contain" && v < 1 || c === "cover" && (h > l || g > d) ? this.fitObject({
2207
- object: u,
2208
- withoutSave: !0
2209
- }) : u.set({
2210
- scaleX: 1,
2211
- scaleY: 1
2212
- });
2213
- }
2214
- u.set({
2215
- flipX: !1,
2216
- flipY: !1,
2217
- angle: 0
2218
- }), r.centerObject(u), r.renderAll(), s.resumeHistory(), a || s.saveState(), r.fire("editor:object-reset");
2219
- }
2220
- }
2221
- }
2222
- class yt {
2223
- /**
2224
- * @param {object} options
2225
- * @param {ImageEditor} options.editor – экземпляр редактора
2226
- */
2227
- constructor(e) {
2228
- var {
2229
- editor: t
2230
- } = e;
2231
- this.editor = t, this.isBlocked = !1, this.overlayMask = null, this._createOverlay();
2232
- }
2233
- /**
2234
- * Создаёт overlay для блокировки монтажной области
2235
- * @private
2236
- * @returns {void}
2237
- */
2238
- _createOverlay() {
2239
- var {
2240
- historyManager: e,
2241
- options: {
2242
- overlayMaskColor: t = "rgba(0,0,0,0.5)"
2243
- }
2244
- } = this.editor;
2245
- e.suspendHistory(), this.overlayMask = this.editor.shapeManager.addRectangle({
2246
- fill: t,
2247
- selectable: !1,
2248
- evented: !0,
2249
- hoverCursor: "not-allowed",
2250
- hasBorders: !1,
2251
- hasControls: !1,
2252
- visible: !1,
2253
- id: "overlay-mask"
2254
- }, {
2255
- withoutSelection: !0
2256
- }), e.resumeHistory();
2257
- }
2258
- /**
2259
- * Обновляет размеры и позицию overlay, выносит его на передний план
2260
- * @returns {void}
2261
- */
2262
- refresh() {
2263
- var {
2264
- canvas: e,
2265
- montageArea: t,
2266
- historyManager: a
2267
- } = this.editor;
2268
- if (!(!t || !this.overlayMask)) {
2269
- a.suspendHistory(), t.setCoords();
2270
- var {
2271
- left: r,
2272
- top: n,
2273
- width: i,
2274
- height: s
2275
- } = t.getBoundingRect();
2276
- this.overlayMask.set({
2277
- left: r,
2278
- top: n,
2279
- width: i,
2280
- height: s
2281
- }), e.discardActiveObject(), this.editor.layerManager.bringToFront(this.overlayMask, {
2282
- withoutSave: !0
2283
- }), a.resumeHistory();
2284
- }
2285
- }
2286
- /**
2287
- * Выключает редактор:
2288
- * - убирает все селекты, события мыши, скейл/драг–н–дроп
2289
- * - делает все объекты не‑evented и не‑selectable
2290
- * - делает видимым overlayMask поверх всех объектов в монтажной области
2291
- * @returns {void}
2292
- */
2293
- block() {
2294
- if (!this.isBlocked) {
2295
- var {
2296
- canvas: e,
2297
- canvasManager: t,
2298
- historyManager: a
2299
- } = this.editor;
2300
- a.suspendHistory(), this.isBlocked = !0, e.discardActiveObject(), e.selection = !1, e.skipTargetFind = !0, t.getObjects().forEach((r) => {
2301
- r.evented = !1, r.selectable = !1;
2302
- }), e.upperCanvasEl.style.pointerEvents = "none", e.lowerCanvasEl.style.pointerEvents = "none", this.overlayMask.visible = !0, this.refresh(), e.fire("editor:disabled"), a.resumeHistory();
2303
- }
2304
- }
2305
- /**
2306
- * Включает редактор
2307
- * @returns {void}
2308
- */
2309
- unblock() {
2310
- if (this.isBlocked) {
2311
- var {
2312
- canvas: e,
2313
- canvasManager: t,
2314
- historyManager: a
2315
- } = this.editor;
2316
- a.suspendHistory(), this.isBlocked = !1, e.selection = !0, e.skipTargetFind = !1, t.getObjects().forEach((r) => {
2317
- r.evented = !0, r.selectable = !0;
2318
- }), e.upperCanvasEl.style.pointerEvents = "", e.lowerCanvasEl.style.pointerEvents = "", this.overlayMask.visible = !1, e.requestRenderAll(), e.fire("editor:enabled"), a.resumeHistory();
2319
- }
2320
- }
2321
- }
2322
- class H {
2323
- /**
2324
- * @param {object} options
2325
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2326
- */
2327
- constructor(e) {
2328
- var {
2329
- editor: t
2330
- } = e;
2331
- this.editor = t;
2332
- }
2333
- /**
2334
- * Поднять объект навверх по оси Z
2335
- * @param {fabric.Object} object
2336
- * @param {Object} options
2337
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2338
- * @fires editor:object-bring-to-front
2339
- */
2340
- bringToFront(e) {
2341
- var {
2342
- withoutSave: t
2343
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2344
- canvas: a,
2345
- historyManager: r
2346
- } = this.editor;
2347
- r.suspendHistory();
2348
- var n = e || a.getActiveObject();
2349
- n && (n.type === "activeselection" ? n.getObjects().forEach((i) => {
2350
- a.bringObjectToFront(i);
2351
- }) : a.bringObjectToFront(n), a.renderAll(), r.resumeHistory(), t || r.saveState(), a.fire("editor:object-bring-to-front"));
2352
- }
2353
- /**
2354
- * Поднять объект на один уровень вверх по оси Z
2355
- * @param {fabric.Object} object
2356
- * @param {Object} options
2357
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2358
- * @fires editor:object-bring-forward
2359
- */
2360
- bringForward(e) {
2361
- var {
2362
- withoutSave: t
2363
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2364
- canvas: a,
2365
- historyManager: r
2366
- } = this.editor;
2367
- r.suspendHistory();
2368
- var n = e || a.getActiveObject();
2369
- n && (n.type === "activeselection" ? H._moveSelectionForward(a, n) : a.bringObjectForward(n), a.renderAll(), r.resumeHistory(), t || r.saveState(), a.fire("editor:object-bring-forward"));
2370
- }
2371
- /**
2372
- * Отправить объект на задний план по оси Z
2373
- * @param {fabric.Object} object
2374
- * @param {Object} options
2375
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2376
- * @fires editor:object-send-to-back
2377
- */
2378
- sendToBack(e) {
2379
- var {
2380
- withoutSave: t
2381
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2382
- canvas: a,
2383
- montageArea: r,
2384
- historyManager: n,
2385
- interactionBlocker: {
2386
- overlayMask: i
2387
- }
2388
- } = this.editor;
2389
- n.suspendHistory();
2390
- var s = e || a.getActiveObject();
2391
- if (s) {
2392
- if (s.type === "activeselection")
2393
- for (var c = s.getObjects(), u = c.length - 1; u >= 0; u -= 1)
2394
- a.sendObjectToBack(c[u]);
2395
- else
2396
- a.sendObjectToBack(s);
2397
- a.sendObjectToBack(r), a.sendObjectToBack(i), a.renderAll(), n.resumeHistory(), t || n.saveState(), a.fire("editor:object-send-to-back");
2398
- }
2399
- }
2400
- /**
2401
- * Отправить объект на один уровень ниже по оси Z
2402
- * @param {fabric.Object} object
2403
- * @param {Object} options
2404
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2405
- */
2406
- sendBackwards(e) {
2407
- var {
2408
- withoutSave: t
2409
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2410
- canvas: a,
2411
- montageArea: r,
2412
- historyManager: n,
2413
- interactionBlocker: {
2414
- overlayMask: i
2415
- }
2416
- } = this.editor;
2417
- n.suspendHistory();
2418
- var s = e || a.getActiveObject();
2419
- s && (s.type === "activeselection" ? H._moveSelectionBackwards(a, s) : a.sendObjectBackwards(s), a.sendObjectToBack(r), a.sendObjectToBack(i), a.renderAll(), n.resumeHistory(), t || n.saveState(), a.fire("editor:object-send-backwards"));
2420
- }
2421
- /**
2422
- * Сдвигает выделенные объекты на один уровень вверх относительно ближайшего верхнего объекта
2423
- * @param {fabric.Canvas} canvas - экземпляр холста
2424
- * @param {fabric.ActiveSelection} activeSelection - активное выделение
2425
- * @returns {void}
2426
- * @private
2427
- */
2428
- static _moveSelectionForward(e, t) {
2429
- for (var a = e.getObjects(), r = t.getObjects(), n = r.map((l) => a.indexOf(l)), i = -1, s = function(d) {
2430
- var h = a[d];
2431
- if (!r.includes(h) && n.some((g) => d > g))
2432
- return i = d, 1;
2433
- }, c = 0; c < a.length && !s(c); c += 1)
2434
- ;
2435
- if (i !== -1) {
2436
- var u = r.map((l) => ({
2437
- obj: l,
2438
- index: a.indexOf(l)
2439
- })).sort((l, d) => d.index - l.index);
2440
- u.forEach((l) => {
2441
- var d = a.indexOf(l.obj);
2442
- d < i && (e.moveObjectTo(l.obj, i), i = d);
2443
- });
2444
- }
2445
- }
2446
- /**
2447
- * Сдвигает выделенные объекты на один уровень вниз относительно ближайшего нижнего объекта
2448
- * @param {fabric.Canvas} canvas - экземпляр холста
2449
- * @param {fabric.ActiveSelection} activeSelection - активное выделение
2450
- * @returns {void}
2451
- * @private
2452
- */
2453
- static _moveSelectionBackwards(e, t) {
2454
- for (var a = e.getObjects(), r = t.getObjects(), n = Math.min(...r.map((s) => a.indexOf(s))), i = r.length - 1; i >= 0; i -= 1)
2455
- e.moveObjectTo(r[i], n - 1);
2456
- }
2457
- }
2458
- var pt = ["id", "left", "top", "width", "height", "fill"], bt = ["id", "left", "top", "radius", "fill"], It = ["id", "left", "top", "width", "height", "fill"];
2459
- function je(o, e) {
2460
- var t = Object.keys(o);
2461
- if (Object.getOwnPropertySymbols) {
2462
- var a = Object.getOwnPropertySymbols(o);
2463
- e && (a = a.filter(function(r) {
2464
- return Object.getOwnPropertyDescriptor(o, r).enumerable;
2465
- })), t.push.apply(t, a);
2466
- }
2467
- return t;
2468
- }
2469
- function F(o) {
2470
- for (var e = 1; e < arguments.length; e++) {
2471
- var t = arguments[e] != null ? arguments[e] : {};
2472
- e % 2 ? je(Object(t), !0).forEach(function(a) {
2473
- St(o, a, t[a]);
2474
- }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(o, Object.getOwnPropertyDescriptors(t)) : je(Object(t)).forEach(function(a) {
2475
- Object.defineProperty(o, a, Object.getOwnPropertyDescriptor(t, a));
2476
- });
2477
- }
2478
- return o;
2479
- }
2480
- function St(o, e, t) {
2481
- return (e = At(e)) in o ? Object.defineProperty(o, e, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : o[e] = t, o;
2482
- }
2483
- function At(o) {
2484
- var e = wt(o, "string");
2485
- return typeof e == "symbol" ? e : e + "";
2486
- }
2487
- function wt(o, e) {
2488
- if (typeof o != "object" || !o) return o;
2489
- var t = o[Symbol.toPrimitive];
2490
- if (t !== void 0) {
2491
- var a = t.call(o, e);
2492
- if (typeof a != "object") return a;
2493
- throw new TypeError("@@toPrimitive must return a primitive value.");
2494
- }
2495
- return (e === "string" ? String : Number)(o);
2496
- }
2497
- function G(o, e) {
2498
- if (o == null) return {};
2499
- var t, a, r = Ct(o, e);
2500
- if (Object.getOwnPropertySymbols) {
2501
- var n = Object.getOwnPropertySymbols(o);
2502
- for (a = 0; a < n.length; a++) t = n[a], e.indexOf(t) === -1 && {}.propertyIsEnumerable.call(o, t) && (r[t] = o[t]);
2503
- }
2504
- return r;
2505
- }
2506
- function Ct(o, e) {
2507
- if (o == null) return {};
2508
- var t = {};
2509
- for (var a in o) if ({}.hasOwnProperty.call(o, a)) {
2510
- if (e.indexOf(a) !== -1) continue;
2511
- t[a] = o[a];
2512
- }
2513
- return t;
2514
- }
2515
- class Dt {
2516
- /**
2517
- * @param {object} options
2518
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2519
- */
2520
- constructor(e) {
2521
- var {
2522
- editor: t
2523
- } = e;
2524
- this.editor = t;
2525
- }
2526
- /**
2527
- * Добавление прямоугольника
2528
- * @param {Object} shapeOptions
2529
- * @param {Number} shapeOptions.left - Координата X
2530
- * @param {Number} shapeOptions.top - Координата Y
2531
- * @param {Number} shapeOptions.width - Ширина
2532
- * @param {Number} shapeOptions.height - Высота
2533
- * @param {String} shapeOptions.fill - Цвет заливки
2534
- * @param {String} shapeOptions.originX - Ориентация по X
2535
- * @param {String} shapeOptions.originY - Ориентация по Y
2536
- * @param {Rect} shapeOptions.rest - Остальные параметры
2537
- *
2538
- * @param {Object} flags.withoutSelection - Не выделять объект
2539
- * @param {Object} flags.withoutAdding - Не добавлять объект в canvas
2540
- */
2541
- addRectangle() {
2542
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2543
- id: t = "rect-".concat(I()),
2544
- left: a,
2545
- top: r,
2546
- width: n = 100,
2547
- height: i = 100,
2548
- fill: s = "blue"
2549
- } = e, c = G(e, pt), {
2550
- withoutSelection: u,
2551
- withoutAdding: l
2552
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2553
- canvas: d
2554
- } = this.editor, h = new xe(F({
2555
- id: t,
2556
- left: a,
2557
- top: r,
2558
- width: n,
2559
- height: i,
2560
- fill: s
2561
- }, c));
2562
- return !a && !r && d.centerObject(h), l || (d.add(h), u || d.setActiveObject(h), d.renderAll()), h;
2563
- }
2564
- /**
2565
- * Добавление круга
2566
- * @param {Object} shapeOptions
2567
- * @param {Number} shapeOptions.left - Координата X
2568
- * @param {Number} shapeOptions.top - Координата Y
2569
- * @param {Number} shapeOptions.radius - Радиус
2570
- * @param {string} shapeOptions.fill - Цвет заливки
2571
- * @param {String} shapeOptions.originX - Ориентация по X
2572
- * @param {String} shapeOptions.originY - Ориентация по Y
2573
- * @param {Circle} shapeOptions.rest - Остальные параметры
2574
- *
2575
- * @param {Object} flags.withoutSelection - Не выделять объект
2576
- * @param {Object} flags.withoutAdding - Не добавлять объект в canvas
2577
- */
2578
- addCircle() {
2579
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2580
- id: t = "circle-".concat(I()),
2581
- left: a,
2582
- top: r,
2583
- radius: n = 50,
2584
- fill: i = "green"
2585
- } = e, s = G(e, bt), {
2586
- withoutSelection: c,
2587
- withoutAdding: u
2588
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2589
- canvas: l
2590
- } = this.editor, d = new Be(F({
2591
- id: t,
2592
- left: a,
2593
- top: r,
2594
- fill: i,
2595
- radius: n
2596
- }, s));
2597
- return !a && !r && l.centerObject(d), u || (l.add(d), c || l.setActiveObject(d), l.renderAll()), d;
2598
- }
2599
- /**
2600
- * Добавление треугольника
2601
- * @param {Object} shapeOptions
2602
- * @param {Number} shapeOptions.left - Координата X
2603
- * @param {Number} shapeOptions.top - Координата Y
2604
- * @param {Number} shapeOptions.width - Ширина
2605
- * @param {Number} shapeOptions.height - Высота
2606
- * @param {String} shapeOptions.originX - Ориентация по X
2607
- * @param {String} shapeOptions.originY - Ориентация по Y
2608
- * @param {String} shapeOptions.fill - Цвет заливки
2609
- * @param {Triangle} shapeOptions.rest - Остальные параметры
2610
- *
2611
- * @param {Object} flags.withoutSelection - Не выделять объект
2612
- * @param {Object} flags.withoutAdding - Не добавлять объект в canvas
2613
- */
2614
- addTriangle() {
2615
- var e = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2616
- id: t = "triangle-".concat(I()),
2617
- left: a,
2618
- top: r,
2619
- width: n = 100,
2620
- height: i = 100,
2621
- fill: s = "yellow"
2622
- } = e, c = G(e, It), {
2623
- withoutSelection: u,
2624
- withoutAdding: l
2625
- } = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, {
2626
- canvas: d
2627
- } = this.editor, h = new Ze(F({
2628
- id: t,
2629
- left: a,
2630
- top: r,
2631
- fill: s,
2632
- width: n,
2633
- height: i
2634
- }, c));
2635
- return !a && !r && d.centerObject(h), l || (d.add(h), u || d.setActiveObject(h), d.renderAll()), h;
2636
- }
2637
- }
2638
- function ye(o, e, t, a, r, n, i) {
2639
- try {
2640
- var s = o[n](i), c = s.value;
2641
- } catch (u) {
2642
- return void t(u);
2643
- }
2644
- s.done ? e(c) : Promise.resolve(c).then(a, r);
2645
- }
2646
- function pe(o) {
2647
- return function() {
2648
- var e = this, t = arguments;
2649
- return new Promise(function(a, r) {
2650
- var n = o.apply(e, t);
2651
- function i(c) {
2652
- ye(n, a, r, i, s, "next", c);
2653
- }
2654
- function s(c) {
2655
- ye(n, a, r, i, s, "throw", c);
2656
- }
2657
- i(void 0);
2658
- });
2659
- };
2660
- }
2661
- class Nt {
2662
- /**
2663
- * @param {object} options
2664
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2665
- */
2666
- constructor(e) {
2667
- var {
2668
- editor: t
2669
- } = e;
2670
- this.editor = t, this.clipboard = null;
2671
- }
2672
- /**
2673
- * Копирование объекта
2674
- * @fires editor:object-copied
2675
- */
2676
- copy() {
2677
- var e = this;
2678
- return pe(function* () {
2679
- var {
2680
- canvas: t
2681
- } = e.editor, a = t.getActiveObject();
2682
- if (a)
2683
- try {
2684
- var r = yield a.clone(["format"]);
2685
- if (e.clipboard = r, typeof ClipboardItem == "undefined" || !navigator.clipboard) {
2686
- console.warn(
2687
- // eslint-disable-next-line max-len
2688
- "ClipboardManager. navigator.clipboard не поддерживается в этом браузере или отсутствует соединение по HTTPS-протоколу."
2689
- ), t.fire("editor:object-copied", {
2690
- object: r
2691
- });
2692
- return;
2693
- }
2694
- if (e.clipboard.type !== "image") {
2695
- yield navigator.clipboard.writeText(["application/image-editor", JSON.stringify(r)]), t.fire("editor:object-copied", {
2696
- object: r
2697
- });
2698
- return;
2699
- }
2700
- var n = r.toCanvasElement(), i = yield new Promise((u) => {
2701
- n.toBlob(u);
2702
- }), s = new ClipboardItem({
2703
- [i.type]: i
2704
- });
2705
- yield navigator.clipboard.write([s]), t.fire("editor:object-copied", {
2706
- object: r
2707
- });
2708
- } catch (u) {
2709
- var c = "ClipboardManager. Ошибка записи в системный буфер обмена:";
2710
- console.error(c, u), t.fire("editor:error", {
2711
- message: "".concat(c, " ").concat(u.message)
2712
- });
2713
- }
2714
- })();
2715
- }
2716
- /**
2717
- * Вставка объекта
2718
- * @fires editor:object-pasted
2719
- */
2720
- paste() {
2721
- var e = this;
2722
- return pe(function* () {
2723
- var {
2724
- canvas: t
2725
- } = e.editor;
2726
- if (e.clipboard) {
2727
- var a = yield e.clipboard.clone(["format"]);
2728
- t.discardActiveObject(), a.set({
2729
- id: "".concat(a.type, "-").concat(I()),
2730
- left: a.left + 10,
2731
- top: a.top + 10,
2732
- evented: !0
2733
- }), a instanceof z ? (a.canvas = t, a.forEachObject((r) => {
2734
- t.add(r);
2735
- })) : t.add(a), t.setActiveObject(a), t.requestRenderAll(), t.fire("editor:object-pasted", {
2736
- object: a
2737
- });
2738
- }
2739
- })();
2740
- }
2741
- }
2742
- class Lt {
2743
- /**
2744
- * @param {object} options
2745
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2746
- */
2747
- constructor(e) {
2748
- var {
2749
- editor: t
2750
- } = e;
2751
- this.editor = t;
2752
- }
2753
- /**
2754
- * Блокирует объект (или группу объектов) на канвасе
2755
- * @param {Object} options
2756
- * @param {fabric.Object} [options.object] - объект, который нужно заблокировать
2757
- * @param {Boolean} [options.withoutSave] - не сохранять состояние
2758
- * @returns
2759
- * @fires editor:object-locked
2760
- */
2761
- lockObject() {
2762
- var {
2763
- object: e,
2764
- skipInnerObjects: t,
2765
- withoutSave: a
2766
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2767
- canvas: r,
2768
- historyManager: n
2769
- } = this.editor, i = e || r.getActiveObject();
2770
- if (!(!i || i.locked)) {
2771
- var s = {
2772
- lockMovementX: !0,
2773
- lockMovementY: !0,
2774
- lockRotation: !0,
2775
- lockScalingX: !0,
2776
- lockScalingY: !0,
2777
- lockSkewingX: !0,
2778
- lockSkewingY: !0,
2779
- locked: !0
2780
- };
2781
- i.set(s);
2782
- var c = !t && ["activeselection", "group"].includes(i.type);
2783
- c && i.getObjects().forEach((u) => {
2784
- u.set(s);
2785
- }), r.renderAll(), a || n.saveState(), r.fire("editor:object-locked", {
2786
- object: i
2787
- });
2788
- }
2789
- }
2790
- /**
2791
- * Разблокирует объект (или группу объектов) на канвасе
2792
- * @param {Object} options
2793
- * @param {fabric.Object} [options.object] - объект, который нужно разблокировать
2794
- * @param {Boolean} [options.withoutSave] - не сохранять состояние
2795
- * @returns
2796
- * @fires editor:object-unlocked
2797
- */
2798
- unlockObject() {
2799
- var {
2800
- object: e,
2801
- withoutSave: t
2802
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2803
- canvas: a,
2804
- historyManager: r
2805
- } = this.editor, n = e || a.getActiveObject();
2806
- if (n) {
2807
- var i = {
2808
- lockMovementX: !1,
2809
- lockMovementY: !1,
2810
- lockRotation: !1,
2811
- lockScalingX: !1,
2812
- lockScalingY: !1,
2813
- lockSkewingX: !1,
2814
- lockSkewingY: !1,
2815
- locked: !1
2816
- };
2817
- n.set(i), ["activeselection", "group"].includes(n.type) && n.getObjects().forEach((s) => {
2818
- s.set(i);
2819
- }), a.renderAll(), t || r.saveState(), a.fire("editor:object-unlocked", {
2820
- object: n
2821
- });
2822
- }
2823
- }
2824
- }
2825
- class Ot {
2826
- /**
2827
- * @param {object} options
2828
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2829
- */
2830
- constructor(e) {
2831
- var {
2832
- editor: t
2833
- } = e;
2834
- this.editor = t;
2835
- }
2836
- /**
2837
- * Группировка объектов
2838
- * @param {Object} options
2839
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2840
- * @param {fabric.Object} options.object - массив объектов для группировки
2841
- * @fires editor:objects-grouped
2842
- */
2843
- group() {
2844
- var {
2845
- object: e,
2846
- withoutSave: t
2847
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2848
- canvas: a,
2849
- historyManager: r
2850
- } = this.editor;
2851
- r.suspendHistory();
2852
- var n = e || a.getActiveObject();
2853
- if (n && n.type === "activeselection") {
2854
- var i = n.getObjects(), s = new ze(i);
2855
- i.forEach((c) => a.remove(c)), s.set("id", "".concat(s.type, "-").concat(I())), a.add(s), a.setActiveObject(s), a.renderAll(), r.resumeHistory(), t || r.saveState(), a.fire("editor:objects-grouped");
2856
- }
2857
- }
2858
- /**
2859
- * Разгруппировка объектов
2860
- * @param {Object} options
2861
- * @param {fabric.Group} options.object - объект для разгруппировки
2862
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2863
- * @fires editor:objects-ungrouped
2864
- */
2865
- ungroup() {
2866
- var {
2867
- object: e,
2868
- withoutSave: t
2869
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2870
- canvas: a,
2871
- historyManager: r
2872
- } = this.editor;
2873
- r.suspendHistory();
2874
- var n = e || a.getActiveObject();
2875
- if (!(!n || n.type !== "group")) {
2876
- var i = n.removeAll();
2877
- a.remove(n), i.forEach((c) => a.add(c));
2878
- var s = new z(i, {
2879
- canvas: a
2880
- });
2881
- a.setActiveObject(s), a.renderAll(), r.resumeHistory(), t || r.saveState(), a.fire("editor:objects-ungrouped");
2882
- }
2883
- }
2884
- }
2885
- class Tt {
2886
- /**
2887
- * @param {object} options
2888
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2889
- */
2890
- constructor(e) {
2891
- var {
2892
- editor: t
2893
- } = e;
2894
- this.editor = t;
2895
- }
2896
- /**
2897
- * Выделить все объекты
2898
- * @fires editor:all-objects-selected
2899
- */
2900
- selectAll() {
2901
- var {
2902
- canvas: e,
2903
- canvasManager: t,
2904
- objectLockManager: a
2905
- } = this.editor;
2906
- e.discardActiveObject();
2907
- var r = t.getObjects(), n = r.some((s) => s.locked), i = r.length > 1 ? new z(t.getObjects(), {
2908
- canvas: e
2909
- }) : r[0];
2910
- n && a.lockObject({
2911
- object: i,
2912
- skipInnerObjects: !0,
2913
- withoutSave: !0
2914
- }), e.setActiveObject(i), e.requestRenderAll(), e.fire("editor:all-objects-selected", {
2915
- selected: i
2916
- });
2917
- }
2918
- }
2919
- class Et {
2920
- /**
2921
- * @param {object} options
2922
- * @param {ImageEditor} options.editor - экземпляр редактора с доступом к canvas
2923
- */
2924
- constructor(e) {
2925
- var {
2926
- editor: t
2927
- } = e;
2928
- this.editor = t;
2929
- }
2930
- /**
2931
- * Удалить выбранные объекты
2932
- * @param {Object} options
2933
- * @param {fabric.Object[]} options.objects - массив объектов для удаления
2934
- * @param {Boolean} options.withoutSave - Не сохранять состояние
2935
- * @fires editor:object-deleted
2936
- */
2937
- deleteSelectedObjects() {
2938
- var {
2939
- objects: e,
2940
- withoutSave: t
2941
- } = arguments.length > 0 && arguments[0] !== void 0 ? arguments[0] : {}, {
2942
- canvas: a,
2943
- historyManager: r
2944
- } = this.editor, n = (e || a.getActiveObjects()).filter((i) => !i.locked);
2945
- n != null && n.length && (r.suspendHistory(), n.forEach((i) => {
2946
- if (i.type === "group" && i.format !== "svg") {
2947
- this.ungroup(i), this.deleteSelectedObjects();
2948
- return;
2949
- }
2950
- a.remove(i);
2951
- }), a.discardActiveObject(), a.renderAll(), r.resumeHistory(), t || r.saveState(), a.fire("editor:objects-deleted"));
2952
- }
2953
- }
2954
- function be(o, e, t, a, r, n, i) {
2955
- try {
2956
- var s = o[n](i), c = s.value;
2957
- } catch (u) {
2958
- return void t(u);
2959
- }
2960
- s.done ? e(c) : Promise.resolve(c).then(a, r);
2961
- }
2962
- function kt(o) {
2963
- return function() {
2964
- var e = this, t = arguments;
2965
- return new Promise(function(a, r) {
2966
- var n = o.apply(e, t);
2967
- function i(c) {
2968
- be(n, a, r, i, s, "next", c);
2969
- }
2970
- function s(c) {
2971
- be(n, a, r, i, s, "throw", c);
2972
- }
2973
- i(void 0);
2974
- });
2975
- };
2976
- }
2977
- class ae {
2978
- constructor(e) {
2979
- var t = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};
2980
- this.options = t;
2981
- var {
2982
- defaultScale: a,
2983
- minZoom: r,
2984
- maxZoom: n
2985
- } = t;
2986
- this.containerId = e, this.editorId = "".concat(e, "-").concat(I()), this.clipboard = null, this.defaultZoom = a, this.minZoom = r || gt, this.maxZoom = n || vt, this.init();
2987
- }
2988
- init() {
2989
- var e = this;
2990
- return kt(function* () {
2991
- var {
2992
- editorContainerWidth: t,
2993
- editorContainerHeight: a,
2994
- canvasWrapperWidth: r,
2995
- canvasWrapperHeight: n,
2996
- canvasCSSWidth: i,
2997
- canvasCSSHeight: s,
2998
- initialImage: c,
2999
- initialStateJSON: u,
3000
- scaleType: l,
3001
- _onReadyCallback: d
3002
- } = e.options;
3003
- if ($e.apply(), e.canvas = new Pe(e.containerId, e.options), e.moduleLoader = new _e(), e.workerManager = new Re(), e.historyManager = new ht({
3004
- editor: e
3005
- }), e.toolbar = new lt({
3006
- editor: e
3007
- }), e.transformManager = new jt({
3008
- editor: e
3009
- }), e.canvasManager = new Mt({
3010
- editor: e
3011
- }), e.imageManager = new C({
3012
- editor: e
3013
- }), e.layerManager = new H({
3014
- editor: e
3015
- }), e.shapeManager = new Dt({
3016
- editor: e
3017
- }), e.interactionBlocker = new yt({
3018
- editor: e
3019
- }), e.clipboardManager = new Nt({
3020
- editor: e
3021
- }), e.objectLockManager = new Lt({
3022
- editor: e
3023
- }), e.groupingManager = new Ot({
3024
- editor: e
3025
- }), e.selectionManager = new Tt({
3026
- editor: e
3027
- }), e.deletionManager = new Et({
3028
- editor: e
3029
- }), e._createMonageArea(), e._createClippingArea(), e.listeners = new Z({
3030
- editor: e,
3031
- options: e.options
3032
- }), e.canvasManager.setEditorContainerWidth(t), e.canvasManager.setEditorContainerHeight(a), e.canvasManager.setCanvasWrapperWidth(r), e.canvasManager.setCanvasWrapperHeight(n), e.canvasManager.setCanvasCSSWidth(i), e.canvasManager.setCanvasCSSHeight(s), c != null && c.source) {
3033
- var {
3034
- source: h,
3035
- scale: g = "image-".concat(l),
3036
- withoutSave: v = !0
3037
- } = c;
3038
- yield e.imageManager.importImage({
3039
- source: h,
3040
- scale: g,
3041
- withoutSave: v
3042
- });
3043
- } else
3044
- e.canvasManager.setDefaultScale({
3045
- withoutSave: !0
3046
- });
3047
- u && e.historyManager.loadStateFromFullState(u), e.historyManager.saveState(), console.log("editor:ready"), e.canvas.fire("editor:ready", e), typeof d == "function" && d(e);
3048
- })();
3049
- }
3050
- /**
3051
- * Создаёт монтажную область
3052
- * @private
3053
- * @returns {void}
3054
- */
3055
- _createMonageArea() {
3056
- var {
3057
- montageAreaWidth: e,
3058
- montageAreaHeight: t
3059
- } = this.options;
3060
- this.montageArea = this.shapeManager.addRectangle({
3061
- width: e,
3062
- height: t,
3063
- fill: ae._createMosaicPattern(),
3064
- stroke: null,
3065
- strokeWidth: 0,
3066
- selectable: !1,
3067
- hasBorders: !1,
3068
- hasControls: !1,
3069
- evented: !1,
3070
- id: "montage-area",
3071
- originX: "center",
3072
- originY: "center",
3073
- objectCaching: !1,
3074
- noScaleCache: !0
3075
- }, {
3076
- withoutSelection: !0
3077
- });
3078
- }
3079
- /**
3080
- * Создаёт область клиппинга
3081
- * @private
3082
- * @returns {void}
3083
- */
3084
- _createClippingArea() {
3085
- var {
3086
- montageAreaWidth: e,
3087
- montageAreaHeight: t
3088
- } = this.options;
3089
- this.canvas.clipPath = this.shapeManager.addRectangle({
3090
- id: "area-clip",
3091
- width: e,
3092
- height: t,
3093
- stroke: null,
3094
- fill: null,
3095
- strokeWidth: 0,
3096
- hasBorders: !1,
3097
- hasControls: !1,
3098
- selectable: !1,
3099
- evented: !1,
3100
- originX: "center",
3101
- originY: "center"
3102
- }, {
3103
- withoutSelection: !0,
3104
- withoutAdding: !0
3105
- });
3106
- }
3107
- /**
3108
- * Метод для удаления редактора и всех слушателей.
3109
- */
3110
- destroy() {
3111
- this.listeners.destroy(), this.toolbar.destroy(), this.canvas.dispose(), this.workerManager.worker.terminate(), this.imageManager.revokeBlobUrls();
3112
- }
3113
- /**
3114
- * Создает паттерн мозаики.
3115
- * @returns {object} паттерн мозаики
3116
- */
3117
- static _createMosaicPattern() {
3118
- var e = document.createElement("canvas");
3119
- e.width = 20, e.height = 20;
3120
- var t = e.getContext("2d");
3121
- return t.fillStyle = "#ddd", t.fillRect(0, 0, 40, 40), t.fillStyle = "#ccc", t.fillRect(0, 0, 10, 10), t.fillRect(10, 10, 10, 10), new Ue({
3122
- source: e,
3123
- repeat: "repeat"
3124
- });
3125
- }
3126
- }
3127
- const xt = {
3128
- // Cохраняют ли объекты свой текущий порядок (z-index) при выделении
3129
- preserveObjectStacking: !0,
3130
- // Возможность взаимодействия с объектом за пределами монтажной области
3131
- controlsAboveOverlay: !0,
3132
- // Зум по центру
3133
- centeredScaling: !0,
3134
- // Поворот объекта по центру
3135
- centeredRotation: !0,
3136
- /*
3137
- * Кастомные опции
3138
- */
3139
- // Canvas Montage Area width and height
3140
- montageAreaWidth: 512,
3141
- montageAreaHeight: 512,
3142
- // Canvas backstore width and height
3143
- canvasBackstoreWidth: "auto",
3144
- canvasBackstoreHeight: "auto",
3145
- // Canvas (upper & lower) CSS width and height
3146
- canvasCSSWidth: "100%",
3147
- canvasCSSHeight: "100%",
3148
- // Wrapper CSS width and height
3149
- canvasWrapperWidth: "100%",
3150
- canvasWrapperHeight: "100%",
3151
- // Container CSS width and height
3152
- editorContainerWidth: "fit-content",
3153
- editorContainerHeight: "100%",
3154
- // Максимальная длина истории действий
3155
- maxHistoryLength: 50,
3156
- // Дефолтный тип скейлинга для объектов (cotain/cover)
3157
- scaleType: "contain",
3158
- /**
3159
- * Массив допустимых форматов изображений для загрузки в редактор.
3160
- */
3161
- acceptContentTypes: ["image/png", "image/jpeg", "image/jpg", "image/svg+xml", "image/webp"],
3162
- // Показывать панель инструментов для выделенного объекта
3163
- showToolbar: !0,
3164
- // Настройки панели инструментов выделенного объекта.
3165
- // Можно передать массив с названиями действий или объект с настройками, кастомными иконками и обработчиками
3166
- // ui/toolbar-manager/default-config.js
3167
- toolbar: {
3168
- lockedActions: [{
3169
- name: "Разблокировать",
3170
- handle: "unlock"
3171
- }],
3172
- actions: [{
3173
- name: "Создать копию",
3174
- handle: "copyPaste"
3175
- }, {
3176
- name: "Заблокировать",
3177
- handle: "lock"
3178
- }, {
3179
- name: "На передний план",
3180
- handle: "bringToFront"
3181
- }, {
3182
- name: "На задний план",
3183
- handle: "sendToBack"
3184
- }, {
3185
- name: "На один уровень вверх",
3186
- handle: "bringForward"
3187
- }, {
3188
- name: "На один уровень вниз",
3189
- handle: "sendBackwards"
3190
- }, {
3191
- name: "Удалить",
3192
- handle: "delete"
3193
- }]
3194
- },
3195
- // Можно передать JSON объект в виде строки для инициализации редактора
3196
- initialStateJSON: null,
3197
- /*
3198
- * Объект изображения с которым редактор будет инициализирован. Может содержать:
3199
- * - {String} url - URL изображения (обязательный)
3200
- * - {String} scaleType - Тип скейлинга (image-contain/image-cover/scale-montage)
3201
- * - {Boolean} withoutSave - Не сохранять состояние редактора (по умолчанию false)
3202
- */
3203
- initialImage: null,
3204
- // Дефолтный масштаб
3205
- defaultScale: 1,
3206
- // Минимальный и максимальный зум
3207
- minZoom: 0.1,
3208
- maxZoom: 2,
3209
- // Максимальная кратность зума относительно текущего defaultZoom
3210
- maxZoomFactor: 2,
3211
- // Шаг зума
3212
- zoomRatio: 0.1,
3213
- overlayMaskColor: "rgba(136, 136, 136, 0.6)",
3214
- /*
3215
- * Настройки слушателей событий
3216
- */
3217
- // Адаптировать канвас при изменении размеров контейнера (например, при изменении размеров окна браузера)
3218
- adaptCanvasToContainer: !0,
3219
- // Поднимать объект на передний план по оси Z при выделении
3220
- bringToFrontOnSelection: !1,
3221
- // Зум по колесику мыши
3222
- mouseWheelZooming: !0,
3223
- // Перемещение канваса при зажатой кнопке ALT
3224
- canvasDragging: !1,
3225
- // Копирование объектов (Ctrl + C, Ctrl + V)
3226
- copyObjectsByHotkey: !0,
3227
- // Вставка изображения из буфера обмена
3228
- pasteImageFromClipboard: !0,
3229
- // Отмена/повтор действия по сочетанию клавиш (Ctrl + Z, Ctrl + Y)
3230
- undoRedoByHotKeys: !0,
3231
- // Выделение всех объектов по сочетанию клавиш (Ctrl + A)
3232
- selectAllByHotkey: !0,
3233
- // Удаление объектов по сочетанию клавиш (Delete)
3234
- deleteObjectsByHotkey: !0,
3235
- // Сброс параметров объекта по двойному клику
3236
- resetObjectFitByDoubleClick: !0
3237
- };
3238
- function Ie(o, e) {
3239
- var t = Object.keys(o);
3240
- if (Object.getOwnPropertySymbols) {
3241
- var a = Object.getOwnPropertySymbols(o);
3242
- e && (a = a.filter(function(r) {
3243
- return Object.getOwnPropertyDescriptor(o, r).enumerable;
3244
- })), t.push.apply(t, a);
3245
- }
3246
- return t;
3247
- }
3248
- function Se(o) {
3249
- for (var e = 1; e < arguments.length; e++) {
3250
- var t = arguments[e] != null ? arguments[e] : {};
3251
- e % 2 ? Ie(Object(t), !0).forEach(function(a) {
3252
- Bt(o, a, t[a]);
3253
- }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(o, Object.getOwnPropertyDescriptors(t)) : Ie(Object(t)).forEach(function(a) {
3254
- Object.defineProperty(o, a, Object.getOwnPropertyDescriptor(t, a));
3255
- });
3256
- }
3257
- return o;
3258
- }
3259
- function Bt(o, e, t) {
3260
- return (e = Zt(e)) in o ? Object.defineProperty(o, e, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : o[e] = t, o;
3261
- }
3262
- function Zt(o) {
3263
- var e = zt(o, "string");
3264
- return typeof e == "symbol" ? e : e + "";
3265
- }
3266
- function zt(o, e) {
3267
- if (typeof o != "object" || !o) return o;
3268
- var t = o[Symbol.toPrimitive];
3269
- if (t !== void 0) {
3270
- var a = t.call(o, e);
3271
- if (typeof a != "object") return a;
3272
- throw new TypeError("@@toPrimitive must return a primitive value.");
3273
- }
3274
- return (e === "string" ? String : Number)(o);
3275
- }
3276
- function Yt(o) {
3277
- var e = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {}, t = Se(Se({}, xt), e), a = document.getElementById(o);
3278
- if (!a)
3279
- return Promise.reject(new Error('Контейнер с ID "'.concat(o, '" не найден.')));
3280
- var r = document.createElement("canvas");
3281
- return r.id = "".concat(o, "-canvas"), a.appendChild(r), t.editorContainer = a, new Promise((n) => {
3282
- t._onReadyCallback = n;
3283
- var i = new ae(r.id, t);
3284
- window[o] = i;
3285
- });
3286
- }
3287
- export {
3288
- Yt as default
3289
- };
3290
- //# sourceMappingURL=main.js.map