@trixwell/ngx-parl 1.5.1 → 1.5.3

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/README.md CHANGED
@@ -76,8 +76,9 @@ assets/ngx-parl/...
76
76
  | messageList | ChatMessage[] | List of chat messages, user information |
77
77
  | messageUpdate | ChatMessage | Incoming message from external source (signal/subject/observable) |
78
78
  | messageAction | MessageActionEvent | Emits chat events: send, edit, delete |
79
+ | loadHistory | boolean | Use scroll for load history |
79
80
 
80
- Use the scrollToBottom() to control scrolling down.
81
+ - use the scrollToBottom() to control scrolling down.
81
82
 
82
83
  # Example Usage
83
84
 
@@ -122,6 +123,7 @@ export interface MessageActionEvent {
122
123
  <ngx-parl [header]="header()"
123
124
  [(messageList)]="messageList"
124
125
  [(messageUpdate)]="messageUpdate"
125
- [(messageAction)]="messageAction">
126
+ [(messageAction)]="messageAction"
127
+ [(isScrolledToTop)]="isScrolledToTop">
126
128
  </ngx-parl>
127
129
  ```
@@ -5,6 +5,7 @@ import { FormsModule } from '@angular/forms';
5
5
  import { MatMenu, MatMenuItem, MatMenuTrigger } from '@angular/material/menu';
6
6
  import * as i2 from '@ngneat/transloco';
7
7
  import { TranslocoPipe, TranslocoModule, provideTransloco } from '@ngneat/transloco';
8
+ import { InfiniteScrollDirective } from 'ngx-infinite-scroll';
8
9
  import * as i1 from '@angular/common/http';
9
10
  import { MatDialogContent, MatDialogTitle } from '@angular/material/dialog';
10
11
  import { MatProgressSpinner } from '@angular/material/progress-spinner';
@@ -133,10 +134,10 @@ class ChatMessageComponent {
133
134
  return message.type === this.messageType.Outgoing;
134
135
  }
135
136
  messageType = MessageType;
136
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
137
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.7", type: ChatMessageComponent, isStandalone: true, selector: "lib-chat-message", inputs: { currentMessage: { classPropertyName: "currentMessage", publicName: "currentMessage", isSignal: true, isRequired: true, transformFunction: null }, edit: { classPropertyName: "edit", publicName: "edit", isSignal: true, isRequired: false, transformFunction: null }, requestEdit: { classPropertyName: "requestEdit", publicName: "requestEdit", isSignal: true, isRequired: false, transformFunction: null }, requestDelete: { classPropertyName: "requestDelete", publicName: "requestDelete", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { edit: "editChange", requestEdit: "requestEditChange", requestDelete: "requestDeleteChange" }, ngImport: i0, template: "<div class=\"message\"\n [ngClass]=\"{\n 'message--outgoing': currentMessage().type === messageType.Outgoing,\n 'message--incoming': currentMessage().type === messageType.Incoming\n }\">\n <div class=\"message__avatar\">\n <img [ngSrc]=\"avatarSrc()\" width=\"36\" height=\"36\" alt=\"avatar\"/>\n </div>\n\n <div class=\"message__body\"\n #menuTrigger=\"matMenuTrigger\"\n [matMenuTriggerFor]=\"messageMenu\"\n (contextmenu)=\"openContextMenu($event, menuTrigger)\"\n (keydown.shift.F10)=\"openContextMenu($event, menuTrigger)\">\n\n @if (attachments().length) {\n <div class=\"message__attachments\">\n @for (file of attachments(); track file) {\n <img [src]=\"file\"\n alt=\"attachment\"\n class=\"message__image\"\n loading=\"lazy\"/>\n }\n </div>\n }\n\n <div class=\"message__bubble\" tabindex=\"0\">\n <div class=\"message__text\">{{ currentMessage().content }}</div>\n\n <div class=\"message__meta\">\n <time class=\"message__time\">{{ currentMessage().cr_time | date:'HH:mm' }}</time>\n @if (currentMessage().type === messageType.Outgoing) {\n @if (currentMessage().checked) {\n <img ngSrc=\"assets/ngx-parl/icons/checked-message.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n } @else {\n <img ngSrc=\"assets/ngx-parl/icons/no-check.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n }\n }\n </div>\n </div>\n </div>\n</div>\n\n<mat-menu #messageMenu=\"matMenu\" xPosition=\"before\" yPosition=\"below\" class=\"message__menu\">\n @if (currentMessage().type === messageType.Outgoing) {\n <button mat-menu-item (click)=\"editMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.edit' | transloco }}</span>\n </button>\n }\n @if (canDelete(currentMessage())) {\n <button mat-menu-item (click)=\"deleteMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/trash.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.remove' | transloco }}</span>\n </button>\n }\n</mat-menu>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.message{display:flex;gap:8px;align-items:flex-end;max-width:80%;animation:pop .12s ease-out}.message__avatar img{display:block;border-radius:1000px}.message__body{padding:8px 12px;min-width:0;font:400 16px/20px inter,sans-serif;border-radius:16px}.message__bubble{display:grid;grid-template-columns:1fr auto;column-gap:8px;align-items:end;max-width:100%;word-break:break-word;white-space:pre-wrap}.message__text{min-width:0}.message__meta{display:inline-flex;align-items:center;gap:4px;-webkit-user-select:none;user-select:none;line-height:1;margin:0}.message__time{font:400 12px/16px inter,sans-serif}.message__attachments{display:grid;gap:8px;justify-content:start;width:100%;margin-bottom:8px}.message__attachments:has(:only-child){grid-template-columns:repeat(1,minmax(0,140px))}.message__attachments:has(:nth-child(2):last-child){grid-template-columns:repeat(2,minmax(0,140px))}.message__attachments:has(:nth-child(3)){grid-template-columns:repeat(3,minmax(0,140px))}.message__image{display:block;width:100%;height:140px;object-fit:cover;border-radius:12px;aspect-ratio:1/1}.message__bubble:after{content:\"\";display:block;clear:both}.message__time{font:400 12px/16px inter,sans-serif;min-width:30px}.message__icon{width:14px;height:14px;vertical-align:middle}.message__icon--menu{width:20px;height:20px}.message--incoming{margin-right:auto}.message--incoming .message__body{border-bottom-right-radius:16px}.message--incoming .message__bubble{border-top-left-radius:4px}.message--incoming .message__time{color:#adb5bd}.message--outgoing{margin-left:auto;flex-direction:row-reverse}.message--outgoing .message__body{border-bottom-left-radius:16px}.message--outgoing .message__bubble{border-top-right-radius:4px}.message--outgoing .message__time{color:#e9ecef}@keyframes pop{0%{transform:translateY(2px);opacity:0}to{transform:translateY(0);opacity:1}}::ng-deep .mat-mdc-menu-content{background:#fff;width:160px;border-radius:8px;gap:4px;border:1px solid #E1E7F8;padding:8px 4px;box-shadow:0 2px 4px #0000001a;overflow:hidden;font:400 14px/18px inter,sans-serif}::ng-deep .mat-mdc-menu-item-text{font:400 14px/18px inter,sans-serif!important;font-weight:500!important;color:#343a40!important;display:flex;justify-content:flex-start;align-items:center;gap:4px}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "component", type: MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "pipe", type: DatePipe, name: "date" }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] });
137
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
138
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.16", type: ChatMessageComponent, isStandalone: true, selector: "lib-chat-message", inputs: { currentMessage: { classPropertyName: "currentMessage", publicName: "currentMessage", isSignal: true, isRequired: true, transformFunction: null }, edit: { classPropertyName: "edit", publicName: "edit", isSignal: true, isRequired: false, transformFunction: null }, requestEdit: { classPropertyName: "requestEdit", publicName: "requestEdit", isSignal: true, isRequired: false, transformFunction: null }, requestDelete: { classPropertyName: "requestDelete", publicName: "requestDelete", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { edit: "editChange", requestEdit: "requestEditChange", requestDelete: "requestDeleteChange" }, ngImport: i0, template: "<div class=\"message\"\n [ngClass]=\"{\n 'message--outgoing': currentMessage().type === messageType.Outgoing,\n 'message--incoming': currentMessage().type === messageType.Incoming\n }\">\n <div class=\"message__avatar\">\n <img [ngSrc]=\"avatarSrc()\" width=\"36\" height=\"36\" alt=\"avatar\"/>\n </div>\n\n <div class=\"message__body\"\n #menuTrigger=\"matMenuTrigger\"\n [matMenuTriggerFor]=\"messageMenu\"\n (contextmenu)=\"openContextMenu($event, menuTrigger)\"\n (keydown.shift.F10)=\"openContextMenu($event, menuTrigger)\">\n\n @if (attachments().length) {\n <div class=\"message__attachments\">\n @for (file of attachments(); track file) {\n <img [src]=\"file\"\n alt=\"attachment\"\n class=\"message__image\"\n loading=\"lazy\"/>\n }\n </div>\n }\n\n <div class=\"message__bubble\" tabindex=\"0\">\n <div class=\"message__text\">{{ currentMessage().content }}</div>\n\n <div class=\"message__meta\">\n <time class=\"message__time\">{{ currentMessage().cr_time | date:'HH:mm' }}</time>\n @if (currentMessage().type === messageType.Outgoing) {\n @if (currentMessage().checked) {\n <img ngSrc=\"assets/ngx-parl/icons/checked-message.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n } @else {\n <img ngSrc=\"assets/ngx-parl/icons/no-check.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n }\n }\n </div>\n </div>\n </div>\n</div>\n\n<mat-menu #messageMenu=\"matMenu\" xPosition=\"before\" yPosition=\"below\" class=\"message__menu\">\n @if (currentMessage().type === messageType.Outgoing) {\n <button mat-menu-item (click)=\"editMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.edit' | transloco }}</span>\n </button>\n }\n @if (canDelete(currentMessage())) {\n <button mat-menu-item (click)=\"deleteMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/trash.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.remove' | transloco }}</span>\n </button>\n }\n</mat-menu>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.message{display:flex;gap:8px;align-items:flex-end;max-width:80%;animation:pop .12s ease-out}.message__avatar img{display:block;border-radius:1000px}.message__body{padding:8px 12px;min-width:0;font:400 16px/20px inter,sans-serif;border-radius:16px}.message__bubble{display:grid;grid-template-columns:1fr auto;column-gap:8px;align-items:end;max-width:100%;word-break:break-word;white-space:pre-wrap}.message__text{min-width:0}.message__meta{display:inline-flex;align-items:center;gap:4px;-webkit-user-select:none;user-select:none;line-height:1;margin:0}.message__time{font:400 12px/16px inter,sans-serif}.message__attachments{display:grid;gap:8px;justify-content:start;width:100%;margin-bottom:8px}.message__attachments:has(:only-child){grid-template-columns:repeat(1,minmax(0,140px))}.message__attachments:has(:nth-child(2):last-child){grid-template-columns:repeat(2,minmax(0,140px))}.message__attachments:has(:nth-child(3)){grid-template-columns:repeat(3,minmax(0,140px))}.message__image{display:block;width:100%;height:140px;object-fit:cover;border-radius:12px;aspect-ratio:1/1}.message__bubble:after{content:\"\";display:block;clear:both}.message__time{font:400 12px/16px inter,sans-serif;min-width:30px}.message__icon{width:14px;height:14px;vertical-align:middle}.message__icon--menu{width:20px;height:20px}.message--incoming{margin-right:auto}.message--incoming .message__body{border-bottom-right-radius:16px}.message--incoming .message__bubble{border-top-left-radius:4px}.message--incoming .message__time{color:#adb5bd}.message--outgoing{margin-left:auto;flex-direction:row-reverse}.message--outgoing .message__body{border-bottom-left-radius:16px}.message--outgoing .message__bubble{border-top-right-radius:4px}.message--outgoing .message__time{color:#e9ecef}@keyframes pop{0%{transform:translateY(2px);opacity:0}to{transform:translateY(0);opacity:1}}::ng-deep .mat-mdc-menu-content{background:#fff;width:160px;border-radius:8px;gap:4px;border:1px solid #E1E7F8;padding:8px 4px;box-shadow:0 2px 4px #0000001a;overflow:hidden;font:400 14px/18px inter,sans-serif}::ng-deep .mat-mdc-menu-item-text{font:400 14px/18px inter,sans-serif!important;font-weight:500!important;color:#343a40!important;display:flex;justify-content:flex-start;align-items:center;gap:4px}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "component", type: MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "pipe", type: DatePipe, name: "date" }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] });
138
139
  }
139
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatMessageComponent, decorators: [{
140
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatMessageComponent, decorators: [{
140
141
  type: Component,
141
142
  args: [{ selector: 'lib-chat-message', imports: [
142
143
  NgClass,
@@ -173,97 +174,137 @@ class UtilsService {
173
174
  const seconds = String(d.getSeconds()).padStart(2, '0');
174
175
  return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}`;
175
176
  }
176
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: UtilsService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
177
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: UtilsService, providedIn: 'root' });
177
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: UtilsService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
178
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: UtilsService, providedIn: 'root' });
178
179
  }
179
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: UtilsService, decorators: [{
180
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: UtilsService, decorators: [{
180
181
  type: Injectable,
181
182
  args: [{
182
183
  providedIn: 'root'
183
184
  }]
184
185
  }], ctorParameters: () => [{ type: i1.HttpClient }] });
185
186
 
186
- class ChatStartDayPipe {
187
+ class ToggleDisplayChatStartDayPipe {
187
188
  utils;
188
189
  transloco;
189
190
  constructor(utils, transloco) {
190
191
  this.utils = utils;
191
192
  this.transloco = transloco;
192
193
  }
193
- transform(value, format = 'd MMMM') {
194
- if (!value) {
195
- return '';
196
- }
194
+ transform(message, messages, i) {
197
195
  const locale = this.utils.langToLocale(this.transloco.getActiveLang());
198
196
  const datePipe = new DatePipe(locale);
199
- const valueDate = new Date(value);
200
- const today = new Date();
201
- const isToday = datePipe.transform(valueDate, 'shortDate') === datePipe.transform(today, 'shortDate');
202
- return isToday
203
- ? (locale.startsWith('uk') ? 'Сьогодні' : 'Today')
204
- : (datePipe.transform(valueDate, format) ?? '');
197
+ const prev = i > 0 ? messages[i - 1] : undefined;
198
+ const currDay = datePipe.transform(new Date(message.cr_time), 'shortDate');
199
+ const prevDay = prev ? datePipe.transform(new Date(prev.cr_time), 'shortDate') : undefined;
200
+ return prev ? currDay !== prevDay : true;
205
201
  }
206
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatStartDayPipe, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Pipe });
207
- static ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "20.3.7", ngImport: i0, type: ChatStartDayPipe, isStandalone: true, name: "chatStartDay" });
202
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ToggleDisplayChatStartDayPipe, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Pipe });
203
+ static ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "20.3.16", ngImport: i0, type: ToggleDisplayChatStartDayPipe, isStandalone: true, name: "toggleDisplayChatStartDay" });
208
204
  }
209
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatStartDayPipe, decorators: [{
205
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ToggleDisplayChatStartDayPipe, decorators: [{
210
206
  type: Pipe,
211
207
  args: [{
212
- name: 'chatStartDay'
208
+ name: 'toggleDisplayChatStartDay'
213
209
  }]
214
210
  }], ctorParameters: () => [{ type: UtilsService }, { type: i2.TranslocoService }] });
215
211
 
216
- class ToggleDisplayChatStartDayPipe {
212
+ class ChatStartDayPipe {
217
213
  utils;
218
214
  transloco;
219
215
  constructor(utils, transloco) {
220
216
  this.utils = utils;
221
217
  this.transloco = transloco;
222
218
  }
223
- transform(message, messages, i) {
219
+ transform(value, format = 'd MMMM') {
220
+ if (!value) {
221
+ return '';
222
+ }
224
223
  const locale = this.utils.langToLocale(this.transloco.getActiveLang());
225
224
  const datePipe = new DatePipe(locale);
226
- const prev = i > 0 ? messages[i - 1] : undefined;
227
- const currDay = datePipe.transform(new Date(message.cr_time), 'shortDate');
228
- const prevDay = prev ? datePipe.transform(new Date(prev.cr_time), 'shortDate') : undefined;
229
- return prev ? currDay !== prevDay : true;
225
+ const valueDate = new Date(value);
226
+ const today = new Date();
227
+ const isToday = datePipe.transform(valueDate, 'shortDate') === datePipe.transform(today, 'shortDate');
228
+ return isToday
229
+ ? (locale.startsWith('uk') ? 'Сьогодні' : 'Today')
230
+ : (datePipe.transform(valueDate, format) ?? '');
230
231
  }
231
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ToggleDisplayChatStartDayPipe, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Pipe });
232
- static ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "20.3.7", ngImport: i0, type: ToggleDisplayChatStartDayPipe, isStandalone: true, name: "toggleDisplayChatStartDay" });
232
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatStartDayPipe, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Pipe });
233
+ static ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "20.3.16", ngImport: i0, type: ChatStartDayPipe, isStandalone: true, name: "chatStartDay" });
233
234
  }
234
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ToggleDisplayChatStartDayPipe, decorators: [{
235
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatStartDayPipe, decorators: [{
235
236
  type: Pipe,
236
237
  args: [{
237
- name: 'toggleDisplayChatStartDay'
238
+ name: 'chatStartDay'
238
239
  }]
239
240
  }], ctorParameters: () => [{ type: UtilsService }, { type: i2.TranslocoService }] });
240
241
 
241
242
  class ChatFlowComponent {
242
243
  flowRef;
243
244
  scrollToBottomTrigger = model(0, ...(ngDevMode ? [{ debugName: "scrollToBottomTrigger" }] : []));
245
+ loadHistory = model(false, ...(ngDevMode ? [{ debugName: "loadHistory" }] : []));
244
246
  messageListInput = model.required(...(ngDevMode ? [{ debugName: "messageListInput" }] : []));
245
247
  messageList = computed(() => this.messageListInput(), ...(ngDevMode ? [{ debugName: "messageList" }] : []));
246
248
  selectedForEdit = model.required(...(ngDevMode ? [{ debugName: "selectedForEdit" }] : []));
247
249
  requestDelete = model(null, ...(ngDevMode ? [{ debugName: "requestDelete" }] : []));
248
250
  viewInitialized = false;
251
+ previousScrollHeight = 0;
252
+ previousMessageCount = 0;
253
+ isUserAtBottom = true;
249
254
  constructor() {
255
+ /** ✅ Реакція на зміну списку */
250
256
  effect(() => {
251
- const length = this.messageList().length;
252
- if (!this.viewInitialized || length === 0) {
257
+ const messages = this.messageList();
258
+ if (!this.viewInitialized) {
259
+ this.previousMessageCount = messages.length;
253
260
  return;
254
261
  }
255
- queueMicrotask(() => this.scrollToBottomSmooth());
262
+ const addedMessages = messages.length > this.previousMessageCount;
263
+ if (addedMessages && !this.isUserAtBottom) {
264
+ this.restoreScrollAfterHistoryPrepend();
265
+ }
266
+ if (addedMessages && this.isUserAtBottom) {
267
+ queueMicrotask(() => this.scrollToBottom());
268
+ }
269
+ this.previousMessageCount = messages.length;
256
270
  });
257
271
  effect(() => {
258
272
  this.scrollToBottomTrigger();
259
- if (this.viewInitialized) {
260
- queueMicrotask(() => this.scrollToBottomSmooth());
273
+ if (!this.viewInitialized) {
274
+ return;
261
275
  }
276
+ queueMicrotask(() => this.scrollToBottom());
262
277
  });
263
278
  }
264
279
  ngAfterViewInit() {
265
280
  this.viewInitialized = true;
266
- queueMicrotask(() => this.scrollToBottomSmooth());
281
+ const element = this.flowRef?.nativeElement;
282
+ if (!element) {
283
+ return;
284
+ }
285
+ this.isUserAtBottom = true;
286
+ element.addEventListener('scroll', () => {
287
+ this.previousScrollHeight = element.scrollHeight;
288
+ this.isUserAtBottom =
289
+ element.scrollTop + element.clientHeight >=
290
+ element.scrollHeight - 10;
291
+ });
292
+ queueMicrotask(() => this.scrollToBottom());
293
+ }
294
+ onScrollUp() {
295
+ this.loadHistory.set(true);
296
+ return this;
297
+ }
298
+ restoreScrollAfterHistoryPrepend() {
299
+ const element = this.flowRef?.nativeElement;
300
+ if (!element) {
301
+ return this;
302
+ }
303
+ queueMicrotask(() => {
304
+ const newHeight = element.scrollHeight;
305
+ element.scrollTop = newHeight - this.previousScrollHeight;
306
+ });
307
+ return this;
267
308
  }
268
309
  scrollToBottomSmooth() {
269
310
  const element = this.flowRef?.nativeElement;
@@ -276,6 +317,14 @@ class ChatFlowComponent {
276
317
  });
277
318
  return this;
278
319
  }
320
+ scrollToBottom() {
321
+ const element = this.flowRef?.nativeElement;
322
+ if (!element) {
323
+ return this;
324
+ }
325
+ element.scrollTop = element.scrollHeight;
326
+ return this;
327
+ }
279
328
  startEdit(message) {
280
329
  this.messageList().forEach(currentMessage => {
281
330
  if (currentMessage.id !== message.id && currentMessage.edit) {
@@ -326,24 +375,25 @@ class ChatFlowComponent {
326
375
  // return message.id;
327
376
  return `${message.chat_id}-${message.type}-${message.id}`;
328
377
  }
329
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatFlowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
330
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.7", type: ChatFlowComponent, isStandalone: true, selector: "app-chat-flow", inputs: { scrollToBottomTrigger: { classPropertyName: "scrollToBottomTrigger", publicName: "scrollToBottomTrigger", isSignal: true, isRequired: false, transformFunction: null }, messageListInput: { classPropertyName: "messageListInput", publicName: "messageListInput", isSignal: true, isRequired: true, transformFunction: null }, selectedForEdit: { classPropertyName: "selectedForEdit", publicName: "selectedForEdit", isSignal: true, isRequired: true, transformFunction: null }, requestDelete: { classPropertyName: "requestDelete", publicName: "requestDelete", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { scrollToBottomTrigger: "scrollToBottomTriggerChange", messageListInput: "messageListInputChange", selectedForEdit: "selectedForEditChange", requestDelete: "requestDeleteChange" }, viewQueries: [{ propertyName: "flowRef", first: true, predicate: ["chatFlowRef"], descendants: true }], ngImport: i0, template: "<div class=\"chat\">\n @if (messageList()) {\n <div class=\"chat__flow\" #chatFlowRef>\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <lib-chat-message [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n", styles: [":host{display:block;height:100%;min-height:0}.chat{background:#fff;height:100%;display:flex;flex-direction:column;min-height:0;width:100%}.chat__flow{flex:1 1 auto;min-height:0;overflow:auto;display:flex;flex-direction:column;gap:12px;padding:12px 16px;scroll-behavior:smooth;overscroll-behavior:contain}.chat__flow__empty{display:flex;gap:16px;flex-direction:column;align-items:center;justify-content:center;padding:72px 16px;height:100%}.chat__flow__empty .message__icon{width:56px;height:56px}.chat__flow__empty--header{width:285px;display:flex;gap:12px;flex-direction:column;align-items:center}.chat__flow__empty--title{margin:0;font:600 16px/20px inter,sans-serif;color:#212529}.chat__flow__empty--subscription{margin:0;font:400 16px/20px inter,sans-serif;color:#495057}.chat__start-day{display:flex;justify-content:center;background:#f8f9fa;color:#6c757d;font:400 12px/16px inter,sans-serif;width:80px;height:22px;margin:0 auto;border-radius:1000px;gap:8px;padding:4px 8px}.chat ::-webkit-scrollbar{width:6px}.chat ::-webkit-scrollbar-track{background:transparent}.chat ::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.chat ::-webkit-scrollbar-button{display:none}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "component", type: ChatMessageComponent, selector: "lib-chat-message", inputs: ["currentMessage", "edit", "requestEdit", "requestDelete"], outputs: ["editChange", "requestEditChange", "requestDeleteChange"] }, { kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "pipe", type: ChatStartDayPipe, name: "chatStartDay" }, { kind: "pipe", type: ToggleDisplayChatStartDayPipe, name: "toggleDisplayChatStartDay" }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] });
378
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatFlowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
379
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.16", type: ChatFlowComponent, isStandalone: true, selector: "app-chat-flow", inputs: { scrollToBottomTrigger: { classPropertyName: "scrollToBottomTrigger", publicName: "scrollToBottomTrigger", isSignal: true, isRequired: false, transformFunction: null }, loadHistory: { classPropertyName: "loadHistory", publicName: "loadHistory", isSignal: true, isRequired: false, transformFunction: null }, messageListInput: { classPropertyName: "messageListInput", publicName: "messageListInput", isSignal: true, isRequired: true, transformFunction: null }, selectedForEdit: { classPropertyName: "selectedForEdit", publicName: "selectedForEdit", isSignal: true, isRequired: true, transformFunction: null }, requestDelete: { classPropertyName: "requestDelete", publicName: "requestDelete", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { scrollToBottomTrigger: "scrollToBottomTriggerChange", loadHistory: "loadHistoryChange", messageListInput: "messageListInputChange", selectedForEdit: "selectedForEditChange", requestDelete: "requestDeleteChange" }, viewQueries: [{ propertyName: "flowRef", first: true, predicate: ["chatFlowRef"], descendants: true }], ngImport: i0, template: "<div class=\"chat\">\n @if (messageList()) {\n <div #chatFlowRef\n class=\"chat__flow\"\n infiniteScroll\n [scrollWindow]=\"false\"\n [infiniteScrollUpDistance]=\"1\"\n [infiniteScrollThrottle]=\"150\"\n (scrolledUp)=\"onScrollUp()\"\n >\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <div class=\"chat__message\">\n <lib-chat-message\n [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n </div>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n\n", styles: [":host{display:block;height:100%;min-height:0}.chat{background:#fff;height:100%;display:flex;flex-direction:column;min-height:0;width:100%}.chat__flow{flex:1 1 auto;min-height:0;overflow:auto;display:flex;flex-direction:column;gap:12px;padding:12px 16px;scroll-behavior:smooth;overscroll-behavior:contain}.chat__flow__empty{display:flex;gap:16px;flex-direction:column;align-items:center;justify-content:center;padding:72px 16px;height:100%}.chat__flow__empty .message__icon{width:56px;height:56px}.chat__flow__empty--header{width:285px;display:flex;gap:12px;flex-direction:column;align-items:center}.chat__flow__empty--title{margin:0;font:600 16px/20px inter,sans-serif;color:#212529}.chat__flow__empty--subscription{margin:0;font:400 16px/20px inter,sans-serif;color:#495057}.chat__start-day{display:flex;justify-content:center;background:#f8f9fa;color:#6c757d;font:400 12px/16px inter,sans-serif;width:80px;height:22px;margin:0 auto;border-radius:1000px;gap:8px;padding:4px 8px}.chat ::-webkit-scrollbar{width:6px}.chat ::-webkit-scrollbar-track{background:transparent}.chat ::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.chat ::-webkit-scrollbar-button{display:none}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "component", type: ChatMessageComponent, selector: "lib-chat-message", inputs: ["currentMessage", "edit", "requestEdit", "requestDelete"], outputs: ["editChange", "requestEditChange", "requestDeleteChange"] }, { kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "directive", type: InfiniteScrollDirective, selector: "[infiniteScroll], [infinite-scroll], [data-infinite-scroll]", inputs: ["infiniteScrollDistance", "infiniteScrollUpDistance", "infiniteScrollThrottle", "infiniteScrollDisabled", "infiniteScrollContainer", "scrollWindow", "immediateCheck", "horizontal", "alwaysCallback", "fromRoot"], outputs: ["scrolled", "scrolledUp"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }, { kind: "pipe", type: ToggleDisplayChatStartDayPipe, name: "toggleDisplayChatStartDay" }, { kind: "pipe", type: ChatStartDayPipe, name: "chatStartDay" }] });
331
380
  }
332
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ChatFlowComponent, decorators: [{
381
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ChatFlowComponent, decorators: [{
333
382
  type: Component,
334
383
  args: [{ selector: 'app-chat-flow', imports: [
335
384
  FormsModule,
336
385
  ChatMessageComponent,
337
- ChatStartDayPipe,
338
- ToggleDisplayChatStartDayPipe,
339
386
  ChatMessageComponent,
340
387
  TranslocoPipe,
341
- NgOptimizedImage
342
- ], standalone: true, template: "<div class=\"chat\">\n @if (messageList()) {\n <div class=\"chat__flow\" #chatFlowRef>\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <lib-chat-message [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n", styles: [":host{display:block;height:100%;min-height:0}.chat{background:#fff;height:100%;display:flex;flex-direction:column;min-height:0;width:100%}.chat__flow{flex:1 1 auto;min-height:0;overflow:auto;display:flex;flex-direction:column;gap:12px;padding:12px 16px;scroll-behavior:smooth;overscroll-behavior:contain}.chat__flow__empty{display:flex;gap:16px;flex-direction:column;align-items:center;justify-content:center;padding:72px 16px;height:100%}.chat__flow__empty .message__icon{width:56px;height:56px}.chat__flow__empty--header{width:285px;display:flex;gap:12px;flex-direction:column;align-items:center}.chat__flow__empty--title{margin:0;font:600 16px/20px inter,sans-serif;color:#212529}.chat__flow__empty--subscription{margin:0;font:400 16px/20px inter,sans-serif;color:#495057}.chat__start-day{display:flex;justify-content:center;background:#f8f9fa;color:#6c757d;font:400 12px/16px inter,sans-serif;width:80px;height:22px;margin:0 auto;border-radius:1000px;gap:8px;padding:4px 8px}.chat ::-webkit-scrollbar{width:6px}.chat ::-webkit-scrollbar-track{background:transparent}.chat ::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.chat ::-webkit-scrollbar-button{display:none}\n"] }]
388
+ NgOptimizedImage,
389
+ InfiniteScrollDirective,
390
+ ToggleDisplayChatStartDayPipe,
391
+ ChatStartDayPipe
392
+ ], standalone: true, template: "<div class=\"chat\">\n @if (messageList()) {\n <div #chatFlowRef\n class=\"chat__flow\"\n infiniteScroll\n [scrollWindow]=\"false\"\n [infiniteScrollUpDistance]=\"1\"\n [infiniteScrollThrottle]=\"150\"\n (scrolledUp)=\"onScrollUp()\"\n >\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <div class=\"chat__message\">\n <lib-chat-message\n [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n </div>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n\n", styles: [":host{display:block;height:100%;min-height:0}.chat{background:#fff;height:100%;display:flex;flex-direction:column;min-height:0;width:100%}.chat__flow{flex:1 1 auto;min-height:0;overflow:auto;display:flex;flex-direction:column;gap:12px;padding:12px 16px;scroll-behavior:smooth;overscroll-behavior:contain}.chat__flow__empty{display:flex;gap:16px;flex-direction:column;align-items:center;justify-content:center;padding:72px 16px;height:100%}.chat__flow__empty .message__icon{width:56px;height:56px}.chat__flow__empty--header{width:285px;display:flex;gap:12px;flex-direction:column;align-items:center}.chat__flow__empty--title{margin:0;font:600 16px/20px inter,sans-serif;color:#212529}.chat__flow__empty--subscription{margin:0;font:400 16px/20px inter,sans-serif;color:#495057}.chat__start-day{display:flex;justify-content:center;background:#f8f9fa;color:#6c757d;font:400 12px/16px inter,sans-serif;width:80px;height:22px;margin:0 auto;border-radius:1000px;gap:8px;padding:4px 8px}.chat ::-webkit-scrollbar{width:6px}.chat ::-webkit-scrollbar-track{background:transparent}.chat ::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.chat ::-webkit-scrollbar-button{display:none}\n"] }]
343
393
  }], ctorParameters: () => [], propDecorators: { flowRef: [{
344
394
  type: ViewChild,
345
395
  args: ['chatFlowRef']
346
- }], scrollToBottomTrigger: [{ type: i0.Input, args: [{ isSignal: true, alias: "scrollToBottomTrigger", required: false }] }, { type: i0.Output, args: ["scrollToBottomTriggerChange"] }], messageListInput: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageListInput", required: true }] }, { type: i0.Output, args: ["messageListInputChange"] }], selectedForEdit: [{ type: i0.Input, args: [{ isSignal: true, alias: "selectedForEdit", required: true }] }, { type: i0.Output, args: ["selectedForEditChange"] }], requestDelete: [{ type: i0.Input, args: [{ isSignal: true, alias: "requestDelete", required: false }] }, { type: i0.Output, args: ["requestDeleteChange"] }] } });
396
+ }], scrollToBottomTrigger: [{ type: i0.Input, args: [{ isSignal: true, alias: "scrollToBottomTrigger", required: false }] }, { type: i0.Output, args: ["scrollToBottomTriggerChange"] }], loadHistory: [{ type: i0.Input, args: [{ isSignal: true, alias: "loadHistory", required: false }] }, { type: i0.Output, args: ["loadHistoryChange"] }], messageListInput: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageListInput", required: true }] }, { type: i0.Output, args: ["messageListInputChange"] }], selectedForEdit: [{ type: i0.Input, args: [{ isSignal: true, alias: "selectedForEdit", required: true }] }, { type: i0.Output, args: ["selectedForEditChange"] }], requestDelete: [{ type: i0.Input, args: [{ isSignal: true, alias: "requestDelete", required: false }] }, { type: i0.Output, args: ["requestDeleteChange"] }] } });
347
397
 
348
398
  class ImageFile {
349
399
  id;
@@ -650,10 +700,10 @@ class InputMessageComponent {
650
700
  return Number.isFinite(n) ? n : fb;
651
701
  }
652
702
  FileType = FileType;
653
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: InputMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
654
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.7", type: InputMessageComponent, isStandalone: true, selector: "app-input-message", inputs: { editMessage: { classPropertyName: "editMessage", publicName: "editMessage", isSignal: true, isRequired: false, transformFunction: null }, cancelEdit: { classPropertyName: "cancelEdit", publicName: "cancelEdit", isSignal: true, isRequired: false, transformFunction: null }, input_text: { classPropertyName: "input_text", publicName: "input_text", isSignal: true, isRequired: false, transformFunction: null }, files: { classPropertyName: "files", publicName: "files", isSignal: true, isRequired: false, transformFunction: null }, previews: { classPropertyName: "previews", publicName: "previews", isSignal: true, isRequired: false, transformFunction: null }, messageEvent: { classPropertyName: "messageEvent", publicName: "messageEvent", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { cancelEdit: "cancelEditChange", input_text: "input_textChange", files: "filesChange", previews: "previewsChange", messageEvent: "messageEventChange" }, viewQueries: [{ propertyName: "inputTextElement", first: true, predicate: ["inputText"], descendants: true }, { propertyName: "mirrorElement", first: true, predicate: ["mirror"], descendants: true }], ngImport: i0, template: "<section class=\"message\"\n [class.message--focus]=\"focused()\"\n [class.message--filled]=\"hasText()\"\n [class.message--sending]=\"sending()\">\n\n @if (isEditMode()) {\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button class=\"message__button message__button--attach\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon message__icon--edit\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__edit-hint\">\n <p>{{ editMessage()?.content }}</p>\n </div>\n </div>\n\n <button class=\"message__button message__button--close\"\n type=\"button\"\n (click)=\"cancelEditMessage()\">\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\"\n class=\"message__icon message__icon--close\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n </div>\n }\n\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button type=\"button\"\n class=\"message__button message__button--attach\"\n [attr.aria-label]=\"'chat.attach' | transloco\"\n (click)=\"fileDialog.click()\">\n <img ngSrc=\"assets/ngx-parl/icons/attach-filled.svg\"\n class=\"message__icon message__icon--attach\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <input #fileDialog\n type=\"file\"\n class=\"hidden\"\n accept=\"image/*\"\n multiple\n (change)=\"inputFileChange($event)\">\n\n <div #inputText\n role=\"textbox\"\n class=\"message__input\"\n spellcheck=\"true\"\n contenteditable=\"true\"\n aria-multiline=\"true\"\n [attr.data-placeholder]=\"isEditMode()\n ? ('chat.edit_placeholder' | transloco)\n : ('chat.placeholder' | transloco)\"\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (input)=\"onInput()\"\n (paste)=\"onPaste()\"\n (keydown)=\"onKeyDown($event)\">\n </div>\n\n <div #mirror class=\"message__input-mirror\" aria-hidden=\"true\"></div>\n\n <ng-content></ng-content>\n </div>\n\n <button class=\"message__button message__button--send\"\n type=\"button\"\n (click)=\"enterDown()\"\n [attr.aria-label]=\"'chat.send' | transloco\"\n [disabled]=\"!canSend()\">\n <img ngSrc=\"assets/ngx-parl/icons/send.svg\"\n class=\"message__icon message__icon--send\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n\n </button>\n </div>\n\n @if (previews().length) {\n <div class=\"message__files\">\n @for (item of previews(); track $index) {\n <div class=\"message__file\" [attr.data-kind]=\"item.originalKind\">\n <button type=\"button\"\n class=\"message__file-remove\"\n [attr.aria-label]=\"'chat.delete_file' | transloco\"\n (click)=\"removeFile($index, $event)\">\n <img ngSrc=\"assets/ngx-parl/icons/remove.svg\"\n class=\"message__icon message__icon--file\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__file-media\" (click)=\"openPreview(item, $index)\">\n <img [src]=\"item.src\" [alt]=\"item.name\" class=\"message__file-img\"/>\n\n @if (item.originalKind === FileType.GIF) {\n <span class=\"message__file-badge message__file-badge--gif\">\n {{ 'chat.gif' | transloco }}\n </span>\n }\n </div>\n </div>\n }\n </div>\n }\n</section>\n\n\n", styles: [":host{display:block}.message{--lh: 24px;--max-rows: 8;display:flex;flex-direction:column;justify-content:space-between;background:#fff;border-top:1px solid #E1E7F8;padding:4px 8px;margin-top:4px;gap:8px;border-bottom-left-radius:16px;border-bottom-right-radius:16px;transition:box-shadow .18s ease,background-color .18s ease,border-color .18s ease}.message__wrap{display:flex;width:100%;min-width:0}.message__wrap--input{position:relative;display:flex;align-items:center;gap:12px;flex:1;min-width:0;padding:8px 0}.message__wrap--input:focus-within{border-color:#f8f9fa}.message__wrap--input::-webkit-scrollbar{width:4px;height:4px}.message__wrap--input::-webkit-scrollbar-track{background:transparent}.message__wrap--input::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.message__wrap--input::-webkit-scrollbar-button{display:none}.message__input{flex:1 1 auto;min-height:var(--lh);line-height:var(--lh);font:400 16px/20px inter,sans-serif;outline:none;border:none;background:transparent;color:#343a40;box-sizing:content-box;resize:none;white-space:pre-wrap;overflow-wrap:break-word;caret-color:#3c46b9;transition:height .16s ease;padding:0 4px 0 0;overflow-y:auto;max-height:calc(var(--lh) * var(--max-rows))}.message__input:empty:before{content:attr(data-placeholder);color:#ced4da;pointer-events:none;display:inline-block;animation:subtle-rise .18s ease both}.message__input-mirror{position:absolute;visibility:hidden;pointer-events:none;z-index:-1;white-space:pre-wrap;overflow-wrap:break-word;word-break:normal}.message__button{cursor:pointer;background:none;border:none;transition:transform .12s ease,opacity .12s ease,background-color .12s ease}.message__button:active{transform:scale(.96)}.message__button:disabled{opacity:.45;pointer-events:none}.message__icon{display:inline-flex;width:22px;height:22px}.message__icon:not(.message__icon--file):hover{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--focus,.message__icon--filled{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--edit{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(73%) contrast(99%)}.message__icon--close{filter:invert(1%) sepia(1%) saturate(1%) hue-rotate(100deg) brightness(100%) contrast(30%)}.message__files{display:flex;flex-wrap:wrap;gap:8px}.message__file{position:relative;width:40px;height:40px;flex:0 0 auto;overflow:visible}.message__file-media{width:100%;height:100%;border-radius:8px;overflow:hidden;cursor:pointer;position:relative;background:#f8f9fa}.message__file-img{position:absolute;inset:0;width:100%;height:100%;object-fit:cover;display:block}.message__file-badge{position:absolute;left:6px;top:6px;z-index:2;display:inline-flex;align-items:center;justify-content:center;height:18px;padding:0 6px;border-radius:4px;font:400 16px/20px inter,sans-serif;background:#0009;color:#fff;letter-spacing:.3px;line-height:1}.message__file-badge--video{width:28px;height:28px;padding:0;left:50%;top:50%;transform:translate(-50%,-50%);border-radius:50%;background:#00000073}.message__file-badge--video img{width:18px;height:18px;filter:invert(100%)}.message__file-badge--gif{font-weight:700}.message__file-duration{position:absolute;right:6px;bottom:6px;background:#0009;color:#fff;border-radius:4px;padding:1px;font:400 16px/20px inter,sans-serif;z-index:2px;line-height:1.1}.message__file-remove{position:absolute;top:-6px;right:-6px;width:16px;height:16px;padding:0;border:none;border-radius:50%;display:inline-flex;align-items:center;justify-content:center;z-index:3;cursor:pointer;transition:transform .12s ease,box-shadow .12s ease}.message__file-remove:hover{transform:scale(1.05)}.message__file-remove:active{transform:scale(.96)}.message__file-remove .message__icon--file{width:14px;height:14px}.message__edit-hint{max-width:100%;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;display:flex;align-items:center;gap:8px;border-radius:4px;border-left:2px solid #4656CA;padding:4px 8px}.message__edit-hint p{color:#ced4da;font:400 16px/20px inter,sans-serif;margin:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.hidden{position:absolute;opacity:0;width:0;height:0;pointer-events:none}@keyframes subtle-rise{0%{transform:translateY(2px);opacity:.6}to{transform:translateY(0);opacity:1}}\n"], dependencies: [{ kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] });
703
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: InputMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
704
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.16", type: InputMessageComponent, isStandalone: true, selector: "app-input-message", inputs: { editMessage: { classPropertyName: "editMessage", publicName: "editMessage", isSignal: true, isRequired: false, transformFunction: null }, cancelEdit: { classPropertyName: "cancelEdit", publicName: "cancelEdit", isSignal: true, isRequired: false, transformFunction: null }, input_text: { classPropertyName: "input_text", publicName: "input_text", isSignal: true, isRequired: false, transformFunction: null }, files: { classPropertyName: "files", publicName: "files", isSignal: true, isRequired: false, transformFunction: null }, previews: { classPropertyName: "previews", publicName: "previews", isSignal: true, isRequired: false, transformFunction: null }, messageEvent: { classPropertyName: "messageEvent", publicName: "messageEvent", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { cancelEdit: "cancelEditChange", input_text: "input_textChange", files: "filesChange", previews: "previewsChange", messageEvent: "messageEventChange" }, viewQueries: [{ propertyName: "inputTextElement", first: true, predicate: ["inputText"], descendants: true }, { propertyName: "mirrorElement", first: true, predicate: ["mirror"], descendants: true }], ngImport: i0, template: "<section class=\"message\"\n [class.message--focus]=\"focused()\"\n [class.message--filled]=\"hasText()\"\n [class.message--sending]=\"sending()\">\n\n @if (isEditMode()) {\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button class=\"message__button message__button--attach\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon message__icon--edit\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__edit-hint\">\n <p>{{ editMessage()?.content }}</p>\n </div>\n </div>\n\n <button class=\"message__button message__button--close\"\n type=\"button\"\n (click)=\"cancelEditMessage()\">\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\"\n class=\"message__icon message__icon--close\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n </div>\n }\n\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button type=\"button\"\n class=\"message__button message__button--attach\"\n [attr.aria-label]=\"'chat.attach' | transloco\"\n (click)=\"fileDialog.click()\">\n <img ngSrc=\"assets/ngx-parl/icons/attach-filled.svg\"\n class=\"message__icon message__icon--attach\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <input #fileDialog\n type=\"file\"\n class=\"hidden\"\n accept=\"image/*\"\n multiple\n (change)=\"inputFileChange($event)\">\n\n <div #inputText\n role=\"textbox\"\n class=\"message__input\"\n spellcheck=\"true\"\n contenteditable=\"true\"\n aria-multiline=\"true\"\n [attr.data-placeholder]=\"isEditMode()\n ? ('chat.edit_placeholder' | transloco)\n : ('chat.placeholder' | transloco)\"\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (input)=\"onInput()\"\n (paste)=\"onPaste()\"\n (keydown)=\"onKeyDown($event)\">\n </div>\n\n <div #mirror class=\"message__input-mirror\" aria-hidden=\"true\"></div>\n\n <ng-content></ng-content>\n </div>\n\n <button class=\"message__button message__button--send\"\n type=\"button\"\n (click)=\"enterDown()\"\n [attr.aria-label]=\"'chat.send' | transloco\"\n [disabled]=\"!canSend()\">\n <img ngSrc=\"assets/ngx-parl/icons/send.svg\"\n class=\"message__icon message__icon--send\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n\n </button>\n </div>\n\n @if (previews().length) {\n <div class=\"message__files\">\n @for (item of previews(); track $index) {\n <div class=\"message__file\" [attr.data-kind]=\"item.originalKind\">\n <button type=\"button\"\n class=\"message__file-remove\"\n [attr.aria-label]=\"'chat.delete_file' | transloco\"\n (click)=\"removeFile($index, $event)\">\n <img ngSrc=\"assets/ngx-parl/icons/remove.svg\"\n class=\"message__icon message__icon--file\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__file-media\" (click)=\"openPreview(item, $index)\">\n <img [src]=\"item.src\" [alt]=\"item.name\" class=\"message__file-img\"/>\n\n @if (item.originalKind === FileType.GIF) {\n <span class=\"message__file-badge message__file-badge--gif\">\n {{ 'chat.gif' | transloco }}\n </span>\n }\n </div>\n </div>\n }\n </div>\n }\n</section>\n\n\n", styles: [":host{display:block}.message{--lh: 24px;--max-rows: 8;display:flex;flex-direction:column;justify-content:space-between;background:#fff;border-top:1px solid #E1E7F8;padding:4px 8px;margin-top:4px;gap:8px;border-bottom-left-radius:16px;border-bottom-right-radius:16px;transition:box-shadow .18s ease,background-color .18s ease,border-color .18s ease}.message__wrap{display:flex;width:100%;min-width:0}.message__wrap--input{position:relative;display:flex;align-items:center;gap:12px;flex:1;min-width:0;padding:8px 0}.message__wrap--input:focus-within{border-color:#f8f9fa}.message__wrap--input::-webkit-scrollbar{width:4px;height:4px}.message__wrap--input::-webkit-scrollbar-track{background:transparent}.message__wrap--input::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.message__wrap--input::-webkit-scrollbar-button{display:none}.message__input{flex:1 1 auto;min-height:var(--lh);line-height:var(--lh);font:400 16px/20px inter,sans-serif;outline:none;border:none;background:transparent;color:#343a40;box-sizing:content-box;resize:none;white-space:pre-wrap;overflow-wrap:break-word;caret-color:#3c46b9;transition:height .16s ease;padding:0 4px 0 0;overflow-y:auto;max-height:calc(var(--lh) * var(--max-rows))}.message__input:empty:before{content:attr(data-placeholder);color:#ced4da;pointer-events:none;display:inline-block;animation:subtle-rise .18s ease both}.message__input-mirror{position:absolute;visibility:hidden;pointer-events:none;z-index:-1;white-space:pre-wrap;overflow-wrap:break-word;word-break:normal}.message__button{cursor:pointer;background:none;border:none;transition:transform .12s ease,opacity .12s ease,background-color .12s ease}.message__button:active{transform:scale(.96)}.message__button:disabled{opacity:.45;pointer-events:none}.message__icon{display:inline-flex;width:22px;height:22px}.message__icon:not(.message__icon--file):hover{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--focus,.message__icon--filled{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--edit{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(73%) contrast(99%)}.message__icon--close{filter:invert(1%) sepia(1%) saturate(1%) hue-rotate(100deg) brightness(100%) contrast(30%)}.message__files{display:flex;flex-wrap:wrap;gap:8px}.message__file{position:relative;width:40px;height:40px;flex:0 0 auto;overflow:visible}.message__file-media{width:100%;height:100%;border-radius:8px;overflow:hidden;cursor:pointer;position:relative;background:#f8f9fa}.message__file-img{position:absolute;inset:0;width:100%;height:100%;object-fit:cover;display:block}.message__file-badge{position:absolute;left:6px;top:6px;z-index:2;display:inline-flex;align-items:center;justify-content:center;height:18px;padding:0 6px;border-radius:4px;font:400 16px/20px inter,sans-serif;background:#0009;color:#fff;letter-spacing:.3px;line-height:1}.message__file-badge--video{width:28px;height:28px;padding:0;left:50%;top:50%;transform:translate(-50%,-50%);border-radius:50%;background:#00000073}.message__file-badge--video img{width:18px;height:18px;filter:invert(100%)}.message__file-badge--gif{font-weight:700}.message__file-duration{position:absolute;right:6px;bottom:6px;background:#0009;color:#fff;border-radius:4px;padding:1px;font:400 16px/20px inter,sans-serif;z-index:2px;line-height:1.1}.message__file-remove{position:absolute;top:-6px;right:-6px;width:16px;height:16px;padding:0;border:none;border-radius:50%;display:inline-flex;align-items:center;justify-content:center;z-index:3;cursor:pointer;transition:transform .12s ease,box-shadow .12s ease}.message__file-remove:hover{transform:scale(1.05)}.message__file-remove:active{transform:scale(.96)}.message__file-remove .message__icon--file{width:14px;height:14px}.message__edit-hint{max-width:100%;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;display:flex;align-items:center;gap:8px;border-radius:4px;border-left:2px solid #4656CA;padding:4px 8px}.message__edit-hint p{color:#ced4da;font:400 16px/20px inter,sans-serif;margin:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.hidden{position:absolute;opacity:0;width:0;height:0;pointer-events:none}@keyframes subtle-rise{0%{transform:translateY(2px);opacity:.6}to{transform:translateY(0);opacity:1}}\n"], dependencies: [{ kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] });
655
705
  }
656
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: InputMessageComponent, decorators: [{
706
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: InputMessageComponent, decorators: [{
657
707
  type: Component,
658
708
  args: [{ selector: 'app-input-message', imports: [TranslocoPipe, NgOptimizedImage], standalone: true, template: "<section class=\"message\"\n [class.message--focus]=\"focused()\"\n [class.message--filled]=\"hasText()\"\n [class.message--sending]=\"sending()\">\n\n @if (isEditMode()) {\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button class=\"message__button message__button--attach\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon message__icon--edit\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__edit-hint\">\n <p>{{ editMessage()?.content }}</p>\n </div>\n </div>\n\n <button class=\"message__button message__button--close\"\n type=\"button\"\n (click)=\"cancelEditMessage()\">\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\"\n class=\"message__icon message__icon--close\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n </div>\n }\n\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button type=\"button\"\n class=\"message__button message__button--attach\"\n [attr.aria-label]=\"'chat.attach' | transloco\"\n (click)=\"fileDialog.click()\">\n <img ngSrc=\"assets/ngx-parl/icons/attach-filled.svg\"\n class=\"message__icon message__icon--attach\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <input #fileDialog\n type=\"file\"\n class=\"hidden\"\n accept=\"image/*\"\n multiple\n (change)=\"inputFileChange($event)\">\n\n <div #inputText\n role=\"textbox\"\n class=\"message__input\"\n spellcheck=\"true\"\n contenteditable=\"true\"\n aria-multiline=\"true\"\n [attr.data-placeholder]=\"isEditMode()\n ? ('chat.edit_placeholder' | transloco)\n : ('chat.placeholder' | transloco)\"\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (input)=\"onInput()\"\n (paste)=\"onPaste()\"\n (keydown)=\"onKeyDown($event)\">\n </div>\n\n <div #mirror class=\"message__input-mirror\" aria-hidden=\"true\"></div>\n\n <ng-content></ng-content>\n </div>\n\n <button class=\"message__button message__button--send\"\n type=\"button\"\n (click)=\"enterDown()\"\n [attr.aria-label]=\"'chat.send' | transloco\"\n [disabled]=\"!canSend()\">\n <img ngSrc=\"assets/ngx-parl/icons/send.svg\"\n class=\"message__icon message__icon--send\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n\n </button>\n </div>\n\n @if (previews().length) {\n <div class=\"message__files\">\n @for (item of previews(); track $index) {\n <div class=\"message__file\" [attr.data-kind]=\"item.originalKind\">\n <button type=\"button\"\n class=\"message__file-remove\"\n [attr.aria-label]=\"'chat.delete_file' | transloco\"\n (click)=\"removeFile($index, $event)\">\n <img ngSrc=\"assets/ngx-parl/icons/remove.svg\"\n class=\"message__icon message__icon--file\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__file-media\" (click)=\"openPreview(item, $index)\">\n <img [src]=\"item.src\" [alt]=\"item.name\" class=\"message__file-img\"/>\n\n @if (item.originalKind === FileType.GIF) {\n <span class=\"message__file-badge message__file-badge--gif\">\n {{ 'chat.gif' | transloco }}\n </span>\n }\n </div>\n </div>\n }\n </div>\n }\n</section>\n\n\n", styles: [":host{display:block}.message{--lh: 24px;--max-rows: 8;display:flex;flex-direction:column;justify-content:space-between;background:#fff;border-top:1px solid #E1E7F8;padding:4px 8px;margin-top:4px;gap:8px;border-bottom-left-radius:16px;border-bottom-right-radius:16px;transition:box-shadow .18s ease,background-color .18s ease,border-color .18s ease}.message__wrap{display:flex;width:100%;min-width:0}.message__wrap--input{position:relative;display:flex;align-items:center;gap:12px;flex:1;min-width:0;padding:8px 0}.message__wrap--input:focus-within{border-color:#f8f9fa}.message__wrap--input::-webkit-scrollbar{width:4px;height:4px}.message__wrap--input::-webkit-scrollbar-track{background:transparent}.message__wrap--input::-webkit-scrollbar-thumb{background-color:#e9ecef;border-radius:4px;border:1px solid #E9ECEF}.message__wrap--input::-webkit-scrollbar-button{display:none}.message__input{flex:1 1 auto;min-height:var(--lh);line-height:var(--lh);font:400 16px/20px inter,sans-serif;outline:none;border:none;background:transparent;color:#343a40;box-sizing:content-box;resize:none;white-space:pre-wrap;overflow-wrap:break-word;caret-color:#3c46b9;transition:height .16s ease;padding:0 4px 0 0;overflow-y:auto;max-height:calc(var(--lh) * var(--max-rows))}.message__input:empty:before{content:attr(data-placeholder);color:#ced4da;pointer-events:none;display:inline-block;animation:subtle-rise .18s ease both}.message__input-mirror{position:absolute;visibility:hidden;pointer-events:none;z-index:-1;white-space:pre-wrap;overflow-wrap:break-word;word-break:normal}.message__button{cursor:pointer;background:none;border:none;transition:transform .12s ease,opacity .12s ease,background-color .12s ease}.message__button:active{transform:scale(.96)}.message__button:disabled{opacity:.45;pointer-events:none}.message__icon{display:inline-flex;width:22px;height:22px}.message__icon:not(.message__icon--file):hover{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--focus,.message__icon--filled{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(93%) contrast(91%)}.message__icon--edit{filter:invert(33%) sepia(79%) saturate(2933%) hue-rotate(205deg) brightness(73%) contrast(99%)}.message__icon--close{filter:invert(1%) sepia(1%) saturate(1%) hue-rotate(100deg) brightness(100%) contrast(30%)}.message__files{display:flex;flex-wrap:wrap;gap:8px}.message__file{position:relative;width:40px;height:40px;flex:0 0 auto;overflow:visible}.message__file-media{width:100%;height:100%;border-radius:8px;overflow:hidden;cursor:pointer;position:relative;background:#f8f9fa}.message__file-img{position:absolute;inset:0;width:100%;height:100%;object-fit:cover;display:block}.message__file-badge{position:absolute;left:6px;top:6px;z-index:2;display:inline-flex;align-items:center;justify-content:center;height:18px;padding:0 6px;border-radius:4px;font:400 16px/20px inter,sans-serif;background:#0009;color:#fff;letter-spacing:.3px;line-height:1}.message__file-badge--video{width:28px;height:28px;padding:0;left:50%;top:50%;transform:translate(-50%,-50%);border-radius:50%;background:#00000073}.message__file-badge--video img{width:18px;height:18px;filter:invert(100%)}.message__file-badge--gif{font-weight:700}.message__file-duration{position:absolute;right:6px;bottom:6px;background:#0009;color:#fff;border-radius:4px;padding:1px;font:400 16px/20px inter,sans-serif;z-index:2px;line-height:1.1}.message__file-remove{position:absolute;top:-6px;right:-6px;width:16px;height:16px;padding:0;border:none;border-radius:50%;display:inline-flex;align-items:center;justify-content:center;z-index:3;cursor:pointer;transition:transform .12s ease,box-shadow .12s ease}.message__file-remove:hover{transform:scale(1.05)}.message__file-remove:active{transform:scale(.96)}.message__file-remove .message__icon--file{width:14px;height:14px}.message__edit-hint{max-width:100%;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;display:flex;align-items:center;gap:8px;border-radius:4px;border-left:2px solid #4656CA;padding:4px 8px}.message__edit-hint p{color:#ced4da;font:400 16px/20px inter,sans-serif;margin:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.hidden{position:absolute;opacity:0;width:0;height:0;pointer-events:none}@keyframes subtle-rise{0%{transform:translateY(2px);opacity:.6}to{transform:translateY(0);opacity:1}}\n"] }]
659
709
  }], ctorParameters: () => [], propDecorators: { inputTextElement: [{
@@ -688,12 +738,12 @@ class NgxParlComponent {
688
738
  hideHandler = input(null, ...(ngDevMode ? [{ debugName: "hideHandler" }] : []));
689
739
  closeHandler = input(null, ...(ngDevMode ? [{ debugName: "closeHandler" }] : []));
690
740
  scrollToBottomTrigger = model(0, ...(ngDevMode ? [{ debugName: "scrollToBottomTrigger" }] : []));
741
+ loadHistory = model(false, ...(ngDevMode ? [{ debugName: "loadHistory" }] : []));
691
742
  constructor(utils, transloco) {
692
743
  this.utils = utils;
693
744
  this.transloco = transloco;
694
745
  effect(() => {
695
- const lang = this.language();
696
- this.transloco.setActiveLang(lang);
746
+ this.transloco.setActiveLang(this.language());
697
747
  });
698
748
  effect(() => {
699
749
  const updatedMessage = this.messageUpdate();
@@ -705,31 +755,27 @@ class NgxParlComponent {
705
755
  return;
706
756
  }
707
757
  this.lastUpdateKey = key;
708
- if (updatedMessage.type !== MessageType.Incoming) {
709
- return;
710
- }
711
- this.messageList.update((currentList) => {
712
- const list = [...currentList];
713
- const incomingIndex = list.findIndex((message) => message.id === updatedMessage.id &&
714
- message.type === MessageType.Incoming);
715
- if (incomingIndex > -1) {
716
- list[incomingIndex] = updatedMessage;
717
- return list;
758
+ this.messageList.update(list => {
759
+ const index = list.findIndex(m => m.id === updatedMessage.id);
760
+ if (index > -1) {
761
+ const updated = [...list];
762
+ updated[index] = updatedMessage;
763
+ return updated;
718
764
  }
719
- list.push(updatedMessage);
720
- return list;
765
+ return [...list, updatedMessage];
721
766
  });
767
+ this.scrollToBottomTrigger.update(v => v + 1);
722
768
  });
723
769
  }
724
770
  onCancelEdit(messageId) {
725
771
  if (messageId != null) {
726
- this.messageList.update((currentList) => {
727
- const updatedList = [...currentList];
728
- const index = updatedList.findIndex((message) => message.id === messageId);
772
+ this.messageList.update(list => {
773
+ const updated = [...list];
774
+ const index = updated.findIndex(m => m.id === messageId);
729
775
  if (index > -1) {
730
- updatedList[index].edit = false;
776
+ updated[index].edit = false;
731
777
  }
732
- return updatedList;
778
+ return updated;
733
779
  });
734
780
  }
735
781
  this.selectedForEdit.set(null);
@@ -739,10 +785,7 @@ class NgxParlComponent {
739
785
  if (messageId == null) {
740
786
  return this;
741
787
  }
742
- this.messageList.update((currentList) => {
743
- const updatedList = currentList.filter((m) => m.id !== messageId);
744
- return [...updatedList];
745
- });
788
+ this.messageList.update(list => list.filter(m => m.id !== messageId));
746
789
  this.pushMessageAction({
747
790
  action: 'delete',
748
791
  chatMessageId: messageId,
@@ -777,6 +820,7 @@ class NgxParlComponent {
777
820
  files: Array.isArray(files) && files.length ? files : [],
778
821
  file_list: Array.isArray(file_list) && file_list.length ? file_list : [],
779
822
  });
823
+ this.scrollToBottomTrigger.update(v => v + 1);
780
824
  return this;
781
825
  }
782
826
  // new message
@@ -800,9 +844,10 @@ class NgxParlComponent {
800
844
  avatar: lastOutgoing?.avatar ?? null,
801
845
  file_path: [],
802
846
  file_list: [],
803
- checked: false,
847
+ checked: true,
804
848
  };
805
849
  this.messageList.update((list) => [...list, new ChatMessage(dto)]);
850
+ this.scrollToBottomTrigger.update(v => v + 1);
806
851
  this.pushMessageAction({
807
852
  action: 'send',
808
853
  chatMessageId: dto.id,
@@ -833,9 +878,10 @@ class NgxParlComponent {
833
878
  avatar: lastOutgoing?.avatar ?? null,
834
879
  file_path: hasFiles ? files : [],
835
880
  file_list: hasFileList ? file_list : [],
836
- checked: false,
881
+ checked: true,
837
882
  };
838
883
  this.messageList.update((list) => [...list, new ChatMessage(dto)]);
884
+ this.scrollToBottomTrigger.update(v => v + 1);
839
885
  this.pushMessageAction({
840
886
  action: 'send',
841
887
  chatMessageId: dto.id,
@@ -864,22 +910,18 @@ class NgxParlComponent {
864
910
  }
865
911
  return this;
866
912
  }
867
- scrollToBottom() {
868
- this.scrollToBottomTrigger.update(value => value + 1);
869
- return this;
870
- }
871
913
  FlowTheme = FlowTheme;
872
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: NgxParlComponent, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Component });
873
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.7", type: NgxParlComponent, isStandalone: true, selector: "ngx-parl", inputs: { theme: { classPropertyName: "theme", publicName: "theme", isSignal: true, isRequired: false, transformFunction: null }, header: { classPropertyName: "header", publicName: "header", isSignal: true, isRequired: false, transformFunction: null }, language: { classPropertyName: "language", publicName: "language", isSignal: true, isRequired: false, transformFunction: null }, messageList: { classPropertyName: "messageList", publicName: "messageList", isSignal: true, isRequired: false, transformFunction: null }, messageUpdate: { classPropertyName: "messageUpdate", publicName: "messageUpdate", isSignal: true, isRequired: false, transformFunction: null }, selectedForEdit: { classPropertyName: "selectedForEdit", publicName: "selectedForEdit", isSignal: true, isRequired: false, transformFunction: null }, messageAction: { classPropertyName: "messageAction", publicName: "messageAction", isSignal: true, isRequired: false, transformFunction: null }, hideHandler: { classPropertyName: "hideHandler", publicName: "hideHandler", isSignal: true, isRequired: false, transformFunction: null }, closeHandler: { classPropertyName: "closeHandler", publicName: "closeHandler", isSignal: true, isRequired: false, transformFunction: null }, scrollToBottomTrigger: { classPropertyName: "scrollToBottomTrigger", publicName: "scrollToBottomTrigger", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { messageList: "messageListChange", messageUpdate: "messageUpdateChange", selectedForEdit: "selectedForEditChange", messageAction: "messageActionChange", scrollToBottomTrigger: "scrollToBottomTriggerChange" }, providers: [], ngImport: i0, template: "<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.modal-chat{width:800px;height:600px;border-radius:16px}.modal-chat__body{display:flex;flex-direction:column;height:100%}.modal-chat__header{height:56px;min-height:56px;max-height:56px;display:flex;align-items:center;justify-content:space-between;padding:0 16px;gap:12px;border-top-left-radius:16px;border-top-right-radius:16px;flex-direction:row}.modal-chat__hide-block{width:48px;color:#5a72d7}.modal-chat__title #modal-chat__heading{color:#fff;font:600 16px/20px inter,sans-serif}.modal-chat__actions{display:flex;gap:8px}.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;background:#fff;border-bottom-left-radius:16px;border-bottom-right-radius:16px}.modal-chat__content--header{border-top-left-radius:16px;border-top-right-radius:16px}.modal-chat__content app-chat-flow{flex:1 1 auto;min-height:0px;overflow:auto;display:flex;align-items:flex-end}.modal-chat__input{flex:0px 0px auto;border-bottom-left-radius:16px;border-bottom-right-radius:16px}:host ::ng-deep .mat-mdc-dialog-content.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;padding:0;max-height:none;overflow:hidden}img{cursor:pointer}\n"], dependencies: [{ kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: MatDialogContent, selector: "[mat-dialog-content], mat-dialog-content, [matDialogContent]" }, { kind: "directive", type: MatDialogTitle, selector: "[mat-dialog-title], [matDialogTitle]", inputs: ["id"], exportAs: ["matDialogTitle"] }, { kind: "component", type: MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: ChatFlowComponent, selector: "app-chat-flow", inputs: ["scrollToBottomTrigger", "messageListInput", "selectedForEdit", "requestDelete"], outputs: ["scrollToBottomTriggerChange", "messageListInputChange", "selectedForEditChange", "requestDeleteChange"] }, { kind: "component", type: InputMessageComponent, selector: "app-input-message", inputs: ["editMessage", "cancelEdit", "input_text", "files", "previews", "messageEvent"], outputs: ["cancelEditChange", "input_textChange", "filesChange", "previewsChange", "messageEventChange"] }, { kind: "ngmodule", type: TranslocoModule }, { kind: "pipe", type: i2.TranslocoPipe, name: "transloco" }] });
914
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: NgxParlComponent, deps: [{ token: UtilsService }, { token: i2.TranslocoService }], target: i0.ɵɵFactoryTarget.Component });
915
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "20.3.16", type: NgxParlComponent, isStandalone: true, selector: "ngx-parl", inputs: { theme: { classPropertyName: "theme", publicName: "theme", isSignal: true, isRequired: false, transformFunction: null }, header: { classPropertyName: "header", publicName: "header", isSignal: true, isRequired: false, transformFunction: null }, language: { classPropertyName: "language", publicName: "language", isSignal: true, isRequired: false, transformFunction: null }, messageList: { classPropertyName: "messageList", publicName: "messageList", isSignal: true, isRequired: false, transformFunction: null }, messageUpdate: { classPropertyName: "messageUpdate", publicName: "messageUpdate", isSignal: true, isRequired: false, transformFunction: null }, selectedForEdit: { classPropertyName: "selectedForEdit", publicName: "selectedForEdit", isSignal: true, isRequired: false, transformFunction: null }, messageAction: { classPropertyName: "messageAction", publicName: "messageAction", isSignal: true, isRequired: false, transformFunction: null }, hideHandler: { classPropertyName: "hideHandler", publicName: "hideHandler", isSignal: true, isRequired: false, transformFunction: null }, closeHandler: { classPropertyName: "closeHandler", publicName: "closeHandler", isSignal: true, isRequired: false, transformFunction: null }, scrollToBottomTrigger: { classPropertyName: "scrollToBottomTrigger", publicName: "scrollToBottomTrigger", isSignal: true, isRequired: false, transformFunction: null }, loadHistory: { classPropertyName: "loadHistory", publicName: "loadHistory", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { messageList: "messageListChange", messageUpdate: "messageUpdateChange", selectedForEdit: "selectedForEditChange", messageAction: "messageActionChange", scrollToBottomTrigger: "scrollToBottomTriggerChange", loadHistory: "loadHistoryChange" }, providers: [], ngImport: i0, template: "<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\"\n [(loadHistory)]=\"loadHistory\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.modal-chat{width:800px;height:600px;border-radius:16px}.modal-chat__body{display:flex;flex-direction:column;height:100%}.modal-chat__header{height:56px;min-height:56px;max-height:56px;display:flex;align-items:center;justify-content:space-between;padding:0 16px;gap:12px;border-top-left-radius:16px;border-top-right-radius:16px;flex-direction:row}.modal-chat__hide-block{width:48px;color:#5a72d7}.modal-chat__title #modal-chat__heading{color:#fff;font:600 16px/20px inter,sans-serif}.modal-chat__actions{display:flex;gap:8px}.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;background:#fff;border-bottom-left-radius:16px;border-bottom-right-radius:16px}.modal-chat__content--header{border-top-left-radius:16px;border-top-right-radius:16px}.modal-chat__content app-chat-flow{flex:1 1 auto;min-height:0px;overflow:auto;display:flex;align-items:flex-end}.modal-chat__input{flex:0px 0px auto;border-bottom-left-radius:16px;border-bottom-right-radius:16px}:host ::ng-deep .mat-mdc-dialog-content.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;padding:0;max-height:none;overflow:hidden}img{cursor:pointer}\n"], dependencies: [{ kind: "directive", type: NgOptimizedImage, selector: "img[ngSrc]", inputs: ["ngSrc", "ngSrcset", "sizes", "width", "height", "decoding", "loading", "priority", "loaderParams", "disableOptimizedSrcset", "fill", "placeholder", "placeholderConfig", "src", "srcset"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: MatDialogContent, selector: "[mat-dialog-content], mat-dialog-content, [matDialogContent]" }, { kind: "directive", type: MatDialogTitle, selector: "[mat-dialog-title], [matDialogTitle]", inputs: ["id"], exportAs: ["matDialogTitle"] }, { kind: "component", type: MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: ChatFlowComponent, selector: "app-chat-flow", inputs: ["scrollToBottomTrigger", "loadHistory", "messageListInput", "selectedForEdit", "requestDelete"], outputs: ["scrollToBottomTriggerChange", "loadHistoryChange", "messageListInputChange", "selectedForEditChange", "requestDeleteChange"] }, { kind: "component", type: InputMessageComponent, selector: "app-input-message", inputs: ["editMessage", "cancelEdit", "input_text", "files", "previews", "messageEvent"], outputs: ["cancelEditChange", "input_textChange", "filesChange", "previewsChange", "messageEventChange"] }, { kind: "ngmodule", type: TranslocoModule }, { kind: "pipe", type: i2.TranslocoPipe, name: "transloco" }] });
874
916
  }
875
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: NgxParlComponent, decorators: [{
917
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: NgxParlComponent, decorators: [{
876
918
  type: Component,
877
919
  args: [{ selector: 'ngx-parl', standalone: true, imports: [
878
920
  NgOptimizedImage, NgClass, MatDialogContent, MatDialogTitle, MatProgressSpinner, ChatFlowComponent, InputMessageComponent,
879
921
  TranslocoModule,
880
922
  TranslocoPipe
881
- ], providers: [], template: "<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.modal-chat{width:800px;height:600px;border-radius:16px}.modal-chat__body{display:flex;flex-direction:column;height:100%}.modal-chat__header{height:56px;min-height:56px;max-height:56px;display:flex;align-items:center;justify-content:space-between;padding:0 16px;gap:12px;border-top-left-radius:16px;border-top-right-radius:16px;flex-direction:row}.modal-chat__hide-block{width:48px;color:#5a72d7}.modal-chat__title #modal-chat__heading{color:#fff;font:600 16px/20px inter,sans-serif}.modal-chat__actions{display:flex;gap:8px}.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;background:#fff;border-bottom-left-radius:16px;border-bottom-right-radius:16px}.modal-chat__content--header{border-top-left-radius:16px;border-top-right-radius:16px}.modal-chat__content app-chat-flow{flex:1 1 auto;min-height:0px;overflow:auto;display:flex;align-items:flex-end}.modal-chat__input{flex:0px 0px auto;border-bottom-left-radius:16px;border-bottom-right-radius:16px}:host ::ng-deep .mat-mdc-dialog-content.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;padding:0;max-height:none;overflow:hidden}img{cursor:pointer}\n"] }]
882
- }], ctorParameters: () => [{ type: UtilsService }, { type: i2.TranslocoService }], propDecorators: { theme: [{ type: i0.Input, args: [{ isSignal: true, alias: "theme", required: false }] }], header: [{ type: i0.Input, args: [{ isSignal: true, alias: "header", required: false }] }], language: [{ type: i0.Input, args: [{ isSignal: true, alias: "language", required: false }] }], messageList: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageList", required: false }] }, { type: i0.Output, args: ["messageListChange"] }], messageUpdate: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageUpdate", required: false }] }, { type: i0.Output, args: ["messageUpdateChange"] }], selectedForEdit: [{ type: i0.Input, args: [{ isSignal: true, alias: "selectedForEdit", required: false }] }, { type: i0.Output, args: ["selectedForEditChange"] }], messageAction: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageAction", required: false }] }, { type: i0.Output, args: ["messageActionChange"] }], hideHandler: [{ type: i0.Input, args: [{ isSignal: true, alias: "hideHandler", required: false }] }], closeHandler: [{ type: i0.Input, args: [{ isSignal: true, alias: "closeHandler", required: false }] }], scrollToBottomTrigger: [{ type: i0.Input, args: [{ isSignal: true, alias: "scrollToBottomTrigger", required: false }] }, { type: i0.Output, args: ["scrollToBottomTriggerChange"] }] } });
923
+ ], providers: [], template: "<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\"\n [(loadHistory)]=\"loadHistory\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n", styles: [".flow-theme-primary ::ng-deep .modal-chat__header{background-color:#5a72d7;color:#fff}.flow-theme-primary ::ng-deep .message--incoming .message__body{background:#e9ecef;color:#343a40}.flow-theme-primary ::ng-deep .message--outgoing .message__body{background:#4656ca;color:#fff}:root{--shadow-sm: 0px 1px 2px 0px rgba(0, 0, 0, .05);--shadow-md: 0px 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0px 8px 24px rgba(0, 0, 0, .12);--shadow-soft: 0px 0px 6px 0px rgba(46, 46, 46, .12) }.flow-theme-secondary ::ng-deep .modal-chat__header{background-color:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--incoming .message__body{background:#848af5;color:#fefefe}.flow-theme-secondary ::ng-deep .message--outgoing .message__body{background:#efefef;color:#464646}.modal-chat{width:800px;height:600px;border-radius:16px}.modal-chat__body{display:flex;flex-direction:column;height:100%}.modal-chat__header{height:56px;min-height:56px;max-height:56px;display:flex;align-items:center;justify-content:space-between;padding:0 16px;gap:12px;border-top-left-radius:16px;border-top-right-radius:16px;flex-direction:row}.modal-chat__hide-block{width:48px;color:#5a72d7}.modal-chat__title #modal-chat__heading{color:#fff;font:600 16px/20px inter,sans-serif}.modal-chat__actions{display:flex;gap:8px}.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;background:#fff;border-bottom-left-radius:16px;border-bottom-right-radius:16px}.modal-chat__content--header{border-top-left-radius:16px;border-top-right-radius:16px}.modal-chat__content app-chat-flow{flex:1 1 auto;min-height:0px;overflow:auto;display:flex;align-items:flex-end}.modal-chat__input{flex:0px 0px auto;border-bottom-left-radius:16px;border-bottom-right-radius:16px}:host ::ng-deep .mat-mdc-dialog-content.modal-chat__content{flex:1 1 auto;display:flex;flex-direction:column;padding:0;max-height:none;overflow:hidden}img{cursor:pointer}\n"] }]
924
+ }], ctorParameters: () => [{ type: UtilsService }, { type: i2.TranslocoService }], propDecorators: { theme: [{ type: i0.Input, args: [{ isSignal: true, alias: "theme", required: false }] }], header: [{ type: i0.Input, args: [{ isSignal: true, alias: "header", required: false }] }], language: [{ type: i0.Input, args: [{ isSignal: true, alias: "language", required: false }] }], messageList: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageList", required: false }] }, { type: i0.Output, args: ["messageListChange"] }], messageUpdate: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageUpdate", required: false }] }, { type: i0.Output, args: ["messageUpdateChange"] }], selectedForEdit: [{ type: i0.Input, args: [{ isSignal: true, alias: "selectedForEdit", required: false }] }, { type: i0.Output, args: ["selectedForEditChange"] }], messageAction: [{ type: i0.Input, args: [{ isSignal: true, alias: "messageAction", required: false }] }, { type: i0.Output, args: ["messageActionChange"] }], hideHandler: [{ type: i0.Input, args: [{ isSignal: true, alias: "hideHandler", required: false }] }], closeHandler: [{ type: i0.Input, args: [{ isSignal: true, alias: "closeHandler", required: false }] }], scrollToBottomTrigger: [{ type: i0.Input, args: [{ isSignal: true, alias: "scrollToBottomTrigger", required: false }] }, { type: i0.Output, args: ["scrollToBottomTriggerChange"] }], loadHistory: [{ type: i0.Input, args: [{ isSignal: true, alias: "loadHistory", required: false }] }, { type: i0.Output, args: ["loadHistoryChange"] }] } });
883
925
 
884
926
  var chat$1 = {
885
927
  title: "Telegram",
@@ -931,10 +973,10 @@ class ParlTranslocoLoader {
931
973
  return of(en);
932
974
  }
933
975
  }
934
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ParlTranslocoLoader, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
935
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ParlTranslocoLoader });
976
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ParlTranslocoLoader, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
977
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ParlTranslocoLoader });
936
978
  }
937
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.7", ngImport: i0, type: ParlTranslocoLoader, decorators: [{
979
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "20.3.16", ngImport: i0, type: ParlTranslocoLoader, decorators: [{
938
980
  type: Injectable
939
981
  }] });
940
982
 
@@ -1 +1 @@
1
- {"version":3,"file":"trixwell-ngx-parl.mjs","sources":["../../../projects/ngx-parl/src/lib/core/entity/chat.ts","../../../projects/ngx-parl/src/lib/core/components/chat-message/chat-message.ts","../../../projects/ngx-parl/src/lib/core/components/chat-message/chat-message.html","../../../projects/ngx-parl/src/lib/core/service/utils/utils.ts","../../../projects/ngx-parl/src/lib/core/pipes/chat-start-day-pipe.ts","../../../projects/ngx-parl/src/lib/core/pipes/toggle-display-chat-start-day-pipe.ts","../../../projects/ngx-parl/src/lib/chat-flow/chat-flow.ts","../../../projects/ngx-parl/src/lib/chat-flow/chat-flow.html","../../../projects/ngx-parl/src/lib/core/entity/file.ts","../../../projects/ngx-parl/src/lib/input-message/input-message.ts","../../../projects/ngx-parl/src/lib/input-message/input-message.html","../../../projects/ngx-parl/src/lib/core/entity/theme.ts","../../../projects/ngx-parl/src/lib/ngx-parl/ngx-parl.ts","../../../projects/ngx-parl/src/lib/ngx-parl/ngx-parl.html","../../../projects/ngx-parl/src/assets/i18n/parl-transloco.loader.ts","../../../projects/ngx-parl/src/lib/ngx-parl.providers.ts","../../../projects/ngx-parl/src/trixwell-ngx-parl.ts"],"sourcesContent":["export class ChatMessage {\n public id: number;\n public chat_id: number;\n public cr_time: string;\n public type: ChatMessageType;\n public user: string;\n public content: string;\n public avatar: string | null;\n public file_path: string[] | null;\n public file_list: File[] | null;\n\n public checked: boolean | null;\n public edit = false;\n\n constructor(data: ChatMessageDTO) {\n this.id = data.id;\n this.chat_id = data.chat_id;\n this.cr_time = data.cr_time;\n this.type = data.type;\n this.user = data.user;\n this.content = data.content;\n this.avatar = data.avatar ?? null;\n this.checked = data.checked ?? null;\n this.file_path = data.file_path ?? null;\n this.file_list = data.file_list ?? null;\n }\n\n get dateSimple(): string {\n const d = new Date(this.cr_time.replace(' ', 'T'));\n const dd = String(d.getDate()).padStart(2, '0');\n const mm = String(d.getMonth() + 1).padStart(2, '0');\n const yyyy = d.getFullYear();\n return `${dd}.${mm}.${yyyy}`;\n }\n\n get timeHHmm(): string {\n const d = new Date(this.cr_time.replace(' ', 'T'));\n const hh = String(d.getHours()).padStart(2, '0');\n const mm = String(d.getMinutes()).padStart(2, '0');\n return `${hh}:${mm}`;\n }\n}\n\nexport interface ChatMessageDTO {\n id: number;\n chat_id: number;\n cr_time: string; // ISO or 'YYYY-MM-DD HH:mm:ss'\n type: ChatMessageType;\n user: string;\n content: string;\n avatar?: string | null;\n file_path?: string[] | null;\n file_list?: File[] | [];\n checked?: boolean | null;\n}\n\nexport type ChatMessageType = 'incoming' | 'outgoing';\n\nexport enum MessageType {\n Incoming = 'incoming',\n Outgoing = 'outgoing'\n}\n\nexport interface CurrMessage {\n id?: number;\n content: string;\n files?: string[];\n file_list?: File[];\n}\n\nexport type MessageActionType = 'send' | 'edit' | 'delete';\n\nexport interface MessageActionEvent {\n action: MessageActionType;\n chatMessageId?: number;\n content: string;\n files?: string[];\n file_list?: File[];\n}\n","import {Component, computed, input, model} from '@angular/core';\nimport {DatePipe, NgClass, NgOptimizedImage} from '@angular/common';\nimport {ChatMessage, MessageType} from '../../entity/chat';\nimport {MatMenu, MatMenuItem, MatMenuTrigger} from '@angular/material/menu';\nimport {TranslocoPipe} from '@ngneat/transloco';\n\n@Component({\n selector: 'lib-chat-message',\n imports: [\n NgClass,\n NgOptimizedImage,\n DatePipe,\n MatMenu,\n MatMenuItem,\n MatMenuTrigger,\n TranslocoPipe,\n ],\n templateUrl: './chat-message.html',\n styleUrl: './chat-message.scss',\n standalone: true,\n})\n\nexport class ChatMessageComponent {\n public currentMessage = input.required<ChatMessage>();\n public edit = model<boolean>(false);\n\n public requestEdit = model<ChatMessage | null>(null);\n public requestDelete = model<number | null>(null);\n\n constructor() {}\n\n private normalizeSourcePath(sourcePath: string): string {\n const cleanedPath = (sourcePath ?? '').trim();\n if (!cleanedPath) {\n return '';\n }\n\n if (cleanedPath.startsWith('data:') || cleanedPath.startsWith('blob:') || /^https?:\\/\\//i.test(cleanedPath)) {\n return cleanedPath;\n }\n\n const assetsIndex = cleanedPath.indexOf('assets/');\n if (assetsIndex >= 0) {\n return '/' + cleanedPath.slice(assetsIndex);\n }\n\n return cleanedPath.replace(/^\\.{1,2}\\//, '/');\n }\n\n attachments = computed(() => {\n const message = this.currentMessage();\n const filePath = message.file_path;\n\n if (Array.isArray(filePath)) {\n return filePath.map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n\n const rawFilePath = (filePath as unknown as string) ?? '';\n if (typeof rawFilePath !== 'string' || !rawFilePath.trim) {\n return [];\n }\n\n if (rawFilePath.trim().startsWith('[')) {\n try {\n const parsed = JSON.parse(rawFilePath);\n if (Array.isArray(parsed)) {\n return parsed\n .map(item => (typeof item === 'string' ? this.normalizeSourcePath(item) : ''))\n .filter(Boolean);\n }\n } catch {}\n }\n\n if (rawFilePath.startsWith('data:')) {\n return [rawFilePath];\n }\n\n if (rawFilePath.includes('|')) {\n return rawFilePath.split('|').map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n if (rawFilePath.includes(',')) {\n return rawFilePath.split(',').map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n\n return [];\n });\n\n avatarSrc = computed(() => {\n const message = this.currentMessage();\n const fallback = message.type === 'incoming'\n ? 'assets/ngx-parl/icons/avatar_anonym.svg'\n : 'assets/ngx-parl/icons/avatar_manager.svg';\n\n return message.avatar || fallback;\n });\n\n openContextMenu(event: Event, trigger: any) {\n event.preventDefault();\n event.stopPropagation();\n trigger.openMenu();\n\n return this;\n }\n\n editMessage(message: ChatMessage) {\n this.edit.set(true);\n this.requestEdit.set(message);\n\n return this;\n }\n\n deleteMessage(message: ChatMessage) {\n this.requestDelete.set(message.id);\n queueMicrotask(() => this.requestDelete.set(null));\n\n return this;\n }\n\n canDelete(message: ChatMessage): boolean {\n return message.type === this.messageType.Outgoing;\n }\n\n public readonly messageType = MessageType;\n}\n","<div class=\"message\"\n [ngClass]=\"{\n 'message--outgoing': currentMessage().type === messageType.Outgoing,\n 'message--incoming': currentMessage().type === messageType.Incoming\n }\">\n <div class=\"message__avatar\">\n <img [ngSrc]=\"avatarSrc()\" width=\"36\" height=\"36\" alt=\"avatar\"/>\n </div>\n\n <div class=\"message__body\"\n #menuTrigger=\"matMenuTrigger\"\n [matMenuTriggerFor]=\"messageMenu\"\n (contextmenu)=\"openContextMenu($event, menuTrigger)\"\n (keydown.shift.F10)=\"openContextMenu($event, menuTrigger)\">\n\n @if (attachments().length) {\n <div class=\"message__attachments\">\n @for (file of attachments(); track file) {\n <img [src]=\"file\"\n alt=\"attachment\"\n class=\"message__image\"\n loading=\"lazy\"/>\n }\n </div>\n }\n\n <div class=\"message__bubble\" tabindex=\"0\">\n <div class=\"message__text\">{{ currentMessage().content }}</div>\n\n <div class=\"message__meta\">\n <time class=\"message__time\">{{ currentMessage().cr_time | date:'HH:mm' }}</time>\n @if (currentMessage().type === messageType.Outgoing) {\n @if (currentMessage().checked) {\n <img ngSrc=\"assets/ngx-parl/icons/checked-message.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n } @else {\n <img ngSrc=\"assets/ngx-parl/icons/no-check.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n }\n }\n </div>\n </div>\n </div>\n</div>\n\n<mat-menu #messageMenu=\"matMenu\" xPosition=\"before\" yPosition=\"below\" class=\"message__menu\">\n @if (currentMessage().type === messageType.Outgoing) {\n <button mat-menu-item (click)=\"editMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.edit' | transloco }}</span>\n </button>\n }\n @if (canDelete(currentMessage())) {\n <button mat-menu-item (click)=\"deleteMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/trash.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.remove' | transloco }}</span>\n </button>\n }\n</mat-menu>\n","import {Injectable} from '@angular/core';\nimport {HttpClient} from '@angular/common/http';\n\n\n@Injectable({\n providedIn: 'root'\n})\nexport class UtilsService {\n\n constructor(protected http: HttpClient) {\n }\n\n langToLocale(lang: string): string {\n switch (lang) {\n case 'uk':\n return 'uk-UA';\n case 'en':\n default:\n return 'en-US';\n }\n }\n\n getLocalISODate(): string {\n const d = new Date();\n\n const year = d.getFullYear();\n const month = String(d.getMonth() + 1).padStart(2, '0');\n const day = String(d.getDate()).padStart(2, '0');\n\n const hours = String(d.getHours()).padStart(2, '0');\n const minutes = String(d.getMinutes()).padStart(2, '0');\n const seconds = String(d.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}`;\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\nimport {DatePipe} from '@angular/common';\nimport {UtilsService} from '../service/utils/utils';\nimport {TranslocoService} from '@ngneat/transloco';\n\n@Pipe({\n name: 'chatStartDay'\n})\nexport class ChatStartDayPipe implements PipeTransform {\n constructor(protected utils: UtilsService, private transloco: TranslocoService) {}\n\n transform(value: string, format: string = 'd MMMM'): string {\n if (!value) {\n return '';\n }\n\n const locale = this.utils.langToLocale(this.transloco.getActiveLang());\n const datePipe = new DatePipe(locale);\n\n const valueDate = new Date(value);\n const today = new Date();\n\n const isToday = datePipe.transform(valueDate, 'shortDate') === datePipe.transform(today, 'shortDate');\n\n return isToday\n ? (locale.startsWith('uk') ? 'Сьогодні' : 'Today')\n : (datePipe.transform(valueDate, format) ?? '');\n }\n\n}\n","import {Pipe, PipeTransform} from '@angular/core';\nimport {DatePipe} from '@angular/common';\nimport {ChatMessage} from '../entity/chat';\nimport {UtilsService} from '../service/utils/utils';\nimport {TranslocoService} from '@ngneat/transloco';\n\n@Pipe({\n name: 'toggleDisplayChatStartDay'\n})\nexport class ToggleDisplayChatStartDayPipe implements PipeTransform {\n\n constructor(protected utils: UtilsService, private transloco: TranslocoService) {}\n\n transform(message: ChatMessage, messages: ChatMessage[], i: number): boolean {\n const locale = this.utils.langToLocale(this.transloco.getActiveLang());\n const datePipe = new DatePipe(locale);\n\n const prev = i > 0 ? messages[i - 1] : undefined;\n\n const currDay = datePipe.transform(new Date(message.cr_time), 'shortDate');\n const prevDay = prev ? datePipe.transform(new Date(prev.cr_time), 'shortDate') : undefined;\n\n return prev ? currDay !== prevDay : true;\n }\n}\n","import {AfterViewInit, Component, computed, effect, ElementRef, model, ViewChild,} from '@angular/core';\nimport {FormsModule} from '@angular/forms';\nimport {ChatMessage} from '../core/entity/chat';\nimport {ChatMessageComponent} from '../core/components/chat-message/chat-message';\nimport {ChatStartDayPipe} from '../core/pipes/chat-start-day-pipe';\nimport {ToggleDisplayChatStartDayPipe} from '../core/pipes/toggle-display-chat-start-day-pipe';\nimport {TranslocoPipe} from '@ngneat/transloco';\nimport {NgOptimizedImage} from '@angular/common';\n\n@Component({\n selector: 'app-chat-flow',\n imports: [\n FormsModule,\n ChatMessageComponent,\n ChatStartDayPipe,\n ToggleDisplayChatStartDayPipe,\n ChatMessageComponent,\n TranslocoPipe,\n NgOptimizedImage\n ],\n templateUrl: './chat-flow.html',\n styleUrl: './chat-flow.scss',\n standalone: true,\n})\n\nexport class ChatFlowComponent implements AfterViewInit {\n @ViewChild('chatFlowRef') flowRef?: ElementRef<HTMLElement>;\n public scrollToBottomTrigger = model<number>(0);\n\n public messageListInput = model.required<ChatMessage[]>();\n public messageList = computed(() => this.messageListInput());\n\n public selectedForEdit = model.required<ChatMessage | null>();\n public requestDelete = model<number | null>(null);\n\n private viewInitialized = false;\n\n constructor() {\n effect(() => {\n const length = this.messageList().length;\n\n if (!this.viewInitialized || length === 0) {\n return;\n }\n\n queueMicrotask(() => this.scrollToBottomSmooth());\n });\n\n effect(() => {\n this.scrollToBottomTrigger();\n if (this.viewInitialized) {\n queueMicrotask(() => this.scrollToBottomSmooth());\n }\n });\n }\n\n ngAfterViewInit() {\n this.viewInitialized = true;\n\n queueMicrotask(() => this.scrollToBottomSmooth());\n }\n\n scrollToBottomSmooth() {\n const element = this.flowRef?.nativeElement;\n\n if (!element) {\n return this;\n }\n\n element.scrollTo({\n top: element.scrollHeight,\n behavior: 'smooth',\n });\n\n return this;\n }\n\n startEdit(message: ChatMessage) {\n this.messageList().forEach(currentMessage => {\n if (currentMessage.id !== message.id && currentMessage.edit) {\n currentMessage.edit = false;\n }\n });\n\n message.edit = true;\n\n if (this.selectedForEdit()?.id === message.id) {\n this.selectedForEdit.set(null);\n queueMicrotask(() => this.selectedForEdit.set(message));\n } else {\n this.selectedForEdit.set(message);\n }\n\n return this;\n }\n\n onEditChange(id: number, isEdit: boolean) {\n const message = this.messageList().find(item => item.id === id);\n\n if (!message) {\n return this;\n }\n\n if (isEdit) {\n return this.startEdit(message);\n }\n\n message.edit = false;\n\n if (this.selectedForEdit()?.id === id) {\n this.selectedForEdit.set(null);\n }\n\n return this;\n }\n\n onRequestEdit(message: ChatMessage | null) {\n if (message) {\n return this.startEdit(message);\n }\n\n this.selectedForEdit.set(null);\n\n return this;\n }\n\n onRequestDelete(messageId: number | null) {\n if (messageId == null) {\n return this;\n }\n\n this.selectedForEdit.set(null);\n\n this.requestDelete.set(messageId);\n queueMicrotask(() => this.requestDelete.set(null));\n\n return this;\n }\n\n trackByMessageId(_index: number, message: ChatMessage): string {\n // return message.id;\n return `${message.chat_id}-${message.type}-${message.id}`;\n }\n}\n","<div class=\"chat\">\n @if (messageList()) {\n <div class=\"chat__flow\" #chatFlowRef>\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <lib-chat-message [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n","export class ImageFile {\n constructor(\n public id: string,\n public url: string,\n public cr_time: string,\n ) {}\n}\n\nexport type OriginalKind = 'image' | 'gif';\n\nexport interface PreviewItem {\n originalKind: OriginalKind;\n duration?: number;\n src: string;\n name: string;\n type: string;\n size: number;\n}\n\nexport enum FileType {\n IMAGE = 'image',\n GIF = 'gif'\n}\n","import {\n AfterViewInit,\n Component,\n computed,\n effect,\n ElementRef,\n input,\n model,\n OnDestroy,\n signal,\n ViewChild\n} from '@angular/core';\nimport {FileType, OriginalKind, PreviewItem} from '../core/entity/file';\nimport {TranslocoPipe} from '@ngneat/transloco';\nimport {ChatMessage, CurrMessage, MessageActionEvent, MessageActionType} from '../core/entity/chat';\nimport {NgOptimizedImage} from '@angular/common';\n\n@Component({\n selector: 'app-input-message',\n imports: [TranslocoPipe, NgOptimizedImage],\n templateUrl: './input-message.html',\n styleUrl: './input-message.scss',\n standalone: true,\n})\n\nexport class InputMessageComponent implements AfterViewInit, OnDestroy {\n @ViewChild('inputText', {static: false}) inputTextElement!: ElementRef<HTMLDivElement>;\n @ViewChild('mirror', {static: false}) mirrorElement!: ElementRef<HTMLDivElement>;\n\n public editMessage = input<ChatMessage | { id: number; content: string; file_path?: string[] | null } | null>(null);\n\n public hasOriginalAttachments = computed(() => {\n const filePaths = this.editFilePaths();\n return filePaths.length > 0;\n });\n\n public hasNewAttachments = computed(() => (this.previews()?.length ?? 0) > 0);\n\n public cancelEdit = model<number | null>(null);\n public input_text = model<string | CurrMessage>('');\n\n public draft = signal<string>('');\n public focused = signal<boolean>(false);\n public sending = signal<boolean>(false);\n public hasText = computed(() => this.draft().trim().length > 0);\n\n public isEditMode = computed(() => !!this.editMessage());\n public canSend = computed(() =>\n this.hasText() ||\n this.hasNewAttachments() ||\n (this.isEditMode() && this.hasOriginalAttachments())\n );\n\n public files = model<File[]>([]);\n public previews = model<PreviewItem[]>([]);\n\n private lastHeightPx = 0;\n private lastRows = 1;\n private resizeRaf: number | null = null;\n\n public messageEvent = model<MessageActionEvent | null>(null);\n\n constructor() {\n effect(() => {\n const message = this.editMessage();\n const element = this.inputTextElement?.nativeElement;\n\n if (!element) {\n return;\n }\n\n if (message) {\n const content = (message as any).content ?? '';\n this.draft.set(content);\n element.innerText = content;\n\n queueMicrotask(() => {\n this.autoResizeByRows();\n element.focus();\n this.focused.set(true);\n });\n }\n });\n }\n\n ngAfterViewInit() {\n const element = this.inputTextElement.nativeElement;\n element.style.transition = 'height 160ms ease';\n this.initMirror();\n\n const computedStyle = getComputedStyle(element);\n const lineHeight = this.cssNum(computedStyle.lineHeight, 24);\n\n element.style.height = `${lineHeight}px`;\n this.lastHeightPx = lineHeight;\n this.lastRows = 1;\n this.updateOverflow(1);\n\n requestAnimationFrame(() => {\n const {rows, nextHeightPx} = this.measureByMirror();\n element.style.height = `${nextHeightPx}px`;\n\n this.lastRows = rows;\n this.lastHeightPx = nextHeightPx;\n this.updateOverflow(rows);\n });\n }\n\n ngOnDestroy() {\n if (this.resizeRaf) {\n cancelAnimationFrame(this.resizeRaf);\n this.resizeRaf = null;\n }\n }\n\n editFilePaths(): string[] {\n const message = this.editMessage();\n if (!message) {\n return [];\n }\n\n const file_path = (message as any).file_path;\n\n return Array.isArray(file_path) ? file_path : [];\n }\n\n collectAttachmentSources(): string[] {\n return (this.previews() ?? []).map(p => p.src).filter(Boolean);\n }\n\n cancelEditMessage() {\n const message = this.editMessage();\n this.cancelEdit.set((message as any)?.id ?? null);\n queueMicrotask(() => this.cancelEdit.set(null));\n\n this.draft.set('');\n const element = this.inputTextElement?.nativeElement;\n\n if (element) {\n element.innerHTML = '';\n this.autoResizeByRows();\n element.focus();\n }\n\n return this;\n }\n\n enterDown() {\n const element = this.inputTextElement.nativeElement;\n const text = this.draft().trim();\n\n if (!this.canSend()) {\n return this;\n }\n\n this.sending.set(true);\n\n const files = this.collectAttachmentSources();\n const fileList = this.files();\n\n const message = this.editMessage();\n const action: MessageActionType = message ? 'edit' : 'send';\n\n this.messageEvent.set({\n action,\n chatMessageId: message ? message.id : undefined,\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n\n queueMicrotask(() => this.messageEvent.set(null));\n\n if (message) {\n this.input_text.set({\n id: message.id,\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n } else {\n this.input_text.set({\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n }\n\n this.draft.set('');\n element.innerHTML = '';\n this.files.set([]);\n this.previews.set([]);\n element.focus();\n this.autoResizeByRows();\n\n setTimeout(() => this.sending.set(false), 150);\n\n return this;\n }\n\n onFocus() {\n if (this.inputTextElement.nativeElement.innerHTML === '<br>') {\n this.inputTextElement.nativeElement.innerHTML = '';\n }\n this.focused.set(true);\n\n return this;\n }\n\n onBlur() {\n this.focused.set(false);\n\n return this;\n }\n\n onKeyDown(event: KeyboardEvent) {\n if (event.key === 'Enter' && !event.shiftKey) {\n event.preventDefault();\n this.enterDown();\n\n return this;\n }\n queueMicrotask(() => this.autoResizeByRows());\n\n return this;\n }\n\n onInput() {\n this.draft.set(this.inputTextElement.nativeElement.innerText ?? '');\n this.autoResizeByRows();\n\n return this;\n }\n\n onPaste() {\n queueMicrotask(() => {\n this.draft.set(this.inputTextElement.nativeElement.innerText ?? '');\n this.autoResizeByRows();\n });\n\n return this;\n }\n\n inputFileChange(event: Event) {\n const inputEl = event.target as HTMLInputElement;\n const selected = inputEl.files;\n\n if (!selected?.length) {\n inputEl.value = '';\n\n return this;\n }\n\n const list = Array.from(selected).filter(f => (f.type || '').startsWith('image/'));\n if (!list.length) {\n inputEl.value = '';\n\n return this;\n }\n\n this.files.set([...(this.files() ?? []), ...list]);\n\n Promise.all(\n list.map(async f => {\n const src = await this.readFileAsDataURL(f);\n const originalKind: OriginalKind = (f.type || '') === 'image/gif' ? FileType.GIF : FileType.IMAGE;\n return <PreviewItem>{src, originalKind, name: f.name, type: f.type || '', size: f.size};\n })\n )\n .then(items => this.previews.set([...(this.previews() ?? []), ...items]))\n .finally(() => (inputEl.value = ''));\n\n return this;\n }\n\n removeFile(index: number, event?: MouseEvent) {\n event?.stopPropagation();\n event?.preventDefault();\n\n const previews = [...(this.previews() ?? [])];\n previews.splice(index, 1);\n this.previews.set(previews);\n\n const filesArr = [...(this.files() ?? [])];\n if (index >= 0 && index < filesArr.length) {\n filesArr.splice(index, 1);\n this.files.set(filesArr);\n }\n return this;\n }\n\n openPreview(_item: PreviewItem, _index: number) {\n return this;\n }\n\n autoResizeByRows() {\n const element = this.inputTextElement.nativeElement;\n const {rows, nextHeightPx} = this.measureByMirror();\n\n if (rows === this.lastRows) {\n this.updateOverflow(rows);\n\n return this;\n }\n\n if (this.resizeRaf) {\n cancelAnimationFrame(this.resizeRaf);\n }\n\n element.style.height = `${this.lastHeightPx}px`;\n\n this.resizeRaf = requestAnimationFrame(() => {\n element.style.height = `${nextHeightPx}px`;\n this.lastHeightPx = nextHeightPx;\n this.lastRows = rows;\n this.updateOverflow(rows);\n });\n\n return this;\n }\n\n measureByMirror(): { rows: number; nextHeightPx: number } {\n const inputEl = this.inputTextElement.nativeElement;\n const mirrorEl = this.mirrorElement.nativeElement;\n const computedStyle = getComputedStyle(inputEl);\n\n let text = inputEl.innerText;\n if (!text || text === '\\n') {\n text = '\\u00A0';\n }\n\n mirrorEl.style.width = computedStyle.width;\n mirrorEl.textContent = text;\n\n const lineHeight = this.cssNum(computedStyle.lineHeight, 24);\n const paddingTop = this.cssNum(computedStyle.paddingTop, 0);\n const paddingBottom = this.cssNum(computedStyle.paddingBottom, 0);\n const paddingY = paddingTop + paddingBottom;\n\n const maxRowsCss = computedStyle.getPropertyValue('--max-rows').trim();\n const maxRows = maxRowsCss ? this.cssNum(maxRowsCss, 8) : 8;\n\n const contentH = mirrorEl.offsetHeight;\n const rawRows = Math.max(1, Math.ceil(contentH / lineHeight));\n const rows = Math.min(rawRows, maxRows);\n\n const nextHeightPx = Math.round(rows * lineHeight + paddingY);\n\n return {rows, nextHeightPx};\n }\n\n initMirror() {\n const mirror = this.mirrorElement.nativeElement;\n const input = this.inputTextElement.nativeElement;\n const computedStyle = getComputedStyle(input);\n\n mirror.style.position = 'absolute';\n mirror.style.visibility = 'hidden';\n mirror.style.pointerEvents = 'none';\n mirror.style.zIndex = '-1';\n mirror.style.whiteSpace = 'pre-wrap';\n mirror.style.overflowWrap = 'break-word';\n mirror.style.wordBreak = 'normal';\n\n const properties = [\n 'font', 'font-size', 'font-family', 'font-weight', 'font-style',\n 'line-height', 'letter-spacing', 'word-spacing',\n 'padding-top', 'padding-bottom', 'padding-left', 'padding-right',\n 'border-top-width', 'border-bottom-width', 'border-left-width', 'border-right-width',\n 'white-space', 'text-transform', 'box-sizing'\n ];\n\n properties.forEach(property => (mirror.style as any)[property] = computedStyle.getPropertyValue(property));\n mirror.style.paddingTop = '0px';\n mirror.style.paddingBottom = '0px';\n }\n\n updateOverflow(rows: number) {\n const element = this.inputTextElement.nativeElement;\n const computedStyle = getComputedStyle(element);\n const maxRowsCss = computedStyle.getPropertyValue('--max-rows').trim();\n const maxRows = maxRowsCss ? this.cssNum(maxRowsCss, 8) : 8;\n element.style.overflowY = rows >= maxRows ? 'auto' : 'hidden';\n\n return this;\n }\n\n readFileAsDataURL(file: File): Promise<string> {\n return new Promise((resolve, reject) => {\n const reader = new FileReader();\n reader.onload = e => resolve((e.target?.result as string) || '');\n reader.onerror = reject;\n reader.readAsDataURL(file);\n });\n }\n\n cssNum(v: string, fb = 0): number {\n const n = parseFloat(v);\n return Number.isFinite(n) ? n : fb;\n }\n\n protected readonly FileType = FileType;\n}\n","<section class=\"message\"\n [class.message--focus]=\"focused()\"\n [class.message--filled]=\"hasText()\"\n [class.message--sending]=\"sending()\">\n\n @if (isEditMode()) {\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button class=\"message__button message__button--attach\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon message__icon--edit\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__edit-hint\">\n <p>{{ editMessage()?.content }}</p>\n </div>\n </div>\n\n <button class=\"message__button message__button--close\"\n type=\"button\"\n (click)=\"cancelEditMessage()\">\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\"\n class=\"message__icon message__icon--close\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n </div>\n }\n\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button type=\"button\"\n class=\"message__button message__button--attach\"\n [attr.aria-label]=\"'chat.attach' | transloco\"\n (click)=\"fileDialog.click()\">\n <img ngSrc=\"assets/ngx-parl/icons/attach-filled.svg\"\n class=\"message__icon message__icon--attach\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <input #fileDialog\n type=\"file\"\n class=\"hidden\"\n accept=\"image/*\"\n multiple\n (change)=\"inputFileChange($event)\">\n\n <div #inputText\n role=\"textbox\"\n class=\"message__input\"\n spellcheck=\"true\"\n contenteditable=\"true\"\n aria-multiline=\"true\"\n [attr.data-placeholder]=\"isEditMode()\n ? ('chat.edit_placeholder' | transloco)\n : ('chat.placeholder' | transloco)\"\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (input)=\"onInput()\"\n (paste)=\"onPaste()\"\n (keydown)=\"onKeyDown($event)\">\n </div>\n\n <div #mirror class=\"message__input-mirror\" aria-hidden=\"true\"></div>\n\n <ng-content></ng-content>\n </div>\n\n <button class=\"message__button message__button--send\"\n type=\"button\"\n (click)=\"enterDown()\"\n [attr.aria-label]=\"'chat.send' | transloco\"\n [disabled]=\"!canSend()\">\n <img ngSrc=\"assets/ngx-parl/icons/send.svg\"\n class=\"message__icon message__icon--send\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n\n </button>\n </div>\n\n @if (previews().length) {\n <div class=\"message__files\">\n @for (item of previews(); track $index) {\n <div class=\"message__file\" [attr.data-kind]=\"item.originalKind\">\n <button type=\"button\"\n class=\"message__file-remove\"\n [attr.aria-label]=\"'chat.delete_file' | transloco\"\n (click)=\"removeFile($index, $event)\">\n <img ngSrc=\"assets/ngx-parl/icons/remove.svg\"\n class=\"message__icon message__icon--file\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__file-media\" (click)=\"openPreview(item, $index)\">\n <img [src]=\"item.src\" [alt]=\"item.name\" class=\"message__file-img\"/>\n\n @if (item.originalKind === FileType.GIF) {\n <span class=\"message__file-badge message__file-badge--gif\">\n {{ 'chat.gif' | transloco }}\n </span>\n }\n </div>\n </div>\n }\n </div>\n }\n</section>\n\n\n","export enum FlowTheme {\n PRIMARY = 'primary',\n SECONDARY = 'secondary',\n}\n","import {Component, computed, effect, input, model} from '@angular/core';\nimport {NgClass, NgOptimizedImage} from '@angular/common';\nimport {ChatFlowComponent} from '../chat-flow/chat-flow';\nimport {MatDialogContent, MatDialogTitle} from '@angular/material/dialog';\nimport {\n ChatMessage,\n ChatMessageDTO,\n ChatMessageType,\n CurrMessage,\n MessageActionEvent,\n MessageType\n} from '../core/entity/chat';\nimport {MatProgressSpinner} from '@angular/material/progress-spinner';\nimport {InputMessageComponent} from '../input-message/input-message';\nimport {\n TranslocoModule,\n TranslocoPipe,\n TranslocoService,\n} from '@ngneat/transloco';\nimport {UtilsService} from '../core/service/utils/utils';\nimport {FlowTheme} from '../core/entity/theme';\n\n@Component({\n selector: 'ngx-parl',\n standalone: true,\n imports: [\n NgOptimizedImage, NgClass, MatDialogContent, MatDialogTitle, MatProgressSpinner, ChatFlowComponent, InputMessageComponent,\n TranslocoModule,\n TranslocoPipe\n ],\n templateUrl: './ngx-parl.html',\n styleUrl: './ngx-parl.scss',\n providers: [],\n})\n\nexport class NgxParlComponent {\n public ai_run_in_progress = false;\n private lastUpdateKey: string | null = null;\n\n public theme = input<FlowTheme>(FlowTheme.PRIMARY);\n public header = input<boolean>(true);\n public language = input<'en' | 'uk'>('en');\n\n public messageList = model<ChatMessage[]>([]);\n public messageUpdate = model<ChatMessage>();\n public selectedForEdit = model<ChatMessage | null>(null);\n public messageAction = model<MessageActionEvent | null>(null);\n\n public incomingUser = computed(() => {\n return (\n this.messageList().find((message) => message.type === MessageType.Incoming)?.user ?? ''\n );\n });\n\n public hideHandler = input<(() => unknown) | null>(null);\n public closeHandler = input<(() => unknown) | null>(null);\n\n public scrollToBottomTrigger = model<number>(0);\n\n constructor(private utils: UtilsService,\n private transloco: TranslocoService) {\n effect(() => {\n const lang = this.language();\n this.transloco.setActiveLang(lang);\n });\n\n effect(() => {\n const updatedMessage = this.messageUpdate();\n\n if (!updatedMessage) {\n return;\n }\n\n const key = `${updatedMessage.id}-${updatedMessage.cr_time}-${updatedMessage.type}`;\n if (this.lastUpdateKey === key) {\n return;\n }\n\n this.lastUpdateKey = key;\n\n if (updatedMessage.type !== MessageType.Incoming) {\n return;\n }\n\n this.messageList.update((currentList) => {\n const list = [...currentList];\n\n const incomingIndex = list.findIndex(\n (message) =>\n message.id === updatedMessage.id &&\n message.type === MessageType.Incoming,\n );\n\n if (incomingIndex > -1) {\n list[incomingIndex] = updatedMessage;\n return list;\n }\n\n list.push(updatedMessage);\n\n return list;\n });\n });\n }\n\n onCancelEdit(messageId: number | null) {\n if (messageId != null) {\n this.messageList.update((currentList) => {\n const updatedList = [...currentList];\n const index = updatedList.findIndex(\n (message) => message.id === messageId,\n );\n\n if (index > -1) {\n updatedList[index].edit = false;\n }\n\n return updatedList;\n });\n }\n\n this.selectedForEdit.set(null);\n\n return this;\n }\n\n onRequestDelete(messageId: number | null) {\n if (messageId == null) {\n return this;\n }\n\n this.messageList.update((currentList) => {\n const updatedList = currentList.filter((m) => m.id !== messageId);\n return [...updatedList];\n });\n\n this.pushMessageAction({\n action: 'delete',\n chatMessageId: messageId,\n content: '',\n });\n\n return this;\n }\n\n sendMessage(\n event:\n | CurrMessage\n | string\n | undefined\n ) {\n if (!event) {\n return this;\n }\n\n // edit message\n if (typeof event !== 'string' && 'id' in event) {\n const {id, content, files, file_list} = event;\n\n this.messageList.update((currentList) => {\n const updatedList = [...currentList];\n const index = updatedList.findIndex(\n (message) => message.id === id,\n );\n\n if (index > -1) {\n updatedList[index].content = (content ?? '').trim();\n\n if (Array.isArray(files)) {\n updatedList[index].file_path = files.length ? files : null;\n }\n\n updatedList[index].edit = false;\n }\n\n return updatedList;\n });\n\n this.selectedForEdit.set(null);\n\n this.pushMessageAction({\n action: 'edit',\n chatMessageId: id,\n content: (content ?? '').trim(),\n files: Array.isArray(files) && files.length ? files : [],\n file_list: Array.isArray(file_list) && file_list.length ? file_list : [],\n });\n\n return this;\n }\n\n // new message\n if (typeof event === 'string') {\n const text = event.trim();\n\n if (!text) {\n return this;\n }\n\n const messages = this.messageList();\n const lastId = messages.at(-1)?.id ?? 0;\n\n const lastOutgoing = [...messages]\n .reverse()\n .find((message) => message.type === MessageType.Outgoing,);\n\n const dto: ChatMessageDTO = {\n id: lastId + 1,\n chat_id: lastOutgoing?.chat_id ?? 1,\n cr_time: this.utils.getLocalISODate(),\n type: MessageType.Outgoing as ChatMessageType,\n user: lastOutgoing?.user ?? '',\n content: text,\n avatar: lastOutgoing?.avatar ?? null,\n file_path: [],\n file_list: [],\n checked: false,\n };\n\n this.messageList.update((list) => [...list, new ChatMessage(dto)]);\n\n this.pushMessageAction({\n action: 'send',\n chatMessageId: dto.id,\n content: text,\n });\n\n return this;\n }\n\n // new message + files\n const {content, files, file_list} = event;\n const text = (content ?? '').trim();\n const hasFiles = Array.isArray(files) && files.length > 0;\n const hasFileList = Array.isArray(file_list) && file_list.length > 0;\n\n if (!text && !hasFiles) {\n return this;\n }\n\n const messages = this.messageList();\n const lastId = messages.at(-1)?.id ?? 0;\n\n const lastOutgoing = [...messages]\n .reverse()\n .find((message) => message.type === MessageType.Outgoing,);\n\n const dto: ChatMessageDTO = {\n id: lastId + 1,\n chat_id: lastOutgoing?.chat_id ?? 1,\n cr_time: this.utils.getLocalISODate(),\n type: MessageType.Outgoing as ChatMessageType,\n user: lastOutgoing?.user ?? '',\n content: text,\n avatar: lastOutgoing?.avatar ?? null,\n file_path: hasFiles ? files : [],\n file_list: hasFileList ? file_list : [],\n checked: false,\n };\n\n this.messageList.update((list) => [...list, new ChatMessage(dto)]);\n\n this.pushMessageAction({\n action: 'send',\n chatMessageId: dto.id,\n content: text,\n files: hasFiles ? files : [],\n file_list: hasFileList ? file_list : [],\n });\n\n return this;\n }\n\n pushMessageAction(event: MessageActionEvent) {\n this.messageAction.set(event);\n\n setTimeout(() => this.messageAction.set(null), 0);\n\n return this;\n }\n\n onHideClick() {\n const handler = this.hideHandler();\n\n if (handler) {\n handler();\n }\n\n return this;\n }\n\n onCloseClick() {\n const handler = this.closeHandler();\n\n if (handler) {\n handler();\n }\n\n return this;\n }\n\n scrollToBottom() {\n this.scrollToBottomTrigger.update(value => value + 1);\n\n return this;\n }\n\n protected readonly FlowTheme = FlowTheme;\n}\n","<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n","import { Injectable } from '@angular/core';\nimport { TranslocoLoader, Translation } from '@ngneat/transloco';\nimport { Observable, of } from 'rxjs';\n\nimport en from '../../assets/i18n/en.json';\nimport uk from '../../assets/i18n/uk.json';\n\n@Injectable()\nexport class ParlTranslocoLoader implements TranslocoLoader {\n getTranslation(lang: string): Observable<Translation> {\n switch (lang) {\n case 'uk':\n return of(uk as Translation);\n case 'en':\n default:\n return of(en as Translation);\n }\n }\n}\n","import { isDevMode } from '@angular/core';\nimport { provideTransloco } from '@ngneat/transloco';\nimport { ParlTranslocoLoader } from '../assets/i18n/parl-transloco.loader';\n\nexport function provideNgxParl() {\n return provideTransloco({\n config: {\n availableLangs: ['en', 'uk'],\n defaultLang: 'en',\n reRenderOnLangChange: true,\n prodMode: !isDevMode(),\n },\n loader: ParlTranslocoLoader,\n });\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i1.UtilsService"],"mappings":"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"}
1
+ {"version":3,"file":"trixwell-ngx-parl.mjs","sources":["../../../projects/ngx-parl/src/lib/core/entity/chat.ts","../../../projects/ngx-parl/src/lib/core/components/chat-message/chat-message.ts","../../../projects/ngx-parl/src/lib/core/components/chat-message/chat-message.html","../../../projects/ngx-parl/src/lib/core/service/utils/utils.ts","../../../projects/ngx-parl/src/lib/core/pipes/toggle-display-chat-start-day-pipe.ts","../../../projects/ngx-parl/src/lib/core/pipes/chat-start-day-pipe.ts","../../../projects/ngx-parl/src/lib/chat-flow/chat-flow.ts","../../../projects/ngx-parl/src/lib/chat-flow/chat-flow.html","../../../projects/ngx-parl/src/lib/core/entity/file.ts","../../../projects/ngx-parl/src/lib/input-message/input-message.ts","../../../projects/ngx-parl/src/lib/input-message/input-message.html","../../../projects/ngx-parl/src/lib/core/entity/theme.ts","../../../projects/ngx-parl/src/lib/ngx-parl/ngx-parl.ts","../../../projects/ngx-parl/src/lib/ngx-parl/ngx-parl.html","../../../projects/ngx-parl/src/assets/i18n/parl-transloco.loader.ts","../../../projects/ngx-parl/src/lib/ngx-parl.providers.ts","../../../projects/ngx-parl/src/trixwell-ngx-parl.ts"],"sourcesContent":["export class ChatMessage {\n public id: number;\n public chat_id: number;\n public cr_time: string;\n public type: ChatMessageType;\n public user: string;\n public content: string;\n public avatar: string | null;\n public file_path: string[] | null;\n public file_list: File[] | null;\n\n public checked: boolean | null;\n public edit = false;\n\n constructor(data: ChatMessageDTO) {\n this.id = data.id;\n this.chat_id = data.chat_id;\n this.cr_time = data.cr_time;\n this.type = data.type;\n this.user = data.user;\n this.content = data.content;\n this.avatar = data.avatar ?? null;\n this.checked = data.checked ?? null;\n this.file_path = data.file_path ?? null;\n this.file_list = data.file_list ?? null;\n }\n\n get dateSimple(): string {\n const d = new Date(this.cr_time.replace(' ', 'T'));\n const dd = String(d.getDate()).padStart(2, '0');\n const mm = String(d.getMonth() + 1).padStart(2, '0');\n const yyyy = d.getFullYear();\n return `${dd}.${mm}.${yyyy}`;\n }\n\n get timeHHmm(): string {\n const d = new Date(this.cr_time.replace(' ', 'T'));\n const hh = String(d.getHours()).padStart(2, '0');\n const mm = String(d.getMinutes()).padStart(2, '0');\n return `${hh}:${mm}`;\n }\n}\n\nexport interface ChatMessageDTO {\n id: number;\n chat_id: number;\n cr_time: string; // ISO or 'YYYY-MM-DD HH:mm:ss'\n type: ChatMessageType;\n user: string;\n content: string;\n avatar?: string | null;\n file_path?: string[] | [] | null;\n file_list?: File[] | [] | null;\n checked?: boolean | null;\n}\n\nexport type ChatMessageType = 'incoming' | 'outgoing';\n\nexport enum MessageType {\n Incoming = 'incoming',\n Outgoing = 'outgoing'\n}\n\nexport interface CurrMessage {\n id?: number;\n content: string;\n files?: string[];\n file_list?: File[];\n}\n\nexport type MessageActionType = 'send' | 'edit' | 'delete';\n\nexport interface MessageActionEvent {\n action: MessageActionType;\n chatMessageId?: number;\n content: string;\n files?: string[];\n file_list?: File[];\n}\n","import {Component, computed, input, model} from '@angular/core';\nimport {DatePipe, NgClass, NgOptimizedImage} from '@angular/common';\nimport {ChatMessage, MessageType} from '../../entity/chat';\nimport {MatMenu, MatMenuItem, MatMenuTrigger} from '@angular/material/menu';\nimport {TranslocoPipe} from '@ngneat/transloco';\n\n@Component({\n selector: 'lib-chat-message',\n imports: [\n NgClass,\n NgOptimizedImage,\n DatePipe,\n MatMenu,\n MatMenuItem,\n MatMenuTrigger,\n TranslocoPipe,\n ],\n templateUrl: './chat-message.html',\n styleUrl: './chat-message.scss',\n standalone: true,\n})\n\nexport class ChatMessageComponent {\n public currentMessage = input.required<ChatMessage>();\n public edit = model<boolean>(false);\n\n public requestEdit = model<ChatMessage | null>(null);\n public requestDelete = model<number | null>(null);\n\n constructor() {}\n\n private normalizeSourcePath(sourcePath: string): string {\n const cleanedPath = (sourcePath ?? '').trim();\n if (!cleanedPath) {\n return '';\n }\n\n if (cleanedPath.startsWith('data:') || cleanedPath.startsWith('blob:') || /^https?:\\/\\//i.test(cleanedPath)) {\n return cleanedPath;\n }\n\n const assetsIndex = cleanedPath.indexOf('assets/');\n if (assetsIndex >= 0) {\n return '/' + cleanedPath.slice(assetsIndex);\n }\n\n return cleanedPath.replace(/^\\.{1,2}\\//, '/');\n }\n\n attachments = computed(() => {\n const message = this.currentMessage();\n const filePath = message.file_path;\n\n if (Array.isArray(filePath)) {\n return filePath.map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n\n const rawFilePath = (filePath as unknown as string) ?? '';\n if (typeof rawFilePath !== 'string' || !rawFilePath.trim) {\n return [];\n }\n\n if (rawFilePath.trim().startsWith('[')) {\n try {\n const parsed = JSON.parse(rawFilePath);\n if (Array.isArray(parsed)) {\n return parsed\n .map(item => (typeof item === 'string' ? this.normalizeSourcePath(item) : ''))\n .filter(Boolean);\n }\n } catch {}\n }\n\n if (rawFilePath.startsWith('data:')) {\n return [rawFilePath];\n }\n\n if (rawFilePath.includes('|')) {\n return rawFilePath.split('|').map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n if (rawFilePath.includes(',')) {\n return rawFilePath.split(',').map(p => this.normalizeSourcePath(p)).filter(Boolean);\n }\n\n return [];\n });\n\n avatarSrc = computed(() => {\n const message = this.currentMessage();\n const fallback = message.type === 'incoming'\n ? 'assets/ngx-parl/icons/avatar_anonym.svg'\n : 'assets/ngx-parl/icons/avatar_manager.svg';\n\n return message.avatar || fallback;\n });\n\n openContextMenu(event: Event, trigger: any) {\n event.preventDefault();\n event.stopPropagation();\n trigger.openMenu();\n\n return this;\n }\n\n editMessage(message: ChatMessage) {\n this.edit.set(true);\n this.requestEdit.set(message);\n\n return this;\n }\n\n deleteMessage(message: ChatMessage) {\n this.requestDelete.set(message.id);\n queueMicrotask(() => this.requestDelete.set(null));\n\n return this;\n }\n\n canDelete(message: ChatMessage): boolean {\n return message.type === this.messageType.Outgoing;\n }\n\n public readonly messageType = MessageType;\n}\n","<div class=\"message\"\n [ngClass]=\"{\n 'message--outgoing': currentMessage().type === messageType.Outgoing,\n 'message--incoming': currentMessage().type === messageType.Incoming\n }\">\n <div class=\"message__avatar\">\n <img [ngSrc]=\"avatarSrc()\" width=\"36\" height=\"36\" alt=\"avatar\"/>\n </div>\n\n <div class=\"message__body\"\n #menuTrigger=\"matMenuTrigger\"\n [matMenuTriggerFor]=\"messageMenu\"\n (contextmenu)=\"openContextMenu($event, menuTrigger)\"\n (keydown.shift.F10)=\"openContextMenu($event, menuTrigger)\">\n\n @if (attachments().length) {\n <div class=\"message__attachments\">\n @for (file of attachments(); track file) {\n <img [src]=\"file\"\n alt=\"attachment\"\n class=\"message__image\"\n loading=\"lazy\"/>\n }\n </div>\n }\n\n <div class=\"message__bubble\" tabindex=\"0\">\n <div class=\"message__text\">{{ currentMessage().content }}</div>\n\n <div class=\"message__meta\">\n <time class=\"message__time\">{{ currentMessage().cr_time | date:'HH:mm' }}</time>\n @if (currentMessage().type === messageType.Outgoing) {\n @if (currentMessage().checked) {\n <img ngSrc=\"assets/ngx-parl/icons/checked-message.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n } @else {\n <img ngSrc=\"assets/ngx-parl/icons/no-check.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n }\n }\n </div>\n </div>\n </div>\n</div>\n\n<mat-menu #messageMenu=\"matMenu\" xPosition=\"before\" yPosition=\"below\" class=\"message__menu\">\n @if (currentMessage().type === messageType.Outgoing) {\n <button mat-menu-item (click)=\"editMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.edit' | transloco }}</span>\n </button>\n }\n @if (canDelete(currentMessage())) {\n <button mat-menu-item (click)=\"deleteMessage(currentMessage())\">\n <img ngSrc=\"assets/ngx-parl/icons/trash.svg\"\n class=\"message__icon--menu\"\n width=\"20\" height=\"20\" alt=\"hide\" priority/>\n <span>{{ 'chat.remove' | transloco }}</span>\n </button>\n }\n</mat-menu>\n","import {Injectable} from '@angular/core';\nimport {HttpClient} from '@angular/common/http';\n\n\n@Injectable({\n providedIn: 'root'\n})\nexport class UtilsService {\n\n constructor(protected http: HttpClient) {\n }\n\n langToLocale(lang: string): string {\n switch (lang) {\n case 'uk':\n return 'uk-UA';\n case 'en':\n default:\n return 'en-US';\n }\n }\n\n getLocalISODate(): string {\n const d = new Date();\n\n const year = d.getFullYear();\n const month = String(d.getMonth() + 1).padStart(2, '0');\n const day = String(d.getDate()).padStart(2, '0');\n\n const hours = String(d.getHours()).padStart(2, '0');\n const minutes = String(d.getMinutes()).padStart(2, '0');\n const seconds = String(d.getSeconds()).padStart(2, '0');\n\n return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}`;\n }\n}\n","import {Pipe, PipeTransform} from '@angular/core';\nimport {DatePipe} from '@angular/common';\nimport {ChatMessage} from '../entity/chat';\nimport {UtilsService} from '../service/utils/utils';\nimport {TranslocoService} from '@ngneat/transloco';\n\n@Pipe({\n name: 'toggleDisplayChatStartDay'\n})\nexport class ToggleDisplayChatStartDayPipe implements PipeTransform {\n\n constructor(protected utils: UtilsService, private transloco: TranslocoService) {}\n\n transform(message: ChatMessage, messages: ChatMessage[], i: number): boolean {\n const locale = this.utils.langToLocale(this.transloco.getActiveLang());\n const datePipe = new DatePipe(locale);\n\n const prev = i > 0 ? messages[i - 1] : undefined;\n\n const currDay = datePipe.transform(new Date(message.cr_time), 'shortDate');\n const prevDay = prev ? datePipe.transform(new Date(prev.cr_time), 'shortDate') : undefined;\n\n return prev ? currDay !== prevDay : true;\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\nimport {DatePipe} from '@angular/common';\nimport {UtilsService} from '../service/utils/utils';\nimport {TranslocoService} from '@ngneat/transloco';\n\n@Pipe({\n name: 'chatStartDay'\n})\nexport class ChatStartDayPipe implements PipeTransform {\n constructor(protected utils: UtilsService, private transloco: TranslocoService) {}\n\n transform(value: string, format: string = 'd MMMM'): string {\n if (!value) {\n return '';\n }\n\n const locale = this.utils.langToLocale(this.transloco.getActiveLang());\n const datePipe = new DatePipe(locale);\n\n const valueDate = new Date(value);\n const today = new Date();\n\n const isToday = datePipe.transform(valueDate, 'shortDate') === datePipe.transform(today, 'shortDate');\n\n return isToday\n ? (locale.startsWith('uk') ? 'Сьогодні' : 'Today')\n : (datePipe.transform(valueDate, format) ?? '');\n }\n\n}\n","import {AfterViewInit, Component, computed, effect, ElementRef, model, ViewChild,} from '@angular/core';\nimport {FormsModule} from '@angular/forms';\nimport {ChatMessage} from '../core/entity/chat';\nimport {ChatMessageComponent} from '../core/components/chat-message/chat-message';\nimport {TranslocoPipe} from '@ngneat/transloco';\nimport {NgOptimizedImage} from '@angular/common';\nimport {InfiniteScrollDirective} from 'ngx-infinite-scroll';\nimport {ToggleDisplayChatStartDayPipe} from '../core/pipes/toggle-display-chat-start-day-pipe';\nimport {ChatStartDayPipe} from '../core/pipes/chat-start-day-pipe';\n\n@Component({\n selector: 'app-chat-flow',\n imports: [\n FormsModule,\n ChatMessageComponent,\n ChatMessageComponent,\n TranslocoPipe,\n NgOptimizedImage,\n InfiniteScrollDirective,\n ToggleDisplayChatStartDayPipe,\n ChatStartDayPipe\n ],\n templateUrl: './chat-flow.html',\n styleUrl: './chat-flow.scss',\n standalone: true,\n})\n\nexport class ChatFlowComponent implements AfterViewInit {\n @ViewChild('chatFlowRef') flowRef?: ElementRef<HTMLElement>;\n\n public scrollToBottomTrigger = model<number>(0);\n public loadHistory = model<boolean>(false);\n\n public messageListInput = model.required<ChatMessage[]>();\n public messageList = computed(() => this.messageListInput());\n\n public selectedForEdit = model.required<ChatMessage | null>();\n public requestDelete = model<number | null>(null);\n\n private viewInitialized = false;\n private previousScrollHeight = 0;\n private previousMessageCount = 0;\n private isUserAtBottom = true;\n\n constructor() {\n\n /** ✅ Реакція на зміну списку */\n effect(() => {\n const messages = this.messageList();\n\n if (!this.viewInitialized) {\n this.previousMessageCount = messages.length;\n return;\n }\n\n const addedMessages = messages.length > this.previousMessageCount;\n\n if (addedMessages && !this.isUserAtBottom) {\n this.restoreScrollAfterHistoryPrepend();\n }\n\n if (addedMessages && this.isUserAtBottom) {\n queueMicrotask(() => this.scrollToBottom());\n }\n\n this.previousMessageCount = messages.length;\n });\n effect(() => {\n this.scrollToBottomTrigger();\n\n if (!this.viewInitialized) {\n return;\n }\n\n queueMicrotask(() => this.scrollToBottom());\n });\n }\n\n ngAfterViewInit() {\n this.viewInitialized = true;\n\n const element = this.flowRef?.nativeElement;\n if (!element) {\n return;\n }\n\n this.isUserAtBottom = true;\n\n element.addEventListener('scroll', () => {\n this.previousScrollHeight = element.scrollHeight;\n\n this.isUserAtBottom =\n element.scrollTop + element.clientHeight >=\n element.scrollHeight - 10;\n });\n\n queueMicrotask(() => this.scrollToBottom());\n }\n\n onScrollUp() {\n this.loadHistory.set(true);\n\n return this;\n }\n\n restoreScrollAfterHistoryPrepend() {\n const element = this.flowRef?.nativeElement;\n if (!element) {\n return this;\n }\n\n queueMicrotask(() => {\n const newHeight = element.scrollHeight;\n element.scrollTop = newHeight - this.previousScrollHeight;\n });\n\n return this;\n }\n\n scrollToBottomSmooth(): this {\n const element = this.flowRef?.nativeElement;\n\n if (!element) {\n return this;\n }\n\n element.scrollTo({\n top: element.scrollHeight,\n behavior: 'smooth',\n });\n\n return this;\n }\n\n scrollToBottom() {\n const element = this.flowRef?.nativeElement;\n if (!element) {\n return this;\n }\n\n element.scrollTop = element.scrollHeight;\n return this;\n }\n\n startEdit(message: ChatMessage): this {\n this.messageList().forEach(currentMessage => {\n if (currentMessage.id !== message.id && currentMessage.edit) {\n currentMessage.edit = false;\n }\n });\n\n message.edit = true;\n\n if (this.selectedForEdit()?.id === message.id) {\n this.selectedForEdit.set(null);\n queueMicrotask(() => this.selectedForEdit.set(message));\n } else {\n this.selectedForEdit.set(message);\n }\n\n return this;\n }\n\n onEditChange(id: number, isEdit: boolean): this {\n const message = this.messageList().find(item => item.id === id);\n\n if (!message) {\n return this;\n }\n\n if (isEdit) {\n return this.startEdit(message);\n }\n\n message.edit = false;\n\n if (this.selectedForEdit()?.id === id) {\n this.selectedForEdit.set(null);\n }\n\n return this;\n }\n\n onRequestEdit(message: ChatMessage | null): this {\n if (message) {\n return this.startEdit(message);\n }\n\n this.selectedForEdit.set(null);\n\n return this;\n }\n\n onRequestDelete(messageId: number | null): this {\n if (messageId == null) {\n return this;\n }\n\n this.selectedForEdit.set(null);\n\n this.requestDelete.set(messageId);\n queueMicrotask(() => this.requestDelete.set(null));\n\n return this;\n }\n\n\n trackByMessageId(_index: number, message: ChatMessage): string {\n // return message.id;\n return `${message.chat_id}-${message.type}-${message.id}`;\n }\n}\n","<div class=\"chat\">\n @if (messageList()) {\n <div #chatFlowRef\n class=\"chat__flow\"\n infiniteScroll\n [scrollWindow]=\"false\"\n [infiniteScrollUpDistance]=\"1\"\n [infiniteScrollThrottle]=\"150\"\n (scrolledUp)=\"onScrollUp()\"\n >\n @for (message of messageList(); track trackByMessageId(i, message); let i = $index) {\n @if (message | toggleDisplayChatStartDay: messageList() : i) {\n <span class=\"chat__start-day\">\n {{ message.cr_time | chatStartDay:'d MMMM' }}\n </span>\n }\n\n <div class=\"chat__message\">\n <lib-chat-message\n [currentMessage]=\"message\"\n [edit]=\"message.edit ?? false\"\n (editChange)=\"onEditChange(message.id, $event)\"\n (requestEditChange)=\"onRequestEdit($event)\"\n (requestDeleteChange)=\"onRequestDelete($event)\">\n </lib-chat-message>\n </div>\n }\n </div>\n } @else {\n <div class=\"chat__flow__empty\">\n <img ngSrc=\"assets/ngx-parl/icons/lucide_send.svg\"\n class=\"message__icon\"\n width=\"18\" height=\"24\" alt=\"hide\" priority/>\n\n\n <div class=\"chat__flow__empty--header\">\n <h3 class=\"chat__flow__empty--title\">{{ 'chat.chat_empty_title' | transloco }}</h3>\n <p class=\"chat__flow__empty--subscription\">{{ 'chat.chat_empty_subscription' | transloco }}</p>\n </div>\n </div>\n }\n</div>\n\n","export class ImageFile {\n constructor(\n public id: string,\n public url: string,\n public cr_time: string,\n ) {}\n}\n\nexport type OriginalKind = 'image' | 'gif';\n\nexport interface PreviewItem {\n originalKind: OriginalKind;\n duration?: number;\n src: string;\n name: string;\n type: string;\n size: number;\n}\n\nexport enum FileType {\n IMAGE = 'image',\n GIF = 'gif'\n}\n","import {\n AfterViewInit,\n Component,\n computed,\n effect,\n ElementRef,\n input,\n model,\n OnDestroy,\n signal,\n ViewChild\n} from '@angular/core';\nimport {FileType, OriginalKind, PreviewItem} from '../core/entity/file';\nimport {TranslocoPipe} from '@ngneat/transloco';\nimport {ChatMessage, CurrMessage, MessageActionEvent, MessageActionType} from '../core/entity/chat';\nimport {NgOptimizedImage} from '@angular/common';\n\n@Component({\n selector: 'app-input-message',\n imports: [TranslocoPipe, NgOptimizedImage],\n templateUrl: './input-message.html',\n styleUrl: './input-message.scss',\n standalone: true,\n})\n\nexport class InputMessageComponent implements AfterViewInit, OnDestroy {\n @ViewChild('inputText', {static: false}) inputTextElement!: ElementRef<HTMLDivElement>;\n @ViewChild('mirror', {static: false}) mirrorElement!: ElementRef<HTMLDivElement>;\n\n public editMessage = input<ChatMessage | { id: number; content: string; file_path?: string[] | null } | null>(null);\n\n public hasOriginalAttachments = computed(() => {\n const filePaths = this.editFilePaths();\n return filePaths.length > 0;\n });\n\n public hasNewAttachments = computed(() => (this.previews()?.length ?? 0) > 0);\n\n public cancelEdit = model<number | null>(null);\n public input_text = model<string | CurrMessage>('');\n\n public draft = signal<string>('');\n public focused = signal<boolean>(false);\n public sending = signal<boolean>(false);\n public hasText = computed(() => this.draft().trim().length > 0);\n\n public isEditMode = computed(() => !!this.editMessage());\n public canSend = computed(() =>\n this.hasText() ||\n this.hasNewAttachments() ||\n (this.isEditMode() && this.hasOriginalAttachments())\n );\n\n public files = model<File[]>([]);\n public previews = model<PreviewItem[]>([]);\n\n private lastHeightPx = 0;\n private lastRows = 1;\n private resizeRaf: number | null = null;\n\n public messageEvent = model<MessageActionEvent | null>(null);\n\n constructor() {\n effect(() => {\n const message = this.editMessage();\n const element = this.inputTextElement?.nativeElement;\n\n if (!element) {\n return;\n }\n\n if (message) {\n const content = (message as any).content ?? '';\n this.draft.set(content);\n element.innerText = content;\n\n queueMicrotask(() => {\n this.autoResizeByRows();\n element.focus();\n this.focused.set(true);\n });\n }\n });\n }\n\n ngAfterViewInit() {\n const element = this.inputTextElement.nativeElement;\n element.style.transition = 'height 160ms ease';\n this.initMirror();\n\n const computedStyle = getComputedStyle(element);\n const lineHeight = this.cssNum(computedStyle.lineHeight, 24);\n\n element.style.height = `${lineHeight}px`;\n this.lastHeightPx = lineHeight;\n this.lastRows = 1;\n this.updateOverflow(1);\n\n requestAnimationFrame(() => {\n const {rows, nextHeightPx} = this.measureByMirror();\n element.style.height = `${nextHeightPx}px`;\n\n this.lastRows = rows;\n this.lastHeightPx = nextHeightPx;\n this.updateOverflow(rows);\n });\n }\n\n ngOnDestroy() {\n if (this.resizeRaf) {\n cancelAnimationFrame(this.resizeRaf);\n this.resizeRaf = null;\n }\n }\n\n editFilePaths(): string[] {\n const message = this.editMessage();\n if (!message) {\n return [];\n }\n\n const file_path = (message as any).file_path;\n\n return Array.isArray(file_path) ? file_path : [];\n }\n\n collectAttachmentSources(): string[] {\n return (this.previews() ?? []).map(p => p.src).filter(Boolean);\n }\n\n cancelEditMessage() {\n const message = this.editMessage();\n this.cancelEdit.set((message as any)?.id ?? null);\n queueMicrotask(() => this.cancelEdit.set(null));\n\n this.draft.set('');\n const element = this.inputTextElement?.nativeElement;\n\n if (element) {\n element.innerHTML = '';\n this.autoResizeByRows();\n element.focus();\n }\n\n return this;\n }\n\n enterDown() {\n const element = this.inputTextElement.nativeElement;\n const text = this.draft().trim();\n\n if (!this.canSend()) {\n return this;\n }\n\n this.sending.set(true);\n\n const files = this.collectAttachmentSources();\n const fileList = this.files();\n\n const message = this.editMessage();\n const action: MessageActionType = message ? 'edit' : 'send';\n\n this.messageEvent.set({\n action,\n chatMessageId: message ? message.id : undefined,\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n\n queueMicrotask(() => this.messageEvent.set(null));\n\n if (message) {\n this.input_text.set({\n id: message.id,\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n } else {\n this.input_text.set({\n content: text,\n files: files.length ? files : [],\n file_list: fileList.length ? fileList : [],\n });\n }\n\n this.draft.set('');\n element.innerHTML = '';\n this.files.set([]);\n this.previews.set([]);\n element.focus();\n this.autoResizeByRows();\n\n setTimeout(() => this.sending.set(false), 150);\n\n return this;\n }\n\n onFocus() {\n if (this.inputTextElement.nativeElement.innerHTML === '<br>') {\n this.inputTextElement.nativeElement.innerHTML = '';\n }\n this.focused.set(true);\n\n return this;\n }\n\n onBlur() {\n this.focused.set(false);\n\n return this;\n }\n\n onKeyDown(event: KeyboardEvent) {\n if (event.key === 'Enter' && !event.shiftKey) {\n event.preventDefault();\n this.enterDown();\n\n return this;\n }\n queueMicrotask(() => this.autoResizeByRows());\n\n return this;\n }\n\n onInput() {\n this.draft.set(this.inputTextElement.nativeElement.innerText ?? '');\n this.autoResizeByRows();\n\n return this;\n }\n\n onPaste() {\n queueMicrotask(() => {\n this.draft.set(this.inputTextElement.nativeElement.innerText ?? '');\n this.autoResizeByRows();\n });\n\n return this;\n }\n\n inputFileChange(event: Event) {\n const inputEl = event.target as HTMLInputElement;\n const selected = inputEl.files;\n\n if (!selected?.length) {\n inputEl.value = '';\n\n return this;\n }\n\n const list = Array.from(selected).filter(f => (f.type || '').startsWith('image/'));\n if (!list.length) {\n inputEl.value = '';\n\n return this;\n }\n\n this.files.set([...(this.files() ?? []), ...list]);\n\n Promise.all(\n list.map(async f => {\n const src = await this.readFileAsDataURL(f);\n const originalKind: OriginalKind = (f.type || '') === 'image/gif' ? FileType.GIF : FileType.IMAGE;\n return <PreviewItem>{src, originalKind, name: f.name, type: f.type || '', size: f.size};\n })\n )\n .then(items => this.previews.set([...(this.previews() ?? []), ...items]))\n .finally(() => (inputEl.value = ''));\n\n return this;\n }\n\n removeFile(index: number, event?: MouseEvent) {\n event?.stopPropagation();\n event?.preventDefault();\n\n const previews = [...(this.previews() ?? [])];\n previews.splice(index, 1);\n this.previews.set(previews);\n\n const filesArr = [...(this.files() ?? [])];\n if (index >= 0 && index < filesArr.length) {\n filesArr.splice(index, 1);\n this.files.set(filesArr);\n }\n return this;\n }\n\n openPreview(_item: PreviewItem, _index: number) {\n return this;\n }\n\n autoResizeByRows() {\n const element = this.inputTextElement.nativeElement;\n const {rows, nextHeightPx} = this.measureByMirror();\n\n if (rows === this.lastRows) {\n this.updateOverflow(rows);\n\n return this;\n }\n\n if (this.resizeRaf) {\n cancelAnimationFrame(this.resizeRaf);\n }\n\n element.style.height = `${this.lastHeightPx}px`;\n\n this.resizeRaf = requestAnimationFrame(() => {\n element.style.height = `${nextHeightPx}px`;\n this.lastHeightPx = nextHeightPx;\n this.lastRows = rows;\n this.updateOverflow(rows);\n });\n\n return this;\n }\n\n measureByMirror(): { rows: number; nextHeightPx: number } {\n const inputEl = this.inputTextElement.nativeElement;\n const mirrorEl = this.mirrorElement.nativeElement;\n const computedStyle = getComputedStyle(inputEl);\n\n let text = inputEl.innerText;\n if (!text || text === '\\n') {\n text = '\\u00A0';\n }\n\n mirrorEl.style.width = computedStyle.width;\n mirrorEl.textContent = text;\n\n const lineHeight = this.cssNum(computedStyle.lineHeight, 24);\n const paddingTop = this.cssNum(computedStyle.paddingTop, 0);\n const paddingBottom = this.cssNum(computedStyle.paddingBottom, 0);\n const paddingY = paddingTop + paddingBottom;\n\n const maxRowsCss = computedStyle.getPropertyValue('--max-rows').trim();\n const maxRows = maxRowsCss ? this.cssNum(maxRowsCss, 8) : 8;\n\n const contentH = mirrorEl.offsetHeight;\n const rawRows = Math.max(1, Math.ceil(contentH / lineHeight));\n const rows = Math.min(rawRows, maxRows);\n\n const nextHeightPx = Math.round(rows * lineHeight + paddingY);\n\n return {rows, nextHeightPx};\n }\n\n initMirror() {\n const mirror = this.mirrorElement.nativeElement;\n const input = this.inputTextElement.nativeElement;\n const computedStyle = getComputedStyle(input);\n\n mirror.style.position = 'absolute';\n mirror.style.visibility = 'hidden';\n mirror.style.pointerEvents = 'none';\n mirror.style.zIndex = '-1';\n mirror.style.whiteSpace = 'pre-wrap';\n mirror.style.overflowWrap = 'break-word';\n mirror.style.wordBreak = 'normal';\n\n const properties = [\n 'font', 'font-size', 'font-family', 'font-weight', 'font-style',\n 'line-height', 'letter-spacing', 'word-spacing',\n 'padding-top', 'padding-bottom', 'padding-left', 'padding-right',\n 'border-top-width', 'border-bottom-width', 'border-left-width', 'border-right-width',\n 'white-space', 'text-transform', 'box-sizing'\n ];\n\n properties.forEach(property => (mirror.style as any)[property] = computedStyle.getPropertyValue(property));\n mirror.style.paddingTop = '0px';\n mirror.style.paddingBottom = '0px';\n }\n\n updateOverflow(rows: number) {\n const element = this.inputTextElement.nativeElement;\n const computedStyle = getComputedStyle(element);\n const maxRowsCss = computedStyle.getPropertyValue('--max-rows').trim();\n const maxRows = maxRowsCss ? this.cssNum(maxRowsCss, 8) : 8;\n element.style.overflowY = rows >= maxRows ? 'auto' : 'hidden';\n\n return this;\n }\n\n readFileAsDataURL(file: File): Promise<string> {\n return new Promise((resolve, reject) => {\n const reader = new FileReader();\n reader.onload = e => resolve((e.target?.result as string) || '');\n reader.onerror = reject;\n reader.readAsDataURL(file);\n });\n }\n\n cssNum(v: string, fb = 0): number {\n const n = parseFloat(v);\n return Number.isFinite(n) ? n : fb;\n }\n\n protected readonly FileType = FileType;\n}\n","<section class=\"message\"\n [class.message--focus]=\"focused()\"\n [class.message--filled]=\"hasText()\"\n [class.message--sending]=\"sending()\">\n\n @if (isEditMode()) {\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button class=\"message__button message__button--attach\">\n <img ngSrc=\"assets/ngx-parl/icons/icon-edit.svg\"\n class=\"message__icon message__icon--edit\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__edit-hint\">\n <p>{{ editMessage()?.content }}</p>\n </div>\n </div>\n\n <button class=\"message__button message__button--close\"\n type=\"button\"\n (click)=\"cancelEditMessage()\">\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\"\n class=\"message__icon message__icon--close\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n </div>\n }\n\n <div class=\"message__wrap\">\n <div class=\"message__wrap--input\">\n <button type=\"button\"\n class=\"message__button message__button--attach\"\n [attr.aria-label]=\"'chat.attach' | transloco\"\n (click)=\"fileDialog.click()\">\n <img ngSrc=\"assets/ngx-parl/icons/attach-filled.svg\"\n class=\"message__icon message__icon--attach\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <input #fileDialog\n type=\"file\"\n class=\"hidden\"\n accept=\"image/*\"\n multiple\n (change)=\"inputFileChange($event)\">\n\n <div #inputText\n role=\"textbox\"\n class=\"message__input\"\n spellcheck=\"true\"\n contenteditable=\"true\"\n aria-multiline=\"true\"\n [attr.data-placeholder]=\"isEditMode()\n ? ('chat.edit_placeholder' | transloco)\n : ('chat.placeholder' | transloco)\"\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (input)=\"onInput()\"\n (paste)=\"onPaste()\"\n (keydown)=\"onKeyDown($event)\">\n </div>\n\n <div #mirror class=\"message__input-mirror\" aria-hidden=\"true\"></div>\n\n <ng-content></ng-content>\n </div>\n\n <button class=\"message__button message__button--send\"\n type=\"button\"\n (click)=\"enterDown()\"\n [attr.aria-label]=\"'chat.send' | transloco\"\n [disabled]=\"!canSend()\">\n <img ngSrc=\"assets/ngx-parl/icons/send.svg\"\n class=\"message__icon message__icon--send\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n\n </button>\n </div>\n\n @if (previews().length) {\n <div class=\"message__files\">\n @for (item of previews(); track $index) {\n <div class=\"message__file\" [attr.data-kind]=\"item.originalKind\">\n <button type=\"button\"\n class=\"message__file-remove\"\n [attr.aria-label]=\"'chat.delete_file' | transloco\"\n (click)=\"removeFile($index, $event)\">\n <img ngSrc=\"assets/ngx-parl/icons/remove.svg\"\n class=\"message__icon message__icon--file\"\n width=\"18\" height=\"18\" alt=\"hide\" priority/>\n </button>\n\n <div class=\"message__file-media\" (click)=\"openPreview(item, $index)\">\n <img [src]=\"item.src\" [alt]=\"item.name\" class=\"message__file-img\"/>\n\n @if (item.originalKind === FileType.GIF) {\n <span class=\"message__file-badge message__file-badge--gif\">\n {{ 'chat.gif' | transloco }}\n </span>\n }\n </div>\n </div>\n }\n </div>\n }\n</section>\n\n\n","export enum FlowTheme {\n PRIMARY = 'primary',\n SECONDARY = 'secondary',\n}\n","import {Component, computed, effect, input, model} from '@angular/core';\nimport {NgClass, NgOptimizedImage} from '@angular/common';\nimport {ChatFlowComponent} from '../chat-flow/chat-flow';\nimport {MatDialogContent, MatDialogTitle} from '@angular/material/dialog';\nimport {\n ChatMessage,\n ChatMessageDTO,\n ChatMessageType,\n CurrMessage,\n MessageActionEvent,\n MessageType\n} from '../core/entity/chat';\nimport {MatProgressSpinner} from '@angular/material/progress-spinner';\nimport {InputMessageComponent} from '../input-message/input-message';\nimport {\n TranslocoModule,\n TranslocoPipe,\n TranslocoService,\n} from '@ngneat/transloco';\nimport {UtilsService} from '../core/service/utils/utils';\nimport {FlowTheme} from '../core/entity/theme';\n\n@Component({\n selector: 'ngx-parl',\n standalone: true,\n imports: [\n NgOptimizedImage, NgClass, MatDialogContent, MatDialogTitle, MatProgressSpinner, ChatFlowComponent, InputMessageComponent,\n TranslocoModule,\n TranslocoPipe\n ],\n templateUrl: './ngx-parl.html',\n styleUrl: './ngx-parl.scss',\n providers: [],\n})\nexport class NgxParlComponent {\n public ai_run_in_progress = false;\n private lastUpdateKey: string | null = null;\n\n public theme = input<FlowTheme>(FlowTheme.PRIMARY);\n public header = input<boolean>(true);\n public language = input<'en' | 'uk'>('en');\n\n public messageList = model<ChatMessage[]>([]);\n public messageUpdate = model<ChatMessage>();\n public selectedForEdit = model<ChatMessage | null>(null);\n public messageAction = model<MessageActionEvent | null>(null);\n\n public incomingUser = computed(() => {\n return (\n this.messageList().find((message) => message.type === MessageType.Incoming)?.user ?? ''\n );\n });\n\n public hideHandler = input<(() => unknown) | null>(null);\n public closeHandler = input<(() => unknown) | null>(null);\n\n public scrollToBottomTrigger = model<number>(0);\n public loadHistory = model<boolean>(false);\n\n constructor(private utils: UtilsService,\n private transloco: TranslocoService) {\n effect(() => {\n this.transloco.setActiveLang(this.language());\n });\n\n effect(() => {\n const updatedMessage = this.messageUpdate();\n\n if (!updatedMessage) {\n return;\n }\n\n const key = `${updatedMessage.id}-${updatedMessage.cr_time}-${updatedMessage.type}`;\n if (this.lastUpdateKey === key) {\n return;\n }\n\n this.lastUpdateKey = key;\n\n this.messageList.update(list => {\n const index = list.findIndex(m => m.id === updatedMessage.id);\n\n if (index > -1) {\n const updated = [...list];\n updated[index] = updatedMessage;\n return updated;\n }\n\n return [...list, updatedMessage];\n });\n\n this.scrollToBottomTrigger.update(v => v + 1);\n });\n }\n\n onCancelEdit(messageId: number | null) {\n if (messageId != null) {\n this.messageList.update(list => {\n const updated = [...list];\n const index = updated.findIndex(m => m.id === messageId);\n if (index > -1) {\n updated[index].edit = false;\n }\n return updated;\n });\n }\n\n this.selectedForEdit.set(null);\n return this;\n }\n\n onRequestDelete(messageId: number | null) {\n if (messageId == null) {\n return this;\n }\n\n this.messageList.update(list => list.filter(m => m.id !== messageId));\n\n this.pushMessageAction({\n action: 'delete',\n chatMessageId: messageId,\n content: '',\n });\n\n return this;\n }\n\n sendMessage(\n event:\n | CurrMessage\n | string\n | undefined\n ) {\n if (!event) {\n return this;\n }\n\n // edit message\n if (typeof event !== 'string' && 'id' in event) {\n const {id, content, files, file_list} = event;\n\n this.messageList.update((currentList) => {\n const updatedList = [...currentList];\n const index = updatedList.findIndex(\n (message) => message.id === id,\n );\n\n if (index > -1) {\n updatedList[index].content = (content ?? '').trim();\n\n if (Array.isArray(files)) {\n updatedList[index].file_path = files.length ? files : null;\n }\n\n updatedList[index].edit = false;\n }\n\n return updatedList;\n });\n\n this.selectedForEdit.set(null);\n\n this.pushMessageAction({\n action: 'edit',\n chatMessageId: id,\n content: (content ?? '').trim(),\n files: Array.isArray(files) && files.length ? files : [],\n file_list: Array.isArray(file_list) && file_list.length ? file_list : [],\n });\n this.scrollToBottomTrigger.update(v => v + 1);\n return this;\n }\n\n // new message\n if (typeof event === 'string') {\n const text = event.trim();\n\n if (!text) {\n return this;\n }\n\n const messages = this.messageList();\n const lastId = messages.at(-1)?.id ?? 0;\n\n const lastOutgoing = [...messages]\n .reverse()\n .find((message) => message.type === MessageType.Outgoing,);\n\n const dto: ChatMessageDTO = {\n id: lastId + 1,\n chat_id: lastOutgoing?.chat_id ?? 1,\n cr_time: this.utils.getLocalISODate(),\n type: MessageType.Outgoing as ChatMessageType,\n user: lastOutgoing?.user ?? '',\n content: text,\n avatar: lastOutgoing?.avatar ?? null,\n file_path: [],\n file_list: [],\n checked: true,\n };\n\n this.messageList.update((list) => [...list, new ChatMessage(dto)]);\n this.scrollToBottomTrigger.update(v => v + 1);\n this.pushMessageAction({\n action: 'send',\n chatMessageId: dto.id,\n content: text,\n });\n\n return this;\n }\n\n // new message + files\n const {content, files, file_list} = event;\n const text = (content ?? '').trim();\n const hasFiles = Array.isArray(files) && files.length > 0;\n const hasFileList = Array.isArray(file_list) && file_list.length > 0;\n\n if (!text && !hasFiles) {\n return this;\n }\n\n const messages = this.messageList();\n const lastId = messages.at(-1)?.id ?? 0;\n\n const lastOutgoing = [...messages]\n .reverse()\n .find((message) => message.type === MessageType.Outgoing,);\n\n const dto: ChatMessageDTO = {\n id: lastId + 1,\n chat_id: lastOutgoing?.chat_id ?? 1,\n cr_time: this.utils.getLocalISODate(),\n type: MessageType.Outgoing as ChatMessageType,\n user: lastOutgoing?.user ?? '',\n content: text,\n avatar: lastOutgoing?.avatar ?? null,\n file_path: hasFiles ? files : [],\n file_list: hasFileList ? file_list : [],\n checked: true,\n };\n\n this.messageList.update((list) => [...list, new ChatMessage(dto)]);\n this.scrollToBottomTrigger.update(v => v + 1);\n this.pushMessageAction({\n action: 'send',\n chatMessageId: dto.id,\n content: text,\n files: hasFiles ? files : [],\n file_list: hasFileList ? file_list : [],\n });\n\n return this;\n }\n\n pushMessageAction(event: MessageActionEvent) {\n this.messageAction.set(event);\n setTimeout(() => this.messageAction.set(null), 0);\n return this;\n }\n\n onHideClick() {\n const handler = this.hideHandler();\n\n if (handler) {\n handler();\n }\n return this;\n }\n\n onCloseClick() {\n const handler = this.closeHandler();\n\n if (handler) {\n handler();\n }\n\n return this;\n }\n\n protected readonly FlowTheme = FlowTheme;\n}\n","<div class=\"modal-chat\" [ngClass]=\"'flow-theme-' + theme()\">\n <div class=\"modal-chat__body\">\n @if (header()) {\n <header class=\"modal-chat__header\" mat-dialog-title>\n <div class=\"modal-chat__hide-block\"></div>\n\n <div class=\"modal-chat__title\">\n <h1 class=\"modal-chat__heading\" id=\"modal-chat__heading\">\n {{ 'chat.title' | transloco }} {{ incomingUser() }}\n </h1>\n </div>\n\n <div class=\"modal-chat__actions\">\n <img ngSrc=\"assets/ngx-parl/icons/hide.svg\" class=\"modal-chat__icon modal-chat__icon--hide\"\n width=\"24\" height=\"24\" alt=\"hide\" priority\n (click)=\"onHideClick()\"/>\n <img ngSrc=\"assets/ngx-parl/icons/close.svg\" class=\"modal-chat__icon modal-chat__icon--close\"\n width=\"24\" height=\"24\" alt=\"close\" priority\n (click)=\"onCloseClick()\"/>\n </div>\n </header>\n }\n\n <mat-dialog-content class=\"modal-chat__content\" [class.modal-chat__content--header]=\"header() === false\">\n <app-chat-flow [messageListInput]=\"messageList()\"\n [(selectedForEdit)]=\"selectedForEdit\"\n (requestDeleteChange)=\"onRequestDelete($event)\"\n [(scrollToBottomTrigger)]=\"scrollToBottomTrigger\"\n [(loadHistory)]=\"loadHistory\">\n </app-chat-flow>\n\n <app-input-message [editMessage]=\"selectedForEdit()\"\n (cancelEditChange)=\"onCancelEdit($event)\"\n (input_textChange)=\"sendMessage($event)\">\n @if (ai_run_in_progress) {\n <mat-spinner [diameter]=\"30\"></mat-spinner>\n }\n </app-input-message>\n </mat-dialog-content>\n </div>\n</div>\n","import { Injectable } from '@angular/core';\nimport { TranslocoLoader, Translation } from '@ngneat/transloco';\nimport { Observable, of } from 'rxjs';\n\nimport en from '../../assets/i18n/en.json';\nimport uk from '../../assets/i18n/uk.json';\n\n@Injectable()\nexport class ParlTranslocoLoader implements TranslocoLoader {\n getTranslation(lang: string): Observable<Translation> {\n switch (lang) {\n case 'uk':\n return of(uk as Translation);\n case 'en':\n default:\n return of(en as Translation);\n }\n }\n}\n","import { isDevMode } from '@angular/core';\nimport { provideTransloco } from '@ngneat/transloco';\nimport { ParlTranslocoLoader } from '../assets/i18n/parl-transloco.loader';\n\nexport function provideNgxParl() {\n return provideTransloco({\n config: {\n availableLangs: ['en', 'uk'],\n defaultLang: 'en',\n reRenderOnLangChange: true,\n prodMode: !isDevMode(),\n },\n loader: ParlTranslocoLoader,\n });\n}\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i1.UtilsService"],"mappings":"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"}
package/index.d.ts CHANGED
@@ -76,6 +76,7 @@ declare class NgxParlComponent {
76
76
  hideHandler: _angular_core.InputSignal<(() => unknown) | null>;
77
77
  closeHandler: _angular_core.InputSignal<(() => unknown) | null>;
78
78
  scrollToBottomTrigger: _angular_core.ModelSignal<number>;
79
+ loadHistory: _angular_core.ModelSignal<boolean>;
79
80
  constructor(utils: UtilsService, transloco: TranslocoService);
80
81
  onCancelEdit(messageId: number | null): this;
81
82
  onRequestDelete(messageId: number | null): this;
@@ -83,10 +84,9 @@ declare class NgxParlComponent {
83
84
  pushMessageAction(event: MessageActionEvent): this;
84
85
  onHideClick(): this;
85
86
  onCloseClick(): this;
86
- scrollToBottom(): this;
87
87
  protected readonly FlowTheme: typeof FlowTheme;
88
88
  static ɵfac: _angular_core.ɵɵFactoryDeclaration<NgxParlComponent, never>;
89
- static ɵcmp: _angular_core.ɵɵComponentDeclaration<NgxParlComponent, "ngx-parl", never, { "theme": { "alias": "theme"; "required": false; "isSignal": true; }; "header": { "alias": "header"; "required": false; "isSignal": true; }; "language": { "alias": "language"; "required": false; "isSignal": true; }; "messageList": { "alias": "messageList"; "required": false; "isSignal": true; }; "messageUpdate": { "alias": "messageUpdate"; "required": false; "isSignal": true; }; "selectedForEdit": { "alias": "selectedForEdit"; "required": false; "isSignal": true; }; "messageAction": { "alias": "messageAction"; "required": false; "isSignal": true; }; "hideHandler": { "alias": "hideHandler"; "required": false; "isSignal": true; }; "closeHandler": { "alias": "closeHandler"; "required": false; "isSignal": true; }; "scrollToBottomTrigger": { "alias": "scrollToBottomTrigger"; "required": false; "isSignal": true; }; }, { "messageList": "messageListChange"; "messageUpdate": "messageUpdateChange"; "selectedForEdit": "selectedForEditChange"; "messageAction": "messageActionChange"; "scrollToBottomTrigger": "scrollToBottomTriggerChange"; }, never, never, true, never>;
89
+ static ɵcmp: _angular_core.ɵɵComponentDeclaration<NgxParlComponent, "ngx-parl", never, { "theme": { "alias": "theme"; "required": false; "isSignal": true; }; "header": { "alias": "header"; "required": false; "isSignal": true; }; "language": { "alias": "language"; "required": false; "isSignal": true; }; "messageList": { "alias": "messageList"; "required": false; "isSignal": true; }; "messageUpdate": { "alias": "messageUpdate"; "required": false; "isSignal": true; }; "selectedForEdit": { "alias": "selectedForEdit"; "required": false; "isSignal": true; }; "messageAction": { "alias": "messageAction"; "required": false; "isSignal": true; }; "hideHandler": { "alias": "hideHandler"; "required": false; "isSignal": true; }; "closeHandler": { "alias": "closeHandler"; "required": false; "isSignal": true; }; "scrollToBottomTrigger": { "alias": "scrollToBottomTrigger"; "required": false; "isSignal": true; }; "loadHistory": { "alias": "loadHistory"; "required": false; "isSignal": true; }; }, { "messageList": "messageListChange"; "messageUpdate": "messageUpdateChange"; "selectedForEdit": "selectedForEditChange"; "messageAction": "messageActionChange"; "scrollToBottomTrigger": "scrollToBottomTriggerChange"; "loadHistory": "loadHistoryChange"; }, never, never, true, never>;
90
90
  }
91
91
 
92
92
  declare function provideNgxParl(): _angular_core.EnvironmentProviders[];
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@trixwell/ngx-parl",
3
- "version": "1.5.1",
3
+ "version": "1.5.3",
4
4
  "license": "LGPL-3.0",
5
5
  "description": "Highly customizable Angular Material chat component",
6
6
  "keywords": [