@anu3ev/fabric-image-editor 0.3.1 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/dist/main.js +42 -38
  2. package/package.json +1 -1
  3. package/readme.md +9 -6
package/dist/main.js CHANGED
@@ -35,7 +35,7 @@ var C = (h, t, e) => new Promise((s, n) => {
35
35
  }, i = (r) => r.done ? s(r.value) : Promise.resolve(r.value).then(o, a);
36
36
  i((e = e.apply(h, t)).next());
37
37
  });
38
- import { ActiveSelection as D, Textbox as gt, util as mt, controlsUtils as pe, InteractiveFabricObject as Me, loadSVGFromURL as Ne, FabricImage as ht, Point as bt, Gradient as ye, Rect as Le, Circle as xe, Triangle as ke, Group as $, Canvas as Be, Pattern as _e } from "fabric";
38
+ import { ActiveSelection as D, Textbox as gt, util as mt, controlsUtils as pe, InteractiveFabricObject as Me, loadSVGFromURL as Ne, FabricImage as ht, Point as bt, Gradient as ye, Rect as Le, Circle as xe, Triangle as ke, Group as $, Canvas as _e, Pattern as Be } from "fabric";
39
39
  import { create as Re } from "jsondiffpatch";
40
40
  import Ue from "diff-match-patch";
41
41
  var ze = "useandom-26T198340PX75pxJACKVERYMINDBUSHWOLF_GQZbfghjklqvwyzrict", x = function() {
@@ -463,18 +463,18 @@ class He {
463
463
  this.worker.terminate();
464
464
  }
465
465
  }
466
- const F = 12, Ze = 2, Dt = 8, Ot = 20, We = 100, Tt = 20, Nt = 8, Pe = 100, vt = 32, xt = 1, Ve = "#2B2D33", kt = "#3D8BF4", Bt = "#FFFFFF";
466
+ const F = 12, Ze = 2, Dt = 8, Ot = 20, We = 100, Tt = 20, Nt = 8, Pe = 100, vt = 32, xt = 1, Ve = "#2B2D33", kt = "#3D8BF4", _t = "#FFFFFF";
467
467
  function Mt(h, t, e, s, n) {
468
468
  const o = F, a = Ze;
469
- h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = Bt, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -o / 2, o, o, a), h.fill(), h.stroke(), h.restore();
469
+ h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = _t, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -o / 2, o, o, a), h.fill(), h.stroke(), h.restore();
470
470
  }
471
471
  function be(h, t, e, s, n) {
472
472
  const o = Dt, a = Ot, i = We;
473
- h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = Bt, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -a / 2, o, a, i), h.fill(), h.stroke(), h.restore();
473
+ h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = _t, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -a / 2, o, a, i), h.fill(), h.stroke(), h.restore();
474
474
  }
475
475
  function ve(h, t, e, s, n) {
476
476
  const o = Tt, a = Nt, i = Pe;
477
- h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = Bt, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -a / 2, o, a, i), h.fill(), h.stroke(), h.restore();
477
+ h.save(), h.translate(t, e), h.rotate(mt.degreesToRadians(n.angle)), h.fillStyle = _t, h.strokeStyle = kt, h.lineWidth = xt, h.beginPath(), h.roundRect(-o / 2, -a / 2, o, a, i), h.fill(), h.stroke(), h.restore();
478
478
  }
479
479
  const Ge = "", we = new Image();
480
480
  we.src = Ge;
@@ -972,7 +972,7 @@ const as = {
972
972
  "white-space": "nowrap",
973
973
  "box-shadow": "0 2px 8px rgba(0, 0, 0, 0.2)"
974
974
  }, je = 16, Ie = 16, is = "fabric-editor-angle-indicator";
975
- class _t {
975
+ class Bt {
976
976
  constructor({ editor: t }) {
977
977
  this.isActive = !1, this.currentAngle = 0, this.editor = t, this.canvas = t.canvas, this.options = t.options, this._createDOM(), this._bindEvents();
978
978
  }
@@ -1000,7 +1000,7 @@ class _t {
1000
1000
  return;
1001
1001
  }
1002
1002
  const s = e.angle || 0;
1003
- this.currentAngle = _t._normalizeAngle(s), this.el.textContent = `${this.currentAngle}°`, this._positionIndicator(t.e), this.isActive || this._showIndicator();
1003
+ this.currentAngle = Bt._normalizeAngle(s), this.el.textContent = `${this.currentAngle}°`, this._positionIndicator(t.e), this.isActive || this._showIndicator();
1004
1004
  }
1005
1005
  /**
1006
1006
  * Обработчик отпускания кнопки мыши
@@ -1538,7 +1538,7 @@ class H {
1538
1538
  [T]
1539
1539
  );
1540
1540
  if (d) {
1541
- const w = m * 0.264583, L = b * 0.264583, B = (yield this.editor.moduleLoader.loadModule("jspdf")).jsPDF, Y = new B({
1541
+ const w = m * 0.264583, L = b * 0.264583, _ = (yield this.editor.moduleLoader.loadModule("jspdf")).jsPDF, Y = new _({
1542
1542
  orientation: w > L ? "landscape" : "portrait",
1543
1543
  unit: "mm",
1544
1544
  format: [w, L]
@@ -1552,8 +1552,8 @@ class H {
1552
1552
  };
1553
1553
  return a.fire("editor:canvas-exported", W), W;
1554
1554
  }
1555
- const _ = Y.output("blob"), Z = {
1556
- image: new File([_], e, { type: "application/pdf" }),
1555
+ const B = Y.output("blob"), Z = {
1556
+ image: new File([B], e, { type: "application/pdf" }),
1557
1557
  format: "pdf",
1558
1558
  contentType: "application/pdf",
1559
1559
  fileName: e
@@ -2442,8 +2442,8 @@ class ps {
2442
2442
  const { canvas: a, montageArea: i } = this.editor, r = a.viewportTransform, c = a.getWidth(), d = a.getHeight(), l = t.x - r[4], u = t.y - r[5], g = Math.abs(n), f = e - s;
2443
2443
  if (Math.abs(f) / g <= 0.1)
2444
2444
  return { x: l, y: u };
2445
- const b = c / 2, p = d / 2, M = i.left, S = i.top, O = b - M * s, T = p - S * s, y = (O - r[4]) / (e - s), A = (T - r[5]) / (e - s), N = y * g, I = A * g, v = N * o, w = I * o, L = Math.abs(v) > Math.abs(l) ? l : v, B = Math.abs(w) > Math.abs(u) ? u : w;
2446
- return { x: L, y: B };
2445
+ const b = c / 2, p = d / 2, M = i.left, S = i.top, O = b - M * s, T = p - S * s, y = (O - r[4]) / (e - s), A = (T - r[5]) / (e - s), N = y * g, I = A * g, v = N * o, w = I * o, L = Math.abs(v) > Math.abs(l) ? l : v, _ = Math.abs(w) > Math.abs(u) ? u : w;
2446
+ return { x: L, y: _ };
2447
2447
  }
2448
2448
  /**
2449
2449
  * Применяет плавное центрирование viewport при приближении к defaultZoom.
@@ -3996,9 +3996,9 @@ const Et = 0.01;
3996
3996
  class k {
3997
3997
  constructor({ editor: t }) {
3998
3998
  var e;
3999
- this.handleTextEditingEntered = () => {
3999
+ this._handleTextEditingEntered = () => {
4000
4000
  this.isTextEditingActive = !0;
4001
- }, this.handleTextChanged = (s) => {
4001
+ }, this._handleTextChanged = (s) => {
4002
4002
  var r, c;
4003
4003
  const { target: n } = s;
4004
4004
  if (!k._isTextbox(n)) return;
@@ -4013,7 +4013,7 @@ class k {
4013
4013
  } else u < l ? n.textCaseRaw = i.slice(0, u) : n.textCaseRaw = o.toLocaleLowerCase();
4014
4014
  } else
4015
4015
  n.textCaseRaw = o;
4016
- }, this.handleTextEditingExited = (s) => {
4016
+ }, this._handleTextEditingExited = (s) => {
4017
4017
  var i, r;
4018
4018
  const { target: n } = s;
4019
4019
  if (!k._isTextbox(n)) return;
@@ -4029,10 +4029,11 @@ class k {
4029
4029
  }), setTimeout(() => {
4030
4030
  this.isTextEditingActive = !1, this.editor.historyManager.saveState();
4031
4031
  }, hs);
4032
- }, this.handleObjectScaling = (s) => {
4032
+ }, this._handleObjectScaling = (s) => {
4033
4033
  var Z, G, W, tt, et, st, nt, ot, at, it, rt, ct, dt, lt;
4034
4034
  const { target: n, transform: o } = s;
4035
4035
  if (!k._isTextbox(n) || !o) return;
4036
+ n.isScaling = !0;
4036
4037
  const a = this._ensureScalingState(n), { baseWidth: i, baseLeft: r, baseFontSize: c } = a, d = typeof ((Z = o.original) == null ? void 0 : Z.width) == "number" ? o.original.width : void 0, l = typeof ((G = o.original) == null ? void 0 : G.left) == "number" ? o.original.left : void 0, u = d != null ? d : i, g = l != null ? l : r, f = (W = o.corner) != null ? W : "", m = (tt = o.action) != null ? tt : "", b = ["ml", "mr"].includes(f) || m === "scaleX", p = ["mt", "mb"].includes(f) || m === "scaleY", M = ["tl", "tr", "bl", "br"].includes(f) || m === "scale";
4037
4038
  if (!b && !p && !M) return;
4038
4039
  const S = Math.abs((st = (et = n.scaleX) != null ? et : o.scaleX) != null ? st : 1) || 1, O = Math.abs((ot = (nt = n.scaleY) != null ? nt : o.scaleY) != null ? ot : 1) || 1, T = Math.max(1, u * S), y = Math.max(1, c * O), A = (it = (at = o.originX) != null ? at : n.originX) != null ? it : "left", N = g + u, I = g + u / 2, v = (rt = n.width) != null ? rt : u, w = Math.abs(T - v) > Et, L = Math.abs(y - ((ct = n.fontSize) != null ? ct : c)) > Et;
@@ -4046,27 +4047,22 @@ class k {
4046
4047
  scaleX: 1,
4047
4048
  scaleY: 1
4048
4049
  });
4049
- const B = (dt = n.width) != null ? dt : T, Y = Math.abs(B - v) > Et;
4050
- let _ = g;
4051
- Y && (b || M) && (A === "right" ? _ = N - B : A === "center" && (_ = I - B / 2)), n.set({ left: _ }), a.baseLeft = _, o.scaleX = 1, o.scaleY = 1;
4050
+ const _ = (dt = n.width) != null ? dt : T, Y = Math.abs(_ - v) > Et;
4051
+ let B = g;
4052
+ Y && (b || M) && (A === "right" ? B = N - _ : A === "center" && (B = I - _ / 2)), n.set({ left: B }), a.baseLeft = B, o.scaleX = 1, o.scaleY = 1;
4052
4053
  const { original: R } = o;
4053
- R && (R.scaleX = 1, R.scaleY = 1, R.width = B, R.height = n.height, R.left = _), n.setCoords(), this.canvas.requestRenderAll(), a.baseWidth = B, a.baseFontSize = (lt = n.fontSize) != null ? lt : y, a.hasWidthChange = Y || L;
4054
- }, this.handleObjectModified = (s) => {
4054
+ R && (R.scaleX = 1, R.scaleY = 1, R.width = _, R.height = n.height, R.left = B), n.setCoords(), this.canvas.requestRenderAll(), a.baseWidth = _, a.baseFontSize = (lt = n.fontSize) != null ? lt : y, a.hasWidthChange = Y || L;
4055
+ }, this._handleObjectModified = (s) => {
4055
4056
  var r, c, d;
4056
4057
  const { target: n } = s;
4057
4058
  if (!k._isTextbox(n)) return;
4059
+ n.isScaling = !1;
4058
4060
  const o = this.scalingState.get(n);
4059
4061
  if (this.scalingState.delete(n), !(o != null && o.hasWidthChange)) return;
4060
4062
  const a = (r = n.width) != null ? r : n.calcTextWidth(), i = (d = (c = n.fontSize) != null ? c : o == null ? void 0 : o.baseFontSize) != null ? d : 16;
4061
- this.updateText(n, { width: a, fontSize: i }), n.set({ scaleX: 1, scaleY: 1 }), n.setCoords();
4063
+ this.updateText({ target: n, style: { width: a, fontSize: i } }), n.set({ scaleX: 1, scaleY: 1 }), n.setCoords();
4062
4064
  }, this.editor = t, this.canvas = t.canvas, this.fonts = (e = t.options.fonts) != null ? e : [], this.scalingState = /* @__PURE__ */ new WeakMap(), this.isTextEditingActive = !1, this._bindEvents();
4063
4065
  }
4064
- /**
4065
- * Уничтожает менеджер и снимает слушатели.
4066
- */
4067
- destroy() {
4068
- this.canvas.off("object:scaling", this.handleObjectScaling), this.canvas.off("object:modified", this.handleObjectModified), this.canvas.off("text:editing:exited", this.handleTextEditingExited), this.canvas.off("text:editing:entered", this.handleTextEditingEntered), this.canvas.off("text:changed", this.handleTextChanged);
4069
- }
4070
4066
  /**
4071
4067
  * Добавляет новый текстовый объект на канвас.
4072
4068
  * @param options — настройки текста
@@ -4159,12 +4155,14 @@ class k {
4159
4155
  }
4160
4156
  /**
4161
4157
  * Обновляет текстовый объект.
4162
- * @param target — объект, его id или активный объект (если не передан)
4163
- * @param style — стиль, который нужно применить
4164
4158
  * @param options — настройки обновления
4159
+ * @param options.target — объект, его id или активный объект (если не передан)
4160
+ * @param options.style — стиль, который нужно применить
4161
+ * @param options.withoutSave — не сохранять состояние в историю
4162
+ * @param options.skipRender — не вызывать перерисовку канваса
4165
4163
  */
4166
- updateText(t, e = {}, { withoutSave: s, skipRender: n } = {}) {
4167
- var Y, _, R, Z, G, W, tt, et, st, nt, ot, at, it, rt, ct, dt, lt, zt, Yt, Ht, Zt, Wt, Pt, Vt, Gt, Xt, Kt, $t, Qt, Jt, qt, te, ee, se, ne, oe, ae, ie, re, ce, de, le;
4164
+ updateText({ target: t, style: e = {}, withoutSave: s, skipRender: n } = {}) {
4165
+ var Y, B, R, Z, G, W, tt, et, st, nt, ot, at, it, rt, ct, dt, lt, zt, Yt, Ht, Zt, Wt, Pt, Vt, Gt, Xt, Kt, $t, Qt, Jt, qt, te, ee, se, ne, oe, ae, ie, re, ce, de, le;
4168
4166
  const o = this._resolveTextObject(t);
4169
4167
  if (!o) return null;
4170
4168
  const { historyManager: a } = this.editor;
@@ -4172,7 +4170,7 @@ class k {
4172
4170
  const i = {
4173
4171
  id: o.id,
4174
4172
  text: (Y = o.text) != null ? Y : void 0,
4175
- textCaseRaw: (_ = o.textCaseRaw) != null ? _ : void 0,
4173
+ textCaseRaw: (B = o.textCaseRaw) != null ? B : void 0,
4176
4174
  uppercase: !!o.uppercase,
4177
4175
  fontFamily: (R = o.fontFamily) != null ? R : void 0,
4178
4176
  fontSize: (Z = o.fontSize) != null ? Z : void 0,
@@ -4232,7 +4230,7 @@ class k {
4232
4230
  y.text = wt, o.textCaseRaw = v;
4233
4231
  } else o.textCaseRaw === void 0 && (o.textCaseRaw = A);
4234
4232
  o.uppercase = w, o.set(y), o.setCoords(), n || this.canvas.requestRenderAll(), a.resumeHistory(), s || a.saveState();
4235
- const B = {
4233
+ const _ = {
4236
4234
  id: o.id,
4237
4235
  text: (Vt = o.text) != null ? Vt : void 0,
4238
4236
  textCaseRaw: (Gt = o.textCaseRaw) != null ? Gt : void 0,
@@ -4266,9 +4264,15 @@ class k {
4266
4264
  },
4267
4265
  updates: y,
4268
4266
  before: i,
4269
- after: B
4267
+ after: _
4270
4268
  }), o;
4271
4269
  }
4270
+ /**
4271
+ * Уничтожает менеджер и снимает слушатели.
4272
+ */
4273
+ destroy() {
4274
+ this.canvas.off("object:scaling", this._handleObjectScaling), this.canvas.off("object:modified", this._handleObjectModified), this.canvas.off("text:editing:exited", this._handleTextEditingExited), this.canvas.off("text:editing:entered", this._handleTextEditingEntered), this.canvas.off("text:changed", this._handleTextChanged);
4275
+ }
4272
4276
  /**
4273
4277
  * Возвращает активный текст или ищет по id.
4274
4278
  */
@@ -4288,7 +4292,7 @@ class k {
4288
4292
  return !!t && t instanceof gt;
4289
4293
  }
4290
4294
  _bindEvents() {
4291
- this.canvas.on("object:scaling", this.handleObjectScaling), this.canvas.on("object:modified", this.handleObjectModified), this.canvas.on("text:editing:entered", this.handleTextEditingEntered), this.canvas.on("text:editing:exited", this.handleTextEditingExited), this.canvas.on("text:changed", this.handleTextChanged);
4295
+ this.canvas.on("object:scaling", this._handleObjectScaling), this.canvas.on("object:modified", this._handleObjectModified), this.canvas.on("text:editing:entered", this._handleTextEditingEntered), this.canvas.on("text:editing:exited", this._handleTextEditingExited), this.canvas.on("text:changed", this._handleTextChanged);
4292
4296
  }
4293
4297
  _ensureScalingState(t) {
4294
4298
  var s, n, o;
@@ -4349,7 +4353,7 @@ class Ut {
4349
4353
  showRotationAngle: d,
4350
4354
  _onReadyCallback: l
4351
4355
  } = this.options;
4352
- if (jt.apply(), this.canvas = new Be(this.containerId, this.options), this.moduleLoader = new Ye(), this.workerManager = new He(), this.errorManager = new St({ editor: this }), this.historyManager = new It({ editor: this }), this.toolbar = new os({ editor: this }), this.transformManager = new ms({ editor: this }), this.zoomManager = new ps({ editor: this }), this.canvasManager = new fs({ editor: this }), this.imageManager = new H({ editor: this }), this.layerManager = new At({ editor: this }), this.shapeManager = new ys({ editor: this }), this.interactionBlocker = new Ms({ editor: this }), this.backgroundManager = new Q({ editor: this }), this.clipboardManager = new bs({ editor: this }), this.objectLockManager = new Ct({ editor: this }), this.groupingManager = new vs({ editor: this }), this.selectionManager = new js({ editor: this }), this.deletionManager = new Rt({ editor: this }), this.panConstraintManager = new As({ editor: this }), this.fontManager = new Lt((u = this.options.fonts) != null ? u : []), this.textManager = new k({ editor: this }), d && (this.angleIndicator = new _t({ editor: this })), this._createMontageArea(), this._createClippingArea(), this.listeners = new ft({ editor: this, options: this.options }), this.canvasManager.setEditorContainerWidth(t), this.canvasManager.setEditorContainerHeight(e), this.canvasManager.setCanvasWrapperWidth(s), this.canvasManager.setCanvasWrapperHeight(n), this.canvasManager.setCanvasCSSWidth(o), this.canvasManager.setCanvasCSSHeight(a), this.canvasManager.updateCanvas(), this.zoomManager.calculateAndApplyDefaultZoom(), yield this.fontManager.loadFonts(), i != null && i.source) {
4356
+ if (jt.apply(), this.canvas = new _e(this.containerId, this.options), this.moduleLoader = new Ye(), this.workerManager = new He(), this.errorManager = new St({ editor: this }), this.historyManager = new It({ editor: this }), this.toolbar = new os({ editor: this }), this.transformManager = new ms({ editor: this }), this.zoomManager = new ps({ editor: this }), this.canvasManager = new fs({ editor: this }), this.imageManager = new H({ editor: this }), this.layerManager = new At({ editor: this }), this.shapeManager = new ys({ editor: this }), this.interactionBlocker = new Ms({ editor: this }), this.backgroundManager = new Q({ editor: this }), this.clipboardManager = new bs({ editor: this }), this.objectLockManager = new Ct({ editor: this }), this.groupingManager = new vs({ editor: this }), this.selectionManager = new js({ editor: this }), this.deletionManager = new Rt({ editor: this }), this.panConstraintManager = new As({ editor: this }), this.fontManager = new Lt((u = this.options.fonts) != null ? u : []), this.textManager = new k({ editor: this }), d && (this.angleIndicator = new Bt({ editor: this })), this._createMontageArea(), this._createClippingArea(), this.listeners = new ft({ editor: this, options: this.options }), this.canvasManager.setEditorContainerWidth(t), this.canvasManager.setEditorContainerHeight(e), this.canvasManager.setCanvasWrapperWidth(s), this.canvasManager.setCanvasWrapperHeight(n), this.canvasManager.setCanvasCSSWidth(o), this.canvasManager.setCanvasCSSHeight(a), this.canvasManager.updateCanvas(), this.zoomManager.calculateAndApplyDefaultZoom(), yield this.fontManager.loadFonts(), i != null && i.source) {
4353
4357
  const {
4354
4358
  source: g,
4355
4359
  scale: f = `image-${c}`,
@@ -4422,7 +4426,7 @@ class Ut {
4422
4426
  const t = document.createElement("canvas");
4423
4427
  t.width = 20, t.height = 20;
4424
4428
  const e = t.getContext("2d");
4425
- 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 _e({
4429
+ 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 Be({
4426
4430
  source: t,
4427
4431
  repeat: "repeat"
4428
4432
  });
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@anu3ev/fabric-image-editor",
3
- "version": "0.03.01",
3
+ "version": "0.4.1",
4
4
  "description": "JavaScript image editor built on FabricJS, allowing you to create instances with an integrated montage area and providing an API to modify and manage state.",
5
5
  "module": "dist/main.js",
6
6
  "files": [
package/readme.md CHANGED
@@ -129,17 +129,20 @@ const textbox = editor.textManager.addText({
129
129
  })
130
130
 
131
131
  // Update existing text
132
- editor.textManager.updateText(textbox, {
133
- text: 'HELLO FABRIC',
134
- uppercase: true,
135
- strokeColor: '#2563eb',
136
- strokeWidth: 2
132
+ editor.textManager.updateText({
133
+ target: textbox,
134
+ style: {
135
+ text: 'HELLO FABRIC',
136
+ uppercase: true,
137
+ strokeColor: '#2563eb',
138
+ strokeWidth: 2
139
+ }
137
140
  })
138
141
  ```
139
142
 
140
143
  ### Configuring Fonts
141
144
 
142
- By default the editor ships with a curated Google Fonts collection (Latin + Cyrillic coverage).
145
+ By default the editor ships with a curated Google Fonts collection (Latin + Cyrillic coverage).
143
146
  If you want to use your own fonts, supply a `fonts` array – the provided list will replace the defaults.
144
147
 
145
148
  ```typescript