@datarailsshared/datarailsshared 1.6.212 → 1.6.216
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/datarailsshared-datarailsshared-1.6.216.tgz +0 -0
- package/esm2022/lib/dr-chat/dr-chat-form/chat-form.component.mjs +3 -3
- package/esm2022/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.mjs +65 -10
- package/fesm2022/datarailsshared-datarailsshared.mjs +66 -11
- package/fesm2022/datarailsshared-datarailsshared.mjs.map +1 -1
- package/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.d.ts +11 -1
- package/package.json +1 -1
- package/datarailsshared-datarailsshared-1.6.212.tgz +0 -0
|
Binary file
|
|
@@ -206,7 +206,7 @@ export class DrChatFormComponent {
|
|
|
206
206
|
i0.ɵɵlistener("drop", function DrChatFormComponent_drop_HostBindingHandler($event) { return ctx.onDrop($event); })("dragover", function DrChatFormComponent_dragover_HostBindingHandler($event) { return ctx.onDragOver($event); })("dragleave", function DrChatFormComponent_dragleave_HostBindingHandler($event) { return ctx.onDragLeave($event); });
|
|
207
207
|
} if (rf & 2) {
|
|
208
208
|
i0.ɵɵclassProp("file-over", ctx.fileOver);
|
|
209
|
-
} }, inputs: { message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply", showDisabledButtonInsteadOfDotFlashing: "showDisabledButtonInsteadOfDotFlashing" }, outputs: { send: "send", abort: "abort", inputChange: "inputChange" }, decls: 10, vars: 14, consts: [["class", "dropped-files", 4, "ngIf"], [1, "message-row"], [1, "message-row__input", 3, "ngClass"], ["type", "text",
|
|
209
|
+
} }, inputs: { message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply", showDisabledButtonInsteadOfDotFlashing: "showDisabledButtonInsteadOfDotFlashing" }, outputs: { send: "send", abort: "abort", inputChange: "inputChange" }, decls: 10, vars: 14, consts: [["class", "dropped-files", 4, "ngIf"], [1, "message-row"], [1, "message-row__input", 3, "ngClass"], ["type", "text", 3, "ngModel", "rows", "placeholder", "focus", "blur", "mouseenter", "mouseleave", "ngModelChange", "keydown.enter"], ["textAreaElement", ""], ["class", "dr-icon-notify send-button", 3, "style", "click", 4, "ngIf"], ["class", "dr-icon-notify send-button-disabled", 3, "click", 4, "ngIf"], ["class", "wait-reply-dot-flashing", 4, "ngIf"], ["theme", "ghost", "class", "abort-button", 3, "click", 4, "ngIf"], [1, "dropped-files"], ["class", "dropped-files__item", 4, "ngFor", "ngForOf"], [1, "dropped-files__item"], [1, "dropped-files__item__preview"], ["class", "dr-icon-file", 4, "ngIf"], [1, "dropped-files__item__name"], [1, "dropped-files__item__remove", "dr-icon-exit", 3, "click"], [1, "dr-icon-file"], [1, "dr-icon-notify", "send-button", 3, "click"], [1, "dr-icon-notify", "send-button-disabled", 3, "click"], [1, "wait-reply-dot-flashing"], ["theme", "ghost", 1, "abort-button", 3, "click"]], template: function DrChatFormComponent_Template(rf, ctx) { if (rf & 1) {
|
|
210
210
|
i0.ɵɵtemplate(0, DrChatFormComponent_div_0_Template, 2, 1, "div", 0);
|
|
211
211
|
i0.ɵɵelementStart(1, "div", 1)(2, "div", 2)(3, "textarea", 3, 4);
|
|
212
212
|
i0.ɵɵlistener("focus", function DrChatFormComponent_Template_textarea_focus_3_listener() { return ctx.inputFocus = true; })("blur", function DrChatFormComponent_Template_textarea_blur_3_listener() { return ctx.inputFocus = false; })("mouseenter", function DrChatFormComponent_Template_textarea_mouseenter_3_listener() { return ctx.inputHover = true; })("mouseleave", function DrChatFormComponent_Template_textarea_mouseleave_3_listener() { return ctx.inputHover = false; })("ngModelChange", function DrChatFormComponent_Template_textarea_ngModelChange_3_listener($event) { return ctx.message = $event; })("ngModelChange", function DrChatFormComponent_Template_textarea_ngModelChange_3_listener($event) { return ctx.onModelChange($event); })("keydown.enter", function DrChatFormComponent_Template_textarea_keydown_enter_3_listener($event) { return ctx.sendMessage($event); });
|
|
@@ -238,7 +238,7 @@ export class DrChatFormComponent {
|
|
|
238
238
|
}
|
|
239
239
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(DrChatFormComponent, [{
|
|
240
240
|
type: Component,
|
|
241
|
-
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
|
|
241
|
+
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 (keydown.enter)=\"sendMessage($event)\">\n </textarea>\n <i\n *ngIf=\"!waitForReply\"\n [style]=\"getSendButtonPosition(textAreaElement)\"\n (click)=\"sendMessage($event)\"\n class=\"dr-icon-notify send-button\"></i>\n <i\n *ngIf=\"waitForReply && showDisabledButtonInsteadOfDotFlashing\"\n (click)=\"sendMessage($event)\"\n class=\"dr-icon-notify send-button-disabled\"></i>\n <dr-dot-flashing\n *ngIf=\"waitForReply && !showDisabledButtonInsteadOfDotFlashing\"\n 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{--send-button-offset: 42px;display:flex;flex-direction:column;align-items:center;padding:0 16px;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,:host .message-row__input .send-button-disabled{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"] }]
|
|
242
242
|
}], function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.DomSanitizer }]; }, { message: [{
|
|
243
243
|
type: Input
|
|
244
244
|
}], messagePlaceholder: [{
|
|
@@ -270,4 +270,4 @@ export class DrChatFormComponent {
|
|
|
270
270
|
type: HostListener,
|
|
271
271
|
args: ['dragleave', ['$event']]
|
|
272
272
|
}] }); })();
|
|
273
|
-
//# 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    _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     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() showDisabledButtonInsteadOfDotFlashing = 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(\n        protected cdr: ChangeDetectorRef,\n        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($event) {\n        if (!$event || !$event.shiftKey) {\n            $event && $event.preventDefault();\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\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        if (this._textareaInitialHeight) {\n            textAreaElement.style.height = '45px';\n        } else {\n            textAreaElement.style.height = 'auto';\n            textAreaElement.style.height = textAreaElement.scrollHeight + 'px';\n        }\n\n        return `${textAreaElement.style.height}`;\n    }\n\n    getSendButtonPosition(textAreaElement: HTMLTextAreaElement) {\n        return `top: calc(${this.getTextAreaHeight(textAreaElement)} - var(--send-button-offset));`;\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            data-analytics=\"ai-chat-actions-send-message\"\n            (keydown.enter)=\"sendMessage($event)\">\n        </textarea>\n        <i\n            *ngIf=\"!waitForReply\"\n            [style]=\"getSendButtonPosition(textAreaElement)\"\n            (click)=\"sendMessage($event)\"\n            data-analytics=\"ai-chat-actions-send-message\"\n            class=\"dr-icon-notify send-button\"></i>\n        <i\n            *ngIf=\"waitForReply && showDisabledButtonInsteadOfDotFlashing\"\n            (click)=\"sendMessage($event)\"\n            class=\"dr-icon-notify send-button-disabled\"></i>\n        <dr-dot-flashing\n            *ngIf=\"waitForReply && !showDisabledButtonInsteadOfDotFlashing\"\n            class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n        <dr-button\n            *ngIf=\"waitForReply\"\n            (click)=\"abortMessage()\"\n            data-analytics=\"ai-chat-actions-abort-message\"\n            theme=\"ghost\"\n            class=\"abort-button\"\n            >Stop generating</dr-button\n        >\n    </div>\n</div>\n"]}
|
|
273
|
+
//# 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    _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     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() showDisabledButtonInsteadOfDotFlashing = 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(\n        protected cdr: ChangeDetectorRef,\n        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($event) {\n        if (!$event || !$event.shiftKey) {\n            $event && $event.preventDefault();\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\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        if (this._textareaInitialHeight) {\n            textAreaElement.style.height = '45px';\n        } else {\n            textAreaElement.style.height = 'auto';\n            textAreaElement.style.height = textAreaElement.scrollHeight + 'px';\n        }\n\n        return `${textAreaElement.style.height}`;\n    }\n\n    getSendButtonPosition(textAreaElement: HTMLTextAreaElement) {\n        return `top: calc(${this.getTextAreaHeight(textAreaElement)} - var(--send-button-offset));`;\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            (keydown.enter)=\"sendMessage($event)\">\n        </textarea>\n        <i\n            *ngIf=\"!waitForReply\"\n            [style]=\"getSendButtonPosition(textAreaElement)\"\n            (click)=\"sendMessage($event)\"\n            class=\"dr-icon-notify send-button\"></i>\n        <i\n            *ngIf=\"waitForReply && showDisabledButtonInsteadOfDotFlashing\"\n            (click)=\"sendMessage($event)\"\n            class=\"dr-icon-notify send-button-disabled\"></i>\n        <dr-dot-flashing\n            *ngIf=\"waitForReply && !showDisabledButtonInsteadOfDotFlashing\"\n            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"]}
|
|
@@ -90,6 +90,7 @@ export class DrChatFormDropdownComponent {
|
|
|
90
90
|
this.send = new EventEmitter();
|
|
91
91
|
this.uploadFiles = new EventEmitter();
|
|
92
92
|
this.abort = new EventEmitter();
|
|
93
|
+
this.filesRejected = new EventEmitter();
|
|
93
94
|
/**
|
|
94
95
|
* Emits when message input value has been changed
|
|
95
96
|
*
|
|
@@ -105,7 +106,7 @@ export class DrChatFormDropdownComponent {
|
|
|
105
106
|
this.fileOver = false;
|
|
106
107
|
const files = event.dataTransfer?.files;
|
|
107
108
|
if (files) {
|
|
108
|
-
this.saveFiles(files);
|
|
109
|
+
this.saveFiles(Array.from(files));
|
|
109
110
|
}
|
|
110
111
|
}
|
|
111
112
|
}
|
|
@@ -175,15 +176,59 @@ export class DrChatFormDropdownComponent {
|
|
|
175
176
|
}
|
|
176
177
|
async saveFiles(files) {
|
|
177
178
|
const uploadedFiles = [];
|
|
179
|
+
const tooLargeFiles = [];
|
|
180
|
+
const unsupportedFiles = [];
|
|
181
|
+
const allowedExtensions = this.getAllowedExtensions();
|
|
182
|
+
let maxExceeded = false;
|
|
183
|
+
const remainingSlotsInit = Math.max(0, this.maxFilesCount - this.droppedFiles$.value.length);
|
|
184
|
+
if (files.length > remainingSlotsInit) {
|
|
185
|
+
this.filesRejected.emit({ oversize: [], unsupported: [], maxExceeded: true });
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
let remainingSlots = remainingSlotsInit;
|
|
178
189
|
for (const file of files) {
|
|
179
|
-
|
|
190
|
+
if (remainingSlots <= 0) {
|
|
191
|
+
maxExceeded = true;
|
|
192
|
+
continue;
|
|
193
|
+
}
|
|
194
|
+
const fileExtension = this.getFileExtension(file.name);
|
|
195
|
+
const isAllowedByType = !allowedExtensions || allowedExtensions.has(fileExtension);
|
|
196
|
+
if (!isAllowedByType) {
|
|
197
|
+
unsupportedFiles.push(file.name);
|
|
198
|
+
continue;
|
|
199
|
+
}
|
|
200
|
+
if (file.size > this.maxFileSizeBytes) {
|
|
201
|
+
tooLargeFiles.push(file.name);
|
|
202
|
+
continue;
|
|
203
|
+
}
|
|
204
|
+
const res = file;
|
|
180
205
|
res.data = (await this.base64Convert(res));
|
|
181
206
|
this.droppedFiles$.next([...this.droppedFiles$.value, res]);
|
|
182
207
|
uploadedFiles.push(res);
|
|
208
|
+
remainingSlots--;
|
|
183
209
|
this.cdr.markForCheck();
|
|
184
210
|
this.cdr.detectChanges();
|
|
185
211
|
}
|
|
186
|
-
|
|
212
|
+
if (tooLargeFiles.length || unsupportedFiles.length || maxExceeded) {
|
|
213
|
+
this.filesRejected.emit({ oversize: tooLargeFiles, unsupported: unsupportedFiles, maxExceeded });
|
|
214
|
+
}
|
|
215
|
+
if (uploadedFiles.length) {
|
|
216
|
+
this.uploadFiles.emit(uploadedFiles.map((item) => ({ data: item.data, name: item.name })));
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
getAllowedExtensions() {
|
|
220
|
+
if (!this.acceptFormatFiles || !this.acceptFormatFiles.trim())
|
|
221
|
+
return null;
|
|
222
|
+
const parts = this.acceptFormatFiles
|
|
223
|
+
.split(',')
|
|
224
|
+
.map((p) => p.trim().toLowerCase())
|
|
225
|
+
.filter(Boolean)
|
|
226
|
+
.map((p) => (p.startsWith('.') ? p.slice(1) : p));
|
|
227
|
+
return parts.length ? new Set(parts) : null;
|
|
228
|
+
}
|
|
229
|
+
getFileExtension(fileName) {
|
|
230
|
+
const idx = fileName.lastIndexOf('.');
|
|
231
|
+
return idx >= 0 ? fileName.slice(idx + 1).toLowerCase() : '';
|
|
187
232
|
}
|
|
188
233
|
base64Convert(file) {
|
|
189
234
|
return new Promise((resolve, reject) => {
|
|
@@ -205,7 +250,7 @@ export class DrChatFormDropdownComponent {
|
|
|
205
250
|
i0.ɵɵlistener("drop", function DrChatFormDropdownComponent_drop_HostBindingHandler($event) { return ctx.onDrop($event); })("dragover", function DrChatFormDropdownComponent_dragover_HostBindingHandler($event) { return ctx.onDragOver($event); })("dragleave", function DrChatFormDropdownComponent_dragleave_HostBindingHandler($event) { return ctx.onDragLeave($event); });
|
|
206
251
|
} if (rf & 2) {
|
|
207
252
|
i0.ɵɵclassProp("file-over", ctx.fileOver);
|
|
208
|
-
} }, inputs: { isUploadingFiles: "isUploadingFiles", message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply", showDotFlashing: "showDotFlashing" }, outputs: { send: "send", uploadFiles: "uploadFiles", abort: "abort", inputChange: "inputChange" }, ngContentSelectors: _c5, decls: 17, vars:
|
|
253
|
+
} }, inputs: { acceptFormatFiles: "acceptFormatFiles", maxFileSizeBytes: "maxFileSizeBytes", maxFilesCount: "maxFilesCount", isUploadingFiles: "isUploadingFiles", message: "message", messagePlaceholder: "messagePlaceholder", dropFiles: "dropFiles", dropFilePlaceholder: "dropFilePlaceholder", waitForReply: "waitForReply", showDotFlashing: "showDotFlashing" }, outputs: { send: "send", uploadFiles: "uploadFiles", abort: "abort", filesRejected: "filesRejected", inputChange: "inputChange" }, ngContentSelectors: _c5, decls: 17, vars: 21, consts: [[1, "message-row", 3, "ngClass"], [1, "message-row__input", 3, "ngClass"], [3, "files", "isRemovable", "maxLengthText", "removeFileEvent", 4, "ngIf"], [1, "message-row__input-textarea-wrap"], ["type", "text", 3, "ngModel", "rows", "placeholder", "focus", "blur", "mouseenter", "mouseleave", "ngModelChange", "keydown.enter"], ["textAreaElement", ""], [1, "message-row__btns"], ["type", "file", "hidden", "", "multiple", "", 3, "accept", "change"], ["fileInput", ""], ["class", "dr-icon-send-arrow-up send-button", 3, "click", 4, "ngIf"], ["class", "dr-icon-stop abort-button", 3, "click", 4, "ngIf"], ["class", "wait-reply-dot-flashing", 4, "ngIf"], [3, "files", "isRemovable", "maxLengthText", "removeFileEvent"], [1, "dr-icon-send-arrow-up", "send-button", 3, "click"], [1, "dr-icon-stop", "abort-button", 3, "click"], [1, "wait-reply-dot-flashing"]], template: function DrChatFormDropdownComponent_Template(rf, ctx) { if (rf & 1) {
|
|
209
254
|
i0.ɵɵprojectionDef(_c2);
|
|
210
255
|
i0.ɵɵelementStart(0, "div", 0)(1, "div", 1);
|
|
211
256
|
i0.ɵɵpipe(2, "async");
|
|
@@ -229,16 +274,18 @@ export class DrChatFormDropdownComponent {
|
|
|
229
274
|
} if (rf & 2) {
|
|
230
275
|
const _r1 = i0.ɵɵreference(7);
|
|
231
276
|
let tmp_1_0;
|
|
232
|
-
i0.ɵɵproperty("ngClass", i0.ɵɵpureFunction1(
|
|
277
|
+
i0.ɵɵproperty("ngClass", i0.ɵɵpureFunction1(16, _c3, ctx.isUploadingFiles));
|
|
233
278
|
i0.ɵɵadvance(1);
|
|
234
|
-
i0.ɵɵproperty("ngClass", i0.ɵɵpureFunction2(
|
|
279
|
+
i0.ɵɵproperty("ngClass", i0.ɵɵpureFunction2(18, _c4, ctx.inputFocus, !!(ctx.message == null ? null : (tmp_1_0 = ctx.message.trim()) == null ? null : tmp_1_0.length) || !!i0.ɵɵpipeBind1(2, 12, ctx.droppedFiles$).length));
|
|
235
280
|
i0.ɵɵadvance(2);
|
|
236
|
-
i0.ɵɵproperty("ngIf", i0.ɵɵpipeBind1(4,
|
|
281
|
+
i0.ɵɵproperty("ngIf", i0.ɵɵpipeBind1(4, 14, ctx.droppedFiles$).length);
|
|
237
282
|
i0.ɵɵadvance(3);
|
|
238
283
|
i0.ɵɵstyleMap(ctx.getTextAreaHeight(_r1));
|
|
239
284
|
i0.ɵɵpropertyInterpolate("placeholder", ctx.fileOver ? ctx.dropFilePlaceholder : ctx.messagePlaceholder);
|
|
240
285
|
i0.ɵɵproperty("ngModel", ctx.message)("rows", 1);
|
|
241
|
-
i0.ɵɵadvance(
|
|
286
|
+
i0.ɵɵadvance(5);
|
|
287
|
+
i0.ɵɵproperty("accept", ctx.acceptFormatFiles);
|
|
288
|
+
i0.ɵɵadvance(2);
|
|
242
289
|
i0.ɵɵproperty("ngIf", !ctx.waitForReply);
|
|
243
290
|
i0.ɵɵadvance(1);
|
|
244
291
|
i0.ɵɵproperty("ngIf", ctx.waitForReply && !ctx.showDotFlashing);
|
|
@@ -248,13 +295,19 @@ export class DrChatFormDropdownComponent {
|
|
|
248
295
|
}
|
|
249
296
|
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(DrChatFormDropdownComponent, [{
|
|
250
297
|
type: Component,
|
|
251
|
-
args: [{ selector: 'dr-chat-form-dropdown', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"message-row\" [ngClass]=\"{ 'message-row_loading': isUploadingFiles }\">\n <div\n class=\"message-row__input\"\n [ngClass]=\"{\n 'message-row__input--focused': inputFocus,\n 'message-row__input--filled': !!message?.trim()?.length || !!(droppedFiles$ | async).length,\n }\">\n <dr-chat-dropped-files\n *ngIf=\"(droppedFiles$ | async).length\"\n [files]=\"droppedFiles$ | async\"\n [isRemovable]=\"true\"\n [maxLengthText]=\"15\"\n (removeFileEvent)=\"removeFile($event)\"></dr-chat-dropped-files>\n\n <div class=\"message-row__input-textarea-wrap\">\n <textarea\n #textAreaElement\n
|
|
298
|
+
args: [{ selector: 'dr-chat-form-dropdown', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"message-row\" [ngClass]=\"{ 'message-row_loading': isUploadingFiles }\">\n <div\n class=\"message-row__input\"\n [ngClass]=\"{\n 'message-row__input--focused': inputFocus,\n 'message-row__input--filled': !!message?.trim()?.length || !!(droppedFiles$ | async).length,\n }\">\n <dr-chat-dropped-files\n *ngIf=\"(droppedFiles$ | async).length\"\n [files]=\"droppedFiles$ | async\"\n [isRemovable]=\"true\"\n [maxLengthText]=\"15\"\n (removeFileEvent)=\"removeFile($event)\"></dr-chat-dropped-files>\n\n <div class=\"message-row__input-textarea-wrap\">\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 (keydown.enter)=\"sendMessage($event)\">\n </textarea>\n </div>\n <div class=\"message-row__btns\">\n <ng-content></ng-content>\n <input #fileInput type=\"file\" [accept]=\"acceptFormatFiles\" hidden multiple (change)=\"onFileSelected($event)\" />\n <i *ngIf=\"!waitForReply\" (click)=\"sendMessage($event)\" class=\"dr-icon-send-arrow-up send-button\"></i>\n\n <i *ngIf=\"waitForReply && !showDotFlashing\" class=\"dr-icon-stop abort-button\" (click)=\"abortMessage()\"></i>\n <dr-dot-flashing *ngIf=\"waitForReply && showDotFlashing\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n </div>\n <ng-content select=\"[dropItem]\"></ng-content>\n </div>\n</div>\n", styles: [":host::ng-deep{--disabled-bg: linear-gradient(270.95deg, rgba(130, 210, 240, .2) 10.98%, rgba(146, 146, 255, .2) 93.38%);display:flex;flex-direction:column;align-items:center;padding:0}:host::ng-deep .message-row{display:flex;justify-content:center;width:100%;padding:0 0 21px;max-width:956px}:host::ng-deep .message-row__input{flex-direction:column;background-color:#fff;position:relative;display:flex;align-items:center;flex-grow:1;padding:14px 8px 8px 15px;height:auto;overflow:visible;min-width:265px;border-radius:24px;border:1.5px solid transparent;box-shadow:0 2px 16px -10px #603cff29;transition:.35s ease;will-change:box-shadow}:host::ng-deep .message-row__input .abort-button,:host::ng-deep .message-row__input .send-button{width:32px;height:32px;display:flex;align-items:center;justify-content:center;cursor:pointer;font-size:24px;border-radius:100px;color:#fff;transition:.15s ease-in-out;background:linear-gradient(266deg,#6969ff 25.2%,#4eb7df 90.24%) border-box}:host::ng-deep .message-row__input .send-button{background:var(--disabled-bg);pointer-events:none}:host::ng-deep .message-row__input--focused{box-shadow:8px 8px 16px #6969ff40,-4px -4px 8px #40b6e340,8px 8px 60px #00000040;height:auto!important;background:#fff}:host::ng-deep .message-row__input--filled{box-shadow:8px 8px 16px #6969ff40,-4px -4px 8px #40b6e340,8px 8px 60px #00000040;background:#fff}:host::ng-deep .message-row__input--filled .send-button{background:linear-gradient(266deg,#6969ff 25.2%,#4eb7df 90.24%) border-box;pointer-events:all}:host::ng-deep .message-row__input .wait-reply-dot-flashing{display:flex;align-items:center;width:32px;height:32px}:host::ng-deep .message-row__input:before{content:\"\";position:absolute;inset:-3px;background:linear-gradient(266.3deg,#6969ff 25.2%,#4eb7df 90.24%);border-radius:25px;z-index:-1}:host::ng-deep .message-row__input textarea{color:#333;width:100%;outline:none;min-height:30px;flex-grow:1;resize:none;padding:4px 48px 4px 4px;margin:auto;border:none;border-radius:22.5px;font-size:14px;line-height:24px;font-weight:400}:host::ng-deep .message-row__input textarea:focus{border:none}:host::ng-deep .message-row__input textarea::placeholder{color:#9ea1aa}:host::ng-deep .message-row__btns{display:flex;justify-content:space-between;width:100%;align-items:center}:host::ng-deep .message-row__input-textarea-wrap{display:flex;width:100%;position:relative}:host::ng-deep dr-chat-dropped-files{margin-right:auto}:host::ng-deep dr-chat-dropped-files .dropped-files{padding:0}:host::ng-deep dr-chat-dropped-files .dropped-files__item{margin:0}:host::ng-deep .message-row_loading .send-button{background:var(--disabled-bg);pointer-events:none}:host::ng-deep .message-row_loading .dropped-files__item{opacity:.5;pointer-events:none}\n"] }]
|
|
252
299
|
}], function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.DomSanitizer }]; }, { textAreaElementRef: [{
|
|
253
300
|
type: ViewChild,
|
|
254
301
|
args: ['textAreaElement']
|
|
255
302
|
}], fileInput: [{
|
|
256
303
|
type: ViewChild,
|
|
257
304
|
args: ['fileInput']
|
|
305
|
+
}], acceptFormatFiles: [{
|
|
306
|
+
type: Input
|
|
307
|
+
}], maxFileSizeBytes: [{
|
|
308
|
+
type: Input
|
|
309
|
+
}], maxFilesCount: [{
|
|
310
|
+
type: Input
|
|
258
311
|
}], isUploadingFiles: [{
|
|
259
312
|
type: Input
|
|
260
313
|
}], message: [{
|
|
@@ -275,6 +328,8 @@ export class DrChatFormDropdownComponent {
|
|
|
275
328
|
type: Output
|
|
276
329
|
}], abort: [{
|
|
277
330
|
type: Output
|
|
331
|
+
}], filesRejected: [{
|
|
332
|
+
type: Output
|
|
278
333
|
}], inputChange: [{
|
|
279
334
|
type: Output
|
|
280
335
|
}], fileOver: [{
|
|
@@ -290,4 +345,4 @@ export class DrChatFormDropdownComponent {
|
|
|
290
345
|
type: HostListener,
|
|
291
346
|
args: ['dragleave', ['$event']]
|
|
292
347
|
}] }); })();
|
|
293
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dr-chat-form-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.ts","../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Input,\n    Output,\n    ViewChild,\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { IFIle } from '../../models/chat';\nimport { BehaviorSubject } from 'rxjs';\n\n@Component({\n    selector: 'dr-chat-form-dropdown',\n    templateUrl: './dr-chat-form-dropdown.component.html',\n    styleUrls: ['./dr-chat-form-dropdown.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DrChatFormDropdownComponent {\n    @ViewChild('textAreaElement') textAreaElementRef: ElementRef;\n    @ViewChild('fileInput') fileInput: ElementRef;\n    _textareaInitialHeight = true;\n    inputFocus = false;\n    inputHover = false;\n\n    droppedFiles$ = new BehaviorSubject<IFIle[]>([]);\n\n    @Input() isUploadingFiles = false;\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     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() showDotFlashing = false;\n\n    /**\n     *\n     * @type {EventEmitter<{ message: string, files: IFile[] }>}\n     */\n    @Output() send = new EventEmitter<{ message: string; files: IFIle[] }>();\n    @Output() uploadFiles = new EventEmitter<IFIle[]>();\n    @Output() abort = new EventEmitter<void>();\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(\n        protected cdr: ChangeDetectorRef,\n        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            const files = event.dataTransfer?.files;\n            if (files) {\n                this.saveFiles(files);\n            }\n        }\n    }\n\n    removeFile(file: IFIle) {\n        const droppedFiles = this.droppedFiles$.value;\n        const index = droppedFiles.indexOf(file);\n        if (index >= 0) {\n            droppedFiles.splice(index, 1);\n            this.droppedFiles$.next(droppedFiles);\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($event: Event | KeyboardEvent) {\n        if (!$event || !($event as KeyboardEvent).shiftKey) {\n            $event?.preventDefault();\n            $event?.stopPropagation();\n            if (this.waitForReply || this.isUploadingFiles) {\n                return;\n            } else if (this.droppedFiles$.value.length || String(this.message).trim().length) {\n                this._textareaInitialHeight = true;\n                this.send.emit({ message: this.message, files: this.droppedFiles$.value });\n                this.message = '';\n                this.droppedFiles$.next([]);\n                this.cdr.markForCheck();\n            }\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        if (this._textareaInitialHeight) {\n            textAreaElement.style.height = '30px';\n        } else {\n            textAreaElement.style.height = 'auto';\n            textAreaElement.style.height = textAreaElement.scrollHeight + 'px';\n        }\n\n        return `${textAreaElement.style.height}`;\n    }\n\n    onFileSelected(event: Event) {\n        const input = event.target as HTMLInputElement;\n        this.fileOver = false;\n        const files = input.files;\n        if (files && files.length) {\n            this.saveFiles(Array.from(files));\n        }\n        this.fileInput.nativeElement.value = '';\n    }\n\n    private async saveFiles(files: Array<File>) {\n        const uploadedFiles = [];\n        for (const file of files) {\n            let res = file as IFIle;\n            res.data = (await this.base64Convert(res)) as string;\n            this.droppedFiles$.next([...this.droppedFiles$.value, res]);\n            uploadedFiles.push(res);\n            this.cdr.markForCheck();\n            this.cdr.detectChanges();\n        }\n        this.uploadFiles.emit(uploadedFiles.map((item) => ({ data: item.data, name: item.name })));\n    }\n\n    private base64Convert(file: any) {\n        return new Promise((resolve, reject) => {\n            const reader = new FileReader();\n            reader.onload = () => resolve(reader.result as string);\n            reader.onerror = (error) => reject(error);\n            reader.readAsDataURL(file);\n        });\n    }\n}\n","<div class=\"message-row\" [ngClass]=\"{ 'message-row_loading': isUploadingFiles }\">\n    <div\n        class=\"message-row__input\"\n        [ngClass]=\"{\n            'message-row__input--focused': inputFocus,\n            'message-row__input--filled': !!message?.trim()?.length || !!(droppedFiles$ | async).length,\n        }\">\n        <dr-chat-dropped-files\n            *ngIf=\"(droppedFiles$ | async).length\"\n            [files]=\"droppedFiles$ | async\"\n            [isRemovable]=\"true\"\n            [maxLengthText]=\"15\"\n            (removeFileEvent)=\"removeFile($event)\"></dr-chat-dropped-files>\n\n        <div class=\"message-row__input-textarea-wrap\">\n            <textarea\n                #textAreaElement\n                data-analytics=\"ai-chat-actions-send-message\"\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                (keydown.enter)=\"sendMessage($event)\">\n            </textarea>\n        </div>\n        <div class=\"message-row__btns\">\n            <ng-content></ng-content>\n            <input #fileInput type=\"file\" hidden multiple (change)=\"onFileSelected($event)\" />\n            <i\n                *ngIf=\"!waitForReply\"\n                (click)=\"sendMessage($event)\"\n                data-analytics=\"ai-chat-actions-send-message\"\n                class=\"dr-icon-send-arrow-up send-button\"></i>\n\n            <i\n                *ngIf=\"waitForReply && !showDotFlashing\"\n                data-analytics=\"ai-chat-actions-abort-message\"\n                class=\"dr-icon-stop abort-button\"\n                (click)=\"abortMessage()\"></i>\n            <dr-dot-flashing *ngIf=\"waitForReply && showDotFlashing\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n        </div>\n        <ng-content select=\"[dropItem]\"></ng-content>\n    </div>\n</div>\n"]}
|
|
348
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dr-chat-form-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.ts","../../../../../../projects/datarailsshared/src/lib/dr-chat/dr-chat-form-dropdown/dr-chat-form-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Input,\n    Output,\n    ViewChild,\n} from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { IFIle } from '../../models/chat';\nimport { BehaviorSubject } from 'rxjs';\n\n@Component({\n    selector: 'dr-chat-form-dropdown',\n    templateUrl: './dr-chat-form-dropdown.component.html',\n    styleUrls: ['./dr-chat-form-dropdown.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DrChatFormDropdownComponent {\n    @ViewChild('textAreaElement') textAreaElementRef: ElementRef;\n    @ViewChild('fileInput') fileInput: ElementRef;\n    _textareaInitialHeight = true;\n    inputFocus = false;\n    inputHover = false;\n\n    droppedFiles$ = new BehaviorSubject<IFIle[]>([]);\n\n    @Input() acceptFormatFiles: string;\n    @Input() maxFileSizeBytes: number;\n    @Input() maxFilesCount: number;\n\n    @Input() isUploadingFiles = false;\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     * Parameter to check is send message function available\n     *\n     * @type {boolean}\n     */\n    @Input() showDotFlashing = false;\n\n    /**\n     *\n     * @type {EventEmitter<{ message: string, files: IFile[] }>}\n     */\n    @Output() send = new EventEmitter<{ message: string; files: IFIle[] }>();\n    @Output() uploadFiles = new EventEmitter<IFIle[]>();\n    @Output() abort = new EventEmitter<void>();\n    @Output() filesRejected = new EventEmitter<{ oversize: string[]; unsupported: string[]; maxExceeded: boolean }>();\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(\n        protected cdr: ChangeDetectorRef,\n        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            const files = event.dataTransfer?.files;\n            if (files) {\n                this.saveFiles(Array.from(files));\n            }\n        }\n    }\n\n    removeFile(file: IFIle) {\n        const droppedFiles = this.droppedFiles$.value;\n        const index = droppedFiles.indexOf(file);\n        if (index >= 0) {\n            droppedFiles.splice(index, 1);\n            this.droppedFiles$.next(droppedFiles);\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($event: Event | KeyboardEvent) {\n        if (!$event || !($event as KeyboardEvent).shiftKey) {\n            $event?.preventDefault();\n            $event?.stopPropagation();\n            if (this.waitForReply || this.isUploadingFiles) {\n                return;\n            } else if (this.droppedFiles$.value.length || String(this.message).trim().length) {\n                this._textareaInitialHeight = true;\n                this.send.emit({ message: this.message, files: this.droppedFiles$.value });\n                this.message = '';\n                this.droppedFiles$.next([]);\n                this.cdr.markForCheck();\n            }\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        if (this._textareaInitialHeight) {\n            textAreaElement.style.height = '30px';\n        } else {\n            textAreaElement.style.height = 'auto';\n            textAreaElement.style.height = textAreaElement.scrollHeight + 'px';\n        }\n\n        return `${textAreaElement.style.height}`;\n    }\n\n    onFileSelected(event: Event) {\n        const input = event.target as HTMLInputElement;\n        this.fileOver = false;\n        const files = input.files;\n        if (files && files.length) {\n            this.saveFiles(Array.from(files));\n        }\n        this.fileInput.nativeElement.value = '';\n    }\n\n    private async saveFiles(files: Array<File>) {\n        const uploadedFiles: IFIle[] = [];\n        const tooLargeFiles: string[] = [];\n        const unsupportedFiles: string[] = [];\n        const allowedExtensions = this.getAllowedExtensions();\n        let maxExceeded = false;\n\n        const remainingSlotsInit = Math.max(0, this.maxFilesCount - this.droppedFiles$.value.length);\n        if (files.length > remainingSlotsInit) {\n            this.filesRejected.emit({ oversize: [], unsupported: [], maxExceeded: true });\n            return;\n        }\n\n        let remainingSlots = remainingSlotsInit;\n\n        for (const file of files) {\n            if (remainingSlots <= 0) {\n                maxExceeded = true;\n                continue;\n            }\n\n            const fileExtension = this.getFileExtension(file.name);\n            const isAllowedByType = !allowedExtensions || allowedExtensions.has(fileExtension);\n            if (!isAllowedByType) {\n                unsupportedFiles.push(file.name);\n                continue;\n            }\n            if (file.size > this.maxFileSizeBytes) {\n                tooLargeFiles.push(file.name);\n                continue;\n            }\n\n            const res = file as IFIle;\n            res.data = (await this.base64Convert(res)) as string;\n            this.droppedFiles$.next([...this.droppedFiles$.value, res]);\n            uploadedFiles.push(res);\n            remainingSlots--;\n            this.cdr.markForCheck();\n            this.cdr.detectChanges();\n        }\n\n        if (tooLargeFiles.length || unsupportedFiles.length || maxExceeded) {\n            this.filesRejected.emit({ oversize: tooLargeFiles, unsupported: unsupportedFiles, maxExceeded });\n        }\n\n        if (uploadedFiles.length) {\n            this.uploadFiles.emit(uploadedFiles.map((item) => ({ data: item.data, name: item.name })));\n        }\n    }\n\n    private getAllowedExtensions(): Set<string> | null {\n        if (!this.acceptFormatFiles || !this.acceptFormatFiles.trim()) return null;\n        const parts = this.acceptFormatFiles\n            .split(',')\n            .map((p) => p.trim().toLowerCase())\n            .filter(Boolean)\n            .map((p) => (p.startsWith('.') ? p.slice(1) : p));\n        return parts.length ? new Set(parts) : null;\n    }\n\n    private getFileExtension(fileName: string): string {\n        const idx = fileName.lastIndexOf('.');\n        return idx >= 0 ? fileName.slice(idx + 1).toLowerCase() : '';\n    }\n\n    private base64Convert(file: any) {\n        return new Promise((resolve, reject) => {\n            const reader = new FileReader();\n            reader.onload = () => resolve(reader.result as string);\n            reader.onerror = (error) => reject(error);\n            reader.readAsDataURL(file);\n        });\n    }\n}\n","<div class=\"message-row\" [ngClass]=\"{ 'message-row_loading': isUploadingFiles }\">\n    <div\n        class=\"message-row__input\"\n        [ngClass]=\"{\n            'message-row__input--focused': inputFocus,\n            'message-row__input--filled': !!message?.trim()?.length || !!(droppedFiles$ | async).length,\n        }\">\n        <dr-chat-dropped-files\n            *ngIf=\"(droppedFiles$ | async).length\"\n            [files]=\"droppedFiles$ | async\"\n            [isRemovable]=\"true\"\n            [maxLengthText]=\"15\"\n            (removeFileEvent)=\"removeFile($event)\"></dr-chat-dropped-files>\n\n        <div class=\"message-row__input-textarea-wrap\">\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                (keydown.enter)=\"sendMessage($event)\">\n            </textarea>\n        </div>\n        <div class=\"message-row__btns\">\n            <ng-content></ng-content>\n            <input #fileInput type=\"file\" [accept]=\"acceptFormatFiles\" hidden multiple (change)=\"onFileSelected($event)\" />\n            <i *ngIf=\"!waitForReply\" (click)=\"sendMessage($event)\" class=\"dr-icon-send-arrow-up send-button\"></i>\n\n            <i *ngIf=\"waitForReply && !showDotFlashing\" class=\"dr-icon-stop abort-button\" (click)=\"abortMessage()\"></i>\n            <dr-dot-flashing *ngIf=\"waitForReply && showDotFlashing\" class=\"wait-reply-dot-flashing\"></dr-dot-flashing>\n        </div>\n        <ng-content select=\"[dropItem]\"></ng-content>\n    </div>\n</div>\n"]}
|