adb-shared 6.0.6 → 6.0.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.
@@ -22,25 +22,25 @@ export class AdbRichEditorComponent {
22
22
  textarea.setSelectionRange(selectionStart, selectionEnd);
23
23
  }
24
24
  onItalicClick() {
25
- const textarea = this.el.nativeElement.querySelector('textarea');
26
- if (textarea) {
27
- const selectionStart = textarea.selectionStart;
28
- const selectionEnd = textarea.selectionEnd;
29
- let currentValue = textarea.value;
30
- let modifiedText = `${currentValue.substring(0, selectionStart)}_${currentValue.substring(selectionStart, selectionEnd)}_${currentValue.substring(selectionEnd)}`;
31
- this.text = modifiedText;
32
- this.onChange(this.text);
33
- }
25
+ this.applyFormatting('_');
34
26
  }
35
27
  onBoldClick() {
28
+ this.applyFormatting('__');
29
+ }
30
+ applyFormatting(marker) {
36
31
  const textarea = this.el.nativeElement.querySelector('textarea');
37
32
  if (textarea) {
38
33
  const selectionStart = textarea.selectionStart;
39
34
  const selectionEnd = textarea.selectionEnd;
40
35
  let currentValue = textarea.value;
41
- let modifiedText = `${currentValue.substring(0, selectionStart)}__${currentValue.substring(selectionStart, selectionEnd)}__${currentValue.substring(selectionEnd)}`;
36
+ let modifiedText = selectionStart !== selectionEnd
37
+ ? `${currentValue.substring(0, selectionStart)}${marker}${currentValue.substring(selectionStart, selectionEnd)}${marker}${currentValue.substring(selectionEnd)}`
38
+ : `${currentValue.substring(0, selectionStart)}${marker}${marker}${currentValue.substring(selectionStart)}`;
42
39
  this.text = modifiedText;
40
+ textarea.value = modifiedText;
41
+ textarea.setSelectionRange(selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length, selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length);
43
42
  this.onChange(this.text);
43
+ textarea.focus();
44
44
  }
45
45
  }
46
46
  onTaxonClick() {
@@ -106,4 +106,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
106
106
  }], hasReference: [{
107
107
  type: Input
108
108
  }] } });
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adb-rich-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/artdata-shared/src/lib/components/adb-rich-editor/adb-rich-editor.component.ts","../../../../../../projects/artdata-shared/src/lib/components/adb-rich-editor/adb-rich-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, forwardRef, Input, OnDestroy, Renderer2 } from \"@angular/core\";\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from \"@angular/forms\";\r\n\r\n@Component({\r\n    selector: 'adb-rich-editor',\r\n    templateUrl: './adb-rich-editor.component.html',\r\n    providers: [{\r\n        provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => AdbRichEditorComponent),\r\n        multi: true\r\n    }]\r\n})\r\nexport class AdbRichEditorComponent implements ControlValueAccessor, OnDestroy {\r\n    text: string;\r\n    @Input() hasTaxon = false;\r\n    @Input() hasReference = false;\r\n\r\n\r\n    constructor(private el: ElementRef) { }\r\n\r\n\r\n    onDoubleClick() {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        const selectionStart = textarea.selectionStart;\r\n        let selectionEnd = textarea.selectionEnd;\r\n        while (selectionEnd > selectionStart && textarea.value[selectionEnd - 1] === ' ') {\r\n            selectionEnd--;\r\n        }\r\n        textarea.setSelectionRange(selectionStart, selectionEnd);\r\n    }\r\n\r\n    onItalicClick(): void {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        if (textarea) {\r\n            const selectionStart = textarea.selectionStart;\r\n            const selectionEnd = textarea.selectionEnd;\r\n            let currentValue = textarea.value;\r\n            let modifiedText = `${currentValue.substring(0, selectionStart)}_${currentValue.substring(selectionStart, selectionEnd)}_${currentValue.substring(selectionEnd)}`;\r\n            this.text = modifiedText;\r\n            this.onChange(this.text);\r\n        }\r\n    }\r\n\r\n    onBoldClick():void{\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        if (textarea) {\r\n            const selectionStart = textarea.selectionStart;\r\n            const selectionEnd = textarea.selectionEnd;\r\n            let currentValue = textarea.value;\r\n            let modifiedText = `${currentValue.substring(0, selectionStart)}__${currentValue.substring(selectionStart, selectionEnd)}__${currentValue.substring(selectionEnd)}`;\r\n            this.text = modifiedText;\r\n            this.onChange(this.text);\r\n        }\r\n    }\r\n\r\n    onTaxonClick(): void {\r\n        this.replaceMarked('[Name](taxon:0)');\r\n    }\r\n\r\n    onReferenceClick(): void {\r\n        this.replaceMarked('[Name](reference:0)');\r\n    }\r\n\r\n    private replaceMarked(replaceTag: string) {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        if (textarea) {\r\n            const start = textarea.selectionStart;\r\n            const end = textarea.selectionEnd;\r\n            const originalText = textarea.value;\r\n            const charBefore = start > 0 ? originalText[start - 1] : '';\r\n            const charAfter = end < originalText.length ? originalText[end] : '';\r\n\r\n            const spaceBefore = charBefore && !/\\s/.test(charBefore) ? ' ' : '';\r\n            const spaceAfter = charAfter && !/\\s/.test(charAfter) ? ' ' : '';\r\n\r\n            const newText = originalText.slice(0, start) + spaceBefore + replaceTag + spaceAfter + originalText.slice(end);\r\n            this.text = newText;\r\n            this.onChange(this.text);\r\n            const nameStart = newText.indexOf('Name', start); // Find 'Name' starting from the previous selection start\r\n            if (nameStart !== -1) {\r\n                const nameEnd = nameStart + 'Name'.length;\r\n    \r\n                // Use setTimeout to make sure the textarea is updated before selecting\r\n                setTimeout(() => {\r\n                    textarea.focus();\r\n                    textarea.setSelectionRange(nameStart, nameEnd); // Select the 'Name' string\r\n                }, 0);\r\n            }\r\n        }\r\n    }\r\n\r\n    onTextChange(): void {\r\n        this.onChange(this.text);\r\n    }\r\n\r\n    //ControlValueAccessor\r\n    onChange: any = () => { };\r\n    onTouched: any = () => { };\r\n\r\n    writeValue(value: string): void {\r\n        this.text = value;\r\n    }\r\n\r\n    registerOnChange(fn: any): void {\r\n        this.onChange = fn;\r\n    }\r\n\r\n    registerOnTouched(fn: any): void {\r\n        this.onTouched = fn;\r\n    }\r\n\r\n    setDisabledState?(isDisabled: boolean): void {\r\n\r\n    }\r\n\r\n    ngOnDestroy(): void {\r\n    }\r\n}","<div>\r\n    <div class=\"d-flex justify-content-end gap-3 mb-1\">\r\n        <button class=\"btn btn-secondary\" (click)=\"onItalicClick()\" title=\"Italic\" aria-label=\"Italic\" type=\"button\"><span class=\"fas fa-italic\"></span></button>\r\n        <button class=\"btn btn-secondary\" (click)=\"onBoldClick()\" title=\"Italic\" aria-label=\"strong\" type=\"button\"><span class=\"fas fa-bold\"></span></button>\r\n        <button *ngIf=\"hasTaxon\" class=\"btn btn-secondary\" (click)=\"onTaxonClick()\" title=\"Taxon\" aria-label=\"Taxon\" type=\"button\"><span class=\"fas fa-bug\"></span></button>\r\n        <button *ngIf=\"hasReference\" class=\"btn btn-secondary\" (click)=\"onReferenceClick()\" title=\"Reference\" aria-label=\"Reference\" type=\"button\"><span class=\"fas fa-asterisk\"></span></button>\r\n    </div>\r\n    <textarea class=\"form-control\" [(ngModel)]=\"text\" (ngModelChange)=\"onTextChange()\" (dblclick)=\"onDoubleClick()\"></textarea>\r\n</div>\r\n"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"adb-rich-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/artdata-shared/src/lib/components/adb-rich-editor/adb-rich-editor.component.ts","../../../../../../projects/artdata-shared/src/lib/components/adb-rich-editor/adb-rich-editor.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAiB,SAAS,EAAc,UAAU,EAAE,KAAK,EAAwB,MAAM,eAAe,CAAC;AAC9G,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;;;;AAUzE,MAAM,OAAO,sBAAsB;IAM/B,YAAoB,EAAc;QAAd,OAAE,GAAF,EAAE,CAAY;QAJzB,aAAQ,GAAG,KAAK,CAAC;QACjB,iBAAY,GAAG,KAAK,CAAC;QAuF9B,sBAAsB;QACtB,aAAQ,GAAQ,GAAG,EAAE,GAAG,CAAC,CAAC;QAC1B,cAAS,GAAQ,GAAG,EAAE,GAAG,CAAC,CAAC;IAtFW,CAAC;IAGvC,aAAa;QACT,MAAM,QAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QACjE,MAAM,cAAc,GAAG,QAAQ,CAAC,cAAc,CAAC;QAC/C,IAAI,YAAY,GAAG,QAAQ,CAAC,YAAY,CAAC;QACzC,OAAO,YAAY,GAAG,cAAc,IAAI,QAAQ,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,KAAK,GAAG,EAAE,CAAC;YAC/E,YAAY,EAAE,CAAC;QACnB,CAAC;QACD,QAAQ,CAAC,iBAAiB,CAAC,cAAc,EAAE,YAAY,CAAC,CAAC;IAC7D,CAAC;IAED,aAAa;QACT,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IAC9B,CAAC;IAED,WAAW;QACP,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;IAC/B,CAAC;IAEO,eAAe,CAAC,MAAc;QAClC,MAAM,QAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QACjE,IAAI,QAAQ,EAAE,CAAC;YACX,MAAM,cAAc,GAAG,QAAQ,CAAC,cAAc,CAAC;YAC/C,MAAM,YAAY,GAAG,QAAQ,CAAC,YAAY,CAAC;YAC3C,IAAI,YAAY,GAAG,QAAQ,CAAC,KAAK,CAAC;YAElC,IAAI,YAAY,GAAG,cAAc,KAAK,YAAY;gBAC9C,CAAC,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC,EAAE,cAAc,CAAC,GAAG,MAAM,GAAG,YAAY,CAAC,SAAS,CAAC,cAAc,EAAE,YAAY,CAAC,GAAG,MAAM,GAAG,YAAY,CAAC,SAAS,CAAC,YAAY,CAAC,EAAE;gBAChK,CAAC,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC,EAAE,cAAc,CAAC,GAAG,MAAM,GAAG,MAAM,GAAG,YAAY,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE,CAAC;YAEhH,IAAI,CAAC,IAAI,GAAG,YAAY,CAAC;YACzB,QAAQ,CAAC,KAAK,GAAG,YAAY,CAAC;YAC9B,QAAQ,CAAC,iBAAiB,CACtB,cAAc,KAAK,YAAY,CAAC,CAAC,CAAC,YAAY,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,MAAM,CAAC,MAAM,EACnG,cAAc,KAAK,YAAY,CAAC,CAAC,CAAC,YAAY,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,cAAc,GAAG,MAAM,CAAC,MAAM,CACtG,CAAC;YAEF,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACzB,QAAQ,CAAC,KAAK,EAAE,CAAC;QACrB,CAAC;IACL,CAAC;IAED,YAAY;QACR,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC,CAAC;IAC1C,CAAC;IAED,gBAAgB;QACZ,IAAI,CAAC,aAAa,CAAC,qBAAqB,CAAC,CAAC;IAC9C,CAAC;IAEO,aAAa,CAAC,UAAkB;QACpC,MAAM,QAAQ,GAAG,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;QACjE,IAAI,QAAQ,EAAE,CAAC;YACX,MAAM,KAAK,GAAG,QAAQ,CAAC,cAAc,CAAC;YACtC,MAAM,GAAG,GAAG,QAAQ,CAAC,YAAY,CAAC;YAClC,MAAM,YAAY,GAAG,QAAQ,CAAC,KAAK,CAAC;YACpC,MAAM,UAAU,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YAC5D,MAAM,SAAS,GAAG,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YAErE,MAAM,WAAW,GAAG,UAAU,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YACpE,MAAM,UAAU,GAAG,SAAS,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YAEjE,MAAM,OAAO,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,GAAG,WAAW,GAAG,UAAU,GAAG,UAAU,GAAG,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC/G,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACzB,MAAM,SAAS,GAAG,OAAO,CAAC,OAAO,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,yDAAyD;YAC3G,IAAI,SAAS,KAAK,CAAC,CAAC,EAAE,CAAC;gBACnB,MAAM,OAAO,GAAG,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC;gBAE1C,uEAAuE;gBACvE,UAAU,CAAC,GAAG,EAAE;oBACZ,QAAQ,CAAC,KAAK,EAAE,CAAC;oBACjB,QAAQ,CAAC,iBAAiB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,2BAA2B;gBAC/E,CAAC,EAAE,CAAC,CAAC,CAAC;YACV,CAAC;QACL,CAAC;IACL,CAAC;IAED,YAAY;QACR,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC7B,CAAC;IAMD,UAAU,CAAC,KAAa;QACpB,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;IACtB,CAAC;IAED,gBAAgB,CAAC,EAAO;QACpB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACvB,CAAC;IAED,iBAAiB,CAAC,EAAO;QACrB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACxB,CAAC;IAED,gBAAgB,CAAE,UAAmB;IAErC,CAAC;IAED,WAAW;IACX,CAAC;iIA/GQ,sBAAsB;qHAAtB,sBAAsB,0GALpB,CAAC;gBACR,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,UAAU,EAAC,GAAG,EAAE,CAAC,sBAAsB,EAAC;gBACjF,KAAK,EAAE,IAAI;aACd,CAAC,0BCTN,09BASA;;2FDEa,sBAAsB;kBARlC,SAAS;+BACI,iBAAiB,aAEhB,CAAC;4BACR,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,UAAU,EAAC,GAAG,EAAE,uBAAuB,EAAC;4BACjF,KAAK,EAAE,IAAI;yBACd,CAAC;+EAIO,QAAQ;sBAAhB,KAAK;gBACG,YAAY;sBAApB,KAAK","sourcesContent":["import { AfterViewInit, Component, ElementRef, forwardRef, Input, OnDestroy, Renderer2 } from \"@angular/core\";\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from \"@angular/forms\";\r\n\r\n@Component({\r\n    selector: 'adb-rich-editor',\r\n    templateUrl: './adb-rich-editor.component.html',\r\n    providers: [{\r\n        provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => AdbRichEditorComponent),\r\n        multi: true\r\n    }]\r\n})\r\nexport class AdbRichEditorComponent implements ControlValueAccessor, OnDestroy {\r\n    text: string;\r\n    @Input() hasTaxon = false;\r\n    @Input() hasReference = false;\r\n\r\n\r\n    constructor(private el: ElementRef) { }\r\n\r\n\r\n    onDoubleClick() {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        const selectionStart = textarea.selectionStart;\r\n        let selectionEnd = textarea.selectionEnd;\r\n        while (selectionEnd > selectionStart && textarea.value[selectionEnd - 1] === ' ') {\r\n            selectionEnd--;\r\n        }\r\n        textarea.setSelectionRange(selectionStart, selectionEnd);\r\n    }\r\n\r\n    onItalicClick(): void {\r\n        this.applyFormatting('_');\r\n    }\r\n    \r\n    onBoldClick(): void {\r\n        this.applyFormatting('__');\r\n    }\r\n\r\n    private applyFormatting(marker: string): void {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        if (textarea) {\r\n            const selectionStart = textarea.selectionStart;\r\n            const selectionEnd = textarea.selectionEnd;\r\n            let currentValue = textarea.value;\r\n    \r\n            let modifiedText = selectionStart !== selectionEnd\r\n                ? `${currentValue.substring(0, selectionStart)}${marker}${currentValue.substring(selectionStart, selectionEnd)}${marker}${currentValue.substring(selectionEnd)}`\r\n                : `${currentValue.substring(0, selectionStart)}${marker}${marker}${currentValue.substring(selectionStart)}`;\r\n    \r\n            this.text = modifiedText;\r\n            textarea.value = modifiedText;\r\n            textarea.setSelectionRange(\r\n                selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length,\r\n                selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length\r\n            );\r\n    \r\n            this.onChange(this.text);\r\n            textarea.focus();\r\n        }\r\n    }\r\n\r\n    onTaxonClick(): void {\r\n        this.replaceMarked('[Name](taxon:0)');\r\n    }\r\n\r\n    onReferenceClick(): void {\r\n        this.replaceMarked('[Name](reference:0)');\r\n    }\r\n\r\n    private replaceMarked(replaceTag: string) {\r\n        const textarea = this.el.nativeElement.querySelector('textarea');\r\n        if (textarea) {\r\n            const start = textarea.selectionStart;\r\n            const end = textarea.selectionEnd;\r\n            const originalText = textarea.value;\r\n            const charBefore = start > 0 ? originalText[start - 1] : '';\r\n            const charAfter = end < originalText.length ? originalText[end] : '';\r\n\r\n            const spaceBefore = charBefore && !/\\s/.test(charBefore) ? ' ' : '';\r\n            const spaceAfter = charAfter && !/\\s/.test(charAfter) ? ' ' : '';\r\n\r\n            const newText = originalText.slice(0, start) + spaceBefore + replaceTag + spaceAfter + originalText.slice(end);\r\n            this.text = newText;\r\n            this.onChange(this.text);\r\n            const nameStart = newText.indexOf('Name', start); // Find 'Name' starting from the previous selection start\r\n            if (nameStart !== -1) {\r\n                const nameEnd = nameStart + 'Name'.length;\r\n\r\n                // Use setTimeout to make sure the textarea is updated before selecting\r\n                setTimeout(() => {\r\n                    textarea.focus();\r\n                    textarea.setSelectionRange(nameStart, nameEnd); // Select the 'Name' string\r\n                }, 0);\r\n            }\r\n        }\r\n    }\r\n\r\n    onTextChange(): void {\r\n        this.onChange(this.text);\r\n    }\r\n\r\n    //ControlValueAccessor\r\n    onChange: any = () => { };\r\n    onTouched: any = () => { };\r\n\r\n    writeValue(value: string): void {\r\n        this.text = value;\r\n    }\r\n\r\n    registerOnChange(fn: any): void {\r\n        this.onChange = fn;\r\n    }\r\n\r\n    registerOnTouched(fn: any): void {\r\n        this.onTouched = fn;\r\n    }\r\n\r\n    setDisabledState?(isDisabled: boolean): void {\r\n\r\n    }\r\n\r\n    ngOnDestroy(): void {\r\n    }\r\n}","<div>\r\n    <div class=\"d-flex justify-content-end gap-3 mb-1\">\r\n        <button class=\"btn btn-secondary\" (click)=\"onItalicClick()\" title=\"Italic\" aria-label=\"Italic\" type=\"button\"><span class=\"fas fa-italic\"></span></button>\r\n        <button class=\"btn btn-secondary\" (click)=\"onBoldClick()\" title=\"Italic\" aria-label=\"strong\" type=\"button\"><span class=\"fas fa-bold\"></span></button>\r\n        <button *ngIf=\"hasTaxon\" class=\"btn btn-secondary\" (click)=\"onTaxonClick()\" title=\"Taxon\" aria-label=\"Taxon\" type=\"button\"><span class=\"fas fa-bug\"></span></button>\r\n        <button *ngIf=\"hasReference\" class=\"btn btn-secondary\" (click)=\"onReferenceClick()\" title=\"Reference\" aria-label=\"Reference\" type=\"button\"><span class=\"fas fa-asterisk\"></span></button>\r\n    </div>\r\n    <textarea class=\"form-control\" [(ngModel)]=\"text\" (ngModelChange)=\"onTextChange()\" (dblclick)=\"onDoubleClick()\"></textarea>\r\n</div>\r\n"]}
@@ -5,7 +5,7 @@ export class EmptyValuePipe {
5
5
  if (!value) {
6
6
  return '-';
7
7
  }
8
- if (value.trim() === '') {
8
+ if (String(value).trim() === '') {
9
9
  return '-';
10
10
  }
11
11
  return value;
@@ -19,4 +19,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
19
19
  name: 'adbEmptyValue'
20
20
  }]
21
21
  }] });
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW1wdHlWYWx1ZS5waXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXJ0ZGF0YS1zaGFyZWQvc3JjL2xpYi9waXBlcy9lbXB0eVZhbHVlLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7O0FBSXBELE1BQU0sT0FBTyxjQUFjO0lBQ3ZCLFNBQVMsQ0FBQyxLQUFVO1FBQ2hCLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNULE9BQU8sR0FBRyxDQUFDO1FBQ2YsQ0FBQztRQUNELElBQUksS0FBSyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsRUFBRSxDQUFDO1lBQ3RCLE9BQU8sR0FBRyxDQUFDO1FBQ2YsQ0FBQztRQUNELE9BQU8sS0FBSyxDQUFDO0lBQ2pCLENBQUM7aUlBVFEsY0FBYzsrSEFBZCxjQUFjOzsyRkFBZCxjQUFjO2tCQUgxQixJQUFJO21CQUFDO29CQUNGLElBQUksRUFBRSxlQUFlO2lCQUN4QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBpcGUsIFBpcGVUcmFuc2Zvcm0gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuQFBpcGUoe1xyXG4gICAgbmFtZTogJ2FkYkVtcHR5VmFsdWUnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBFbXB0eVZhbHVlUGlwZSBpbXBsZW1lbnRzIFBpcGVUcmFuc2Zvcm0ge1xyXG4gICAgdHJhbnNmb3JtKHZhbHVlOiBhbnkpIHtcclxuICAgICAgICBpZiAoIXZhbHVlKSB7XHJcbiAgICAgICAgICAgIHJldHVybiAnLSc7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmICh2YWx1ZS50cmltKCkgPT09ICcnKSB7XHJcbiAgICAgICAgICAgIHJldHVybiAnLSc7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJldHVybiB2YWx1ZTtcclxuICAgIH1cclxufVxyXG4iXX0=
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW1wdHlWYWx1ZS5waXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXJ0ZGF0YS1zaGFyZWQvc3JjL2xpYi9waXBlcy9lbXB0eVZhbHVlLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7O0FBSXBELE1BQU0sT0FBTyxjQUFjO0lBQ3ZCLFNBQVMsQ0FBQyxLQUFVO1FBQ2hCLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNULE9BQU8sR0FBRyxDQUFDO1FBQ2YsQ0FBQztRQUNELElBQUksTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsRUFBRSxDQUFDO1lBQzlCLE9BQU8sR0FBRyxDQUFDO1FBQ2YsQ0FBQztRQUNELE9BQU8sS0FBSyxDQUFDO0lBQ2pCLENBQUM7aUlBVFEsY0FBYzsrSEFBZCxjQUFjOzsyRkFBZCxjQUFjO2tCQUgxQixJQUFJO21CQUFDO29CQUNGLElBQUksRUFBRSxlQUFlO2lCQUN4QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBpcGUsIFBpcGVUcmFuc2Zvcm0gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuQFBpcGUoe1xyXG4gICAgbmFtZTogJ2FkYkVtcHR5VmFsdWUnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBFbXB0eVZhbHVlUGlwZSBpbXBsZW1lbnRzIFBpcGVUcmFuc2Zvcm0ge1xyXG4gICAgdHJhbnNmb3JtKHZhbHVlOiBhbnkpIHtcclxuICAgICAgICBpZiAoIXZhbHVlKSB7XHJcbiAgICAgICAgICAgIHJldHVybiAnLSc7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmIChTdHJpbmcodmFsdWUpLnRyaW0oKSA9PT0gJycpIHtcclxuICAgICAgICAgICAgcmV0dXJuICctJztcclxuICAgICAgICB9XHJcbiAgICAgICAgcmV0dXJuIHZhbHVlO1xyXG4gICAgfVxyXG59XHJcbiJdfQ==
@@ -1026,7 +1026,7 @@ class EmptyValuePipe {
1026
1026
  if (!value) {
1027
1027
  return '-';
1028
1028
  }
1029
- if (value.trim() === '') {
1029
+ if (String(value).trim() === '') {
1030
1030
  return '-';
1031
1031
  }
1032
1032
  return value;
@@ -1795,25 +1795,25 @@ class AdbRichEditorComponent {
1795
1795
  textarea.setSelectionRange(selectionStart, selectionEnd);
1796
1796
  }
1797
1797
  onItalicClick() {
1798
- const textarea = this.el.nativeElement.querySelector('textarea');
1799
- if (textarea) {
1800
- const selectionStart = textarea.selectionStart;
1801
- const selectionEnd = textarea.selectionEnd;
1802
- let currentValue = textarea.value;
1803
- let modifiedText = `${currentValue.substring(0, selectionStart)}_${currentValue.substring(selectionStart, selectionEnd)}_${currentValue.substring(selectionEnd)}`;
1804
- this.text = modifiedText;
1805
- this.onChange(this.text);
1806
- }
1798
+ this.applyFormatting('_');
1807
1799
  }
1808
1800
  onBoldClick() {
1801
+ this.applyFormatting('__');
1802
+ }
1803
+ applyFormatting(marker) {
1809
1804
  const textarea = this.el.nativeElement.querySelector('textarea');
1810
1805
  if (textarea) {
1811
1806
  const selectionStart = textarea.selectionStart;
1812
1807
  const selectionEnd = textarea.selectionEnd;
1813
1808
  let currentValue = textarea.value;
1814
- let modifiedText = `${currentValue.substring(0, selectionStart)}__${currentValue.substring(selectionStart, selectionEnd)}__${currentValue.substring(selectionEnd)}`;
1809
+ let modifiedText = selectionStart !== selectionEnd
1810
+ ? `${currentValue.substring(0, selectionStart)}${marker}${currentValue.substring(selectionStart, selectionEnd)}${marker}${currentValue.substring(selectionEnd)}`
1811
+ : `${currentValue.substring(0, selectionStart)}${marker}${marker}${currentValue.substring(selectionStart)}`;
1815
1812
  this.text = modifiedText;
1813
+ textarea.value = modifiedText;
1814
+ textarea.setSelectionRange(selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length, selectionStart !== selectionEnd ? selectionEnd + marker.length * 2 : selectionStart + marker.length);
1816
1815
  this.onChange(this.text);
1816
+ textarea.focus();
1817
1817
  }
1818
1818
  }
1819
1819
  onTaxonClick() {