@sinequa/assistant 3.1.1 → 3.2.1
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/chat/chat-message/chat-message.component.d.ts +3 -0
- package/chat/chat-settings-v3/chat-settings-v3.component.d.ts +8 -3
- package/chat/chat.component.d.ts +174 -146
- package/chat/chat.service.d.ts +31 -917
- package/chat/prompt.component.d.ts +21 -0
- package/chat/public-api.d.ts +1 -0
- package/chat/rest-chat.service.d.ts +5 -2
- package/chat/saved-chats/saved-chats.component.d.ts +4 -3
- package/chat/styles/assistant.scss +21 -1
- package/chat/types.d.ts +284 -866
- package/chat/websocket-chat.service.d.ts +16 -12
- package/esm2020/chat/chat-message/chat-message.component.mjs +25 -3
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +3 -3
- package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +22 -24
- package/esm2020/chat/chat.component.mjs +123 -67
- package/esm2020/chat/chat.service.mjs +68 -24
- package/esm2020/chat/prompt.component.mjs +88 -0
- package/esm2020/chat/public-api.mjs +2 -1
- package/esm2020/chat/rest-chat.service.mjs +73 -26
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +37 -7
- package/esm2020/chat/types.mjs +50 -54
- package/esm2020/chat/websocket-chat.service.mjs +116 -56
- package/fesm2015/sinequa-assistant-chat.mjs +583 -251
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
- package/fesm2020/sinequa-assistant-chat.mjs +584 -254
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +4 -2
|
@@ -3,7 +3,7 @@ import { Action } from "@sinequa/components/action";
|
|
|
3
3
|
import { AbstractFacet } from "@sinequa/components/facet";
|
|
4
4
|
import { SearchService } from "@sinequa/components/search";
|
|
5
5
|
import { PrincipalWebService } from "@sinequa/core/web-services";
|
|
6
|
-
import { BehaviorSubject, Subscription, filter,
|
|
6
|
+
import { BehaviorSubject, Subscription, filter, fromEvent, map, merge, switchMap, tap } from "rxjs";
|
|
7
7
|
import { ChatService } from "./chat.service";
|
|
8
8
|
import { InstanceManagerService } from "./instance-manager.service";
|
|
9
9
|
import { WebSocketChatService } from "./websocket-chat.service";
|
|
@@ -18,14 +18,21 @@ import * as i2 from "@angular/forms";
|
|
|
18
18
|
export class ChatComponent extends AbstractFacet {
|
|
19
19
|
constructor() {
|
|
20
20
|
super();
|
|
21
|
+
this.loginService = inject(LoginService);
|
|
22
|
+
this.websocketService = inject(WebSocketChatService);
|
|
23
|
+
this.restService = inject(RestChatService);
|
|
24
|
+
this.instanceManagerService = inject(InstanceManagerService);
|
|
25
|
+
this.searchService = inject(SearchService);
|
|
26
|
+
this.principalService = inject(PrincipalWebService);
|
|
27
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
28
|
+
/** Define the query to use to fetch answers */
|
|
29
|
+
this.query = this.searchService.query;
|
|
21
30
|
/** Define the protocol to be used for this chat instance*/
|
|
22
31
|
this.protocol = "WEBSOCKET";
|
|
23
32
|
/** Map of listeners overriding default registered ones*/
|
|
24
33
|
this.messageHandlers = new Map();
|
|
25
34
|
/** When the assistant answer a user question, automatically scroll down to the bottom of the discussion */
|
|
26
35
|
this.automaticScrollToLastResponse = false;
|
|
27
|
-
this.enableChat = true;
|
|
28
|
-
this.showCredits = true;
|
|
29
36
|
this.customAssistantIcon = '';
|
|
30
37
|
this.data = new EventEmitter();
|
|
31
38
|
this.referenceClicked = new EventEmitter();
|
|
@@ -38,16 +45,10 @@ export class ChatComponent extends AbstractFacet {
|
|
|
38
45
|
this._actions = [];
|
|
39
46
|
this.sub = new Subscription();
|
|
40
47
|
this.changes$ = new BehaviorSubject(undefined);
|
|
41
|
-
this.
|
|
48
|
+
this.firstChangesHandled = false;
|
|
42
49
|
this.isAtBottom = true;
|
|
43
50
|
this.initializationError = false;
|
|
44
|
-
this.
|
|
45
|
-
this.websocketService = inject(WebSocketChatService);
|
|
46
|
-
this.restService = inject(RestChatService);
|
|
47
|
-
this.instanceManagerService = inject(InstanceManagerService);
|
|
48
|
-
this.searchService = inject(SearchService);
|
|
49
|
-
this.principalService = inject(PrincipalWebService);
|
|
50
|
-
this.cdr = inject(ChangeDetectorRef);
|
|
51
|
+
this.enabledUserInput = false;
|
|
51
52
|
this._actions.push(new Action({
|
|
52
53
|
icon: 'fas fa-sync',
|
|
53
54
|
title: 'Reset chat',
|
|
@@ -57,13 +58,14 @@ export class ChatComponent extends AbstractFacet {
|
|
|
57
58
|
ngOnInit() {
|
|
58
59
|
this.sub.add(this.loginService.events.pipe(filter(e => e.type === 'login-complete'), tap(_ => this.instantiateChatService()), map(_ => this.chatService.initChatConfig()), switchMap(() => this.chatService.initConfig$), filter(initConfig => !!initConfig), switchMap(_ => this.chatService.init()), filter(success => !!success), tap(_ => this.onLoadChat()), switchMap(_ => this.chatService.chatConfig$), tap(config => {
|
|
59
60
|
this.config = config;
|
|
61
|
+
this.enabledUserInput = this.config.modeSettings.enabledUserInput;
|
|
60
62
|
this._config.emit(config);
|
|
61
63
|
try {
|
|
62
64
|
this.updateModelDescription();
|
|
63
|
-
if (!this.
|
|
65
|
+
if (!this.firstChangesHandled) {
|
|
64
66
|
this.handleChanges();
|
|
65
67
|
this.addScrollListener();
|
|
66
|
-
this.
|
|
68
|
+
this.firstChangesHandled = true;
|
|
67
69
|
}
|
|
68
70
|
}
|
|
69
71
|
catch (error) {
|
|
@@ -99,12 +101,50 @@ export class ChatComponent extends AbstractFacet {
|
|
|
99
101
|
get actions() { return this._actions; }
|
|
100
102
|
handleChanges() {
|
|
101
103
|
const changes = this.changes$.value;
|
|
104
|
+
// If the chat service is a WebSocketChatService, handle the override of the message handlers if exists
|
|
102
105
|
if (changes?.messageHandlers && this.messageHandlers && this.chatService instanceof WebSocketChatService) {
|
|
103
106
|
this.chatService.overrideMessageHandlers(this.messageHandlers);
|
|
104
107
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
+
/**
|
|
109
|
+
* Initialize the chat with the provided chat messages if exists, otherwise load the default chat
|
|
110
|
+
* Once the chat is initialized (firstChangesHandled is true), allow opening the chat with the new provided messages (if exists)
|
|
111
|
+
*/
|
|
112
|
+
if (!this.firstChangesHandled || changes?.chat) {
|
|
113
|
+
const openChat = () => {
|
|
114
|
+
if (this.messages$.value && this.config.savedChatSettings.enabled) {
|
|
115
|
+
this.chatService.listSavedChat(); // Refresh the list of saved chats
|
|
116
|
+
}
|
|
117
|
+
this.openChat(this.chat.messages);
|
|
118
|
+
};
|
|
119
|
+
this.chat ? openChat() : this.loadDefaultChat();
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* If the chat is initialized, the initialization event is "Query", the query changes and the queryChangeShouldTriggerReload function is provided,
|
|
123
|
+
* then the chat should be reloaded if the function returns true
|
|
124
|
+
* Otherwise, the chat should be reloaded by default
|
|
125
|
+
*/
|
|
126
|
+
if (this.firstChangesHandled && changes?.query && this.config.modeSettings.initialization.event === 'Query') {
|
|
127
|
+
const previousQuery = changes.query.previousValue;
|
|
128
|
+
const currentQuery = changes.query.currentValue;
|
|
129
|
+
if (this.queryChangeShouldTriggerReload ? this.queryChangeShouldTriggerReload(previousQuery, currentQuery) : true) {
|
|
130
|
+
this.triggerReloadAfterQueryChange(currentQuery);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
triggerReloadAfterQueryChange(query) {
|
|
135
|
+
const systemMsg = { role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: { display: false } };
|
|
136
|
+
const userMsg = { role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, { principal: this.principalService.principal }), additionalProperties: { display: this.config.modeSettings.displayUserPrompt } };
|
|
137
|
+
/**
|
|
138
|
+
* If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant
|
|
139
|
+
* Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results
|
|
140
|
+
*/
|
|
141
|
+
if (!!this.query.text) {
|
|
142
|
+
const userQueryMsg = { role: 'user', content: this.query.text, additionalProperties: { display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true } };
|
|
143
|
+
this.openChat(this.config.modeSettings.sendUserPrompt ? [systemMsg, userMsg, userQueryMsg] : [systemMsg, userQueryMsg]);
|
|
144
|
+
}
|
|
145
|
+
else {
|
|
146
|
+
const warningMsg = { role: 'assistant', content: "You must perform a full text search to retrieve some results", additionalProperties: { display: true } };
|
|
147
|
+
this.openChat([warningMsg]);
|
|
108
148
|
}
|
|
109
149
|
}
|
|
110
150
|
addScrollListener() {
|
|
@@ -114,7 +154,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
114
154
|
}));
|
|
115
155
|
}
|
|
116
156
|
updateModelDescription() {
|
|
117
|
-
this.modelDescription = this.chatService.getModel(this.config.
|
|
157
|
+
this.modelDescription = this.chatService.getModel(this.config.defaultValues.service_id, this.config.defaultValues.model_id);
|
|
118
158
|
this.assistantIcon = !!this.customAssistantIcon ? this.customAssistantIcon : 'sq-sinequa';
|
|
119
159
|
switch (this.modelDescription?.provider) {
|
|
120
160
|
case 'Google':
|
|
@@ -141,17 +181,15 @@ export class ChatComponent extends AbstractFacet {
|
|
|
141
181
|
this.chatService.chatHistory = this.chatService.chatHistory.slice(0, this.messageToEdit);
|
|
142
182
|
this.messageToEdit = undefined;
|
|
143
183
|
}
|
|
144
|
-
// Re-attach the $progress and $attachment of the last response to the last assistant's response in the chat history
|
|
145
|
-
this.chatService.chatHistory.at(-1).additionalProperties.$progress = this.messages$.value.at(-1).additionalProperties.$progress;
|
|
146
|
-
this.chatService.chatHistory.at(-1).additionalProperties.$attachment = this.messages$.value.at(-1).additionalProperties.$attachment;
|
|
147
184
|
// Fetch the answer
|
|
148
185
|
this.fetchAnswer(this.question.trim(), this.chatService.chatHistory);
|
|
149
186
|
// Clear the input value in the UI
|
|
150
187
|
this.questionInput.nativeElement.value = '';
|
|
188
|
+
this.questionInput.nativeElement.style.height = `auto`;
|
|
151
189
|
}
|
|
152
190
|
}
|
|
153
191
|
fetchAnswer(question, conversation) {
|
|
154
|
-
const userMsg = { role: 'user', content: question, additionalProperties: { display: true } };
|
|
192
|
+
const userMsg = { role: 'user', content: question, additionalProperties: { display: true, isUserInput: true } };
|
|
155
193
|
const messages = [...conversation, userMsg];
|
|
156
194
|
this.messages$.next(messages);
|
|
157
195
|
this.fetch(messages);
|
|
@@ -208,22 +246,56 @@ export class ChatComponent extends AbstractFacet {
|
|
|
208
246
|
}
|
|
209
247
|
}, 10);
|
|
210
248
|
}
|
|
249
|
+
/**
|
|
250
|
+
* Start a new chat with the defaultValues settings
|
|
251
|
+
* The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat
|
|
252
|
+
* If the savedChat feature is enabled, the list of saved chats will be refreshed
|
|
253
|
+
*/
|
|
211
254
|
newChat() {
|
|
212
|
-
this.chatService.
|
|
255
|
+
this.chatService.setSavedChatId(undefined); // Reset the savedChatId
|
|
256
|
+
if (this.config.savedChatSettings.enabled) {
|
|
257
|
+
this.chatService.listSavedChat(); // Refresh the list of saved chats
|
|
258
|
+
}
|
|
213
259
|
this.loadDefaultChat(); // Start a new chat
|
|
214
260
|
}
|
|
261
|
+
/**
|
|
262
|
+
* Start the default chat with the defaultValues settings
|
|
263
|
+
* If the chat is meant to be initialized with event === "Query", the corresponding user query message will be added to the chat history
|
|
264
|
+
*/
|
|
215
265
|
loadDefaultChat() {
|
|
216
|
-
this.
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
266
|
+
const systemMsg = { role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: { display: false } };
|
|
267
|
+
const userMsg = { role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, { principal: this.principalService.principal }), additionalProperties: { display: this.config.modeSettings.displayUserPrompt } };
|
|
268
|
+
if (this.config.modeSettings.initialization.event === 'Query') {
|
|
269
|
+
// If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant
|
|
270
|
+
// Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results
|
|
271
|
+
if (!!this.query.text) {
|
|
272
|
+
const userQueryMsg = { role: 'user', content: this.query.text, additionalProperties: { display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true } };
|
|
273
|
+
this.openChat(this.config.modeSettings.sendUserPrompt ? [systemMsg, userMsg, userQueryMsg] : [systemMsg, userQueryMsg]);
|
|
274
|
+
}
|
|
275
|
+
else {
|
|
276
|
+
const warningMsg = { role: 'assistant', content: "You must perform a full text search to retrieve some results", additionalProperties: { display: true } };
|
|
277
|
+
this.openChat([warningMsg]);
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
else {
|
|
281
|
+
this.openChat([systemMsg, userMsg]);
|
|
282
|
+
}
|
|
220
283
|
}
|
|
284
|
+
/**
|
|
285
|
+
* Start/open a new chat with the provided messages and chatId
|
|
286
|
+
* If the last message is from the user, a request to the assistant is made to get an answer
|
|
287
|
+
* If the last message is from the assistant, the conversation is loaded right away
|
|
288
|
+
* @param messages The list of messages of the chat
|
|
289
|
+
* @param chatId The id of the chat. If provided (ie. an existing discussion in the saved chat index), update the savedChatId in the chat service for the upcoming saved chat operations
|
|
290
|
+
*/
|
|
221
291
|
openChat(messages, chatId) {
|
|
222
292
|
if (!messages || !Array.isArray(messages)) {
|
|
223
293
|
console.error('Error occurs while trying to load the chat discussion. Invalid messages received :', messages);
|
|
224
294
|
return;
|
|
225
295
|
}
|
|
226
|
-
|
|
296
|
+
if (chatId) {
|
|
297
|
+
this.chatService.setSavedChatId(chatId);
|
|
298
|
+
}
|
|
227
299
|
this.resetChat();
|
|
228
300
|
this.messages$.next(messages);
|
|
229
301
|
this.chatService.chatHistory = messages;
|
|
@@ -236,6 +308,11 @@ export class ChatComponent extends AbstractFacet {
|
|
|
236
308
|
this.terminateFetch();
|
|
237
309
|
}
|
|
238
310
|
}
|
|
311
|
+
/**
|
|
312
|
+
* Reset the chat by clearing the messages and the chat history
|
|
313
|
+
* The question input will be focused after the chat is reset
|
|
314
|
+
* The fetch subscription will be terminated
|
|
315
|
+
*/
|
|
239
316
|
resetChat() {
|
|
240
317
|
if (this.messages$.value) {
|
|
241
318
|
this.messages$.next(undefined); // Reset chat
|
|
@@ -247,8 +324,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
247
324
|
onLoadChat() {
|
|
248
325
|
this.loading$.next(true);
|
|
249
326
|
this.sub.add(this.chatService.loadSavedChat$
|
|
250
|
-
.pipe(filter(savedChat => !!savedChat), switchMap(savedChat => this.chatService.getSavedChat(savedChat.id)), filter(savedChatHistory => !!savedChatHistory), tap(savedChatHistory => this.openChat(savedChatHistory.
|
|
251
|
-
).subscribe());
|
|
327
|
+
.pipe(filter(savedChat => !!savedChat), switchMap(savedChat => this.chatService.getSavedChat(savedChat.id)), filter(savedChatHistory => !!savedChatHistory), tap(savedChatHistory => this.openChat(savedChatHistory.history, savedChatHistory.id))).subscribe());
|
|
252
328
|
}
|
|
253
329
|
terminateFetch() {
|
|
254
330
|
this.dataSubscription?.unsubscribe();
|
|
@@ -274,60 +350,44 @@ export class ChatComponent extends AbstractFacet {
|
|
|
274
350
|
}
|
|
275
351
|
onKeyUp(event) {
|
|
276
352
|
switch (event.key) {
|
|
277
|
-
case '
|
|
278
|
-
this.
|
|
279
|
-
break;
|
|
280
|
-
case 'ArrowDown':
|
|
281
|
-
this.navigateMessage(1);
|
|
353
|
+
case 'Backspace':
|
|
354
|
+
this.calculateHeight();
|
|
282
355
|
break;
|
|
283
356
|
case 'Enter':
|
|
284
|
-
|
|
357
|
+
if (!event.shiftKey)
|
|
358
|
+
this.submitQuestion();
|
|
359
|
+
this.calculateHeight();
|
|
285
360
|
break;
|
|
286
361
|
default:
|
|
287
362
|
break;
|
|
288
363
|
}
|
|
289
|
-
// Handle Shift + Enter
|
|
290
|
-
if (event.shiftKey && event.key === 'Enter') {
|
|
291
|
-
this.submitQuestion();
|
|
292
|
-
}
|
|
293
364
|
}
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
}
|
|
302
|
-
this.currentMessageIndex = (this.currentMessageIndex ?? userMessages.length) + direction;
|
|
303
|
-
if (this.currentMessageIndex < 0) {
|
|
304
|
-
// If the user presses up arrow on the first message, stay at the first message
|
|
305
|
-
this.currentMessageIndex = 0;
|
|
306
|
-
}
|
|
307
|
-
else if (this.currentMessageIndex >= userMessages.length) {
|
|
308
|
-
// If the user presses down arrow on the last previous message, clear the input
|
|
309
|
-
this.currentMessageIndex = undefined;
|
|
310
|
-
this.question = '';
|
|
311
|
-
return;
|
|
312
|
-
}
|
|
313
|
-
this.question = userMessages[this.currentMessageIndex].content;
|
|
365
|
+
calculateHeight() {
|
|
366
|
+
const maxHeight = 170;
|
|
367
|
+
const el = this.questionInput.nativeElement;
|
|
368
|
+
el.style.maxHeight = `${maxHeight}px`;
|
|
369
|
+
el.style.height = 'auto';
|
|
370
|
+
el.style.height = `${el.scrollHeight}px`;
|
|
371
|
+
el.style.overflowY = el.scrollHeight >= maxHeight ? 'scroll' : 'hidden';
|
|
314
372
|
}
|
|
315
373
|
}
|
|
316
374
|
ChatComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
317
|
-
ChatComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatComponent, isStandalone: true, selector: "sq-chat-v3", inputs: { instanceId: "instanceId", query: "query", protocol: "protocol", messageHandlers: "messageHandlers", automaticScrollToLastResponse: "automaticScrollToLastResponse", chat: "chat",
|
|
375
|
+
ChatComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatComponent, isStandalone: true, selector: "sq-chat-v3", inputs: { instanceId: "instanceId", query: "query", queryChangeShouldTriggerReload: "queryChangeShouldTriggerReload", protocol: "protocol", messageHandlers: "messageHandlers", automaticScrollToLastResponse: "automaticScrollToLastResponse", chat: "chat", customAssistantIcon: "customAssistantIcon" }, outputs: { data: "data", referenceClicked: "referenceClicked", openPreview: "openPreview", loading$: "loading", error: "error", _config: "config" }, providers: [
|
|
318
376
|
RestChatService,
|
|
319
377
|
WebSocketChatService
|
|
320
|
-
], queries: [{ propertyName: "loadingTpl", first: true, predicate: ["loadingTpl"], descendants: true }], viewQueries: [{ propertyName: "messageList", first: true, predicate: ["messageList"], descendants: true }, { propertyName: "questionInput", first: true, predicate: ["questionInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "\n<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n <ul class=\"list-group list-group-flush overflow-auto pb-
|
|
378
|
+
], queries: [{ propertyName: "loadingTpl", first: true, predicate: ["loadingTpl"], descendants: true }], viewQueries: [{ propertyName: "messageList", first: true, predicate: ["messageList"], descendants: true }, { propertyName: "questionInput", first: true, predicate: ["questionInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "\n<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n <ul class=\"list-group list-group-flush overflow-auto pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\" *ngIf=\"message.additionalProperties.display\"\n [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role !== 'assistant'\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [assistantIcon]=\"customAssistantIcon || assistantIcon\"\n [streaming]=\"last && (chatService.streaming$ | async)\"\n [canEdit]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role !== 'assistant'\"\n [canRegenerate]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'assistant' && last\"\n [canCopy]=\"!(last && (chatService.streaming$ | async)) && messageToEdit === undefined && message.role === 'assistant'\"\n (edit)=\"editMessage(index)\" (regenerate)=\"regenerateMessage(index)\"\n (referenceClicked)=\"referenceClicked.emit($event)\"\n (openPreview)=\"openPreview.emit($event)\">\n </sq-chat-message>\n </li>\n </ng-container>\n\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <div class=\"user-input mt-auto\" *ngIf=\"enabledUserInput\">\n <div class=\"py-2\">\n <ng-container *ngTemplateOutlet=\"inputTpl\"></ng-container>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight()\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <!--<button\n *ngIf=\"(chatService.streaming$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Stop generating\"\n (click)=\"terminateFetch()\">\n <i class=\"fas fa-stop\"></i>\n </button>-->\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <div class=\"sq-floating-scroll\" *ngIf=\"!isAtBottom\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;bottom:75px;text-align:center}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:end;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: ChatMessageComponent, selector: "sq-chat-message", inputs: ["message", "conversation", "assistantIcon", "streaming", "canEdit", "canRegenerate", "canCopy"], outputs: ["referenceClicked", "edit", "regenerate", "openPreview"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
321
379
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatComponent, decorators: [{
|
|
322
380
|
type: Component,
|
|
323
381
|
args: [{ selector: 'sq-chat-v3', providers: [
|
|
324
382
|
RestChatService,
|
|
325
383
|
WebSocketChatService
|
|
326
|
-
], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, FormsModule, ChatMessageComponent], template: "\n<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n <ul class=\"list-group list-group-flush overflow-auto pb-
|
|
384
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, FormsModule, ChatMessageComponent], template: "\n<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n <ul class=\"list-group list-group-flush overflow-auto pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\" *ngIf=\"message.additionalProperties.display\"\n [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role !== 'assistant'\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [assistantIcon]=\"customAssistantIcon || assistantIcon\"\n [streaming]=\"last && (chatService.streaming$ | async)\"\n [canEdit]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role !== 'assistant'\"\n [canRegenerate]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'assistant' && last\"\n [canCopy]=\"!(last && (chatService.streaming$ | async)) && messageToEdit === undefined && message.role === 'assistant'\"\n (edit)=\"editMessage(index)\" (regenerate)=\"regenerateMessage(index)\"\n (referenceClicked)=\"referenceClicked.emit($event)\"\n (openPreview)=\"openPreview.emit($event)\">\n </sq-chat-message>\n </li>\n </ng-container>\n\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <div class=\"user-input mt-auto\" *ngIf=\"enabledUserInput\">\n <div class=\"py-2\">\n <ng-container *ngTemplateOutlet=\"inputTpl\"></ng-container>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight()\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <!--<button\n *ngIf=\"(chatService.streaming$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Stop generating\"\n (click)=\"terminateFetch()\">\n <i class=\"fas fa-stop\"></i>\n </button>-->\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n title=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <div class=\"sq-floating-scroll\" *ngIf=\"!isAtBottom\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;bottom:75px;text-align:center}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:end;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}\n"] }]
|
|
327
385
|
}], ctorParameters: function () { return []; }, propDecorators: { instanceId: [{
|
|
328
386
|
type: Input
|
|
329
387
|
}], query: [{
|
|
330
388
|
type: Input
|
|
389
|
+
}], queryChangeShouldTriggerReload: [{
|
|
390
|
+
type: Input
|
|
331
391
|
}], protocol: [{
|
|
332
392
|
type: Input
|
|
333
393
|
}], messageHandlers: [{
|
|
@@ -336,10 +396,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
336
396
|
type: Input
|
|
337
397
|
}], chat: [{
|
|
338
398
|
type: Input
|
|
339
|
-
}], enableChat: [{
|
|
340
|
-
type: Input
|
|
341
|
-
}], showCredits: [{
|
|
342
|
-
type: Input
|
|
343
399
|
}], customAssistantIcon: [{
|
|
344
400
|
type: Input
|
|
345
401
|
}], data: [{
|
|
@@ -366,4 +422,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
366
422
|
type: ContentChild,
|
|
367
423
|
args: ['loadingTpl']
|
|
368
424
|
}] } });
|
|
369
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.component.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.component.ts","../../../../projects/assistant/chat/chat.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,SAAS,EAAE,YAAY,EAAc,YAAY,EAAE,KAAK,EAAgC,MAAM,EAA8B,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1N,OAAO,EAAE,MAAM,EAAE,MAAM,4BAA4B,CAAC;AACpD,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAE3D,OAAO,EAAE,mBAAmB,EAAU,MAAM,4BAA4B,CAAC;AACzE,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAC9G,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;;;;AAkBtD,MAAM,OAAO,aAAc,SAAQ,aAAa;IA0D9C;QACE,KAAK,EAAE,CAAC;QAtDV,2DAA2D;QAClD,aAAQ,GAAyB,WAAW,CAAC;QACtD,yDAAyD;QAChD,oBAAe,GAAqC,IAAI,GAAG,EAAE,CAAC;QACvE,2GAA2G;QAClG,kCAA6B,GAAG,KAAK,CAAC;QAEtC,eAAU,GAAG,IAAI,CAAC;QAClB,gBAAW,GAAG,IAAI,CAAC;QACnB,wBAAmB,GAAG,EAAE,CAAC;QACxB,SAAI,GAAG,IAAI,YAAY,EAAiB,CAAC;QACzC,qBAAgB,GAAG,IAAI,YAAY,EAAU,CAAC;QAC9C,gBAAW,GAAG,IAAI,YAAY,EAAyB,CAAC;QAC/C,aAAQ,GAAG,IAAI,YAAY,CAAU,KAAK,CAAC,CAAC;QACrD,UAAK,GAAG,IAAI,YAAY,EAAO,CAAC;QACxB,YAAO,GAAG,IAAI,YAAY,EAAc,CAAC;QAS3D,cAAS,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAEtE,aAAQ,GAAG,EAAE,CAAC;QAEd,aAAQ,GAAa,EAAE,CAAC;QAExB,QAAG,GAAG,IAAI,YAAY,EAAE,CAAC;QASzB,aAAQ,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAErE,uBAAkB,GAAG,KAAK,CAAC;QAC3B,eAAU,GAAG,IAAI,CAAC;QAClB,wBAAmB,GAAG,KAAK,CAAC;QAErB,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,qBAAgB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAChD,gBAAW,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QACtC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,kBAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QACtC,qBAAgB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAC/C,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAKrC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC;YAC5B,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,YAAY;YACnB,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE;SACrC,CAAC,CAAC,CAAC;IACN,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAC3B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,gBAAgB,CAAC,EACxC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,EACvC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,EAAE,CAAC,EAC3C,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAC7C,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,EAClC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,EACvC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,EAC5B,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,EAC3B,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAC5C,GAAG,CAAC,MAAM,CAAC,EAAE;YACX,IAAI,CAAC,MAAM,GAAG,MAAO,CAAC;YACtB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC1B,IAAI;gBACF,IAAI,CAAC,sBAAsB,EAAE,CAAC;gBAC9B,IAAG,CAAC,IAAI,CAAC,kBAAkB,EAAE;oBAC3B,IAAI,CAAC,aAAa,EAAE,CAAC;oBACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;oBACzB,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;iBAChC;aACF;YAAC,OAAO,KAAK,EAAE;gBACd,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAA;gBAC/B,MAAM,KAAK,CAAC;aACb;QACH,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC5B,IAAI,IAAI,CAAC,MAAM,EAAE;YACf,IAAI,CAAC,aAAa,EAAE,CAAC;SACtB;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;QACvB,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;IACvC,CAAC;IAED,sBAAsB;QACpB,QAAQ,IAAI,CAAC,QAAQ,EAAE;YACrB,KAAK,MAAM;gBACT,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;gBACpC,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC;gBACzC,MAAM;YACR;gBACE,MAAM,IAAI,KAAK,CAAC,yFAAyF,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;SAC9H;QACD,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACpD,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAC/E,CAAC;IAED,IAAa,OAAO,KAAK,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAExC,aAAa;QACnB,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC;QACpC,IAAI,OAAO,EAAE,eAAe,IAAI,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,WAAW,YAAY,oBAAoB,EAAE;YACxG,IAAI,CAAC,WAAW,CAAC,uBAAuB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;SAChE;QACD,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,IAAI,OAAO,EAAE,IAAI,EAAE;YAC1C,gBAAgB;YAChB,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC;SACxE;IACH,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,SAAS,CAAC,IAAI,CAAC,WAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACrI,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,4BAA4B,EAAE,CAAC;YACtD,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QAC3B,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;QAChI,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,YAAY,CAAC;QAC1F,QAAO,IAAI,CAAC,gBAAgB,EAAE,QAAQ,EAAE;YACtC,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;gBACrB,MAAM;YACR,KAAK,aAAa;gBAChB,IAAI,CAAC,UAAU,GAAG,gFAAgF,CAAC;gBACnG,MAAM;YACR,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,uCAAuC,CAAC;gBAC1D,MAAM;YACR,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,6BAA6B,CAAC;gBAChD,MAAM;SACT;QACD,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3B,CAAC;IAED,cAAc;QACZ,IAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,IAAI,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;YAC/E,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS,EAAE;gBACpC,gCAAgC;gBAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;gBACvE,uGAAuG;gBACvG,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;gBACzF,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;aAChC;YACD,oHAAoH;YACpH,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,SAAS,CAAC;YAClI,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,WAAW,CAAC;YACtI,mBAAmB;YACnB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,IAAI,CAAC,WAAW,CAAC,WAAY,CAAC,CAAC;YACtE,kCAAkC;YAClC,IAAI,CAAC,aAAc,CAAC,aAAa,CAAC,KAAK,GAAG,EAAE,CAAC;SAC9C;IACH,CAAC;IAEO,WAAW,CAAC,QAAgB,EAAE,YAA2B;QAC/D,MAAM,OAAO,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,QAAQ,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAC,CAAC;QACzF,MAAM,QAAQ,GAAG,CAAC,GAAG,YAAY,EAAE,OAAO,CAAC,CAAC;QAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IACvB,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,QAAuB;QAClC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QACzB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzB,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;QACrC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC;aACjE,SAAS,CAAC;YACT,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC;YACzC,KAAK,EAAE,GAAG,CAAC,EAAE;gBACX,IAAI,CAAC,cAAc,EAAE,CAAC;gBACtB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACnB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACvB,CAAC;YACD,QAAQ,EAAE,GAAG,EAAE;gBACb,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEL,IAAG,IAAI,CAAC,6BAA6B,EAAE;YACrC,IAAI,CAAC,UAAU,EAAE,CAAC;SACnB;IACH,CAAC;IAED;;;OAGG;IACH,UAAU,CAAC,QAAuB;QAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACzB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAG,IAAI,CAAC,6BAA6B,EAAE;YACrC,IAAI,CAAC,UAAU,EAAE,CAAC;SACnB;IACH,CAAC;IAEO,4BAA4B;QAClC,IAAG,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE;YAClC,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,YAAY,GAAG,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,SAAS,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,YAAY,CAAC;SACjK;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,UAAU;QACR,UAAU,CAAC,GAAG,EAAE;YACd,IAAG,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE;gBAClC,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,YAAY,CAAC;gBACvF,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;aAC1B;QACH,CAAC,EAAE,EAAE,CAAC,CAAC;IACT,CAAC;IAED,OAAO;QACL,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC,CAAC,kCAAkC;QACpE,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,mBAAmB;IAC7C,CAAC;IAED,eAAe;QACb,IAAI,CAAC,QAAQ,CAAC;YACZ,EAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,YAAY,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,KAAK,EAAC,EAAC;YACtG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,EAAC,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAC,CAAC,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAC;SAC1K,CAAC,CAAC;IACL,CAAC;IAED,QAAQ,CAAC,QAAsB,EAAE,MAAe;QAC9C,IAAI,CAAC,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;YACzC,OAAO,CAAC,KAAK,CAAC,oFAAoF,EAAE,QAAQ,CAAC,CAAC;YAC9G,OAAO;SACR;QACD,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,MAAM,IAAI,WAAW,CAAC,YAAY,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,QAAQ,CAAC;QACxC,MAAM,WAAW,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACpC,IAAG,WAAW,IAAI,WAAW,CAAC,IAAI,KAAK,MAAM,EAAE;YAC7C,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,+EAA+E;SACtG;aACI;YACH,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,qFAAqF;YAChH,IAAI,CAAC,cAAc,EAAE,CAAC;SACvB;IACH,CAAC;IAED,SAAS;QACP,IAAG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE;YACvB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,aAAa;SAC9C;QACD,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,SAAS,CAAC,CAAC,qBAAqB;QAC/D,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAED,UAAU;QACR,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzB,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,IAAI,CAAC,WAAW,CAAC,cAAc;aAC5B,IAAI,CACH,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,EAChC,SAAS,CAAC,SAAS,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,SAAU,CAAC,EAAE,CAAC,CAAC,EACpE,MAAM,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC,CAAC,gBAAgB,CAAC,EAC9C,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAiB,CAAC,OAAO,EAAE,gBAAiB,CAAC,EAAE,CAAC,CAAC,EACvF,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC,CAAC,kCAAkC;SACpF,CAAC,SAAS,EAAE,CAChB,CAAC;IACJ,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;QACrC,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;QAClC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,aAAa,EAAE,aAAa,CAAC,KAAK,EAAE,CAAC;QAC5C,CAAC,CAAC,CAAA;QACF,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3B,CAAC;IAED,WAAW,CAAC,KAAa;QACvB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC;QAC7D,IAAI,CAAC,aAAa,EAAE,aAAa,CAAC,KAAK,EAAE,CAAC;IAC5C,CAAC;IAED,iBAAiB,CAAC,KAAa;QAC5B,kFAAkF;QACnF,MAAM,cAAc,GAAG,IAAI,CAAC,WAAW,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;QACrE,4CAA4C;QAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACpC,mBAAmB;QACnB,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;IAC7B,CAAC;IAED,OAAO,CAAC,KAAoB;QAC1B,QAAQ,KAAK,CAAC,GAAG,EAAE;YACjB,KAAK,SAAS;gBACZ,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC;gBACzB,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;gBACxB,MAAM;YACR,KAAK,OAAO;gBACV,IAAI,CAAC,cAAc,EAAE,CAAC;gBACtB,MAAM;YACR;gBACE,MAAM;SACT;QAED,uBAAuB;QACvB,IAAI,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC,GAAG,KAAK,OAAO,EAAE;YAC3C,IAAI,CAAC,cAAc,EAAE,CAAC;SACvB;IACH,CAAC;IAEO,eAAe,CAAC,SAAiB;QACvC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE;YAC5E,OAAO;SACR;QACD,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC;QAEjF,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YAC3B,OAAO;SACR;QAED,IAAI,CAAC,mBAAmB,GAAG,CAAC,IAAI,CAAC,mBAAmB,IAAI,YAAY,CAAC,MAAM,CAAC,GAAG,SAAS,CAAC;QAEzF,IAAI,IAAI,CAAC,mBAAmB,GAAG,CAAC,EAAE;YAChC,+EAA+E;YAC/E,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC;SAC9B;aAAM,IAAI,IAAI,CAAC,mBAAmB,IAAI,YAAY,CAAC,MAAM,EAAE;YAC1D,+EAA+E;YAC/E,IAAI,CAAC,mBAAmB,GAAG,SAAS,CAAC;YACrC,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;YACnB,OAAO;SACR;QAED,IAAI,CAAC,QAAQ,GAAG,YAAY,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,OAAO,CAAC;IACjE,CAAC;;0GAtXU,aAAa;8FAAb,aAAa,8eARb;QACT,eAAe;QACf,oBAAoB;KACrB,yXC5BH,i6HA2FA,2oFD5DY,YAAY,2dAAE,WAAW,+mBAAE,oBAAoB;2FAE9C,aAAa;kBAZzB,SAAS;+BACE,YAAY,aAGX;wBACT,eAAe;wBACf,oBAAoB;qBACrB,mBACgB,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,oBAAoB,CAAC;0EAIjD,UAAU;sBAAlB,KAAK;gBAEG,KAAK;sBAAb,KAAK;gBAEG,QAAQ;sBAAhB,KAAK;gBAEG,eAAe;sBAAvB,KAAK;gBAEG,6BAA6B;sBAArC,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,mBAAmB;sBAA3B,KAAK;gBACI,IAAI;sBAAb,MAAM;gBACG,gBAAgB;sBAAzB,MAAM;gBACG,WAAW;sBAApB,MAAM;gBACY,QAAQ;sBAA1B,MAAM;uBAAC,SAAS;gBACP,KAAK;sBAAd,MAAM;gBACW,OAAO;sBAAxB,MAAM;uBAAC,QAAQ;gBAEU,WAAW;sBAApC,SAAS;uBAAC,aAAa;gBACI,aAAa;sBAAxC,SAAS;uBAAC,eAAe;gBAEE,UAAU;sBAArC,YAAY;uBAAC,YAAY","sourcesContent":["import { inject, ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChanges, TemplateRef, ViewChild } from \"@angular/core\";\nimport { Action } from \"@sinequa/components/action\";\nimport { AbstractFacet } from \"@sinequa/components/facet\";\nimport { SearchService } from \"@sinequa/components/search\";\nimport { Query } from \"@sinequa/core/app-utils\";\nimport { PrincipalWebService, Record } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Subscription, filter, finalize, fromEvent, map, merge, switchMap, tap } from \"rxjs\";\nimport { ChatService } from \"./chat.service\";\nimport { ChatContextAttachment, ChatConfig, ChatMessage, GllmModelDescription, MessageHandler, RawMessage } from \"./types\";\nimport { InstanceManagerService } from \"./instance-manager.service\";\nimport { WebSocketChatService } from \"./websocket-chat.service\";\nimport { ChatMessageComponent } from \"./chat-message/chat-message.component\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { RestChatService } from \"./rest-chat.service\";\n\nexport interface InitChat {\n  messages: RawMessage[];\n}\n\n@Component({\n  selector: 'sq-chat-v3', // mandatory since @sinequa/components already has the same tag-name \"sq-chat\"\n  templateUrl: './chat.component.html',\n  styleUrls: ['./chat.component.scss'],\n  providers: [\n    RestChatService,\n    WebSocketChatService\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, FormsModule, ChatMessageComponent]\n})\nexport class ChatComponent extends AbstractFacet implements OnInit, OnChanges, OnDestroy {\n  /** Define the key based on it, the chat service instance will be stored */\n  @Input() instanceId: string;\n  /** Define the query to use to fetch answers */\n  @Input() query?: Query;\n  /** Define the protocol to be used for this chat instance*/\n  @Input() protocol: 'REST' | 'WEBSOCKET' = \"WEBSOCKET\";\n  /** Map of listeners overriding default registered ones*/\n  @Input() messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  /** When the assistant answer a user question, automatically scroll down to the bottom of the discussion */\n  @Input() automaticScrollToLastResponse = false;\n  @Input() chat?: InitChat;\n  @Input() enableChat = true;\n  @Input() showCredits = true;\n  @Input() customAssistantIcon = '';\n  @Output() data = new EventEmitter<ChatMessage[]>();\n  @Output() referenceClicked = new EventEmitter<Record>();\n  @Output() openPreview = new EventEmitter<ChatContextAttachment>();\n  @Output(\"loading\") loading$ = new EventEmitter<boolean>(false);\n  @Output() error = new EventEmitter<any>();\n  @Output(\"config\") _config = new EventEmitter<ChatConfig>();\n\n  @ViewChild('messageList') messageList?: ElementRef<HTMLUListElement>;\n  @ViewChild('questionInput') questionInput?: ElementRef<HTMLInputElement>;\n\n  @ContentChild('loadingTpl') loadingTpl?: TemplateRef<any>;\n\n  chatService: ChatService;\n  config: ChatConfig;\n  messages$ = new BehaviorSubject<ChatMessage[] | undefined>(undefined);\n\n  question = '';\n\n  _actions: Action[] = [];\n\n  sub = new Subscription();\n  dataSubscription: Subscription | undefined;\n\n  /** Variables that depend on the type of model in use */\n  modelDescription?: GllmModelDescription;\n  assistantIcon: string;\n  privacyUrl: string;\n\n  messageToEdit?: number;\n  changes$ = new BehaviorSubject<SimpleChanges | undefined>(undefined);\n  currentMessageIndex: number | undefined;\n  handleFirstChanges = false;\n  isAtBottom = true;\n  initializationError = false;\n\n  public loginService = inject(LoginService);\n  public websocketService = inject(WebSocketChatService);\n  public restService = inject(RestChatService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public searchService = inject(SearchService);\n  public principalService = inject(PrincipalWebService);\n  public cdr = inject(ChangeDetectorRef);\n\n  constructor() {\n    super();\n\n    this._actions.push(new Action({\n      icon: 'fas fa-sync',\n      title: 'Reset chat',\n      action: () => this.loadDefaultChat()\n    }));\n  }\n\n  ngOnInit(): void {\n    this.sub.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        map(_ => this.chatService.initChatConfig()),\n        switchMap(() => this.chatService.initConfig$),\n        filter(initConfig => !!initConfig),\n        switchMap(_ => this.chatService.init()),\n        filter(success => !!success),\n        tap(_ => this.onLoadChat()),\n        switchMap(_ => this.chatService.chatConfig$),\n        tap(config => {\n          this.config = config!;\n          this._config.emit(config);\n          try {\n            this.updateModelDescription();\n            if(!this.handleFirstChanges) {\n              this.handleChanges();\n              this.addScrollListener();\n              this.handleFirstChanges = true;\n            }\n          } catch (error) {\n            this.initializationError = true\n            throw error;\n          }\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    this.changes$.next(changes);\n    if (this.config) {\n      this.handleChanges();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.sub.unsubscribe();\n    this.dataSubscription?.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    switch (this.protocol) {\n      case 'REST':\n        this.chatService = this.restService;\n        break;\n      case 'WEBSOCKET':\n        this.chatService = this.websocketService;\n        break;\n      default:\n        throw new Error(`Could not found a ChatService implementation corresponding to the provided protocol: '${this.protocol}'`);\n    }\n    this.chatService.setChatInstanceId(this.instanceId);\n    this.instanceManagerService.storeInstance(this.instanceId, this.chatService);\n  }\n\n  override get actions() { return this._actions; }\n\n  private handleChanges() {\n    const changes = this.changes$.value;\n    if (changes?.messageHandlers && this.messageHandlers && this.chatService instanceof WebSocketChatService) {\n      this.chatService.overrideMessageHandlers(this.messageHandlers);\n    }\n    if (!this.messages$.value || changes?.chat) {\n      // Load the chat\n      this.chat ? this.openChat(this.chat.messages) : this.loadDefaultChat();\n    }\n  }\n\n  private addScrollListener() {\n    this.sub.add(\n      merge(this.loading$, this.messages$, this.chatService.streaming$, fromEvent(this.messageList!.nativeElement, 'scroll')).subscribe(() => {\n        this.isAtBottom = this.toggleScrollButtonVisibility();\n        this.cdr.detectChanges();\n      })\n    );\n  }\n\n  updateModelDescription() {\n    this.modelDescription = this.chatService.getModel(this.config.serviceSettings.service_id, this.config.serviceSettings.model_id);\n    this.assistantIcon = !!this.customAssistantIcon ? this.customAssistantIcon : 'sq-sinequa';\n    switch(this.modelDescription?.provider) {\n      case 'Google':\n        this.privacyUrl = '';\n        break;\n      case 'AzureOpenAI':\n        this.privacyUrl = 'https://learn.microsoft.com/en-us/legal/cognitive-services/openai/data-privacy';\n        break;\n      case 'OpenAI':\n        this.privacyUrl = 'https://openai.com/enterprise-privacy';\n        break;\n      case 'Cohere':\n        this.privacyUrl = 'https://cohere.com/security';\n        break;\n    }\n    this.cdr.detectChanges();\n  }\n\n  submitQuestion() {\n    if(this.question.trim() && this.messages$.value && this.chatService.chatHistory) {\n      if (this.messageToEdit !== undefined) {\n        // Update the messages in the UI\n        this.messages$.next(this.messages$.value.slice(0, this.messageToEdit));\n        // Update the raw messages in the chat history which is the clean version used to make the next request\n        this.chatService.chatHistory = this.chatService.chatHistory.slice(0, this.messageToEdit);\n        this.messageToEdit = undefined;\n      }\n      // Re-attach the $progress and $attachment of the last response to the last assistant's response in the chat history\n      this.chatService.chatHistory.at(-1)!.additionalProperties.$progress = this.messages$.value.at(-1)!.additionalProperties.$progress;\n      this.chatService.chatHistory.at(-1)!.additionalProperties.$attachment = this.messages$.value.at(-1)!.additionalProperties.$attachment;\n      // Fetch the answer\n      this.fetchAnswer(this.question.trim(), this.chatService.chatHistory!);\n      // Clear the input value in the UI\n      this.questionInput!.nativeElement.value = '';\n    }\n  }\n\n  private fetchAnswer(question: string, conversation: ChatMessage[]) {\n    const userMsg = {role: 'user', content: question, additionalProperties: {display: true}};\n    const messages = [...conversation, userMsg];\n    this.messages$.next(messages);\n    this.fetch(messages);\n  }\n\n  /**\n   * Given a list of messages, fetch the server for a continuation and updates\n   * the list of messages accordingly.\n   * @param messages\n   */\n  public fetch(messages: ChatMessage[]) {\n    this.cdr.detectChanges();\n    this.loading$.next(true);\n    this.dataSubscription?.unsubscribe();\n    this.dataSubscription = this.chatService.fetch(messages, this.query)\n      .subscribe({\n        next: res => this.updateData(res.history),\n        error: err => {\n          this.terminateFetch();\n          console.error(err);\n          this.error.emit(err);\n        },\n        complete: () => {\n          this.terminateFetch();\n        }\n      });\n\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  /**\n   * Update the UI with the new messages\n   * @param messages\n   */\n  updateData(messages: ChatMessage[]) {\n    this.messages$.next(messages);\n    this.data.emit(messages);\n    this.loading$.next(false);\n    this.question = '';\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  private toggleScrollButtonVisibility(): boolean {\n    if(this.messageList?.nativeElement) {\n      return Math.round(this.messageList?.nativeElement.scrollHeight - this.messageList?.nativeElement.scrollTop - 1) <= this.messageList?.nativeElement.clientHeight;\n    }\n    return true;\n  }\n\n  scrollDown() {\n    setTimeout(() => {\n      if(this.messageList?.nativeElement) {\n        this.messageList.nativeElement.scrollTop = this.messageList.nativeElement.scrollHeight;\n        this.cdr.detectChanges();\n      }\n    }, 10);\n  }\n\n  newChat() {\n    this.chatService.listSavedChat(); // Refresh the list of saved chats\n    this.loadDefaultChat(); // Start a new chat\n  }\n\n  loadDefaultChat() {\n    this.openChat([\n      {role: 'system', content: this.config.uiSettings.systemPrompt, additionalProperties: {display: false}},\n      {role: 'user', content: ChatService.formatPrompt(this.config.uiSettings.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: true}},\n    ]);\n  }\n\n  openChat(messages: RawMessage[], chatId?: string) {\n    if (!messages || !Array.isArray(messages)) {\n      console.error('Error occurs while trying to load the chat discussion. Invalid messages received :', messages);\n      return;\n    }\n    this.chatService.setSavedChatId(chatId || ChatService.generateGUID());\n    this.resetChat();\n    this.messages$.next(messages);\n    this.chatService.chatHistory = messages;\n    const lastMessage = messages.at(-1);\n    if(lastMessage && lastMessage.role === 'user') {\n      this.fetch(messages); // If the last message if from a user, an answer from the assistant is expected\n    }\n    else {\n      this.updateData(messages); // If the last message if from the assistant, we can load the conversation right away\n      this.terminateFetch();\n    }\n  }\n\n  resetChat() {\n    if(this.messages$.value) {\n      this.messages$.next(undefined); // Reset chat\n    }\n    this.chatService.chatHistory = undefined; // Reset chat history\n    this.question = '';\n    this.terminateFetch();\n  }\n\n  onLoadChat() {\n    this.loading$.next(true);\n    this.sub.add(\n      this.chatService.loadSavedChat$\n        .pipe(\n          filter(savedChat => !!savedChat),\n          switchMap(savedChat => this.chatService.getSavedChat(savedChat!.id)),\n          filter(savedChatHistory => !!savedChatHistory),\n          tap(savedChatHistory => this.openChat(savedChatHistory!.History, savedChatHistory!.id)),\n          finalize(() => this.chatService.listSavedChat()) // Refresh the list of saved chats\n        ).subscribe()\n    );\n  }\n\n  terminateFetch() {\n    this.dataSubscription?.unsubscribe();\n    this.dataSubscription = undefined;\n    this.loading$.next(false);\n    setTimeout(() => {\n      this.questionInput?.nativeElement.focus();\n    })\n    this.cdr.detectChanges();\n  }\n\n  editMessage(index: number) {\n    this.messageToEdit = index;\n    this.question = this.chatService.chatHistory![index].content;\n    this.questionInput?.nativeElement.focus();\n  }\n\n  regenerateMessage(index: number) {\n     // Define the chat history based on which the assistant will generate a new answer\n    const slicedMessages = this.chatService.chatHistory!.slice(0, index);\n    // Accordingly update the messages in the UI\n    this.messages$.next(slicedMessages);\n    // Fetch the answer\n    this.fetch(slicedMessages);\n  }\n\n  onKeyUp(event: KeyboardEvent): void {\n    switch (event.key) {\n      case 'ArrowUp':\n        this.navigateMessage(-1);\n        break;\n      case 'ArrowDown':\n        this.navigateMessage(1);\n        break;\n      case 'Enter':\n        this.submitQuestion();\n        break;\n      default:\n        break;\n    }\n\n    // Handle Shift + Enter\n    if (event.shiftKey && event.key === 'Enter') {\n      this.submitQuestion();\n    }\n  }\n\n  private navigateMessage(direction: number): void {\n    if (!this.chatService.chatHistory || this.chatService.chatHistory.length < 1) {\n      return;\n    }\n    const userMessages = this.chatService.chatHistory.filter(m => m.role === 'user');\n\n    if (userMessages.length < 1) {\n      return;\n    }\n\n    this.currentMessageIndex = (this.currentMessageIndex ?? userMessages.length) + direction;\n\n    if (this.currentMessageIndex < 0) {\n      // If the user presses up arrow on the first message, stay at the first message\n      this.currentMessageIndex = 0;\n    } else if (this.currentMessageIndex >= userMessages.length) {\n      // If the user presses down arrow on the last previous message, clear the input\n      this.currentMessageIndex = undefined;\n      this.question = '';\n      return;\n    }\n\n    this.question = userMessages[this.currentMessageIndex].content;\n  }\n}\n","\n<ng-container *ngIf=\"!initializationError\">\n  <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n    <ul class=\"list-group list-group-flush overflow-auto pb-5\" #messageList>\n      <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n        <!-- Regular messages -->\n        <li class=\"list-group-item\" *ngIf=\"message.additionalProperties.display\"\n            [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n          <sq-chat-message\n            [class.sq-user-message]=\"message.role !== 'assistant'\"\n            [message]=\"message\"\n            [conversation]=\"messages\"\n            [assistantIcon]=\"customAssistantIcon || assistantIcon\"\n            [streaming]=\"last && (chatService.streaming$ | async)\"\n            [canEdit]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role !== 'assistant'\"\n            [canRegenerate]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'assistant' && last\"\n            [canCopy]=\"!(last && (chatService.streaming$ | async)) && messageToEdit === undefined && message.role === 'assistant'\"\n            (edit)=\"editMessage(index)\" (regenerate)=\"regenerateMessage(index)\"\n            (referenceClicked)=\"referenceClicked.emit($event)\"\n            (openPreview)=\"openPreview.emit($event)\">\n          </sq-chat-message>\n        </li>\n      </ng-container>\n\n      <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n        <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n      </li>\n    </ul>\n\n    <div class=\"user-input mt-auto\" *ngIf=\"enableChat\">\n      <div class=\"py-2\">\n        <ng-container *ngTemplateOutlet=\"inputTpl\"></ng-container>\n        <div class=\"text-end small text-muted px-3\" *ngIf=\"showCredits\">\n          powered by {{modelDescription?.displayName}} <ng-container *ngIf=\"privacyUrl\"> - <a [href]=\"privacyUrl\" target=\"_blank\">privacy notice</a></ng-container>\n        </div>\n      </div>\n    </div>\n  </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n  <div class=\"spinner-grow text-success d-block mx-auto my-5\" role=\"status\">\n    <span class=\"visually-hidden\">Loading...</span>\n  </div>\n</ng-template>\n\n<ng-template #inputTpl>\n  <div class=\"px-3 py-1\">\n    <div class=\"ast-input-container\">\n      <i class=\"fas fa-search\"></i>\n      <input #questionInput\n        type=\"text\" class=\"form-control\"\n        placeholder=\"Ask something\" autofocus\n        [(ngModel)]=\"question\"\n        (keyup)=\"onKeyUp($event)\"\n        [disabled]=\"(loading$ | async) || (chatService.streaming$ | async)\">\n      <button\n        *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Send message\"\n        (click)=\"submitQuestion()\">\n        <i class=\"fas fa-paper-plane\"></i>\n      </button>\n      <!--<button\n        *ngIf=\"(chatService.streaming$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Stop generating\"\n        (click)=\"terminateFetch()\">\n        <i class=\"fas fa-stop\"></i>\n      </button>-->\n      <button\n        *ngIf=\"messageToEdit\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Cancel edition\"\n        (click)=\"messageToEdit = undefined; question = ''\">\n        <i class=\"fas fa-undo-alt\"></i>\n      </button>\n      <div class=\"sq-floating-scroll\" *ngIf=\"!isAtBottom\">\n        <button class=\"btn shadow\" (click)=\"scrollDown()\">\n          <i class=\"fas fa-angle-double-down\"></i>\n        </button>\n      </div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
425
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.component.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.component.ts","../../../../projects/assistant/chat/chat.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,SAAS,EAAE,YAAY,EAAc,YAAY,EAAE,KAAK,EAAgC,MAAM,EAA8B,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1N,OAAO,EAAE,MAAM,EAAE,MAAM,4BAA4B,CAAC;AACpD,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;AAE3D,OAAO,EAAE,mBAAmB,EAAU,MAAM,4BAA4B,CAAC;AACzE,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AACpG,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;;;;AAkBtD,MAAM,OAAO,aAAc,SAAQ,aAAa;IAkE9C;QACE,KAAK,EAAE,CAAC;QAjEH,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,qBAAgB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAChD,gBAAW,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QACtC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,kBAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QACtC,qBAAgB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAC/C,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAIvC,+CAA+C;QACtC,UAAK,GAAU,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC;QAQjD,2DAA2D;QAClD,aAAQ,GAAyB,WAAW,CAAC;QACtD,yDAAyD;QAChD,oBAAe,GAAqC,IAAI,GAAG,EAAE,CAAC;QACvE,2GAA2G;QAClG,kCAA6B,GAAG,KAAK,CAAC;QAEtC,wBAAmB,GAAG,EAAE,CAAC;QAExB,SAAI,GAAG,IAAI,YAAY,EAAiB,CAAC;QACzC,qBAAgB,GAAG,IAAI,YAAY,EAAU,CAAC;QAC9C,gBAAW,GAAG,IAAI,YAAY,EAAyB,CAAC;QAC/C,aAAQ,GAAG,IAAI,YAAY,CAAU,KAAK,CAAC,CAAC;QACrD,UAAK,GAAG,IAAI,YAAY,EAAO,CAAC;QACxB,YAAO,GAAG,IAAI,YAAY,EAAc,CAAC;QAS3D,cAAS,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAEtE,aAAQ,GAAG,EAAE,CAAC;QAEd,aAAQ,GAAa,EAAE,CAAC;QAExB,QAAG,GAAG,IAAI,YAAY,EAAE,CAAC;QASzB,aAAQ,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAErE,wBAAmB,GAAG,KAAK,CAAC;QAC5B,eAAU,GAAG,IAAI,CAAC;QAClB,wBAAmB,GAAG,KAAK,CAAC;QAC5B,qBAAgB,GAAG,KAAK,CAAC;QAKvB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC;YAC5B,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,YAAY;YACnB,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE;SACrC,CAAC,CAAC,CAAC;IACN,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAC3B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,gBAAgB,CAAC,EACxC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,EACvC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,EAAE,CAAC,EAC3C,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAC7C,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,EAClC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,EACvC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,EAC5B,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,EAC3B,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,EAC5C,GAAG,CAAC,MAAM,CAAC,EAAE;YACX,IAAI,CAAC,MAAM,GAAG,MAAO,CAAC;YACtB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,gBAAgB,CAAC;YAClE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC1B,IAAI;gBACF,IAAI,CAAC,sBAAsB,EAAE,CAAC;gBAC9B,IAAG,CAAC,IAAI,CAAC,mBAAmB,EAAE;oBAC5B,IAAI,CAAC,aAAa,EAAE,CAAC;oBACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;oBACzB,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;iBACjC;aACF;YAAC,OAAO,KAAK,EAAE;gBACd,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAA;gBAC/B,MAAM,KAAK,CAAC;aACb;QACH,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC5B,IAAI,IAAI,CAAC,MAAM,EAAE;YACf,IAAI,CAAC,aAAa,EAAE,CAAC;SACtB;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;QACvB,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;IACvC,CAAC;IAED,sBAAsB;QACpB,QAAQ,IAAI,CAAC,QAAQ,EAAE;YACrB,KAAK,MAAM;gBACT,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;gBACpC,MAAM;YACR,KAAK,WAAW;gBACd,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC;gBACzC,MAAM;YACR;gBACE,MAAM,IAAI,KAAK,CAAC,yFAAyF,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;SAC9H;QACD,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACpD,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAC/E,CAAC;IAED,IAAa,OAAO,KAAK,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAExC,aAAa;QACnB,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC;QACpC,uGAAuG;QACvG,IAAI,OAAO,EAAE,eAAe,IAAI,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,WAAW,YAAY,oBAAoB,EAAE;YACxG,IAAI,CAAC,WAAW,CAAC,uBAAuB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;SAChE;QACD;;;WAGG;QACH,IAAI,CAAC,IAAI,CAAC,mBAAmB,IAAI,OAAO,EAAE,IAAI,EAAE;YAC9C,MAAM,QAAQ,GAAG,GAAG,EAAE;gBACpB,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE;oBACjE,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC,CAAC,kCAAkC;iBACrE;gBACD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAK,CAAC,QAAQ,CAAC,CAAA;YACpC,CAAC,CAAC;YACF,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC;SACjD;QACD;;;;WAIG;QACH,IAAI,IAAI,CAAC,mBAAmB,IAAI,OAAO,EAAE,KAAK,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,KAAK,KAAK,OAAO,EAAE;YAC3G,MAAM,aAAa,GAAG,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC;YAClD,MAAM,YAAY,GAAG,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC;YAEhD,IAAI,IAAI,CAAC,8BAA8B,CAAC,CAAC,CAAC,IAAI,CAAC,8BAA8B,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE;gBACjH,IAAI,CAAC,6BAA6B,CAAC,YAAY,CAAC,CAAC;aAClD;SACF;IACH,CAAC;IAEO,6BAA6B,CAAC,KAAY;QAChD,MAAM,SAAS,GAAG,EAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,YAAY,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,KAAK,EAAC,EAAC,CAAC;QAC5H,MAAM,OAAO,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,UAAU,EAAE,EAAC,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAC,CAAC,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,iBAAiB,EAAC,EAAC,CAAC;QACnO;;;WAGG;QACH,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;YACrB,MAAM,YAAY,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,gBAAgB,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,cAAc,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,cAAc,EAAE,WAAW,EAAE,IAAI,EAAC,EAAC,CAAC;YACvQ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC,CAAC;SACzH;aAAM;YACL,MAAM,UAAU,GAAG,EAAC,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,8DAA8D,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAC,CAAC;YACvJ,IAAI,CAAC,QAAQ,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;SAC7B;IACH,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,SAAS,CAAC,IAAI,CAAC,WAAY,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACrI,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,4BAA4B,EAAE,CAAC;YACtD,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QAC3B,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAC5H,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,YAAY,CAAC;QAC1F,QAAO,IAAI,CAAC,gBAAgB,EAAE,QAAQ,EAAE;YACtC,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;gBACrB,MAAM;YACR,KAAK,aAAa;gBAChB,IAAI,CAAC,UAAU,GAAG,gFAAgF,CAAC;gBACnG,MAAM;YACR,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,uCAAuC,CAAC;gBAC1D,MAAM;YACR,KAAK,QAAQ;gBACX,IAAI,CAAC,UAAU,GAAG,6BAA6B,CAAC;gBAChD,MAAM;SACT;QACD,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3B,CAAC;IAED,cAAc;QACZ,IAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,IAAI,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;YAC/E,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS,EAAE;gBACpC,gCAAgC;gBAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;gBACvE,uGAAuG;gBACvG,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;gBACzF,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;aAChC;YACD,mBAAmB;YACnB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;YACrE,kCAAkC;YAClC,IAAI,CAAC,aAAc,CAAC,aAAa,CAAC,KAAK,GAAG,EAAE,CAAC;YAC7C,IAAI,CAAC,aAAc,CAAC,aAAa,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC;SACzD;IACH,CAAC;IAEO,WAAW,CAAC,QAAgB,EAAE,YAA2B;QAC/D,MAAM,OAAO,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,QAAQ,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAC,EAAC,CAAC;QAC5G,MAAM,QAAQ,GAAG,CAAC,GAAG,YAAY,EAAE,OAAO,CAAC,CAAC;QAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;IACvB,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,QAAuB;QAClC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;QACzB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzB,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;QACrC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC;aACjE,SAAS,CAAC;YACT,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC;YACzC,KAAK,EAAE,GAAG,CAAC,EAAE;gBACX,IAAI,CAAC,cAAc,EAAE,CAAC;gBACtB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACnB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACvB,CAAC;YACD,QAAQ,EAAE,GAAG,EAAE;gBACb,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,CAAC;SACF,CAAC,CAAC;QAEL,IAAG,IAAI,CAAC,6BAA6B,EAAE;YACrC,IAAI,CAAC,UAAU,EAAE,CAAC;SACnB;IACH,CAAC;IAED;;;OAGG;IACH,UAAU,CAAC,QAAuB;QAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACzB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAG,IAAI,CAAC,6BAA6B,EAAE;YACrC,IAAI,CAAC,UAAU,EAAE,CAAC;SACnB;IACH,CAAC;IAEO,4BAA4B;QAClC,IAAG,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE;YAClC,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,YAAY,GAAG,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,SAAS,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,YAAY,CAAC;SACjK;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,UAAU;QACR,UAAU,CAAC,GAAG,EAAE;YACd,IAAG,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE;gBAClC,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,YAAY,CAAC;gBACvF,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;aAC1B;QACH,CAAC,EAAE,EAAE,CAAC,CAAC;IACT,CAAC;IAED;;;;OAIG;IACH,OAAO;QACL,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,CAAC,wBAAwB;QACpE,IAAI,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE;YACzC,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC,CAAC,kCAAkC;SACrE;QACD,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,mBAAmB;IAC7C,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,MAAM,SAAS,GAAG,EAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,YAAY,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,KAAK,EAAC,EAAC,CAAC;QAC5H,MAAM,OAAO,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,UAAU,EAAE,EAAC,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAC,CAAC,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,iBAAiB,EAAC,EAAC,CAAC;QACnO,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,KAAK,KAAK,OAAO,EAAE;YAC7D,wHAAwH;YACxH,mIAAmI;YACnI,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;gBACrB,MAAM,YAAY,GAAG,EAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,gBAAgB,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,cAAc,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,cAAc,EAAE,WAAW,EAAE,IAAI,EAAC,EAAC,CAAC;gBACvQ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC,CAAC;aACzH;iBAAM;gBACL,MAAM,UAAU,GAAG,EAAC,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,8DAA8D,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAC,CAAC;gBACvJ,IAAI,CAAC,QAAQ,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;aAC7B;SACF;aAAM;YACL,IAAI,CAAC,QAAQ,CAAC,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;SACrC;IACH,CAAC;IAED;;;;;;OAMG;IACH,QAAQ,CAAC,QAAsB,EAAE,MAAe;QAC9C,IAAI,CAAC,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;YACzC,OAAO,CAAC,KAAK,CAAC,oFAAoF,EAAE,QAAQ,CAAC,CAAC;YAC9G,OAAO;SACR;QACD,IAAI,MAAM,EAAE;YACV,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;SACzC;QACD,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,QAAQ,CAAC;QACxC,MAAM,WAAW,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACpC,IAAG,WAAW,IAAI,WAAW,CAAC,IAAI,KAAK,MAAM,EAAE;YAC7C,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,+EAA+E;SACtG;aACI;YACH,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,qFAAqF;YAChH,IAAI,CAAC,cAAc,EAAE,CAAC;SACvB;IACH,CAAC;IAED;;;;OAIG;IACH,SAAS;QACP,IAAG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE;YACvB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,aAAa;SAC9C;QACD,IAAI,CAAC,WAAW,CAAC,WAAW,GAAG,SAAS,CAAC,CAAC,qBAAqB;QAC/D,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAED,UAAU;QACR,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzB,IAAI,CAAC,GAAG,CAAC,GAAG,CACV,IAAI,CAAC,WAAW,CAAC,cAAc;aAC5B,IAAI,CACH,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,EAChC,SAAS,CAAC,SAAS,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,SAAU,CAAC,EAAE,CAAC,CAAC,EACpE,MAAM,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC,CAAC,gBAAgB,CAAC,EAC9C,GAAG,CAAC,gBAAgB,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAiB,CAAC,OAAO,EAAE,gBAAiB,CAAC,EAAE,CAAC,CAAC,CACxF,CAAC,SAAS,EAAE,CAChB,CAAC;IACJ,CAAC;IAED,cAAc;QACZ,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,CAAC;QACrC,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;QAClC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,aAAa,EAAE,aAAa,CAAC,KAAK,EAAE,CAAC;QAC5C,CAAC,CAAC,CAAA;QACF,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3B,CAAC;IAED,WAAW,CAAC,KAAa;QACvB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC;QAC7D,IAAI,CAAC,aAAa,EAAE,aAAa,CAAC,KAAK,EAAE,CAAC;IAC5C,CAAC;IAED,iBAAiB,CAAC,KAAa;QAC5B,kFAAkF;QACnF,MAAM,cAAc,GAAG,IAAI,CAAC,WAAW,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;QACrE,4CAA4C;QAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACpC,mBAAmB;QACnB,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;IAC7B,CAAC;IAED,OAAO,CAAC,KAAoB;QAC1B,QAAQ,KAAK,CAAC,GAAG,EAAE;YACjB,KAAK,WAAW;gBACd,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,MAAM;YACR,KAAK,OAAO;gBACV,IAAI,CAAC,KAAK,CAAC,QAAQ;oBAAE,IAAI,CAAC,cAAc,EAAE,CAAC;gBAC3C,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,MAAM;YACR;gBACE,MAAM;SACT;IACH,CAAC;IAED,eAAe;QACb,MAAM,SAAS,GAAG,GAAG,CAAC;QACtB,MAAM,EAAE,GAAG,IAAI,CAAC,aAAc,CAAC,aAAa,CAAC;QAC7C,EAAE,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,SAAS,IAAI,CAAC;QACtC,EAAE,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC;QACzB,EAAE,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,IAAI,CAAC;QACzC,EAAE,CAAC,KAAK,CAAC,SAAS,GAAG,EAAE,CAAC,YAAY,IAAI,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC;IAC1E,CAAC;;0GAjbU,aAAa;8FAAb,aAAa,0fARb;QACT,eAAe;QACf,oBAAoB;KACrB,yXC5BH,49HA+FA,03GDhEY,YAAY,2dAAE,WAAW,+mBAAE,oBAAoB;2FAE9C,aAAa;kBAZzB,SAAS;+BACE,YAAY,aAGX;wBACT,eAAe;wBACf,oBAAoB;qBACrB,mBACgB,uBAAuB,CAAC,MAAM,cACnC,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,oBAAoB,CAAC;0EAajD,UAAU;sBAAlB,KAAK;gBAEG,KAAK;sBAAb,KAAK;gBAOG,8BAA8B;sBAAtC,KAAK;gBAEG,QAAQ;sBAAhB,KAAK;gBAEG,eAAe;sBAAvB,KAAK;gBAEG,6BAA6B;sBAArC,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,mBAAmB;sBAA3B,KAAK;gBAEI,IAAI;sBAAb,MAAM;gBACG,gBAAgB;sBAAzB,MAAM;gBACG,WAAW;sBAApB,MAAM;gBACY,QAAQ;sBAA1B,MAAM;uBAAC,SAAS;gBACP,KAAK;sBAAd,MAAM;gBACW,OAAO;sBAAxB,MAAM;uBAAC,QAAQ;gBAEU,WAAW;sBAApC,SAAS;uBAAC,aAAa;gBACI,aAAa;sBAAxC,SAAS;uBAAC,eAAe;gBAEE,UAAU;sBAArC,YAAY;uBAAC,YAAY","sourcesContent":["import { inject, ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChanges, TemplateRef, ViewChild } from \"@angular/core\";\nimport { Action } from \"@sinequa/components/action\";\nimport { AbstractFacet } from \"@sinequa/components/facet\";\nimport { SearchService } from \"@sinequa/components/search\";\nimport { Query } from \"@sinequa/core/app-utils\";\nimport { PrincipalWebService, Record } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Subscription, filter, fromEvent, map, merge, switchMap, tap } from \"rxjs\";\nimport { ChatService } from \"./chat.service\";\nimport { ChatContextAttachment, ChatConfig, ChatMessage, GllmModelDescription, MessageHandler, RawMessage } from \"./types\";\nimport { InstanceManagerService } from \"./instance-manager.service\";\nimport { WebSocketChatService } from \"./websocket-chat.service\";\nimport { ChatMessageComponent } from \"./chat-message/chat-message.component\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { RestChatService } from \"./rest-chat.service\";\n\nexport interface InitChat {\n  messages: RawMessage[];\n}\n\n@Component({\n  selector: 'sq-chat-v3', // mandatory since @sinequa/components already has the same tag-name \"sq-chat\"\n  templateUrl: './chat.component.html',\n  styleUrls: ['./chat.component.scss'],\n  providers: [\n    RestChatService,\n    WebSocketChatService\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, FormsModule, ChatMessageComponent]\n})\nexport class ChatComponent extends AbstractFacet implements OnInit, OnChanges, OnDestroy {\n\n  public loginService = inject(LoginService);\n  public websocketService = inject(WebSocketChatService);\n  public restService = inject(RestChatService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public searchService = inject(SearchService);\n  public principalService = inject(PrincipalWebService);\n  public cdr = inject(ChangeDetectorRef);\n\n  /** Define the key based on it, the chat service instance will be stored */\n  @Input() instanceId: string;\n  /** Define the query to use to fetch answers */\n  @Input() query: Query = this.searchService.query;\n  /** Function that determines whether the chat should be reloaded after the query changes\n   * If not provided, the chat will be reloaded by default\n   * @param prevQuery The previous query\n   * @param newQuery The new query\n   * @returns true if the chat should be reloaded, false otherwise\n   */\n  @Input() queryChangeShouldTriggerReload: (prevQuery: any, newQuery: any) => boolean;\n  /** Define the protocol to be used for this chat instance*/\n  @Input() protocol: 'REST' | 'WEBSOCKET' = \"WEBSOCKET\";\n  /** Map of listeners overriding default registered ones*/\n  @Input() messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  /** When the assistant answer a user question, automatically scroll down to the bottom of the discussion */\n  @Input() automaticScrollToLastResponse = false;\n  @Input() chat?: InitChat;\n  @Input() customAssistantIcon = '';\n\n  @Output() data = new EventEmitter<ChatMessage[]>();\n  @Output() referenceClicked = new EventEmitter<Record>();\n  @Output() openPreview = new EventEmitter<ChatContextAttachment>();\n  @Output(\"loading\") loading$ = new EventEmitter<boolean>(false);\n  @Output() error = new EventEmitter<any>();\n  @Output(\"config\") _config = new EventEmitter<ChatConfig>();\n\n  @ViewChild('messageList') messageList?: ElementRef<HTMLUListElement>;\n  @ViewChild('questionInput') questionInput?: ElementRef<HTMLTextAreaElement>;\n\n  @ContentChild('loadingTpl') loadingTpl?: TemplateRef<any>;\n\n  chatService: ChatService;\n  config: ChatConfig;\n  messages$ = new BehaviorSubject<ChatMessage[] | undefined>(undefined);\n\n  question = '';\n\n  _actions: Action[] = [];\n\n  sub = new Subscription();\n  dataSubscription: Subscription | undefined;\n\n  /** Variables that depend on the type of model in use */\n  modelDescription?: GllmModelDescription;\n  assistantIcon: string;\n  privacyUrl: string;\n\n  messageToEdit?: number;\n  changes$ = new BehaviorSubject<SimpleChanges | undefined>(undefined);\n  currentMessageIndex: number | undefined;\n  firstChangesHandled = false;\n  isAtBottom = true;\n  initializationError = false;\n  enabledUserInput = false;\n\n  constructor() {\n    super();\n\n    this._actions.push(new Action({\n      icon: 'fas fa-sync',\n      title: 'Reset chat',\n      action: () => this.loadDefaultChat()\n    }));\n  }\n\n  ngOnInit(): void {\n    this.sub.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        map(_ => this.chatService.initChatConfig()),\n        switchMap(() => this.chatService.initConfig$),\n        filter(initConfig => !!initConfig),\n        switchMap(_ => this.chatService.init()),\n        filter(success => !!success),\n        tap(_ => this.onLoadChat()),\n        switchMap(_ => this.chatService.chatConfig$),\n        tap(config => {\n          this.config = config!;\n          this.enabledUserInput = this.config.modeSettings.enabledUserInput;\n          this._config.emit(config);\n          try {\n            this.updateModelDescription();\n            if(!this.firstChangesHandled) {\n              this.handleChanges();\n              this.addScrollListener();\n              this.firstChangesHandled = true;\n            }\n          } catch (error) {\n            this.initializationError = true\n            throw error;\n          }\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    this.changes$.next(changes);\n    if (this.config) {\n      this.handleChanges();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.sub.unsubscribe();\n    this.dataSubscription?.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    switch (this.protocol) {\n      case 'REST':\n        this.chatService = this.restService;\n        break;\n      case 'WEBSOCKET':\n        this.chatService = this.websocketService;\n        break;\n      default:\n        throw new Error(`Could not found a ChatService implementation corresponding to the provided protocol: '${this.protocol}'`);\n    }\n    this.chatService.setChatInstanceId(this.instanceId);\n    this.instanceManagerService.storeInstance(this.instanceId, this.chatService);\n  }\n\n  override get actions() { return this._actions; }\n\n  private handleChanges() {\n    const changes = this.changes$.value;\n    // If the chat service is a WebSocketChatService, handle the override of the message handlers if exists\n    if (changes?.messageHandlers && this.messageHandlers && this.chatService instanceof WebSocketChatService) {\n      this.chatService.overrideMessageHandlers(this.messageHandlers);\n    }\n    /**\n     * Initialize the chat with the provided chat messages if exists, otherwise load the default chat\n     * Once the chat is initialized (firstChangesHandled is true), allow opening the chat with the new provided messages (if exists)\n     */\n    if (!this.firstChangesHandled || changes?.chat) {\n      const openChat = () => {\n        if (this.messages$.value && this.config.savedChatSettings.enabled) {\n          this.chatService.listSavedChat(); // Refresh the list of saved chats\n        }\n        this.openChat(this.chat!.messages)\n      };\n      this.chat ? openChat() : this.loadDefaultChat();\n    }\n    /**\n     * If the chat is initialized, the initialization event is \"Query\", the query changes and the queryChangeShouldTriggerReload function is provided,\n     * then the chat should be reloaded if the function returns true\n     * Otherwise, the chat should be reloaded by default\n     */\n    if (this.firstChangesHandled && changes?.query && this.config.modeSettings.initialization.event === 'Query') {\n      const previousQuery = changes.query.previousValue;\n      const currentQuery = changes.query.currentValue;\n\n      if (this.queryChangeShouldTriggerReload ? this.queryChangeShouldTriggerReload(previousQuery, currentQuery) : true) {\n        this.triggerReloadAfterQueryChange(currentQuery);\n      }\n    }\n  }\n\n  private triggerReloadAfterQueryChange(query: Query) {\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n    /**\n     * If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant\n     * Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results\n     */\n    if (!!this.query.text) {\n      const userQueryMsg = {role: 'user', content: this.query.text, additionalProperties: {display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true}};\n      this.openChat(this.config.modeSettings.sendUserPrompt ? [systemMsg, userMsg, userQueryMsg] : [systemMsg, userQueryMsg]);\n    } else {\n      const warningMsg = {role: 'assistant', content: \"You must perform a full text search to retrieve some results\", additionalProperties: {display: true}};\n      this.openChat([warningMsg]);\n    }\n  }\n\n  private addScrollListener() {\n    this.sub.add(\n      merge(this.loading$, this.messages$, this.chatService.streaming$, fromEvent(this.messageList!.nativeElement, 'scroll')).subscribe(() => {\n        this.isAtBottom = this.toggleScrollButtonVisibility();\n        this.cdr.detectChanges();\n      })\n    );\n  }\n\n  updateModelDescription() {\n    this.modelDescription = this.chatService.getModel(this.config.defaultValues.service_id, this.config.defaultValues.model_id);\n    this.assistantIcon = !!this.customAssistantIcon ? this.customAssistantIcon : 'sq-sinequa';\n    switch(this.modelDescription?.provider) {\n      case 'Google':\n        this.privacyUrl = '';\n        break;\n      case 'AzureOpenAI':\n        this.privacyUrl = 'https://learn.microsoft.com/en-us/legal/cognitive-services/openai/data-privacy';\n        break;\n      case 'OpenAI':\n        this.privacyUrl = 'https://openai.com/enterprise-privacy';\n        break;\n      case 'Cohere':\n        this.privacyUrl = 'https://cohere.com/security';\n        break;\n    }\n    this.cdr.detectChanges();\n  }\n\n  submitQuestion() {\n    if(this.question.trim() && this.messages$.value && this.chatService.chatHistory) {\n      if (this.messageToEdit !== undefined) {\n        // Update the messages in the UI\n        this.messages$.next(this.messages$.value.slice(0, this.messageToEdit));\n        // Update the raw messages in the chat history which is the clean version used to make the next request\n        this.chatService.chatHistory = this.chatService.chatHistory.slice(0, this.messageToEdit);\n        this.messageToEdit = undefined;\n      }\n      // Fetch the answer\n      this.fetchAnswer(this.question.trim(), this.chatService.chatHistory);\n      // Clear the input value in the UI\n      this.questionInput!.nativeElement.value = '';\n      this.questionInput!.nativeElement.style.height = `auto`;\n    }\n  }\n\n  private fetchAnswer(question: string, conversation: ChatMessage[]) {\n    const userMsg = {role: 'user', content: question, additionalProperties: {display: true, isUserInput: true}};\n    const messages = [...conversation, userMsg];\n    this.messages$.next(messages);\n    this.fetch(messages);\n  }\n\n  /**\n   * Given a list of messages, fetch the server for a continuation and updates\n   * the list of messages accordingly.\n   * @param messages\n   */\n  public fetch(messages: ChatMessage[]) {\n    this.cdr.detectChanges();\n    this.loading$.next(true);\n    this.dataSubscription?.unsubscribe();\n    this.dataSubscription = this.chatService.fetch(messages, this.query)\n      .subscribe({\n        next: res => this.updateData(res.history),\n        error: err => {\n          this.terminateFetch();\n          console.error(err);\n          this.error.emit(err);\n        },\n        complete: () => {\n          this.terminateFetch();\n        }\n      });\n\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  /**\n   * Update the UI with the new messages\n   * @param messages\n   */\n  updateData(messages: ChatMessage[]) {\n    this.messages$.next(messages);\n    this.data.emit(messages);\n    this.loading$.next(false);\n    this.question = '';\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  private toggleScrollButtonVisibility(): boolean {\n    if(this.messageList?.nativeElement) {\n      return Math.round(this.messageList?.nativeElement.scrollHeight - this.messageList?.nativeElement.scrollTop - 1) <= this.messageList?.nativeElement.clientHeight;\n    }\n    return true;\n  }\n\n  scrollDown() {\n    setTimeout(() => {\n      if(this.messageList?.nativeElement) {\n        this.messageList.nativeElement.scrollTop = this.messageList.nativeElement.scrollHeight;\n        this.cdr.detectChanges();\n      }\n    }, 10);\n  }\n\n  /**\n   * Start a new chat with the defaultValues settings\n   * The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat\n   * If the savedChat feature is enabled, the list of saved chats will be refreshed\n   */\n  newChat() {\n    this.chatService.setSavedChatId(undefined); // Reset the savedChatId\n    if (this.config.savedChatSettings.enabled) {\n      this.chatService.listSavedChat(); // Refresh the list of saved chats\n    }\n    this.loadDefaultChat(); // Start a new chat\n  }\n\n  /**\n   * Start the default chat with the defaultValues settings\n   * If the chat is meant to be initialized with event === \"Query\", the corresponding user query message will be added to the chat history\n   */\n  loadDefaultChat() {\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n    if (this.config.modeSettings.initialization.event === 'Query') {\n      // If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant\n      // Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results\n      if (!!this.query.text) {\n        const userQueryMsg = {role: 'user', content: this.query.text, additionalProperties: {display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true}};\n        this.openChat(this.config.modeSettings.sendUserPrompt ? [systemMsg, userMsg, userQueryMsg] : [systemMsg, userQueryMsg]);\n      } else {\n        const warningMsg = {role: 'assistant', content: \"You must perform a full text search to retrieve some results\", additionalProperties: {display: true}};\n        this.openChat([warningMsg]);\n      }\n    } else {\n      this.openChat([systemMsg, userMsg]);\n    }\n  }\n\n  /**\n   * Start/open a new chat with the provided messages and chatId\n   * If the last message is from the user, a request to the assistant is made to get an answer\n   * If the last message is from the assistant, the conversation is loaded right away\n   * @param messages The list of messages of the chat\n   * @param chatId  The id of the chat. If provided (ie. an existing discussion in the saved chat index), update the savedChatId in the chat service for the upcoming saved chat operations\n   */\n  openChat(messages: RawMessage[], chatId?: string) {\n    if (!messages || !Array.isArray(messages)) {\n      console.error('Error occurs while trying to load the chat discussion. Invalid messages received :', messages);\n      return;\n    }\n    if (chatId) {\n      this.chatService.setSavedChatId(chatId);\n    }\n    this.resetChat();\n    this.messages$.next(messages);\n    this.chatService.chatHistory = messages;\n    const lastMessage = messages.at(-1);\n    if(lastMessage && lastMessage.role === 'user') {\n      this.fetch(messages); // If the last message if from a user, an answer from the assistant is expected\n    }\n    else {\n      this.updateData(messages); // If the last message if from the assistant, we can load the conversation right away\n      this.terminateFetch();\n    }\n  }\n\n  /**\n   * Reset the chat by clearing the messages and the chat history\n   * The question input will be focused after the chat is reset\n   * The fetch subscription will be terminated\n   */\n  resetChat() {\n    if(this.messages$.value) {\n      this.messages$.next(undefined); // Reset chat\n    }\n    this.chatService.chatHistory = undefined; // Reset chat history\n    this.question = '';\n    this.terminateFetch();\n  }\n\n  onLoadChat() {\n    this.loading$.next(true);\n    this.sub.add(\n      this.chatService.loadSavedChat$\n        .pipe(\n          filter(savedChat => !!savedChat),\n          switchMap(savedChat => this.chatService.getSavedChat(savedChat!.id)),\n          filter(savedChatHistory => !!savedChatHistory),\n          tap(savedChatHistory => this.openChat(savedChatHistory!.history, savedChatHistory!.id))\n        ).subscribe()\n    );\n  }\n\n  terminateFetch() {\n    this.dataSubscription?.unsubscribe();\n    this.dataSubscription = undefined;\n    this.loading$.next(false);\n    setTimeout(() => {\n      this.questionInput?.nativeElement.focus();\n    })\n    this.cdr.detectChanges();\n  }\n\n  editMessage(index: number) {\n    this.messageToEdit = index;\n    this.question = this.chatService.chatHistory![index].content;\n    this.questionInput?.nativeElement.focus();\n  }\n\n  regenerateMessage(index: number) {\n     // Define the chat history based on which the assistant will generate a new answer\n    const slicedMessages = this.chatService.chatHistory!.slice(0, index);\n    // Accordingly update the messages in the UI\n    this.messages$.next(slicedMessages);\n    // Fetch the answer\n    this.fetch(slicedMessages);\n  }\n\n  onKeyUp(event: KeyboardEvent): void {\n    switch (event.key) {\n      case 'Backspace':\n        this.calculateHeight();\n        break;\n      case 'Enter':\n        if (!event.shiftKey) this.submitQuestion();\n        this.calculateHeight();\n        break;\n      default:\n        break;\n    }\n  }\n\n  calculateHeight(): void {\n    const maxHeight = 170;\n    const el = this.questionInput!.nativeElement;\n    el.style.maxHeight = `${maxHeight}px`;\n    el.style.height = 'auto';\n    el.style.height = `${el.scrollHeight}px`;\n    el.style.overflowY = el.scrollHeight >= maxHeight ? 'scroll' : 'hidden';\n  }\n}\n","\n<ng-container *ngIf=\"!initializationError\">\n  <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n\n    <ul class=\"list-group list-group-flush overflow-auto pe-2 pb-2\" #messageList>\n      <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n        <!-- Regular messages -->\n        <li class=\"list-group-item\" *ngIf=\"message.additionalProperties.display\"\n            [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n          <sq-chat-message\n            [class.sq-user-message]=\"message.role !== 'assistant'\"\n            [message]=\"message\"\n            [conversation]=\"messages\"\n            [assistantIcon]=\"customAssistantIcon || assistantIcon\"\n            [streaming]=\"last && (chatService.streaming$ | async)\"\n            [canEdit]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role !== 'assistant'\"\n            [canRegenerate]=\"(loading$ | async) === false && (chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'assistant' && last\"\n            [canCopy]=\"!(last && (chatService.streaming$ | async)) && messageToEdit === undefined && message.role === 'assistant'\"\n            (edit)=\"editMessage(index)\" (regenerate)=\"regenerateMessage(index)\"\n            (referenceClicked)=\"referenceClicked.emit($event)\"\n            (openPreview)=\"openPreview.emit($event)\">\n          </sq-chat-message>\n        </li>\n      </ng-container>\n\n      <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n        <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n      </li>\n    </ul>\n\n    <div class=\"user-input mt-auto\" *ngIf=\"enabledUserInput\">\n      <div class=\"py-2\">\n        <ng-container *ngTemplateOutlet=\"inputTpl\"></ng-container>\n        <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n          {{ config?.globalSettings?.disclaimer }}\n        </div>\n      </div>\n    </div>\n  </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n  <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n    <span class=\"visually-hidden\">Loading...</span>\n  </div>\n</ng-template>\n\n<ng-template #inputTpl>\n  <div class=\"px-3 py-1\">\n    <div class=\"ast-input-container\">\n      <button disabled class=\"btn btn-light\">\n        <i class=\"fas fa-search\"></i>\n      </button>\n      <textarea #questionInput rows=\"1\"\n        type=\"text\" class=\"form-control\"\n        placeholder=\"Ask something\" autofocus\n        [(ngModel)]=\"question\"\n        (keyup)=\"onKeyUp($event)\"\n        (keydown)=\"calculateHeight()\"\n        [disabled]=\"(loading$ | async) || (chatService.streaming$ | async)\">\n      </textarea>\n      <button\n        *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Send message\"\n        (click)=\"submitQuestion()\">\n        <i class=\"fas fa-paper-plane\"></i>\n      </button>\n      <!--<button\n        *ngIf=\"(chatService.streaming$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Stop generating\"\n        (click)=\"terminateFetch()\">\n        <i class=\"fas fa-stop\"></i>\n      </button>-->\n      <button\n        *ngIf=\"messageToEdit\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        title=\"Cancel edition\"\n        (click)=\"messageToEdit = undefined; question = ''\">\n        <i class=\"fas fa-undo-alt\"></i>\n      </button>\n      <div class=\"sq-floating-scroll\" *ngIf=\"!isAtBottom\">\n        <button class=\"btn shadow\" (click)=\"scrollDown()\">\n          <i class=\"fas fa-angle-double-down\"></i>\n        </button>\n      </div>\n    </div>\n  </div>\n</ng-template>\n"]}
|