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