@datarailsshared/datarailsshared 1.5.283 → 1.5.285
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/assets/styles/_styles.scss +4 -0
- package/datarailsshared-datarailsshared-1.5.285.tgz +0 -0
- package/esm2022/lib/dr-chat/dr-chat-form/chat-form.component.mjs +3 -3
- package/esm2022/lib/dr-code-editor/components/code-editor-hint-wrapper.component.mjs +15 -0
- package/esm2022/lib/dr-code-editor/components/dr-codemirror.component.mjs +201 -0
- package/esm2022/lib/dr-code-editor/dr-code-editor.component.mjs +292 -0
- package/esm2022/lib/dr-code-editor/dr-code-editor.module.mjs +42 -0
- package/esm2022/lib/dr-code-editor/models/code-editor-hint.mjs +6 -0
- package/esm2022/lib/utils/dr-shared-utils.mjs +4 -1
- package/esm2022/public-api.mjs +6 -1
- package/fesm2022/datarailsshared-datarailsshared.mjs +600 -67
- package/fesm2022/datarailsshared-datarailsshared.mjs.map +1 -1
- package/lib/dr-code-editor/components/code-editor-hint-wrapper.component.d.ts +7 -0
- package/lib/dr-code-editor/components/dr-codemirror.component.d.ts +47 -0
- package/lib/dr-code-editor/dr-code-editor.component.d.ts +44 -0
- package/lib/dr-code-editor/dr-code-editor.module.d.ts +17 -0
- package/lib/dr-code-editor/models/code-editor-hint.d.ts +14 -0
- package/lib/utils/dr-shared-utils.d.ts +1 -0
- package/package.json +1 -1
- package/public-api.d.ts +5 -0
- package/styles.css +526 -0
- package/datarailsshared-datarailsshared-1.5.283.tgz +0 -0
|
@@ -1,3 +1,7 @@
|
|
|
1
1
|
@import '@angular/material/prebuilt-themes/deeppurple-amber.css';
|
|
2
2
|
@import '@ng-select/ng-select/themes/default.theme.css';
|
|
3
3
|
@import 'ngx-toastr/toastr.css';
|
|
4
|
+
|
|
5
|
+
@import 'codemirror/lib/codemirror.css';
|
|
6
|
+
@import 'codemirror/theme/material.css';
|
|
7
|
+
@import 'codemirror/addon/hint/show-hint.css';
|
|
Binary file
|
|
@@ -131,11 +131,11 @@ export class DrChatFormComponent {
|
|
|
131
131
|
return `height: ${height}px;`;
|
|
132
132
|
}
|
|
133
133
|
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DrChatFormComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
134
|
-
/** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DrChatFormComponent, selector: "dr-chat-form", inputs: { message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply" }, outputs: { send: "send", abort: "abort", inputChange: "inputChange" }, host: { listeners: { "drop": "onDrop($event)", "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)" }, properties: { "class.file-over": "this.fileOver" } }, ngImport: i0, template: "<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n </div>\n <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n </div>\n</div>\n<div class=\"message-row\">\n <div
|
|
134
|
+
/** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DrChatFormComponent, selector: "dr-chat-form", inputs: { message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply" }, outputs: { send: "send", abort: "abort", inputChange: "inputChange" }, host: { listeners: { "drop": "onDrop($event)", "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)" }, properties: { "class.file-over": "this.fileOver" } }, ngImport: i0, template: "<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n </div>\n <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n </div>\n</div>\n<div class=\"message-row\">\n <div\n class=\"message-row__input\"\n [ngClass]=\"{ 'message-row__input--focused': inputFocus, 'message-row__input--filled': !!message?.length }\">\n <textarea\n #textAreaElement\n (focus)=\"inputFocus = true\"\n (blur)=\"inputFocus = false\"\n (mouseenter)=\"inputHover = true\"\n (mouseleave)=\"inputHover = false\"\n [(ngModel)]=\"message\"\n [rows]=\"1\"\n [style]=\"getTextAreaHeight(textAreaElement)\"\n (ngModelChange)=\"onModelChange($event)\"\n type=\"text\"\n placeholder=\"{{ fileOver ? dropFilePlaceholder : messagePlaceholder }}\"\n (keyup.enter)=\"sendMessage()\">\n </textarea>\n <i *ngIf=\"!waitForReply\" (click)=\"sendMessage()\" class=\"dr-icon-notify send-button\"></i>\n <dr-dot-flashing *ngIf=\"waitForReply\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n <dr-button *ngIf=\"waitForReply\" (click)=\"abortMessage()\" theme=\"ghost\" class=\"abort-button\">Stop generating</dr-button>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column;align-items:center;padding:0 24px;margin-top:12px}:host .message-row{display:flex;justify-content:center;width:100%;padding:0 0 21px;max-width:956px}:host .message-row__input{position:relative;display:flex;align-items:center;flex-grow:1;flex-direction:row;height:auto;overflow:visible;min-width:265px;border-radius:24px;background:#dfe0e3 border-box;border:1.5px solid transparent;box-shadow:0 2px 16px -10px #603cff29}:host .message-row__input .send-button{position:absolute;right:4px;top:2.5px;cursor:pointer;font-size:28px;width:68px;border-radius:100px;display:flex;align-items:center;justify-content:center;height:40px;background:#f0f1f4;color:#aeabac}:host .message-row__input--focused{background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input--filled{background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input--filled .send-button{color:#fff;background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input textarea{font-size:14px;color:#333;line-height:19px;flex-grow:1;resize:none;padding:14px 76px 12px 23px;margin:auto;border:none;border-radius:22.5px}:host .message-row__input textarea:focus{border:none}:host .message-row__input textarea::placeholder{color:#9ea1aa}:host .message-row__input .wait-reply-dot-flashing{position:absolute;right:20px}:host .message-row__input .abort-button{position:absolute;right:0;top:-44px}:host .message-row__input .abort-button::ng-deep button{background:#f2f2ff!important;border-radius:4px}:host input{flex:1}:host input.with-button{border-bottom-right-radius:0;border-top-right-radius:0}:host .dropped-files{display:flex;flex-direction:row;margin-bottom:.5rem;flex-wrap:wrap}:host .dropped-files__item{display:flex;flex-direction:column;justify-content:center;margin:0 10px 10px 0;position:relative}:host .dropped-files__item__preview{background-size:cover;background-position:center;width:64px;height:64px;border-radius:8px;border:1px solid #ccc}:host .dropped-files__item__preview i{font-size:62px}:host .dropped-files__item__name{white-space:nowrap;font-size:12px;color:#8f929e;margin-top:4px;max-width:64px;overflow:hidden;text-overflow:ellipsis}:host .dropped-files__item__remove{position:absolute;right:-4px;top:-4px;cursor:pointer;background:#fff;border-radius:12px;color:#8f929e;border:1px solid #8f929e;font-size:14px}\n"], dependencies: [{ kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.DrButtonComponent, selector: "dr-button", inputs: ["theme", "icon", "iconColor", "iconSize", "iconAfter", "iconAfterColor", "iconAfterSize", "disabled", "isLoading", "isActive"] }, { kind: "component", type: i5.DrDotFlashingComponent, selector: "dr-dot-flashing" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
135
135
|
}
|
|
136
136
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DrChatFormComponent, decorators: [{
|
|
137
137
|
type: Component,
|
|
138
|
-
args: [{ selector: 'dr-chat-form', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n </div>\n <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n </div>\n</div>\n<div class=\"message-row\">\n <div
|
|
138
|
+
args: [{ selector: 'dr-chat-form', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n </div>\n <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n </div>\n</div>\n<div class=\"message-row\">\n <div\n class=\"message-row__input\"\n [ngClass]=\"{ 'message-row__input--focused': inputFocus, 'message-row__input--filled': !!message?.length }\">\n <textarea\n #textAreaElement\n (focus)=\"inputFocus = true\"\n (blur)=\"inputFocus = false\"\n (mouseenter)=\"inputHover = true\"\n (mouseleave)=\"inputHover = false\"\n [(ngModel)]=\"message\"\n [rows]=\"1\"\n [style]=\"getTextAreaHeight(textAreaElement)\"\n (ngModelChange)=\"onModelChange($event)\"\n type=\"text\"\n placeholder=\"{{ fileOver ? dropFilePlaceholder : messagePlaceholder }}\"\n (keyup.enter)=\"sendMessage()\">\n </textarea>\n <i *ngIf=\"!waitForReply\" (click)=\"sendMessage()\" class=\"dr-icon-notify send-button\"></i>\n <dr-dot-flashing *ngIf=\"waitForReply\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n <dr-button *ngIf=\"waitForReply\" (click)=\"abortMessage()\" theme=\"ghost\" class=\"abort-button\">Stop generating</dr-button>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:column;align-items:center;padding:0 24px;margin-top:12px}:host .message-row{display:flex;justify-content:center;width:100%;padding:0 0 21px;max-width:956px}:host .message-row__input{position:relative;display:flex;align-items:center;flex-grow:1;flex-direction:row;height:auto;overflow:visible;min-width:265px;border-radius:24px;background:#dfe0e3 border-box;border:1.5px solid transparent;box-shadow:0 2px 16px -10px #603cff29}:host .message-row__input .send-button{position:absolute;right:4px;top:2.5px;cursor:pointer;font-size:28px;width:68px;border-radius:100px;display:flex;align-items:center;justify-content:center;height:40px;background:#f0f1f4;color:#aeabac}:host .message-row__input--focused{background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input--filled{background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input--filled .send-button{color:#fff;background:linear-gradient(-90deg,#6969ff,#4eb7df) border-box}:host .message-row__input textarea{font-size:14px;color:#333;line-height:19px;flex-grow:1;resize:none;padding:14px 76px 12px 23px;margin:auto;border:none;border-radius:22.5px}:host .message-row__input textarea:focus{border:none}:host .message-row__input textarea::placeholder{color:#9ea1aa}:host .message-row__input .wait-reply-dot-flashing{position:absolute;right:20px}:host .message-row__input .abort-button{position:absolute;right:0;top:-44px}:host .message-row__input .abort-button::ng-deep button{background:#f2f2ff!important;border-radius:4px}:host input{flex:1}:host input.with-button{border-bottom-right-radius:0;border-top-right-radius:0}:host .dropped-files{display:flex;flex-direction:row;margin-bottom:.5rem;flex-wrap:wrap}:host .dropped-files__item{display:flex;flex-direction:column;justify-content:center;margin:0 10px 10px 0;position:relative}:host .dropped-files__item__preview{background-size:cover;background-position:center;width:64px;height:64px;border-radius:8px;border:1px solid #ccc}:host .dropped-files__item__preview i{font-size:62px}:host .dropped-files__item__name{white-space:nowrap;font-size:12px;color:#8f929e;margin-top:4px;max-width:64px;overflow:hidden;text-overflow:ellipsis}:host .dropped-files__item__remove{position:absolute;right:-4px;top:-4px;cursor:pointer;background:#fff;border-radius:12px;color:#8f929e;border:1px solid #8f929e;font-size:14px}\n"] }]
|
|
139
139
|
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.DomSanitizer }]; }, propDecorators: { message: [{
|
|
140
140
|
type: Input
|
|
141
141
|
}], messagePlaceholder: [{
|
|
@@ -165,4 +165,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
165
165
|
type: HostListener,
|
|
166
166
|
args: ['dragleave', ['$event']]
|
|
167
167
|
}] } });
|
|
168
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-form.component.js","sourceRoot":"","sources":["../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form/chat-form.component.ts","../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form/chat-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Input,\n    Output,\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { IMAGE_TYPES } from '../../models/chat';\n\n@Component({\n    selector: 'dr-chat-form',\n    templateUrl: 'chat-form.component.html',\n    styleUrls: ['./chat-form.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DrChatFormComponent {\n\n    _textareaInitialHeight = true;\n    inputFocus = false;\n    inputHover = false;\n\n    droppedFiles: any[] = [];\n\n    /**\n     * Predefined message text\n     *\n     * @type {string}\n     */\n    @Input() message = '';\n\n    /**\n     * Message placeholder text\n     *\n     * @type {string}\n     */\n    @Input() messagePlaceholder = 'Type a message';\n\n    /**\n     * Show send button\n     *\n     * @type {boolean}\n     */\n    @Input() dropFiles = false;\n\n    /**\n     * File drop placeholder text\n     *\n     * @type {string}\n     */\n    @Input() dropFilePlaceholder = 'Drop file to send';\n\n    /**\n     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() waitForReply = false;\n\n    /**\n     *\n     * @type {EventEmitter<{ message: string, files: File[] }>}\n     */\n    @Output() send = new EventEmitter<{ message: string; files: File[] }>();\n    @Output() abort = new EventEmitter<any>();\n\n    /**\n     * Emits when message input value has been changed\n     *\n     * @type {EventEmitter<string>}\n     */\n\n    @Output() inputChange = new EventEmitter<string>();\n\n    @HostBinding('class.file-over') fileOver = false;\n\n    constructor(protected cdr: ChangeDetectorRef, protected domSanitizer: DomSanitizer) {\n    }\n\n    @HostListener('drop', ['$event'])\n    onDrop(event: any) {\n        if (this.dropFiles) {\n            event.preventDefault();\n            event.stopPropagation();\n\n            this.fileOver = false;\n            if (event.dataTransfer?.files) {\n                for (const file of event.dataTransfer.files) {\n                    const res = file;\n\n                    if (IMAGE_TYPES.includes(file.type)) {\n                        const fr = new FileReader();\n                        fr.onload = (e: any) => {\n                            res.src = e.target.result;\n                            res.urlStyle = this.domSanitizer.bypassSecurityTrustStyle(`url(${ res.src })`);\n                            this.cdr.detectChanges();\n                        };\n\n                        fr.readAsDataURL(file);\n                    }\n                    this.droppedFiles.push(res);\n                }\n            }\n        }\n    }\n\n    removeFile(file) {\n        const index = this.droppedFiles.indexOf(file);\n        if (index >= 0) {\n            this.droppedFiles.splice(index, 1);\n        }\n    }\n\n    @HostListener('dragover', ['$event'])\n    onDragOver(event: DragEvent) {\n        event.preventDefault();\n        event.stopPropagation();\n        if (this.dropFiles) {\n            this.fileOver = true;\n        }\n    }\n\n    @HostListener('dragleave', ['$event'])\n    onDragLeave(event: DragEvent) {\n        event.preventDefault();\n        event.stopPropagation();\n        if (this.dropFiles) {\n            this.fileOver = false;\n        }\n    }\n\n    sendMessage() {\n        if (this.waitForReply) {\n            return;\n        }\n\n        if (this.droppedFiles.length || String(this.message).trim().length) {\n            this._textareaInitialHeight = true;\n            this.send.emit({ message: this.message, files: this.droppedFiles });\n            this.message = '';\n            this.droppedFiles = [];\n            this.cdr.markForCheck();\n        }\n    }\n\n    abortMessage() {\n        this.abort.emit();\n    }\n\n    onModelChange(value: string): void {\n        this._textareaInitialHeight = false;\n        this.inputChange.emit(value);\n    }\n\n    getTextAreaHeight(textAreaElement: HTMLTextAreaElement) {\n        let height;\n        if (this._textareaInitialHeight) {\n            height = 45;\n        } else {\n            height = textAreaElement.scrollHeight;\n        }\n        return `height: ${ height }px;`;\n    }\n}\n","<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n    <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n        <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n            <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n        </div>\n        <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n        <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n    </div>\n</div>\n<div class=\"message-row\">\n    <div class=\"message-row__input\">\n            <textarea #textAreaElement\n                      (focus)=\"inputFocus = true\"\n                      (blur)=\"inputFocus = false\"\n                      (mouseenter)=\"inputHover = true\"\n                      (mouseleave)=\"inputHover = false\"\n                      [(ngModel)]=\"message\"\n                      [rows]=\"1\"\n                      [style]=\"getTextAreaHeight(textAreaElement)\"\n                      (ngModelChange)=\"onModelChange($event)\"\n                      type=\"text\"\n                      placeholder=\"{{ fileOver ? dropFilePlaceholder : messagePlaceholder }}\"\n                      (keyup.enter)=\"sendMessage()\">\n            </textarea>\n        <i *ngIf=\"!waitForReply\" (click)=\"sendMessage()\" class=\"dr-icon-notify send-button\"></i>\n        <dr-dot-flashing *ngIf=\"waitForReply\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n        <dr-button *ngIf=\"waitForReply\" (click)=\"abortMessage()\" theme=\"ghost\" class=\"abort-button\">Stop generating</dr-button>\n    </div>\n</div>\n"]}
|
|
168
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-form.component.js","sourceRoot":"","sources":["../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form/chat-form.component.ts","../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form/chat-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Input,\n    Output,\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { IMAGE_TYPES } from '../../models/chat';\n\n@Component({\n    selector: 'dr-chat-form',\n    templateUrl: 'chat-form.component.html',\n    styleUrls: ['./chat-form.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DrChatFormComponent {\n\n    _textareaInitialHeight = true;\n    inputFocus = false;\n    inputHover = false;\n\n    droppedFiles: any[] = [];\n\n    /**\n     * Predefined message text\n     *\n     * @type {string}\n     */\n    @Input() message = '';\n\n    /**\n     * Message placeholder text\n     *\n     * @type {string}\n     */\n    @Input() messagePlaceholder = 'Type a message';\n\n    /**\n     * Show send button\n     *\n     * @type {boolean}\n     */\n    @Input() dropFiles = false;\n\n    /**\n     * File drop placeholder text\n     *\n     * @type {string}\n     */\n    @Input() dropFilePlaceholder = 'Drop file to send';\n\n    /**\n     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() waitForReply = false;\n\n    /**\n     *\n     * @type {EventEmitter<{ message: string, files: File[] }>}\n     */\n    @Output() send = new EventEmitter<{ message: string; files: File[] }>();\n    @Output() abort = new EventEmitter<any>();\n\n    /**\n     * Emits when message input value has been changed\n     *\n     * @type {EventEmitter<string>}\n     */\n\n    @Output() inputChange = new EventEmitter<string>();\n\n    @HostBinding('class.file-over') fileOver = false;\n\n    constructor(protected cdr: ChangeDetectorRef, protected domSanitizer: DomSanitizer) {\n    }\n\n    @HostListener('drop', ['$event'])\n    onDrop(event: any) {\n        if (this.dropFiles) {\n            event.preventDefault();\n            event.stopPropagation();\n\n            this.fileOver = false;\n            if (event.dataTransfer?.files) {\n                for (const file of event.dataTransfer.files) {\n                    const res = file;\n\n                    if (IMAGE_TYPES.includes(file.type)) {\n                        const fr = new FileReader();\n                        fr.onload = (e: any) => {\n                            res.src = e.target.result;\n                            res.urlStyle = this.domSanitizer.bypassSecurityTrustStyle(`url(${ res.src })`);\n                            this.cdr.detectChanges();\n                        };\n\n                        fr.readAsDataURL(file);\n                    }\n                    this.droppedFiles.push(res);\n                }\n            }\n        }\n    }\n\n    removeFile(file) {\n        const index = this.droppedFiles.indexOf(file);\n        if (index >= 0) {\n            this.droppedFiles.splice(index, 1);\n        }\n    }\n\n    @HostListener('dragover', ['$event'])\n    onDragOver(event: DragEvent) {\n        event.preventDefault();\n        event.stopPropagation();\n        if (this.dropFiles) {\n            this.fileOver = true;\n        }\n    }\n\n    @HostListener('dragleave', ['$event'])\n    onDragLeave(event: DragEvent) {\n        event.preventDefault();\n        event.stopPropagation();\n        if (this.dropFiles) {\n            this.fileOver = false;\n        }\n    }\n\n    sendMessage() {\n        if (this.waitForReply) {\n            return;\n        }\n\n        if (this.droppedFiles.length || String(this.message).trim().length) {\n            this._textareaInitialHeight = true;\n            this.send.emit({ message: this.message, files: this.droppedFiles });\n            this.message = '';\n            this.droppedFiles = [];\n            this.cdr.markForCheck();\n        }\n    }\n\n    abortMessage() {\n        this.abort.emit();\n    }\n\n    onModelChange(value: string): void {\n        this._textareaInitialHeight = false;\n        this.inputChange.emit(value);\n    }\n\n    getTextAreaHeight(textAreaElement: HTMLTextAreaElement) {\n        let height;\n        if (this._textareaInitialHeight) {\n            height = 45;\n        } else {\n            height = textAreaElement.scrollHeight;\n        }\n        return `height: ${ height }px;`;\n    }\n}\n","<div class=\"dropped-files\" *ngIf=\"droppedFiles?.length\">\n    <div class=\"dropped-files__item\" *ngFor=\"let file of droppedFiles\">\n        <div class=\"dropped-files__item__preview\" [style.background-image]=\"file.urlStyle || 'none'\">\n            <i class=\"dr-icon-file\" *ngIf=\"!file.urlStyle\"></i>\n        </div>\n        <div class=\"dropped-files__item__name\">{{ file.name }}</div>\n        <i class=\"dropped-files__item__remove dr-icon-exit\" (click)=\"removeFile(file)\"></i>\n    </div>\n</div>\n<div class=\"message-row\">\n    <div\n        class=\"message-row__input\"\n        [ngClass]=\"{ 'message-row__input--focused': inputFocus, 'message-row__input--filled': !!message?.length }\">\n        <textarea\n            #textAreaElement\n            (focus)=\"inputFocus = true\"\n            (blur)=\"inputFocus = false\"\n            (mouseenter)=\"inputHover = true\"\n            (mouseleave)=\"inputHover = false\"\n            [(ngModel)]=\"message\"\n            [rows]=\"1\"\n            [style]=\"getTextAreaHeight(textAreaElement)\"\n            (ngModelChange)=\"onModelChange($event)\"\n            type=\"text\"\n            placeholder=\"{{ fileOver ? dropFilePlaceholder : messagePlaceholder }}\"\n            (keyup.enter)=\"sendMessage()\">\n        </textarea>\n        <i *ngIf=\"!waitForReply\" (click)=\"sendMessage()\" class=\"dr-icon-notify send-button\"></i>\n        <dr-dot-flashing *ngIf=\"waitForReply\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n        <dr-button *ngIf=\"waitForReply\" (click)=\"abortMessage()\" theme=\"ghost\" class=\"abort-button\">Stop generating</dr-button>\n    </div>\n</div>\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../../dr-tooltip/dr-tooltip.directive";
|
|
4
|
+
export class CodeEditorHintWrapperComponent {
|
|
5
|
+
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CodeEditorHintWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
6
|
+
/** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: CodeEditorHintWrapperComponent, selector: "dr-code-editor-hint-wrapper", ngImport: i0, template: ` <span [innerHTML]="content" [drTooltip]="tooltip"></span> `, isInline: true, dependencies: [{ kind: "directive", type: i1.DrTooltipDirective, selector: "[drTooltip]", inputs: ["drTooltip", "drTooltipContext", "drTooltipPosition", "drTooltipClass", "drTooltipTheme", "drTooltipMousleaveTimeout", "drTooltipMouseEnterTimeout", "drTooltipOptions", "drTooltipShow"] }] }); }
|
|
7
|
+
}
|
|
8
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CodeEditorHintWrapperComponent, decorators: [{
|
|
9
|
+
type: Component,
|
|
10
|
+
args: [{
|
|
11
|
+
selector: 'dr-code-editor-hint-wrapper',
|
|
12
|
+
template: ` <span [innerHTML]="content" [drTooltip]="tooltip"></span> `,
|
|
13
|
+
}]
|
|
14
|
+
}] });
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29kZS1lZGl0b3ItaGludC13cmFwcGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2RhdGFyYWlsc3NoYXJlZC9zcmMvbGliL2RyLWNvZGUtZWRpdG9yL2NvbXBvbmVudHMvY29kZS1lZGl0b3ItaGludC13cmFwcGVyLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFNMUMsTUFBTSxPQUFPLDhCQUE4QjtrSUFBOUIsOEJBQThCO3NIQUE5Qiw4QkFBOEIsbUVBRjdCLDZEQUE2RDs7NEZBRTlELDhCQUE4QjtrQkFKMUMsU0FBUzttQkFBQztvQkFDUCxRQUFRLEVBQUUsNkJBQTZCO29CQUN2QyxRQUFRLEVBQUUsNkRBQTZEO2lCQUMxRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ2RyLWNvZGUtZWRpdG9yLWhpbnQtd3JhcHBlcicsXG4gICAgdGVtcGxhdGU6IGAgPHNwYW4gW2lubmVySFRNTF09XCJjb250ZW50XCIgW2RyVG9vbHRpcF09XCJ0b29sdGlwXCI+PC9zcGFuPiBgLFxufSlcbmV4cG9ydCBjbGFzcyBDb2RlRWRpdG9ySGludFdyYXBwZXJDb21wb25lbnQge1xuICAgIGNvbnRlbnQ6IHN0cmluZztcbiAgICB0b29sdGlwOiBzdHJpbmc7XG59XG4iXX0=
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, forwardRef, Input, KeyValueDiffers, NgZone, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
3
|
+
import * as CodeMirror from 'codemirror';
|
|
4
|
+
import { DrSharedUtils } from '../../utils/dr-shared-utils';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class DrCodemirrorComponent {
|
|
7
|
+
set options(value) {
|
|
8
|
+
this._options = value;
|
|
9
|
+
if (!this._differ && value) {
|
|
10
|
+
this._differ = this._differs.find(value).create();
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
constructor(_differs, _ngZone) {
|
|
14
|
+
this._differs = _differs;
|
|
15
|
+
this._ngZone = _ngZone;
|
|
16
|
+
this.className = '';
|
|
17
|
+
this.name = 'codemirror';
|
|
18
|
+
this.autoFocus = false;
|
|
19
|
+
this.preserveScrollPosition = false;
|
|
20
|
+
this.cursorActivity = new EventEmitter();
|
|
21
|
+
this.focusChange = new EventEmitter();
|
|
22
|
+
this.scroll = new EventEmitter();
|
|
23
|
+
this.drop = new EventEmitter();
|
|
24
|
+
this.codeMirrorLoaded = new EventEmitter();
|
|
25
|
+
this.value = '';
|
|
26
|
+
this.disabled = false;
|
|
27
|
+
this.isFocused = false;
|
|
28
|
+
this.onChange = (_) => { };
|
|
29
|
+
this.onTouched = () => { };
|
|
30
|
+
}
|
|
31
|
+
get codeMirrorGlobal() {
|
|
32
|
+
if (this._codeMirror) {
|
|
33
|
+
return this._codeMirror;
|
|
34
|
+
}
|
|
35
|
+
this._codeMirror = CodeMirror;
|
|
36
|
+
return this._codeMirror;
|
|
37
|
+
}
|
|
38
|
+
ngAfterViewInit() {
|
|
39
|
+
this._ngZone.runOutsideAngular(async () => {
|
|
40
|
+
const codeMirrorObj = await this.codeMirrorGlobal;
|
|
41
|
+
const codeMirror = codeMirrorObj?.default ? codeMirrorObj.default : codeMirrorObj;
|
|
42
|
+
this.codeMirror = codeMirror.fromTextArea(this.ref.nativeElement, this._options);
|
|
43
|
+
this.codeMirror.on('cursorActivity', cm => this._ngZone.run(() => this.cursorActive(cm)));
|
|
44
|
+
this.codeMirror.on('scroll', this.scrollChanged.bind(this));
|
|
45
|
+
this.codeMirror.on('blur', () => this._ngZone.run(() => this.focusChanged(false)));
|
|
46
|
+
this.codeMirror.on('focus', () => this._ngZone.run(() => this.focusChanged(true)));
|
|
47
|
+
this.codeMirror.on('change', (cm, change) => this._ngZone.run(() => this.codemirrorValueChanged(cm, change)));
|
|
48
|
+
this.codeMirror.on('drop', (cm, e) => {
|
|
49
|
+
this._ngZone.run(() => this.dropFiles(cm, e));
|
|
50
|
+
});
|
|
51
|
+
this.codeMirror.setValue(this.value);
|
|
52
|
+
this.codeMirrorLoaded.emit(this);
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
ngDoCheck() {
|
|
56
|
+
if (!this._differ) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
// check options have not changed
|
|
60
|
+
const changes = this._differ.diff(this._options);
|
|
61
|
+
if (changes) {
|
|
62
|
+
changes.forEachChangedItem(option => this.setOptionIfChanged(option.key, option.currentValue));
|
|
63
|
+
changes.forEachAddedItem(option => this.setOptionIfChanged(option.key, option.currentValue));
|
|
64
|
+
changes.forEachRemovedItem(option => this.setOptionIfChanged(option.key, option.currentValue));
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
ngOnDestroy() {
|
|
68
|
+
// is there a lighter-weight way to remove the cm instance?
|
|
69
|
+
if (this.codeMirror) {
|
|
70
|
+
this.codeMirror.toTextArea();
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
codemirrorValueChanged(cm, change) {
|
|
74
|
+
const cmVal = cm.getValue();
|
|
75
|
+
if (this.value !== cmVal) {
|
|
76
|
+
this.value = cmVal;
|
|
77
|
+
this.onChange(this.value);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
setOptionIfChanged(optionName, newValue) {
|
|
81
|
+
if (!this.codeMirror) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
// cast to any to handle strictly typed option names
|
|
85
|
+
// could possibly import settings strings available in the future
|
|
86
|
+
this.codeMirror.setOption(optionName, newValue);
|
|
87
|
+
}
|
|
88
|
+
focusChanged(focused) {
|
|
89
|
+
this.onTouched();
|
|
90
|
+
this.isFocused = focused;
|
|
91
|
+
this.focusChange.emit(focused);
|
|
92
|
+
}
|
|
93
|
+
scrollChanged(cm) {
|
|
94
|
+
this.scroll.emit(cm.getScrollInfo());
|
|
95
|
+
}
|
|
96
|
+
cursorActive(cm) {
|
|
97
|
+
this.cursorActivity.emit(cm);
|
|
98
|
+
}
|
|
99
|
+
dropFiles(cm, e) {
|
|
100
|
+
this.drop.emit([cm, e]);
|
|
101
|
+
}
|
|
102
|
+
writeValue(value) {
|
|
103
|
+
if (value === null || value === undefined) {
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
if (!this.codeMirror) {
|
|
107
|
+
this.value = value;
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
const cur = this.codeMirror.getValue();
|
|
111
|
+
if (value !== cur && DrSharedUtils.normalizeLineEndings(cur) !== DrSharedUtils.normalizeLineEndings(value)) {
|
|
112
|
+
this.value = value;
|
|
113
|
+
if (this.preserveScrollPosition) {
|
|
114
|
+
const prevScrollPosition = this.codeMirror.getScrollInfo();
|
|
115
|
+
this.codeMirror.setValue(this.value);
|
|
116
|
+
this.codeMirror.scrollTo(prevScrollPosition.left, prevScrollPosition.top);
|
|
117
|
+
}
|
|
118
|
+
else {
|
|
119
|
+
this.codeMirror.setValue(this.value);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
registerOnChange(fn) {
|
|
124
|
+
this.onChange = fn;
|
|
125
|
+
}
|
|
126
|
+
registerOnTouched(fn) {
|
|
127
|
+
this.onTouched = fn;
|
|
128
|
+
}
|
|
129
|
+
setDisabledState(isDisabled) {
|
|
130
|
+
this.disabled = isDisabled;
|
|
131
|
+
this.setOptionIfChanged('readOnly', this.disabled);
|
|
132
|
+
}
|
|
133
|
+
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DrCodemirrorComponent, deps: [{ token: i0.KeyValueDiffers }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
134
|
+
/** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DrCodemirrorComponent, selector: "dr-codemirror", inputs: { className: "className", name: "name", autoFocus: "autoFocus", options: "options", preserveScrollPosition: "preserveScrollPosition" }, outputs: { cursorActivity: "cursorActivity", focusChange: "focusChange", scroll: "scroll", drop: "drop", codeMirrorLoaded: "codeMirrorLoaded" }, providers: [
|
|
135
|
+
{
|
|
136
|
+
provide: NG_VALUE_ACCESSOR,
|
|
137
|
+
useExisting: forwardRef((() => DrCodemirrorComponent)),
|
|
138
|
+
multi: true,
|
|
139
|
+
},
|
|
140
|
+
], viewQueries: [{ propertyName: "ref", first: true, predicate: ["ref"], descendants: true }], ngImport: i0, template: `
|
|
141
|
+
<textarea
|
|
142
|
+
[name]="name"
|
|
143
|
+
class="dr-codemirror {{ className }}"
|
|
144
|
+
[class.dr-codemirror--focused]="isFocused"
|
|
145
|
+
autocomplete="off"
|
|
146
|
+
[autofocus]="autoFocus"
|
|
147
|
+
#ref
|
|
148
|
+
>
|
|
149
|
+
</textarea>
|
|
150
|
+
`, isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
151
|
+
}
|
|
152
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DrCodemirrorComponent, decorators: [{
|
|
153
|
+
type: Component,
|
|
154
|
+
args: [{
|
|
155
|
+
selector: 'dr-codemirror',
|
|
156
|
+
template: `
|
|
157
|
+
<textarea
|
|
158
|
+
[name]="name"
|
|
159
|
+
class="dr-codemirror {{ className }}"
|
|
160
|
+
[class.dr-codemirror--focused]="isFocused"
|
|
161
|
+
autocomplete="off"
|
|
162
|
+
[autofocus]="autoFocus"
|
|
163
|
+
#ref
|
|
164
|
+
>
|
|
165
|
+
</textarea>
|
|
166
|
+
`,
|
|
167
|
+
providers: [
|
|
168
|
+
{
|
|
169
|
+
provide: NG_VALUE_ACCESSOR,
|
|
170
|
+
useExisting: forwardRef((() => DrCodemirrorComponent)),
|
|
171
|
+
multi: true,
|
|
172
|
+
},
|
|
173
|
+
],
|
|
174
|
+
preserveWhitespaces: false,
|
|
175
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
176
|
+
}]
|
|
177
|
+
}], ctorParameters: function () { return [{ type: i0.KeyValueDiffers }, { type: i0.NgZone }]; }, propDecorators: { className: [{
|
|
178
|
+
type: Input
|
|
179
|
+
}], name: [{
|
|
180
|
+
type: Input
|
|
181
|
+
}], autoFocus: [{
|
|
182
|
+
type: Input
|
|
183
|
+
}], options: [{
|
|
184
|
+
type: Input
|
|
185
|
+
}], preserveScrollPosition: [{
|
|
186
|
+
type: Input
|
|
187
|
+
}], cursorActivity: [{
|
|
188
|
+
type: Output
|
|
189
|
+
}], focusChange: [{
|
|
190
|
+
type: Output
|
|
191
|
+
}], scroll: [{
|
|
192
|
+
type: Output
|
|
193
|
+
}], drop: [{
|
|
194
|
+
type: Output
|
|
195
|
+
}], codeMirrorLoaded: [{
|
|
196
|
+
type: Output
|
|
197
|
+
}], ref: [{
|
|
198
|
+
type: ViewChild,
|
|
199
|
+
args: ['ref']
|
|
200
|
+
}] } });
|
|
201
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dr-codemirror.component.js","sourceRoot":"","sources":["../../../../../../projects/datarailsshared/src/lib/dr-code-editor/components/dr-codemirror.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    ChangeDetectionStrategy,\n    Component,\n    DoCheck,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    Input,\n    KeyValueDiffer,\n    KeyValueDiffers,\n    NgZone,\n    OnDestroy,\n    Output,\n    ViewChild,\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { Editor, EditorChange, EditorFromTextArea, ScrollInfo } from 'codemirror';\nimport * as CodeMirror from 'codemirror'\nimport { DrSharedUtils } from '../../utils/dr-shared-utils';\n\n@Component({\n    selector: 'dr-codemirror',\n    template: `\n    <textarea\n      [name]=\"name\"\n      class=\"dr-codemirror {{ className }}\"\n      [class.dr-codemirror--focused]=\"isFocused\"\n      autocomplete=\"off\"\n      [autofocus]=\"autoFocus\"\n      #ref\n    >\n    </textarea>\n  `,\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => DrCodemirrorComponent),\n            multi: true,\n        },\n    ],\n    preserveWhitespaces: false,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DrCodemirrorComponent implements AfterViewInit, OnDestroy, ControlValueAccessor, DoCheck\n{\n    @Input() className = '';\n    @Input() name = 'codemirror';\n    @Input() autoFocus = false;\n\n    @Input()\n    set options(value: { [key: string]: any }) {\n        this._options = value;\n        if (!this._differ && value) {\n            this._differ = this._differs.find(value).create();\n        }\n    }\n    @Input() preserveScrollPosition = false;\n    @Output() cursorActivity = new EventEmitter<Editor>();\n    @Output() focusChange = new EventEmitter<boolean>();\n    @Output() scroll = new EventEmitter<ScrollInfo>();\n    @Output() drop = new EventEmitter<[Editor, DragEvent]>();\n    @Output() codeMirrorLoaded = new EventEmitter<DrCodemirrorComponent>();\n    @ViewChild('ref') ref!: ElementRef<HTMLTextAreaElement>;\n    value = '';\n    disabled = false;\n    isFocused = false;\n    codeMirror?: EditorFromTextArea;\n\n    private _codeMirror: any;\n    private _differ?: KeyValueDiffer<string, any>;\n    private _options: any;\n\n    constructor(private _differs: KeyValueDiffers, private _ngZone: NgZone) {}\n\n    get codeMirrorGlobal(): any {\n        if (this._codeMirror) {\n            return this._codeMirror;\n        }\n\n        this._codeMirror = CodeMirror;\n        return this._codeMirror;\n    }\n\n    ngAfterViewInit() {\n        this._ngZone.runOutsideAngular(async () => {\n            const codeMirrorObj = await this.codeMirrorGlobal;\n            const codeMirror = codeMirrorObj?.default ? codeMirrorObj.default : codeMirrorObj;\n            this.codeMirror = codeMirror.fromTextArea(\n                this.ref.nativeElement,\n                this._options,\n            ) as EditorFromTextArea;\n            this.codeMirror.on('cursorActivity', cm => this._ngZone.run(() => this.cursorActive(cm)));\n            this.codeMirror.on('scroll', this.scrollChanged.bind(this));\n            this.codeMirror.on('blur', () => this._ngZone.run(() => this.focusChanged(false)));\n            this.codeMirror.on('focus', () => this._ngZone.run(() => this.focusChanged(true)));\n            this.codeMirror.on('change', (cm, change) =>\n                this._ngZone.run(() => this.codemirrorValueChanged(cm, change)),\n            );\n            this.codeMirror.on('drop', (cm, e) => {\n                this._ngZone.run(() => this.dropFiles(cm, e));\n            });\n            this.codeMirror.setValue(this.value);\n            this.codeMirrorLoaded.emit(this);\n        });\n    }\n\n    ngDoCheck() {\n        if (!this._differ) {\n            return;\n        }\n        // check options have not changed\n        const changes = this._differ.diff(this._options);\n        if (changes) {\n            changes.forEachChangedItem(option =>\n                this.setOptionIfChanged(option.key, option.currentValue),\n            );\n            changes.forEachAddedItem(option => this.setOptionIfChanged(option.key, option.currentValue));\n            changes.forEachRemovedItem(option =>\n                this.setOptionIfChanged(option.key, option.currentValue),\n            );\n        }\n    }\n\n    ngOnDestroy() {\n        // is there a lighter-weight way to remove the cm instance?\n        if (this.codeMirror) {\n            this.codeMirror.toTextArea();\n        }\n    }\n\n    codemirrorValueChanged(cm: Editor, change: EditorChange) {\n        const cmVal = cm.getValue();\n        if (this.value !== cmVal) {\n            this.value = cmVal;\n            this.onChange(this.value);\n        }\n    }\n\n    setOptionIfChanged(optionName: string, newValue: any) {\n        if (!this.codeMirror) {\n            return;\n        }\n\n        // cast to any to handle strictly typed option names\n        // could possibly import settings strings available in the future\n        this.codeMirror.setOption(optionName as any, newValue);\n    }\n\n    focusChanged(focused: boolean) {\n        this.onTouched();\n        this.isFocused = focused;\n        this.focusChange.emit(focused);\n    }\n\n    scrollChanged(cm: Editor) {\n        this.scroll.emit(cm.getScrollInfo());\n    }\n\n    cursorActive(cm: Editor) {\n        this.cursorActivity.emit(cm);\n    }\n\n    dropFiles(cm: Editor, e: DragEvent) {\n        this.drop.emit([cm, e]);\n    }\n\n    writeValue(value: string) {\n        if (value === null || value === undefined) {\n            return;\n        }\n        if (!this.codeMirror) {\n            this.value = value;\n            return;\n        }\n        const cur = this.codeMirror.getValue();\n        if (value !== cur && DrSharedUtils.normalizeLineEndings(cur) !== DrSharedUtils.normalizeLineEndings(value)) {\n            this.value = value;\n            if (this.preserveScrollPosition) {\n                const prevScrollPosition = this.codeMirror.getScrollInfo();\n                this.codeMirror.setValue(this.value);\n                this.codeMirror.scrollTo(prevScrollPosition.left, prevScrollPosition.top);\n            } else {\n                this.codeMirror.setValue(this.value);\n            }\n        }\n    }\n\n    registerOnChange(fn: (value: string) => void) {\n        this.onChange = fn;\n    }\n\n    registerOnTouched(fn: () => void) {\n        this.onTouched = fn;\n    }\n\n    setDisabledState(isDisabled: boolean) {\n        this.disabled = isDisabled;\n        this.setOptionIfChanged('readOnly', this.disabled);\n    }\n\n    private onChange = (_: any) => {};\n\n    private onTouched = () => {};\n}\n"]}
|