@sinequa/assistant 3.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/chat/chat-message/chat-message.component.d.ts +157 -0
- package/chat/chat-reference/chat-reference.component.d.ts +118 -0
- package/chat/chat-settings-v3/chat-settings-v3.component.d.ts +41 -0
- package/chat/chat.component.d.ts +1112 -0
- package/chat/chat.service.d.ts +1046 -0
- package/chat/format-icon/format-icon.component.d.ts +10 -0
- package/chat/format-icon/icons.d.ts +5 -0
- package/chat/index.d.ts +5 -0
- package/chat/initials-avatar/initials-avatar.component.d.ts +35 -0
- package/chat/instance-manager.service.d.ts +28 -0
- package/chat/messages/de.d.ts +4 -0
- package/chat/messages/en.d.ts +4 -0
- package/chat/messages/fr.d.ts +4 -0
- package/chat/messages/index.d.ts +4 -0
- package/chat/public-api.d.ts +11 -0
- package/chat/rest-chat.service.d.ts +28 -0
- package/chat/saved-chats/saved-chats.component.d.ts +37 -0
- package/chat/styles/assistant.scss +61 -0
- package/chat/styles/references.scss +23 -0
- package/chat/types.d.ts +1241 -0
- package/chat/websocket-chat.service.d.ts +97 -0
- package/esm2020/chat/chat-message/chat-message.component.mjs +181 -0
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +40 -0
- package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +103 -0
- package/esm2020/chat/chat.component.mjs +369 -0
- package/esm2020/chat/chat.service.mjs +185 -0
- package/esm2020/chat/format-icon/format-icon.component.mjs +23 -0
- package/esm2020/chat/format-icon/icons.mjs +138 -0
- package/esm2020/chat/initials-avatar/initials-avatar.component.mjs +60 -0
- package/esm2020/chat/instance-manager.service.mjs +46 -0
- package/esm2020/chat/messages/de.mjs +4 -0
- package/esm2020/chat/messages/en.mjs +4 -0
- package/esm2020/chat/messages/fr.mjs +4 -0
- package/esm2020/chat/messages/index.mjs +9 -0
- package/esm2020/chat/public-api.mjs +12 -0
- package/esm2020/chat/rest-chat.service.mjs +164 -0
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +132 -0
- package/esm2020/chat/sinequa-assistant-chat.mjs +5 -0
- package/esm2020/chat/types.mjs +85 -0
- package/esm2020/chat/websocket-chat.service.mjs +430 -0
- package/esm2020/public-api.mjs +3 -0
- package/esm2020/sinequa-assistant.mjs +5 -0
- package/fesm2015/sinequa-assistant-chat.mjs +1925 -0
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -0
- package/fesm2015/sinequa-assistant.mjs +9 -0
- package/fesm2015/sinequa-assistant.mjs.map +1 -0
- package/fesm2020/sinequa-assistant-chat.mjs +1911 -0
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -0
- package/fesm2020/sinequa-assistant.mjs +9 -0
- package/fesm2020/sinequa-assistant.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/package.json +46 -0
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { AuthenticationService } from "@sinequa/core/login";
|
|
2
|
+
import { ConnectionOptions, SignalRWebService } from "@sinequa/core/web-services";
|
|
3
|
+
import { HubConnection } from "@microsoft/signalr";
|
|
4
|
+
import { ChatMessage, ChatResponse, GllmFunction, GllmModelDescription, MessageHandler, SavedChatHistory } from "./types";
|
|
5
|
+
import { ChatService } from "./chat.service";
|
|
6
|
+
import { Observable, Subject } from "rxjs";
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export declare class WebSocketChatService extends ChatService {
|
|
9
|
+
connection: HubConnection | undefined;
|
|
10
|
+
connectionBuilt$: Subject<void>;
|
|
11
|
+
connectionStarted$: Subject<void>;
|
|
12
|
+
private messageHandlers;
|
|
13
|
+
private actionMap;
|
|
14
|
+
private content;
|
|
15
|
+
private executionTime;
|
|
16
|
+
private attachments;
|
|
17
|
+
signalRService: SignalRWebService;
|
|
18
|
+
authenticationService: AuthenticationService;
|
|
19
|
+
constructor();
|
|
20
|
+
/**
|
|
21
|
+
* Initialize the chat process after the login is complete.
|
|
22
|
+
* It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.
|
|
23
|
+
*
|
|
24
|
+
* @returns An Observable<boolean> indicating the success of the initialization process.
|
|
25
|
+
*/
|
|
26
|
+
init(): Observable<boolean>;
|
|
27
|
+
/**
|
|
28
|
+
* Define the assistant endpoint to use for the websocket requests
|
|
29
|
+
* It can be overridden by the app config
|
|
30
|
+
*/
|
|
31
|
+
getRequestsUrl(): void;
|
|
32
|
+
listModels(): Observable<GllmModelDescription[] | undefined>;
|
|
33
|
+
listFunctions(): Observable<GllmFunction[] | undefined>;
|
|
34
|
+
fetch(messages: ChatMessage[], query?: import("@sinequa/core/app-utils").Query): Observable<ChatResponse>;
|
|
35
|
+
listSavedChat(): void;
|
|
36
|
+
getSavedChat(id: string): Observable<SavedChatHistory | undefined>;
|
|
37
|
+
deleteSavedChat(ids: string[]): Observable<number>;
|
|
38
|
+
/**
|
|
39
|
+
* Initialize out-of-the-box handlers
|
|
40
|
+
* It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with a single message or a result
|
|
41
|
+
*/
|
|
42
|
+
initMessageHandlers(): void;
|
|
43
|
+
/**
|
|
44
|
+
* Override and register the entire messageHandlers map by merging the provided map with the default one
|
|
45
|
+
* @param messageHandlers
|
|
46
|
+
*/
|
|
47
|
+
overrideMessageHandlers<T>(messageHandlers: Map<string, MessageHandler<T>>): void;
|
|
48
|
+
/**
|
|
49
|
+
* Add a listener for a specific event.
|
|
50
|
+
* If a listener for this same event already exists, it will be overridden.
|
|
51
|
+
* If the listener has "isChatGlobalHandler" set to true, it will be registered to the hub connection.
|
|
52
|
+
* @param eventName Name of the event to register a listener for
|
|
53
|
+
* @param eventHandler The handler to be called when the event is received
|
|
54
|
+
*/
|
|
55
|
+
addMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>): void;
|
|
56
|
+
/**
|
|
57
|
+
* Dynamically register a listener for a specific event.
|
|
58
|
+
* If a listener for this event already exists, it will be overridden.
|
|
59
|
+
* @param eventName Name of the event to register a listener for
|
|
60
|
+
* @param eventHandler The handler to be called when the event is received
|
|
61
|
+
*/
|
|
62
|
+
protected registerMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>): void;
|
|
63
|
+
/**
|
|
64
|
+
* Remove a listener for a specific event from the messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.
|
|
65
|
+
* @param eventName Name of the event to remove the listener for
|
|
66
|
+
*/
|
|
67
|
+
removeMessageHandler(eventName: string): void;
|
|
68
|
+
/**
|
|
69
|
+
* Unsubscribe from receiving messages for a specific event from the SignalR hub.
|
|
70
|
+
* ALL its related listeners will be removed from hub connection
|
|
71
|
+
* This is needed to prevent accumulating old listeners when overriding the entire messageHandlers map
|
|
72
|
+
* @param eventName Name of the event
|
|
73
|
+
*/
|
|
74
|
+
protected unsubscribeMessageHandler(eventName: string): void;
|
|
75
|
+
/**
|
|
76
|
+
* Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class
|
|
77
|
+
* @param options The options for the connection. It overrides the default options
|
|
78
|
+
* @param logLevel Define the log level displayed in the console
|
|
79
|
+
* @returns Promise that resolves when the connection is built
|
|
80
|
+
*/
|
|
81
|
+
buildConnection(options?: ConnectionOptions): Promise<void>;
|
|
82
|
+
/**
|
|
83
|
+
* Start the connection
|
|
84
|
+
* @returns Promise that resolves when the connection is started
|
|
85
|
+
*/
|
|
86
|
+
startConnection(): Promise<void>;
|
|
87
|
+
/**
|
|
88
|
+
* Stop the connection
|
|
89
|
+
* @returns Promise that resolves when the connection is stopped
|
|
90
|
+
*/
|
|
91
|
+
stopConnection(): Promise<void>;
|
|
92
|
+
private getTransports;
|
|
93
|
+
private getLogLevel;
|
|
94
|
+
get defaultOptions(): ConnectionOptions;
|
|
95
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<WebSocketChatService, never>;
|
|
96
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<WebSocketChatService>;
|
|
97
|
+
}
|
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from "@angular/core";
|
|
2
|
+
import { unified } from "unified";
|
|
3
|
+
import remarkParse from "remark-parse";
|
|
4
|
+
import { visit, CONTINUE, EXIT } from "unist-util-visit";
|
|
5
|
+
import { UtilsModule } from "@sinequa/components/utils";
|
|
6
|
+
import remarkGfm from "remark-gfm";
|
|
7
|
+
import { CommonModule } from "@angular/common";
|
|
8
|
+
import { CollapseModule } from "@sinequa/components/collapse";
|
|
9
|
+
import { RemarkModule } from "ngx-remark";
|
|
10
|
+
import { InitialsAvatarComponent } from "../initials-avatar/initials-avatar.component";
|
|
11
|
+
import { ChatReferenceComponent } from "../chat-reference/chat-reference.component";
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "@sinequa/components/search";
|
|
14
|
+
import * as i2 from "@sinequa/components/utils";
|
|
15
|
+
import * as i3 from "@sinequa/core/web-services";
|
|
16
|
+
import * as i4 from "@angular/common";
|
|
17
|
+
import * as i5 from "@sinequa/components/collapse";
|
|
18
|
+
import * as i6 from "ngx-remark";
|
|
19
|
+
export class ChatMessageComponent {
|
|
20
|
+
constructor(searchService, ui, principalService, cdr, el) {
|
|
21
|
+
this.searchService = searchService;
|
|
22
|
+
this.ui = ui;
|
|
23
|
+
this.principalService = principalService;
|
|
24
|
+
this.cdr = cdr;
|
|
25
|
+
this.el = el;
|
|
26
|
+
this.canEdit = false;
|
|
27
|
+
this.canRegenerate = false;
|
|
28
|
+
this.canCopy = false;
|
|
29
|
+
this.referenceClicked = new EventEmitter();
|
|
30
|
+
this.edit = new EventEmitter();
|
|
31
|
+
this.regenerate = new EventEmitter();
|
|
32
|
+
this.openPreview = new EventEmitter();
|
|
33
|
+
this.references = [];
|
|
34
|
+
this.referenceMap = new Map();
|
|
35
|
+
this.showReferences = true;
|
|
36
|
+
/**
|
|
37
|
+
* This Unified plugin looks a text nodes and replaces any reference in the
|
|
38
|
+
* form [1], [2.3], etc. with custom nodes of type "chat-reference".
|
|
39
|
+
*/
|
|
40
|
+
this.referencePlugin = (tree) => {
|
|
41
|
+
const references = new Set();
|
|
42
|
+
// Visit all text nodes
|
|
43
|
+
visit(tree, "text", (node, index, parent) => {
|
|
44
|
+
let text = node.value;
|
|
45
|
+
text = this.reformatReferences(text);
|
|
46
|
+
const matches = this.getReferenceMatches(text);
|
|
47
|
+
// Quit if no references were found
|
|
48
|
+
if (matches.length === 0) {
|
|
49
|
+
return CONTINUE;
|
|
50
|
+
}
|
|
51
|
+
const nodes = [];
|
|
52
|
+
for (let match of matches) {
|
|
53
|
+
const refId = match[1].trim();
|
|
54
|
+
const [ref] = refId.split(".");
|
|
55
|
+
// We find a valid reference in the text
|
|
56
|
+
if (!isNaN(+ref)) {
|
|
57
|
+
references.add(+ref); // Add it to the set of used references
|
|
58
|
+
// If needed, insert a text node before the reference
|
|
59
|
+
const current = nodes.at(-1) ?? { end: 0 };
|
|
60
|
+
if (match.index > current.end) {
|
|
61
|
+
nodes.push({ type: "text", value: text.substring(current.end, match.index), end: match.index });
|
|
62
|
+
}
|
|
63
|
+
// Add a custom reference node
|
|
64
|
+
nodes.push({ type: "chat-reference", refId, end: match.index + match[0].length });
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
// Quit if no references were found
|
|
68
|
+
if (nodes.length === 0) {
|
|
69
|
+
return CONTINUE;
|
|
70
|
+
}
|
|
71
|
+
if (nodes.at(-1).end < text.length) {
|
|
72
|
+
nodes.push({ type: "text", value: text.substring(nodes.at(-1).end, text.length), end: text.length });
|
|
73
|
+
}
|
|
74
|
+
// Delete the current text node from the parent and replace it with the new nodes
|
|
75
|
+
parent.children.splice(index, 1, ...nodes);
|
|
76
|
+
return index + nodes.length; // Visit the next node after the inserted ones
|
|
77
|
+
});
|
|
78
|
+
if (references.size > 0) {
|
|
79
|
+
this.references = Array.from(references.values())
|
|
80
|
+
.sort((a, b) => a - b)
|
|
81
|
+
.map(r => '' + r);
|
|
82
|
+
this.cdr.detectChanges();
|
|
83
|
+
}
|
|
84
|
+
return tree;
|
|
85
|
+
};
|
|
86
|
+
this.placeholderPlugin = (tree) => {
|
|
87
|
+
visit(tree, "text", (node, index, parent) => {
|
|
88
|
+
parent.children.push({ type: "streaming-placeholder" });
|
|
89
|
+
return EXIT;
|
|
90
|
+
}, true);
|
|
91
|
+
return tree;
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
get name() {
|
|
95
|
+
return !this.principalService.principal ? ''
|
|
96
|
+
: this.principalService.principal['fullName'] || this.principalService.principal.name;
|
|
97
|
+
}
|
|
98
|
+
ngOnChanges(changes) {
|
|
99
|
+
if (changes.streaming) {
|
|
100
|
+
this.collapseProgress = !this.streaming;
|
|
101
|
+
}
|
|
102
|
+
if (changes.message && this.message.role === "assistant") {
|
|
103
|
+
this.references = [];
|
|
104
|
+
this.referenceMap.clear();
|
|
105
|
+
for (let m of this.conversation) {
|
|
106
|
+
if (m.additionalProperties.$attachment) {
|
|
107
|
+
for (const attachment of m.additionalProperties.$attachment) {
|
|
108
|
+
this.referenceMap.set('' + attachment.contextId, { ...attachment, $partId: attachment.parts?.[0].partId });
|
|
109
|
+
for (let i = 0; i < attachment.parts.length; i++) {
|
|
110
|
+
const refId = `${attachment.contextId}.${attachment.parts[i].partId}`;
|
|
111
|
+
this.referenceMap.set(refId, { ...attachment, $partId: attachment.parts[i].partId });
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
this.processor = unified()
|
|
117
|
+
.use(remarkParse)
|
|
118
|
+
.use(remarkGfm)
|
|
119
|
+
.use(() => this.referencePlugin);
|
|
120
|
+
if (this.streaming) {
|
|
121
|
+
this.processor = this.processor.use(() => this.placeholderPlugin);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
ngAfterViewInit() {
|
|
126
|
+
Prism?.highlightAllUnder?.(this.el.nativeElement);
|
|
127
|
+
}
|
|
128
|
+
openDocument(record) {
|
|
129
|
+
const url = record.url1 || record.originalUrl;
|
|
130
|
+
if (url) {
|
|
131
|
+
// Open the URL in a new tab
|
|
132
|
+
window.open(url, '_blank');
|
|
133
|
+
}
|
|
134
|
+
this.referenceClicked.emit(record);
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Reformat [ids: 12.2, 42.5] to [12.2][42.5]
|
|
138
|
+
*/
|
|
139
|
+
reformatReferences(content) {
|
|
140
|
+
return content.replace(/\[(?:ids?:?\s*)?(?:documents?:?\s*)?(\s*(?:,?\s*\d+(?:\.\d+)?(?:\.part)?\s*)+)\]/g, (str, match) => `[${match.replace(/\.part/g, "").split(',').join("] [")}]`);
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Match all references in a given message
|
|
144
|
+
*/
|
|
145
|
+
getReferenceMatches(content) {
|
|
146
|
+
return Array.from(content.matchAll(/\[(\s*\d+(?:\.\d+)?\s*)\]/g));
|
|
147
|
+
}
|
|
148
|
+
copyToClipboard(text) {
|
|
149
|
+
this.ui.copyToClipboard(text);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
ChatMessageComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatMessageComponent, deps: [{ token: i1.SearchService }, { token: i2.UIService }, { token: i3.PrincipalWebService }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
153
|
+
ChatMessageComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatMessageComponent, isStandalone: true, selector: "sq-chat-message", inputs: { message: "message", conversation: "conversation", assistantIcon: "assistantIcon", streaming: "streaming", canEdit: "canEdit", canRegenerate: "canRegenerate", canCopy: "canCopy" }, outputs: { referenceClicked: "referenceClicked", edit: "edit", regenerate: "regenerate", openPreview: "openPreview" }, usesOnChanges: true, ngImport: i0, template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\" [title]=\"step.time ?? ''\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n \n </div>\n\n <!-- Edit / Regenerate floating actions -->\n <div class=\"sq-chat-message-action\">\n <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n (click)=\"edit.emit(message)\">\n <i class=\"fas fa-edit\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n (click)=\"copyToClipboard(message.content)\">\n <i class=\"far fa-clipboard\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n (click)=\"regenerate.emit(message)\">\n <i class=\"fas fa-sync-alt\"></i>\n </button>\n </div>\n\n <ng-template #tooltipTpl let-ref>\n <sq-chat-reference class=\"expanded\"\n [attachment]=\"ref\"\n [reference]=\"ref.contextId\"\n [partId]=\"ref.$partId\"\n (openPreview)=\"openPreview.emit($event)\"\n (openDocument)=\"openDocument($event)\">\n </sq-chat-reference>\n </ng-template>\n\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-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}:host{display:flex}:host:not(:hover) .sq-chat-message-action{display:none}.message-content{padding:var(--ast-message-padding, var(--ast-size-3, .75rem));border-radius:var(--ast-message-border-radius, var(--ast-size-4, 1rem));display:inline-block;max-width:100%}.message-content .references{margin-top:var(--ast-size-5, 1.25rem)}.message-content .references ul{border-left:.2rem solid var(--ast-secondary-color, #FF732E);padding-left:var(--ast-size-5, 1.25rem);margin-top:var(--ast-size-2, .5rem)}.message-content .references .references-title{font-weight:var(--font-weight-bold, 500)}.message-content .references .references-collapse{cursor:pointer;margin-left:var(--ast-size-2, .5rem)}.message-content ::ng-deep p:last-child{margin-bottom:0}.message-content ::ng-deep .placeholder-glow .placeholder{animation-duration:.4s;width:12px;height:var(--ast-size-4, 1rem);vertical-align:text-bottom}.message-content ::ng-deep img{max-width:100%}.message-content ::ng-deep table{display:block;border:1px solid #ccc;border-collapse:collapse;margin:0;padding:0;min-width:100%;overflow-x:auto;table-layout:fixed}.message-content ::ng-deep table tr{background-color:#f8f8f8;border:1px solid #ddd;padding:.35em}.message-content ::ng-deep table th,.message-content ::ng-deep table td{padding:.625em;text-align:center}.message-content ::ng-deep table th{font-size:.85em;letter-spacing:.1em;text-transform:uppercase}.message-assistant .message-content{background:var(--ast-secondary-bg, #FFF8F1)}.message-user .message-content{background:var(--ast-primary-bg, #f2f8fe);font-weight:var(--ast-user-font-weight, var(--font-weight-bold, 500))}.sq-chat-message-action{position:absolute;right:calc(0rem - var(--ast-size-2, .5rem));top:var(--ast-size-2, .5rem);display:flex;flex-direction:column}.message-icon{margin-top:var(--ast-size-3, .75rem);margin-right:var(--ast-size-4, 1rem)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.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"] }, { kind: "ngmodule", type: CollapseModule }, { kind: "component", type: i5.Collapse, selector: "sq-collapse", inputs: ["collapsed"] }, { kind: "ngmodule", type: RemarkModule }, { kind: "component", type: i6.RemarkComponent, selector: "remark", inputs: ["markdown", "processor", "debug"] }, { kind: "directive", type: i6.RemarkTemplateDirective, selector: "[remarkTemplate]", inputs: ["remarkTemplate"] }, { kind: "component", type: InitialsAvatarComponent, selector: "sq-initials-avatar", inputs: ["fullName", "size"] }, { kind: "component", type: ChatReferenceComponent, selector: "sq-chat-reference", inputs: ["reference", "attachment", "partId"], outputs: ["openDocument", "openPreview"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
154
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatMessageComponent, decorators: [{
|
|
155
|
+
type: Component,
|
|
156
|
+
args: [{ selector: "sq-chat-message", changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, UtilsModule, CollapseModule, RemarkModule,
|
|
157
|
+
InitialsAvatarComponent, ChatReferenceComponent], template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\" [title]=\"step.time ?? ''\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n \n </div>\n\n <!-- Edit / Regenerate floating actions -->\n <div class=\"sq-chat-message-action\">\n <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n (click)=\"edit.emit(message)\">\n <i class=\"fas fa-edit\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n (click)=\"copyToClipboard(message.content)\">\n <i class=\"far fa-clipboard\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n (click)=\"regenerate.emit(message)\">\n <i class=\"fas fa-sync-alt\"></i>\n </button>\n </div>\n\n <ng-template #tooltipTpl let-ref>\n <sq-chat-reference class=\"expanded\"\n [attachment]=\"ref\"\n [reference]=\"ref.contextId\"\n [partId]=\"ref.$partId\"\n (openPreview)=\"openPreview.emit($event)\"\n (openDocument)=\"openDocument($event)\">\n </sq-chat-reference>\n </ng-template>\n\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-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}:host{display:flex}:host:not(:hover) .sq-chat-message-action{display:none}.message-content{padding:var(--ast-message-padding, var(--ast-size-3, .75rem));border-radius:var(--ast-message-border-radius, var(--ast-size-4, 1rem));display:inline-block;max-width:100%}.message-content .references{margin-top:var(--ast-size-5, 1.25rem)}.message-content .references ul{border-left:.2rem solid var(--ast-secondary-color, #FF732E);padding-left:var(--ast-size-5, 1.25rem);margin-top:var(--ast-size-2, .5rem)}.message-content .references .references-title{font-weight:var(--font-weight-bold, 500)}.message-content .references .references-collapse{cursor:pointer;margin-left:var(--ast-size-2, .5rem)}.message-content ::ng-deep p:last-child{margin-bottom:0}.message-content ::ng-deep .placeholder-glow .placeholder{animation-duration:.4s;width:12px;height:var(--ast-size-4, 1rem);vertical-align:text-bottom}.message-content ::ng-deep img{max-width:100%}.message-content ::ng-deep table{display:block;border:1px solid #ccc;border-collapse:collapse;margin:0;padding:0;min-width:100%;overflow-x:auto;table-layout:fixed}.message-content ::ng-deep table tr{background-color:#f8f8f8;border:1px solid #ddd;padding:.35em}.message-content ::ng-deep table th,.message-content ::ng-deep table td{padding:.625em;text-align:center}.message-content ::ng-deep table th{font-size:.85em;letter-spacing:.1em;text-transform:uppercase}.message-assistant .message-content{background:var(--ast-secondary-bg, #FFF8F1)}.message-user .message-content{background:var(--ast-primary-bg, #f2f8fe);font-weight:var(--ast-user-font-weight, var(--font-weight-bold, 500))}.sq-chat-message-action{position:absolute;right:calc(0rem - var(--ast-size-2, .5rem));top:var(--ast-size-2, .5rem);display:flex;flex-direction:column}.message-icon{margin-top:var(--ast-size-3, .75rem);margin-right:var(--ast-size-4, 1rem)}\n"] }]
|
|
158
|
+
}], ctorParameters: function () { return [{ type: i1.SearchService }, { type: i2.UIService }, { type: i3.PrincipalWebService }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { message: [{
|
|
159
|
+
type: Input
|
|
160
|
+
}], conversation: [{
|
|
161
|
+
type: Input
|
|
162
|
+
}], assistantIcon: [{
|
|
163
|
+
type: Input
|
|
164
|
+
}], streaming: [{
|
|
165
|
+
type: Input
|
|
166
|
+
}], canEdit: [{
|
|
167
|
+
type: Input
|
|
168
|
+
}], canRegenerate: [{
|
|
169
|
+
type: Input
|
|
170
|
+
}], canCopy: [{
|
|
171
|
+
type: Input
|
|
172
|
+
}], referenceClicked: [{
|
|
173
|
+
type: Output
|
|
174
|
+
}], edit: [{
|
|
175
|
+
type: Output
|
|
176
|
+
}], regenerate: [{
|
|
177
|
+
type: Output
|
|
178
|
+
}], openPreview: [{
|
|
179
|
+
type: Output
|
|
180
|
+
}] } });
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-message.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/chat-message/chat-message.component.ts","../../../../../projects/assistant/chat/chat-message/chat-message.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnChanges, Output, SimpleChanges, ChangeDetectorRef, AfterViewInit, ElementRef } from \"@angular/core\";\nimport { PrincipalWebService, Record } from \"@sinequa/core/web-services\";\nimport { SearchService } from \"@sinequa/components/search\";\nimport { ChatContextAttachment, ChatMessage } from \"../types\";\n\nimport { unified, Processor } from \"unified\";\nimport remarkParse from \"remark-parse\";\nimport { visit, CONTINUE, EXIT } from \"unist-util-visit\";\nimport { Text, Parent, Content } from \"mdast\";\nimport { Node } from \"unist\";\nimport { UIService, UtilsModule } from \"@sinequa/components/utils\";\nimport remarkGfm from \"remark-gfm\";\nimport { CommonModule } from \"@angular/common\";\nimport { CollapseModule } from \"@sinequa/components/collapse\";\nimport { RemarkModule } from \"ngx-remark\";\nimport { InitialsAvatarComponent } from \"../initials-avatar/initials-avatar.component\";\nimport { ChatReferenceComponent } from \"../chat-reference/chat-reference.component\";\n\ndeclare module Prism {\n  function highlightAllUnder(el: HTMLElement): void;\n}\n\n@Component({\n  selector: \"sq-chat-message\",\n  templateUrl: \"./chat-message.component.html\",\n  styleUrls: [\"./chat-message.component.scss\"],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, UtilsModule, CollapseModule, RemarkModule,\n    InitialsAvatarComponent, ChatReferenceComponent]\n})\nexport class ChatMessageComponent implements OnChanges, AfterViewInit {\n  @Input() message: ChatMessage;\n  @Input() conversation: ChatMessage[];\n  @Input() assistantIcon: string;\n  @Input() streaming: boolean;\n  @Input() canEdit: boolean = false;\n  @Input() canRegenerate: boolean = false;\n  @Input() canCopy: boolean = false;\n  @Output() referenceClicked = new EventEmitter<Record>();\n  @Output() edit = new EventEmitter<ChatMessage>();\n  @Output() regenerate = new EventEmitter<ChatMessage>();\n  @Output() openPreview = new EventEmitter<ChatContextAttachment>();\n\n  processor: Processor;\n\n  references: string[] = [];\n  referenceMap = new Map<string, ChatContextAttachment>();\n  showReferences: boolean = true;\n  collapseProgress: boolean;\n\n  get name(): string {\n    return !this.principalService.principal ? ''\n      : this.principalService.principal['fullName'] as string || this.principalService.principal.name;\n  }\n\n  constructor(\n    public searchService: SearchService,\n    public ui: UIService,\n    public principalService: PrincipalWebService,\n    public cdr: ChangeDetectorRef,\n    public el: ElementRef\n  ) { }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.streaming) {\n      this.collapseProgress = !this.streaming;\n    }\n\n    if (changes.message && this.message.role === \"assistant\") {\n      this.references = [];\n      this.referenceMap.clear();\n      for (let m of this.conversation) {\n        if (m.additionalProperties.$attachment) {\n          for (const attachment of m.additionalProperties.$attachment) {\n            this.referenceMap.set('' + attachment.contextId, { ...attachment, $partId: attachment.parts?.[0].partId });\n            for (let i = 0; i < attachment.parts.length; i++) {\n              const refId = `${attachment.contextId}.${attachment.parts[i].partId}`;\n              this.referenceMap.set(refId, { ...attachment, $partId: attachment.parts[i].partId });\n            }\n          }\n        }\n      }\n\n      this.processor = unified()\n        .use(remarkParse)\n        .use(remarkGfm)\n        .use(() => this.referencePlugin);\n\n      if (this.streaming) {\n        this.processor = this.processor.use(() => this.placeholderPlugin);\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    Prism?.highlightAllUnder?.(this.el.nativeElement);\n  }\n\n  openDocument(record: Record) {\n    const url = record.url1 || record.originalUrl;\n    if (url) {\n      // Open the URL in a new tab\n      window.open(url, '_blank');\n    }\n    this.referenceClicked.emit(record);\n  }\n\n  /**\n   * This Unified plugin looks a text nodes and replaces any reference in the\n   * form [1], [2.3], etc. with custom nodes of type \"chat-reference\".\n   */\n  referencePlugin = (tree: Node) => {\n\n    const references = new Set<number>();\n\n    // Visit all text nodes\n    visit(tree, \"text\", (node: Text, index: number, parent: Parent) => {\n      let text = node.value;\n\n      text = this.reformatReferences(text);\n      const matches = this.getReferenceMatches(text);\n\n      // Quit if no references were found\n      if (matches.length === 0) {\n        return CONTINUE;\n      }\n\n      const nodes: (Content & { end: number })[] = [];\n\n      for (let match of matches) {\n        const refId = match[1].trim();\n        const [ref] = refId.split(\".\");\n        // We find a valid reference in the text\n        if (!isNaN(+ref)) {\n          references.add(+ref); // Add it to the set of used references\n\n          // If needed, insert a text node before the reference\n          const current = nodes.at(-1) ?? { end: 0 };\n          if (match.index! > current.end) {\n            nodes.push({ type: \"text\", value: text.substring(current.end, match.index!), end: match.index! });\n          }\n\n          // Add a custom reference node\n          nodes.push({ type: \"chat-reference\", refId, end: match.index! + match[0].length } as any);\n        }\n      }\n\n      // Quit if no references were found\n      if (nodes.length === 0) {\n        return CONTINUE;\n      }\n\n      if (nodes.at(-1)!.end < text.length) {\n        nodes.push({ type: \"text\", value: text.substring(nodes.at(-1)!.end, text.length), end: text.length });\n      }\n\n      // Delete the current text node from the parent and replace it with the new nodes\n      parent.children.splice(index, 1, ...nodes);\n\n      return index + nodes.length; // Visit the next node after the inserted ones\n    });\n\n    if (references.size > 0) {\n      this.references = Array.from(references.values())\n        .sort((a, b) => a - b)\n        .map(r => '' + r);\n      this.cdr.detectChanges();\n    }\n\n    return tree;\n  }\n\n  placeholderPlugin = (tree: Node) => {\n\n    visit(tree, \"text\", (node: Text, index: number, parent: Parent) => {\n      parent.children.push({ type: \"streaming-placeholder\" } as any);\n      return EXIT;\n    }, true);\n\n    return tree;\n  }\n\n  /**\n   * Reformat [ids: 12.2, 42.5] to [12.2][42.5]\n   */\n  reformatReferences(content: string) {\n    return content.replace(/\\[(?:ids?:?\\s*)?(?:documents?:?\\s*)?(\\s*(?:,?\\s*\\d+(?:\\.\\d+)?(?:\\.part)?\\s*)+)\\]/g,\n      (str, match: string) => `[${match.replace(/\\.part/g, \"\").split(',').join(\"] [\")}]`\n    );\n  }\n\n  /**\n   * Match all references in a given message\n   */\n  getReferenceMatches(content: string) {\n    return Array.from(content.matchAll(/\\[(\\s*\\d+(?:\\.\\d+)?\\s*)\\]/g));\n  }\n\n  copyToClipboard(text: string) {\n    this.ui.copyToClipboard(text);\n  }\n}\n","<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n  <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n  <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n  <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n    <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n      View progress\n      <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n    </a>\n    <sq-collapse [collapsed]=\"collapseProgress\">\n      <ng-template>\n        <ul class=\"list-unstyled\">\n          <li *ngFor=\"let step of progress\" [title]=\"step.time ?? ''\">\n            <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n            <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n            <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n            <span *ngIf=\"step.content\">: {{step.content}}</span>\n          </li>\n        </ul>\n      </ng-template>\n    </sq-collapse>\n  </div>\n\n  <div class=\"message-content\" *ngIf=\"message.content\">\n\n    <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n      <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n        <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n          (click)=\"openDocument(attachment.record)\"\n          class=\"reference\"\n          [sqTooltip]=\"attachment\"\n          [sqTooltipTemplate]=\"tooltipTpl\"\n          [hoverableTooltip]=\"true\"\n          >{{ref.refId}}</a>\n\n        <ng-template #staticRefTpl>\n          <span class=\"reference\">{{ref.refId}}</span>\n        </ng-template>\n\n      </ng-template>\n\n      <ng-template remarkTemplate=\"streaming-placeholder\">\n        <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n          <span class=\"placeholder ms-1\"></span>\n        </span>\n      </ng-template>\n\n      <ng-template remarkTemplate=\"code\" let-node>\n        <div class=\"card mb-2\">\n          <div class=\"card-header d-flex justify-content-between align-items-center\">\n            <span>{{node.lang}}</span>\n            <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n          </div>\n          <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n        </div>\n      </ng-template>\n\n    </remark>\n\n    <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n    <!-- List of reference, if any -->\n    <div *ngIf=\"references?.length\" class=\"references\">\n      <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n      <sq-collapse [collapsed]=\"!showReferences\">\n        <ng-template>\n          <ul>\n            <ng-container *ngFor=\"let reference of references\">\n              <li *ngIf=\"referenceMap.get(reference) as attachment\"  class=\"text-truncate\">\n                <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n                  (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n              </li>\n            </ng-container>\n          </ul>\n        </ng-template>\n      </sq-collapse>\n    </div>\n  \n  </div>\n\n    <!-- Edit / Regenerate floating actions -->\n  <div class=\"sq-chat-message-action\">\n    <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n            (click)=\"edit.emit(message)\">\n      <i class=\"fas fa-edit\"></i>\n    </button>\n    <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n            (click)=\"copyToClipboard(message.content)\">\n      <i class=\"far fa-clipboard\"></i>\n    </button>\n    <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n            (click)=\"regenerate.emit(message)\">\n      <i class=\"fas fa-sync-alt\"></i>\n    </button>\n  </div>\n\n  <ng-template #tooltipTpl let-ref>\n    <sq-chat-reference class=\"expanded\"\n                        [attachment]=\"ref\"\n                        [reference]=\"ref.contextId\"\n                        [partId]=\"ref.$partId\"\n                        (openPreview)=\"openPreview.emit($event)\"\n                        (openDocument)=\"openDocument($event)\">\n      </sq-chat-reference>\n  </ng-template>\n\n</div>\n"]}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
3
|
+
import { UtilsModule } from '@sinequa/components/utils';
|
|
4
|
+
import { FormatIconComponent } from '../format-icon/format-icon.component';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/common";
|
|
7
|
+
import * as i2 from "@sinequa/components/utils";
|
|
8
|
+
export class ChatReferenceComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.openDocument = new EventEmitter();
|
|
11
|
+
this.openPreview = new EventEmitter();
|
|
12
|
+
}
|
|
13
|
+
get parts() {
|
|
14
|
+
if (!this.attachment)
|
|
15
|
+
return [];
|
|
16
|
+
return this.attachment.parts.filter(part => (!this.partId || part.partId === this.partId) && !!part.text);
|
|
17
|
+
}
|
|
18
|
+
expandAttachment() {
|
|
19
|
+
if (this.partId)
|
|
20
|
+
return;
|
|
21
|
+
this.attachment['$expanded'] = !this.attachment['$expanded'];
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
ChatReferenceComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatReferenceComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
25
|
+
ChatReferenceComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatReferenceComponent, isStandalone: true, selector: "sq-chat-reference", inputs: { reference: "reference", attachment: "attachment", partId: "partId" }, outputs: { openDocument: "openDocument", openPreview: "openPreview" }, ngImport: i0, template: "<div [class.reference-tooltip]=\"!!partId\">\n <div class=\"reference-data\" [class.expanded]=\"attachment['$expanded'] || !!partId\">\n <span class=\"reference me-1\">{{reference}}</span>\n <sq-format-icon [extension]=\"attachment.record.fileext\"></sq-format-icon>\n <a [id]=\"'attachment-'+attachment.recordId\" (click)=\"expandAttachment()\">\n {{attachment.record.title}}\n </a>\n <i class=\"fas fa-eye\" (click)=\"openPreview.emit(attachment)\" [sqTooltip]=\"!partId ? 'Preview document' : ''\"></i>\n <i class=\"fas fa-arrow-up-right-from-square\" *ngIf=\"attachment.record.url1 || attachment.record.originalUrl\"\n (click)=\"openDocument.emit(attachment.record)\" [sqTooltip]=\"!partId ? 'Open document' : ''\"></i>\n </div>\n <div class=\"reference-passages\" *ngIf=\"!!partId || (attachment['$expanded']) && parts.length\">\n <div class=\"reference-passage\" *ngFor=\"let part of parts\">\n <span class=\"reference me-1\">{{reference}}.{{part.partId}}</span>\n <span [innerHTML]=\"part.text\"></span>\n </div>\n </div>\n</div>", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{display:block}:host.expanded,:host:hover{background-color:#fff}.reference-data{display:flex;flex-direction:row;align-items:baseline;padding:var(--ast-size-1, .25rem)}.reference-data a{color:var(--ast-secondary-color, #FF732E);flex-grow:1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;font-weight:var(--font-weight-bold, 500);cursor:pointer}.reference-data i{padding:var(--ast-size-1, .25rem);margin-left:var(--ast-size-1, .25rem);cursor:pointer;color:#000}.reference-data i.active{color:#fff;background-color:var(--ast-secondary-color, #FF732E)}.reference-data:not(.expanded) i{opacity:0}.reference-data:not(.expanded):hover i{opacity:1}.reference-passages{white-space:normal;font-style:italic;font-weight:400;padding:1rem 0;color:#000}.reference-passages .reference-passage{display:flex;align-items:baseline;padding-left:2.5rem;padding-right:1rem;word-wrap:break-word}.reference-passages .reference-passage+.reference-passage{padding-top:1rem}.reference-passages .reference-passage .reference{margin-right:var(--ast-size-2, .5rem)}sq-format-icon{margin-left:var(--ast-size-1, .25rem);margin-right:var(--ast-size-2, .5rem);color:var(--ast-secondary-color, #FF732E)}.reference-tooltip{max-width:600px!important;box-shadow:0 .5rem 1rem #00000026;padding:.5rem;font-size:.875rem}\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: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }, { kind: "component", type: FormatIconComponent, selector: "sq-format-icon", inputs: ["extension"] }] });
|
|
26
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatReferenceComponent, decorators: [{
|
|
27
|
+
type: Component,
|
|
28
|
+
args: [{ selector: 'sq-chat-reference', standalone: true, imports: [CommonModule, UtilsModule, FormatIconComponent], template: "<div [class.reference-tooltip]=\"!!partId\">\n <div class=\"reference-data\" [class.expanded]=\"attachment['$expanded'] || !!partId\">\n <span class=\"reference me-1\">{{reference}}</span>\n <sq-format-icon [extension]=\"attachment.record.fileext\"></sq-format-icon>\n <a [id]=\"'attachment-'+attachment.recordId\" (click)=\"expandAttachment()\">\n {{attachment.record.title}}\n </a>\n <i class=\"fas fa-eye\" (click)=\"openPreview.emit(attachment)\" [sqTooltip]=\"!partId ? 'Preview document' : ''\"></i>\n <i class=\"fas fa-arrow-up-right-from-square\" *ngIf=\"attachment.record.url1 || attachment.record.originalUrl\"\n (click)=\"openDocument.emit(attachment.record)\" [sqTooltip]=\"!partId ? 'Open document' : ''\"></i>\n </div>\n <div class=\"reference-passages\" *ngIf=\"!!partId || (attachment['$expanded']) && parts.length\">\n <div class=\"reference-passage\" *ngFor=\"let part of parts\">\n <span class=\"reference me-1\">{{reference}}.{{part.partId}}</span>\n <span [innerHTML]=\"part.text\"></span>\n </div>\n </div>\n</div>", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{display:block}:host.expanded,:host:hover{background-color:#fff}.reference-data{display:flex;flex-direction:row;align-items:baseline;padding:var(--ast-size-1, .25rem)}.reference-data a{color:var(--ast-secondary-color, #FF732E);flex-grow:1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;font-weight:var(--font-weight-bold, 500);cursor:pointer}.reference-data i{padding:var(--ast-size-1, .25rem);margin-left:var(--ast-size-1, .25rem);cursor:pointer;color:#000}.reference-data i.active{color:#fff;background-color:var(--ast-secondary-color, #FF732E)}.reference-data:not(.expanded) i{opacity:0}.reference-data:not(.expanded):hover i{opacity:1}.reference-passages{white-space:normal;font-style:italic;font-weight:400;padding:1rem 0;color:#000}.reference-passages .reference-passage{display:flex;align-items:baseline;padding-left:2.5rem;padding-right:1rem;word-wrap:break-word}.reference-passages .reference-passage+.reference-passage{padding-top:1rem}.reference-passages .reference-passage .reference{margin-right:var(--ast-size-2, .5rem)}sq-format-icon{margin-left:var(--ast-size-1, .25rem);margin-right:var(--ast-size-2, .5rem);color:var(--ast-secondary-color, #FF732E)}.reference-tooltip{max-width:600px!important;box-shadow:0 .5rem 1rem #00000026;padding:.5rem;font-size:.875rem}\n"] }]
|
|
29
|
+
}], propDecorators: { reference: [{
|
|
30
|
+
type: Input
|
|
31
|
+
}], attachment: [{
|
|
32
|
+
type: Input
|
|
33
|
+
}], partId: [{
|
|
34
|
+
type: Input
|
|
35
|
+
}], openDocument: [{
|
|
36
|
+
type: Output
|
|
37
|
+
}], openPreview: [{
|
|
38
|
+
type: Output
|
|
39
|
+
}] } });
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1yZWZlcmVuY2UuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXNzaXN0YW50L2NoYXQvY2hhdC1yZWZlcmVuY2UvY2hhdC1yZWZlcmVuY2UuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXNzaXN0YW50L2NoYXQvY2hhdC1yZWZlcmVuY2UvY2hhdC1yZWZlcmVuY2UuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdkUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBRXhELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHNDQUFzQyxDQUFDOzs7O0FBVTNFLE1BQU0sT0FBTyxzQkFBc0I7SUFQbkM7UUFhWSxpQkFBWSxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFDMUMsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBeUIsQ0FBQztLQVduRTtJQVRDLElBQUksS0FBSztRQUNQLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVTtZQUFFLE9BQU8sRUFBRSxDQUFDO1FBQ2hDLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLElBQUksSUFBSSxDQUFDLE1BQU0sS0FBSyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM1RyxDQUFDO0lBRUQsZ0JBQWdCO1FBQ2QsSUFBSSxJQUFJLENBQUMsTUFBTTtZQUFFLE9BQU87UUFDeEIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDL0QsQ0FBQzs7bUhBakJVLHNCQUFzQjt1R0FBdEIsc0JBQXNCLG9PQ2RuQyw4bkNBaUJNLDA3RURMTSxZQUFZLCtQQUFFLFdBQVcsdVBBQUUsbUJBQW1COzJGQUU3QyxzQkFBc0I7a0JBUGxDLFNBQVM7K0JBQ0UsbUJBQW1CLGNBR2pCLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxXQUFXLEVBQUUsbUJBQW1CLENBQUM7OEJBSWhELFNBQVM7c0JBQWpCLEtBQUs7Z0JBQ0csVUFBVTtzQkFBbEIsS0FBSztnQkFDRyxNQUFNO3NCQUFkLEtBQUs7Z0JBRUksWUFBWTtzQkFBckIsTUFBTTtnQkFDRyxXQUFXO3NCQUFwQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBVdGlsc01vZHVsZSB9IGZyb20gJ0BzaW5lcXVhL2NvbXBvbmVudHMvdXRpbHMnO1xuaW1wb3J0IHsgUmVjb3JkIH0gZnJvbSAnQHNpbmVxdWEvY29yZS93ZWItc2VydmljZXMnO1xuaW1wb3J0IHsgRm9ybWF0SWNvbkNvbXBvbmVudCB9IGZyb20gJy4uL2Zvcm1hdC1pY29uL2Zvcm1hdC1pY29uLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDaGF0Q29udGV4dEF0dGFjaG1lbnQsIERvY3VtZW50UGFydCB9IGZyb20gJy4uL3R5cGVzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc3EtY2hhdC1yZWZlcmVuY2UnLFxuICB0ZW1wbGF0ZVVybDogJy4vY2hhdC1yZWZlcmVuY2UuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9jaGF0LXJlZmVyZW5jZS5jb21wb25lbnQuc2NzcyddLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBVdGlsc01vZHVsZSwgRm9ybWF0SWNvbkNvbXBvbmVudF1cbn0pXG5leHBvcnQgY2xhc3MgQ2hhdFJlZmVyZW5jZUNvbXBvbmVudCB7XG5cbiAgQElucHV0KCkgcmVmZXJlbmNlOiBzdHJpbmc7XG4gIEBJbnB1dCgpIGF0dGFjaG1lbnQ6IENoYXRDb250ZXh0QXR0YWNobWVudDtcbiAgQElucHV0KCkgcGFydElkPzogbnVtYmVyO1xuXG4gIEBPdXRwdXQoKSBvcGVuRG9jdW1lbnQgPSBuZXcgRXZlbnRFbWl0dGVyPFJlY29yZD4oKTtcbiAgQE91dHB1dCgpIG9wZW5QcmV2aWV3ID0gbmV3IEV2ZW50RW1pdHRlcjxDaGF0Q29udGV4dEF0dGFjaG1lbnQ+KCk7XG5cbiAgZ2V0IHBhcnRzKCk6IERvY3VtZW50UGFydFtdIHtcbiAgICBpZiAoIXRoaXMuYXR0YWNobWVudCkgcmV0dXJuIFtdO1xuICAgIHJldHVybiB0aGlzLmF0dGFjaG1lbnQucGFydHMuZmlsdGVyKHBhcnQgPT4gKCF0aGlzLnBhcnRJZCB8fCBwYXJ0LnBhcnRJZCA9PT0gdGhpcy5wYXJ0SWQpICYmICEhcGFydC50ZXh0KTtcbiAgfVxuXG4gIGV4cGFuZEF0dGFjaG1lbnQoKSB7XG4gICAgaWYgKHRoaXMucGFydElkKSByZXR1cm47XG4gICAgdGhpcy5hdHRhY2htZW50WyckZXhwYW5kZWQnXSA9ICF0aGlzLmF0dGFjaG1lbnRbJyRleHBhbmRlZCddO1xuICB9XG59XG4iLCI8ZGl2IFtjbGFzcy5yZWZlcmVuY2UtdG9vbHRpcF09XCIhIXBhcnRJZFwiPlxuICAgIDxkaXYgY2xhc3M9XCJyZWZlcmVuY2UtZGF0YVwiIFtjbGFzcy5leHBhbmRlZF09XCJhdHRhY2htZW50WyckZXhwYW5kZWQnXSB8fCAhIXBhcnRJZFwiPlxuICAgICAgICA8c3BhbiBjbGFzcz1cInJlZmVyZW5jZSBtZS0xXCI+e3tyZWZlcmVuY2V9fTwvc3Bhbj5cbiAgICAgICAgPHNxLWZvcm1hdC1pY29uIFtleHRlbnNpb25dPVwiYXR0YWNobWVudC5yZWNvcmQuZmlsZWV4dFwiPjwvc3EtZm9ybWF0LWljb24+XG4gICAgICAgIDxhIFtpZF09XCInYXR0YWNobWVudC0nK2F0dGFjaG1lbnQucmVjb3JkSWRcIiAoY2xpY2spPVwiZXhwYW5kQXR0YWNobWVudCgpXCI+XG4gICAgICAgICAgICB7e2F0dGFjaG1lbnQucmVjb3JkLnRpdGxlfX1cbiAgICAgICAgPC9hPlxuICAgICAgICA8aSBjbGFzcz1cImZhcyBmYS1leWVcIiAoY2xpY2spPVwib3BlblByZXZpZXcuZW1pdChhdHRhY2htZW50KVwiIFtzcVRvb2x0aXBdPVwiIXBhcnRJZCA/ICdQcmV2aWV3IGRvY3VtZW50JyA6ICcnXCI+PC9pPlxuICAgICAgICA8aSBjbGFzcz1cImZhcyBmYS1hcnJvdy11cC1yaWdodC1mcm9tLXNxdWFyZVwiICpuZ0lmPVwiYXR0YWNobWVudC5yZWNvcmQudXJsMSB8fCBhdHRhY2htZW50LnJlY29yZC5vcmlnaW5hbFVybFwiXG4gICAgICAgICAgICAoY2xpY2spPVwib3BlbkRvY3VtZW50LmVtaXQoYXR0YWNobWVudC5yZWNvcmQpXCIgW3NxVG9vbHRpcF09XCIhcGFydElkID8gJ09wZW4gZG9jdW1lbnQnIDogJydcIj48L2k+XG4gICAgPC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cInJlZmVyZW5jZS1wYXNzYWdlc1wiICpuZ0lmPVwiISFwYXJ0SWQgfHwgKGF0dGFjaG1lbnRbJyRleHBhbmRlZCddKSAmJiBwYXJ0cy5sZW5ndGhcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInJlZmVyZW5jZS1wYXNzYWdlXCIgKm5nRm9yPVwibGV0IHBhcnQgb2YgcGFydHNcIj5cbiAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwicmVmZXJlbmNlIG1lLTFcIj57e3JlZmVyZW5jZX19Lnt7cGFydC5wYXJ0SWR9fTwvc3Bhbj5cbiAgICAgICAgICAgIDxzcGFuIFtpbm5lckhUTUxdPVwicGFydC50ZXh0XCI+PC9zcGFuPlxuICAgICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cbjwvZGl2PiJdfQ==
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output, inject } from "@angular/core";
|
|
2
|
+
import { Subscription, filter, switchMap, tap } from "rxjs";
|
|
3
|
+
import { InstanceManagerService } from "../instance-manager.service";
|
|
4
|
+
import { PrincipalWebService } from "@sinequa/core/web-services";
|
|
5
|
+
import { CommonModule } from "@angular/common";
|
|
6
|
+
import { FormsModule } from "@angular/forms";
|
|
7
|
+
import { LoginService } from "@sinequa/core/login";
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/common";
|
|
10
|
+
import * as i2 from "@angular/forms";
|
|
11
|
+
export class ChatSettingsV3Component {
|
|
12
|
+
constructor() {
|
|
13
|
+
this._update = new EventEmitter();
|
|
14
|
+
this._cancel = new EventEmitter();
|
|
15
|
+
this.subscription = new Subscription();
|
|
16
|
+
this.isAdmin = false;
|
|
17
|
+
this.loginService = inject(LoginService);
|
|
18
|
+
this.instanceManagerService = inject(InstanceManagerService);
|
|
19
|
+
this.principalService = inject(PrincipalWebService);
|
|
20
|
+
}
|
|
21
|
+
ngOnInit() {
|
|
22
|
+
this.subscription.add(this.loginService.events.pipe(filter(e => e.type === 'login-complete'), tap(_ => this.instantiateChatService()), switchMap(() => this.chatService.initConfig$), filter(initConfig => !!initConfig)).subscribe(_ => {
|
|
23
|
+
this.isAdmin = this.principalService.principal.isAdministrator;
|
|
24
|
+
// Init config with a copy of the original chat config, so that it won't be modified by the user until he clicks on save
|
|
25
|
+
this.config = JSON.parse(JSON.stringify(this.chatService.chatConfig$.value));
|
|
26
|
+
this.selectedModel = this.chatService.getModel(this.config.serviceSettings.service_id, this.config.serviceSettings.model_id);
|
|
27
|
+
this.updateFunctionsCheckboxes();
|
|
28
|
+
}));
|
|
29
|
+
}
|
|
30
|
+
ngOnDestroy() {
|
|
31
|
+
this.subscription.unsubscribe();
|
|
32
|
+
}
|
|
33
|
+
get hasPrompts() {
|
|
34
|
+
return this.isAdmin
|
|
35
|
+
|| !!this.config.uiSettings.systemPrompt
|
|
36
|
+
|| !!this.config.uiSettings.userPrompt;
|
|
37
|
+
}
|
|
38
|
+
get hasAdvancedParameters() {
|
|
39
|
+
return this.isAdmin
|
|
40
|
+
|| !!this.config.uiSettings.temperature
|
|
41
|
+
|| !!this.config.uiSettings.top_p
|
|
42
|
+
|| !!this.config.uiSettings.maxTokens;
|
|
43
|
+
}
|
|
44
|
+
get hasModel() {
|
|
45
|
+
return this.isAdmin
|
|
46
|
+
|| !!this.config.uiSettings.servicesModels
|
|
47
|
+
|| !!this.config.uiSettings.functions
|
|
48
|
+
|| !!this.config.uiSettings.debug
|
|
49
|
+
|| !!this.config.uiSettings.temperature
|
|
50
|
+
|| !!this.config.uiSettings.top_p
|
|
51
|
+
|| !!this.config.uiSettings.maxTokens;
|
|
52
|
+
}
|
|
53
|
+
instantiateChatService() {
|
|
54
|
+
this.chatService = this.instanceManagerService.getInstance(this.instanceId);
|
|
55
|
+
}
|
|
56
|
+
onChatModelChange(selectedModel) {
|
|
57
|
+
// Update properties based on the selected model
|
|
58
|
+
this.config.serviceSettings.service_id = selectedModel.serviceId;
|
|
59
|
+
this.config.serviceSettings.model_id = selectedModel.modelId;
|
|
60
|
+
}
|
|
61
|
+
toggleFunctionsSelection(name) {
|
|
62
|
+
if (this.config.functions.includes(name)) {
|
|
63
|
+
this.config.functions = this.config.functions.filter(item => item !== name);
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
this.config.functions.push(name);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
updateFunctionsCheckboxes() {
|
|
70
|
+
// Update the checkboxes based on config.functions
|
|
71
|
+
(this.chatService.functions || []).forEach(item => {
|
|
72
|
+
item['selected'] = this.config.functions.includes(item.functionName);
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Save the new chat config in the chat service and the user preferences
|
|
77
|
+
*/
|
|
78
|
+
save() {
|
|
79
|
+
this.chatService.updateChatConfig(this.config);
|
|
80
|
+
this._update.emit();
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Cancel the current changes
|
|
84
|
+
*/
|
|
85
|
+
cancel() {
|
|
86
|
+
this._cancel.emit();
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
ChatSettingsV3Component.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatSettingsV3Component, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
90
|
+
ChatSettingsV3Component.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatSettingsV3Component, isStandalone: true, selector: "sq-chat-settings-v3", inputs: { instanceId: "instanceId" }, outputs: { _update: "update", _cancel: "cancel" }, ngImport: i0, template: "<div class=\"sq-chat-settings\">\n <div class=\"settings-panel card-body small\" *ngIf=\"config\">\n\n <h5 *ngIf=\"hasModel\">Model</h5>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.servicesModels\">\n <label for=\"gllmModel\" class=\"form-label\">Model</label>\n <select class=\"form-select\" id=\"gllmModel\" [(ngModel)]=\"selectedModel\" (ngModelChange)=\"onChatModelChange($event)\">\n <option *ngFor=\"let model of chatService.models\" [ngValue]=\"model\">{{model.displayName}}</option>\n </select>\n </div>\n \n <div class=\"mb-4\" *ngIf=\"isAdmin || config.uiSettings.functions\">\n <label for=\"gllmFunctions\" class=\"form-label\">Functions</label>\n <div id=\"gllmFunctions\" *ngFor=\"let func of chatService.functions\" class=\"multi-option form-check form-switch\">\n <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" [id]=\"func.functionName\" [(ngModel)]=\"func.selected\"\n (ngModelChange)=\"toggleFunctionsSelection(func.functionName)\">\n <label class=\"form-check-label\" [for]=\"func.functionName\" [title]=\"func.description\">{{ func.functionName }}</label>\n </div>\n </div>\n \n <div class=\"form-check form-switch mb-2\" *ngIf=\"isAdmin || config.uiSettings.debug\">\n <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" id=\"debug\" [(ngModel)]=\"config.debug\">\n <label class=\"form-check-label\" for=\"debug\">Debug</label>\n </div>\n \n <details *ngIf=\"hasAdvancedParameters\">\n <summary>Advanced parameters</summary>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.temperature\">\n <label for=\"temperature\" class=\"form-label\">Temperature: {{config.serviceSettings.temperature}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"2\" step=\"0.1\" id=\"temperature\"\n [(ngModel)]=\"config.serviceSettings.temperature\">\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.top_p\">\n <label for=\"top-p\" class=\"form-label\">Top P: {{config.serviceSettings.top_p}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"1\" step=\"0.05\" id=\"top-p\"\n [(ngModel)]=\"config.serviceSettings.top_p\">\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.maxTokens\">\n <label for=\"max-tokens\" class=\"form-label\">Max generated tokens per answer:\n {{config.serviceSettings.maxTokens}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"1\" max=\"2048\" step=\"1\" id=\"max-tokens\"\n [(ngModel)]=\"config.serviceSettings.maxTokens\">\n </div>\n </details>\n \n <hr>\n \n <h5 *ngIf=\"hasPrompts\">Prompts</h5>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.systemPrompt\">\n <label for=\"initialSystemPrompt\" class=\"form-label\">System prompt (hidden)</label>\n <textarea class=\"form-control\" id=\"initialSystemPrompt\" [(ngModel)]=\"config.uiSettings.systemPrompt\"></textarea>\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.userPrompt\">\n <label for=\"initialUserPrompt\" class=\"form-label\">Initial user prompt</label>\n <textarea class=\"form-control\" id=\"initialUserPrompt\" [(ngModel)]=\"config.uiSettings.userPrompt\"></textarea>\n </div>\n \n </div>\n \n <div class=\"buttons-panel d-flex justify-content-end\">\n <button class=\"btn btn-light\" (click)=\"cancel()\">Cancel</button>\n <button class=\"btn btn-primary\" *ngIf=\"config\" (click)=\"save()\">Save</button>\n </div>\n \n</div>", styles: [":host{display:block;width:var(--ast-chat-settings-width, 100%);max-width:100%;height:100%;margin-left:auto;margin-right:auto;padding-top:var(--ast-chat-settings-padding-top, 0);padding-bottom:var(--ast-chat-settings-padding-bottom, 0)}.sq-chat-settings{display:flex;flex-direction:column;height:100%}.sq-chat-settings .settings-panel{flex-grow:1;overflow:auto}.sq-chat-settings .buttons-panel{padding-top:.5rem}\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: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.RangeValueAccessor, selector: "input[type=range][formControlName],input[type=range][formControl],input[type=range][ngModel]" }, { kind: "directive", type: i2.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i2.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
|
|
91
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatSettingsV3Component, decorators: [{
|
|
92
|
+
type: Component,
|
|
93
|
+
args: [{ selector: 'sq-chat-settings-v3', standalone: true, imports: [CommonModule, FormsModule], template: "<div class=\"sq-chat-settings\">\n <div class=\"settings-panel card-body small\" *ngIf=\"config\">\n\n <h5 *ngIf=\"hasModel\">Model</h5>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.servicesModels\">\n <label for=\"gllmModel\" class=\"form-label\">Model</label>\n <select class=\"form-select\" id=\"gllmModel\" [(ngModel)]=\"selectedModel\" (ngModelChange)=\"onChatModelChange($event)\">\n <option *ngFor=\"let model of chatService.models\" [ngValue]=\"model\">{{model.displayName}}</option>\n </select>\n </div>\n \n <div class=\"mb-4\" *ngIf=\"isAdmin || config.uiSettings.functions\">\n <label for=\"gllmFunctions\" class=\"form-label\">Functions</label>\n <div id=\"gllmFunctions\" *ngFor=\"let func of chatService.functions\" class=\"multi-option form-check form-switch\">\n <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" [id]=\"func.functionName\" [(ngModel)]=\"func.selected\"\n (ngModelChange)=\"toggleFunctionsSelection(func.functionName)\">\n <label class=\"form-check-label\" [for]=\"func.functionName\" [title]=\"func.description\">{{ func.functionName }}</label>\n </div>\n </div>\n \n <div class=\"form-check form-switch mb-2\" *ngIf=\"isAdmin || config.uiSettings.debug\">\n <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" id=\"debug\" [(ngModel)]=\"config.debug\">\n <label class=\"form-check-label\" for=\"debug\">Debug</label>\n </div>\n \n <details *ngIf=\"hasAdvancedParameters\">\n <summary>Advanced parameters</summary>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.temperature\">\n <label for=\"temperature\" class=\"form-label\">Temperature: {{config.serviceSettings.temperature}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"2\" step=\"0.1\" id=\"temperature\"\n [(ngModel)]=\"config.serviceSettings.temperature\">\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.top_p\">\n <label for=\"top-p\" class=\"form-label\">Top P: {{config.serviceSettings.top_p}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"1\" step=\"0.05\" id=\"top-p\"\n [(ngModel)]=\"config.serviceSettings.top_p\">\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.maxTokens\">\n <label for=\"max-tokens\" class=\"form-label\">Max generated tokens per answer:\n {{config.serviceSettings.maxTokens}}</label>\n <input type=\"range\" class=\"form-range form-range-sm\" min=\"1\" max=\"2048\" step=\"1\" id=\"max-tokens\"\n [(ngModel)]=\"config.serviceSettings.maxTokens\">\n </div>\n </details>\n \n <hr>\n \n <h5 *ngIf=\"hasPrompts\">Prompts</h5>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.systemPrompt\">\n <label for=\"initialSystemPrompt\" class=\"form-label\">System prompt (hidden)</label>\n <textarea class=\"form-control\" id=\"initialSystemPrompt\" [(ngModel)]=\"config.uiSettings.systemPrompt\"></textarea>\n </div>\n <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.userPrompt\">\n <label for=\"initialUserPrompt\" class=\"form-label\">Initial user prompt</label>\n <textarea class=\"form-control\" id=\"initialUserPrompt\" [(ngModel)]=\"config.uiSettings.userPrompt\"></textarea>\n </div>\n \n </div>\n \n <div class=\"buttons-panel d-flex justify-content-end\">\n <button class=\"btn btn-light\" (click)=\"cancel()\">Cancel</button>\n <button class=\"btn btn-primary\" *ngIf=\"config\" (click)=\"save()\">Save</button>\n </div>\n \n</div>", styles: [":host{display:block;width:var(--ast-chat-settings-width, 100%);max-width:100%;height:100%;margin-left:auto;margin-right:auto;padding-top:var(--ast-chat-settings-padding-top, 0);padding-bottom:var(--ast-chat-settings-padding-bottom, 0)}.sq-chat-settings{display:flex;flex-direction:column;height:100%}.sq-chat-settings .settings-panel{flex-grow:1;overflow:auto}.sq-chat-settings .buttons-panel{padding-top:.5rem}\n"] }]
|
|
94
|
+
}], propDecorators: { instanceId: [{
|
|
95
|
+
type: Input
|
|
96
|
+
}], _update: [{
|
|
97
|
+
type: Output,
|
|
98
|
+
args: ["update"]
|
|
99
|
+
}], _cancel: [{
|
|
100
|
+
type: Output,
|
|
101
|
+
args: ["cancel"]
|
|
102
|
+
}] } });
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-settings-v3.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/chat-settings-v3/chat-settings-v3.component.ts","../../../../../projects/assistant/chat/chat-settings-v3/chat-settings-v3.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { ChatService } from \"../chat.service\";\nimport { ChatConfig, GllmModelDescription } from \"../types\";\nimport { Subscription, filter, switchMap, tap } from \"rxjs\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\nimport { PrincipalWebService } from \"@sinequa/core/web-services\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { LoginService } from \"@sinequa/core/login\";\n\n@Component({\n  selector: 'sq-chat-settings-v3',\n  templateUrl: './chat-settings-v3.component.html',\n  styleUrls: [\"./chat-settings-v3.component.scss\"],\n  standalone: true,\n  imports: [CommonModule, FormsModule]\n})\nexport class ChatSettingsV3Component 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(\"update\") _update = new EventEmitter();\n  @Output(\"cancel\") _cancel = new EventEmitter();\n\n  chatService: ChatService;\n  config: ChatConfig;\n  subscription = new Subscription();\n  selectedModel: GllmModelDescription | undefined;\n  isAdmin = false;\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public principalService = inject(PrincipalWebService);\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.initConfig$),\n        filter(initConfig => !!initConfig)\n      ).subscribe(_ => {\n        this.isAdmin = this.principalService.principal!.isAdministrator;\n        // Init config with a copy of the original chat config, so that it won't be modified by the user until he clicks on save\n        this.config = JSON.parse(JSON.stringify(this.chatService.chatConfig$.value));\n        this.selectedModel = this.chatService.getModel(this.config.serviceSettings.service_id, this.config.serviceSettings.model_id);\n        this.updateFunctionsCheckboxes();\n      })\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  get hasPrompts(): boolean {\n    return this.isAdmin\n      || !!this.config.uiSettings.systemPrompt\n      || !!this.config.uiSettings.userPrompt;\n  }\n\n  get hasAdvancedParameters(): boolean {\n    return this.isAdmin\n      || !!this.config.uiSettings.temperature\n      || !!this.config.uiSettings.top_p\n      || !!this.config.uiSettings.maxTokens;\n  }\n\n  get hasModel(): boolean {\n    return this.isAdmin\n      || !!this.config.uiSettings.servicesModels\n      || !!this.config.uiSettings.functions\n      || !!this.config.uiSettings.debug\n      || !!this.config.uiSettings.temperature\n      || !!this.config.uiSettings.top_p\n      || !!this.config.uiSettings.maxTokens;\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onChatModelChange(selectedModel: GllmModelDescription) {\n    // Update properties based on the selected model\n    this.config.serviceSettings.service_id = selectedModel.serviceId;\n    this.config.serviceSettings.model_id = selectedModel.modelId;\n  }\n\n  toggleFunctionsSelection(name: string) {\n    if (this.config.functions.includes(name)) {\n      this.config.functions = this.config.functions.filter(item => item !== name);\n    } else {\n      this.config.functions.push(name);\n    }\n  }\n\n  private updateFunctionsCheckboxes(): void {\n    // Update the checkboxes based on config.functions\n    (this.chatService.functions || []).forEach(item => {\n      item['selected'] = this.config.functions.includes(item.functionName);\n    });\n  }\n\n  /**\n   * Save the new chat config in the chat service and the user preferences\n   */\n  save() {\n    this.chatService.updateChatConfig(this.config);\n    this._update.emit();\n  }\n\n  /**\n   * Cancel the current changes\n   */\n  cancel() {\n    this._cancel.emit();\n  }\n}\n","<div class=\"sq-chat-settings\">\n  <div class=\"settings-panel card-body small\" *ngIf=\"config\">\n\n    <h5 *ngIf=\"hasModel\">Model</h5>\n    <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.servicesModels\">\n      <label for=\"gllmModel\" class=\"form-label\">Model</label>\n      <select class=\"form-select\" id=\"gllmModel\" [(ngModel)]=\"selectedModel\" (ngModelChange)=\"onChatModelChange($event)\">\n        <option *ngFor=\"let model of chatService.models\" [ngValue]=\"model\">{{model.displayName}}</option>\n      </select>\n    </div>\n  \n    <div class=\"mb-4\" *ngIf=\"isAdmin || config.uiSettings.functions\">\n      <label for=\"gllmFunctions\" class=\"form-label\">Functions</label>\n      <div id=\"gllmFunctions\" *ngFor=\"let func of chatService.functions\" class=\"multi-option form-check form-switch\">\n        <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" [id]=\"func.functionName\" [(ngModel)]=\"func.selected\"\n          (ngModelChange)=\"toggleFunctionsSelection(func.functionName)\">\n        <label class=\"form-check-label\" [for]=\"func.functionName\" [title]=\"func.description\">{{ func.functionName }}</label>\n      </div>\n    </div>\n  \n    <div class=\"form-check form-switch mb-2\" *ngIf=\"isAdmin || config.uiSettings.debug\">\n      <input class=\"form-check-input\" type=\"checkbox\" role=\"switch\" id=\"debug\" [(ngModel)]=\"config.debug\">\n      <label class=\"form-check-label\" for=\"debug\">Debug</label>\n    </div>\n  \n    <details *ngIf=\"hasAdvancedParameters\">\n      <summary>Advanced parameters</summary>\n      <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.temperature\">\n        <label for=\"temperature\" class=\"form-label\">Temperature: {{config.serviceSettings.temperature}}</label>\n        <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"2\" step=\"0.1\" id=\"temperature\"\n          [(ngModel)]=\"config.serviceSettings.temperature\">\n      </div>\n      <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.top_p\">\n        <label for=\"top-p\" class=\"form-label\">Top P: {{config.serviceSettings.top_p}}</label>\n        <input type=\"range\" class=\"form-range form-range-sm\" min=\"0\" max=\"1\" step=\"0.05\" id=\"top-p\"\n          [(ngModel)]=\"config.serviceSettings.top_p\">\n      </div>\n      <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.maxTokens\">\n        <label for=\"max-tokens\" class=\"form-label\">Max generated tokens per answer:\n          {{config.serviceSettings.maxTokens}}</label>\n        <input type=\"range\" class=\"form-range form-range-sm\" min=\"1\" max=\"2048\" step=\"1\" id=\"max-tokens\"\n          [(ngModel)]=\"config.serviceSettings.maxTokens\">\n      </div>\n    </details>\n  \n    <hr>\n  \n    <h5 *ngIf=\"hasPrompts\">Prompts</h5>\n    <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.systemPrompt\">\n      <label for=\"initialSystemPrompt\" class=\"form-label\">System prompt (hidden)</label>\n      <textarea class=\"form-control\" id=\"initialSystemPrompt\" [(ngModel)]=\"config.uiSettings.systemPrompt\"></textarea>\n    </div>\n    <div class=\"mb-2\" *ngIf=\"isAdmin || config.uiSettings.userPrompt\">\n      <label for=\"initialUserPrompt\" class=\"form-label\">Initial user prompt</label>\n      <textarea class=\"form-control\" id=\"initialUserPrompt\" [(ngModel)]=\"config.uiSettings.userPrompt\"></textarea>\n    </div>\n  \n  </div>\n  \n  <div class=\"buttons-panel d-flex justify-content-end\">\n    <button class=\"btn btn-light\" (click)=\"cancel()\">Cancel</button>\n    <button class=\"btn btn-primary\" *ngIf=\"config\" (click)=\"save()\">Save</button>\n  </div>\n  \n</div>"]}
|