@sinequa/assistant 3.7.4 → 3.8.0
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-settings-v3/chat-settings-v3.component.d.ts +6 -5
- package/chat/chat.component.d.ts +11 -7
- package/chat/chat.service.d.ts +5 -5
- package/chat/documents-upload/document-list/document-list.component.d.ts +77 -0
- package/chat/documents-upload/document-overview/document-overview.component.d.ts +41 -0
- package/chat/documents-upload/document-upload/document-upload.component.d.ts +98 -0
- package/chat/documents-upload/documents-upload.model.d.ts +66 -0
- package/chat/documents-upload/documents-upload.service.d.ts +174 -0
- package/chat/public-api.d.ts +5 -0
- package/chat/styles/assistant.scss +2 -0
- package/chat/types.d.ts +77 -21
- package/chat/websocket-chat.service.d.ts +4 -4
- package/esm2020/chat/chat-message/chat-message.component.mjs +3 -3
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +3 -3
- package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +13 -10
- package/esm2020/chat/chat.component.mjs +25 -17
- package/esm2020/chat/chat.service.mjs +9 -9
- package/esm2020/chat/documents-upload/document-list/document-list.component.mjs +191 -0
- package/esm2020/chat/documents-upload/document-overview/document-overview.component.mjs +80 -0
- package/esm2020/chat/documents-upload/document-upload/document-upload.component.mjs +258 -0
- package/esm2020/chat/documents-upload/documents-upload.model.mjs +2 -0
- package/esm2020/chat/documents-upload/documents-upload.service.mjs +289 -0
- package/esm2020/chat/public-api.mjs +6 -1
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +4 -4
- package/esm2020/chat/token-progress-bar/token-progress-bar.component.mjs +3 -3
- package/esm2020/chat/types.mjs +17 -6
- package/esm2020/chat/websocket-chat.service.mjs +3 -3
- package/fesm2015/sinequa-assistant-chat.mjs +3654 -2859
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
- package/fesm2020/sinequa-assistant-chat.mjs +3639 -2845
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +4 -2
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { LoginService } from '@sinequa/core/login';
|
|
3
|
+
import { JsonMethodPluginService } from '@sinequa/core/web-services';
|
|
4
|
+
import { BehaviorSubject, catchError, filter, map, take } from 'rxjs';
|
|
5
|
+
import { HttpClient, HttpEventType, HttpHeaders } from '@angular/common/http';
|
|
6
|
+
import { AppService } from '@sinequa/core/app-utils';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export class DocumentsUploadService {
|
|
9
|
+
constructor() {
|
|
10
|
+
/** Documents' upload configuration. */
|
|
11
|
+
this.uploadConfig$ = new BehaviorSubject(undefined);
|
|
12
|
+
this.jsonMethodWebService = inject(JsonMethodPluginService);
|
|
13
|
+
this.loginService = inject(LoginService);
|
|
14
|
+
this.appService = inject(AppService);
|
|
15
|
+
this.http = inject(HttpClient);
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Initializes the file upload service with the provided chat service instance.
|
|
19
|
+
*
|
|
20
|
+
* @param chatService - An instance of ChatService. If not provided, an error is thrown.
|
|
21
|
+
* @throws {Error} If the chatService instance is not provided or if there is an error during initialization.
|
|
22
|
+
*/
|
|
23
|
+
init(chatService) {
|
|
24
|
+
if (!chatService)
|
|
25
|
+
throw new Error('A chatService instance must be provided');
|
|
26
|
+
this.chatService = chatService;
|
|
27
|
+
try {
|
|
28
|
+
this.getRequestsUrl();
|
|
29
|
+
this.getUploadConfig();
|
|
30
|
+
}
|
|
31
|
+
catch (error) {
|
|
32
|
+
throw new Error(error);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Fetch the endpoint to use for the file upload related requests
|
|
37
|
+
* @throws {Error} If the property 'restEndpoint' is not provided in the assistant configuration
|
|
38
|
+
*/
|
|
39
|
+
getRequestsUrl() {
|
|
40
|
+
if (!!this.chatService.assistantConfig$.value.connectionSettings.restEndpoint) {
|
|
41
|
+
this.REQUEST_URL = this.chatService.assistantConfig$.value.connectionSettings.restEndpoint;
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Retrieves the upload configuration for documents.
|
|
49
|
+
* The response, which contains the upload configuration, is emitted to the `uploadConfig$` observable.
|
|
50
|
+
*
|
|
51
|
+
* @throws {Error} If there is an error invoking the `documentsUploadConfigGet` action.
|
|
52
|
+
*/
|
|
53
|
+
getUploadConfig() {
|
|
54
|
+
const data = {
|
|
55
|
+
action: "documentsUploadConfigGet",
|
|
56
|
+
appName: this.appService.appName,
|
|
57
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
58
|
+
};
|
|
59
|
+
this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(take(1), catchError((error) => {
|
|
60
|
+
throw new Error(`Error invoking documentsUploadConfigGet: ${error}`);
|
|
61
|
+
})).subscribe((res) => this.uploadConfig$.next(res));
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Uploads documents to the server.
|
|
65
|
+
*
|
|
66
|
+
* This method takes a FormData object which should contain all files to be uploaded.
|
|
67
|
+
* Each file should be appended to the FormData object with its original name.
|
|
68
|
+
*
|
|
69
|
+
* For example:
|
|
70
|
+
* ```typescript
|
|
71
|
+
* const formData = new FormData();
|
|
72
|
+
* formData.append(file.name, file.file);
|
|
73
|
+
* ```
|
|
74
|
+
*
|
|
75
|
+
* @param formData - The FormData object containing the files to be uploaded.
|
|
76
|
+
* - Each file should be appended to the FormData object with its original name.
|
|
77
|
+
* For example:
|
|
78
|
+
* ```typescript
|
|
79
|
+
* const formData = new FormData();
|
|
80
|
+
* formData.append(file.name, file.file);
|
|
81
|
+
* ```
|
|
82
|
+
*
|
|
83
|
+
* @param enableProgress - (Optional) A boolean parameter that controls whether progress reporting is enabled.
|
|
84
|
+
* - Defaults to `true`.
|
|
85
|
+
* - If `true`, the `reportProgress` option is set in the request options, enabling the track of upload progress based on the underlying browser's progress events.
|
|
86
|
+
* - If `false`, progress events are not reported.
|
|
87
|
+
*
|
|
88
|
+
* @returns An Observable that emits events of type `UploadEvent`:
|
|
89
|
+
* - `UploadProgressEvent`: Emitted during the upload process, containing:
|
|
90
|
+
* - * `type`: "uploadProgress" - The type of the event.
|
|
91
|
+
* - * `loaded`: number - The number of bytes uploaded so far.
|
|
92
|
+
* - * `total`: number - The total number of bytes to be uploaded (if available).
|
|
93
|
+
* - * `progress`: number - The percentage of upload progress (calculated as `(loaded / total) * 100`).
|
|
94
|
+
* - `UploadResponseEvent`: Emitted when the upload is complete, containing:
|
|
95
|
+
* - * `type`: "response" - The type of the event.
|
|
96
|
+
* - * `body`: `UploadResponse` - The server's response, including:
|
|
97
|
+
* - * - * `statusToken`: string - A token to track the status of the indexing process.
|
|
98
|
+
* - * - * `executionTime`: string - The time taken to execute the upload.
|
|
99
|
+
* - * - * `connectorResponse` (optional): Additional response data from the connector, containing:
|
|
100
|
+
* - * - * - * `collection`: string - The collection name.
|
|
101
|
+
* - * - * - * `action`: string - The action performed.
|
|
102
|
+
* - * - * - * `stats`: object - Statistics about the upload, including:
|
|
103
|
+
* - * - * - * - * `nbDocUpdated`: number - The number of documents updated.
|
|
104
|
+
*
|
|
105
|
+
* @throws {Error} If there is an error invoking the `documentsupload` action.
|
|
106
|
+
*
|
|
107
|
+
* Example usage:
|
|
108
|
+
* ```typescript
|
|
109
|
+
* const formData = new FormData();
|
|
110
|
+
* formData.append(file.name, file.file);
|
|
111
|
+
* this.documentsUploadService.uploadDocuments(formData).subscribe(event => {
|
|
112
|
+
* if (event.type === 'uploadProgress') {
|
|
113
|
+
* console.log(`Progress: ${event.progress}%`);
|
|
114
|
+
* } else if (event.type === 'response') {
|
|
115
|
+
* console.log('Upload complete:', event.body);
|
|
116
|
+
* }
|
|
117
|
+
* });
|
|
118
|
+
* ```
|
|
119
|
+
*/
|
|
120
|
+
uploadDocuments(formData, enableProgress = true) {
|
|
121
|
+
// Add the required "data" field as a JSON string
|
|
122
|
+
const data = {
|
|
123
|
+
action: "documentsupload",
|
|
124
|
+
appName: this.appService.appName,
|
|
125
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
126
|
+
};
|
|
127
|
+
formData.append('data', JSON.stringify(data));
|
|
128
|
+
const options = {
|
|
129
|
+
headers: new HttpHeaders({ 'Accept': 'application/json' }),
|
|
130
|
+
observe: 'events',
|
|
131
|
+
};
|
|
132
|
+
if (enableProgress)
|
|
133
|
+
options['reportProgress'] = true;
|
|
134
|
+
const url = this.jsonMethodWebService.makeUrl(this.REQUEST_URL);
|
|
135
|
+
/**
|
|
136
|
+
* Need to use the HttpClient directly to avoid the JSONMethodPluginService's behavior
|
|
137
|
+
* because it is altering the request and thus making it unable to properly set the Content-Type header with the boundary
|
|
138
|
+
*/
|
|
139
|
+
return this.http.post(url, formData, options).pipe(filter((event) => event.type === HttpEventType.Response || (enableProgress && event.type === HttpEventType.UploadProgress)), map((event) => {
|
|
140
|
+
switch (event.type) {
|
|
141
|
+
case HttpEventType.UploadProgress: {
|
|
142
|
+
const { loaded = 0, total = 0 } = event;
|
|
143
|
+
return { type: 'uploadProgress', loaded, total, progress: total > 0 ? Math.round((loaded / total) * 100) : 0 };
|
|
144
|
+
}
|
|
145
|
+
case HttpEventType.Response:
|
|
146
|
+
return { type: 'response', body: event.body };
|
|
147
|
+
default:
|
|
148
|
+
throw new Error("Unexpected event type"); // Should not reach here, but for typescript sake.
|
|
149
|
+
}
|
|
150
|
+
}), catchError((error) => {
|
|
151
|
+
throw new Error(`Error invoking documentsupload: ${error}`);
|
|
152
|
+
}));
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Retrieves the indexing status of a predefined uploaded document(s)
|
|
156
|
+
* based on the provided status token returned from the upload process.
|
|
157
|
+
*
|
|
158
|
+
* @param token - The status token used to track the indexing process.
|
|
159
|
+
* @returns An Observable that emits the server response of type `IndexingResponse`.
|
|
160
|
+
* The `IndexingResponse` object contains:
|
|
161
|
+
* - `docs`: array - An array of document status objects, each containing:
|
|
162
|
+
* - * `id`: string - The document ID.
|
|
163
|
+
* - * `fileName`: string - The name of the file.
|
|
164
|
+
* - * `operation`: "Add" | "Update" - The operation performed on the document.
|
|
165
|
+
* - * `status`: "Indexing" | "Indexed" | "Error" - The status of the document.
|
|
166
|
+
* - * `previousIndexationTime`: string | null - The previous indexation time.
|
|
167
|
+
* - * `currentIndexationTime`: string | null - The current indexation time.
|
|
168
|
+
* - `executionTime`: string - The time taken to retrieve the indexing status.
|
|
169
|
+
* @throws {Error} If there is an error invoking the `documentsUploadStatus` action.
|
|
170
|
+
*/
|
|
171
|
+
getIndexingStatus(token) {
|
|
172
|
+
const data = {
|
|
173
|
+
action: "documentsUploadStatus",
|
|
174
|
+
appName: this.appService.appName,
|
|
175
|
+
statusToken: token,
|
|
176
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
177
|
+
};
|
|
178
|
+
return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(catchError((error) => {
|
|
179
|
+
throw new Error(`Error invoking documentsUploadStatus: ${error}`);
|
|
180
|
+
}));
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Retrieves a list of uploaded documents.
|
|
184
|
+
*
|
|
185
|
+
* @param {string[]} [columns] - Optional array of additional index columns to include for each document.
|
|
186
|
+
* @param {number} [skip] - Optional number of documents to skip for pagination. If set, count also needs to be set. No skip if not set.
|
|
187
|
+
* @param {number} [count] - Optional number of documents to take for pagination (unlimited if not set).
|
|
188
|
+
* @returns {Observable<UploadedDocuments>} An observable that emits the list of uploaded documents.
|
|
189
|
+
* The `UploadedDocuments` object contains:
|
|
190
|
+
* - `docs`: array - An array of objects representing a single document with the following properties:
|
|
191
|
+
* - * `id`: string - A unique identifier for the document.
|
|
192
|
+
* - * `fileName`: string - The name of the file.
|
|
193
|
+
* - * `title`: string - The title of the document.
|
|
194
|
+
* - * `fileExt`: string - The file extension of the document.
|
|
195
|
+
* - * `indexationTime`: string - The time when the document was indexed.
|
|
196
|
+
* - * `size`: number - The size of the document.
|
|
197
|
+
* - * `sizeDisplay`: string - A human-readable representation of the document size.
|
|
198
|
+
* - * `[key: string]`: any - Additional columns asked in the request, value of the column for the document.
|
|
199
|
+
* - `count`: number - The count of documents in the current batch or subset.
|
|
200
|
+
* - `totalCount`: number - The total count of documents available.
|
|
201
|
+
* - `executionTime`: string - The time taken to process the documents.
|
|
202
|
+
* @throws {Error} Throws an error if the request fails.
|
|
203
|
+
*/
|
|
204
|
+
getDocumentsList(columns, skip, count) {
|
|
205
|
+
const data = {
|
|
206
|
+
action: "documentsList",
|
|
207
|
+
appName: this.appService.appName,
|
|
208
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
209
|
+
};
|
|
210
|
+
if (columns)
|
|
211
|
+
data["columns"] = columns;
|
|
212
|
+
if (skip)
|
|
213
|
+
data["skip"] = skip;
|
|
214
|
+
if (count)
|
|
215
|
+
data["count"] = count;
|
|
216
|
+
return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(catchError((error) => {
|
|
217
|
+
throw new Error(`Error invoking documentsList: ${error}`);
|
|
218
|
+
}));
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Deletes the uploaded documents with the specified IDs.
|
|
222
|
+
*
|
|
223
|
+
* @param docIds - An array of document IDs to delete.
|
|
224
|
+
* @returns An observable that emits the server response of type `deleteDocumentsResponse`.
|
|
225
|
+
* The `deleteDocumentsResponse` object contains:
|
|
226
|
+
* - `deletedCount`: number - The number of deleted documents.
|
|
227
|
+
* - `executionTime`: string - The time taken to delete the documents.
|
|
228
|
+
* @throws {Error} If there is an error invoking the `documentsDelete` action.
|
|
229
|
+
*/
|
|
230
|
+
deleteDocuments(docIds) {
|
|
231
|
+
const data = {
|
|
232
|
+
action: "documentsDelete",
|
|
233
|
+
docIds,
|
|
234
|
+
appName: this.appService.appName,
|
|
235
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
236
|
+
};
|
|
237
|
+
return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(catchError((error) => {
|
|
238
|
+
throw new Error(`Error invoking documentsDelete: ${error}`);
|
|
239
|
+
}));
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Deletes all uploaded documents.
|
|
243
|
+
*
|
|
244
|
+
* @returns An observable that emits the server response of type `deleteDocumentsResponse`.
|
|
245
|
+
* The `deleteDocumentsResponse` object contains:
|
|
246
|
+
* - `deletedCount`: number - The number of deleted documents.
|
|
247
|
+
* - `executionTime`: string - The time taken to delete the documents.
|
|
248
|
+
* @throws {Error} If there is an error invoking the `documentsDeleteAll` action.
|
|
249
|
+
*/
|
|
250
|
+
deleteAllDocuments() {
|
|
251
|
+
const data = {
|
|
252
|
+
action: "documentsDeleteAll",
|
|
253
|
+
appName: this.appService.appName,
|
|
254
|
+
debug: this.chatService.assistantConfig$.value.defaultValues.debug
|
|
255
|
+
};
|
|
256
|
+
return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(catchError((error) => {
|
|
257
|
+
throw new Error(`Error invoking documentsDeleteAll: ${error}`);
|
|
258
|
+
}));
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Formats the given file size in bytes into a human-readable string.
|
|
262
|
+
*
|
|
263
|
+
* @param bytes - The size of the file in bytes.
|
|
264
|
+
* @returns A string representing the file size in a human-readable format (e.g., "10.24 KB", "1.00 MB").
|
|
265
|
+
*/
|
|
266
|
+
formatFileSize(bytes) {
|
|
267
|
+
if (bytes === 0)
|
|
268
|
+
return "0 Bytes";
|
|
269
|
+
const k = 1024;
|
|
270
|
+
const sizes = ["Bytes", "KB", "MB", "GB", "TB"];
|
|
271
|
+
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
272
|
+
return (Number.parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + " " + sizes[i]);
|
|
273
|
+
}
|
|
274
|
+
/**
|
|
275
|
+
* Converts the given file size in bytes into megabytes.
|
|
276
|
+
*
|
|
277
|
+
* @param bytes - The size of the file in bytes.
|
|
278
|
+
* @returns The size of the file in megabytes.
|
|
279
|
+
*/
|
|
280
|
+
convertBytesToMB(bytes) {
|
|
281
|
+
return bytes / (1024 * 1024);
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
DocumentsUploadService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DocumentsUploadService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
285
|
+
DocumentsUploadService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DocumentsUploadService });
|
|
286
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DocumentsUploadService, decorators: [{
|
|
287
|
+
type: Injectable
|
|
288
|
+
}] });
|
|
289
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"documents-upload.service.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/documents-upload/documents-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { LoginService } from '@sinequa/core/login';\nimport { JsonMethodPluginService } from '@sinequa/core/web-services';\nimport { ChatService } from '../chat.service';\nimport { BehaviorSubject, catchError, filter, map, Observable, take } from 'rxjs';\nimport { deleteDocumentsResponse, IndexingInfos, UploadConfig, UploadedDocuments, UploadEvent } from './documents-upload.model';\nimport { HttpClient, HttpEventType, HttpHeaders, HttpResponse, HttpUploadProgressEvent } from '@angular/common/http';\nimport { AppService } from '@sinequa/core/app-utils';\n\n@Injectable()\nexport class DocumentsUploadService {\n\n  /** Name of the assistant plugin. */\n  public REQUEST_URL: string;\n  /** Documents' upload configuration. */\n  uploadConfig$ = new BehaviorSubject<UploadConfig | undefined>(undefined);\n  public chatService: ChatService;\n\n  public jsonMethodWebService = inject(JsonMethodPluginService);\n  public loginService = inject(LoginService);\n  public appService = inject(AppService);\n  public http = inject(HttpClient);\n\n  /**\n   * Initializes the file upload service with the provided chat service instance.\n   *\n   * @param chatService - An instance of ChatService. If not provided, an error is thrown.\n   * @throws {Error} If the chatService instance is not provided or if there is an error during initialization.\n   */\n  init(chatService: ChatService | undefined): void {\n    if (!chatService) throw new Error('A chatService instance must be provided');\n\n    this.chatService = chatService;\n    try {\n      this.getRequestsUrl();\n      this.getUploadConfig();\n    } catch (error) {\n      throw new Error(error);\n    }\n  }\n\n  /**\n   * Fetch the endpoint to use for the file upload related requests\n   * @throws {Error} If the property 'restEndpoint' is not provided in the assistant configuration\n   */\n  getRequestsUrl() {\n    if (!!this.chatService.assistantConfig$.value!.connectionSettings.restEndpoint) {\n      this.REQUEST_URL = this.chatService.assistantConfig$.value!.connectionSettings.restEndpoint;\n    } else {\n      throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);\n    }\n  }\n\n  /**\n   * Retrieves the upload configuration for documents.\n   * The response, which contains the upload configuration, is emitted to the `uploadConfig$` observable.\n   *\n   * @throws {Error} If there is an error invoking the `documentsUploadConfigGet` action.\n   */\n  public getUploadConfig(): void {\n    const data = {\n      action: \"documentsUploadConfigGet\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      take(1),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadConfigGet: ${error}`);\n      })\n    ).subscribe((res: UploadConfig) => this.uploadConfig$.next(res));\n  }\n\n  /**\n   * Uploads documents to the server.\n   *\n   * This method takes a FormData object which should contain all files to be uploaded.\n   * Each file should be appended to the FormData object with its original name.\n   *\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param formData - The FormData object containing the files to be uploaded.\n   * - Each file should be appended to the FormData object with its original name.\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param enableProgress - (Optional) A boolean parameter that controls whether progress reporting is enabled.\n   * - Defaults to `true`.\n   * - If `true`, the `reportProgress` option is set in the request options, enabling the track of upload progress based on the underlying browser's progress events.\n   * - If `false`, progress events are not reported.\n   *\n   * @returns An Observable that emits events of type `UploadEvent`:\n   * - `UploadProgressEvent`: Emitted during the upload process, containing:\n   *   - *  `type`: \"uploadProgress\" - The type of the event.\n   *   - *  `loaded`: number - The number of bytes uploaded so far.\n   *   - *  `total`: number - The total number of bytes to be uploaded (if available).\n   *   - *  `progress`: number - The percentage of upload progress (calculated as `(loaded / total) * 100`).\n   * - `UploadResponseEvent`: Emitted when the upload is complete, containing:\n   *   - *  `type`: \"response\" - The type of the event.\n   *   - *  `body`: `UploadResponse` - The server's response, including:\n   *     - * - * `statusToken`: string - A token to track the status of the indexing process.\n   *     - * - *  `executionTime`: string - The time taken to execute the upload.\n   *     - * - *  `connectorResponse` (optional): Additional response data from the connector, containing:\n   *       - * - * - *  `collection`: string - The collection name.\n   *       - * - * - *  `action`: string - The action performed.\n   *       - * - * - *  `stats`: object - Statistics about the upload, including:\n   *         - * - * - * - * `nbDocUpdated`: number - The number of documents updated.\n   *\n   * @throws {Error} If there is an error invoking the `documentsupload` action.\n   *\n   * Example usage:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * this.documentsUploadService.uploadDocuments(formData).subscribe(event => {\n   *   if (event.type === 'uploadProgress') {\n   *     console.log(`Progress: ${event.progress}%`);\n   *   } else if (event.type === 'response') {\n   *     console.log('Upload complete:', event.body);\n   *   }\n   * });\n   * ```\n   */\n  public uploadDocuments(formData: FormData, enableProgress: boolean = true): Observable<UploadEvent> {\n    // Add the required \"data\" field as a JSON string\n    const data = {\n      action: \"documentsupload\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n    formData.append('data', JSON.stringify(data));\n\n    const options: any  = {\n      headers: new HttpHeaders({ 'Accept': 'application/json' }),\n      observe: 'events',\n    };\n    if (enableProgress) options['reportProgress'] = true;\n    const url = this.jsonMethodWebService.makeUrl(this.REQUEST_URL);\n\n    /**\n     * Need to use the HttpClient directly to avoid the JSONMethodPluginService's behavior\n     * because it is altering the request and thus making it unable to properly set the Content-Type header with the boundary\n     */\n    return this.http.post<any>(url, formData, options).pipe(\n      filter((event): event is HttpUploadProgressEvent | HttpResponse<any> =>\n        event.type === HttpEventType.Response || (enableProgress && event.type === HttpEventType.UploadProgress)\n      ),\n      map((event): UploadEvent => {\n        switch (event.type) {\n          case HttpEventType.UploadProgress: {\n            const { loaded = 0, total = 0 } = event;\n            return { type: 'uploadProgress', loaded, total, progress: total > 0 ? Math.round((loaded / total) * 100) : 0 };\n          }\n          case HttpEventType.Response:\n            return { type: 'response', body: event.body };\n          default:\n            throw new Error(\"Unexpected event type\"); // Should not reach here, but for typescript sake.\n        }\n      }),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsupload: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves the indexing status of a predefined uploaded document(s)\n   * based on the provided status token returned from the upload process.\n   *\n   * @param token - The status token used to track the indexing process.\n   * @returns An Observable that emits the server response of type `IndexingResponse`.\n   * The `IndexingResponse` object contains:\n   * - `docs`: array - An array of document status objects, each containing:\n   *   - * `id`: string - The document ID.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `operation`: \"Add\" | \"Update\" - The operation performed on the document.\n   *   - * `status`: \"Indexing\" | \"Indexed\" | \"Error\" - The status of the document.\n   *   - * `previousIndexationTime`: string | null - The previous indexation time.\n   *   - * `currentIndexationTime`: string | null - The current indexation time.\n   * - `executionTime`: string - The time taken to retrieve the indexing status.\n   * @throws {Error} If there is an error invoking the `documentsUploadStatus` action.\n   */\n  public getIndexingStatus(token: string): Observable<IndexingInfos> {\n    const data = {\n      action: \"documentsUploadStatus\",\n      appName: this.appService.appName,\n      statusToken: token,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadStatus: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves a list of uploaded documents.\n   *\n   * @param {string[]} [columns] - Optional array of additional index columns to include for each document.\n   * @param {number} [skip] - Optional number of documents to skip for pagination. If set, count also needs to be set. No skip if not set.\n   * @param {number} [count] - Optional number of documents to take for pagination (unlimited if not set).\n   * @returns {Observable<UploadedDocuments>} An observable that emits the list of uploaded documents.\n   * The `UploadedDocuments` object contains:\n   * - `docs`: array - An array of objects representing a single document with the following properties:\n   *   - * `id`: string - A unique identifier for the document.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `title`: string - The title of the document.\n   *   - * `fileExt`: string - The file extension of the document.\n   *   - * `indexationTime`: string - The time when the document was indexed.\n   *   - * `size`: number - The size of the document.\n   *   - * `sizeDisplay`: string - A human-readable representation of the document size.\n   *   - * `[key: string]`: any - Additional columns asked in the request, value of the column for the document.\n   * - `count`: number - The count of documents in the current batch or subset.\n   * - `totalCount`: number - The total count of documents available.\n   * - `executionTime`: string - The time taken to process the documents.\n   * @throws {Error} Throws an error if the request fails.\n   */\n  public getDocumentsList(columns?: string[], skip?: number, count?: number): Observable<UploadedDocuments> {\n    const data = {\n      action: \"documentsList\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    if (columns) data[\"columns\"] = columns;\n    if (skip) data[\"skip\"] = skip;\n    if (count) data[\"count\"] = count;\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsList: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes the uploaded documents with the specified IDs.\n   *\n   * @param docIds - An array of document IDs to delete.\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDelete` action.\n   */\n  public deleteDocuments(docIds: string[]): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDelete\",\n      docIds,\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDelete: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes all uploaded documents.\n   *\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDeleteAll` action.\n   */\n  public deleteAllDocuments(): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDeleteAll\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return this.jsonMethodWebService.post(this.REQUEST_URL, data).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDeleteAll: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Formats the given file size in bytes into a human-readable string.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns A string representing the file size in a human-readable format (e.g., \"10.24 KB\", \"1.00 MB\").\n   */\n  public formatFileSize(bytes: number): string {\n    if (bytes === 0) return \"0 Bytes\";\n    const k = 1024;\n    const sizes = [\"Bytes\", \"KB\", \"MB\", \"GB\", \"TB\"];\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\n    return (\n      Number.parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + \" \" + sizes[i]\n    );\n  }\n\n  /**\n   * Converts the given file size in bytes into megabytes.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns The size of the file in megabytes.\n   */\n  public convertBytesToMB(bytes: number): number {\n    return bytes / (1024 * 1024);\n  }\n\n}\n"]}
|
|
@@ -10,4 +10,9 @@ export * from './websocket-chat.service';
|
|
|
10
10
|
export * from './rest-chat.service';
|
|
11
11
|
export * from './instance-manager.service';
|
|
12
12
|
export * from './prompt.component';
|
|
13
|
-
|
|
13
|
+
export * from './documents-upload/document-upload/document-upload.component';
|
|
14
|
+
export * from './documents-upload/document-list/document-list.component';
|
|
15
|
+
export * from './documents-upload/documents-upload.service';
|
|
16
|
+
export * from './documents-upload/documents-upload.model';
|
|
17
|
+
export * from './documents-upload/document-overview/document-overview.component';
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2Fzc2lzdGFudC9jaGF0L3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYywrQ0FBK0MsQ0FBQztBQUM5RCxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLDZDQUE2QyxDQUFDO0FBQzVELGNBQWMscUNBQXFDLENBQUM7QUFDcEQsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLG9CQUFvQixDQUFDO0FBQ25DLGNBQWMsOERBQThELENBQUM7QUFDN0UsY0FBYywwREFBMEQsQ0FBQztBQUN6RSxjQUFjLDZDQUE2QyxDQUFDO0FBQzVELGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyxrRUFBa0UsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vY2hhdC1zZXR0aW5ncy12My9jaGF0LXNldHRpbmdzLXYzLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2NoYXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY2hhdC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vc2F2ZWQtY2hhdHMvc2F2ZWQtY2hhdHMuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vaW5pdGlhbHMtYXZhdGFyL2luaXRpYWxzLWF2YXRhci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9mb3JtYXQtaWNvbi9mb3JtYXQtaWNvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL21lc3NhZ2VzL2luZGV4JztcbmV4cG9ydCAqIGZyb20gJy4vd2Vic29ja2V0LWNoYXQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3Jlc3QtY2hhdC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vaW5zdGFuY2UtbWFuYWdlci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vcHJvbXB0LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2RvY3VtZW50cy11cGxvYWQvZG9jdW1lbnQtdXBsb2FkL2RvY3VtZW50LXVwbG9hZC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9kb2N1bWVudHMtdXBsb2FkL2RvY3VtZW50LWxpc3QvZG9jdW1lbnQtbGlzdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9kb2N1bWVudHMtdXBsb2FkL2RvY3VtZW50cy11cGxvYWQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2RvY3VtZW50cy11cGxvYWQvZG9jdW1lbnRzLXVwbG9hZC5tb2RlbCc7XG5leHBvcnQgKiBmcm9tICcuL2RvY3VtZW50cy11cGxvYWQvZG9jdW1lbnQtb3ZlcnZpZXcvZG9jdW1lbnQtb3ZlcnZpZXcuY29tcG9uZW50JzsiXX0=
|
|
@@ -23,7 +23,7 @@ export class SavedChatsComponent {
|
|
|
23
23
|
this.notificationsService = inject(NotificationsService);
|
|
24
24
|
}
|
|
25
25
|
ngOnInit() {
|
|
26
|
-
this.subscription.add(this.loginService.events.pipe(filter(e => e.type === 'login-complete'), tap(_ => this.instantiateChatService()), switchMap(_ => this.chatService.userOverride$), tap(_ => {
|
|
26
|
+
this.subscription.add(this.loginService.events.pipe(filter(e => e.type === 'login-complete'), tap(_ => this.instantiateChatService()), switchMap(_ => this.chatService.userOverride$), filter(isOverridden => isOverridden !== undefined), tap(_ => {
|
|
27
27
|
this.onListSavedChat();
|
|
28
28
|
this.chatService.listSavedChat();
|
|
29
29
|
})).subscribe());
|
|
@@ -161,10 +161,10 @@ export class SavedChatsComponent {
|
|
|
161
161
|
}
|
|
162
162
|
}
|
|
163
163
|
SavedChatsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: SavedChatsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
164
|
-
SavedChatsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, ngImport: i0, template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.savedChatId === savedChat.id\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"onRename($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"onDelete($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\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: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: ModalModule }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
|
|
164
|
+
SavedChatsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, ngImport: i0, template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.savedChatId === savedChat.id\">\n <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"onRename($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"onDelete($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.saved-chats .saved-chat-date{font-size:12px;font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\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: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: ModalModule }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
|
|
165
165
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: SavedChatsComponent, decorators: [{
|
|
166
166
|
type: Component,
|
|
167
|
-
args: [{ selector: 'sq-saved-chats-v3', standalone: true, imports: [CommonModule, ModalModule, UtilsModule], template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.savedChatId === savedChat.id\">\n <span class=\"title me-1\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"onRename($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"onDelete($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.saved-chats .saved-chat-date{font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
|
|
167
|
+
args: [{ selector: 'sq-saved-chats-v3', standalone: true, imports: [CommonModule, ModalModule, UtilsModule], template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.savedChatId === savedChat.id\">\n <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n (click)=\"onRename($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n (click)=\"onDelete($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.saved-chats .saved-chat-date{font-size:12px;font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
|
|
168
168
|
}], propDecorators: { instanceId: [{
|
|
169
169
|
type: Input
|
|
170
170
|
}], load: [{
|
|
@@ -172,4 +172,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
172
172
|
}], delete: [{
|
|
173
173
|
type: Output
|
|
174
174
|
}] } });
|
|
175
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAqB,MAAM,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAClG,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,SAAS,EAAE,GAAG,EAAE,UAAU,EAAE,eAAe,EAAE,MAAM,MAAM,CAAC;AACrG,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAe,WAAW,EAAe,YAAY,EAAiB,WAAW,EAAE,MAAM,qBAAqB,CAAC;AACtH,OAAO,EAAE,oBAAoB,EAAE,MAAM,4BAA4B,CAAC;AAClE,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AACxD,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,aAAa,EAAE,UAAU,EAAE,cAAc,EAAE,gBAAgB,EAAE,kBAAkB,EAAE,iBAAiB,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAGvM,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;;;;AASrE,MAAM,OAAO,mBAAmB;IAPhC;QAWY,SAAI,GAAG,IAAI,YAAY,EAAa,CAAC;QACrC,WAAM,GAAG,IAAI,YAAY,EAAa,CAAC;QAGjD,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;QAClC,uBAAkB,GAAG,IAAI,eAAe,CAAuC,EAAE,CAAC,CAAC;QAE5E,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;KAmK5D;IAjKC,QAAQ;QACN,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,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,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,EAC9C,GAAG,CAAC,CAAC,CAAC,EAAE;YACN,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QACnC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,sBAAsB,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9E,CAAC;IAED,eAAe;QACb,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,SAAS,CACpC,CAAC,UAAuB,EAAE,EAAE;YAC1B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,UAAU,CAAC,CAAC,CAAC;QACxE,CAAC,CACF,CACF,CAAC;IACJ,CAAC;IAED,MAAM,CAAC,SAAoB;QACzB,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,KAAK,EAAE;YACvF,OAAO;SACR;QACD,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;QAC9C,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;QAC9C,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAChD,IAAI,CAAC,WAAW,CAAC,kBAAkB,CAAC,iBAAiB,EAAE,EAAE,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC;QACzE,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QACjC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAED,QAAQ,CAAC,KAAY,EAAE,SAAoB;QACzC,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,KAAK,EAAE;YACvF,OAAO;SACR;QACD,MAAM,KAAK,GAAkB;YAC3B,KAAK,EAAE,yBAAyB;YAChC,OAAO,EAAE,+CAA+C,SAAS,CAAC,KAAK,IAAI;YAC3E,OAAO,EAAE;gBACL,IAAI,WAAW,CAAC,EAAC,MAAM,6BAAoB,EAAC,CAAC;gBAC7C,IAAI,WAAW,CAAC,EAAC,MAAM,yBAAgB,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAC,CAAC;aAC3E;YACD,MAAM,EAAE,SAAS,CAAC,KAAK;YACvB,UAAU,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC;SAClC,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;YACzC,IAAG,GAAG,4BAAmB,EAAE;gBACvB,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,WAAW,CAAC,eAAe,CAAC,SAAS,CAAC,EAAE,EAAE,KAAK,CAAC,MAAM,CAAC;qBACzD,IAAI,CACH,GAAG,CAAC,GAAG,EAAE;oBACP,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,yBAAyB,SAAS,CAAC,KAAK,uCAAuC,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC;oBACnI,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;gBACnC,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;oBACnB,OAAO,CAAC,KAAK,CAAC,+CAA+C,EAAE,KAAK,CAAC,CAAC;oBACtE,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,uDAAuD,SAAS,CAAC,KAAK,GAAG,CAAC,CAAC;oBAC3G,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;gBACjC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CAChB,CAAC;gBACF,IAAI,CAAC,WAAW,CAAC,kBAAkB,CAAC,mBAAmB,EAAE,EAAC,MAAM,EAAE,KAAK,CAAC,MAAM,EAAC,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC;aAClG;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,QAAQ,CAAC,KAAY,EAAE,SAAoB;QACzC,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,KAAK,EAAE;YACvF,OAAO;SACR;QACD,IAAI,CAAC,YAAY;aACd,OAAO,CAAC;YACL,KAAK,EAAE,yBAAyB;YAChC,OAAO,EAAE,2CAA2C,SAAS,CAAC,KAAK,6BAA6B;YAChG,OAAO,EAAE;gBACL,IAAI,WAAW,CAAC,EAAC,MAAM,6BAAoB,EAAC,CAAC;gBAC7C,IAAI,WAAW,CAAC,EAAC,MAAM,yBAAgB,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,EAAC,CAAC;aAC5E;YACD,WAAW,6BAAqB;SACnC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;YACV,IAAG,GAAG,4BAAmB,EAAE;gBACzB,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;qBAC7C,IAAI,CACH,GAAG,CAAC,GAAG,EAAE;oBACP,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,yBAAyB,SAAS,CAAC,KAAK,kCAAkC,CAAC,CAAC;oBAC9G,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBAC5B,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;gBACnC,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;oBACnB,OAAO,CAAC,KAAK,CAAC,+CAA+C,EAAE,KAAK,CAAC,CAAC;oBACtE,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,uDAAuD,SAAS,CAAC,KAAK,GAAG,CAAC,CAAC;oBAC3G,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;gBACjC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CAChB,CAAC;gBACF,IAAI,CAAC,WAAW,CAAC,kBAAkB,CAAC,mBAAmB,EAAE,EAAE,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC;aAC5E;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,sBAAsB,CAAC,UAAuB;QACpD,MAAM,iBAAiB,GAAG,IAAI,GAAG,EAAuB,CAAC;QAEzD,UAAU;aACP,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,OAAO,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,OAAO,EAAE,CAAC;aACrF,OAAO,CAAC,SAAS,CAAC,EAAE;YACjB,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,wEAAwE;YAEpJ,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;gBACpC,iBAAiB,CAAC,GAAG,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;aACrC;YAED,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEL,OAAO,KAAK,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;QAAA,CAAC;IAC5E,CAAC;IAEO,WAAW,CAAC,IAAU;QAC5B,IAAI,OAAO,CAAC,IAAI,CAAC,EAAE;YACf,OAAO,OAAO,CAAC;SAClB;aAAM,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE;YAC1B,OAAO,WAAW,CAAC;SACtB;aAAM,IAAI,UAAU,CAAC,IAAI,CAAC,EAAE;YACzB,OAAO,WAAW,CAAC;SACtB;aAAM,IAAI,gBAAgB,CAAC,cAAc,EAAE,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE;YACxD,OAAO,WAAW,CAAC;SACpB;aAAM,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE;YAC1B,OAAO,YAAY,CAAC;SACvB;aAAM,IAAI,kBAAkB,CAAC,cAAc,EAAE,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE;YAC1D,OAAO,YAAY,CAAC;SACrB;aAAM,IAAI,aAAa,CAAC,IAAI,CAAC,EAAE;YAC5B,OAAO,cAAc,CAAC;SACzB;aAAM,IAAI,kBAAkB,CAAC,cAAc,EAAE,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE;YACxD,OAAO,cAAc,CAAC;SACzB;aAAM,IAAI,UAAU,CAAC,IAAI,CAAC,EAAE;YACzB,OAAO,WAAW,CAAC;SACtB;aAAM,IAAI,iBAAiB,CAAC,cAAc,EAAE,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE;YACxD,OAAO,WAAW,CAAC;SACtB;aAAM;YACH,OAAO,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;SAC/B;IACH,CAAC;;gHA/KU,mBAAmB;oGAAnB,mBAAmB,gKCpBhC,k4BAgBA,yiEDEY,YAAY,oTAAE,WAAW,8BAAE,WAAW;2FAErC,mBAAmB;kBAP/B,SAAS;+BACE,mBAAmB,cAGjB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,WAAW,CAAC;8BAIxC,UAAU;sBAAlB,KAAK;gBAEI,IAAI;sBAAb,MAAM;gBACG,MAAM;sBAAf,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { Validators } from '@angular/forms';\nimport { Subscription, catchError, filter, switchMap, tap, throwError, BehaviorSubject } from \"rxjs\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService, PromptOptions, ModalModule } from \"@sinequa/core/modal\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { format, parseISO, isToday, isYesterday, isThisWeek, isThisMonth, isThisQuarter, isThisYear, endOfYesterday, differenceInDays, differenceInMonths, differenceInYears, toDate } from 'date-fns';\nimport { ChatService } from \"../chat.service\";\nimport { SavedChat } from \"../types\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  imports: [CommonModule, ModalModule, UtilsModule]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string; value: SavedChat[]}[]>([]);\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public modalService = inject(ModalService);\n  public notificationsService = inject(NotificationsService);\n\n  ngOnInit(): void {\n    this.subscription.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.userOverride$),\n        tap(_ => {\n          this.onListSavedChat();\n          this.chatService.listSavedChat();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      this.chatService.savedChats$.subscribe(\n        (savedChats: SavedChat[]) => {\n          this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));\n        }\n      )\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.setSavedChatId(savedChat.id);\n    this.chatService.generateChatId(savedChat.id);\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.generateAuditEvent('saved-chat.load', {}, savedChat.id);\n    this.chatService.listSavedChat();\n    this.load.emit(savedChat);\n  }\n\n  onRename(event: Event, savedChat: SavedChat) {\n    event.stopPropagation();\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    const model: PromptOptions = {\n      title: 'Rename saved discussion',\n      message: `Please enter a new name for the discussion \"${savedChat.title}\".`,\n      buttons: [\n          new ModalButton({result: ModalResult.Cancel}),\n          new ModalButton({result: ModalResult.OK, text: \"Rename\", primary: true})\n      ],\n      output: savedChat.title,\n      validators: [Validators.required]\n    };\n\n    this.modalService.prompt(model).then(res => {\n      if(res === ModalResult.OK) {\n          this.subscription.add(\n            this.chatService.updateSavedChat(savedChat.id, model.output)\n              .pipe(\n                tap(() => {\n                  this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully renamed to \"${model.output}\".`);\n                  this.chatService.listSavedChat();\n                }),\n                catchError((error) => {\n                  console.error('Error occurred while updating the saved chat:', error);\n                  this.notificationsService.error(`Error occurred while updating the saved discussion \"${savedChat.title}\"`);\n                  return throwError(() => error);\n                })\n              ).subscribe()\n          );\n          this.chatService.generateAuditEvent('saved-chat.rename', {'text': model.output}, savedChat.id);\n      }\n    });\n  }\n\n  onDelete(event: Event, savedChat: SavedChat) {\n    event.stopPropagation();\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.modalService\n      .confirm({\n          title: \"Delete saved discussion\",\n          message: `You are about to delete the discussion \"${savedChat.title}\". Do you want to continue?`,\n          buttons: [\n              new ModalButton({result: ModalResult.Cancel}),\n              new ModalButton({result: ModalResult.OK, text: \"Confirm\", primary: true})\n          ],\n          confirmType: ConfirmType.Warning\n      }).then(res => {\n          if(res === ModalResult.OK) {\n            this.subscription.add(\n              this.chatService.deleteSavedChat([savedChat.id])\n                .pipe(\n                  tap(() => {\n                    this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully deleted.`);\n                    this.delete.emit(savedChat);\n                    this.chatService.listSavedChat();\n                  }),\n                  catchError((error) => {\n                    console.error('Error occurred while deleting the saved chat:', error);\n                    this.notificationsService.error(`Error occurred while deleting the saved discussion \"${savedChat.title}\"`);\n                    return throwError(() => error);\n                  })\n                ).subscribe()\n            );\n            this.chatService.generateAuditEvent('saved-chat.delete', {}, savedChat.id);\n          }\n      });\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string; value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n          const groupKey = this._getTimeKey(toDate(parseISO(savedChat.modifiedUTC))); // Must convert the UTC date to local date before passing to _getTimeKey\n\n          if (!groupedSavedChats.has(groupKey)) {\n            groupedSavedChats.set(groupKey, []);\n          }\n\n          groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n\n  private _getTimeKey(date: Date): string {\n    if (isToday(date)) {\n        return 'Today';\n    } else if (isYesterday(date)) {\n        return 'Yesterday';\n    } else if (isThisWeek(date)) {\n        return 'This week';\n    } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n      return 'Last week';\n    } else if (isThisMonth(date)) {\n        return 'This month';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n      return 'Last month';\n    } else if (isThisQuarter(date)) {\n        return 'This quarter';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return 'Last quarter';\n    } else if (isThisYear(date)) {\n        return 'This year';\n    } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return 'Last year';\n    } else {\n        return format(date, 'yyyy');\n    }\n  }\n\n}\n","<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\"\n      (click)=\"onLoad(savedChat)\"\n      class=\"saved-chat p-2\"\n      [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n      [class.active]=\"chatService.savedChatId === savedChat.id\">\n      <span class=\"title me-1\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n        (click)=\"onRename($event, savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n        (click)=\"onDelete($event, savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n"]}
|
|
175
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { Validators } from '@angular/forms';\nimport { Subscription, catchError, filter, switchMap, tap, throwError, BehaviorSubject } from \"rxjs\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { ConfirmType, ModalButton, ModalResult, ModalService, PromptOptions, ModalModule } from \"@sinequa/core/modal\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { format, parseISO, isToday, isYesterday, isThisWeek, isThisMonth, isThisQuarter, isThisYear, endOfYesterday, differenceInDays, differenceInMonths, differenceInYears, toDate } from 'date-fns';\nimport { ChatService } from \"../chat.service\";\nimport { SavedChat } from \"../types\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  imports: [CommonModule, ModalModule, UtilsModule]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string; value: SavedChat[]}[]>([]);\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public modalService = inject(ModalService);\n  public notificationsService = inject(NotificationsService);\n\n  ngOnInit(): void {\n    this.subscription.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.userOverride$),\n        filter(isOverridden => isOverridden !== undefined),\n        tap(_ => {\n          this.onListSavedChat();\n          this.chatService.listSavedChat();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      this.chatService.savedChats$.subscribe(\n        (savedChats: SavedChat[]) => {\n          this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));\n        }\n      )\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.setSavedChatId(savedChat.id);\n    this.chatService.generateChatId(savedChat.id);\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.generateAuditEvent('saved-chat.load', {}, savedChat.id);\n    this.chatService.listSavedChat();\n    this.load.emit(savedChat);\n  }\n\n  onRename(event: Event, savedChat: SavedChat) {\n    event.stopPropagation();\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    const model: PromptOptions = {\n      title: 'Rename saved discussion',\n      message: `Please enter a new name for the discussion \"${savedChat.title}\".`,\n      buttons: [\n          new ModalButton({result: ModalResult.Cancel}),\n          new ModalButton({result: ModalResult.OK, text: \"Rename\", primary: true})\n      ],\n      output: savedChat.title,\n      validators: [Validators.required]\n    };\n\n    this.modalService.prompt(model).then(res => {\n      if(res === ModalResult.OK) {\n          this.subscription.add(\n            this.chatService.updateSavedChat(savedChat.id, model.output)\n              .pipe(\n                tap(() => {\n                  this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully renamed to \"${model.output}\".`);\n                  this.chatService.listSavedChat();\n                }),\n                catchError((error) => {\n                  console.error('Error occurred while updating the saved chat:', error);\n                  this.notificationsService.error(`Error occurred while updating the saved discussion \"${savedChat.title}\"`);\n                  return throwError(() => error);\n                })\n              ).subscribe()\n          );\n          this.chatService.generateAuditEvent('saved-chat.rename', {'text': model.output}, savedChat.id);\n      }\n    });\n  }\n\n  onDelete(event: Event, savedChat: SavedChat) {\n    event.stopPropagation();\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.modalService\n      .confirm({\n          title: \"Delete saved discussion\",\n          message: `You are about to delete the discussion \"${savedChat.title}\". Do you want to continue?`,\n          buttons: [\n              new ModalButton({result: ModalResult.Cancel}),\n              new ModalButton({result: ModalResult.OK, text: \"Confirm\", primary: true})\n          ],\n          confirmType: ConfirmType.Warning\n      }).then(res => {\n          if(res === ModalResult.OK) {\n            this.subscription.add(\n              this.chatService.deleteSavedChat([savedChat.id])\n                .pipe(\n                  tap(() => {\n                    this.notificationsService.success(`The saved discussion \"${savedChat.title}\" has been successfully deleted.`);\n                    this.delete.emit(savedChat);\n                    this.chatService.listSavedChat();\n                  }),\n                  catchError((error) => {\n                    console.error('Error occurred while deleting the saved chat:', error);\n                    this.notificationsService.error(`Error occurred while deleting the saved discussion \"${savedChat.title}\"`);\n                    return throwError(() => error);\n                  })\n                ).subscribe()\n            );\n            this.chatService.generateAuditEvent('saved-chat.delete', {}, savedChat.id);\n          }\n      });\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string; value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n          const groupKey = this._getTimeKey(toDate(parseISO(savedChat.modifiedUTC))); // Must convert the UTC date to local date before passing to _getTimeKey\n\n          if (!groupedSavedChats.has(groupKey)) {\n            groupedSavedChats.set(groupKey, []);\n          }\n\n          groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n\n  private _getTimeKey(date: Date): string {\n    if (isToday(date)) {\n        return 'Today';\n    } else if (isYesterday(date)) {\n        return 'Yesterday';\n    } else if (isThisWeek(date)) {\n        return 'This week';\n    } else if (differenceInDays(endOfYesterday(), date) <= 7) {\n      return 'Last week';\n    } else if (isThisMonth(date)) {\n        return 'This month';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 1) {\n      return 'Last month';\n    } else if (isThisQuarter(date)) {\n        return 'This quarter';\n    } else if (differenceInMonths(endOfYesterday(), date) <= 3) {\n        return 'Last quarter';\n    } else if (isThisYear(date)) {\n        return 'This year';\n    } else if (differenceInYears(endOfYesterday(), date) === 1) {\n        return 'Last year';\n    } else {\n        return format(date, 'yyyy');\n    }\n  }\n\n}\n","<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\"\n      (click)=\"onLoad(savedChat)\"\n      class=\"saved-chat p-2\"\n      [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n      [class.active]=\"chatService.savedChatId === savedChat.id\">\n      <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'Rename'\"\n        (click)=\"onRename($event, savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'Delete'\"\n        (click)=\"onDelete($event, savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n"]}
|
|
@@ -44,11 +44,11 @@ export class TokenProgressBarComponent {
|
|
|
44
44
|
}
|
|
45
45
|
}
|
|
46
46
|
TokenProgressBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: TokenProgressBarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
47
|
-
TokenProgressBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: TokenProgressBarComponent, isStandalone: true, selector: "sq-token-progress-bar", inputs: { instanceId: "instanceId" }, ngImport: i0, template: "<div class=\"bars-container d-flex flex-row gap-2 p-2 me-4\" *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined) || (config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\">\n <div *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"userTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + userPercentage + '%, #0040BF 0)'\">\n </div>\n <div *ngIf=\"(config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"chatTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + chatPercentage + '%, #0040BF 0)'\">\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.bars-container{background-color:var(--ast-primary-bg, #f2f8fe);border-bottom-left-radius:1rem;border-bottom-right-radius:1rem}.token-progress-bar{width:1.5rem;height:1.5rem;border-radius:50%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
|
|
47
|
+
TokenProgressBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: TokenProgressBarComponent, isStandalone: true, selector: "sq-token-progress-bar", inputs: { instanceId: "instanceId" }, ngImport: i0, template: "<div class=\"bars-container d-flex flex-row gap-2 p-2 me-4\" *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined) || (config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\">\n <div *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"userTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + userPercentage + '%, #0040BF 0)'\">\n </div>\n <div *ngIf=\"(config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"chatTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + chatPercentage + '%, #0040BF 0)'\">\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.bars-container{background-color:var(--ast-primary-bg, #f2f8fe);border-bottom-left-radius:1rem;border-bottom-right-radius:1rem}.token-progress-bar{width:1.5rem;height:1.5rem;border-radius:50%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
|
|
48
48
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: TokenProgressBarComponent, decorators: [{
|
|
49
49
|
type: Component,
|
|
50
|
-
args: [{ selector: 'sq-token-progress-bar', standalone: true, imports: [CommonModule, UtilsModule], template: "<div class=\"bars-container d-flex flex-row gap-2 p-2 me-4\" *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined) || (config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\">\n <div *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"userTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + userPercentage + '%, #0040BF 0)'\">\n </div>\n <div *ngIf=\"(config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"chatTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + chatPercentage + '%, #0040BF 0)'\">\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.bars-container{background-color:var(--ast-primary-bg, #f2f8fe);border-bottom-left-radius:1rem;border-bottom-right-radius:1rem}.token-progress-bar{width:1.5rem;height:1.5rem;border-radius:50%}\n"] }]
|
|
50
|
+
args: [{ selector: 'sq-token-progress-bar', standalone: true, imports: [CommonModule, UtilsModule], template: "<div class=\"bars-container d-flex flex-row gap-2 p-2 me-4\" *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined) || (config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\">\n <div *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"userTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + userPercentage + '%, #0040BF 0)'\">\n </div>\n <div *ngIf=\"(config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"chatTitle\"\n [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + chatPercentage + '%, #0040BF 0)'\">\n </div>\n</div>\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-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.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-uploaded-doc-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;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}.bars-container{background-color:var(--ast-primary-bg, #f2f8fe);border-bottom-left-radius:1rem;border-bottom-right-radius:1rem}.token-progress-bar{width:1.5rem;height:1.5rem;border-radius:50%}\n"] }]
|
|
51
51
|
}], propDecorators: { instanceId: [{
|
|
52
52
|
type: Input
|
|
53
53
|
}] } });
|
|
54
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
54
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token-progress-bar.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/token-progress-bar/token-progress-bar.component.ts","../../../../../projects/assistant/chat/token-progress-bar/token-progress-bar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAqB,MAAM,EAAE,MAAM,eAAe,CAAC;AAC5E,OAAO,EAAE,YAAY,EAAE,MAAM,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAC5D,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAE/C,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AAErE,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;;;;AASxD,MAAM,OAAO,yBAAyB;IAPtC;QAaE,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;QAU3B,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;KAoDhE;IAlDC,QAAQ;QACN,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,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,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,EAC7C,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,EAC5B,GAAG,CAAC,CAAC,CAAC,EAAE;YACN,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,KAAM,CAAC;YACvD,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAC/B,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,sBAAsB,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9E,CAAC;IAED,uBAAuB;QACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,WAAW,CAAC,qBAAqB,CAAC,SAAS,CAC9C,CAAC,IAAsC,EAAE,EAAE;YACzC,IAAI,IAAI,EAAE;gBACR,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC;gBACtC,IAAI,CAAC,SAAS,GAAG,2BAA2B,IAAI,CAAC,cAAc,yBAAyB,IAAI,CAAC,aAAa,EAAE,CAAC;aAC9G;QACH,CAAC,CACF,CACF,CAAC;IACJ,CAAC;IAED,uBAAuB;QACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,WAAW,CAAC,qBAAqB,CAAC,SAAS,CAC9C,CAAC,IAAkC,EAAE,EAAE;YACrC,IAAI,IAAI,EAAE;gBACR,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC;gBACtC,IAAI,CAAC,SAAS,GAAG,+BAA+B,IAAI,CAAC,cAAc,YAAY,CAAC;aACjF;QACH,CAAC,CACF,CACF,CAAC;IACJ,CAAC;;sHAnEU,yBAAyB;0GAAzB,yBAAyB,uHChBtC,48BAQA,mlDDMY,YAAY,kIAAE,WAAW;2FAExB,yBAAyB;kBAPrC,SAAS;+BACE,uBAAuB,cAGrB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,CAAC;8BAI3B,UAAU;sBAAlB,KAAK","sourcesContent":["import { Component, Input, OnDestroy, OnInit, inject } from '@angular/core';\nimport { Subscription, filter, switchMap, tap } from 'rxjs';\nimport { CommonModule } from '@angular/common';\nimport { ChatConfig, TokenConsumption, UserTokenConsumption } from '../types';\nimport { LoginService } from '@sinequa/core/login';\nimport { InstanceManagerService } from '../instance-manager.service';\nimport { ChatService } from '../chat.service';\nimport { UtilsModule } from '@sinequa/components/utils';\n\n@Component({\n  selector: 'sq-token-progress-bar',\n  templateUrl: './token-progress-bar.component.html',\n  styleUrls: ['./token-progress-bar.component.scss'],\n  standalone: true,\n  imports: [CommonModule, UtilsModule]\n})\nexport class TokenProgressBarComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  chatService: ChatService;\n  config: ChatConfig;\n  subscription = new Subscription();\n\n  // User token consumption progress bar\n  userPercentage?: number;\n  userTitle?: string;\n\n  // Current chat token consumption progress bar\n  chatPercentage?: number;\n  chatTitle?: string;\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n\n  ngOnInit() {\n    this.subscription.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          this.config = this.chatService.assistantConfig$.value!;\n          this.onUserTokensConsumption();\n          this.onChatTokensConsumption();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onUserTokensConsumption() {\n    this.subscription.add(\n      this.chatService.userTokenConsumption$.subscribe(\n        (data: UserTokenConsumption | undefined) => {\n          if (data) {\n            this.userPercentage = data.percentage;\n            this.userTitle = `Max generation allowed: ${this.userPercentage}% consumed. Resets at ${data.nextResetDate}`;\n          }\n        }\n      )\n    );\n  }\n\n  onChatTokensConsumption() {\n    this.subscription.add(\n      this.chatService.chatTokenConsumption$.subscribe(\n        (data: TokenConsumption | undefined) => {\n          if (data) {\n            this.chatPercentage = data.percentage;\n            this.chatTitle = `Max length of conversation: ${this.chatPercentage}% reached.`;\n          }\n        }\n      )\n    );\n  }\n\n}\n","<div class=\"bars-container d-flex flex-row gap-2 p-2 me-4\" *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined) || (config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\">\n    <div *ngIf=\"(config?.globalSettings?.displayUserQuotaConsumption && userPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"userTitle\"\n        [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + userPercentage + '%, #0040BF 0)'\">\n    </div>\n    <div *ngIf=\"(config?.globalSettings?.displayChatTokensConsumption && chatPercentage !== undefined)\" class=\"token-progress-bar\" [sqTooltip]=\"chatTitle\"\n        [style.background]=\"'radial-gradient(closest-side, var(--ast-primary-bg, #F8F8F8) 70%, transparent 75% 100%), conic-gradient(#FF854A ' + chatPercentage + '%, #0040BF 0)'\">\n    </div>\n</div>\n"]}
|