myrta-ui 17.2.6 → 17.2.7

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.
@@ -79,6 +79,31 @@ export class EditorComponent {
79
79
  blur: () => {
80
80
  this.changeFocused(false);
81
81
  },
82
+ keydown: (event) => {
83
+ const editor = this.editorElementRef?.jodit;
84
+ if (!editor || this.maxLength <= 0)
85
+ return true;
86
+ // 1. Получаем текущую длину текста
87
+ const currentLength = editor.text.length;
88
+ // 2. Список системных клавиш, которые нельзя блокировать
89
+ const isNavigationKey = [
90
+ 'Backspace', 'Delete', 'ArrowLeft', 'ArrowRight',
91
+ 'ArrowUp', 'ArrowDown', 'End', 'Home', 'Tab', 'Enter'
92
+ ].includes(event.key);
93
+ // Разрешаем сочетания с Ctrl/Cmd (Copy, Paste, Select All)
94
+ const isModifier = event.ctrlKey || event.metaKey;
95
+ if (currentLength >= this.maxLength && !isNavigationKey && !isModifier) {
96
+ // 3. Самый надежный способ проверить, выделено ли что-то:
97
+ // Если метод sel.isCollapsed недоступен, проверяем длину выделения через строку
98
+ const selectedText = editor.s.sel ? editor.s.sel.toString() : '';
99
+ // Если ничего не выделено, блокируем ввод нового символа
100
+ if (!selectedText.length) {
101
+ event.preventDefault();
102
+ return false;
103
+ }
104
+ }
105
+ return true;
106
+ },
82
107
  paste: (event) => {
83
108
  this.handlePaste(event);
84
109
  },
@@ -189,22 +214,35 @@ export class EditorComponent {
189
214
  this.changeDetection.detectChanges();
190
215
  }
191
216
  handlePaste(event) {
192
- if (!event.clipboardData || this.hasImageButton()) {
217
+ const editor = this.editorElementRef?.jodit;
218
+ if (!event.clipboardData || !editor)
193
219
  return;
220
+ const textData = event.clipboardData.getData('text/plain');
221
+ const currentLength = editor.text.length;
222
+ // Если вставка превышает лимит
223
+ if (this.maxLength > 0 && (currentLength + textData.length) > this.maxLength) {
224
+ event.preventDefault();
225
+ const remainingSpace = this.maxLength - currentLength;
226
+ if (remainingSpace > 0) {
227
+ // Вставляем только разрешенный кусочек
228
+ const truncatedText = textData.substring(0, remainingSpace);
229
+ editor.s.insertHTML(truncatedText);
230
+ }
231
+ return; // Прекращаем выполнение, чтобы не сработала логика картинок ниже
194
232
  }
195
- const { clipboardData } = event;
196
- const htmlData = clipboardData.getData('text/html');
197
- const hasHtmlImage = htmlData?.includes('<img');
198
- const hasImage = Array.from(clipboardData.items).some(item => item.type.startsWith('image/'));
199
- if (!hasImage && !hasHtmlImage) {
233
+ // Ваша существующая логика защиты от картинок
234
+ if (this.hasImageButton())
200
235
  return;
201
- }
202
- event.preventDefault();
203
- const sanitizedHtml = this.removeImagesFromHtml(htmlData);
204
- const textData = clipboardData.getData('text/plain');
205
- const contentToInsert = sanitizedHtml || textData;
206
- if (contentToInsert) {
207
- this.editorElementRef.jodit.s.insertHTML(contentToInsert);
236
+ const htmlData = event.clipboardData.getData('text/html');
237
+ const hasHtmlImage = htmlData?.includes('<img');
238
+ const hasImage = Array.from(event.clipboardData.items).some(item => item.type.startsWith('image/'));
239
+ if (hasImage || hasHtmlImage) {
240
+ event.preventDefault();
241
+ const sanitizedHtml = this.removeImagesFromHtml(htmlData);
242
+ const contentToInsert = sanitizedHtml || textData;
243
+ if (contentToInsert) {
244
+ editor.s.insertHTML(contentToInsert);
245
+ }
208
246
  }
209
247
  }
210
248
  removeImagesFromHtml(html) {
@@ -250,6 +288,7 @@ export class EditorComponent {
250
288
  }
251
289
  }
252
290
  updateValue(insideValue) {
291
+ // Сначала считаем, потом проверяем инициализацию
253
292
  if (this.maxLength) {
254
293
  this._calculateChartsCount();
255
294
  }
@@ -265,7 +304,12 @@ export class EditorComponent {
265
304
  return value ? value.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, '') : value;
266
305
  }
267
306
  _calculateChartsCount() {
268
- this.valueLength = this.editorElementRef.editor.text.replace(/\s+/g, '').length;
307
+ // В Jodit правильно считать длину через .text,
308
+ // но в ngx-jodit инстанс доступен через .jodit
309
+ if (this.editorElementRef?.jodit) {
310
+ this.valueLength = this.editorElementRef.jodit.text.length;
311
+ this.changeDetection.detectChanges(); // Форсируем обновление UI для счетчика
312
+ }
269
313
  }
270
314
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: EditorComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
271
315
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: EditorComponent, selector: "mrx-editor", inputs: { fields: "fields", toolbar: "toolbar", maxLength: "maxLength", minHeight: "minHeight", customClasses: "customClasses", placeholder: "placeholder", disabled: "disabled", readonly: "readonly", iframe: "iframe", defaultInlineStyle: "defaultInlineStyle", config: "config", editHTMLDocumentMode: "editHTMLDocumentMode", defaultMode: "defaultMode", invalid: "invalid", invalidMessage: "invalidMessage", checkInvalid: "checkInvalid" }, outputs: { changed: "changed", modelChange: "modelChange" }, providers: [
@@ -325,4 +369,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
325
369
  }], modelChange: [{
326
370
  type: Output
327
371
  }] } });
328
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrta-ui/src/lib/components/form/editor/editor.component.ts","../../../../../../../projects/myrta-ui/src/lib/components/form/editor/editor.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EAEvB,SAAS,EACT,YAAY,EACZ,UAAU,EACV,KAAK,EAEL,MAAM,EAEN,SAAS,EACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAC1C,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,aAAa,EAAE,MAAM,yBAAyB,CAAC;AAExD,OAAO,EAAE,EAAE,IAAI,MAAM,EAAE,MAAM,MAAM,CAAC;AAEpC,OAAO,EAAE,0BAA0B,EAAE,MAAM,+BAA+B,CAAC;AAE3E,OAAO,EAAE,KAAK,EAAE,MAAM,OAAO,CAAC;AAG9B,OAAO,EAAE,MAAM,uBAAuB,CAAC;AAEvC,OAAO,oCAAoC,CAAC;AAC5C,OAAO,8CAA8C,CAAC;AACtD,OAAO,oCAAoC,CAAC;AAC5C,OAAO,kCAAkC,CAAC;AAC1C,OAAO,oCAAoC,CAAC;AAC5C,OAAO,sCAAsC,CAAC;AAC9C,OAAO,EAAE,mBAAmB,EAA0B,MAAM,eAAe,CAAC;;;;;;;;AAE5E,KAAK,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;AAEnB,2DAA2D;AAC3D,+BAA+B;AAC/B,eAAe;AACf,oBAAoB;AACpB,IAAI;AAeJ,MAAM,OAAO,eAAe;IA2FN;IA1FZ,OAAO,GAAG,KAAK,CAAC;IACjB,KAAK,GAAG,EAAE,CAAC;IACX,WAAW,GAAG,CAAC,CAAC;IAChB,SAAS,GAAmB,KAAK,CAAC;IAElC,OAAO,GAAG;QACf,QAAQ,EAAE,IAAI;KACf,CAAC;IAEK,aAAa,GAAQ;QAC1B,GAAG,cAAc;QACjB,KAAK,EAAE,aAAa;QACpB,WAAW,EAAE,mBAAmB,CAAC,YAAY;QAC7C,QAAQ,EAAE,IAAI;QACd,gBAAgB,EAAE,KAAK;QACvB,gBAAgB,EAAE,KAAK;QACvB,oBAAoB,EAAE,KAAK;QAC3B,IAAI,EAAE;YACJ,gBAAgB,EAAE,KAAK;SACxB;QACD,SAAS,EAAE,GAAG;QACd,MAAM,EAAE,KAAK;QACb,iBAAiB,EAAE,GAAG;QACtB,oBAAoB,EAAE,IAAI;QAC1B,SAAS,EAAE;YACT,kBAAkB,EAAE,IAAI;SACzB;QACD,WAAW,EAAE,EAAE;QACf,eAAe,EAAE,KAAK;QACtB,aAAa,EAAE,IAAI;QACnB,yBAAyB,EAAE,KAAK;QAChC,uBAAuB,EAAE,EAAE;QAC3B,4BAA4B,EAAE,CAAC,QAAQ,CAAC;QACxC,QAAQ,EAAE,KAAK;QACf,QAAQ,EAAE,KAAK;QACf,kBAAkB,EAAE,KAAK;QACzB,oBAAoB,EAAE,kBAAkB;QACxC,sBAAsB,EAAE,KAAK;QAC7B,kBAAkB,EAAE,IAAI;QACxB,YAAY,EAAE,IAAI;QAClB,UAAU,EAAE,IAAI;QAChB,gBAAgB,EAAE,EAAE;QACpB,OAAO,EAAE,CAAC,QAAgB,EAAE,EAAE;YAC5B,OAAO,mBAAmB,CAAC,QAAQ,CAAC,CAAC;QACvC,CAAC;QACD,MAAM,EAAE;YACN,KAAK,EAAE,GAAG,EAAE;gBACV,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;YAC3B,CAAC;YACD,IAAI,EAAE,GAAG,EAAE;gBACT,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;YAC5B,CAAC;YACD,KAAK,EAAE,CAAC,KAAqB,EAAE,EAAE;gBAC/B,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAC1B,CAAC;SACF;QACD,cAAc;QACd,iCAAiC;QACjC,KAAK;QACL,UAAU,EAAE,EAAE;QACd,kBAAkB;KACnB,CAAC;IAEF,aAAa;IACN,IAAI,GAAW,MAAM,EAAE,CAAC;IACf,MAAM,GAAY,EAAE,CAAC;IAErB,OAAO,CAAiB;IACxB,SAAS,GAAG,CAAC,CAAC;IACd,SAAS,GAAG,GAAG,CAAC;IAChB,aAAa,GAAG,EAAE,CAAC;IACnB,WAAW,GAAG,EAAE,CAAC;IACjB,QAAQ,GAAG,KAAK,CAAC;IACjB,QAAQ,GAAG,KAAK,CAAC;IACjB,MAAM,GAAG,KAAK,CAAC;IACf,kBAAkB,GAAG,KAAK,CAAC;IAC3B,MAAM,GAAQ,IAAI,CAAC,aAAa,CAAC;IAEjC,oBAAoB,GAAG,KAAK,CAAC;IAC7B,WAAW,GAAwB,mBAAmB,CAAC,YAAY,CAAC;IAEpE,OAAO,GAAG,KAAK,CAAC;IAChB,cAAc,GAAsB,EAAE,CAAC;IACvC,YAAY,GAAwB,IAAI,CAAC;IAE1B,gBAAgB,CAAO;IAErC,OAAO,GAAyB,IAAI,YAAY,EAAU,CAAC;IAC3D,WAAW,GAAyC,IAAI,YAAY,EAA0B,CAAC;IAEhH,YAAoB,eAAkC;QAAlC,oBAAe,GAAf,eAAe,CAAmB;IACtD,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YACvB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,YAAY,EAAC,CAAC;QACnF,CAAC;QACD,IAAI,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,UAAU,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC,YAAY,EAAC,CAAC;QAC9F,CAAC;QACD,IAAI,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC;YAC3B,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,WAAW,EAAE,OAAO,CAAC,aAAa,CAAC,CAAC,YAAY,EAAC,CAAC;QACjG,CAAC;QACD,IAAI,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YACxB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,QAAQ,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC,YAAY,EAAC,CAAC;QAC3F,CAAC;QACD,IAAI,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YACxB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,QAAQ,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC,YAAY,EAAC,CAAC;QAC3F,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YACtB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,MAAM,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC,YAAY,EAAC,CAAC;QACvF,CAAC;QACD,IAAI,OAAO,CAAC,oBAAoB,CAAC,EAAE,CAAC;YAClC,IAAI,CAAC,aAAa,GAAG;gBACnB,GAAG,IAAI,CAAC,aAAa;gBACrB,gBAAgB,EAAE,EAAC,GAAG,IAAI,CAAC,aAAa,CAAC,gBAAgB,EAAE,GAAG,0BAA0B,EAAC;aAC1F,CAAC;QACJ,CAAC;QACD,IAAI,OAAO,CAAC,sBAAsB,CAAC,EAAE,CAAC;YACpC,IAAI,CAAC,aAAa,GAAG;gBACnB,GAAG,IAAI,CAAC,aAAa;gBACrB,oBAAoB,EAAE,OAAO,CAAC,sBAAsB,CAAC,CAAC,YAAY;aACnE,CAAC;QACJ,CAAC;QACD,IAAI,OAAO,CAAC,oBAAoB,CAAC,EAAE,CAAC;YAClC,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,kBAAkB,EAAE,OAAO,CAAC,oBAAoB,CAAC,CAAC,YAAY,EAAC,CAAC;QAC/G,CAAC;QACD,IAAI,OAAO,CAAC,wBAAwB,CAAC,EAAE,CAAC;YACtC,IAAI,CAAC,aAAa,GAAG;gBACnB,GAAG,IAAI,CAAC,aAAa;gBACrB,sBAAsB,EAAE,OAAO,CAAC,wBAAwB,CAAC,CAAC,YAAY;aACvE,CAAC;QACJ,CAAC;QACD,IAAI,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC;YAC3B,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,WAAW,EAAE,OAAO,CAAC,aAAa,CAAC,CAAC,YAAY,EAAC,CAAC;QACjG,CAAC;QACD,IAAI,OAAO,CAAC,sBAAsB,CAAC,EAAE,CAAC;YACpC,IAAI,CAAC,aAAa,GAAG;gBACnB,GAAG,IAAI,CAAC,aAAa;gBACrB,oBAAoB,EAAE,OAAO,CAAC,sBAAsB,CAAC,CAAC,YAAY;aACnE,CAAC;QACJ,CAAC;QACD,IAAI,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,SAAS,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC,YAAY,EAAC,CAAC;QAC7F,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YACtB,IAAI,CAAC,aAAa,GAAG,EAAC,GAAG,IAAI,CAAC,aAAa,EAAE,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC,YAAY,EAAC,CAAC;QAClF,CAAC;IACH,CAAC;IAED,IAAW,OAAO;QAChB,OAAO,IAAI,CAAC;IACd,CAAC;IAED,IAAW,gBAAgB;QACzB,OAAO,CAAC,CAAC,IAAI,CAAC,cAAc,IAAI,CAAC,CAAC,IAAI,CAAC,cAAc,EAAE,MAAM,CAAC;IAChE,CAAC;IAED,IAAW,iBAAiB;QAC1B,OAAO,IAAI,CAAC,YAAY,KAAK,KAAK,CAAC,CAAC,CAAC,2BAA2B,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC,EAAE,CAAC;IACjI,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,GAAG,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3D,CAAC;IAEM,kBAAkB,CAAC,IAAY;QACpC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IACjD,CAAC;IAEM,aAAa,CAAC,KAAc;QACjC,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,CAAC;IACvC,CAAC;IAEM,WAAW,CAAC,KAAqB;QACtC,IAAI,CAAC,KAAK,CAAC,aAAa,IAAI,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC;YAClD,OAAO;QACT,CAAC;QAED,MAAM,EAAC,aAAa,EAAC,GAAG,KAAK,CAAC;QAC9B,MAAM,QAAQ,GAAG,aAAa,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QACpD,MAAM,YAAY,GAAG,QAAQ,EAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;QAChD,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC;QAE9F,IAAI,CAAC,QAAQ,IAAI,CAAC,YAAY,EAAE,CAAC;YAC/B,OAAO;QACT,CAAC;QAED,KAAK,CAAC,cAAc,EAAE,CAAC;QAEvB,MAAM,aAAa,GAAG,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;QAC1D,MAAM,QAAQ,GAAG,aAAa,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACrD,MAAM,eAAe,GAAG,aAAa,IAAI,QAAQ,CAAC;QAElD,IAAI,eAAe,EAAE,CAAC;YACpB,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;QAC5D,CAAC;IAEH,CAAC;IAEM,oBAAoB,CAAC,IAAY;QACtC,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,GAAG,GAAG,QAAQ,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAC1C,GAAG,CAAC,SAAS,GAAG,IAAI,CAAC;QACrB,GAAG,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;QAEzD,OAAO,GAAG,CAAC,SAAS,CAAC;IACvB,CAAC;IAEM,cAAc;QACnB,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,EAAE,OAAO,CAAC;QAE5C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE,CAAC;YAChC,OAAO,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QACnC,CAAC;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3B,OAAO,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACzB,OAAO,IAAI,KAAK,QAAQ;gBACtB,CAAC,CAAC,IAAI,KAAK,OAAO;gBAClB,CAAC,CAAC,IAAI,EAAE,IAAI,KAAK,OAAO,CAC3B,CAAC;QACJ,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,QAAQ,GAAG,CAAC,KAAU,EAAE,EAAE;IAClC,CAAC,CAAC;IACM,SAAS,GAAG,GAAG,EAAE;IACzB,CAAC,CAAC;IAEK,gBAAgB,CAAC,EAAO;QAC7B,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAEM,iBAAiB,CAAC,EAAY;QACnC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAEM,UAAU,CAAC,YAAiB;QACjC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC,CAAC;QAC/C,IAAI,YAAY,KAAK,IAAI,EAAE,CAAC;YAC1B,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;YACtB,CAAC,EAAE,GAAG,CAAC,CAAC;QACV,CAAC;IACH,CAAC;IAEM,WAAW,CAAC,WAAmB;QAEpC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/B,CAAC;QACD,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC;YACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAC/B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAC,KAAK,EAAE,WAAW,EAAE,EAAE,EAAE,IAAI,CAAC,IAAI,EAAC,CAAC,CAAC;YAC3D,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;YAC3B,IAAI,CAAC,SAAS,EAAE,CAAC;QACnB,CAAC;IACH,CAAC;IAEO,cAAc,CAAC,KAAa;QAClC,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,qDAAqD,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;IAClG,CAAC;IAEO,qBAAqB;QAC3B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC;IAClF,CAAC;wGAtRU,eAAe;4FAAf,eAAe,whBARf;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC;gBAC9C,KAAK,EAAE,IAAI;aACZ;SACF,qKCrDH,4vBAwBA;;4FD+Ba,eAAe;kBAb3B,SAAS;+BACE,YAAY,mBAGL,uBAAuB,CAAC,MAAM,aACpC;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,gBAAgB,CAAC;4BAC9C,KAAK,EAAE,IAAI;yBACZ;qBACF;sFAoEe,MAAM;sBAArB,KAAK;gBAEU,OAAO;sBAAtB,KAAK;gBACU,SAAS;sBAAxB,KAAK;gBACU,SAAS;sBAAxB,KAAK;gBACU,aAAa;sBAA5B,KAAK;gBACU,WAAW;sBAA1B,KAAK;gBACU,QAAQ;sBAAvB,KAAK;gBACU,QAAQ;sBAAvB,KAAK;gBACU,MAAM;sBAArB,KAAK;gBACU,kBAAkB;sBAAjC,KAAK;gBACU,MAAM;sBAArB,KAAK;gBAEU,oBAAoB;sBAAnC,KAAK;gBACU,WAAW;sBAA1B,KAAK;gBAEU,OAAO;sBAAtB,KAAK;gBACU,cAAc;sBAA7B,KAAK;gBACU,YAAY;sBAA3B,KAAK;gBAEyB,gBAAgB;sBAA9C,SAAS;uBAAC,kBAAkB;gBAEZ,OAAO;sBAAvB,MAAM;gBACU,WAAW;sBAA3B,MAAM","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  EventEmitter,\r\n  forwardRef,\r\n  Input,\r\n  OnChanges,\r\n  Output,\r\n  SimpleChanges,\r\n  ViewChild\r\n} from '@angular/core';\r\nimport { changeIconsFunction } from './modules/change-icon-module';\r\nimport { defaultToolbar } from './config';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\r\nimport { popupToolbars } from './config/popups-toolbar';\r\nimport { ToolbarConfig } from './models/toolbar.model';\r\nimport { v4 as uuidv4 } from 'uuid';\r\nimport { Field } from '../../../services/save-store/models';\r\nimport { defaultInlineStyleConstant } from './config/default-inline-style';\r\n\r\nimport { Jodit } from 'jodit';\r\nimport { JoditConfig } from 'ngx-jodit';\r\n\r\nimport ru from 'jodit/esm/langs/ru.js';\r\n\r\nimport 'jodit/esm/plugins/source/source.js';\r\nimport 'jodit/esm/plugins/line-height/line-height.js';\r\nimport 'jodit/esm/plugins/indent/indent.js';\r\nimport 'jodit/esm/plugins/video/video.js';\r\nimport 'jodit/esm/plugins/search/search.js';\r\nimport 'jodit/esm/plugins/resizer/resizer.js';\r\nimport { InputEditorModeEnum, InputEditorValueWithId } from './editor.enum';\r\n\r\nJodit.lang.de = ru;\r\n\r\n// export interface CustomJoditConfig extends JoditConfig {\r\n//   getIcon: (a: any) => void,\r\n//   link: any,\r\n//   cleanHTML: any,\r\n// }\r\n\r\n@Component({\r\n  selector: 'mrx-editor',\r\n  templateUrl: './editor.component.html',\r\n  styleUrls: ['./editor.component.less'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => EditorComponent),\r\n      multi: true,\r\n    },\r\n  ]\r\n})\r\nexport class EditorComponent implements ControlValueAccessor, OnChanges {\r\n  private _isInit = false;\r\n  public value = '';\r\n  public valueLength = 0;\r\n  public isFocused: boolean | null = false;\r\n\r\n  public options = {\r\n    language: 'ru'\r\n  };\r\n\r\n  public defaultConfig: any = {\r\n    ...defaultToolbar,\r\n    popup: popupToolbars,\r\n    defaultMode: InputEditorModeEnum.MODE_WYSIWYG,\r\n    language: 'ru',\r\n    showCharsCounter: false,\r\n    showWordsCounter: false,\r\n    showXPathInStatusbar: false,\r\n    link: {\r\n      noFollowCheckbox: false\r\n    },\r\n    minHeight: 110,\r\n    iframe: false,\r\n    imageDefaultWidth: 200,\r\n    editHTMLDocumentMode: true,\r\n    cleanHTML: {\r\n      fillEmptyParagraph: true\r\n    },\r\n    placeholder: '',\r\n    toolbarAdaptive: false,\r\n    toolbarInline: true,\r\n    toolbarInlineForSelection: false,\r\n    toolbarInlineDisableFor: [],\r\n    toolbarInlineDisabledButtons: ['source'],\r\n    disabled: false,\r\n    readonly: false,\r\n    askBeforePasteHTML: false,\r\n    defaultActionOnPaste: 'insert_only_text',\r\n    askBeforePasteFromWord: false,\r\n    hidePoweredByJodit: true,\r\n    beautifyHTML: true,\r\n    addNewLine: true,\r\n    createAttributes: {},\r\n    getIcon: (iconName: string) => {\r\n      return changeIconsFunction(iconName);\r\n    },\r\n    events: {\r\n      focus: () => {\r\n        this.changeFocused(true);\r\n      },\r\n      blur: () => {\r\n        this.changeFocused(false);\r\n      },\r\n      paste: (event: ClipboardEvent) => {\r\n        this.handlePaste(event);\r\n      },\r\n    },\r\n    // uploader: {\r\n    //   insertImageAsBase64URI: true\r\n    // },\r\n    limitChars: 10,\r\n    // limitHTML: 170,\r\n  };\r\n\r\n  // SAVE STATE\r\n  public uuid: string = uuidv4();\r\n  @Input() public fields: Field[] = [];\r\n\r\n  @Input() public toolbar!: ToolbarConfig;\r\n  @Input() public maxLength = 0;\r\n  @Input() public minHeight = 110;\r\n  @Input() public customClasses = '';\r\n  @Input() public placeholder = '';\r\n  @Input() public disabled = false;\r\n  @Input() public readonly = false;\r\n  @Input() public iframe = false;\r\n  @Input() public defaultInlineStyle = false;\r\n  @Input() public config: any = this.defaultConfig;\r\n\r\n  @Input() public editHTMLDocumentMode = false;\r\n  @Input() public defaultMode: InputEditorModeEnum = InputEditorModeEnum.MODE_WYSIWYG;\r\n\r\n  @Input() public invalid = false;\r\n  @Input() public invalidMessage: string | string[] = '';\r\n  @Input() public checkInvalid: true | false | null = null;\r\n\r\n  @ViewChild('editorElementRef') editorElementRef!: any;\r\n\r\n  @Output() public changed: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() public modelChange: EventEmitter<InputEditorValueWithId> = new EventEmitter<InputEditorValueWithId>();\r\n\r\n  constructor(private changeDetection: ChangeDetectorRef) {\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['toolbar']) {\r\n      this.defaultConfig = {...this.defaultConfig, ...changes['toolbar'].currentValue};\r\n    }\r\n    if (changes['maxLength']) {\r\n      this.defaultConfig = {...this.defaultConfig, limitChars: changes['maxLength'].currentValue};\r\n    }\r\n    if (changes['placeholder']) {\r\n      this.defaultConfig = {...this.defaultConfig, placeholder: changes['placeholder'].currentValue};\r\n    }\r\n    if (changes['disabled']) {\r\n      this.defaultConfig = {...this.defaultConfig, disabled: changes['disabled'].currentValue};\r\n    }\r\n    if (changes['readonly']) {\r\n      this.defaultConfig = {...this.defaultConfig, readonly: changes['readonly'].currentValue};\r\n    }\r\n    if (changes['iframe']) {\r\n      this.defaultConfig = {...this.defaultConfig, iframe: changes['iframe'].currentValue};\r\n    }\r\n    if (changes['defaultInlineStyle']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        createAttributes: {...this.defaultConfig.createAttributes, ...defaultInlineStyleConstant}\r\n      };\r\n    }\r\n    if (changes['editHTMLDocumentMode']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        editHTMLDocumentMode: changes['editHTMLDocumentMode'].currentValue\r\n      };\r\n    }\r\n    if (changes['askBeforePasteHTML']) {\r\n      this.defaultConfig = {...this.defaultConfig, askBeforePasteHTML: changes['askBeforePasteHTML'].currentValue};\r\n    }\r\n    if (changes['askBeforePasteFromWord']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        askBeforePasteFromWord: changes['askBeforePasteFromWord'].currentValue\r\n      };\r\n    }\r\n    if (changes['defaultMode']) {\r\n      this.defaultConfig = {...this.defaultConfig, defaultMode: changes['defaultMode'].currentValue};\r\n    }\r\n    if (changes['defaultActionOnPaste']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        defaultActionOnPaste: changes['defaultActionOnPaste'].currentValue\r\n      };\r\n    }\r\n    if (changes['minHeight']) {\r\n      this.defaultConfig = {...this.defaultConfig, minHeight: changes['minHeight'].currentValue};\r\n    }\r\n    if (changes['config']) {\r\n      this.defaultConfig = {...this.defaultConfig, ...changes['config'].currentValue};\r\n    }\r\n  }\r\n\r\n  public get isValid(): boolean {\r\n    return true;\r\n  }\r\n\r\n  public get isInvalidMessage(): boolean {\r\n    return !!this.invalidMessage || !!this.invalidMessage?.length;\r\n  }\r\n\r\n  public get checkValidClasses(): string {\r\n    return this.checkInvalid === false ? 'mrx-input-checked-success' : this.checkInvalid === true ? 'mrx-input-checked-error' : '';\r\n  }\r\n\r\n  public get getClasses(): string {\r\n    return `${this.customClasses} ${this.checkValidClasses}`;\r\n  }\r\n\r\n  public insertPositionText(text: string) {\r\n    this.editorElementRef.jodit.s.insertHTML(text);\r\n  }\r\n\r\n  public changeFocused(value: boolean): void {\r\n    this.isFocused = value;\r\n    this.changeDetection.detectChanges();\r\n  }\r\n\r\n  public handlePaste(event: ClipboardEvent): void {\r\n    if (!event.clipboardData || this.hasImageButton()) {\r\n      return;\r\n    }\r\n\r\n    const {clipboardData} = event;\r\n    const htmlData = clipboardData.getData('text/html');\r\n    const hasHtmlImage = htmlData?.includes('<img');\r\n    const hasImage = Array.from(clipboardData.items).some(item => item.type.startsWith('image/'));\r\n\r\n    if (!hasImage && !hasHtmlImage) {\r\n      return;\r\n    }\r\n\r\n    event.preventDefault();\r\n\r\n    const sanitizedHtml = this.removeImagesFromHtml(htmlData);\r\n    const textData = clipboardData.getData('text/plain');\r\n    const contentToInsert = sanitizedHtml || textData;\r\n\r\n    if (contentToInsert) {\r\n      this.editorElementRef.jodit.s.insertHTML(contentToInsert);\r\n    }\r\n\r\n  }\r\n\r\n  public removeImagesFromHtml(html: string): string | null {\r\n    if (!html) {\r\n      return null;\r\n    }\r\n\r\n    const div = document.createElement('div');\r\n    div.innerHTML = html;\r\n    div.querySelectorAll('img').forEach(img => img.remove());\r\n\r\n    return div.innerHTML;\r\n  }\r\n\r\n  public hasImageButton(): boolean {\r\n    const buttons = this.defaultConfig?.buttons;\r\n\r\n    if (!buttons) {\r\n      return false;\r\n    }\r\n\r\n    if (typeof buttons === 'string') {\r\n      return buttons.includes('image');\r\n    }\r\n\r\n    if (Array.isArray(buttons)) {\r\n      return buttons.some(item =>\r\n        typeof item === 'string'\r\n          ? item === 'image'\r\n          : item?.name === 'image'\r\n      );\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  private onChange = (value: any) => {\r\n  };\r\n  private onTouched = () => {\r\n  };\r\n\r\n  public registerOnChange(fn: any) {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  public registerOnTouched(fn: () => {}): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  public writeValue(outsideValue: any) {\r\n    this.value = this._sanitizeValue(outsideValue);\r\n    if (outsideValue !== null) {\r\n      setTimeout(() => {\r\n        this._isInit = true;\r\n      }, 100);\r\n    }\r\n  }\r\n\r\n  public updateValue(insideValue: string) {\r\n\r\n    if (this.maxLength) {\r\n      this._calculateChartsCount();\r\n    }\r\n    if (this._isInit) {\r\n      this.value = insideValue;\r\n      this.changed.emit(insideValue);\r\n      this.modelChange.emit({value: insideValue, id: this.uuid});\r\n      this.onChange(insideValue);\r\n      this.onTouched();\r\n    }\r\n  }\r\n\r\n  private _sanitizeValue(value: string): string {\r\n    return value ? value.replace(/<script\\b[^<]*(?:(?!<\\/script>)<[^<]*)*<\\/script>/gi, '') : value;\r\n  }\r\n\r\n  private _calculateChartsCount() {\r\n    this.valueLength = this.editorElementRef.editor.text.replace(/\\s+/g, '').length;\r\n  }\r\n}\r\n","<div\r\n  class=\"mrx-editor\"\r\n  [class.mrx-input-focused]=\"isFocused\"\r\n  [class.mrx-input-error]=\"!isValid || invalid\"\r\n  [class]=\"getClasses\"\r\n  [class.-readonly]=\"readonly\"\r\n>\r\n  <ngx-jodit\r\n    #editorElementRef\r\n    [(ngModel)]=\"value\"\r\n    [options]=\"defaultConfig\"\r\n    [disabled]=\"disabled\"\r\n    (ngModelChange)=\"updateValue($event)\"\r\n  ></ngx-jodit>\r\n\r\n  <mrx-error-message\r\n    *ngIf=\"(!isValid || invalid) && isInvalidMessage\"\r\n    [invalidMessage]=\"invalidMessage\"\r\n  ></mrx-error-message>\r\n\r\n  <mrx-chars-left [maxlength]=\"maxLength\" [valueLength]=\"valueLength\"></mrx-chars-left>\r\n\r\n  <mrx-save-state type=\"editor\" [id]=\"uuid\" [fields]=\"fields\"></mrx-save-state>\r\n</div>\r\n"]}
372
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrta-ui/src/lib/components/form/editor/editor.component.ts","../../../../../../../projects/myrta-ui/src/lib/components/form/editor/editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  EventEmitter,\r\n  forwardRef,\r\n  Input,\r\n  OnChanges,\r\n  Output,\r\n  SimpleChanges,\r\n  ViewChild\r\n} from '@angular/core';\r\nimport { changeIconsFunction } from './modules/change-icon-module';\r\nimport { defaultToolbar } from './config';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\r\nimport { popupToolbars } from './config/popups-toolbar';\r\nimport { ToolbarConfig } from './models/toolbar.model';\r\nimport { v4 as uuidv4 } from 'uuid';\r\nimport { Field } from '../../../services/save-store/models';\r\nimport { defaultInlineStyleConstant } from './config/default-inline-style';\r\n\r\nimport { Jodit } from 'jodit';\r\nimport { JoditConfig } from 'ngx-jodit';\r\n\r\nimport ru from 'jodit/esm/langs/ru.js';\r\n\r\nimport 'jodit/esm/plugins/source/source.js';\r\nimport 'jodit/esm/plugins/line-height/line-height.js';\r\nimport 'jodit/esm/plugins/indent/indent.js';\r\nimport 'jodit/esm/plugins/video/video.js';\r\nimport 'jodit/esm/plugins/search/search.js';\r\nimport 'jodit/esm/plugins/resizer/resizer.js';\r\nimport { InputEditorModeEnum, InputEditorValueWithId } from './editor.enum';\r\n\r\nJodit.lang.de = ru;\r\n\r\n// export interface CustomJoditConfig extends JoditConfig {\r\n//   getIcon: (a: any) => void,\r\n//   link: any,\r\n//   cleanHTML: any,\r\n// }\r\n\r\n@Component({\r\n  selector: 'mrx-editor',\r\n  templateUrl: './editor.component.html',\r\n  styleUrls: ['./editor.component.less'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => EditorComponent),\r\n      multi: true,\r\n    },\r\n  ]\r\n})\r\nexport class EditorComponent implements ControlValueAccessor, OnChanges {\r\n  private _isInit = false;\r\n  public value = '';\r\n  public valueLength = 0;\r\n  public isFocused: boolean | null = false;\r\n\r\n  public options = {\r\n    language: 'ru'\r\n  };\r\n\r\n  public defaultConfig: any = {\r\n    ...defaultToolbar,\r\n    popup: popupToolbars,\r\n    defaultMode: InputEditorModeEnum.MODE_WYSIWYG,\r\n    language: 'ru',\r\n    showCharsCounter: false,\r\n    showWordsCounter: false,\r\n    showXPathInStatusbar: false,\r\n    link: {\r\n      noFollowCheckbox: false\r\n    },\r\n    minHeight: 110,\r\n    iframe: false,\r\n    imageDefaultWidth: 200,\r\n    editHTMLDocumentMode: true,\r\n    cleanHTML: {\r\n      fillEmptyParagraph: true\r\n    },\r\n    placeholder: '',\r\n    toolbarAdaptive: false,\r\n    toolbarInline: true,\r\n    toolbarInlineForSelection: false,\r\n    toolbarInlineDisableFor: [],\r\n    toolbarInlineDisabledButtons: ['source'],\r\n    disabled: false,\r\n    readonly: false,\r\n    askBeforePasteHTML: false,\r\n    defaultActionOnPaste: 'insert_only_text',\r\n    askBeforePasteFromWord: false,\r\n    hidePoweredByJodit: true,\r\n    beautifyHTML: true,\r\n    addNewLine: true,\r\n    createAttributes: {},\r\n    getIcon: (iconName: string) => {\r\n      return changeIconsFunction(iconName);\r\n    },\r\n    events: {\r\n      focus: () => {\r\n        this.changeFocused(true);\r\n      },\r\n      blur: () => {\r\n        this.changeFocused(false);\r\n      },\r\n      keydown: (event: KeyboardEvent): boolean | void => {\r\n        const editor = this.editorElementRef?.jodit;\r\n        if (!editor || this.maxLength <= 0) return true;\r\n\r\n        // 1. Получаем текущую длину текста\r\n        const currentLength = editor.text.length;\r\n\r\n        // 2. Список системных клавиш, которые нельзя блокировать\r\n        const isNavigationKey = [\r\n          'Backspace', 'Delete', 'ArrowLeft', 'ArrowRight', \r\n          'ArrowUp', 'ArrowDown', 'End', 'Home', 'Tab', 'Enter'\r\n        ].includes(event.key);\r\n        \r\n        // Разрешаем сочетания с Ctrl/Cmd (Copy, Paste, Select All)\r\n        const isModifier = event.ctrlKey || event.metaKey;\r\n\r\n        if (currentLength >= this.maxLength && !isNavigationKey && !isModifier) {\r\n          // 3. Самый надежный способ проверить, выделено ли что-то:\r\n          // Если метод sel.isCollapsed недоступен, проверяем длину выделения через строку\r\n          const selectedText = editor.s.sel ? editor.s.sel.toString() : '';\r\n\r\n          // Если ничего не выделено, блокируем ввод нового символа\r\n          if (!selectedText.length) {\r\n            event.preventDefault();\r\n            return false;\r\n          }\r\n        }\r\n\r\n        return true;\r\n      },\r\n\r\n      paste: (event: ClipboardEvent) => {\r\n        this.handlePaste(event);\r\n      },\r\n    },\r\n    // uploader: {\r\n    //   insertImageAsBase64URI: true\r\n    // },\r\n    limitChars: 10,\r\n    // limitHTML: 170,\r\n  };\r\n\r\n  // SAVE STATE\r\n  public uuid: string = uuidv4();\r\n  @Input() public fields: Field[] = [];\r\n\r\n  @Input() public toolbar!: ToolbarConfig;\r\n  @Input() public maxLength = 0;\r\n  @Input() public minHeight = 110;\r\n  @Input() public customClasses = '';\r\n  @Input() public placeholder = '';\r\n  @Input() public disabled = false;\r\n  @Input() public readonly = false;\r\n  @Input() public iframe = false;\r\n  @Input() public defaultInlineStyle = false;\r\n  @Input() public config: any = this.defaultConfig;\r\n\r\n  @Input() public editHTMLDocumentMode = false;\r\n  @Input() public defaultMode: InputEditorModeEnum = InputEditorModeEnum.MODE_WYSIWYG;\r\n\r\n  @Input() public invalid = false;\r\n  @Input() public invalidMessage: string | string[] = '';\r\n  @Input() public checkInvalid: true | false | null = null;\r\n\r\n  @ViewChild('editorElementRef') editorElementRef!: any;\r\n\r\n  @Output() public changed: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() public modelChange: EventEmitter<InputEditorValueWithId> = new EventEmitter<InputEditorValueWithId>();\r\n\r\n  constructor(private changeDetection: ChangeDetectorRef) {\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['toolbar']) {\r\n      this.defaultConfig = {...this.defaultConfig, ...changes['toolbar'].currentValue};\r\n    }\r\n    if (changes['maxLength']) {\r\n      this.defaultConfig = {...this.defaultConfig, limitChars: changes['maxLength'].currentValue};\r\n    }\r\n    if (changes['placeholder']) {\r\n      this.defaultConfig = {...this.defaultConfig, placeholder: changes['placeholder'].currentValue};\r\n    }\r\n    if (changes['disabled']) {\r\n      this.defaultConfig = {...this.defaultConfig, disabled: changes['disabled'].currentValue};\r\n    }\r\n    if (changes['readonly']) {\r\n      this.defaultConfig = {...this.defaultConfig, readonly: changes['readonly'].currentValue};\r\n    }\r\n    if (changes['iframe']) {\r\n      this.defaultConfig = {...this.defaultConfig, iframe: changes['iframe'].currentValue};\r\n    }\r\n    if (changes['defaultInlineStyle']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        createAttributes: {...this.defaultConfig.createAttributes, ...defaultInlineStyleConstant}\r\n      };\r\n    }\r\n    if (changes['editHTMLDocumentMode']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        editHTMLDocumentMode: changes['editHTMLDocumentMode'].currentValue\r\n      };\r\n    }\r\n    if (changes['askBeforePasteHTML']) {\r\n      this.defaultConfig = {...this.defaultConfig, askBeforePasteHTML: changes['askBeforePasteHTML'].currentValue};\r\n    }\r\n    if (changes['askBeforePasteFromWord']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        askBeforePasteFromWord: changes['askBeforePasteFromWord'].currentValue\r\n      };\r\n    }\r\n    if (changes['defaultMode']) {\r\n      this.defaultConfig = {...this.defaultConfig, defaultMode: changes['defaultMode'].currentValue};\r\n    }\r\n    if (changes['defaultActionOnPaste']) {\r\n      this.defaultConfig = {\r\n        ...this.defaultConfig,\r\n        defaultActionOnPaste: changes['defaultActionOnPaste'].currentValue\r\n      };\r\n    }\r\n    if (changes['minHeight']) {\r\n      this.defaultConfig = {...this.defaultConfig, minHeight: changes['minHeight'].currentValue};\r\n    }\r\n    if (changes['config']) {\r\n      this.defaultConfig = {...this.defaultConfig, ...changes['config'].currentValue};\r\n    }\r\n  }\r\n\r\n  public get isValid(): boolean {\r\n    return true;\r\n  }\r\n\r\n  public get isInvalidMessage(): boolean {\r\n    return !!this.invalidMessage || !!this.invalidMessage?.length;\r\n  }\r\n\r\n  public get checkValidClasses(): string {\r\n    return this.checkInvalid === false ? 'mrx-input-checked-success' : this.checkInvalid === true ? 'mrx-input-checked-error' : '';\r\n  }\r\n\r\n  public get getClasses(): string {\r\n    return `${this.customClasses} ${this.checkValidClasses}`;\r\n  }\r\n\r\n  public insertPositionText(text: string) {\r\n    this.editorElementRef.jodit.s.insertHTML(text);\r\n  }\r\n\r\n  public changeFocused(value: boolean): void {\r\n    this.isFocused = value;\r\n    this.changeDetection.detectChanges();\r\n  }\r\n\r\n  public handlePaste(event: ClipboardEvent): void {\r\n    const editor = this.editorElementRef?.jodit;\r\n    if (!event.clipboardData || !editor) return;\r\n\r\n    const textData = event.clipboardData.getData('text/plain');\r\n    const currentLength = editor.text.length;\r\n\r\n    // Если вставка превышает лимит\r\n    if (this.maxLength > 0 && (currentLength + textData.length) > this.maxLength) {\r\n      event.preventDefault();\r\n      const remainingSpace = this.maxLength - currentLength;\r\n\r\n      if (remainingSpace > 0) {\r\n        // Вставляем только разрешенный кусочек\r\n        const truncatedText = textData.substring(0, remainingSpace);\r\n        editor.s.insertHTML(truncatedText);\r\n      }\r\n      return; // Прекращаем выполнение, чтобы не сработала логика картинок ниже\r\n    }\r\n\r\n    // Ваша существующая логика защиты от картинок\r\n    if (this.hasImageButton()) return;\r\n    \r\n    const htmlData = event.clipboardData.getData('text/html');\r\n    const hasHtmlImage = htmlData?.includes('<img');\r\n    const hasImage = Array.from(event.clipboardData.items).some(item => item.type.startsWith('image/'));\r\n\r\n    if (hasImage || hasHtmlImage) {\r\n      event.preventDefault();\r\n      const sanitizedHtml = this.removeImagesFromHtml(htmlData);\r\n      const contentToInsert = sanitizedHtml || textData;\r\n      if (contentToInsert) {\r\n        editor.s.insertHTML(contentToInsert);\r\n      }\r\n    }\r\n  }\r\n\r\n  public removeImagesFromHtml(html: string): string | null {\r\n    if (!html) {\r\n      return null;\r\n    }\r\n\r\n    const div = document.createElement('div');\r\n    div.innerHTML = html;\r\n    div.querySelectorAll('img').forEach(img => img.remove());\r\n\r\n    return div.innerHTML;\r\n  }\r\n\r\n  public hasImageButton(): boolean {\r\n    const buttons = this.defaultConfig?.buttons;\r\n\r\n    if (!buttons) {\r\n      return false;\r\n    }\r\n\r\n    if (typeof buttons === 'string') {\r\n      return buttons.includes('image');\r\n    }\r\n\r\n    if (Array.isArray(buttons)) {\r\n      return buttons.some(item =>\r\n        typeof item === 'string'\r\n          ? item === 'image'\r\n          : item?.name === 'image'\r\n      );\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  private onChange = (value: any) => {\r\n  };\r\n  private onTouched = () => {\r\n  };\r\n\r\n  public registerOnChange(fn: any) {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  public registerOnTouched(fn: () => {}): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  public writeValue(outsideValue: any) {\r\n    this.value = this._sanitizeValue(outsideValue);\r\n    if (outsideValue !== null) {\r\n      setTimeout(() => {\r\n        this._isInit = true;\r\n      }, 100);\r\n    }\r\n  }\r\n\r\n  public updateValue(insideValue: string) {\r\n    // Сначала считаем, потом проверяем инициализацию\r\n    if (this.maxLength) {\r\n      this._calculateChartsCount();\r\n    }\r\n    \r\n    if (this._isInit) {\r\n      this.value = insideValue;\r\n      this.changed.emit(insideValue);\r\n      this.modelChange.emit({value: insideValue, id: this.uuid});\r\n      this.onChange(insideValue);\r\n      this.onTouched();\r\n    }\r\n  }\r\n\r\n  private _sanitizeValue(value: string): string {\r\n    return value ? value.replace(/<script\\b[^<]*(?:(?!<\\/script>)<[^<]*)*<\\/script>/gi, '') : value;\r\n  }\r\n\r\n  private _calculateChartsCount() {\r\n    // В Jodit правильно считать длину через .text, \r\n    // но в ngx-jodit инстанс доступен через .jodit\r\n    if (this.editorElementRef?.jodit) {\r\n      this.valueLength = this.editorElementRef.jodit.text.length;\r\n      this.changeDetection.detectChanges(); // Форсируем обновление UI для счетчика\r\n    }\r\n  }\r\n}\r\n","<div\r\n  class=\"mrx-editor\"\r\n  [class.mrx-input-focused]=\"isFocused\"\r\n  [class.mrx-input-error]=\"!isValid || invalid\"\r\n  [class]=\"getClasses\"\r\n  [class.-readonly]=\"readonly\"\r\n>\r\n  <ngx-jodit\r\n    #editorElementRef\r\n    [(ngModel)]=\"value\"\r\n    [options]=\"defaultConfig\"\r\n    [disabled]=\"disabled\"\r\n    (ngModelChange)=\"updateValue($event)\"\r\n  ></ngx-jodit>\r\n\r\n  <mrx-error-message\r\n    *ngIf=\"(!isValid || invalid) && isInvalidMessage\"\r\n    [invalidMessage]=\"invalidMessage\"\r\n  ></mrx-error-message>\r\n\r\n  <mrx-chars-left [maxlength]=\"maxLength\" [valueLength]=\"valueLength\"></mrx-chars-left>\r\n\r\n  <mrx-save-state type=\"editor\" [id]=\"uuid\" [fields]=\"fields\"></mrx-save-state>\r\n</div>\r\n"]}
@@ -11621,6 +11621,31 @@ class EditorComponent {
11621
11621
  blur: () => {
11622
11622
  this.changeFocused(false);
11623
11623
  },
11624
+ keydown: (event) => {
11625
+ const editor = this.editorElementRef?.jodit;
11626
+ if (!editor || this.maxLength <= 0)
11627
+ return true;
11628
+ // 1. Получаем текущую длину текста
11629
+ const currentLength = editor.text.length;
11630
+ // 2. Список системных клавиш, которые нельзя блокировать
11631
+ const isNavigationKey = [
11632
+ 'Backspace', 'Delete', 'ArrowLeft', 'ArrowRight',
11633
+ 'ArrowUp', 'ArrowDown', 'End', 'Home', 'Tab', 'Enter'
11634
+ ].includes(event.key);
11635
+ // Разрешаем сочетания с Ctrl/Cmd (Copy, Paste, Select All)
11636
+ const isModifier = event.ctrlKey || event.metaKey;
11637
+ if (currentLength >= this.maxLength && !isNavigationKey && !isModifier) {
11638
+ // 3. Самый надежный способ проверить, выделено ли что-то:
11639
+ // Если метод sel.isCollapsed недоступен, проверяем длину выделения через строку
11640
+ const selectedText = editor.s.sel ? editor.s.sel.toString() : '';
11641
+ // Если ничего не выделено, блокируем ввод нового символа
11642
+ if (!selectedText.length) {
11643
+ event.preventDefault();
11644
+ return false;
11645
+ }
11646
+ }
11647
+ return true;
11648
+ },
11624
11649
  paste: (event) => {
11625
11650
  this.handlePaste(event);
11626
11651
  },
@@ -11731,22 +11756,35 @@ class EditorComponent {
11731
11756
  this.changeDetection.detectChanges();
11732
11757
  }
11733
11758
  handlePaste(event) {
11734
- if (!event.clipboardData || this.hasImageButton()) {
11759
+ const editor = this.editorElementRef?.jodit;
11760
+ if (!event.clipboardData || !editor)
11735
11761
  return;
11762
+ const textData = event.clipboardData.getData('text/plain');
11763
+ const currentLength = editor.text.length;
11764
+ // Если вставка превышает лимит
11765
+ if (this.maxLength > 0 && (currentLength + textData.length) > this.maxLength) {
11766
+ event.preventDefault();
11767
+ const remainingSpace = this.maxLength - currentLength;
11768
+ if (remainingSpace > 0) {
11769
+ // Вставляем только разрешенный кусочек
11770
+ const truncatedText = textData.substring(0, remainingSpace);
11771
+ editor.s.insertHTML(truncatedText);
11772
+ }
11773
+ return; // Прекращаем выполнение, чтобы не сработала логика картинок ниже
11736
11774
  }
11737
- const { clipboardData } = event;
11738
- const htmlData = clipboardData.getData('text/html');
11739
- const hasHtmlImage = htmlData?.includes('<img');
11740
- const hasImage = Array.from(clipboardData.items).some(item => item.type.startsWith('image/'));
11741
- if (!hasImage && !hasHtmlImage) {
11775
+ // Ваша существующая логика защиты от картинок
11776
+ if (this.hasImageButton())
11742
11777
  return;
11743
- }
11744
- event.preventDefault();
11745
- const sanitizedHtml = this.removeImagesFromHtml(htmlData);
11746
- const textData = clipboardData.getData('text/plain');
11747
- const contentToInsert = sanitizedHtml || textData;
11748
- if (contentToInsert) {
11749
- this.editorElementRef.jodit.s.insertHTML(contentToInsert);
11778
+ const htmlData = event.clipboardData.getData('text/html');
11779
+ const hasHtmlImage = htmlData?.includes('<img');
11780
+ const hasImage = Array.from(event.clipboardData.items).some(item => item.type.startsWith('image/'));
11781
+ if (hasImage || hasHtmlImage) {
11782
+ event.preventDefault();
11783
+ const sanitizedHtml = this.removeImagesFromHtml(htmlData);
11784
+ const contentToInsert = sanitizedHtml || textData;
11785
+ if (contentToInsert) {
11786
+ editor.s.insertHTML(contentToInsert);
11787
+ }
11750
11788
  }
11751
11789
  }
11752
11790
  removeImagesFromHtml(html) {
@@ -11792,6 +11830,7 @@ class EditorComponent {
11792
11830
  }
11793
11831
  }
11794
11832
  updateValue(insideValue) {
11833
+ // Сначала считаем, потом проверяем инициализацию
11795
11834
  if (this.maxLength) {
11796
11835
  this._calculateChartsCount();
11797
11836
  }
@@ -11807,7 +11846,12 @@ class EditorComponent {
11807
11846
  return value ? value.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, '') : value;
11808
11847
  }
11809
11848
  _calculateChartsCount() {
11810
- this.valueLength = this.editorElementRef.editor.text.replace(/\s+/g, '').length;
11849
+ // В Jodit правильно считать длину через .text,
11850
+ // но в ngx-jodit инстанс доступен через .jodit
11851
+ if (this.editorElementRef?.jodit) {
11852
+ this.valueLength = this.editorElementRef.jodit.text.length;
11853
+ this.changeDetection.detectChanges(); // Форсируем обновление UI для счетчика
11854
+ }
11811
11855
  }
11812
11856
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: EditorComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
11813
11857
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: EditorComponent, selector: "mrx-editor", inputs: { fields: "fields", toolbar: "toolbar", maxLength: "maxLength", minHeight: "minHeight", customClasses: "customClasses", placeholder: "placeholder", disabled: "disabled", readonly: "readonly", iframe: "iframe", defaultInlineStyle: "defaultInlineStyle", config: "config", editHTMLDocumentMode: "editHTMLDocumentMode", defaultMode: "defaultMode", invalid: "invalid", invalidMessage: "invalidMessage", checkInvalid: "checkInvalid" }, outputs: { changed: "changed", modelChange: "modelChange" }, providers: [