@promptbook/components 0.110.0 → 0.111.0-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/README.md +4 -0
- package/esm/index.es.js +2662 -509
- package/esm/index.es.js.map +1 -1
- package/esm/typings/src/_packages/utils.index.d.ts +12 -0
- package/esm/typings/src/book-2.0/agent-source/BookEditable.d.ts +41 -0
- package/esm/typings/src/book-2.0/agent-source/CreateAgentModelRequirementsOptions.d.ts +5 -0
- package/esm/typings/src/book-components/Chat/Chat/ImagePromptRenderer.d.ts +21 -0
- package/esm/typings/src/book-components/Chat/LlmChat/LlmChatProps.d.ts +5 -0
- package/esm/typings/src/book-components/Chat/LlmChat/defaults.d.ts +9 -0
- package/esm/typings/src/book-components/Chat/save/_common/ChatSaveFormatDefinition.d.ts +7 -1
- package/esm/typings/src/book-components/Chat/save/html/htmlSaveFormatDefinition.d.ts +6 -5
- package/esm/typings/src/book-components/Chat/save/index.d.ts +3 -3
- package/esm/typings/src/book-components/Chat/save/pdf/buildChatPdf.d.ts +11 -0
- package/esm/typings/src/book-components/Chat/save/pdf/pdfSaveFormatDefinition.d.ts +2 -2
- package/esm/typings/src/book-components/Chat/utils/parseImagePrompts.d.ts +42 -0
- package/esm/typings/src/book-components/Chat/utils/parseImagePrompts.test.d.ts +1 -0
- package/esm/typings/src/commitments/MEMORY/MEMORY.d.ts +67 -0
- package/esm/typings/src/commitments/MEMORY/MEMORY.test.d.ts +1 -0
- package/esm/typings/src/commitments/_common/toolRuntimeContext.d.ts +49 -0
- package/esm/typings/src/constants/streaming.d.ts +20 -0
- package/esm/typings/src/llm-providers/openai/utils/buildToolInvocationScript.d.ts +9 -0
- package/esm/typings/src/utils/clientVersion.d.ts +51 -0
- package/esm/typings/src/utils/knowledge/inlineKnowledgeSource.d.ts +13 -9
- package/esm/typings/src/utils/normalization/constructImageFilename.d.ts +18 -0
- package/esm/typings/src/utils/normalization/constructImageFilename.test.d.ts +1 -0
- package/esm/typings/src/version.d.ts +1 -1
- package/package.json +2 -1
- package/umd/index.umd.js +2667 -515
- package/umd/index.umd.js.map +1 -1
package/umd/index.umd.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
(function (global, factory) {
|
|
2
|
-
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('react/jsx-runtime'), require('react'), require('spacetrim'), require('crypto-js'), require('crypto-js/enc-hex'), require('path'), require('crypto'), require('moment'), require('react-dom'), require('@monaco-editor/react'), require('monaco-editor'), require('destroyable'), require('katex'), require('react-dom/client'), require('showdown'), require('rxjs'), require('waitasecond'), require('crypto-js/sha256'), require('mime-types'), require('papaparse'), require('colors'), require('@openai/agents'), require('bottleneck'), require('openai'), require('
|
|
3
|
-
typeof define === 'function' && define.amd ? define(['exports', 'react/jsx-runtime', 'react', 'spacetrim', 'crypto-js', 'crypto-js/enc-hex', 'path', 'crypto', 'moment', 'react-dom', '@monaco-editor/react', 'monaco-editor', 'destroyable', 'katex', 'react-dom/client', 'showdown', 'rxjs', 'waitasecond', 'crypto-js/sha256', 'mime-types', 'papaparse', 'colors', '@openai/agents', 'bottleneck', 'openai', '
|
|
4
|
-
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["promptbook-components"] = {}, global.jsxRuntime, global.react, global.spaceTrim$1, global.cryptoJs, global.hexEncoder, global.path, global.crypto$1, global.moment, global.reactDom, global.Editor, global.monacoEditor, global.destroyable, global.katex, global.client, global.showdown, global.rxjs, global.waitasecond, global.sha256, global.mimeTypes, global.papaparse, global.colors, global.agents, global.Bottleneck, global.OpenAI, global.
|
|
5
|
-
})(this, (function (exports, jsxRuntime, react, spaceTrim$1, cryptoJs, hexEncoder, path, crypto$1, moment, reactDom, Editor, monacoEditor, destroyable, katex, client, showdown, rxjs, waitasecond, sha256, mimeTypes, papaparse, colors, agents, Bottleneck, OpenAI,
|
|
2
|
+
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('react/jsx-runtime'), require('react'), require('spacetrim'), require('crypto-js'), require('crypto-js/enc-hex'), require('path'), require('crypto'), require('moment'), require('react-dom'), require('@monaco-editor/react'), require('monaco-editor'), require('destroyable'), require('katex'), require('react-dom/client'), require('showdown'), require('rxjs'), require('jspdf'), require('lucide-react'), require('waitasecond'), require('crypto-js/sha256'), require('mime-types'), require('papaparse'), require('colors'), require('@openai/agents'), require('bottleneck'), require('openai'), require('qrcode')) :
|
|
3
|
+
typeof define === 'function' && define.amd ? define(['exports', 'react/jsx-runtime', 'react', 'spacetrim', 'crypto-js', 'crypto-js/enc-hex', 'path', 'crypto', 'moment', 'react-dom', '@monaco-editor/react', 'monaco-editor', 'destroyable', 'katex', 'react-dom/client', 'showdown', 'rxjs', 'jspdf', 'lucide-react', 'waitasecond', 'crypto-js/sha256', 'mime-types', 'papaparse', 'colors', '@openai/agents', 'bottleneck', 'openai', 'qrcode'], factory) :
|
|
4
|
+
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["promptbook-components"] = {}, global.jsxRuntime, global.react, global.spaceTrim$1, global.cryptoJs, global.hexEncoder, global.path, global.crypto$1, global.moment, global.reactDom, global.Editor, global.monacoEditor, global.destroyable, global.katex, global.client, global.showdown, global.rxjs, global.jspdf, global.lucideReact, global.waitasecond, global.sha256, global.mimeTypes, global.papaparse, global.colors, global.agents, global.Bottleneck, global.OpenAI, global.QRCode));
|
|
5
|
+
})(this, (function (exports, jsxRuntime, react, spaceTrim$1, cryptoJs, hexEncoder, path, crypto$1, moment, reactDom, Editor, monacoEditor, destroyable, katex, client, showdown, rxjs, jspdf, lucideReact, waitasecond, sha256, mimeTypes, papaparse, colors, agents, Bottleneck, OpenAI, QRCode) { 'use strict';
|
|
6
6
|
|
|
7
7
|
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
|
|
8
8
|
|
|
@@ -31,7 +31,7 @@
|
|
|
31
31
|
* @generated
|
|
32
32
|
* @see https://github.com/webgptorg/promptbook
|
|
33
33
|
*/
|
|
34
|
-
const PROMPTBOOK_ENGINE_VERSION = '0.
|
|
34
|
+
const PROMPTBOOK_ENGINE_VERSION = '0.111.0-0';
|
|
35
35
|
/**
|
|
36
36
|
* TODO: string_promptbook_version should be constrained to the all versions of Promptbook engine
|
|
37
37
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
@@ -3430,6 +3430,36 @@
|
|
|
3430
3430
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
3431
3431
|
*/
|
|
3432
3432
|
|
|
3433
|
+
/**
|
|
3434
|
+
* HTTP header used by Promptbook clients to advertise their release version.
|
|
3435
|
+
*
|
|
3436
|
+
* @public exported from `@promptbook/utils`
|
|
3437
|
+
*/
|
|
3438
|
+
const CLIENT_VERSION_HEADER = 'x-promptbook-client-version';
|
|
3439
|
+
/**
|
|
3440
|
+
* The latest client (engine) version that the server expects.
|
|
3441
|
+
*
|
|
3442
|
+
* @public exported from `@promptbook/utils`
|
|
3443
|
+
*/
|
|
3444
|
+
const CLIENT_LATEST_VERSION = PROMPTBOOK_ENGINE_VERSION;
|
|
3445
|
+
/**
|
|
3446
|
+
* Creates a headers object that includes the client version header.
|
|
3447
|
+
*
|
|
3448
|
+
* @param headers - Optional base headers to clone.
|
|
3449
|
+
* @returns New headers object augmented with `CLIENT_VERSION_HEADER`.
|
|
3450
|
+
*
|
|
3451
|
+
* @public exported from `@promptbook/utils`
|
|
3452
|
+
*/
|
|
3453
|
+
function attachClientVersionHeader(headers) {
|
|
3454
|
+
return {
|
|
3455
|
+
...(headers !== null && headers !== void 0 ? headers : {}),
|
|
3456
|
+
[CLIENT_VERSION_HEADER]: CLIENT_LATEST_VERSION,
|
|
3457
|
+
};
|
|
3458
|
+
}
|
|
3459
|
+
/**
|
|
3460
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
3461
|
+
*/
|
|
3462
|
+
|
|
3433
3463
|
/**
|
|
3434
3464
|
* Detects if the code is running in a browser environment in main thread (Not in a web worker)
|
|
3435
3465
|
*
|
|
@@ -6645,28 +6675,14 @@
|
|
|
6645
6675
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
6646
6676
|
*/
|
|
6647
6677
|
|
|
6648
|
-
/**
|
|
6649
|
-
* @@@
|
|
6650
|
-
*
|
|
6651
|
-
* @private thing of inline knowledge
|
|
6652
|
-
*/
|
|
6678
|
+
/** @private The default base name for inline knowledge files when the content lacks identifying text */
|
|
6653
6679
|
const INLINE_KNOWLEDGE_BASE_NAME = 'inline-knowledge';
|
|
6654
|
-
/**
|
|
6655
|
-
* @@@
|
|
6656
|
-
*
|
|
6657
|
-
* @private thing of inline knowledge
|
|
6658
|
-
*/
|
|
6680
|
+
/** @private The default file extension used for inline knowledge uploads */
|
|
6659
6681
|
const INLINE_KNOWLEDGE_EXTENSION = '.txt';
|
|
6660
|
-
/**
|
|
6661
|
-
* @@@
|
|
6662
|
-
*
|
|
6663
|
-
* @private thing of inline knowledge
|
|
6664
|
-
*/
|
|
6682
|
+
/** @private Prefix that identifies base64 data URLs */
|
|
6665
6683
|
const DATA_URL_PREFIX = 'data:';
|
|
6666
6684
|
/**
|
|
6667
|
-
*
|
|
6668
|
-
*
|
|
6669
|
-
* @private thing of inline knowledge
|
|
6685
|
+
* @private Retrieves the first meaningful line from the inline content.
|
|
6670
6686
|
*/
|
|
6671
6687
|
function getFirstNonEmptyLine(content) {
|
|
6672
6688
|
const lines = content.split(/\r?\n/);
|
|
@@ -6679,9 +6695,7 @@
|
|
|
6679
6695
|
return null;
|
|
6680
6696
|
}
|
|
6681
6697
|
/**
|
|
6682
|
-
*
|
|
6683
|
-
*
|
|
6684
|
-
* @private thing of inline knowledge
|
|
6698
|
+
* @private Determines the base file name by normalizing the first non-empty line.
|
|
6685
6699
|
*/
|
|
6686
6700
|
function deriveBaseFilename(content) {
|
|
6687
6701
|
const firstLine = getFirstNonEmptyLine(content);
|
|
@@ -6692,22 +6706,18 @@
|
|
|
6692
6706
|
return normalized || INLINE_KNOWLEDGE_BASE_NAME;
|
|
6693
6707
|
}
|
|
6694
6708
|
/**
|
|
6695
|
-
*
|
|
6696
|
-
*
|
|
6697
|
-
* @private thing of inline knowledge
|
|
6709
|
+
* @private Converts inline knowledge into the internal metadata form used for uploads.
|
|
6698
6710
|
*/
|
|
6699
6711
|
function createInlineKnowledgeSourceFile(content) {
|
|
6700
6712
|
const trimmedContent = content.trim();
|
|
6701
6713
|
const baseName = deriveBaseFilename(trimmedContent);
|
|
6702
6714
|
const filename = `${baseName}${INLINE_KNOWLEDGE_EXTENSION}`;
|
|
6703
6715
|
const mimeType = 'text/plain';
|
|
6704
|
-
const
|
|
6705
|
-
const encodedFilename = encodeURIComponent(filename);
|
|
6706
|
-
const url = `${DATA_URL_PREFIX}${mimeType};name=${encodedFilename};charset=utf-8;base64,${base64}`;
|
|
6716
|
+
const buffer = Buffer.from(trimmedContent, 'utf-8');
|
|
6707
6717
|
return {
|
|
6708
6718
|
filename,
|
|
6709
6719
|
mimeType,
|
|
6710
|
-
|
|
6720
|
+
buffer,
|
|
6711
6721
|
};
|
|
6712
6722
|
}
|
|
6713
6723
|
/**
|
|
@@ -6718,10 +6728,18 @@
|
|
|
6718
6728
|
function isDataUrlKnowledgeSource(source) {
|
|
6719
6729
|
return typeof source === 'string' && source.startsWith(DATA_URL_PREFIX);
|
|
6720
6730
|
}
|
|
6731
|
+
/**
|
|
6732
|
+
* @private Converts a stored inline knowledge file into a data URL for backwards compatibility.
|
|
6733
|
+
*/
|
|
6734
|
+
function inlineKnowledgeSourceToDataUrl(source) {
|
|
6735
|
+
const base64 = source.buffer.toString('base64');
|
|
6736
|
+
const encodedFilename = encodeURIComponent(source.filename);
|
|
6737
|
+
return `${DATA_URL_PREFIX}${source.mimeType};name=${encodedFilename};charset=utf-8;base64,${base64}`;
|
|
6738
|
+
}
|
|
6721
6739
|
/**
|
|
6722
6740
|
* Parses a data URL-based knowledge source into its raw buffer, filename, and MIME type.
|
|
6723
6741
|
*
|
|
6724
|
-
* @private
|
|
6742
|
+
* @private utility of inline knowledge processing
|
|
6725
6743
|
*/
|
|
6726
6744
|
function parseDataUrlKnowledgeSource(source) {
|
|
6727
6745
|
if (!isDataUrlKnowledgeSource(source)) {
|
|
@@ -6854,6 +6872,7 @@
|
|
|
6854
6872
|
`);
|
|
6855
6873
|
}
|
|
6856
6874
|
applyToAgentModelRequirements(requirements, content) {
|
|
6875
|
+
var _a;
|
|
6857
6876
|
const trimmedContent = content.trim();
|
|
6858
6877
|
if (!trimmedContent) {
|
|
6859
6878
|
return requirements;
|
|
@@ -6874,9 +6893,13 @@
|
|
|
6874
6893
|
}
|
|
6875
6894
|
else {
|
|
6876
6895
|
const inlineSource = createInlineKnowledgeSourceFile(trimmedContent);
|
|
6896
|
+
const existingInlineSources = (((_a = requirements._metadata) === null || _a === void 0 ? void 0 : _a.inlineKnowledgeSources) || []).slice();
|
|
6877
6897
|
const updatedRequirements = {
|
|
6878
6898
|
...requirements,
|
|
6879
|
-
|
|
6899
|
+
_metadata: {
|
|
6900
|
+
...requirements._metadata,
|
|
6901
|
+
inlineKnowledgeSources: [...existingInlineSources, inlineSource],
|
|
6902
|
+
},
|
|
6880
6903
|
};
|
|
6881
6904
|
const knowledgeInfo = `Knowledge Source Inline: ${inlineSource.filename} (derived from inline content and processed for retrieval during chat)`;
|
|
6882
6905
|
return this.appendToSystemMessage(updatedRequirements, knowledgeInfo, '\n\n');
|
|
@@ -6958,6 +6981,237 @@
|
|
|
6958
6981
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
6959
6982
|
*/
|
|
6960
6983
|
|
|
6984
|
+
/**
|
|
6985
|
+
* @@@
|
|
6986
|
+
*
|
|
6987
|
+
* @private utility for commitments
|
|
6988
|
+
*/
|
|
6989
|
+
function formatOptionalInstructionBlock(label, content) {
|
|
6990
|
+
const trimmedContent = spaceTrim$1.spaceTrim(content);
|
|
6991
|
+
if (!trimmedContent) {
|
|
6992
|
+
return '';
|
|
6993
|
+
}
|
|
6994
|
+
return spaceTrim$1.spaceTrim((block) => `
|
|
6995
|
+
- ${label}:
|
|
6996
|
+
${block(trimmedContent
|
|
6997
|
+
.split(/\r?\n/)
|
|
6998
|
+
.map((line) => `- ${line}`)
|
|
6999
|
+
.join('\n'))}
|
|
7000
|
+
`);
|
|
7001
|
+
}
|
|
7002
|
+
|
|
7003
|
+
/**
|
|
7004
|
+
* Prompt parameter key used to pass hidden runtime context to tool execution.
|
|
7005
|
+
*
|
|
7006
|
+
* @private internal runtime wiring for commitment tools
|
|
7007
|
+
*/
|
|
7008
|
+
const TOOL_RUNTIME_CONTEXT_PARAMETER = 'promptbookToolRuntimeContext';
|
|
7009
|
+
/**
|
|
7010
|
+
* Hidden argument key used to pass runtime context into individual tool calls.
|
|
7011
|
+
*
|
|
7012
|
+
* @private internal runtime wiring for commitment tools
|
|
7013
|
+
*/
|
|
7014
|
+
const TOOL_RUNTIME_CONTEXT_ARGUMENT = '__promptbookToolRuntimeContext';
|
|
7015
|
+
/**
|
|
7016
|
+
* Parses unknown runtime context payload into a normalized object.
|
|
7017
|
+
*
|
|
7018
|
+
* @private internal runtime wiring for commitment tools
|
|
7019
|
+
*/
|
|
7020
|
+
function parseToolRuntimeContext(rawValue) {
|
|
7021
|
+
if (!rawValue) {
|
|
7022
|
+
return null;
|
|
7023
|
+
}
|
|
7024
|
+
let parsed = rawValue;
|
|
7025
|
+
if (typeof rawValue === 'string') {
|
|
7026
|
+
try {
|
|
7027
|
+
parsed = JSON.parse(rawValue);
|
|
7028
|
+
}
|
|
7029
|
+
catch (_a) {
|
|
7030
|
+
return null;
|
|
7031
|
+
}
|
|
7032
|
+
}
|
|
7033
|
+
if (!parsed || typeof parsed !== 'object') {
|
|
7034
|
+
return null;
|
|
7035
|
+
}
|
|
7036
|
+
return parsed;
|
|
7037
|
+
}
|
|
7038
|
+
/**
|
|
7039
|
+
* Reads runtime context attached to tool call arguments.
|
|
7040
|
+
*
|
|
7041
|
+
* @private internal runtime wiring for commitment tools
|
|
7042
|
+
*/
|
|
7043
|
+
function readToolRuntimeContextFromToolArgs(args) {
|
|
7044
|
+
return parseToolRuntimeContext(args[TOOL_RUNTIME_CONTEXT_ARGUMENT]);
|
|
7045
|
+
}
|
|
7046
|
+
/**
|
|
7047
|
+
* Serializes runtime context for prompt parameters.
|
|
7048
|
+
*
|
|
7049
|
+
* @private internal runtime wiring for commitment tools
|
|
7050
|
+
*/
|
|
7051
|
+
function serializeToolRuntimeContext(context) {
|
|
7052
|
+
return JSON.stringify(context);
|
|
7053
|
+
}
|
|
7054
|
+
/**
|
|
7055
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
7056
|
+
*/
|
|
7057
|
+
|
|
7058
|
+
/**
|
|
7059
|
+
* Tool name used to retrieve persisted user memory.
|
|
7060
|
+
*
|
|
7061
|
+
* @private internal MEMORY commitment constant
|
|
7062
|
+
*/
|
|
7063
|
+
const RETRIEVE_USER_MEMORY_TOOL_NAME = 'retrieve_user_memory';
|
|
7064
|
+
/**
|
|
7065
|
+
* Tool name used to store persisted user memory.
|
|
7066
|
+
*
|
|
7067
|
+
* @private internal MEMORY commitment constant
|
|
7068
|
+
*/
|
|
7069
|
+
const STORE_USER_MEMORY_TOOL_NAME = 'store_user_memory';
|
|
7070
|
+
const UPDATE_USER_MEMORY_TOOL_NAME = 'update_user_memory';
|
|
7071
|
+
const DELETE_USER_MEMORY_TOOL_NAME = 'delete_user_memory';
|
|
7072
|
+
/**
|
|
7073
|
+
* Resolves runtime context from hidden tool arguments.
|
|
7074
|
+
*
|
|
7075
|
+
* @private utility of MEMORY commitment
|
|
7076
|
+
*/
|
|
7077
|
+
function resolveMemoryRuntimeContext(args) {
|
|
7078
|
+
const runtimeContext = readToolRuntimeContextFromToolArgs(args);
|
|
7079
|
+
const memoryContext = runtimeContext === null || runtimeContext === void 0 ? void 0 : runtimeContext.memory;
|
|
7080
|
+
return {
|
|
7081
|
+
enabled: (memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.enabled) === true,
|
|
7082
|
+
userId: memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.userId,
|
|
7083
|
+
username: memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.username,
|
|
7084
|
+
agentId: memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.agentId,
|
|
7085
|
+
agentName: memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.agentName,
|
|
7086
|
+
isTeamConversation: (memoryContext === null || memoryContext === void 0 ? void 0 : memoryContext.isTeamConversation) === true,
|
|
7087
|
+
};
|
|
7088
|
+
}
|
|
7089
|
+
/**
|
|
7090
|
+
* Builds a disabled memory-tool response payload.
|
|
7091
|
+
*
|
|
7092
|
+
* @private utility of MEMORY commitment
|
|
7093
|
+
*/
|
|
7094
|
+
function createDisabledMemoryResult(action, message) {
|
|
7095
|
+
if (action === 'retrieve') {
|
|
7096
|
+
return {
|
|
7097
|
+
action,
|
|
7098
|
+
status: 'disabled',
|
|
7099
|
+
memories: [],
|
|
7100
|
+
message,
|
|
7101
|
+
};
|
|
7102
|
+
}
|
|
7103
|
+
if (action === 'store') {
|
|
7104
|
+
return {
|
|
7105
|
+
action,
|
|
7106
|
+
status: 'disabled',
|
|
7107
|
+
message,
|
|
7108
|
+
};
|
|
7109
|
+
}
|
|
7110
|
+
if (action === 'update') {
|
|
7111
|
+
return {
|
|
7112
|
+
action,
|
|
7113
|
+
status: 'disabled',
|
|
7114
|
+
message,
|
|
7115
|
+
};
|
|
7116
|
+
}
|
|
7117
|
+
if (action === 'delete') {
|
|
7118
|
+
return {
|
|
7119
|
+
action,
|
|
7120
|
+
status: 'disabled',
|
|
7121
|
+
message,
|
|
7122
|
+
};
|
|
7123
|
+
}
|
|
7124
|
+
throw new Error(`Unsupported memory tool action: ${action}`);
|
|
7125
|
+
}
|
|
7126
|
+
/**
|
|
7127
|
+
* Gets the runtime adapter and returns a disabled result when unavailable.
|
|
7128
|
+
*
|
|
7129
|
+
* @private utility of MEMORY commitment
|
|
7130
|
+
*/
|
|
7131
|
+
function getRuntimeAdapterOrDisabledResult(action, runtimeContext) {
|
|
7132
|
+
if (!runtimeContext.enabled || runtimeContext.isTeamConversation) {
|
|
7133
|
+
return {
|
|
7134
|
+
adapter: null,
|
|
7135
|
+
disabledResult: createDisabledMemoryResult(action, runtimeContext.isTeamConversation
|
|
7136
|
+
? 'Memory is disabled for TEAM conversations.'
|
|
7137
|
+
: 'Memory is disabled for unauthenticated users.'),
|
|
7138
|
+
};
|
|
7139
|
+
}
|
|
7140
|
+
{
|
|
7141
|
+
return {
|
|
7142
|
+
adapter: null,
|
|
7143
|
+
disabledResult: createDisabledMemoryResult(action, 'Memory runtime is not available in this environment.'),
|
|
7144
|
+
};
|
|
7145
|
+
}
|
|
7146
|
+
}
|
|
7147
|
+
/**
|
|
7148
|
+
* Parses retrieve memory arguments.
|
|
7149
|
+
*
|
|
7150
|
+
* @private utility of MEMORY commitment
|
|
7151
|
+
*/
|
|
7152
|
+
function parseRetrieveMemoryArgs(args) {
|
|
7153
|
+
const query = typeof args.query === 'string' ? args.query.trim() : undefined;
|
|
7154
|
+
const limit = typeof args.limit === 'number' && Number.isFinite(args.limit) ? Math.floor(args.limit) : undefined;
|
|
7155
|
+
return {
|
|
7156
|
+
query: query && query.length > 0 ? query : undefined,
|
|
7157
|
+
limit: limit && limit > 0 ? Math.min(limit, 20) : undefined,
|
|
7158
|
+
};
|
|
7159
|
+
}
|
|
7160
|
+
/**
|
|
7161
|
+
* Parses store memory arguments.
|
|
7162
|
+
*
|
|
7163
|
+
* @private utility of MEMORY commitment
|
|
7164
|
+
*/
|
|
7165
|
+
function parseStoreMemoryArgs(args) {
|
|
7166
|
+
const content = typeof args.content === 'string' ? args.content.trim() : '';
|
|
7167
|
+
if (!content) {
|
|
7168
|
+
throw new Error('Memory content is required.');
|
|
7169
|
+
}
|
|
7170
|
+
return {
|
|
7171
|
+
content,
|
|
7172
|
+
isGlobal: args.isGlobal === true,
|
|
7173
|
+
};
|
|
7174
|
+
}
|
|
7175
|
+
/**
|
|
7176
|
+
* Parses a memory identifier argument shared across MEMORY tools.
|
|
7177
|
+
*
|
|
7178
|
+
* @private utility of MEMORY commitment
|
|
7179
|
+
*/
|
|
7180
|
+
function parseMemoryIdArg(value) {
|
|
7181
|
+
const memoryId = typeof value === 'string' ? value.trim() : '';
|
|
7182
|
+
if (!memoryId) {
|
|
7183
|
+
throw new Error('Memory id is required.');
|
|
7184
|
+
}
|
|
7185
|
+
return memoryId;
|
|
7186
|
+
}
|
|
7187
|
+
/**
|
|
7188
|
+
* Parses update memory arguments.
|
|
7189
|
+
*
|
|
7190
|
+
* @private utility of MEMORY commitment
|
|
7191
|
+
*/
|
|
7192
|
+
function parseUpdateMemoryArgs(args) {
|
|
7193
|
+
const memoryId = parseMemoryIdArg(args.memoryId);
|
|
7194
|
+
const content = typeof args.content === 'string' ? args.content.trim() : '';
|
|
7195
|
+
if (!content) {
|
|
7196
|
+
throw new Error('Memory content is required.');
|
|
7197
|
+
}
|
|
7198
|
+
const isGlobal = typeof args.isGlobal === 'boolean' ? args.isGlobal : undefined;
|
|
7199
|
+
return {
|
|
7200
|
+
memoryId,
|
|
7201
|
+
content,
|
|
7202
|
+
isGlobal,
|
|
7203
|
+
};
|
|
7204
|
+
}
|
|
7205
|
+
/**
|
|
7206
|
+
* Parses delete memory arguments.
|
|
7207
|
+
*
|
|
7208
|
+
* @private utility of MEMORY commitment
|
|
7209
|
+
*/
|
|
7210
|
+
function parseDeleteMemoryArgs(args) {
|
|
7211
|
+
return {
|
|
7212
|
+
memoryId: parseMemoryIdArg(args.memoryId),
|
|
7213
|
+
};
|
|
7214
|
+
}
|
|
6961
7215
|
/**
|
|
6962
7216
|
* MEMORY commitment definition
|
|
6963
7217
|
*
|
|
@@ -6979,6 +7233,9 @@
|
|
|
6979
7233
|
constructor(type = 'MEMORY') {
|
|
6980
7234
|
super(type);
|
|
6981
7235
|
}
|
|
7236
|
+
get requiresContent() {
|
|
7237
|
+
return false;
|
|
7238
|
+
}
|
|
6982
7239
|
/**
|
|
6983
7240
|
* Short one-line description of MEMORY.
|
|
6984
7241
|
*/
|
|
@@ -6998,21 +7255,14 @@
|
|
|
6998
7255
|
return spaceTrim$1.spaceTrim(`
|
|
6999
7256
|
# ${this.type}
|
|
7000
7257
|
|
|
7001
|
-
|
|
7258
|
+
Enables persistent user memory for the current agent. The memory is stored by the runtime and can be retrieved in future conversations.
|
|
7002
7259
|
|
|
7003
7260
|
## Key aspects
|
|
7004
7261
|
|
|
7005
|
-
- Both
|
|
7006
|
-
-
|
|
7007
|
-
-
|
|
7008
|
-
-
|
|
7009
|
-
|
|
7010
|
-
## Differences from KNOWLEDGE
|
|
7011
|
-
|
|
7012
|
-
- \`KNOWLEDGE\` is for domain expertise and factual information
|
|
7013
|
-
- \`MEMORY\` is for user-specific context and preferences
|
|
7014
|
-
- \`MEMORY\` creates more personalized interactions
|
|
7015
|
-
- \`MEMORY\` often includes temporal or preference-based information
|
|
7262
|
+
- Both \`MEMORY\` and \`MEMORIES\` work identically.
|
|
7263
|
+
- Stores user-specific details through runtime tools.
|
|
7264
|
+
- Retrieves relevant memories for personalized responses.
|
|
7265
|
+
- Supports optional extra instructions in the commitment content.
|
|
7016
7266
|
|
|
7017
7267
|
## Examples
|
|
7018
7268
|
|
|
@@ -7020,10 +7270,7 @@
|
|
|
7020
7270
|
Personal Assistant
|
|
7021
7271
|
|
|
7022
7272
|
PERSONA You are a personal productivity assistant
|
|
7023
|
-
MEMORY
|
|
7024
|
-
MEMORY User prefers morning work sessions and afternoon meetings
|
|
7025
|
-
MEMORY Previously helped with project planning for mobile apps
|
|
7026
|
-
MEMORY User timezone: UTC-8 (Pacific Time)
|
|
7273
|
+
MEMORY Remember user projects and long-term preferences.
|
|
7027
7274
|
GOAL Help optimize daily productivity and workflow
|
|
7028
7275
|
\`\`\`
|
|
7029
7276
|
|
|
@@ -7031,10 +7278,7 @@
|
|
|
7031
7278
|
Learning Companion
|
|
7032
7279
|
|
|
7033
7280
|
PERSONA You are an educational companion for programming students
|
|
7034
|
-
MEMORY
|
|
7035
|
-
MEMORY Previous topics covered: variables, loops, functions
|
|
7036
|
-
MEMORY Student learns best with practical examples and exercises
|
|
7037
|
-
MEMORY Last session: working on list comprehensions
|
|
7281
|
+
MEMORY Remember only the student's learning progress and preferred study style.
|
|
7038
7282
|
GOAL Provide progressive learning experiences tailored to student's pace
|
|
7039
7283
|
\`\`\`
|
|
7040
7284
|
|
|
@@ -7042,41 +7286,263 @@
|
|
|
7042
7286
|
Customer Support Agent
|
|
7043
7287
|
|
|
7044
7288
|
PERSONA You are a customer support representative
|
|
7045
|
-
MEMORY
|
|
7046
|
-
MEMORY Previous issue: billing question resolved last month
|
|
7047
|
-
MEMORY Customer prefers email communication over phone calls
|
|
7048
|
-
MEMORY Account shows frequent use of advanced features
|
|
7289
|
+
MEMORY Remember only important support history and communication preferences.
|
|
7049
7290
|
GOAL Provide personalized support based on customer history
|
|
7050
7291
|
\`\`\`
|
|
7051
7292
|
`);
|
|
7052
7293
|
}
|
|
7053
7294
|
applyToAgentModelRequirements(requirements, content) {
|
|
7054
|
-
const
|
|
7055
|
-
|
|
7056
|
-
|
|
7295
|
+
const extraInstructions = formatOptionalInstructionBlock('Memory instructions', content);
|
|
7296
|
+
const existingTools = requirements.tools || [];
|
|
7297
|
+
const tools = [...existingTools];
|
|
7298
|
+
if (!tools.some((tool) => tool.name === RETRIEVE_USER_MEMORY_TOOL_NAME)) {
|
|
7299
|
+
tools.push({
|
|
7300
|
+
name: RETRIEVE_USER_MEMORY_TOOL_NAME,
|
|
7301
|
+
description: spaceTrim$1.spaceTrim(`
|
|
7302
|
+
Retrieve previously stored user memories relevant to the current conversation.
|
|
7303
|
+
Use this before responding when user context can improve the answer.
|
|
7304
|
+
`),
|
|
7305
|
+
parameters: {
|
|
7306
|
+
type: 'object',
|
|
7307
|
+
properties: {
|
|
7308
|
+
query: {
|
|
7309
|
+
type: 'string',
|
|
7310
|
+
description: 'Optional query used to filter relevant memories.',
|
|
7311
|
+
},
|
|
7312
|
+
limit: {
|
|
7313
|
+
type: 'integer',
|
|
7314
|
+
description: 'Optional maximum number of memories to return (default 5, max 20).',
|
|
7315
|
+
},
|
|
7316
|
+
},
|
|
7317
|
+
},
|
|
7318
|
+
});
|
|
7319
|
+
}
|
|
7320
|
+
if (!tools.some((tool) => tool.name === STORE_USER_MEMORY_TOOL_NAME)) {
|
|
7321
|
+
tools.push({
|
|
7322
|
+
name: STORE_USER_MEMORY_TOOL_NAME,
|
|
7323
|
+
description: spaceTrim$1.spaceTrim(`
|
|
7324
|
+
Store a durable user memory that should be remembered in future conversations.
|
|
7325
|
+
Store only stable and useful user-specific facts or preferences.
|
|
7326
|
+
`),
|
|
7327
|
+
parameters: {
|
|
7328
|
+
type: 'object',
|
|
7329
|
+
properties: {
|
|
7330
|
+
content: {
|
|
7331
|
+
type: 'string',
|
|
7332
|
+
description: 'Memory text to store.',
|
|
7333
|
+
},
|
|
7334
|
+
isGlobal: {
|
|
7335
|
+
type: 'boolean',
|
|
7336
|
+
description: 'Set true to make this memory global across all user agents.',
|
|
7337
|
+
},
|
|
7338
|
+
},
|
|
7339
|
+
required: ['content'],
|
|
7340
|
+
},
|
|
7341
|
+
});
|
|
7342
|
+
}
|
|
7343
|
+
if (!tools.some((tool) => tool.name === UPDATE_USER_MEMORY_TOOL_NAME)) {
|
|
7344
|
+
tools.push({
|
|
7345
|
+
name: UPDATE_USER_MEMORY_TOOL_NAME,
|
|
7346
|
+
description: spaceTrim$1.spaceTrim(`
|
|
7347
|
+
Update an existing user memory after retrieving it, so the stored fact stays accurate.
|
|
7348
|
+
Always pass the memory id you retrieved along with the new content.
|
|
7349
|
+
`),
|
|
7350
|
+
parameters: {
|
|
7351
|
+
type: 'object',
|
|
7352
|
+
properties: {
|
|
7353
|
+
memoryId: {
|
|
7354
|
+
type: 'string',
|
|
7355
|
+
description: 'Unique identifier of the memory entry to update.',
|
|
7356
|
+
},
|
|
7357
|
+
content: {
|
|
7358
|
+
type: 'string',
|
|
7359
|
+
description: 'Updated memory text.',
|
|
7360
|
+
},
|
|
7361
|
+
isGlobal: {
|
|
7362
|
+
type: 'boolean',
|
|
7363
|
+
description: 'Set true to keep the fact global; omit or false to keep it agent-scoped.',
|
|
7364
|
+
},
|
|
7365
|
+
},
|
|
7366
|
+
required: ['memoryId', 'content'],
|
|
7367
|
+
},
|
|
7368
|
+
});
|
|
7369
|
+
}
|
|
7370
|
+
if (!tools.some((tool) => tool.name === DELETE_USER_MEMORY_TOOL_NAME)) {
|
|
7371
|
+
tools.push({
|
|
7372
|
+
name: DELETE_USER_MEMORY_TOOL_NAME,
|
|
7373
|
+
description: spaceTrim$1.spaceTrim(`
|
|
7374
|
+
Delete a user memory that is no longer relevant. Deletions are soft so the record is hidden from future queries.
|
|
7375
|
+
`),
|
|
7376
|
+
parameters: {
|
|
7377
|
+
type: 'object',
|
|
7378
|
+
properties: {
|
|
7379
|
+
memoryId: {
|
|
7380
|
+
type: 'string',
|
|
7381
|
+
description: 'Unique identifier of the memory entry to delete.',
|
|
7382
|
+
},
|
|
7383
|
+
},
|
|
7384
|
+
required: ['memoryId'],
|
|
7385
|
+
},
|
|
7386
|
+
});
|
|
7057
7387
|
}
|
|
7058
|
-
|
|
7059
|
-
|
|
7060
|
-
|
|
7061
|
-
|
|
7388
|
+
return this.appendToSystemMessage({
|
|
7389
|
+
...requirements,
|
|
7390
|
+
tools,
|
|
7391
|
+
_metadata: {
|
|
7392
|
+
...requirements._metadata,
|
|
7393
|
+
useMemory: content || true,
|
|
7394
|
+
},
|
|
7395
|
+
}, spaceTrim$1.spaceTrim((block) => `
|
|
7396
|
+
Memory:
|
|
7397
|
+
- Prefer storing agent-scoped memories; only make them global when the fact should apply across all your agents.
|
|
7398
|
+
- You can use persistent user memory tools.
|
|
7399
|
+
- Use "${RETRIEVE_USER_MEMORY_TOOL_NAME}" to load relevant memory before answering.
|
|
7400
|
+
- Use "${STORE_USER_MEMORY_TOOL_NAME}" to save stable user-specific facts that improve future help.
|
|
7401
|
+
- Use "${UPDATE_USER_MEMORY_TOOL_NAME}" to refresh an existing memory when the content changes.
|
|
7402
|
+
- Use "${DELETE_USER_MEMORY_TOOL_NAME}" to delete memories that are no longer accurate (deletions are soft and hidden from future queries).
|
|
7403
|
+
- Store concise memory items and avoid duplicates.
|
|
7404
|
+
- Never claim memory was saved or loaded unless the tool confirms it.
|
|
7405
|
+
${block(extraInstructions)}
|
|
7406
|
+
`));
|
|
7062
7407
|
}
|
|
7063
|
-
|
|
7064
|
-
|
|
7065
|
-
|
|
7066
|
-
|
|
7067
|
-
|
|
7068
|
-
|
|
7069
|
-
|
|
7070
|
-
|
|
7071
|
-
|
|
7072
|
-
|
|
7073
|
-
|
|
7074
|
-
|
|
7075
|
-
|
|
7076
|
-
|
|
7077
|
-
|
|
7078
|
-
|
|
7079
|
-
|
|
7408
|
+
/**
|
|
7409
|
+
* Gets human-readable titles for MEMORY tool functions.
|
|
7410
|
+
*/
|
|
7411
|
+
getToolTitles() {
|
|
7412
|
+
return {
|
|
7413
|
+
[RETRIEVE_USER_MEMORY_TOOL_NAME]: 'User memory',
|
|
7414
|
+
[STORE_USER_MEMORY_TOOL_NAME]: 'Store user memory',
|
|
7415
|
+
[UPDATE_USER_MEMORY_TOOL_NAME]: 'Update user memory',
|
|
7416
|
+
[DELETE_USER_MEMORY_TOOL_NAME]: 'Delete user memory',
|
|
7417
|
+
};
|
|
7418
|
+
}
|
|
7419
|
+
/**
|
|
7420
|
+
* Gets MEMORY tool function implementations.
|
|
7421
|
+
*/
|
|
7422
|
+
getToolFunctions() {
|
|
7423
|
+
return {
|
|
7424
|
+
async [RETRIEVE_USER_MEMORY_TOOL_NAME](args) {
|
|
7425
|
+
const runtimeContext = resolveMemoryRuntimeContext(args);
|
|
7426
|
+
const { adapter, disabledResult } = getRuntimeAdapterOrDisabledResult('retrieve', runtimeContext);
|
|
7427
|
+
if (!adapter || disabledResult) {
|
|
7428
|
+
return JSON.stringify(disabledResult);
|
|
7429
|
+
}
|
|
7430
|
+
const parsedArgs = parseRetrieveMemoryArgs(args);
|
|
7431
|
+
try {
|
|
7432
|
+
const memories = await adapter.retrieveMemories(parsedArgs, runtimeContext);
|
|
7433
|
+
const result = {
|
|
7434
|
+
action: 'retrieve',
|
|
7435
|
+
status: 'ok',
|
|
7436
|
+
query: parsedArgs.query,
|
|
7437
|
+
memories,
|
|
7438
|
+
};
|
|
7439
|
+
return JSON.stringify(result);
|
|
7440
|
+
}
|
|
7441
|
+
catch (error) {
|
|
7442
|
+
const result = {
|
|
7443
|
+
action: 'retrieve',
|
|
7444
|
+
status: 'error',
|
|
7445
|
+
query: parsedArgs.query,
|
|
7446
|
+
memories: [],
|
|
7447
|
+
message: error instanceof Error ? error.message : String(error),
|
|
7448
|
+
};
|
|
7449
|
+
return JSON.stringify(result);
|
|
7450
|
+
}
|
|
7451
|
+
},
|
|
7452
|
+
async [STORE_USER_MEMORY_TOOL_NAME](args) {
|
|
7453
|
+
const runtimeContext = resolveMemoryRuntimeContext(args);
|
|
7454
|
+
const { adapter, disabledResult } = getRuntimeAdapterOrDisabledResult('store', runtimeContext);
|
|
7455
|
+
if (!adapter || disabledResult) {
|
|
7456
|
+
return JSON.stringify(disabledResult);
|
|
7457
|
+
}
|
|
7458
|
+
try {
|
|
7459
|
+
const parsedArgs = parseStoreMemoryArgs(args);
|
|
7460
|
+
const memory = await adapter.storeMemory(parsedArgs, runtimeContext);
|
|
7461
|
+
const result = {
|
|
7462
|
+
action: 'store',
|
|
7463
|
+
status: 'stored',
|
|
7464
|
+
memory,
|
|
7465
|
+
};
|
|
7466
|
+
return JSON.stringify(result);
|
|
7467
|
+
}
|
|
7468
|
+
catch (error) {
|
|
7469
|
+
const result = {
|
|
7470
|
+
action: 'store',
|
|
7471
|
+
status: 'error',
|
|
7472
|
+
message: error instanceof Error ? error.message : String(error),
|
|
7473
|
+
};
|
|
7474
|
+
return JSON.stringify(result);
|
|
7475
|
+
}
|
|
7476
|
+
},
|
|
7477
|
+
async [UPDATE_USER_MEMORY_TOOL_NAME](args) {
|
|
7478
|
+
const runtimeContext = resolveMemoryRuntimeContext(args);
|
|
7479
|
+
const { adapter, disabledResult } = getRuntimeAdapterOrDisabledResult('update', runtimeContext);
|
|
7480
|
+
if (!adapter || disabledResult) {
|
|
7481
|
+
return JSON.stringify(disabledResult);
|
|
7482
|
+
}
|
|
7483
|
+
try {
|
|
7484
|
+
const parsedArgs = parseUpdateMemoryArgs(args);
|
|
7485
|
+
const memory = await adapter.updateMemory(parsedArgs, runtimeContext);
|
|
7486
|
+
const result = {
|
|
7487
|
+
action: 'update',
|
|
7488
|
+
status: 'updated',
|
|
7489
|
+
memory,
|
|
7490
|
+
};
|
|
7491
|
+
return JSON.stringify(result);
|
|
7492
|
+
}
|
|
7493
|
+
catch (error) {
|
|
7494
|
+
const result = {
|
|
7495
|
+
action: 'update',
|
|
7496
|
+
status: 'error',
|
|
7497
|
+
message: error instanceof Error ? error.message : String(error),
|
|
7498
|
+
};
|
|
7499
|
+
return JSON.stringify(result);
|
|
7500
|
+
}
|
|
7501
|
+
},
|
|
7502
|
+
async [DELETE_USER_MEMORY_TOOL_NAME](args) {
|
|
7503
|
+
const runtimeContext = resolveMemoryRuntimeContext(args);
|
|
7504
|
+
const { adapter, disabledResult } = getRuntimeAdapterOrDisabledResult('delete', runtimeContext);
|
|
7505
|
+
if (!adapter || disabledResult) {
|
|
7506
|
+
return JSON.stringify(disabledResult);
|
|
7507
|
+
}
|
|
7508
|
+
try {
|
|
7509
|
+
const parsedArgs = parseDeleteMemoryArgs(args);
|
|
7510
|
+
const deleted = await adapter.deleteMemory(parsedArgs, runtimeContext);
|
|
7511
|
+
const result = {
|
|
7512
|
+
action: 'delete',
|
|
7513
|
+
status: 'deleted',
|
|
7514
|
+
memoryId: deleted.id,
|
|
7515
|
+
};
|
|
7516
|
+
return JSON.stringify(result);
|
|
7517
|
+
}
|
|
7518
|
+
catch (error) {
|
|
7519
|
+
const result = {
|
|
7520
|
+
action: 'delete',
|
|
7521
|
+
status: 'error',
|
|
7522
|
+
message: error instanceof Error ? error.message : String(error),
|
|
7523
|
+
};
|
|
7524
|
+
return JSON.stringify(result);
|
|
7525
|
+
}
|
|
7526
|
+
},
|
|
7527
|
+
};
|
|
7528
|
+
}
|
|
7529
|
+
}
|
|
7530
|
+
/**
|
|
7531
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
7532
|
+
*/
|
|
7533
|
+
|
|
7534
|
+
/**
|
|
7535
|
+
* AGENT MESSAGE commitment definition
|
|
7536
|
+
*
|
|
7537
|
+
* The AGENT MESSAGE commitment defines a message from the agent in the conversation history.
|
|
7538
|
+
* It is used to pre-fill the chat with a conversation history or to provide few-shot examples.
|
|
7539
|
+
*
|
|
7540
|
+
* Example usage in agent source:
|
|
7541
|
+
*
|
|
7542
|
+
* ```book
|
|
7543
|
+
* AGENT MESSAGE What seems to be the issue?
|
|
7544
|
+
* ```
|
|
7545
|
+
*
|
|
7080
7546
|
* @private [🪔] Maybe export the commitments through some package
|
|
7081
7547
|
*/
|
|
7082
7548
|
class AgentMessageCommitmentDefinition extends BaseCommitmentDefinition {
|
|
@@ -9171,14 +9637,30 @@
|
|
|
9171
9637
|
/**
|
|
9172
9638
|
* Builds a minimal chat prompt for teammate calls.
|
|
9173
9639
|
*/
|
|
9174
|
-
function buildTeammatePrompt(request) {
|
|
9640
|
+
function buildTeammatePrompt(request, runtimeContext) {
|
|
9175
9641
|
return {
|
|
9176
9642
|
title: 'Teammate consultation',
|
|
9177
9643
|
modelRequirements: {
|
|
9178
9644
|
modelVariant: 'CHAT',
|
|
9179
9645
|
},
|
|
9180
9646
|
content: request,
|
|
9181
|
-
parameters: {
|
|
9647
|
+
parameters: {
|
|
9648
|
+
[TOOL_RUNTIME_CONTEXT_PARAMETER]: serializeToolRuntimeContext(runtimeContext),
|
|
9649
|
+
},
|
|
9650
|
+
};
|
|
9651
|
+
}
|
|
9652
|
+
/**
|
|
9653
|
+
* Creates teammate runtime context and marks conversation as team-only memory-disabled.
|
|
9654
|
+
*/
|
|
9655
|
+
function createTeamConversationRuntimeContext(value) {
|
|
9656
|
+
const runtimeContext = parseToolRuntimeContext(value) || {};
|
|
9657
|
+
return {
|
|
9658
|
+
...runtimeContext,
|
|
9659
|
+
memory: {
|
|
9660
|
+
...(runtimeContext.memory || {}),
|
|
9661
|
+
enabled: false,
|
|
9662
|
+
isTeamConversation: true,
|
|
9663
|
+
},
|
|
9182
9664
|
};
|
|
9183
9665
|
}
|
|
9184
9666
|
/**
|
|
@@ -9222,7 +9704,7 @@
|
|
|
9222
9704
|
let toolCalls;
|
|
9223
9705
|
try {
|
|
9224
9706
|
const remoteAgent = await getRemoteTeammateAgent(entry.teammate.url);
|
|
9225
|
-
const prompt = buildTeammatePrompt(request);
|
|
9707
|
+
const prompt = buildTeammatePrompt(request, createTeamConversationRuntimeContext(args[TOOL_RUNTIME_CONTEXT_ARGUMENT]));
|
|
9226
9708
|
const teammateResult = await remoteAgent.callChatModel(prompt);
|
|
9227
9709
|
response = teammateResult.content || '';
|
|
9228
9710
|
toolCalls =
|
|
@@ -9772,25 +10254,6 @@
|
|
|
9772
10254
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
9773
10255
|
*/
|
|
9774
10256
|
|
|
9775
|
-
/**
|
|
9776
|
-
* @@@
|
|
9777
|
-
*
|
|
9778
|
-
* @private utility for commitments
|
|
9779
|
-
*/
|
|
9780
|
-
function formatOptionalInstructionBlock(label, content) {
|
|
9781
|
-
const trimmedContent = spaceTrim$1.spaceTrim(content);
|
|
9782
|
-
if (!trimmedContent) {
|
|
9783
|
-
return '';
|
|
9784
|
-
}
|
|
9785
|
-
return spaceTrim$1.spaceTrim((block) => `
|
|
9786
|
-
- ${label}:
|
|
9787
|
-
${block(trimmedContent
|
|
9788
|
-
.split(/\r?\n/)
|
|
9789
|
-
.map((line) => `- ${line}`)
|
|
9790
|
-
.join('\n'))}
|
|
9791
|
-
`);
|
|
9792
|
-
}
|
|
9793
|
-
|
|
9794
10257
|
/**
|
|
9795
10258
|
* Client-side safe wrapper for sending emails.
|
|
9796
10259
|
*
|
|
@@ -11407,6 +11870,63 @@
|
|
|
11407
11870
|
var styles$c = {"BookEditor":"BookEditor-module_BookEditor__s-0PU","bookEditorContainer":"BookEditor-module_bookEditorContainer__wLMwM","isVerbose":"BookEditor-module_isVerbose__VQ6iL","bookEditorWrapper":"BookEditor-module_bookEditorWrapper__twppD","isBorderRadiusDisabled":"BookEditor-module_isBorderRadiusDisabled__h1I3v","dropOverlay":"BookEditor-module_dropOverlay__xWWoX","bookEditorActionbar":"BookEditor-module_bookEditorActionbar__KW6dc","fullscreen":"BookEditor-module_fullscreen__rktsl","button":"BookEditor-module_button__hS390","savedNotification":"BookEditor-module_savedNotification__OiX9L","fadeOut":"BookEditor-module_fadeOut__q8JnR","uploadPanel":"BookEditor-module_uploadPanel__2JJtD","uploadPanelHeader":"BookEditor-module_uploadPanelHeader__pdJd2","uploadPanelTitle":"BookEditor-module_uploadPanelTitle__TJIVF","uploadPanelHeaderMeta":"BookEditor-module_uploadPanelHeaderMeta__Xw0uI","uploadPanelSummary":"BookEditor-module_uploadPanelSummary__rwSbG","uploadPanelProgressBar":"BookEditor-module_uploadPanelProgressBar__a6pjf","uploadPanelProgressFill":"BookEditor-module_uploadPanelProgressFill__l-TKR","uploadPanelList":"BookEditor-module_uploadPanelList__VxEd5","uploadRow":"BookEditor-module_uploadRow__QiSFg","uploadRowHeader":"BookEditor-module_uploadRowHeader__po0j5","uploadRowName":"BookEditor-module_uploadRowName__doQRO","uploadRowStatus":"BookEditor-module_uploadRowStatus__jsUb-","uploadRowMeta":"BookEditor-module_uploadRowMeta__1lz9h","uploadRowProgressBar":"BookEditor-module_uploadRowProgressBar__NoBA7","uploadRowProgressFill":"BookEditor-module_uploadRowProgressFill__TrP7e","uploadRowActions":"BookEditor-module_uploadRowActions__5Y1Mq","uploadActionButton":"BookEditor-module_uploadActionButton__CqJrr","uploadRowError":"BookEditor-module_uploadRowError__eEHWw"};
|
|
11408
11871
|
styleInject(css_248z$c);
|
|
11409
11872
|
|
|
11873
|
+
/**
|
|
11874
|
+
* Helper for manipulating agent books in-memory.
|
|
11875
|
+
*
|
|
11876
|
+
* @private internal helper for `@promptbook/core`
|
|
11877
|
+
*/
|
|
11878
|
+
class BookEditable {
|
|
11879
|
+
/**
|
|
11880
|
+
* Creates a mutable view over the supplied book content.
|
|
11881
|
+
*
|
|
11882
|
+
* @param book - Agent source text to work with.
|
|
11883
|
+
*/
|
|
11884
|
+
constructor(book) {
|
|
11885
|
+
this.lines = book.split(/\r?\n/);
|
|
11886
|
+
}
|
|
11887
|
+
/**
|
|
11888
|
+
* Creates a new instance from the supplied book text.
|
|
11889
|
+
*
|
|
11890
|
+
* @param book - Agent source text to work with.
|
|
11891
|
+
* @returns Editable view over the book content.
|
|
11892
|
+
*/
|
|
11893
|
+
static from(book) {
|
|
11894
|
+
return new BookEditable(book);
|
|
11895
|
+
}
|
|
11896
|
+
/**
|
|
11897
|
+
* Returns the editable content as a string_book.
|
|
11898
|
+
*
|
|
11899
|
+
* @returns Current book content with original line breaks preserved.
|
|
11900
|
+
*/
|
|
11901
|
+
toString() {
|
|
11902
|
+
return this.lines.join('\n');
|
|
11903
|
+
}
|
|
11904
|
+
/**
|
|
11905
|
+
* Checks whether the book contains any non-empty lines.
|
|
11906
|
+
*
|
|
11907
|
+
* @returns `true` when the book has meaningful content beyond blank lines.
|
|
11908
|
+
*/
|
|
11909
|
+
hasNonEmptyContent() {
|
|
11910
|
+
return this.lines.some((line) => line.trim() !== '');
|
|
11911
|
+
}
|
|
11912
|
+
/**
|
|
11913
|
+
* Finds the last line index whose trimmed text matches the commitment name.
|
|
11914
|
+
*
|
|
11915
|
+
* @param commitment - Commitment keyword to locate (for example `CLOSED`).
|
|
11916
|
+
* @returns Zero-based line index or `null` when the commitment is absent.
|
|
11917
|
+
*/
|
|
11918
|
+
findLastCommitmentLineIndex(commitment) {
|
|
11919
|
+
const normalized = commitment.trim().toUpperCase();
|
|
11920
|
+
for (let lineIndex = this.lines.length - 1; lineIndex >= 0; lineIndex -= 1) {
|
|
11921
|
+
const line = this.lines[lineIndex];
|
|
11922
|
+
if ((line === null || line === void 0 ? void 0 : line.trim().toUpperCase()) === normalized) {
|
|
11923
|
+
return lineIndex;
|
|
11924
|
+
}
|
|
11925
|
+
}
|
|
11926
|
+
return null;
|
|
11927
|
+
}
|
|
11928
|
+
}
|
|
11929
|
+
|
|
11410
11930
|
/**
|
|
11411
11931
|
* Gets all available commitment definitions
|
|
11412
11932
|
* @returns Array of all commitment definitions
|
|
@@ -11867,6 +12387,39 @@
|
|
|
11867
12387
|
{ regex: /(^|[^\\])\\\(([\s\S]+?)\\\)/g, displayMode: false },
|
|
11868
12388
|
{ regex: /(^|[^\\])\$([^$\n]+?)\$/g, displayMode: false },
|
|
11869
12389
|
];
|
|
12390
|
+
const CODE_FENCE_REGEX = /(`{3,}|~{3,})(?:[^\n\r]*)\r?\n[\s\S]*?\r?\n\1[^\n\r]*/g;
|
|
12391
|
+
const INLINE_CODE_REGEX = /(`+)([\s\S]*?)(\1)/g;
|
|
12392
|
+
const CODE_PLACEHOLDER_PREFIX = '@@PROMPTBOOK_CODE_PLACEHOLDER__';
|
|
12393
|
+
const CODE_PLACEHOLDER_REGEX = new RegExp(`${CODE_PLACEHOLDER_PREFIX}(\\d+)__`, 'g');
|
|
12394
|
+
/**
|
|
12395
|
+
* Masks inline and fenced code segments so math rendering never touches them.
|
|
12396
|
+
*
|
|
12397
|
+
* @param markdown - Markdown text to mask.
|
|
12398
|
+
* @returns Masked markdown and a restore helper.
|
|
12399
|
+
*
|
|
12400
|
+
* @private utility of `MarkdownContent` component
|
|
12401
|
+
*/
|
|
12402
|
+
function maskMarkdownCodeSegments(markdown) {
|
|
12403
|
+
const segments = [];
|
|
12404
|
+
let masked = markdown;
|
|
12405
|
+
const addPlaceholder = (segment) => {
|
|
12406
|
+
const placeholder = `${CODE_PLACEHOLDER_PREFIX}${segments.length}__`;
|
|
12407
|
+
segments.push(segment);
|
|
12408
|
+
return placeholder;
|
|
12409
|
+
};
|
|
12410
|
+
const maskWith = (regex) => {
|
|
12411
|
+
regex.lastIndex = 0;
|
|
12412
|
+
masked = masked.replace(regex, (match) => addPlaceholder(match));
|
|
12413
|
+
};
|
|
12414
|
+
maskWith(CODE_FENCE_REGEX);
|
|
12415
|
+
maskWith(INLINE_CODE_REGEX);
|
|
12416
|
+
return {
|
|
12417
|
+
masked: masked,
|
|
12418
|
+
restore(value) {
|
|
12419
|
+
return value.replace(CODE_PLACEHOLDER_REGEX, (_match, index) => { var _a; return (_a = segments[Number(index)]) !== null && _a !== void 0 ? _a : ''; });
|
|
12420
|
+
},
|
|
12421
|
+
};
|
|
12422
|
+
}
|
|
11870
12423
|
function replaceMathDelimiter(md, delimiter) {
|
|
11871
12424
|
return md.replace(delimiter.regex, (...args) => {
|
|
11872
12425
|
var _a, _b, _c;
|
|
@@ -11896,11 +12449,13 @@
|
|
|
11896
12449
|
* @private utility of `MarkdownContent` component
|
|
11897
12450
|
*/
|
|
11898
12451
|
function renderMathInMarkdown(md) {
|
|
12452
|
+
const { masked, restore } = maskMarkdownCodeSegments(md);
|
|
12453
|
+
let processed = masked;
|
|
11899
12454
|
for (const delimiter of mathDelimiterDefinitions) {
|
|
11900
|
-
|
|
12455
|
+
processed = replaceMathDelimiter(processed, delimiter);
|
|
11901
12456
|
}
|
|
11902
|
-
|
|
11903
|
-
return
|
|
12457
|
+
processed = processed.replace(/\\$/g, '$');
|
|
12458
|
+
return restore(processed);
|
|
11904
12459
|
}
|
|
11905
12460
|
/**
|
|
11906
12461
|
* Convert markdown content to HTML
|
|
@@ -13276,6 +13831,7 @@
|
|
|
13276
13831
|
* Inserts upload placeholders and queues uploads for processing.
|
|
13277
13832
|
*/
|
|
13278
13833
|
const handleFiles = react.useCallback(async (files) => {
|
|
13834
|
+
var _a;
|
|
13279
13835
|
if (!onFileUpload || !editor || !monaco) {
|
|
13280
13836
|
return;
|
|
13281
13837
|
}
|
|
@@ -13291,15 +13847,21 @@
|
|
|
13291
13847
|
file,
|
|
13292
13848
|
placeholder: getUploadPlaceholderText(file.name),
|
|
13293
13849
|
}));
|
|
13294
|
-
const
|
|
13295
|
-
const
|
|
13296
|
-
const
|
|
13297
|
-
const
|
|
13298
|
-
const
|
|
13850
|
+
const currentValue = ((_a = model.getValue()) !== null && _a !== void 0 ? _a : '');
|
|
13851
|
+
const bookEditable = new BookEditable(currentValue);
|
|
13852
|
+
const closedLineIndex = bookEditable.findLastCommitmentLineIndex('CLOSED');
|
|
13853
|
+
const insertingBeforeClosed = closedLineIndex !== null;
|
|
13854
|
+
const insertLine = insertingBeforeClosed ? closedLineIndex + 1 : model.getLineCount();
|
|
13855
|
+
const insertColumn = insertingBeforeClosed ? 1 : model.getLineMaxColumn(insertLine);
|
|
13856
|
+
const shouldAddLeadingLineBreak = !insertingBeforeClosed && Boolean(model.getValue());
|
|
13857
|
+
const prefix = shouldAddLeadingLineBreak ? '\n' : '';
|
|
13858
|
+
const placeholderBlock = placeholders.map((entry) => `${entry.placeholder}\n`).join('');
|
|
13859
|
+
const textToInsert = `${prefix}${placeholderBlock}`;
|
|
13860
|
+
const insertStartOffset = model.getOffsetAt(new monaco.Position(insertLine, insertColumn));
|
|
13299
13861
|
editor.executeEdits('upload-placeholders', [
|
|
13300
13862
|
{
|
|
13301
|
-
range: new monaco.Range(
|
|
13302
|
-
text:
|
|
13863
|
+
range: new monaco.Range(insertLine, insertColumn, insertLine, insertColumn),
|
|
13864
|
+
text: textToInsert,
|
|
13303
13865
|
forceMoveMarkers: true,
|
|
13304
13866
|
},
|
|
13305
13867
|
]);
|
|
@@ -14562,17 +15124,274 @@
|
|
|
14562
15124
|
*/
|
|
14563
15125
|
const TemplateIcon = ({ size }) => (jsxRuntime.jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", children: jsxRuntime.jsx("path", { d: "M14,2H6A2,2 0 0,0 4,4V20A2,2 0 0,0 6,22H18A2,2 0 0,0 20,20V8L14,2M18,20H6V4H13V9H18V20Z" }) }));
|
|
14564
15126
|
|
|
15127
|
+
/**
|
|
15128
|
+
* Creates an inline data URL for an SVG illustration so the ex-port stays self-contained.
|
|
15129
|
+
*
|
|
15130
|
+
* @private Internal helper used by `htmlSaveFormatDefinition`.
|
|
15131
|
+
*/
|
|
15132
|
+
function createSvgDataUrl(svg) {
|
|
15133
|
+
return `data:image/svg+xml;utf8,${encodeURIComponent(svg)}`;
|
|
15134
|
+
}
|
|
14565
15135
|
/**
|
|
14566
15136
|
* Utility to compute readable text color based on background
|
|
15137
|
+
*
|
|
15138
|
+
* @private Internal helper of the HTML export renderer.
|
|
14567
15139
|
*/
|
|
14568
15140
|
function getTextColor(bgColor) {
|
|
14569
|
-
// Simple luminance check for contrast
|
|
14570
15141
|
const hex = bgColor.replace('#', '');
|
|
14571
15142
|
const r = parseInt(hex.substring(0, 2), 16);
|
|
14572
15143
|
const g = parseInt(hex.substring(2, 4), 16);
|
|
14573
15144
|
const b = parseInt(hex.substring(4, 6), 16);
|
|
14574
15145
|
const luminance = 0.299 * r + 0.587 * g + 0.114 * b;
|
|
14575
|
-
return luminance > 186 ? '#
|
|
15146
|
+
return luminance > 186 ? '#0f172a' : '#f8fafc';
|
|
15147
|
+
}
|
|
15148
|
+
const HERO_ILLUSTRATION_SVG = spaceTrim__default["default"](() => `
|
|
15149
|
+
<svg width="320" height="220" viewBox="0 0 320 220" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
15150
|
+
<defs>
|
|
15151
|
+
<linearGradient id="heroGradient" x1="0" y1="0" x2="320" y2="220">
|
|
15152
|
+
<stop offset="0%" stop-color="#2563eb" stop-opacity="0.9" />
|
|
15153
|
+
<stop offset="100%" stop-color="#0ea5e9" stop-opacity="0.45" />
|
|
15154
|
+
</linearGradient>
|
|
15155
|
+
</defs>
|
|
15156
|
+
<rect width="320" height="220" rx="28" fill="#0f172a" />
|
|
15157
|
+
<rect x="20" y="24" width="280" height="128" rx="18" fill="url(#heroGradient)" />
|
|
15158
|
+
<rect x="36" y="52" width="248" height="16" rx="8" fill="rgba(255,255,255,0.75)" />
|
|
15159
|
+
<rect x="36" y="84" width="212" height="12" rx="6" fill="rgba(255,255,255,0.55)" />
|
|
15160
|
+
<rect x="36" y="104" width="260" height="10" rx="5" fill="rgba(255,255,255,0.35)" />
|
|
15161
|
+
<circle cx="78" cy="168" r="42" fill="#22d3ee" opacity="0.85" />
|
|
15162
|
+
<circle cx="222" cy="178" r="32" fill="#facc15" opacity="0.8" />
|
|
15163
|
+
<rect x="62" y="130" width="196" height="20" rx="10" fill="rgba(255,255,255,0.15)" />
|
|
15164
|
+
</svg>
|
|
15165
|
+
`);
|
|
15166
|
+
const BRAND_MARK_SVG = spaceTrim__default["default"](() => `
|
|
15167
|
+
<svg width="92" height="92" viewBox="0 0 92 92" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
15168
|
+
<defs>
|
|
15169
|
+
<linearGradient id="badgeGradient" x1="0" y1="0" x2="92" y2="92">
|
|
15170
|
+
<stop offset="0%" stop-color="#38bdf8" />
|
|
15171
|
+
<stop offset="100%" stop-color="#0ea5e9" />
|
|
15172
|
+
</linearGradient>
|
|
15173
|
+
</defs>
|
|
15174
|
+
<rect width="92" height="92" rx="22" fill="url(#badgeGradient)" />
|
|
15175
|
+
<path
|
|
15176
|
+
d="M58 66H42V26H58C67.388 26 74 32.658 74 42C74 51.342 67.388 58 58 58H48V66Z"
|
|
15177
|
+
fill="white"
|
|
15178
|
+
/>
|
|
15179
|
+
<path
|
|
15180
|
+
d="M42 66H34V26H42V66Z"
|
|
15181
|
+
fill="#0f172a"
|
|
15182
|
+
fill-opacity="0.6"
|
|
15183
|
+
/>
|
|
15184
|
+
</svg>
|
|
15185
|
+
`);
|
|
15186
|
+
const HERO_ILLUSTRATION_URL = createSvgDataUrl(HERO_ILLUSTRATION_SVG);
|
|
15187
|
+
const BRAND_MARK_URL = createSvgDataUrl(BRAND_MARK_SVG);
|
|
15188
|
+
const ROLE_COLOR_FALLBACKS = {
|
|
15189
|
+
USER: '#0ea5e9',
|
|
15190
|
+
ASSISTANT: '#2563eb',
|
|
15191
|
+
SYSTEM: '#475569',
|
|
15192
|
+
};
|
|
15193
|
+
/**
|
|
15194
|
+
* Escapes HTML-special characters so user-provided content stays safe.
|
|
15195
|
+
*
|
|
15196
|
+
* @private Internal helper of the HTML export renderer.
|
|
15197
|
+
*/
|
|
15198
|
+
function escapeHtml(value) {
|
|
15199
|
+
return String(value)
|
|
15200
|
+
.replace(/&/g, '&')
|
|
15201
|
+
.replace(/</g, '<')
|
|
15202
|
+
.replace(/>/g, '>')
|
|
15203
|
+
.replace(/"/g, '"')
|
|
15204
|
+
.replace(/'/g, ''');
|
|
15205
|
+
}
|
|
15206
|
+
/**
|
|
15207
|
+
* Splits a message body into paragraphs while preserving line breaks.
|
|
15208
|
+
*
|
|
15209
|
+
* @private Internal helper of the HTML export renderer.
|
|
15210
|
+
*/
|
|
15211
|
+
function formatMessageContent(content) {
|
|
15212
|
+
if (!content.trim()) {
|
|
15213
|
+
return '<p class="message-empty">No text provided.</p>';
|
|
15214
|
+
}
|
|
15215
|
+
const paragraphs = content.split(/\n{2,}/).map((paragraph) => paragraph.trim());
|
|
15216
|
+
const rendered = paragraphs
|
|
15217
|
+
.filter(Boolean)
|
|
15218
|
+
.map((paragraph) => `<p>${escapeHtml(paragraph).replace(/\n/g, '<br />')}</p>`)
|
|
15219
|
+
.join('');
|
|
15220
|
+
if (rendered) {
|
|
15221
|
+
return rendered;
|
|
15222
|
+
}
|
|
15223
|
+
return '<p class="message-empty">No text provided.</p>';
|
|
15224
|
+
}
|
|
15225
|
+
/**
|
|
15226
|
+
* Formats arbitrary date values into a consistent human-friendly label.
|
|
15227
|
+
*
|
|
15228
|
+
* @private Internal helper of the HTML export renderer.
|
|
15229
|
+
*/
|
|
15230
|
+
function formatTimestamp(value) {
|
|
15231
|
+
if (!value) {
|
|
15232
|
+
return '';
|
|
15233
|
+
}
|
|
15234
|
+
const date = typeof value === 'string' ? new Date(value) : value;
|
|
15235
|
+
if (Number.isNaN(date.getTime())) {
|
|
15236
|
+
return '';
|
|
15237
|
+
}
|
|
15238
|
+
return new Intl.DateTimeFormat('en-US', {
|
|
15239
|
+
weekday: 'short',
|
|
15240
|
+
month: 'short',
|
|
15241
|
+
day: 'numeric',
|
|
15242
|
+
year: 'numeric',
|
|
15243
|
+
hour: 'numeric',
|
|
15244
|
+
minute: '2-digit',
|
|
15245
|
+
}).format(date);
|
|
15246
|
+
}
|
|
15247
|
+
/**
|
|
15248
|
+
* Builds a lookup map keyed by participant name to quickly resolve visuals.
|
|
15249
|
+
*
|
|
15250
|
+
* @private Internal helper of the HTML export renderer.
|
|
15251
|
+
*/
|
|
15252
|
+
function buildParticipantMap(participants) {
|
|
15253
|
+
const map = new Map();
|
|
15254
|
+
for (const participant of participants) {
|
|
15255
|
+
const normalized = String(participant.name);
|
|
15256
|
+
map.set(normalized, participant);
|
|
15257
|
+
map.set(normalized.toUpperCase(), participant);
|
|
15258
|
+
}
|
|
15259
|
+
return map;
|
|
15260
|
+
}
|
|
15261
|
+
/**
|
|
15262
|
+
* Normalizes participant colors (either raw strings or Color instances) into hex strings.
|
|
15263
|
+
*
|
|
15264
|
+
* @private Internal helper of the HTML export renderer.
|
|
15265
|
+
*/
|
|
15266
|
+
function normalizeParticipantColor(color) {
|
|
15267
|
+
if (!color) {
|
|
15268
|
+
return undefined;
|
|
15269
|
+
}
|
|
15270
|
+
if (typeof color === 'string') {
|
|
15271
|
+
return color;
|
|
15272
|
+
}
|
|
15273
|
+
const asHelper = color;
|
|
15274
|
+
if (typeof (asHelper === null || asHelper === void 0 ? void 0 : asHelper.toString) === 'function') {
|
|
15275
|
+
return asHelper.toString();
|
|
15276
|
+
}
|
|
15277
|
+
return undefined;
|
|
15278
|
+
}
|
|
15279
|
+
/**
|
|
15280
|
+
* Resolves the visuals (name, avatar, accent color) for a message sender.
|
|
15281
|
+
*
|
|
15282
|
+
* @private Internal helper of the HTML export renderer.
|
|
15283
|
+
*/
|
|
15284
|
+
function resolveParticipantVisuals(participants, sender) {
|
|
15285
|
+
var _a, _b, _c, _d;
|
|
15286
|
+
const normalized = String(sender || 'SYSTEM');
|
|
15287
|
+
const upper = normalized.toUpperCase();
|
|
15288
|
+
const participant = (_a = participants.get(normalized)) !== null && _a !== void 0 ? _a : participants.get(upper);
|
|
15289
|
+
const accentColor = (_c = (_b = normalizeParticipantColor(participant === null || participant === void 0 ? void 0 : participant.color)) !== null && _b !== void 0 ? _b : ROLE_COLOR_FALLBACKS[upper]) !== null && _c !== void 0 ? _c : '#2563eb';
|
|
15290
|
+
const displayName = ((_d = participant === null || participant === void 0 ? void 0 : participant.fullname) === null || _d === void 0 ? void 0 : _d.trim()) || normalized;
|
|
15291
|
+
const avatarLabel = displayName.charAt(0).toUpperCase() || '?';
|
|
15292
|
+
return {
|
|
15293
|
+
displayName,
|
|
15294
|
+
avatarLabel,
|
|
15295
|
+
accentColor,
|
|
15296
|
+
avatarSrc: participant === null || participant === void 0 ? void 0 : participant.avatarSrc,
|
|
15297
|
+
};
|
|
15298
|
+
}
|
|
15299
|
+
/**
|
|
15300
|
+
* Builds markup for attachments inside a message bubble.
|
|
15301
|
+
*
|
|
15302
|
+
* @private Internal helper of the HTML export renderer.
|
|
15303
|
+
*/
|
|
15304
|
+
function buildAttachmentsMarkup(message) {
|
|
15305
|
+
const attachments = message.attachments;
|
|
15306
|
+
if (!attachments || attachments.length === 0) {
|
|
15307
|
+
return '';
|
|
15308
|
+
}
|
|
15309
|
+
const chips = attachments.map((attachment) => {
|
|
15310
|
+
var _a;
|
|
15311
|
+
const hasUrl = Boolean(attachment.url);
|
|
15312
|
+
const tag = hasUrl ? 'a' : 'span';
|
|
15313
|
+
const href = hasUrl ? ` href="${escapeHtml((_a = attachment.url) !== null && _a !== void 0 ? _a : '#')}" target="_blank" rel="noopener"` : '';
|
|
15314
|
+
const name = escapeHtml(attachment.name || 'Attachment');
|
|
15315
|
+
const meta = escapeHtml(attachment.type || 'file');
|
|
15316
|
+
return spaceTrim__default["default"](`
|
|
15317
|
+
<${tag} class="attachment-chip"${href}>
|
|
15318
|
+
<span class="attachment-icon">📎</span>
|
|
15319
|
+
<span class="attachment-name">${name}</span>
|
|
15320
|
+
<span class="attachment-meta">${meta}</span>
|
|
15321
|
+
</${tag}>
|
|
15322
|
+
`);
|
|
15323
|
+
});
|
|
15324
|
+
return `<div class="message-attachments">${chips.join('')}</div>`;
|
|
15325
|
+
}
|
|
15326
|
+
/**
|
|
15327
|
+
* Builds markup for citations within a message bubble.
|
|
15328
|
+
*
|
|
15329
|
+
* @private Internal helper of the HTML export renderer.
|
|
15330
|
+
*/
|
|
15331
|
+
function buildCitationsMarkup(message) {
|
|
15332
|
+
const citations = message.citations;
|
|
15333
|
+
if (!citations || citations.length === 0) {
|
|
15334
|
+
return '';
|
|
15335
|
+
}
|
|
15336
|
+
const chips = citations.map((citation) => {
|
|
15337
|
+
const excerpt = citation.excerpt ? `<p class="citation-excerpt">${escapeHtml(citation.excerpt)}</p>` : '';
|
|
15338
|
+
const urlLink = citation.url
|
|
15339
|
+
? `<a class="citation-link" href="${escapeHtml(citation.url)}" target="_blank" rel="noopener">Open source</a>`
|
|
15340
|
+
: '';
|
|
15341
|
+
return spaceTrim__default["default"](`
|
|
15342
|
+
<article class="citation-chip">
|
|
15343
|
+
<div class="citation-header">
|
|
15344
|
+
<span class="citation-badge">${escapeHtml(citation.id)}</span>
|
|
15345
|
+
<span class="citation-source">${escapeHtml(citation.source)}</span>
|
|
15346
|
+
</div>
|
|
15347
|
+
${excerpt}
|
|
15348
|
+
${urlLink}
|
|
15349
|
+
</article>
|
|
15350
|
+
`);
|
|
15351
|
+
});
|
|
15352
|
+
return `<div class="message-citations">${chips.join('')}</div>`;
|
|
15353
|
+
}
|
|
15354
|
+
/**
|
|
15355
|
+
* Renders a single message block enriched with avatar, metadata, attachments, and citations.
|
|
15356
|
+
*
|
|
15357
|
+
* @private Internal helper of the HTML export renderer.
|
|
15358
|
+
*/
|
|
15359
|
+
function renderMessageBlock(message, participants) {
|
|
15360
|
+
var _a;
|
|
15361
|
+
const sender = String((_a = message.sender) !== null && _a !== void 0 ? _a : 'SYSTEM');
|
|
15362
|
+
const upperSender = sender.toUpperCase();
|
|
15363
|
+
const visuals = resolveParticipantVisuals(participants, sender);
|
|
15364
|
+
const bubbleTextColor = getTextColor(visuals.accentColor);
|
|
15365
|
+
const timestamp = formatTimestamp(message.createdAt);
|
|
15366
|
+
const durationLabel = typeof message.generationDurationMs === 'number' ? `${(message.generationDurationMs / 1000).toFixed(1)}s` : '';
|
|
15367
|
+
const attachments = buildAttachmentsMarkup(message);
|
|
15368
|
+
const citations = buildCitationsMarkup(message);
|
|
15369
|
+
const alignmentClass = upperSender === 'USER' ? 'message-user' : 'message-assistant';
|
|
15370
|
+
const avatarMarkup = visuals.avatarSrc
|
|
15371
|
+
? `<img class="message-avatar-img" src="${escapeHtml(visuals.avatarSrc)}" alt="${escapeHtml(visuals.displayName)}" />`
|
|
15372
|
+
: `<span class="message-avatar-fallback" style="background:${visuals.accentColor};color:${bubbleTextColor};">${escapeHtml(visuals.avatarLabel)}</span>`;
|
|
15373
|
+
return spaceTrim__default["default"](`
|
|
15374
|
+
<article class="message-block ${alignmentClass}">
|
|
15375
|
+
<div class="message-avatar">${avatarMarkup}</div>
|
|
15376
|
+
<div class="message-card" style="--bubble-color:${visuals.accentColor};--bubble-text:${bubbleTextColor};">
|
|
15377
|
+
<div class="message-card-inner">
|
|
15378
|
+
<header class="message-card-header">
|
|
15379
|
+
<div>
|
|
15380
|
+
<strong class="message-name">${escapeHtml(visuals.displayName)}</strong>
|
|
15381
|
+
<span class="message-role">${escapeHtml(upperSender)}</span>
|
|
15382
|
+
</div>
|
|
15383
|
+
${timestamp ? `<time class="message-time">${escapeHtml(timestamp)}</time>` : ''}
|
|
15384
|
+
</header>
|
|
15385
|
+
<div class="message-content">${formatMessageContent(message.content)}</div>
|
|
15386
|
+
${durationLabel
|
|
15387
|
+
? `<div class="message-duration">Responded in ${escapeHtml(durationLabel)}</div>`
|
|
15388
|
+
: ''}
|
|
15389
|
+
${attachments}
|
|
15390
|
+
${citations}
|
|
15391
|
+
</div>
|
|
15392
|
+
</div>
|
|
15393
|
+
</article>
|
|
15394
|
+
`);
|
|
14576
15395
|
}
|
|
14577
15396
|
/**
|
|
14578
15397
|
* HTML export plugin
|
|
@@ -14582,121 +15401,483 @@
|
|
|
14582
15401
|
const htmlSaveFormatDefinition = {
|
|
14583
15402
|
formatName: 'html',
|
|
14584
15403
|
label: 'Html',
|
|
14585
|
-
getContent: ({ messages }) =>
|
|
15404
|
+
getContent: ({ title, messages, participants }) => {
|
|
15405
|
+
const safeTitle = escapeHtml(title || 'Chat');
|
|
15406
|
+
const participantLookup = buildParticipantMap(participants);
|
|
15407
|
+
const exportedAt = new Date();
|
|
15408
|
+
const exportedLabel = formatTimestamp(exportedAt);
|
|
15409
|
+
const uniqueParticipants = Array.from(new Set(participants
|
|
15410
|
+
.map((participant) => { var _a; return ((_a = participant.fullname) !== null && _a !== void 0 ? _a : String(participant.name)).trim(); })
|
|
15411
|
+
.filter(Boolean)));
|
|
15412
|
+
const heroSubtitle = uniqueParticipants.length > 0
|
|
15413
|
+
? `Featuring ${uniqueParticipants.slice(0, 3).join(', ')}${uniqueParticipants.length > 3 ? ' and more' : ''}`
|
|
15414
|
+
: 'Featuring your conversation';
|
|
15415
|
+
const participantCount = new Set(participants.map((participant) => String(participant.name).trim())).size;
|
|
15416
|
+
const statCards = [
|
|
15417
|
+
{ label: 'Messages', value: messages.length.toString() },
|
|
15418
|
+
{ label: 'Participants', value: participantCount.toString() },
|
|
15419
|
+
{ label: 'Exported on', value: exportedLabel },
|
|
15420
|
+
];
|
|
15421
|
+
const messageMarkup = messages.length > 0
|
|
15422
|
+
? messages.map((message) => renderMessageBlock(message, participantLookup)).join('')
|
|
15423
|
+
: '<div class="empty-state">No messages yet. Send a note to capture this chat.</div>';
|
|
15424
|
+
return spaceTrim__default["default"](`
|
|
14586
15425
|
<!DOCTYPE html>
|
|
14587
15426
|
<html lang="en">
|
|
14588
15427
|
<head>
|
|
14589
15428
|
<meta charset="UTF-8" />
|
|
14590
15429
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
14591
|
-
<title
|
|
15430
|
+
<title>${safeTitle} · Promptbook</title>
|
|
14592
15431
|
<style>
|
|
15432
|
+
:root {
|
|
15433
|
+
color-scheme: dark;
|
|
15434
|
+
}
|
|
15435
|
+
|
|
15436
|
+
* {
|
|
15437
|
+
box-sizing: border-box;
|
|
15438
|
+
}
|
|
15439
|
+
|
|
14593
15440
|
body {
|
|
14594
|
-
font-family: 'Segoe UI', Arial, sans-serif;
|
|
14595
|
-
background: #f7f8fa;
|
|
14596
15441
|
margin: 0;
|
|
14597
|
-
|
|
15442
|
+
background: radial-gradient(circle at top, rgba(14, 165, 233, 0.18), transparent 55%),
|
|
15443
|
+
#030617;
|
|
15444
|
+
font-family: 'Inter', 'Segoe UI', system-ui, sans-serif;
|
|
15445
|
+
color: #f8fafc;
|
|
14598
15446
|
}
|
|
14599
|
-
|
|
14600
|
-
|
|
14601
|
-
|
|
14602
|
-
|
|
14603
|
-
border-radius: 12px;
|
|
14604
|
-
box-shadow: 0 2px 16px rgba(0,0,0,0.07);
|
|
14605
|
-
padding: 24px 18px 18px 18px;
|
|
15447
|
+
|
|
15448
|
+
a {
|
|
15449
|
+
color: inherit;
|
|
15450
|
+
text-decoration: none;
|
|
14606
15451
|
}
|
|
14607
|
-
|
|
15452
|
+
|
|
15453
|
+
.export-shell {
|
|
15454
|
+
max-width: 1040px;
|
|
15455
|
+
margin: 0 auto;
|
|
15456
|
+
padding: 40px 24px 64px;
|
|
14608
15457
|
display: flex;
|
|
14609
|
-
|
|
14610
|
-
|
|
14611
|
-
border-radius: 10px;
|
|
14612
|
-
padding: 0;
|
|
15458
|
+
flex-direction: column;
|
|
15459
|
+
gap: 32px;
|
|
14613
15460
|
}
|
|
14614
|
-
|
|
14615
|
-
|
|
14616
|
-
|
|
14617
|
-
border
|
|
14618
|
-
|
|
14619
|
-
|
|
14620
|
-
flex-shrink: 0;
|
|
15461
|
+
|
|
15462
|
+
.hero {
|
|
15463
|
+
background: rgba(15, 23, 42, 0.85);
|
|
15464
|
+
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
15465
|
+
border-radius: 32px;
|
|
15466
|
+
padding: 32px;
|
|
14621
15467
|
display: flex;
|
|
14622
15468
|
align-items: center;
|
|
14623
|
-
justify-content:
|
|
14624
|
-
|
|
14625
|
-
|
|
14626
|
-
|
|
14627
|
-
box-shadow: 0 1px 4px rgba(0,0,0,0.04);
|
|
15469
|
+
justify-content: space-between;
|
|
15470
|
+
gap: 32px;
|
|
15471
|
+
backdrop-filter: blur(18px);
|
|
15472
|
+
box-shadow: 0 30px 60px rgba(15, 23, 42, 0.65);
|
|
14628
15473
|
}
|
|
14629
|
-
|
|
14630
|
-
|
|
14631
|
-
|
|
14632
|
-
|
|
14633
|
-
|
|
14634
|
-
|
|
14635
|
-
|
|
15474
|
+
|
|
15475
|
+
.hero-content h1 {
|
|
15476
|
+
margin: 8px 0;
|
|
15477
|
+
font-size: 32px;
|
|
15478
|
+
line-height: 1.2;
|
|
15479
|
+
font-weight: 600;
|
|
15480
|
+
}
|
|
15481
|
+
|
|
15482
|
+
.hero-edge {
|
|
15483
|
+
color: rgba(255, 255, 255, 0.7);
|
|
15484
|
+
}
|
|
15485
|
+
|
|
15486
|
+
.hero-kicker {
|
|
15487
|
+
letter-spacing: 0.18em;
|
|
15488
|
+
text-transform: uppercase;
|
|
15489
|
+
font-size: 12px;
|
|
15490
|
+
color: rgba(248, 250, 252, 0.7);
|
|
15491
|
+
margin: 0;
|
|
15492
|
+
}
|
|
15493
|
+
|
|
15494
|
+
.hero-subtitle {
|
|
15495
|
+
margin: 0;
|
|
14636
15496
|
font-size: 16px;
|
|
14637
|
-
|
|
14638
|
-
margin-top: 2px;
|
|
15497
|
+
color: rgba(248, 250, 252, 0.65);
|
|
14639
15498
|
}
|
|
14640
|
-
|
|
14641
|
-
|
|
14642
|
-
|
|
14643
|
-
|
|
14644
|
-
|
|
14645
|
-
|
|
15499
|
+
|
|
15500
|
+
.stat-grid {
|
|
15501
|
+
margin-top: 18px;
|
|
15502
|
+
display: flex;
|
|
15503
|
+
gap: 16px;
|
|
15504
|
+
flex-wrap: wrap;
|
|
14646
15505
|
}
|
|
14647
|
-
|
|
14648
|
-
|
|
14649
|
-
|
|
14650
|
-
|
|
14651
|
-
|
|
14652
|
-
|
|
15506
|
+
|
|
15507
|
+
.stat-card {
|
|
15508
|
+
background: rgba(15, 23, 42, 0.6);
|
|
15509
|
+
border: 1px solid rgba(255, 255, 255, 0.05);
|
|
15510
|
+
border-radius: 18px;
|
|
15511
|
+
padding: 12px 18px;
|
|
15512
|
+
min-width: 120px;
|
|
14653
15513
|
}
|
|
14654
|
-
|
|
14655
|
-
|
|
14656
|
-
|
|
14657
|
-
font-
|
|
15514
|
+
|
|
15515
|
+
.stat-value {
|
|
15516
|
+
display: block;
|
|
15517
|
+
font-size: 20px;
|
|
15518
|
+
font-weight: 600;
|
|
14658
15519
|
}
|
|
14659
|
-
|
|
14660
|
-
|
|
14661
|
-
|
|
14662
|
-
|
|
14663
|
-
|
|
14664
|
-
|
|
14665
|
-
|
|
14666
|
-
|
|
14667
|
-
|
|
14668
|
-
|
|
14669
|
-
|
|
14670
|
-
|
|
14671
|
-
|
|
14672
|
-
|
|
14673
|
-
|
|
14674
|
-
|
|
14675
|
-
|
|
14676
|
-
|
|
14677
|
-
|
|
14678
|
-
|
|
14679
|
-
|
|
14680
|
-
|
|
14681
|
-
|
|
14682
|
-
|
|
14683
|
-
|
|
14684
|
-
|
|
14685
|
-
|
|
14686
|
-
|
|
14687
|
-
|
|
14688
|
-
|
|
14689
|
-
|
|
14690
|
-
|
|
14691
|
-
|
|
14692
|
-
|
|
14693
|
-
|
|
14694
|
-
|
|
14695
|
-
|
|
14696
|
-
|
|
14697
|
-
|
|
14698
|
-
|
|
14699
|
-
|
|
15520
|
+
|
|
15521
|
+
.stat-label {
|
|
15522
|
+
font-size: 12px;
|
|
15523
|
+
color: rgba(248, 250, 252, 0.55);
|
|
15524
|
+
text-transform: uppercase;
|
|
15525
|
+
letter-spacing: 0.1em;
|
|
15526
|
+
}
|
|
15527
|
+
|
|
15528
|
+
.hero-visual {
|
|
15529
|
+
display: grid;
|
|
15530
|
+
place-items: center;
|
|
15531
|
+
gap: 12px;
|
|
15532
|
+
}
|
|
15533
|
+
|
|
15534
|
+
.hero-illustration {
|
|
15535
|
+
width: 280px;
|
|
15536
|
+
border-radius: 20px;
|
|
15537
|
+
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
15538
|
+
box-shadow: 0 18px 36px rgba(2, 6, 23, 0.8);
|
|
15539
|
+
}
|
|
15540
|
+
|
|
15541
|
+
.hero-badge {
|
|
15542
|
+
width: 70px;
|
|
15543
|
+
height: 70px;
|
|
15544
|
+
}
|
|
15545
|
+
|
|
15546
|
+
.chat-panel {
|
|
15547
|
+
background: rgba(15, 23, 42, 0.95);
|
|
15548
|
+
border-radius: 28px;
|
|
15549
|
+
padding: 32px;
|
|
15550
|
+
border: 1px solid rgba(255, 255, 255, 0.06);
|
|
15551
|
+
box-shadow: 0 30px 50px rgba(2, 6, 23, 0.7);
|
|
15552
|
+
}
|
|
15553
|
+
|
|
15554
|
+
.chat-panel-header {
|
|
15555
|
+
display: flex;
|
|
15556
|
+
align-items: center;
|
|
15557
|
+
justify-content: space-between;
|
|
15558
|
+
gap: 16px;
|
|
15559
|
+
margin-bottom: 18px;
|
|
15560
|
+
}
|
|
15561
|
+
|
|
15562
|
+
.chat-panel-subhead {
|
|
15563
|
+
margin: 0;
|
|
15564
|
+
font-size: 14px;
|
|
15565
|
+
color: rgba(248, 250, 252, 0.75);
|
|
15566
|
+
}
|
|
15567
|
+
|
|
15568
|
+
.chat-panel-meta {
|
|
15569
|
+
display: flex;
|
|
15570
|
+
gap: 12px;
|
|
15571
|
+
font-size: 13px;
|
|
15572
|
+
color: rgba(248, 250, 252, 0.55);
|
|
15573
|
+
}
|
|
15574
|
+
|
|
15575
|
+
.chat-messages {
|
|
15576
|
+
display: flex;
|
|
15577
|
+
flex-direction: column;
|
|
15578
|
+
gap: 24px;
|
|
15579
|
+
}
|
|
15580
|
+
|
|
15581
|
+
.message-block {
|
|
15582
|
+
display: flex;
|
|
15583
|
+
gap: 16px;
|
|
15584
|
+
align-items: flex-start;
|
|
15585
|
+
}
|
|
15586
|
+
|
|
15587
|
+
.message-block.message-user {
|
|
15588
|
+
flex-direction: row-reverse;
|
|
15589
|
+
}
|
|
15590
|
+
|
|
15591
|
+
.message-card {
|
|
15592
|
+
flex: 1;
|
|
15593
|
+
}
|
|
15594
|
+
|
|
15595
|
+
.message-card-inner {
|
|
15596
|
+
background: radial-gradient(circle at top, rgba(255, 255, 255, 0.08), transparent),
|
|
15597
|
+
var(--bubble-color, #2563eb);
|
|
15598
|
+
border-radius: 24px;
|
|
15599
|
+
padding: 20px;
|
|
15600
|
+
box-shadow: 0 16px 32px rgba(3, 7, 18, 0.6);
|
|
15601
|
+
color: var(--bubble-text, #f8fafc);
|
|
15602
|
+
border: 1px solid rgba(255, 255, 255, 0.05);
|
|
15603
|
+
}
|
|
15604
|
+
|
|
15605
|
+
.message-card-header {
|
|
15606
|
+
display: flex;
|
|
15607
|
+
justify-content: space-between;
|
|
15608
|
+
align-items: baseline;
|
|
15609
|
+
margin-bottom: 12px;
|
|
15610
|
+
gap: 12px;
|
|
15611
|
+
}
|
|
15612
|
+
|
|
15613
|
+
.message-name {
|
|
15614
|
+
font-size: 14px;
|
|
15615
|
+
margin: 0;
|
|
15616
|
+
}
|
|
15617
|
+
|
|
15618
|
+
.message-role {
|
|
15619
|
+
font-size: 11px;
|
|
15620
|
+
text-transform: uppercase;
|
|
15621
|
+
letter-spacing: 0.14em;
|
|
15622
|
+
margin-left: 8px;
|
|
15623
|
+
opacity: 0.75;
|
|
15624
|
+
}
|
|
15625
|
+
|
|
15626
|
+
.message-time {
|
|
15627
|
+
font-size: 11px;
|
|
15628
|
+
opacity: 0.7;
|
|
15629
|
+
}
|
|
15630
|
+
|
|
15631
|
+
.message-content {
|
|
15632
|
+
font-size: 16px;
|
|
15633
|
+
line-height: 1.7;
|
|
15634
|
+
}
|
|
15635
|
+
|
|
15636
|
+
.message-duration {
|
|
15637
|
+
margin-top: 10px;
|
|
15638
|
+
font-size: 12px;
|
|
15639
|
+
opacity: 0.65;
|
|
15640
|
+
}
|
|
15641
|
+
|
|
15642
|
+
.message-avatar {
|
|
15643
|
+
width: 52px;
|
|
15644
|
+
height: 52px;
|
|
15645
|
+
flex-shrink: 0;
|
|
15646
|
+
border-radius: 12px;
|
|
15647
|
+
background: rgba(255, 255, 255, 0.04);
|
|
15648
|
+
display: grid;
|
|
15649
|
+
place-items: center;
|
|
15650
|
+
position: relative;
|
|
15651
|
+
}
|
|
15652
|
+
|
|
15653
|
+
.message-avatar-img {
|
|
15654
|
+
width: 48px;
|
|
15655
|
+
height: 48px;
|
|
15656
|
+
border-radius: 50%;
|
|
15657
|
+
object-fit: cover;
|
|
15658
|
+
}
|
|
15659
|
+
|
|
15660
|
+
.message-avatar-fallback {
|
|
15661
|
+
width: 44px;
|
|
15662
|
+
height: 44px;
|
|
15663
|
+
border-radius: 50%;
|
|
15664
|
+
display: grid;
|
|
15665
|
+
place-items: center;
|
|
15666
|
+
font-weight: 600;
|
|
15667
|
+
font-size: 18px;
|
|
15668
|
+
border: 2px solid rgba(255, 255, 255, 0.2);
|
|
15669
|
+
}
|
|
15670
|
+
|
|
15671
|
+
.message-attachments {
|
|
15672
|
+
margin-top: 14px;
|
|
15673
|
+
display: flex;
|
|
15674
|
+
gap: 10px;
|
|
15675
|
+
flex-wrap: wrap;
|
|
15676
|
+
}
|
|
15677
|
+
|
|
15678
|
+
.attachment-chip {
|
|
15679
|
+
padding: 8px 12px;
|
|
15680
|
+
background: rgba(15, 23, 42, 0.55);
|
|
15681
|
+
border-radius: 14px;
|
|
15682
|
+
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
15683
|
+
display: flex;
|
|
15684
|
+
align-items: center;
|
|
15685
|
+
gap: 6px;
|
|
15686
|
+
font-size: 13px;
|
|
15687
|
+
}
|
|
15688
|
+
|
|
15689
|
+
.attachment-icon {
|
|
15690
|
+
font-size: 14px;
|
|
15691
|
+
}
|
|
15692
|
+
|
|
15693
|
+
.attachment-name {
|
|
15694
|
+
font-weight: 500;
|
|
15695
|
+
}
|
|
15696
|
+
|
|
15697
|
+
.attachment-meta {
|
|
15698
|
+
font-size: 11px;
|
|
15699
|
+
opacity: 0.6;
|
|
15700
|
+
}
|
|
15701
|
+
|
|
15702
|
+
.message-citations {
|
|
15703
|
+
margin-top: 18px;
|
|
15704
|
+
display: grid;
|
|
15705
|
+
gap: 10px;
|
|
15706
|
+
}
|
|
15707
|
+
|
|
15708
|
+
.citation-chip {
|
|
15709
|
+
border-radius: 16px;
|
|
15710
|
+
background: rgba(255, 255, 255, 0.03);
|
|
15711
|
+
padding: 12px 16px;
|
|
15712
|
+
border: 1px solid rgba(255, 255, 255, 0.05);
|
|
15713
|
+
}
|
|
15714
|
+
|
|
15715
|
+
.citation-header {
|
|
15716
|
+
display: flex;
|
|
15717
|
+
align-items: center;
|
|
15718
|
+
gap: 8px;
|
|
15719
|
+
font-size: 12px;
|
|
15720
|
+
text-transform: uppercase;
|
|
15721
|
+
letter-spacing: 0.2em;
|
|
15722
|
+
}
|
|
15723
|
+
|
|
15724
|
+
.citation-badge {
|
|
15725
|
+
background: rgba(15, 23, 42, 0.4);
|
|
15726
|
+
padding: 2px 8px;
|
|
15727
|
+
border-radius: 999px;
|
|
15728
|
+
font-weight: 600;
|
|
15729
|
+
}
|
|
15730
|
+
|
|
15731
|
+
.citation-source {
|
|
15732
|
+
opacity: 0.65;
|
|
15733
|
+
}
|
|
15734
|
+
|
|
15735
|
+
.citation-excerpt {
|
|
15736
|
+
margin: 8px 0 0;
|
|
15737
|
+
font-size: 14px;
|
|
15738
|
+
line-height: 1.6;
|
|
15739
|
+
}
|
|
15740
|
+
|
|
15741
|
+
.citation-link {
|
|
15742
|
+
margin-top: 6px;
|
|
15743
|
+
display: inline-flex;
|
|
15744
|
+
font-size: 12px;
|
|
15745
|
+
color: #38bdf8;
|
|
15746
|
+
}
|
|
15747
|
+
|
|
15748
|
+
.empty-state {
|
|
15749
|
+
text-align: center;
|
|
15750
|
+
color: rgba(248, 250, 252, 0.6);
|
|
15751
|
+
padding: 40px;
|
|
15752
|
+
border-radius: 20px;
|
|
15753
|
+
background: rgba(15, 23, 42, 0.6);
|
|
15754
|
+
border: 1px dashed rgba(255, 255, 255, 0.2);
|
|
15755
|
+
}
|
|
15756
|
+
|
|
15757
|
+
.export-footer {
|
|
15758
|
+
border-radius: 26px;
|
|
15759
|
+
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
15760
|
+
padding: 24px 32px;
|
|
15761
|
+
display: flex;
|
|
15762
|
+
justify-content: space-between;
|
|
15763
|
+
align-items: center;
|
|
15764
|
+
background: linear-gradient(135deg, rgba(15, 23, 42, 0.8), rgba(15, 23, 42, 0.4));
|
|
15765
|
+
box-shadow: 0 24px 40px rgba(2, 6, 23, 0.7);
|
|
15766
|
+
}
|
|
15767
|
+
|
|
15768
|
+
.footer-meta {
|
|
15769
|
+
margin: 4px 0 0;
|
|
15770
|
+
opacity: 0.7;
|
|
15771
|
+
font-size: 13px;
|
|
15772
|
+
}
|
|
15773
|
+
|
|
15774
|
+
.footer-brand {
|
|
15775
|
+
display: flex;
|
|
15776
|
+
align-items: center;
|
|
15777
|
+
gap: 16px;
|
|
15778
|
+
}
|
|
15779
|
+
|
|
15780
|
+
.footer-brand img {
|
|
15781
|
+
width: 46px;
|
|
15782
|
+
height: 46px;
|
|
15783
|
+
}
|
|
15784
|
+
|
|
15785
|
+
.footer-brand span {
|
|
15786
|
+
opacity: 0.65;
|
|
15787
|
+
font-size: 13px;
|
|
15788
|
+
}
|
|
15789
|
+
|
|
15790
|
+
@media (max-width: 960px) {
|
|
15791
|
+
.hero {
|
|
15792
|
+
flex-direction: column;
|
|
15793
|
+
text-align: center;
|
|
15794
|
+
}
|
|
15795
|
+
|
|
15796
|
+
.hero-visual {
|
|
15797
|
+
order: -1;
|
|
15798
|
+
}
|
|
15799
|
+
|
|
15800
|
+
.message-block {
|
|
15801
|
+
flex-direction: column;
|
|
15802
|
+
}
|
|
15803
|
+
|
|
15804
|
+
.message-block.message-user {
|
|
15805
|
+
flex-direction: column;
|
|
15806
|
+
}
|
|
15807
|
+
}
|
|
15808
|
+
|
|
15809
|
+
@media (max-width: 640px) {
|
|
15810
|
+
.chat-panel {
|
|
15811
|
+
padding: 24px;
|
|
15812
|
+
}
|
|
15813
|
+
|
|
15814
|
+
.export-shell {
|
|
15815
|
+
padding: 28px 16px 48px;
|
|
15816
|
+
}
|
|
15817
|
+
}
|
|
15818
|
+
</style>
|
|
15819
|
+
</head>
|
|
15820
|
+
<body>
|
|
15821
|
+
<div class="export-shell">
|
|
15822
|
+
<header class="hero">
|
|
15823
|
+
<div class="hero-content">
|
|
15824
|
+
<p class="hero-kicker">Chat export</p>
|
|
15825
|
+
<h1>Conversation with ${safeTitle}</h1>
|
|
15826
|
+
<p class="hero-subtitle">${escapeHtml(heroSubtitle)}</p>
|
|
15827
|
+
<div class="stat-grid">
|
|
15828
|
+
${statCards
|
|
15829
|
+
.map((stat) => spaceTrim__default["default"](`
|
|
15830
|
+
<div class="stat-card">
|
|
15831
|
+
<span class="stat-value">${escapeHtml(stat.value)}</span>
|
|
15832
|
+
<span class="stat-label">${escapeHtml(stat.label)}</span>
|
|
15833
|
+
</div>
|
|
15834
|
+
`))
|
|
15835
|
+
.join('')}
|
|
15836
|
+
</div>
|
|
15837
|
+
</div>
|
|
15838
|
+
<div class="hero-visual">
|
|
15839
|
+
<img class="hero-illustration" src="${HERO_ILLUSTRATION_URL}" alt="" aria-hidden="true" />
|
|
15840
|
+
<img class="hero-badge" src="${BRAND_MARK_URL}" alt="Promptbook badge" width="72" height="72" />
|
|
15841
|
+
</div>
|
|
15842
|
+
</header>
|
|
15843
|
+
<section class="chat-panel">
|
|
15844
|
+
<div class="chat-panel-header">
|
|
15845
|
+
<div>
|
|
15846
|
+
<p class="chat-panel-subhead">Captured messages, ready to share</p>
|
|
15847
|
+
</div>
|
|
15848
|
+
<div class="chat-panel-meta">
|
|
15849
|
+
<span>${escapeHtml(`${messages.length} messages`)}</span>
|
|
15850
|
+
<span>${escapeHtml(`${participantCount} participants`)}</span>
|
|
15851
|
+
</div>
|
|
15852
|
+
</div>
|
|
15853
|
+
<div class="chat-messages">
|
|
15854
|
+
${messageMarkup}
|
|
15855
|
+
</div>
|
|
15856
|
+
</section>
|
|
15857
|
+
<footer class="export-footer">
|
|
15858
|
+
<div>
|
|
15859
|
+
<p>Generated by Promptbook</p>
|
|
15860
|
+
<p class="footer-meta">Exported ${escapeHtml(exportedLabel)}</p>
|
|
15861
|
+
</div>
|
|
15862
|
+
<div class="footer-brand">
|
|
15863
|
+
<img src="${BRAND_MARK_URL}" alt="" aria-hidden="true" />
|
|
15864
|
+
<div>
|
|
15865
|
+
<strong>ptbk.io</strong>
|
|
15866
|
+
<span>Share with your team</span>
|
|
15867
|
+
</div>
|
|
15868
|
+
</div>
|
|
15869
|
+
</footer>
|
|
15870
|
+
</div>
|
|
15871
|
+
</body>
|
|
15872
|
+
</html>
|
|
15873
|
+
`);
|
|
15874
|
+
},
|
|
15875
|
+
mimeType: 'text/html',
|
|
15876
|
+
fileExtension: 'html',
|
|
15877
|
+
};
|
|
15878
|
+
/**
|
|
15879
|
+
* TODO: Enhance branding
|
|
15880
|
+
*/
|
|
14700
15881
|
|
|
14701
15882
|
/**
|
|
14702
15883
|
* JSON export plugin (full metadata)
|
|
@@ -14736,6 +15917,139 @@
|
|
|
14736
15917
|
fileExtension: 'md',
|
|
14737
15918
|
};
|
|
14738
15919
|
|
|
15920
|
+
/**
|
|
15921
|
+
* Builds display-ready timestamp and duration labels for a chat message.
|
|
15922
|
+
*
|
|
15923
|
+
* @private utility of `<Chat/>` component
|
|
15924
|
+
*/
|
|
15925
|
+
function getChatMessageTimingDisplay(message) {
|
|
15926
|
+
if (!message.createdAt) {
|
|
15927
|
+
return null;
|
|
15928
|
+
}
|
|
15929
|
+
const timestamp = moment__default["default"](message.createdAt);
|
|
15930
|
+
if (!timestamp.isValid()) {
|
|
15931
|
+
return null;
|
|
15932
|
+
}
|
|
15933
|
+
const durationLabel = message.generationDurationMs === undefined
|
|
15934
|
+
? undefined
|
|
15935
|
+
: formatGenerationDurationLabel(message.generationDurationMs);
|
|
15936
|
+
return {
|
|
15937
|
+
timeLabel: timestamp.format('LT'),
|
|
15938
|
+
fullLabel: timestamp.format('YYYY-MM-DD HH:mm:ss'),
|
|
15939
|
+
durationLabel,
|
|
15940
|
+
};
|
|
15941
|
+
}
|
|
15942
|
+
/**
|
|
15943
|
+
* Formats a generation duration value into a compact label.
|
|
15944
|
+
*/
|
|
15945
|
+
function formatGenerationDurationLabel(durationMs) {
|
|
15946
|
+
if (!Number.isFinite(durationMs) || durationMs < 0) {
|
|
15947
|
+
return '0ms';
|
|
15948
|
+
}
|
|
15949
|
+
const duration = moment__default["default"].duration(durationMs);
|
|
15950
|
+
const totalSeconds = duration.asSeconds();
|
|
15951
|
+
if (totalSeconds < 1) {
|
|
15952
|
+
return `${Math.max(1, Math.round(duration.asMilliseconds()))}ms`;
|
|
15953
|
+
}
|
|
15954
|
+
if (totalSeconds < 60) {
|
|
15955
|
+
const precision = totalSeconds < 10 ? 1 : 0;
|
|
15956
|
+
return `${totalSeconds.toFixed(precision)}s`;
|
|
15957
|
+
}
|
|
15958
|
+
const minutes = Math.floor(duration.asMinutes());
|
|
15959
|
+
const seconds = Math.round(totalSeconds - minutes * 60);
|
|
15960
|
+
return `${minutes}m ${seconds}s`;
|
|
15961
|
+
}
|
|
15962
|
+
|
|
15963
|
+
/**
|
|
15964
|
+
* Generates Promptbook branding header for exported files
|
|
15965
|
+
*
|
|
15966
|
+
* @private utility of `<Chat/>` component
|
|
15967
|
+
*/
|
|
15968
|
+
function getPromptbookBranding(shareUrl) {
|
|
15969
|
+
const timestamp = new Date().toLocaleString();
|
|
15970
|
+
return `Generated by Promptbook Studio - ${timestamp}\nShare link: ${shareUrl}\nhttps://promptbook.studio\n\n`;
|
|
15971
|
+
}
|
|
15972
|
+
|
|
15973
|
+
/**
|
|
15974
|
+
* Converts chat messages to plain text format
|
|
15975
|
+
*
|
|
15976
|
+
* @private utility of `<Chat/>` component
|
|
15977
|
+
*/
|
|
15978
|
+
function messagesToText(messages, shareUrl, headerMarkdown, participants) {
|
|
15979
|
+
const branding = getPromptbookBranding(shareUrl);
|
|
15980
|
+
const header = headerMarkdown ? `${headerMarkdown}\n\n` : '';
|
|
15981
|
+
const content = messages
|
|
15982
|
+
.map((message) => {
|
|
15983
|
+
const participant = (participants || []).find((participant) => participant.name === message.sender);
|
|
15984
|
+
const fullname = (participant === null || participant === void 0 ? void 0 : participant.fullname) || message.sender;
|
|
15985
|
+
const timing = getChatMessageTimingDisplay(message);
|
|
15986
|
+
const timestampLabel = timing ? ` [${timing.fullLabel}]` : '';
|
|
15987
|
+
return `${fullname}${timestampLabel}:\n${message.content}\n`;
|
|
15988
|
+
})
|
|
15989
|
+
.join('\n');
|
|
15990
|
+
return header + branding + content;
|
|
15991
|
+
}
|
|
15992
|
+
|
|
15993
|
+
const PDF_PAGE_MARGIN_PT = 40;
|
|
15994
|
+
const PDF_FONT_SIZE_PT = 10;
|
|
15995
|
+
const PDF_LINE_HEIGHT_MULTIPLIER = 1.3;
|
|
15996
|
+
const PDF_PARAGRAPH_SPACING_MULTIPLIER = 0.25;
|
|
15997
|
+
const PDF_LINE_HEIGHT_PT = PDF_FONT_SIZE_PT * PDF_LINE_HEIGHT_MULTIPLIER;
|
|
15998
|
+
/**
|
|
15999
|
+
* Builds the share URL used inside exported files.
|
|
16000
|
+
*
|
|
16001
|
+
* @returns The current page URL or a fallback to Promptbook home when unavailable.
|
|
16002
|
+
* @private Internal helper used by chat save format implementations.
|
|
16003
|
+
*/
|
|
16004
|
+
function resolveShareUrl() {
|
|
16005
|
+
if (typeof window === 'undefined') {
|
|
16006
|
+
return 'https://promptbook.studio';
|
|
16007
|
+
}
|
|
16008
|
+
return window.location.href;
|
|
16009
|
+
}
|
|
16010
|
+
/**
|
|
16011
|
+
* Creates a minimal PDF representation of the provided chat.
|
|
16012
|
+
*
|
|
16013
|
+
* @param messages - Messages that should be included in the PDF export.
|
|
16014
|
+
* @param participants - Optional participant metadata to resolve sender names.
|
|
16015
|
+
* @returns Binary data for the generated PDF file.
|
|
16016
|
+
* @private Internal helper used by `pdfSaveFormatDefinition`.
|
|
16017
|
+
*/
|
|
16018
|
+
function buildChatPdf(messages, participants) {
|
|
16019
|
+
const shareUrl = resolveShareUrl();
|
|
16020
|
+
const textContent = messagesToText([...messages], shareUrl, undefined, participants);
|
|
16021
|
+
const pdf = new jspdf.jsPDF({
|
|
16022
|
+
unit: 'pt',
|
|
16023
|
+
format: 'letter',
|
|
16024
|
+
});
|
|
16025
|
+
pdf.setFont('Helvetica', 'normal');
|
|
16026
|
+
pdf.setFontSize(PDF_FONT_SIZE_PT);
|
|
16027
|
+
const pageWidth = pdf.internal.pageSize.getWidth();
|
|
16028
|
+
const pageHeight = pdf.internal.pageSize.getHeight();
|
|
16029
|
+
const availableWidth = pageWidth - PDF_PAGE_MARGIN_PT * 2;
|
|
16030
|
+
const availableHeight = pageHeight - PDF_PAGE_MARGIN_PT;
|
|
16031
|
+
let cursorY = PDF_PAGE_MARGIN_PT;
|
|
16032
|
+
const ensureSpace = (height) => {
|
|
16033
|
+
if (cursorY + height > availableHeight) {
|
|
16034
|
+
pdf.addPage();
|
|
16035
|
+
cursorY = PDF_PAGE_MARGIN_PT;
|
|
16036
|
+
}
|
|
16037
|
+
};
|
|
16038
|
+
const paragraphs = textContent.split('\n');
|
|
16039
|
+
for (const paragraph of paragraphs) {
|
|
16040
|
+
if (paragraph.trim() === '') {
|
|
16041
|
+
cursorY += PDF_LINE_HEIGHT_PT;
|
|
16042
|
+
continue;
|
|
16043
|
+
}
|
|
16044
|
+
const wrappedLines = pdf.splitTextToSize(paragraph, availableWidth);
|
|
16045
|
+
const paragraphHeight = wrappedLines.length * PDF_LINE_HEIGHT_PT;
|
|
16046
|
+
ensureSpace(paragraphHeight);
|
|
16047
|
+
pdf.text(wrappedLines, PDF_PAGE_MARGIN_PT, cursorY);
|
|
16048
|
+
cursorY += paragraphHeight + PDF_LINE_HEIGHT_PT * PDF_PARAGRAPH_SPACING_MULTIPLIER;
|
|
16049
|
+
}
|
|
16050
|
+
return new Uint8Array(pdf.output('arraybuffer'));
|
|
16051
|
+
}
|
|
16052
|
+
|
|
14739
16053
|
/**
|
|
14740
16054
|
* PDF export plugin
|
|
14741
16055
|
*
|
|
@@ -14744,13 +16058,7 @@
|
|
|
14744
16058
|
const pdfSaveFormatDefinition = {
|
|
14745
16059
|
formatName: 'pdf',
|
|
14746
16060
|
label: 'Pdf',
|
|
14747
|
-
getContent:
|
|
14748
|
-
// const { messages } = chatExportData;
|
|
14749
|
-
await htmlSaveFormatDefinition.getContent(chatExportData);
|
|
14750
|
-
// PDF conversion should be implemented here (sync or pre-generated)
|
|
14751
|
-
// For now, return a placeholder string
|
|
14752
|
-
return '[PDF chat save not implemented. Integrate a PDF library for conversion.]';
|
|
14753
|
-
},
|
|
16061
|
+
getContent: ({ messages, participants }) => buildChatPdf(messages, participants),
|
|
14754
16062
|
mimeType: 'application/pdf',
|
|
14755
16063
|
fileExtension: 'pdf',
|
|
14756
16064
|
};
|
|
@@ -14841,8 +16149,8 @@
|
|
|
14841
16149
|
return CHAT_SAVE_FORMATS.filter((saveFormatDefinition) => formatNames.includes(saveFormatDefinition.formatName));
|
|
14842
16150
|
}
|
|
14843
16151
|
|
|
14844
|
-
var css_248z$4 = "@font-face{font-family:OpenMojiBlack;src:url(https://s6.ptbk.io/fonts/OpenMoji-black-glyf.woff2) format(\"woff2\");unicode-range:u+23,u+2a,u+2d,u+30-39,u+a9,u+ae,u+200d,u+203c,u+2049,u+20e3,u+2117,u+2120,u+2122,u+2139,u+2194-2199,u+21a9,u+21aa,u+229c,u+231a,u+231b,u+2328,u+23cf,u+23e9-23f3,u+23f8-23fe,u+24c2,u+25a1,u+25aa-25ae,u+25b6,u+25c0,u+25c9,u+25d0,u+25d1,u+25e7-25ea,u+25ed,u+25ee,u+25fb-25fe,u+2600-2605,u+260e,u+2611,u+2614,u+2615,u+2618,u+261d,u+2620,u+2622,u+2623,u+2626,u+262a,u+262e,u+262f,u+2638-263a,u+2640,u+2642,u+2648-2653,u+265f,u+2660,u+2663,u+2665,u+2666,u+2668,u+267b,u+267e,u+267f,u+2691-2697,u+2699,u+269b,u+269c,u+26a0,u+26a1,u+26a7,u+26aa,u+26ab,u+26b0,u+26b1,u+26bd,u+26be,u+26c4,u+26c5,u+26c8,u+26ce,u+26cf,u+26d1,u+26d3,u+26d4,u+26e9,u+26ea,u+26f0-26f5,u+26f7-26fa,u+26fd,u+2702,u+2705,u+2708-270d,u+270f,u+2712,u+2714,u+2716,u+271d,u+2721,u+2728,u+2733,u+2734,u+2744,u+2747,u+274c,u+274e,u+2753-2755,u+2757,u+2763,u+2764,u+2795-2797,u+27a1,u+27b0,u+27bf,u+2934,u+2935,u+2b05-2b07,u+2b0c,u+2b0d,u+2b1b,u+2b1c,u+2b1f-2b24,u+2b2e,u+2b2f,u+2b50,u+2b55,u+2b58,u+2b8f,u+2bba-2bbc,u+2bc3,u+2bc4,u+2bea,u+2beb,u+3030,u+303d,u+3297,u+3299,u+e000-e009,u+e010,u+e011,u+e040-e06d,u+e080-e0b4,u+e0c0-e0cc,u+e0ff-e10d,u+e140-e14a,u+e150-e157,u+e181-e189,u+e1c0-e1c4,u+e1c6-e1d9,u+e200-e216,u+e240-e269,u+e280-e283,u+e2c0-e2c4,u+e2c6-e2da,u+e300-e303,u+e305-e30f,u+e312-e316,u+e318-e322,u+e324-e329,u+e32b,u+e340-e348,u+e380,u+e381,u+f000,u+f77a,u+f8ff,u+fe0f,u+1f004,u+1f0cf,u+1f10d-1f10f,u+1f12f,u+1f16d-1f171,u+1f17e,u+1f17f,u+1f18e,u+1f191-1f19a,u+1f1e6-1f1ff,u+1f201,u+1f202,u+1f21a,u+1f22f,u+1f232-1f23a,u+1f250,u+1f251,u+1f260-1f265,u+1f300-1f321,u+1f324-1f393,u+1f396,u+1f397,u+1f399-1f39b,u+1f39e-1f3f0,u+1f3f3-1f3f5,u+1f3f7-1f4fd,u+1f4ff-1f53d,u+1f549-1f54e,u+1f550-1f567,u+1f56f,u+1f570,u+1f573-1f57a,u+1f587,u+1f58a-1f58d,u+1f590,u+1f595,u+1f596,u+1f5a4,u+1f5a5,u+1f5a8,u+1f5b1,u+1f5b2,u+1f5bc,u+1f5c2-1f5c4,u+1f5d1-1f5d3,u+1f5dc-1f5de,u+1f5e1,u+1f5e3,u+1f5e8,u+1f5ef,u+1f5f3,u+1f5fa-1f64f,u+1f680-1f6c5,u+1f6cb-1f6d2,u+1f6d5-1f6d7,u+1f6dc-1f6e5,u+1f6e9,u+1f6eb,u+1f6ec,u+1f6f0,u+1f6f3-1f6fc,u+1f7e0-1f7eb,u+1f7f0,u+1f90c-1f93a,u+1f93c-1f945,u+1f947-1f9ff,u+1fa70-1fa7c,u+1fa80-1fa88,u+1fa90-1fabd,u+1fabf-1fac5,u+1face-1fadb,u+1fae0-1fae8,u+1faf0-1faf8,u+1fbc5-1fbc9,u+e0061-e0067,u+e0069,u+e006c-e0079,u+e007f}.Chat-module_copiedToClipboardMessage__apCPY{background:#222;border-radius:8px;box-shadow:0 2px 12px rgba(0,0,0,.18);color:#fff;font-size:1.1em;left:50%;opacity:.97;padding:10px 24px;pointer-events:none;position:fixed;top:32px;transform:translateX(-50%);z-index:9999}.Chat-module_Chat__j2eE5{--chat-message-min-width:150px;--chat-message-max-width:min(70%,600px);--chat-message-avatar-gap:12px;display:flex;flex-direction:column;font-family:Arial,Helvetica,sans-serif,OpenMojiBlack;height:100%;width:100%}.Chat-module_chatMainFlow__--8FE{display:grid;grid-template:\"🟦\" min-content \"💬\" 1fr \"📝\" min-content/1fr;height:100%;max-width:100vw;width:100%}.Chat-module_chatMainFlow__--8FE .Chat-module_chatBar__fLECN{background-color:#fff;border-bottom:1px solid rgba(15,23,36,.06);color:#0f1724;font-weight:500;grid-area:🟦;padding:16px 20px;text-align:center;width:100%}.Chat-module_TasksInProgress__fQfei{align-self:center;grid-area:🟦;height:min-content;justify-self:self-end;margin:8px 16px;width:auto}.Chat-module_actions__gTZ5T{align-items:center;align-self:self-start;display:flex;gap:8px;grid-area:💬;height:min-content;justify-self:self-end;margin:16px 20px 0;transition:opacity .2s ease;width:auto;will-change:opacity;z-index:40}.Chat-module_actions__gTZ5T.Chat-module_portal__uTOT8{margin:0}.Chat-module_actions__gTZ5T.Chat-module_left__7l5Mn{justify-self:self-start}.Chat-module_actions__gTZ5T.Chat-module_right__ABZrW{justify-self:self-end}.Chat-module_actionsFaded__qqAVQ{opacity:.3}.Chat-module_actionsScrolling__fIawW{opacity:.15;pointer-events:none}@media (max-width:900px){.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}}@media (max-width:600px){.Chat-module_actions__gTZ5T{gap:7px;margin:14px 18px 0}}.Chat-module_chatMainFlow__--8FE .Chat-module_chatChildren__flOPK{grid-area:💬;height:100%;width:100%;z-index:300}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{grid-area:💬;height:100%;overflow-x:hidden;overflow-y:auto;padding:24px 20px 16px;scroll-behavior:smooth;width:100%;z-index:10}.Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ{padding-top:80px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar{width:6px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-track{background:transparent}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-thumb{background:hsla(0,0%,49%,.2);border-radius:3px;transition:all .2s ease}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-thumb:hover{background:hsla(0,0%,49%,.3)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ{align-items:flex-end;animation:Chat-module_messageSlideIn__soTy2 .4s cubic-bezier(.25,.46,.45,.94);column-gap:var(--chat-message-avatar-gap);display:flex;flex-direction:row;margin-bottom:20px;max-width:100%;position:relative}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageStack__n0xNL{align-items:flex-start;display:flex;flex-direction:column;max-width:var(--chat-message-max-width)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageStack__n0xNL{align-items:flex-end}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageStack__n0xNL>*{max-width:100%}.Chat-module_messageMeta__eqCRu{align-items:center;color:#6b7280;display:flex;font-size:11px;gap:6px;line-height:1.4;margin:4px 6px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageMeta__eqCRu{align-self:flex-end;text-align:right}.Chat-module_messageTimestamp__LDhRw{letter-spacing:.02em}.Chat-module_messageDuration__Y9Kg9{opacity:.85}.Chat-module_participantLabel__0Oo-3{color:#64748b;font-size:12px;font-weight:600;margin:0 0 6px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_participantLabel__0Oo-3{text-align:right}@keyframes Chat-module_messageSlideIn__soTy2{0%{opacity:0;transform:translateY(20px) scale(.95)}to{opacity:1;transform:translateY(0) scale(1)}}.Chat-module_isNotCompleteMessage__Hj2K7{opacity:.7;position:relative}.Chat-module_NonCompleteMessageFiller__G5-Ve{color:transparent}.Chat-module_isNotCompleteMessage__Hj2K7 .Chat-module_messageText__XgNyQ:after{animation:Chat-module_loadingPulse__VomRm 1.5s ease-in-out infinite;background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.6),transparent);border-radius:2px;bottom:8px;content:\"\";height:4px;position:absolute;right:12px;width:20px}.Chat-module_completedToolCalls__vI1Qt,.Chat-module_ongoingToolCalls__NZkQN,.Chat-module_sourceCitations__oDtBX{border-top:1px solid hsla(0,0%,100%,.2);display:flex;flex-direction:row;flex-wrap:wrap;gap:8px;margin-top:8px;padding-top:8px}.Chat-module_completedToolCall__-q4Cs,.Chat-module_ongoingToolCall__WT3Rc{align-items:center;background:hsla(0,0%,100%,.15);border:1px solid hsla(0,0%,100%,.2);border-radius:12px;box-shadow:0 1px 3px rgba(0,0,0,.1);color:inherit;display:flex;font-family:OpenMojiColor,OpenMojiBlack,Arial,Helvetica,sans-serif;font-size:.8em;gap:8px;padding:4px 10px}.Chat-module_completedToolCall__-q4Cs{cursor:pointer;transition:all .2s ease}.Chat-module_completedToolCall__-q4Cs:hover{background:hsla(0,0%,100%,.3);box-shadow:0 2px 6px rgba(0,0,0,.15);transform:translateY(-1px)}.Chat-module_toolCallOrigin__-Io6Y{font-size:.85em;font-weight:500;opacity:.75;white-space:nowrap}.Chat-module_toolCallDetails__WUFlD{margin-bottom:24px}.Chat-module_toolCallDetails__WUFlD p{margin-bottom:8px}.Chat-module_toolCallData__UauCO{background:#f1f5f9;border:1px solid #e2e8f0;border-radius:8px;font-size:13px;max-height:300px;overflow-x:auto;padding:12px;word-break:break-all}.Chat-module_ongoingToolCallSpinner__7g-Ay{animation:Chat-module_toolCallSpinner__LSiK6 .8s linear infinite;border:2px solid hsla(0,0%,100%,.3);border-radius:50%;border-top-color:#fff;height:14px;width:14px}.Chat-module_ongoingToolCallName__y59-0{font-weight:500}@keyframes Chat-module_toolCallSpinner__LSiK6{to{transform:rotate(1turn)}}@keyframes Chat-module_loadingPulse__VomRm{0%,to{opacity:.3;transform:scaleX(.8)}50%{opacity:1;transform:scaleX(1.2)}}.Chat-module_typingIndicator__S-CT-{align-items:flex-end;animation:Chat-module_messageSlideIn__soTy2 .4s cubic-bezier(.25,.46,.45,.94);display:flex;margin-bottom:20px}.Chat-module_typingIndicator__S-CT- .Chat-module_avatar__gL6bm{flex-shrink:0;height:40px;margin:0 12px 4px;width:40px}.Chat-module_typingBubble__0Lb7B{backdrop-filter:blur(10px);border:1px solid hsla(0,0%,49%,.1);border-radius:20px;border-bottom-left-radius:6px;box-shadow:0 2px 8px rgba(0,0,0,.1);min-height:24px;padding:16px 20px}.Chat-module_typingBubble__0Lb7B,.Chat-module_typingDots__srOBB{align-items:center;display:flex;gap:4px}.Chat-module_typingDot__dnhKT{animation:Chat-module_typingBounce__1yp2v 1.4s ease-in-out infinite;background:linear-gradient(135deg,#6b7280,hsla(0,0%,49%,.6));border-radius:50%;box-shadow:0 2px 4px rgba(0,0,0,.1);height:8px;width:8px}.Chat-module_typingDot__dnhKT:first-child{animation-delay:-.32s}.Chat-module_typingDot__dnhKT:nth-child(2){animation-delay:-.16s}.Chat-module_typingDot__dnhKT:nth-child(3){animation-delay:0s}@keyframes Chat-module_typingBounce__1yp2v{0%,80%,to{opacity:.5;transform:scale(.8) translateY(0)}40%{opacity:1;transform:scale(1.2) translateY(-8px)}}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV{align-items:flex-end;flex-direction:row-reverse;justify-content:flex-start}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageText__XgNyQ{border-bottom-right-radius:6px}.Chat-module_ratingStar__rRfqC{color:var(--star-inactive-color,#ccc);cursor:pointer;font-size:20px;transition:color .2s}.Chat-module_ratingStar__rRfqC.Chat-module_active__lbYL-{color:gold}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{aspect-ratio:1/1;flex-shrink:0;margin:0 0 4px;position:relative;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{word-wrap:break-word;--chat-heading-surface:hsla(0,0%,100%,.12);--chat-heading-accent:rgba(56,189,248,.85);--chat-heading-text-color:var(--message-text-color,#0f172a);--chat-heading-border-color:rgba(15,23,42,.18);--chat-heading-shadow-color:rgba(15,23,42,.65);backdrop-filter:blur(10px);background-color:var(--message-bg-color);border-radius:20px;box-shadow:0 2px 8px rgba(0,0,0,.1);color:var(--message-text-color);font-size:15px;line-height:1.5;margin-bottom:4px;max-width:100%;min-width:var(--chat-message-min-width);padding:14px 18px;position:relative;text-align:left;transition:all .2s ease}@supports (color:color-mix(in srgb,#000 50%,#fff 50%)){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{--chat-heading-surface:color-mix(in srgb,var(--message-bg-color,#fff) 70%,rgba(15,23,42,.08) 30%);--chat-heading-accent:color-mix(in srgb,var(--message-text-color,#0f172a) 65%,rgba(59,130,246,.9) 35%);--chat-heading-border-color:color-mix(in srgb,var(--message-bg-color,#fff) 60%,rgba(15,23,42,.12) 40%);--chat-heading-shadow-color:color-mix(in srgb,rgba(15,23,42,.65) 80%,rgba(96,165,250,.45) 20%);--chat-heading-text-color:var(--message-text-color,#0f172a)}}.Chat-module_copyButtonContainer__Rij0U{align-items:center;float:right;justify-content:flex-end;pointer-events:none;right:10px;top:8px;visibility:hidden;z-index:2}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ:hover .Chat-module_copyButtonContainer__Rij0U,.Chat-module_copyButtonContainer__Rij0U:focus-within{pointer-events:auto;visibility:visible}.Chat-module_copyButton__DcxT5{align-items:center;background:hsla(0,0%,100%,.2);border:1px solid #ddd;border-radius:6px;box-shadow:0 1px 4px rgba(0,0,0,.07);cursor:pointer;display:flex;opacity:.7;padding:2px 5px;position:relative;transition:all .15s,box-shadow .15s,border .15s}.Chat-module_copiedTooltip__LH81j{animation:Chat-module_copiedTooltipFadeIn__QekO1 .2s;background:#222;border-radius:8px;box-shadow:0 2px 12px rgba(0,0,0,.18);color:#fff;font-size:.98em;left:50%;margin-top:4px;max-width:220px;opacity:.97;overflow-wrap:break-word;padding:6px 16px;pointer-events:none;position:absolute;top:110%;transform:translateX(-50%);white-space:nowrap;word-break:break-word;z-index:100}.Chat-module_copiedTooltipLeft__j-S-5{left:0!important;transform:none!important}.Chat-module_copiedTooltipRight__R-2cE{left:auto!important;right:0!important;transform:none!important}@keyframes Chat-module_copiedTooltipFadeIn__QekO1{0%{opacity:0;transform:translateX(-50%) translateY(8px) scale(.97)}to{opacity:.97;transform:translateX(-50%) translateY(0) scale(1)}}.Chat-module_copyButton__DcxT5:focus,.Chat-module_copyButton__DcxT5:hover{border:1.5px solid #bbb;box-shadow:0 2px 8px rgba(0,132,255,.1);opacity:1}.Chat-module_copyButton__DcxT5 svg{display:block}.Chat-module_messageText__XgNyQ ul{list-style:disc;margin-left:20px}.Chat-module_messageText__XgNyQ ol{list-style:decimal;margin-left:20px}.Chat-module_messageText__XgNyQ blockquote,.Chat-module_messageText__XgNyQ img,.Chat-module_messageText__XgNyQ pre,.Chat-module_messageText__XgNyQ table{border-radius:8px;margin-bottom:10px;margin-top:10px}.Chat-module_messageText__XgNyQ pre{background:#000;color:#fff}.Chat-module_messageText__XgNyQ blockquote,.Chat-module_messageText__XgNyQ pre{border:none;box-shadow:none;display:block;font-size:inherit;line-height:inherit;padding:1em}.Chat-module_messageText__XgNyQ blockquote{background:#ffffffcc;color:#000}.Chat-module_messageText__XgNyQ code{background:#cccccc55;border:none;box-shadow:none;color:inherit;display:inline-block;font-size:inherit;line-height:inherit;margin:0;padding:0}.Chat-module_messageText__XgNyQ pre code{background-color:#000000cc;border-radius:8px}.Chat-module_messageText__XgNyQ .Chat-module_chat-code-block__k8IyS{background:#181c23;border-color:#23272f;box-shadow:0 2px 8px rgba(0,0,0,.12);color:#f8fafc;font-family:Fira Mono,Menlo,Consolas,Liberation Mono,monospace;font-size:14px;line-height:1.6;overflow-x:auto}.Chat-module_messageText__XgNyQ .Chat-module_chat-code-block__k8IyS code{background:none!important;border:none!important;box-shadow:none!important;color:inherit!important;display:block;font-family:inherit!important;font-size:inherit!important;overflow-x:auto;padding:0!important;white-space:pre;word-break:break-word}.Chat-module_messageText__XgNyQ table{background:#f8fafc;border-collapse:separate;border-spacing:0;box-shadow:0 2px 8px rgba(0,0,0,.08);color:#17223b;font-size:14px;margin:16px 0;width:100%}.Chat-module_messageText__XgNyQ td,.Chat-module_messageText__XgNyQ th{background:none;border-bottom:1px solid #d1dbe8;color:#17223b;padding:10px 16px;text-align:left}.Chat-module_messageText__XgNyQ th{background:linear-gradient(90deg,#eaf3fa 80%,#d1e3f8);border-bottom:2px solid #b5c7de;color:#17223b;font-weight:700}.Chat-module_messageText__XgNyQ tr:last-child td{border-bottom:none}.Chat-module_messageText__XgNyQ tr:nth-child(2n) td{background:#eaf3fa}.Chat-module_messageText__XgNyQ tr:hover td{background:#cbe0f7;transition:background .2s}.Chat-module_messageText__XgNyQ table{border-radius:12px;overflow:hidden}.Chat-module_messageText__XgNyQ td:first-child,.Chat-module_messageText__XgNyQ th:first-child{border-top-left-radius:12px}.Chat-module_messageText__XgNyQ td:last-child,.Chat-module_messageText__XgNyQ th:last-child{border-top-right-radius:12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ:hover{box-shadow:0 4px 16px rgba(0,0,0,.15);transform:translateY(-1px)}.Chat-module_attachments__m1Fts{border-top:1px solid hsla(0,0%,49%,.2);display:flex;flex-wrap:wrap;gap:8px;margin-top:10px;padding-top:10px}.Chat-module_attachment__aE9hK{align-items:center;background:hsla(0,0%,100%,.2);border:1px solid hsla(0,0%,49%,.3);border-radius:12px;color:inherit;display:flex;font-size:13px;gap:6px;max-width:200px;padding:6px 12px;text-decoration:none;transition:all .2s ease}.Chat-module_attachment__aE9hK:hover{background:hsla(0,0%,100%,.3);border-color:hsla(0,0%,49%,.5);transform:translateY(-1px)}.Chat-module_attachmentIcon__BX3Cy{font-size:14px;opacity:.8}.Chat-module_attachmentName__aMx56{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_messageButtons__WaOob{border-top:1px solid hsla(0,0%,49%,.83);display:flex;flex-wrap:wrap;gap:8px;margin-top:12px;padding-top:12px}.Chat-module_messageButton__mRnn-{-webkit-tap-highlight-color:transparent;align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.9);border-radius:16px;cursor:pointer;display:inline-flex;font-size:13px;font-weight:500;padding:8px 14px;touch-action:manipulation;transition:all .2s ease;user-select:none}.Chat-module_messageButton__mRnn-:hover{background:rgba(0,132,255,.1);border-color:rgba(0,132,255,.3);box-shadow:0 2px 8px rgba(0,132,255,.15);transform:translateY(-1px)}.Chat-module_messageButton__mRnn-:active{transform:scale(.98);transition:transform .1s ease}.Chat-module_messageButton__mRnn- p{line-height:inherit;margin:0;padding:0}.Chat-module_messageButton__mRnn- strong{font-weight:600}.Chat-module_messageButton__mRnn- em{font-style:italic}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M{align-items:center;backdrop-filter:blur(10px);background:rgba(0,0,0,.8);border-radius:12px;bottom:-8px;display:flex;gap:2px;min-width:24px;opacity:0;padding:4px 6px;position:absolute;right:8px;transform:translateY(4px);transition:all .3s cubic-bezier(.25,.46,.45,.94);z-index:1}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ:hover .Chat-module_rating__soc3M{opacity:1;transform:translateY(0)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M:hover{background:rgba(0,0,0,.9);box-shadow:0 4px 12px rgba(0,0,0,.3);padding:6px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M span{cursor:pointer;display:inline-block;font-size:16px;transition:transform .2s ease,color .2s ease}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M:hover span{transform:scale(1.1)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{display:flex;flex-direction:column;gap:12px;grid-area:📝;padding:24px;position:relative;width:100%;z-index:10}.Chat-module_Chat__j2eE5.Chat-module_fullPageVisual__zNAEy .Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{backdrop-filter:blur(25px);background:linear-gradient(180deg,hsla(0,0%,100%,0),hsla(0,0%,100%,.8))}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan.Chat-module_dragOver__bkS-g{background:linear-gradient(0deg,rgba(0,132,255,.1) 0,rgba(0,132,255,.05))}.Chat-module_filePreviewContainer__R70hm{display:flex;flex-wrap:wrap;gap:8px;margin-bottom:8px}.Chat-module_filePreview__kq2aX{align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.2);border-radius:8px;display:flex;font-size:12px;gap:8px;padding:8px 12px;transition:all .2s ease}.Chat-module_filePreview__kq2aX:hover{background:hsla(0,0%,49%,.15);border-color:hsla(0,0%,49%,.3)}.Chat-module_fileIcon__zoSKW{font-size:14px;opacity:.7}.Chat-module_fileInfo__wBLi0{display:flex;flex-direction:column;gap:2px;min-width:0}.Chat-module_fileName__bBujo{color:#000;font-weight:500;max-width:150px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_fileSize__ivliq{color:#6b7280;font-size:11px}.Chat-module_removeFileButton__0gakR{align-items:center;background:rgba(255,0,0,.1);border:none;border-radius:50%;color:#f44;cursor:pointer;display:flex;flex-shrink:0;height:20px;justify-content:center;transition:all .2s ease;width:20px}.Chat-module_removeFileButton__0gakR:hover{background:rgba(255,0,0,.2);transform:scale(1.1)}.Chat-module_inputContainer__bPt99{align-items:flex-end;background:#fff;border:1px solid rgba(0,0,0,.08);border-radius:28px;box-shadow:0 4px 20px rgba(0,0,0,.06);display:flex;gap:8px;padding:8px 12px;transition:all .3s cubic-bezier(.25,.46,.45,.94)}.Chat-module_inputContainer__bPt99:focus-within{border-color:var(--brand-color);box-shadow:0 8px 32px rgba(0,132,255,.12);transform:translateY(-2px)}.Chat-module_inputContainer__bPt99 textarea::placeholder{color:#94a3b8;opacity:1}.Chat-module_attachmentButton__qLO47{align-items:center;background:transparent;border:none;border-radius:50%;color:#64748b;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;margin-bottom:2px;transition:all .2s ease;width:40px}.Chat-module_attachmentButton__qLO47:hover:not(:disabled){background:#f1f5f9;color:#0f172a}.Chat-module_attachmentButton__qLO47:disabled{cursor:not-allowed;opacity:.3}.Chat-module_voiceButton__d2zlP{align-items:center;background:transparent;border:none;border-radius:50%;color:#64748b;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;margin-bottom:2px;transition:all .2s ease;width:40px}.Chat-module_voiceButton__d2zlP:hover:not(:disabled){background:#f1f5f9;color:#0f172a}.Chat-module_voiceButtonActive__Uoi3W{animation:Chat-module_voiceRecordingPulse__y2wJ5 1.5s infinite;background:rgba(239,68,68,.1)!important;color:#ef4444!important}@keyframes Chat-module_voiceRecordingPulse__y2wJ5{0%{box-shadow:0 0 0 0 rgba(255,0,0,.4)}70%{box-shadow:0 0 0 10px rgba(255,0,0,0)}to{box-shadow:0 0 0 0 rgba(255,0,0,0)}}.Chat-module_uploadProgress__jBTKe{align-items:center;background:rgba(0,132,255,.1);border:1px solid rgba(0,132,255,.2);border-radius:8px;color:#0084ff;display:flex;font-size:13px;gap:12px;padding:8px 12px}.Chat-module_uploadProgressBar__Gutnt{background:rgba(0,132,255,.2);border-radius:2px;flex:1;height:4px;overflow:hidden}.Chat-module_uploadProgressFill__EgubT{animation:Chat-module_uploadProgress__jBTKe 1.5s ease-in-out infinite;background:linear-gradient(90deg,#0084ff,rgba(0,132,255,.8));border-radius:2px;height:100%}@keyframes Chat-module_uploadProgress__jBTKe{0%{transform:translateX(-100%)}50%{transform:translateX(0)}to{transform:translateX(100%)}}.Chat-module_dragOverlay__SEGoS{align-items:center;backdrop-filter:blur(10px);background:rgba(0,132,255,.1);border:2px dashed rgba(0,132,255,.5);border-radius:12px;bottom:0;display:flex;justify-content:center;left:0;pointer-events:none;position:absolute;right:0;top:0;z-index:20}.Chat-module_dragOverlayContent__gb9kF{align-items:center;color:#0084ff;display:flex;flex-direction:column;font-weight:600;gap:12px;text-align:center}.Chat-module_dragOverlayContent__gb9kF svg{opacity:.7}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{-webkit-tap-highlight-color:transparent;appearance:none;-webkit-appearance:none;background:transparent;border:none;color:#0f172a;flex:1;font-size:15px;line-height:1.5;margin:0;max-height:200px;min-width:100px;outline:none;padding:10px 12px;resize:none;touch-action:manipulation}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea:disabled{cursor:not-allowed;opacity:.6;transform:none}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea::placeholder{color:inherit;opacity:.7}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]{-webkit-tap-highlight-color:transparent;align-items:center;aspect-ratio:1/1;border:none;border-radius:50%!important;box-shadow:0 4px 12px rgba(0,132,255,.25);color:#fff;display:flex;height:40px;justify-content:center;margin:0 0 2px!important;min-height:unset!important;min-width:unset!important;overflow:visible;padding:0!important;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94);user-select:none;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]:hover{box-shadow:0 6px 16px rgba(0,132,255,.35);transform:scale(1.05)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]:active{transform:scale(.95)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action] svg{height:20px;width:20px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button img{height:100%;object-fit:contain;width:50%}.Chat-module_scrollToBottomContainer__5rXpK{align-items:flex-start;display:flex;grid-area:📝;height:100%;justify-content:center;pointer-events:none;width:100%;z-index:20}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1{align-items:center;animation:Chat-module_scrollButtonSlideIn__XnImg .3s ease-out;display:inline-flex;justify-content:center;pointer-events:all;position:relative;transform:translate(-50%,-150%);transition:transform .3s cubic-bezier(.25,.46,.45,.94)}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottom__nzxdZ{-webkit-tap-highlight-color:transparent;align-items:center;backdrop-filter:blur(3px);background:rgba(0,0,0,.5);border:none;border-radius:50%;box-shadow:0 4px 16px rgba(0,0,0,.3);color:#fff;cursor:pointer;display:flex;font-weight:700;height:48px;justify-content:center;outline:none;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94);user-select:none;width:48px}@keyframes Chat-module_scrollButtonSlideIn__XnImg{0%{opacity:0;transform:translate(-50%,20px) scale(.8)}to{opacity:.9;transform:translate(-50%) scale(1)}}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1:hover{transform:translate(-50%,-160%) scale(1.05)}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1:active{transform:translate(-50%,-160%) scale(.95);transition:transform .1s ease}.Chat-module_scrollToBottomBadge__N6T6-{align-items:center;background:#2563eb;border-radius:999px;box-shadow:0 2px 10px rgba(37,99,235,.4);color:#fff;display:inline-flex;font-size:10px;font-weight:600;justify-content:center;letter-spacing:.04em;min-width:32px;padding:2px 6px;pointer-events:none;position:absolute;right:-8px;text-transform:uppercase;top:-8px;white-space:nowrap}.Chat-module_ratingModal__XVKYm{align-items:center;animation:Chat-module_modalFadeIn__RPc3w .3s ease-out;backdrop-filter:blur(8px);background-color:rgba(0,0,0,.6);bottom:0;display:flex;justify-content:center;left:0;position:fixed;right:0;top:0;z-index:1000}@keyframes Chat-module_modalFadeIn__RPc3w{0%{backdrop-filter:blur(0);opacity:0}to{backdrop-filter:blur(8px);opacity:1}}.Chat-module_ratingModalContent__CCdq7{animation:Chat-module_modalSlideIn__XXtgN .3s cubic-bezier(.25,.46,.45,.94);backdrop-filter:blur(20px);background:#fff;border:1px solid hsla(0,0%,49%,.1);border-radius:16px;box-shadow:0 20px 60px rgba(0,0,0,.3);color:#0f1724;font-family:Arial,Helvetica,sans-serif,OpenMojiBlack;max-width:480px;padding:32px;width:90%}@keyframes Chat-module_modalSlideIn__XXtgN{0%{opacity:0;transform:translateY(20px) scale(.95)}to{opacity:1;transform:translateY(0) scale(1)}}.Chat-module_ratingModalContent__CCdq7 h3{color:#000;font-size:20px;font-weight:600;margin:0 0 24px;text-align:center}.Chat-module_stars__PCzNO{display:flex;gap:8px;justify-content:center;margin-bottom:24px}.Chat-module_stars__PCzNO span{border-radius:8px;cursor:pointer;font-size:28px;padding:4px;transition:all .2s ease}.Chat-module_stars__PCzNO span:hover{background:rgba(255,215,0,.1);transform:scale(1.2)}.Chat-module_ratingModalStar__XkbHr{cursor:pointer;font-size:24px;transition:color .2s}.Chat-module_toolCallModal__uNaIK{max-width:800px;overflow:hidden;padding:0;position:relative}.Chat-module_searchModalHeader__Y8V-w{align-items:center;background:#fdfdfd;border-bottom:1px solid #eee;display:flex;gap:16px;padding:24px 72px 24px 32px}.Chat-module_selfLearningModalHeader__PDIt1{align-items:center;background:linear-gradient(135deg,#f8fafc,#fef9c3 45%,#ecfccb);border-bottom:1px solid #e2e8f0;gap:18px}.Chat-module_selfLearningAvatarGroup__jgy2k{align-items:center;display:flex;gap:10px}.Chat-module_selfLearningAvatar__NHXKm{align-items:center;background:#e2e8f0;background-position:50%;background-size:cover;border-radius:50%;box-shadow:0 10px 20px rgba(15,23,42,.18);color:#0f172a;display:inline-flex;font-size:18px;font-weight:600;height:48px;justify-content:center;letter-spacing:.02em;width:48px}.Chat-module_selfLearningAvatarInitial__cuw19{line-height:1}.Chat-module_selfLearningTeacher__fGhX8{background:linear-gradient(135deg,#fde68a,#86efac)}.Chat-module_selfLearningHeaderText__LTwHL{display:flex;flex-direction:column;gap:6px;text-align:left}.Chat-module_selfLearningTitle__G0SAk{color:#0f172a;font-size:20px;font-weight:600;margin:0}.Chat-module_selfLearningMetaRow__z4Tg0{align-items:center;color:#64748b;display:flex;flex-wrap:wrap;font-size:13px;gap:10px;margin-bottom:12px}.Chat-module_selfLearningMeta__58Pet{color:#64748b;font-size:13px}.Chat-module_selfLearningMetaChip__0go88{background:#e2e8f0;border-radius:999px;color:#0f172a;font-size:12px;font-weight:600;padding:4px 10px}.Chat-module_selfLearningCommitments__7ECnr{background:#f8fafc;border:1px solid #e2e8f0;border-radius:16px;padding:16px}.Chat-module_selfLearningCommitmentsLabel__OkDO1{color:#94a3b8;font-size:12px;font-weight:600;letter-spacing:.08em;text-transform:uppercase}.Chat-module_selfLearningBookEditor__CZy5T{border-radius:12px;margin-top:12px;overflow:hidden}.Chat-module_selfLearningEmpty__vCKHL{color:#475569;font-size:13px;margin-top:12px}.Chat-module_modalCloseButton__AAfWS{align-items:center;background:#fff;border:1px solid #e2e8f0;border-radius:999px;box-shadow:0 6px 16px rgba(15,23,42,.08);color:#475569;cursor:pointer;display:inline-flex;height:36px;justify-content:center;position:absolute;right:16px;top:16px;transition:background .2s ease,color .2s ease,box-shadow .2s ease,transform .2s ease;width:36px;z-index:1}.Chat-module_modalCloseButton__AAfWS:hover{background:#f8fafc;box-shadow:0 10px 24px rgba(15,23,42,.12);color:#0f172a;transform:translateY(-1px)}.Chat-module_modalCloseButton__AAfWS:focus-visible{outline:2px solid #94a3b8;outline-offset:2px}.Chat-module_teamModalHeader__SB5NL{flex-wrap:wrap;gap:12px}.Chat-module_teamHeaderParticipants__KS7aV{align-items:center;display:flex;flex-wrap:wrap;gap:12px}.Chat-module_teamHeaderProfile__xRlcS{align-items:center;color:inherit;display:inline-flex;gap:12px;text-decoration:none}.Chat-module_teamHeaderProfileLink__fYTfT:hover{color:#0f172a}.Chat-module_teamHeaderAvatar__vXcTp{background-position:50%;background-repeat:no-repeat;background-size:cover;border:1px solid #e2e8f0;border-radius:999px;box-shadow:0 6px 14px rgba(15,23,42,.08);height:40px;width:40px}.Chat-module_teamHeaderName__ygJR9{color:#0f172a;font-size:18px;font-weight:600}.Chat-module_teamHeaderDivider__cnRMd{color:#94a3b8;font-size:14px;font-weight:600}.Chat-module_searchModalIcon__AR5KY{align-items:center;color:#475569;display:inline-flex;font-size:24px;justify-content:center}.Chat-module_searchModalQuery__5x-Ra{color:#202124!important;font-size:22px!important;font-weight:500!important;margin:0!important;text-align:left!important}.Chat-module_searchModalContent__mWLIE{background:#fff;max-height:60vh;overflow-y:auto;padding:24px 32px}.Chat-module_teamChatContainer__xdLtU{background:#f8fafc;border:1px solid #e5e7eb;border-radius:16px;margin-bottom:20px;overflow:hidden}.Chat-module_teamToolCallSection__FH9l7{display:flex;flex-direction:column;gap:16px}.Chat-module_teamToolCallGroup__PO6rl{display:flex;flex-direction:column;gap:8px}.Chat-module_teamToolCallHeading__G-JSj{color:#64748b;font-size:12px;font-weight:600;letter-spacing:.08em;text-transform:uppercase}.Chat-module_teamToolCallChips__hRWqh{display:flex;flex-wrap:wrap;gap:8px}.Chat-module_teamToolCallDetails__ZaTtt{background:#f8fafc;border:1px solid rgba(148,163,184,.4);border-radius:12px;padding:16px}.Chat-module_teamToolCallDetailsHeader__FPIgI{align-items:center;display:flex;gap:12px;justify-content:space-between;margin-bottom:12px}.Chat-module_teamToolCallDetailsTitle__50SeL{align-items:center;color:#0f172a;display:flex;font-size:14px;font-weight:600;gap:6px}.Chat-module_teamToolCallDetailsClear__8yvu7{background:rgba(148,163,184,.2);border:1px solid rgba(148,163,184,.4);border-radius:999px;color:#475569;cursor:pointer;font-size:12px;padding:4px 10px;transition:all .2s ease}.Chat-module_teamToolCallDetailsClear__8yvu7:hover{background:rgba(148,163,184,.35)}.Chat-module_searchResultsList__xVDUZ{display:flex;flex-direction:column;gap:28px;text-align:left}.Chat-module_searchResultsRaw__i3GkD{max-width:650px;text-align:left}.Chat-module_searchResultItem__bTzq5{max-width:650px}.Chat-module_searchResultUrl__SbrH4{color:#202124;font-size:14px;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_searchResultUrl__SbrH4 a{color:#202124;text-decoration:none}.Chat-module_searchResultTitle__XS7zN{font-size:20px!important;font-weight:400!important;line-height:1.3;margin:0 0 4px!important}.Chat-module_searchResultTitle__XS7zN a{color:#1a0dab;text-decoration:none}.Chat-module_searchResultTitle__XS7zN a:hover{text-decoration:underline}.Chat-module_searchResultSnippet__Wsun2{color:#4d5156;font-size:14px;line-height:1.58;margin:0}.Chat-module_noResults__AAv6s{color:#70757a;font-style:italic;padding:40px;text-align:center}.Chat-module_citationDetails__2v2e1,.Chat-module_toolCallDetails__WUFlD{margin:20px 0;text-align:left}.Chat-module_citationPreview__rGhCR{background:#f8f8f8;border:1px solid #ddd;border-radius:8px;margin-top:16px;overflow:hidden}.Chat-module_citationIframe__ngzFY{border:none;display:block;height:60vh;min-height:400px;width:100%}.Chat-module_citationMetadata__DPaCF{background:#f8f8f8;border:1px solid #eee;border-radius:8px;margin-bottom:20px;padding:16px}.Chat-module_citationMetadata__DPaCF p{font-size:.9em;margin:8px 0}.Chat-module_citationMetadata__DPaCF strong{color:#333}.Chat-module_citationMetadata__DPaCF a{color:#007bff;text-decoration:none;word-break:break-all}.Chat-module_citationMetadata__DPaCF a:hover{text-decoration:underline}.Chat-module_citationExcerpt__VU4BS{background:#fff;border:1px solid #ddd;border-radius:8px;max-height:400px;overflow-y:auto;padding:16px}.Chat-module_citationExcerpt__VU4BS h4{color:#333;font-size:1em;margin:0 0 12px}.Chat-module_citationHint__5AkLx{font-size:.85em;margin-top:8px;opacity:.7}.Chat-module_toolCallDataContainer__rdEzC{background:#f8f8f8;border:1px solid #eee;border-radius:6px;margin-bottom:15px;max-height:300px;overflow-y:auto;padding:12px}.Chat-module_toolCallData__UauCO{font-size:.85em;margin:0;white-space:pre-wrap;word-break:break-word}.Chat-module_toolCallArgsList__LD3xH{list-style:none;margin:0;padding:0}.Chat-module_toolCallArgsList__LD3xH li{font-size:.9em;margin-bottom:5px}.Chat-module_toolCallModal__uNaIK .Chat-module_toolCallData__UauCO{background:#f8fafc;color:#0f172a}.Chat-module_ratingInput__z8Pv-{background:hsla(0,0%,49%,.05);border:2px solid hsla(0,0%,49%,.1);border-radius:12px;color:#0b1220;font-size:14px;line-height:1.5;margin-bottom:18px;min-height:100px;padding:16px;resize:vertical;transition:all .2s ease;width:100%}.Chat-module_ratingInput__z8Pv-:focus{background:hsla(0,0%,49%,.08);border-color:#0084ff;box-shadow:0 0 0 4px rgba(0,132,255,.1);outline:none}.Chat-module_ratingInput__z8Pv-[readonly]{background:hsla(0,0%,49%,.01);border:1px solid hsla(0,0%,49%,.5)}.Chat-module_ratingActions__nXcss{display:flex;gap:12px;justify-content:flex-end}.Chat-module_ratingActions__nXcss button{border:none;border-radius:8px;cursor:pointer;font-size:14px;font-weight:500;min-width:80px;padding:12px 24px;transition:all .2s ease}.Chat-module_ratingActions__nXcss button:first-child{background-color:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.2);color:#0b1220}.Chat-module_ratingActions__nXcss button:first-child:hover:not(:disabled){background-color:hsla(0,0%,49%,.2);color:#0b1220}.Chat-module_ratingActions__nXcss button:last-child{background:linear-gradient(135deg,#0084ff,#06c);color:#fff}.Chat-module_ratingActions__nXcss button:last-child:hover:not(:disabled){background:linear-gradient(135deg,#0071d1,#0052a3);box-shadow:0 4px 12px rgba(0,132,255,.3);transform:translateY(-1px)}.Chat-module_ratingActions__nXcss button:last-child:disabled{cursor:not-allowed;opacity:.5}.Chat-module_downloadButton__8--bk{align-items:center;background:linear-gradient(135deg,#0084ff,#06c);border:none;border-radius:8px;box-shadow:0 2px 8px rgba(0,132,255,.2);color:#fff;cursor:pointer;display:inline-flex;font-size:14px;font-weight:600;gap:8px;justify-content:center;padding:12px 24px;transition:all .2s ease}.Chat-module_downloadButton__8--bk:hover{background:linear-gradient(135deg,#09f,#07d);box-shadow:0 4px 16px rgba(0,132,255,.3);transform:translateY(-2px)}.Chat-module_downloadButton__8--bk:active{box-shadow:0 2px 8px rgba(0,132,255,.2);transform:translateY(0)}.Chat-module_downloadButton__8--bk svg{flex-shrink:0}.Chat-module_downloadButton__8--bk span{font-weight:600}.Chat-module_chatButton__d9VgA{-webkit-tap-highlight-color:transparent!important;align-items:center!important;backdrop-filter:blur(20px)!important;background:linear-gradient(135deg,#0084ff,#06c)!important;border:none!important;border-radius:20px!important;box-shadow:0 4px 16px rgba(0,132,255,.3)!important;color:#fff!important;cursor:pointer!important;display:inline-flex!important;font-size:13px!important;font-weight:600!important;gap:8px!important;justify-content:center!important;line-height:1.3!important;margin:0!important;min-height:40px!important;min-width:110px!important;overflow:hidden!important;padding:12px 16px!important;position:relative!important;touch-action:manipulation!important;transition:all .3s cubic-bezier(.25,.46,.45,.94)!important;user-select:none!important}.Chat-module_chatButton__d9VgA:before{background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.2),transparent);content:\"\";height:100%;left:-100%;position:absolute;top:0;transition:left .5s ease;width:100%}.Chat-module_chatButton__d9VgA:hover:before{left:100%}.Chat-module_chatButton__d9VgA:hover:not(:disabled){background:linear-gradient(135deg,#09f,#07d);box-shadow:0 8px 24px rgba(0,132,255,.4);transform:translateY(-2px) scale(1.02)}.Chat-module_chatButton__d9VgA:active{box-shadow:0 4px 16px rgba(0,132,255,.3);transform:scale(.98) translateY(-1px);transition:transform .1s ease}.Chat-module_chatButton__d9VgA:focus{box-shadow:0 4px 16px rgba(0,132,255,.3),0 0 0 3px rgba(0,132,255,.3);outline:none}.Chat-module_chatButton__d9VgA svg{filter:drop-shadow(0 1px 2px rgba(0,0,0,.2));flex-shrink:0;height:16px;opacity:1;transition:all .3s cubic-bezier(.25,.46,.45,.94);width:16px}.Chat-module_chatButton__d9VgA:hover svg{filter:drop-shadow(0 2px 4px rgba(0,0,0,.3));transform:rotate(-90deg) scale(1.1)}.Chat-module_chatButtonText__RkGB-{font-weight:600;opacity:1;text-shadow:0 1px 2px rgba(0,0,0,.1);transition:all .2s ease;white-space:nowrap}.Chat-module_chatButton__d9VgA:hover .Chat-module_chatButtonText__RkGB-{transform:translateX(1px)}.Chat-module_useTemplateButton__xcJNR{-webkit-tap-highlight-color:transparent!important;align-items:center!important;backdrop-filter:blur(20px)!important;background:linear-gradient(135deg,#0084ff,#06c)!important;border:none!important;border-radius:20px!important;box-shadow:0 4px 16px rgba(0,132,255,.3)!important;color:#fff!important;cursor:pointer!important;display:inline-flex!important;font-size:13px!important;font-weight:600!important;gap:8px!important;justify-content:center!important;line-height:1.3!important;margin:0!important;min-height:40px!important;min-width:110px!important;overflow:hidden!important;padding:12px 16px!important;position:relative!important;touch-action:manipulation!important;transition:all .3s cubic-bezier(.25,.46,.45,.94)!important;user-select:none!important}.Chat-module_useTemplateButton__xcJNR:before{background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.2),transparent);content:\"\";height:100%;left:-100%;position:absolute;top:0;transition:left .5s ease;width:100%}.Chat-module_useTemplateButton__xcJNR:hover:before{left:100%}.Chat-module_useTemplateButton__xcJNR:hover:not(:disabled){background:linear-gradient(135deg,#09f,#07d);box-shadow:0 8px 24px rgba(0,132,255,.4);transform:translateY(-2px) scale(1.02)}.Chat-module_useTemplateButton__xcJNR:active{box-shadow:0 4px 16px rgba(0,132,255,.3);transform:scale(.98) translateY(-1px);transition:transform .1s ease}.Chat-module_useTemplateButton__xcJNR:focus{box-shadow:0 4px 16px rgba(0,132,255,.3),0 0 0 3px rgba(0,132,255,.3);outline:none}.Chat-module_useTemplateButton__xcJNR svg{filter:drop-shadow(0 1px 2px rgba(0,0,0,.2));flex-shrink:0;height:16px;opacity:1;transition:all .3s cubic-bezier(.25,.46,.45,.94);width:16px}.Chat-module_useTemplateButton__xcJNR:hover svg{filter:drop-shadow(0 2px 4px rgba(0,0,0,.3));transform:scale(1.1)}.Chat-module_useTemplateButton__xcJNR:hover .Chat-module_chatButtonText__RkGB-{transform:translateX(1px)}.Chat-module_saveButtonContainer__lSNUJ{display:inline-block;position:relative}.Chat-module_saveMenu__-ph8y{background:#fff;border:1px solid #ddd;box-shadow:0 2px 8px rgba(0,0,0,.08);left:0;min-width:120px;position:absolute;top:100%;z-index:10}.Chat-module_saveMenuItem__ISApL{background:none;border:none;color:#111;cursor:pointer;display:block;padding:8px 16px;text-align:left;width:100%}.Chat-module_saveMenuItem__ISApL:hover{background-color:#f0f0f0}.Chat-module_saveMenuDivider__3ktTM{background-color:#ddd;height:1px;margin:4px 0}.Chat-module_chatFeedbackPanel__gfOzk{background:#fff;border:1px solid #e5e7eb;border-radius:14px;box-shadow:0 8px 18px rgba(15,23,42,.08);display:flex;flex-direction:column;gap:6px;padding:6px 8px;width:min(260px,100%)}.Chat-module_chatFeedbackToggle__IeOM6{align-items:center;background:transparent;border:none;border-radius:12px;cursor:pointer;display:flex;gap:10px;padding:6px 8px;text-align:left;transition:background .2s ease,transform .2s ease;width:100%}.Chat-module_chatFeedbackToggle__IeOM6:hover{background:#f5f5f5;transform:translateY(-.5px)}.Chat-module_chatFeedbackToggle__IeOM6:focus-visible{outline:2px solid #2563eb;outline-offset:2px}.Chat-module_chatFeedbackIcon__XcB7A{background:#eef2ff;border-radius:10px;color:#6366f1;display:grid;flex-shrink:0;height:34px;place-items:center;transition:background .2s ease,color .2s ease;width:34px}.Chat-module_chatFeedbackToggle__IeOM6[data-enabled=true] .Chat-module_chatFeedbackIcon__XcB7A{background:#e0f2fe;color:#0284c7}.Chat-module_chatFeedbackToggle__IeOM6[data-enabled=false] .Chat-module_chatFeedbackIcon__XcB7A{background:#f3f4f6;color:#94a3b8}.Chat-module_chatFeedbackMeta__6t2be{display:flex;flex:1;flex-direction:column;gap:2px}.Chat-module_chatFeedbackLabel__qQgH3{color:#0f172a;font-size:.85rem;font-weight:600}.Chat-module_chatFeedbackDescription__I9bNg{color:#64748b;font-size:.75rem}.Chat-module_chatFeedbackIndicator__uHCr7{border-radius:999px;font-size:.72rem;font-weight:700;letter-spacing:.08em;padding:2px 8px;text-transform:uppercase}.Chat-module_chatFeedbackIndicatorActive__bkw-w{background:#dcfce7;border:1px solid #34d399;color:#166534}.Chat-module_chatFeedbackIndicatorInactive__qMBkh{background:#fef9c3;border:1px solid #facc15;color:#713f12}.Chat-module_pauseButton__eeu7K{background:linear-gradient(135deg,#ffb347,#ff8c42)!important}.Chat-module_pauseButton__eeu7K:hover:not(:disabled){background:linear-gradient(135deg,#ffc067,#ff9e5f)!important}.Chat-module_pauseButton__eeu7K.Chat-module_pausing__pTx8b{cursor:wait!important;opacity:.6!important}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya{background:linear-gradient(135deg,#10b981,#059669)!important}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya:hover:not(:disabled){background:linear-gradient(135deg,#34d399,#059669)!important;box-shadow:0 8px 24px rgba(16,185,129,.35);transform:translateY(-2px) scale(1.02)}.Chat-module_pauseButton__eeu7K svg{transition:transform .3s ease}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya svg{transform:scale(1.1)}.Chat-module_pauseButton__eeu7K.Chat-module_pausing__pTx8b svg{opacity:.8}.Chat-module_voiceCallIndicatorBar__N2sWN{align-items:center;backdrop-filter:blur(10px);background:linear-gradient(135deg,rgba(34,197,94,.1),rgba(16,185,129,.1));border-bottom:1px solid rgba(34,197,94,.3);display:flex;grid-area:🟦;justify-content:center;padding:12px 20px;width:100%}.Chat-module_voiceCallIndicator__tsaaG{align-items:center;backdrop-filter:blur(10px);background:linear-gradient(135deg,rgba(34,197,94,.2),rgba(16,185,129,.2));border:1px solid rgba(34,197,94,.4);border-radius:20px;box-shadow:0 2px 8px rgba(34,197,94,.2);color:#10b981;display:inline-flex;font-size:13px;font-weight:600;gap:8px;padding:8px 16px;position:relative}.Chat-module_voiceCallIndicator__tsaaG svg{animation:Chat-module_voiceCallIconPulse__zZbJn 2s ease-in-out infinite;flex-shrink:0;height:16px;width:16px}.Chat-module_voiceCallIndicator__tsaaG span{font-weight:600;text-shadow:0 1px 2px rgba(0,0,0,.1)}.Chat-module_voiceCallPulse__XcGU4{animation:Chat-module_voiceCallPulse__XcGU4 1.5s ease-in-out infinite;background:#10b981;border-radius:50%;height:8px;position:absolute;right:8px;top:50%;transform:translateY(-50%);width:8px}@keyframes Chat-module_voiceCallIconPulse__zZbJn{0%,to{opacity:1;transform:scale(1)}50%{opacity:.8;transform:scale(1.1)}}@keyframes Chat-module_voiceCallPulse__XcGU4{0%,to{opacity:1;transform:translateY(-50%) scale(1)}50%{opacity:.6;transform:translateY(-50%) scale(1.3)}}.Chat-module_chatMessage__nmLaZ .Chat-module_voiceCallIndicator__tsaaG{border-radius:16px;font-size:12px;margin-bottom:8px;padding:6px 12px}.Chat-module_chatMessage__nmLaZ .Chat-module_voiceCallIndicator__tsaaG svg{height:14px;width:14px}.Chat-module_feedbackStatus__-df7l{word-wrap:break-word;animation:Chat-module_feedbackStatusSlideIn__n6bbB .28s ease-out;backdrop-filter:blur(20px);border-radius:14px;box-shadow:0 18px 32px rgba(15,23,42,.35);font-weight:500;max-width:340px;padding:14px 22px;pointer-events:none;position:fixed;right:20px;top:20px;z-index:10000}.Chat-module_feedbackStatusSuccess__Ax2q0{background:linear-gradient(135deg,#10b981,#059669);color:#f8fafc}.Chat-module_feedbackStatusError__dnv6d{background:linear-gradient(135deg,#ef4444,#dc2626);color:#fef2f2}@keyframes Chat-module_feedbackStatusSlideIn__n6bbB{0%{opacity:0;transform:translateX(100%)}to{opacity:1;transform:translateX(0)}}@media (max-width:768px){.Chat-module_Chat__j2eE5{--chat-message-min-width:100px;--chat-message-max-width:min(85%,560px);--chat-message-avatar-gap:10px}.Chat-module_actions__gTZ5T{gap:6px;margin:12px 16px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{padding:16px 12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ{margin-bottom:16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border-radius:18px;font-size:14px;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:36px;margin:0 0 4px;width:36px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{gap:10px;padding:16px 12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-radius:22px;font-size:16px;padding:14px 18px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{height:44px;width:44px}.Chat-module_scrollToBottom__nzxdZ{font-size:18px;height:44px;top:calc(100% - 160px);width:44px}.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_ratingModalContent__CCdq7{border-radius:16px;margin:16px;max-height:80vh;overflow-y:auto;padding:24px 20px}.Chat-module_stars__PCzNO{gap:6px;margin-bottom:20px}.Chat-module_stars__PCzNO span{font-size:32px;padding:8px}.Chat-module_ratingActions__nXcss{flex-direction:column-reverse;gap:8px}.Chat-module_ratingActions__nXcss button{border-radius:10px;font-size:16px;padding:14px;width:100%}.Chat-module_selfLearningModalHeader__PDIt1{align-items:flex-start;flex-direction:column}.Chat-module_selfLearningAvatarGroup__jgy2k{justify-content:flex-start}.Chat-module_selfLearningMetaRow__z4Tg0{align-items:flex-start}}@media (max-width:480px){.Chat-module_Chat__j2eE5{--chat-message-min-width:80px;--chat-message-max-width:min(90%,520px);--chat-message-avatar-gap:8px}.Chat-module_actions__gTZ5T{gap:4px;margin:8px 12px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{padding:12px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border-radius:16px;font-size:14px;padding:10px 14px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:32px;margin:0 0 4px;width:32px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{gap:8px;padding:12px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-radius:20px;font-size:16px;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{height:40px;width:40px}.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_scrollToBottom__nzxdZ{font-size:16px;height:40px;top:calc(100% - 140px);width:40px}.Chat-module_ratingModal__XVKYm{align-items:flex-end;padding:0}.Chat-module_ratingModalContent__CCdq7{border-radius:20px 20px 0 0;margin:0;max-height:70vh;padding:24px 16px 20px;width:100%}}@media (prefers-reduced-motion:reduce){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ,.Chat-module_feedbackStatus__-df7l,.Chat-module_ratingModalContent__CCdq7,.Chat-module_ratingModal__XVKYm,.Chat-module_scrollToBottom__nzxdZ{animation:none}.Chat-module_chatButton__d9VgA,.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button,.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea,.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{transition:none}}@media (prefers-contrast:high){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border:2px solid}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-width:3px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{border:2px solid}}.Chat-module_emailContainer__TBxXZ{display:flex;flex-direction:column;gap:16px;margin-bottom:20px}.Chat-module_emailModalHeader__Z4-CJ{background:#f8fafc}.Chat-module_emailHeaderText__1dR6R{display:flex;flex-direction:column;gap:2px}.Chat-module_emailHeaderLabel__7iAEH{color:#64748b;font-size:12px;letter-spacing:.08em;text-transform:uppercase}.Chat-module_emailMetadata__8XXGf{background:#fff;border:1px solid #e2e8f0;border-radius:12px;box-shadow:0 1px 4px rgba(15,23,42,.08);margin-bottom:16px;padding:18px 20px}.Chat-module_emailField__pcb03{display:flex;font-size:14px;gap:8px;line-height:1.5;margin-bottom:8px}.Chat-module_emailField__pcb03:last-child{margin-bottom:0}.Chat-module_emailField__pcb03 strong{color:#64748b;font-weight:600;min-width:60px}.Chat-module_emailRecipients__YVVys{color:#0f172a;word-break:break-word}.Chat-module_emailBody__OBt-w{text-align:left}.Chat-module_emailBody__OBt-w strong{color:#64748b;display:block;font-size:14px;font-weight:600;margin-bottom:12px}.Chat-module_emailBodyContent__-vFo2{background:#fff;border:1px solid #e2e8f0;border-radius:12px;box-shadow:0 1px 4px rgba(15,23,42,.08);line-height:1.6;max-height:400px;overflow-y:auto;padding:22px}.Chat-module_emailBodyContent__-vFo2 p{margin:0 0 12px}.Chat-module_emailBodyContent__-vFo2 p:last-child{margin-bottom:0}.Chat-module_emailStatus__T4UuL{color:#0f172a;font-weight:600;text-transform:capitalize}\n/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Chat.module.css"],"names":[],"mappings":"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","file":"Chat.module.css","sourcesContent":["@font-face {\n    font-family: 'OpenMojiBlack';\n    src: url('https://s6.ptbk.io/fonts/OpenMoji-black-glyf.woff2') format('woff2'); /* <- TODO: [🐱‍🚀] Dynamically load from /servers.ts */\n    unicode-range: U+23, U+2A, U+2D, U+30-39, U+A9, U+AE, U+200D, U+203C, U+2049, U+20E3, U+2117, U+2120, U+2122, U+2139,\n        U+2194-2199, U+21A9, U+21AA, U+229C, U+231A, U+231B, U+2328, U+23CF, U+23E9-23F3, U+23F8-23FE, U+24C2, U+25A1,\n        U+25AA-25AE, U+25B6, U+25C0, U+25C9, U+25D0, U+25D1, U+25E7-25EA, U+25ED, U+25EE, U+25FB-25FE, U+2600-2605,\n        U+260E, U+2611, U+2614, U+2615, U+2618, U+261D, U+2620, U+2622, U+2623, U+2626, U+262A, U+262E, U+262F,\n        U+2638-263A, U+2640, U+2642, U+2648-2653, U+265F, U+2660, U+2663, U+2665, U+2666, U+2668, U+267B, U+267E, U+267F,\n        U+2691-2697, U+2699, U+269B, U+269C, U+26A0, U+26A1, U+26A7, U+26AA, U+26AB, U+26B0, U+26B1, U+26BD, U+26BE,\n        U+26C4, U+26C5, U+26C8, U+26CE, U+26CF, U+26D1, U+26D3, U+26D4, U+26E9, U+26EA, U+26F0-26F5, U+26F7-26FA, U+26FD,\n        U+2702, U+2705, U+2708-270D, U+270F, U+2712, U+2714, U+2716, U+271D, U+2721, U+2728, U+2733, U+2734, U+2744,\n        U+2747, U+274C, U+274E, U+2753-2755, U+2757, U+2763, U+2764, U+2795-2797, U+27A1, U+27B0, U+27BF, U+2934, U+2935,\n        U+2B05-2B07, U+2B0C, U+2B0D, U+2B1B, U+2B1C, U+2B1F-2B24, U+2B2E, U+2B2F, U+2B50, U+2B55, U+2B58, U+2B8F,\n        U+2BBA-2BBC, U+2BC3, U+2BC4, U+2BEA, U+2BEB, U+3030, U+303D, U+3297, U+3299, U+E000-E009, U+E010, U+E011,\n        U+E040-E06D, U+E080-E0B4, U+E0C0-E0CC, U+E0FF-E10D, U+E140-E14A, U+E150-E157, U+E181-E189, U+E1C0-E1C4,\n        U+E1C6-E1D9, U+E200-E216, U+E240-E269, U+E280-E283, U+E2C0-E2C4, U+E2C6-E2DA, U+E300-E303, U+E305-E30F,\n        U+E312-E316, U+E318-E322, U+E324-E329, U+E32B, U+E340-E348, U+E380, U+E381, U+F000, U+F77A, U+F8FF, U+FE0F,\n        U+1F004, U+1F0CF, U+1F10D-1F10F, U+1F12F, U+1F16D-1F171, U+1F17E, U+1F17F, U+1F18E, U+1F191-1F19A, U+1F1E6-1F1FF,\n        U+1F201, U+1F202, U+1F21A, U+1F22F, U+1F232-1F23A, U+1F250, U+1F251, U+1F260-1F265, U+1F300-1F321, U+1F324-1F393,\n        U+1F396, U+1F397, U+1F399-1F39B, U+1F39E-1F3F0, U+1F3F3-1F3F5, U+1F3F7-1F4FD, U+1F4FF-1F53D, U+1F549-1F54E,\n        U+1F550-1F567, U+1F56F, U+1F570, U+1F573-1F57A, U+1F587, U+1F58A-1F58D, U+1F590, U+1F595, U+1F596, U+1F5A4,\n        U+1F5A5, U+1F5A8, U+1F5B1, U+1F5B2, U+1F5BC, U+1F5C2-1F5C4, U+1F5D1-1F5D3, U+1F5DC-1F5DE, U+1F5E1, U+1F5E3,\n        U+1F5E8, U+1F5EF, U+1F5F3, U+1F5FA-1F64F, U+1F680-1F6C5, U+1F6CB-1F6D2, U+1F6D5-1F6D7, U+1F6DC-1F6E5, U+1F6E9,\n        U+1F6EB, U+1F6EC, U+1F6F0, U+1F6F3-1F6FC, U+1F7E0-1F7EB, U+1F7F0, U+1F90C-1F93A, U+1F93C-1F945, U+1F947-1F9FF,\n        U+1FA70-1FA7C, U+1FA80-1FA88, U+1FA90-1FABD, U+1FABF-1FAC5, U+1FACE-1FADB, U+1FAE0-1FAE8, U+1FAF0-1FAF8,\n        U+1FBC5-1FBC9, U+E0061-E0067, U+E0069, U+E006C-E0079, U+E007F;\n}\n\n.copiedToClipboardMessage {\n    position: fixed;\n    top: 32px;\n    left: 50%;\n    transform: translateX(-50%);\n    background: #222;\n    color: #fff;\n    padding: 10px 24px;\n    border-radius: 8px;\n    font-size: 1.1em;\n    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.18);\n    opacity: 0.97;\n    pointer-events: none;\n    z-index: 9999;\n}\n\n.Chat {\n    width: 100%;\n    height: 100%;\n    display: flex;\n    flex-direction: column;\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiBlack';\n\n    --chat-message-min-width: 150px;\n    --chat-message-max-width: min(70%, 600px);\n    --chat-message-avatar-gap: 12px;\n    /* <- TODO: [🧠][🎱] Better, define other fonts */\n}\n\n.chatMainFlow {\n    width: 100%;\n    height: 100%;\n    max-width: 100vw;\n    display: grid;\n    grid-template:\n        '🟦' min-content\n        '💬' 1fr\n        '📝' min-content\n        / 1fr;\n}\n\n.chatMainFlow .chatBar {\n    grid-area: 🟦;\n    width: 100%;\n    padding: 16px 20px;\n    color: #0f1724;\n    background-color: #ffffff;\n    border-bottom: 1px solid rgba(15, 23, 36, 0.06);\n    text-align: center;\n    font-weight: 500;\n}\n\n.TasksInProgress {\n    grid-area: 🟦;\n    width: auto;\n    height: min-content;\n    align-self: center;\n    justify-self: self-end;\n    margin: 8px 16px;\n}\n\n.actions {\n    grid-area: 💬;\n    width: auto;\n    height: min-content;\n    z-index: 40;\n    align-self: self-start;\n    justify-self: self-end;\n    margin: 16px 20px 0;\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    transition: opacity 0.2s ease;\n    will-change: opacity;\n}\n\n.actions.portal {\n    margin: 0;\n}\n\n.actions.left {\n    justify-self: self-start;\n}\n\n.actions.right {\n    justify-self: self-end;\n}\n\n.actionsFaded {\n    opacity: 0.3;\n}\n\n.actionsScrolling {\n    opacity: 0.15;\n    pointer-events: none;\n}\n\n/* Large tablet and small desktop screens */\n@media (max-width: 900px) {\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n}\n\n/* Medium screens */\n@media (max-width: 600px) {\n    .actions {\n        margin: 14px 18px 0;\n        gap: 7px;\n    }\n}\n\n.chatMainFlow .chatChildren {\n    grid-area: 💬;\n    width: 100%;\n    height: 100%;\n    z-index: 300;\n}\n\n/* Chat messages area */\n.chatMainFlow .chatMessages {\n    grid-area: 💬;\n    width: 100%;\n    height: 100%;\n    z-index: 10;\n    padding: 24px 20px 16px;\n    overflow-y: auto;\n    overflow-x: hidden;\n    scroll-behavior: smooth;\n}\n\n.hasActionsAndFirstMessageIsLong {\n    padding-top: 80px;\n}\n\n/* Custom scrollbar styling */\n.chatMainFlow .chatMessages::-webkit-scrollbar {\n    width: 6px;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-track {\n    background: transparent;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-thumb {\n    background: rgba(125, 125, 125, 0.2);\n    border-radius: 3px;\n    transition: all 0.2s ease;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-thumb:hover {\n    background: rgba(125, 125, 125, 0.3);\n}\n\n/* Individual chat message */\n.chatMainFlow .chatMessage {\n    display: flex;\n    margin-bottom: 20px;\n    align-items: flex-end;\n    flex-direction: row;\n    column-gap: var(--chat-message-avatar-gap);\n    position: relative;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    max-width: 100%;\n}\n\n.chatMainFlow .chatMessage .messageStack {\n    display: flex;\n    flex-direction: column;\n    align-items: flex-start;\n    max-width: var(--chat-message-max-width);\n}\n\n.chatMainFlow .chatMessage.isMe .messageStack {\n    align-items: flex-end;\n}\n\n.chatMainFlow .chatMessage .messageStack > * {\n    max-width: 100%;\n}\n\n.messageMeta {\n    display: flex;\n    align-items: center;\n    gap: 6px;\n    margin: 4px 6px 0;\n    font-size: 11px;\n    color: #6b7280;\n    line-height: 1.4;\n}\n\n.chatMainFlow .chatMessage.isMe .messageMeta {\n    align-self: flex-end;\n    text-align: right;\n}\n\n.messageTimestamp {\n    letter-spacing: 0.02em;\n}\n\n.messageDuration {\n    opacity: 0.85;\n}\n\n.participantLabel {\n    font-size: 12px;\n    font-weight: 600;\n    color: #64748b;\n    margin: 0 0 6px;\n}\n\n.chatMainFlow .chatMessage.isMe .participantLabel {\n    text-align: right;\n}\n\n@keyframes messageSlideIn {\n    from {\n        opacity: 0;\n        transform: translateY(20px) scale(0.95);\n    }\n    to {\n        opacity: 1;\n        transform: translateY(0) scale(1);\n    }\n}\n\n.isNotCompleteMessage {\n    /*/\n    outline: 1px dotted #ff0000 !important;\n    /**/\n\n    opacity: 0.7;\n    position: relative;\n}\n\n.NonCompleteMessageFiller {\n    color: transparent;\n}\n\n/* Enhanced loading states for messages */\n.isNotCompleteMessage .messageText::after {\n    content: '';\n    position: absolute;\n    bottom: 8px;\n    right: 12px;\n    width: 20px;\n    height: 4px;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.6), transparent);\n    border-radius: 2px;\n    animation: loadingPulse 1.5s ease-in-out infinite;\n}\n\n.ongoingToolCalls,\n.completedToolCalls {\n    display: flex;\n    flex-direction: row;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 8px;\n    padding-top: 8px;\n    border-top: 1px solid rgba(255, 255, 255, 0.2);\n}\n\n.sourceCitations {\n    display: flex;\n    flex-direction: row;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 8px;\n    padding-top: 8px;\n    border-top: 1px solid rgba(255, 255, 255, 0.2);\n}\n\n.ongoingToolCall,\n.completedToolCall {\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    font-size: 0.8em;\n    padding: 4px 10px;\n    background: rgba(255, 255, 255, 0.15);\n    border: 1px solid rgba(255, 255, 255, 0.2);\n    border-radius: 12px;\n    box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);\n    color: inherit;\n    font-family: 'OpenMojiColor', 'OpenMojiBlack', Arial, Helvetica, sans-serif;\n}\n\n.completedToolCall {\n    cursor: pointer;\n    transition: all 0.2s ease;\n}\n\n.completedToolCall:hover {\n    background: rgba(255, 255, 255, 0.3);\n    transform: translateY(-1px);\n    box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);\n}\n\n.toolCallOrigin {\n    font-size: 0.85em;\n    opacity: 0.75;\n    font-weight: 500;\n    white-space: nowrap;\n}\n\n.toolCallDetails {\n    margin-bottom: 24px;\n    text-align: left;\n}\n\n.toolCallDetails p {\n    margin-bottom: 8px;\n}\n\n.toolCallData {\n    background: #f1f5f9;\n    padding: 12px;\n    border-radius: 8px;\n    font-size: 13px;\n    overflow-x: auto;\n    white-space: pre-wrap;\n    word-break: break-all;\n    max-height: 300px;\n    border: 1px solid #e2e8f0;\n}\n\n.ongoingToolCallSpinner {\n    width: 14px;\n    height: 14px;\n    border: 2px solid rgba(255, 255, 255, 0.3);\n    border-top-color: #fff;\n    border-radius: 50%;\n    animation: toolCallSpinner 0.8s linear infinite;\n}\n\n.ongoingToolCallName {\n    font-weight: 500;\n}\n\n@keyframes toolCallSpinner {\n    to {\n        transform: rotate(360deg);\n    }\n}\n\n@keyframes loadingPulse {\n    0%,\n    100% {\n        opacity: 0.3;\n        transform: scaleX(0.8);\n    }\n    50% {\n        opacity: 1;\n        transform: scaleX(1.2);\n    }\n}\n\n/* Typing indicator for AI messages */\n.typingIndicator {\n    display: flex;\n    align-items: flex-end;\n    margin-bottom: 20px;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.typingIndicator .avatar {\n    width: 40px;\n    height: 40px;\n    margin: 0 12px 4px;\n    flex-shrink: 0;\n}\n\n/* [㊗️]\n.typingIndicator .avatar img {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    border-radius: 50%;\n    object-fit: cover;\n    background-color: #eef6fb;\n    border: 2px solid rgba(125, 125, 125, 0.1);\n}\n*/\n\n.typingBubble {\n    padding: 16px 20px;\n    border-radius: 20px;\n    border-bottom-left-radius: 6px;\n    border: 1px solid rgba(125, 125, 125, 0.1);\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);\n    backdrop-filter: blur(10px);\n    display: flex;\n    align-items: center;\n    gap: 4px;\n    min-height: 24px;\n}\n\n.typingDots {\n    display: flex;\n    gap: 4px;\n    align-items: center;\n}\n\n.typingDot {\n    width: 8px;\n    height: 8px;\n    border-radius: 50%;\n    background: linear-gradient(135deg, #6b7280 0%, rgba(125, 125, 125, 0.6) 100%);\n    animation: typingBounce 1.4s infinite ease-in-out;\n    box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);\n}\n\n.typingDot:nth-child(1) {\n    animation-delay: -0.32s;\n}\n\n.typingDot:nth-child(2) {\n    animation-delay: -0.16s;\n}\n\n.typingDot:nth-child(3) {\n    animation-delay: 0s;\n}\n\n@keyframes typingBounce {\n    0%,\n    80%,\n    100% {\n        transform: scale(0.8) translateY(0);\n        opacity: 0.5;\n    }\n    40% {\n        transform: scale(1.2) translateY(-8px);\n        opacity: 1;\n    }\n}\n\n.chatMainFlow .chatMessage.isMe {\n    align-items: flex-end;\n    flex-direction: row-reverse;\n    justify-content: flex-start;\n}\n\n.chatMainFlow .chatMessage.isMe .messageText {\n    border-bottom-right-radius: 6px;\n}\n\n.ratingStar {\n    cursor: pointer;\n    font-size: 20px;\n    transition: color 0.2s;\n    color: var(--star-inactive-color, #ccc);\n}\n\n.ratingStar.active {\n    color: #ffd700;\n}\n\n/* Sender Avatar */\n.chatMainFlow .chatMessage .avatar {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    margin: 0 0 4px;\n    flex-shrink: 0;\n    position: relative;\n}\n\n/* [㊗️]\n.chatMainFlow .chatMessage .avatar img {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    border-radius: 50%;\n    object-fit: cover;\n    background-color: var(--avatar-bg-color, #eef6fb);\n    border: 2px solid rgba(125, 125, 125, 0.1);\n    transition: transform 0.2s ease, box-shadow 0.2s ease;\n}\n\n\n.chatMainFlow .chatMessage .avatar img:hover {\n    transform: scale(1.05);\n    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\n}\n*/\n\n/* Message text bubble */\n.chatMainFlow .chatMessage .messageText {\n    background-color: var(--message-bg-color);\n    color: var(--message-text-color);\n    position: relative;\n    padding: 14px 18px;\n    border-radius: 20px;\n    max-width: 100%;\n    min-width: var(--chat-message-min-width);\n    text-align: left;\n    margin-bottom: 4px;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);\n    line-height: 1.5;\n    word-wrap: break-word;\n    transition: all 0.2s ease;\n    font-size: 15px;\n    backdrop-filter: blur(10px);\n    --chat-heading-surface: rgba(255, 255, 255, 0.12);\n    --chat-heading-accent: rgba(56, 189, 248, 0.85);\n    --chat-heading-text-color: var(--message-text-color, #0f172a);\n    --chat-heading-border-color: rgba(15, 23, 42, 0.18);\n    --chat-heading-shadow-color: rgba(15, 23, 42, 0.65);\n}\n\n@supports (color: color-mix(in srgb, #000 50%, #fff 50%)) {\n    .chatMainFlow .chatMessage .messageText {\n        --chat-heading-surface: color-mix(in srgb, var(--message-bg-color, #ffffff) 70%, rgba(15, 23, 42, 0.08) 30%);\n        --chat-heading-accent: color-mix(in srgb, var(--message-text-color, #0f172a) 65%, rgba(59, 130, 246, 0.9) 35%);\n        --chat-heading-border-color: color-mix(in srgb, var(--message-bg-color, #ffffff) 60%, rgba(15, 23, 42, 0.12) 40%);\n        --chat-heading-shadow-color: color-mix(in srgb, rgba(15, 23, 42, 0.65) 80%, rgba(96, 165, 250, 0.45) 20%);\n        --chat-heading-text-color: var(--message-text-color, #0f172a);\n    }\n}\n\n/* Copy button styles */\n.copyButtonContainer {\n    float: right;\n    top: 8px;\n    right: 10px;\n    z-index: 2;\n    align-items: center;\n    justify-content: flex-end;\n    pointer-events: none;\n\n    visibility: hidden;\n}\n.chatMainFlow .chatMessage .messageText:hover .copyButtonContainer,\n.copyButtonContainer:focus-within {\n    visibility: visible;\n    pointer-events: auto;\n}\n\n.copyButton {\n    background: rgba(255, 255, 255, 0.2);\n    border: 1px solid #ddd;\n    border-radius: 6px;\n    padding: 2px 5px;\n    cursor: pointer;\n    box-shadow: 0 1px 4px rgba(0, 0, 0, 0.07);\n    transition: all 0.15s, box-shadow 0.15s, border 0.15s;\n    display: flex;\n    align-items: center;\n    opacity: 0.7;\n    position: relative;\n}\n\n.copiedTooltip {\n    position: absolute;\n    left: 50%;\n    top: 110%;\n    transform: translateX(-50%);\n    background: #222;\n    color: #fff;\n    padding: 6px 16px;\n    border-radius: 8px;\n    font-size: 0.98em;\n    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.18);\n    opacity: 0.97;\n    pointer-events: none;\n    z-index: 100;\n    white-space: nowrap;\n    margin-top: 4px;\n    animation: copiedTooltipFadeIn 0.2s;\n    max-width: 220px;\n    overflow-wrap: break-word;\n    word-break: break-word;\n}\n\n.copiedTooltipLeft {\n    left: 0 !important;\n    transform: none !important;\n}\n\n.copiedTooltipRight {\n    left: auto !important;\n    right: 0 !important;\n    transform: none !important;\n}\n\n/* Removed right-aligned override to keep tooltip position stable */\n\n@keyframes copiedTooltipFadeIn {\n    from {\n        opacity: 0;\n        transform: translateX(-50%) translateY(8px) scale(0.97);\n    }\n    to {\n        opacity: 0.97;\n        transform: translateX(-50%) translateY(0) scale(1);\n    }\n}\n\n.copyButton:hover,\n.copyButton:focus {\n    border: 1.5px solid #bbb;\n    opacity: 1;\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.1);\n}\n\n.copyButton svg {\n    display: block;\n}\n\n.messageText ul {\n    list-style: disc;\n    margin-left: 20px;\n}\n\n.messageText ol {\n    list-style: decimal;\n    margin-left: 20px;\n}\n\n.messageText img,\n.messageText pre,\n.messageText blockquote,\n.messageText table {\n    margin-top: 10px;\n    margin-bottom: 10px;\n    border-radius: 8px;\n}\n\n.messageText pre {\n    display: block;\n    border: none;\n    box-shadow: none;\n    background: #000000ff;\n    font-size: inherit;\n    line-height: inherit;\n    color: #fff;\n    padding: 1em;\n}\n\n.messageText blockquote {\n    display: block;\n    border: none;\n    box-shadow: none;\n    background: #ffffffcc;\n    font-size: inherit;\n    line-height: inherit;\n    color: #000;\n    padding: 1em;\n}\n\n.messageText code {\n    display: inline-block;\n    margin: 0;\n    padding: 0;\n    border: none;\n    box-shadow: none;\n    background: #cccccc55;\n    font-size: inherit;\n    line-height: inherit;\n    color: inherit;\n}\n\n.messageText pre code {\n    background-color: #000000cc;\n    border-radius: 8px;\n}\n\n.messageText .chat-code-block {\n    background: #181c23;\n    color: #f8fafc;\n    font-size: 14px;\n    line-height: 1.6;\n    overflow-x: auto;\n    border-color: #23272f;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.12);\n    font-family: 'Fira Mono', 'Menlo', 'Consolas', 'Liberation Mono', monospace;\n}\n.messageText .chat-code-block code {\n    background: none !important;\n    color: inherit !important;\n    font-family: inherit !important;\n    font-size: inherit !important;\n    padding: 0 !important;\n    border: none !important;\n    box-shadow: none !important;\n    white-space: pre;\n    word-break: break-word;\n    overflow-x: auto;\n    display: block;\n}\n.messageText table {\n    width: 100%;\n    border-collapse: separate;\n    border-spacing: 0;\n    margin: 16px 0;\n    background: #f8fafc; /* Stronger light background for contrast */\n    border-radius: 12px;\n    overflow: hidden;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);\n    font-size: 14px;\n    color: #17223b; /* Dark text for contrast */\n}\n.messageText th,\n.messageText td {\n    padding: 10px 16px;\n    border-bottom: 1px solid #d1dbe8;\n    text-align: left;\n    color: #17223b; /* Ensure strong text color for all cells */\n    background: none;\n}\n.messageText th {\n    background: linear-gradient(90deg, #eaf3fa 80%, #d1e3f8 100%);\n    font-weight: 700;\n    color: #17223b; /* Strong header text */\n    border-bottom: 2px solid #b5c7de;\n}\n.messageText tr:last-child td {\n    border-bottom: none;\n}\n.messageText tr:nth-child(even) td {\n    background: #eaf3fa;\n}\n.messageText tr:hover td {\n    background: #cbe0f7;\n    transition: background 0.2s;\n}\n.messageText table {\n    border-radius: 12px;\n    overflow: hidden;\n}\n.messageText th:first-child,\n.messageText td:first-child {\n    border-top-left-radius: 12px;\n}\n.messageText th:last-child,\n.messageText td:last-child {\n    border-top-right-radius: 12px;\n}\n\n.chatMainFlow .chatMessage .messageText:hover {\n    box-shadow: 0 4px 16px rgba(0, 0, 0, 0.15);\n    transform: translateY(-1px);\n}\n\n/* Attachments styles */\n.attachments {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 10px;\n    padding-top: 10px;\n    border-top: 1px solid rgba(125, 125, 125, 0.2);\n}\n\n.attachment {\n    display: flex;\n    align-items: center;\n    gap: 6px;\n    padding: 6px 12px;\n    background: rgba(255, 255, 255, 0.2);\n    border: 1px solid rgba(125, 125, 125, 0.3);\n    border-radius: 12px;\n    font-size: 13px;\n    color: inherit;\n    text-decoration: none;\n    transition: all 0.2s ease;\n    max-width: 200px;\n}\n\n.attachment:hover {\n    background: rgba(255, 255, 255, 0.3);\n    border-color: rgba(125, 125, 125, 0.5);\n    transform: translateY(-1px);\n}\n\n.attachmentIcon {\n    font-size: 14px;\n    opacity: 0.8;\n}\n\n.attachmentName {\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n}\n\n/* Message buttons container */\n.messageButtons {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 12px;\n    padding-top: 12px;\n    border-top: 1px solid rgba(125 125 125 / 0.83);\n}\n\n/* Individual message button */\n.messageButton {\n    display: inline-flex;\n    align-items: center;\n    padding: 8px 14px;\n    background: rgba(125, 125, 125, 0.1);\n    border: 1px solid rgba(125, 125, 125, 0.9);\n    border-radius: 16px;\n    font-size: 13px;\n    font-weight: 500;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    backdrop-filter: blur(10px);\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n}\n\n.messageButton:hover {\n    background: rgba(0, 132, 255, 0.1);\n    border-color: rgba(0, 132, 255, 0.3);\n    transform: translateY(-1px);\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.15);\n}\n\n.messageButton:active {\n    transform: scale(0.98);\n    transition: transform 0.1s ease;\n}\n\n/* Remove default markdown styles from button content */\n.messageButton p {\n    margin: 0;\n    padding: 0;\n    line-height: inherit;\n}\n\n.messageButton strong {\n    font-weight: 600;\n}\n\n.messageButton em {\n    font-style: italic;\n}\n\n/* Rating system */\n.chatMainFlow .chatMessage .rating {\n    position: absolute;\n    bottom: -8px;\n    right: 8px;\n    display: flex;\n    gap: 2px;\n    align-items: center;\n    min-width: 24px;\n    z-index: 1;\n    background: rgba(0, 0, 0, 0.8);\n    border-radius: 12px;\n    padding: 4px 6px;\n    backdrop-filter: blur(10px);\n    opacity: 0;\n    transform: translateY(4px);\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.chatMainFlow .chatMessage:hover .rating {\n    opacity: 1;\n    transform: translateY(0);\n}\n\n.chatMainFlow .chatMessage .rating:hover {\n    background: rgba(0, 0, 0, 0.9);\n    padding: 6px 8px;\n    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);\n}\n\n.chatMainFlow .chatMessage .rating span {\n    transition: transform 0.2s ease, color 0.2s ease;\n    display: inline-block;\n    cursor: pointer;\n    font-size: 16px;\n}\n\n.chatMainFlow .chatMessage .rating:hover span {\n    transform: scale(1.1);\n}\n\n/* Chat input area */\n.chatMainFlow .chatInput {\n    z-index: 10;\n    grid-area: 📝;\n    width: 100%;\n    padding: 24px;\n\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n    position: relative;\n}\n\n.Chat.fullPageVisual .chatMainFlow .chatInput {\n    backdrop-filter: blur(25px);\n    background: linear-gradient(180deg, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.8) 100%);\n}\n\n/* File upload drag-and-drop styles */\n.chatMainFlow .chatInput.dragOver {\n    background: linear-gradient(to top, rgba(0, 132, 255, 0.1) 0%, rgba(0, 132, 255, 0.05) 100%);\n}\n\n/* File preview container */\n.filePreviewContainer {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-bottom: 8px;\n}\n\n/* Individual file preview */\n.filePreview {\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    padding: 8px 12px;\n    background: rgba(125, 125, 125, 0.1);\n    border: 1px solid rgba(125, 125, 125, 0.2);\n    border-radius: 8px;\n    font-size: 12px;\n    backdrop-filter: blur(10px);\n    transition: all 0.2s ease;\n}\n\n.filePreview:hover {\n    background: rgba(125, 125, 125, 0.15);\n    border-color: rgba(125, 125, 125, 0.3);\n}\n\n.fileIcon {\n    font-size: 14px;\n    opacity: 0.7;\n}\n\n.fileInfo {\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n    min-width: 0;\n}\n\n.fileName {\n    font-weight: 500;\n    color: black;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    max-width: 150px;\n}\n\n.fileSize {\n    color: #6b7280;\n    font-size: 11px;\n}\n\n.removeFileButton {\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    width: 20px;\n    height: 20px;\n    border: none;\n    background: rgba(255, 0, 0, 0.1);\n    color: #ff4444;\n    border-radius: 50%;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n}\n\n.removeFileButton:hover {\n    background: rgba(255, 0, 0, 0.2);\n    transform: scale(1.1);\n}\n\n/* Input container for textarea and buttons */\n.inputContainer {\n    display: flex;\n    align-items: flex-end;\n    gap: 8px;\n    background: #ffffff;\n    border: 1px solid rgba(0, 0, 0, 0.08);\n    border-radius: 28px;\n    padding: 8px 12px;\n    box-shadow: 0 4px 20px rgba(0, 0, 0, 0.06);\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.inputContainer:focus-within {\n    border-color: var(--brand-color);\n    box-shadow: 0 8px 32px rgba(0, 132, 255, 0.12);\n    transform: translateY(-2px);\n}\n\n.inputContainer textarea::placeholder {\n    color: #94a3b8;\n    opacity: 1;\n}\n\n/* Attachment button */\n.attachmentButton {\n    width: 40px;\n    height: 40px;\n    border: none;\n    background: transparent;\n    color: #64748b;\n    border-radius: 50%;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n    margin-bottom: 2px;\n}\n\n.attachmentButton:hover:not(:disabled) {\n    background: #f1f5f9;\n    color: #0f172a;\n}\n\n.attachmentButton:disabled {\n    opacity: 0.3;\n    cursor: not-allowed;\n}\n\n/* Voice Button */\n.voiceButton {\n    width: 40px;\n    height: 40px;\n    border: none;\n    background: transparent;\n    color: #64748b;\n    border-radius: 50%;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n    margin-bottom: 2px;\n}\n\n.voiceButton:hover:not(:disabled) {\n    background: #f1f5f9;\n    color: #0f172a;\n}\n\n.voiceButtonActive {\n    background: rgba(239, 68, 68, 0.1) !important;\n    color: #ef4444 !important;\n    animation: voiceRecordingPulse 1.5s infinite;\n}\n\n@keyframes voiceRecordingPulse {\n    0% {\n        box-shadow: 0 0 0 0 rgba(255, 0, 0, 0.4);\n    }\n    70% {\n        box-shadow: 0 0 0 10px rgba(255, 0, 0, 0);\n    }\n    100% {\n        box-shadow: 0 0 0 0 rgba(255, 0, 0, 0);\n    }\n}\n\n/* Upload progress indicator */\n.uploadProgress {\n    display: flex;\n    align-items: center;\n    gap: 12px;\n    padding: 8px 12px;\n    background: rgba(0, 132, 255, 0.1);\n    border: 1px solid rgba(0, 132, 255, 0.2);\n    border-radius: 8px;\n    font-size: 13px;\n    color: #0084ff;\n}\n\n.uploadProgressBar {\n    flex: 1;\n    height: 4px;\n    background: rgba(0, 132, 255, 0.2);\n    border-radius: 2px;\n    overflow: hidden;\n}\n\n.uploadProgressFill {\n    height: 100%;\n    background: linear-gradient(90deg, #0084ff, rgba(0, 132, 255, 0.8));\n    border-radius: 2px;\n    animation: uploadProgress 1.5s ease-in-out infinite;\n}\n\n@keyframes uploadProgress {\n    0% {\n        transform: translateX(-100%);\n    }\n    50% {\n        transform: translateX(0%);\n    }\n    100% {\n        transform: translateX(100%);\n    }\n}\n\n/* Drag overlay */\n.dragOverlay {\n    position: absolute;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    background: rgba(0, 132, 255, 0.1);\n    border: 2px dashed rgba(0, 132, 255, 0.5);\n    border-radius: 12px;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    backdrop-filter: blur(10px);\n    z-index: 20;\n    pointer-events: none;\n}\n\n.dragOverlayContent {\n    display: flex;\n    flex-direction: column;\n    align-items: center;\n    gap: 12px;\n    color: #0084ff;\n    font-weight: 600;\n    text-align: center;\n}\n\n.dragOverlayContent svg {\n    opacity: 0.7;\n}\n\n/* Chat input field */\n.chatMainFlow .chatInput textarea {\n    flex: 1;\n    padding: 10px 12px;\n    margin: 0;\n    border: none;\n    outline: none;\n    background: transparent;\n    color: #0f172a;\n    min-width: 100px;\n    max-height: 200px;\n    font-size: 15px;\n    line-height: 1.5;\n    resize: none;\n    appearance: none;\n    -webkit-appearance: none;\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n}\n\n.chatMainFlow .chatInput textarea:focus {\n    /* Focus is handled by .inputContainer */\n}\n\n.chatMainFlow .chatInput textarea:disabled {\n    opacity: 0.6;\n    cursor: not-allowed;\n    transform: none;\n}\n\n.chatMainFlow .chatInput textarea::placeholder {\n    color: inherit;\n    opacity: 0.7;\n}\n\n/* Chat send button */\n.chatMainFlow .chatInput button[data-button-type='call-to-action'] {\n    width: 40px;\n    height: 40px;\n    margin: 0 0 2px 0 !important;\n    padding: 0 !important;\n\n    border: none;\n    color: #ffffff;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    border-radius: 50% !important;\n    aspect-ratio: 1 / 1;\n    min-width: unset !important;\n    min-height: unset !important;\n\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    box-shadow: 0 4px 12px rgba(0, 132, 255, 0.25);\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n    overflow: visible;\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action']:hover {\n    transform: scale(1.05);\n    box-shadow: 0 6px 16px rgba(0, 132, 255, 0.35);\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action']:active {\n    transform: scale(0.95);\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action'] svg {\n    width: 20px;\n    height: 20px;\n}\n\n.chatMainFlow .chatInput button img {\n    width: 50%;\n    height: 100%;\n    object-fit: contain;\n}\n\n/* Scroll to bottom button */\n\n.scrollToBottomContainer {\n    /*/\n    outline: 1px dotted red;\n    /**/\n\n    z-index: 20;\n    grid-area: 📝;\n    width: 100%;\n    height: 100%;\n    display: flex;\n    justify-content: center;\n    align-items: flex-start;\n\n    pointer-events: none;\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper {\n    pointer-events: all;\n    position: relative;\n    transform: translate(-50%, -150%);\n    display: inline-flex;\n    justify-content: center;\n    align-items: center;\n    animation: scrollButtonSlideIn 0.3s ease-out;\n    transition: transform 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.scrollToBottomContainer .scrollToBottom {\n    width: 48px;\n    height: 48px;\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    border: none;\n    outline: none;\n    background: rgba(0 0 0 / 0.5);\n    backdrop-filter: blur(3px);\n    border-radius: 50%;\n    font-weight: bold;\n    color: white;\n    cursor: pointer;\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    box-shadow: 0 4px 16px rgba(0, 0, 0, 0.3);\n}\n\n@keyframes scrollButtonSlideIn {\n    from {\n        opacity: 0;\n        transform: translate(-50%, 20px) scale(0.8);\n    }\n    to {\n        opacity: 0.9;\n        transform: translate(-50%, 0) scale(1);\n    }\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper:hover {\n    transform: translate(-50%, -160%) scale(1.05);\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper:active {\n    transform: translate(-50%, -160%) scale(0.95);\n    transition: transform 0.1s ease;\n}\n\n.scrollToBottomBadge {\n    position: absolute;\n    top: -8px;\n    right: -8px;\n    min-width: 32px;\n    padding: 2px 6px;\n    font-size: 10px;\n    font-weight: 600;\n    background: #2563eb;\n    color: #fff;\n    border-radius: 999px;\n    box-shadow: 0 2px 10px rgba(37, 99, 235, 0.4);\n    text-transform: uppercase;\n    letter-spacing: 0.04em;\n    white-space: nowrap;\n    pointer-events: none;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n}\n\n/* Rating modal */\n.ratingModal {\n    position: fixed;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n\n    background-color: rgba(0, 0, 0, 0.6);\n    backdrop-filter: blur(8px);\n\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    z-index: 1000;\n    animation: modalFadeIn 0.3s ease-out;\n}\n\n@keyframes modalFadeIn {\n    from {\n        opacity: 0;\n        backdrop-filter: blur(0px);\n    }\n    to {\n        opacity: 1;\n        backdrop-filter: blur(8px);\n    }\n}\n\n.ratingModalContent {\n    background: #ffffff;\n    color: #0f1724;\n    padding: 32px;\n    border-radius: 16px;\n    width: 90%;\n    max-width: 480px;\n    box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3);\n    border: 1px solid rgba(125, 125, 125, 0.1);\n    backdrop-filter: blur(20px);\n    animation: modalSlideIn 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiBlack';\n    /* <- TODO: [🧠][🎱] Better, define other fonts */\n}\n\n@keyframes modalSlideIn {\n    from {\n        opacity: 0;\n        transform: translateY(20px) scale(0.95);\n    }\n    to {\n        opacity: 1;\n        transform: translateY(0) scale(1);\n    }\n}\n\n.ratingModalContent h3 {\n    margin: 0 0 24px 0;\n    text-align: center;\n    color: black;\n    font-size: 20px;\n    font-weight: 600;\n}\n\n.stars {\n    display: flex;\n    justify-content: center;\n    gap: 8px;\n    margin-bottom: 24px;\n}\n\n.stars span {\n    font-size: 28px;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    padding: 4px;\n    border-radius: 8px;\n}\n\n.stars span:hover {\n    transform: scale(1.2);\n    background: rgba(255, 215, 0, 0.1);\n}\n\n.ratingModalStar {\n    cursor: pointer;\n    font-size: 24px;\n    transition: color 0.2s;\n}\n\n.toolCallModal {\n    max-width: 800px;\n    padding: 0;\n    overflow: hidden;\n    position: relative;\n}\n\n.searchModalHeader {\n    padding: 24px 32px;\n    padding-right: 72px;\n    border-bottom: 1px solid #eee;\n    display: flex;\n    align-items: center;\n    gap: 16px;\n    background: #fdfdfd;\n}\n\n.selfLearningModalHeader {\n    align-items: center;\n    gap: 18px;\n    background: linear-gradient(135deg, #f8fafc 0%, #fef9c3 45%, #ecfccb 100%);\n    border-bottom: 1px solid #e2e8f0;\n}\n\n.selfLearningAvatarGroup {\n    display: flex;\n    align-items: center;\n    gap: 10px;\n}\n\n.selfLearningAvatar {\n    width: 48px;\n    height: 48px;\n    border-radius: 50%;\n    background: #e2e8f0;\n    color: #0f172a;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    font-weight: 600;\n    font-size: 18px;\n    letter-spacing: 0.02em;\n    background-size: cover;\n    background-position: center;\n    box-shadow: 0 10px 20px rgba(15, 23, 42, 0.18);\n}\n\n.selfLearningAvatarInitial {\n    line-height: 1;\n}\n\n.selfLearningTeacher {\n    background: linear-gradient(135deg, #fde68a 0%, #86efac 100%);\n}\n\n.selfLearningHeaderText {\n    display: flex;\n    flex-direction: column;\n    gap: 6px;\n    text-align: left;\n}\n\n.selfLearningTitle {\n    margin: 0;\n    font-size: 20px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.selfLearningMetaRow {\n    display: flex;\n    flex-wrap: wrap;\n    align-items: center;\n    gap: 10px;\n    font-size: 13px;\n    color: #64748b;\n    margin-bottom: 12px;\n}\n\n.selfLearningMeta {\n    font-size: 13px;\n    color: #64748b;\n}\n\n.selfLearningMetaChip {\n    background: #e2e8f0;\n    color: #0f172a;\n    border-radius: 999px;\n    padding: 4px 10px;\n    font-size: 12px;\n    font-weight: 600;\n}\n\n.selfLearningCommitments {\n    background: #f8fafc;\n    border: 1px solid #e2e8f0;\n    border-radius: 16px;\n    padding: 16px;\n}\n\n.selfLearningCommitmentsLabel {\n    font-size: 12px;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    color: #94a3b8;\n    font-weight: 600;\n}\n\n.selfLearningBookEditor {\n    margin-top: 12px;\n    overflow: hidden;\n    border-radius: 12px;\n}\n\n.selfLearningEmpty {\n    margin-top: 12px;\n    font-size: 13px;\n    color: #475569;\n}\n\n.modalCloseButton {\n    position: absolute;\n    top: 16px;\n    right: 16px;\n    z-index: 1;\n    width: 36px;\n    height: 36px;\n    border-radius: 999px;\n    border: 1px solid #e2e8f0;\n    background: #fff;\n    color: #475569;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    box-shadow: 0 6px 16px rgba(15, 23, 42, 0.08);\n    transition: background 0.2s ease, color 0.2s ease, box-shadow 0.2s ease, transform 0.2s ease;\n}\n\n.modalCloseButton:hover {\n    background: #f8fafc;\n    color: #0f172a;\n    box-shadow: 0 10px 24px rgba(15, 23, 42, 0.12);\n    transform: translateY(-1px);\n}\n\n.modalCloseButton:focus-visible {\n    outline: 2px solid #94a3b8;\n    outline-offset: 2px;\n}\n\n.teamModalHeader {\n    gap: 12px;\n    flex-wrap: wrap;\n}\n\n.teamHeaderParticipants {\n    display: flex;\n    align-items: center;\n    gap: 12px;\n    flex-wrap: wrap;\n}\n\n.teamHeaderProfile {\n    display: inline-flex;\n    align-items: center;\n    gap: 12px;\n    text-decoration: none;\n    color: inherit;\n}\n\n.teamHeaderProfileLink:hover {\n    color: #0f172a;\n}\n\n.teamHeaderAvatar {\n    width: 40px;\n    height: 40px;\n    border-radius: 999px;\n    border: 1px solid #e2e8f0;\n    background-size: cover;\n    background-position: center;\n    background-repeat: no-repeat;\n    box-shadow: 0 6px 14px rgba(15, 23, 42, 0.08);\n}\n\n.teamHeaderName {\n    font-size: 18px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.teamHeaderDivider {\n    font-size: 14px;\n    font-weight: 600;\n    color: #94a3b8;\n}\n\n.searchModalIcon {\n    font-size: 24px;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    color: #475569;\n}\n\n.searchModalQuery {\n    margin: 0 !important;\n    font-size: 22px !important;\n    font-weight: 500 !important;\n    color: #202124 !important;\n    text-align: left !important;\n}\n\n.searchModalContent {\n    padding: 24px 32px;\n    background: #fff;\n    max-height: 60vh;\n    overflow-y: auto;\n}\n\n.teamChatContainer {\n    border: 1px solid #e5e7eb;\n    border-radius: 16px;\n    overflow: hidden;\n    background: #f8fafc;\n    margin-bottom: 20px;\n}\n\n.teamToolCallSection {\n    display: flex;\n    flex-direction: column;\n    gap: 16px;\n}\n\n.teamToolCallGroup {\n    display: flex;\n    flex-direction: column;\n    gap: 8px;\n}\n\n.teamToolCallHeading {\n    font-size: 12px;\n    font-weight: 600;\n    text-transform: uppercase;\n    letter-spacing: 0.08em;\n    color: #64748b;\n}\n\n.teamToolCallChips {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n}\n\n.teamToolCallDetails {\n    border: 1px solid rgba(148, 163, 184, 0.4);\n    border-radius: 12px;\n    padding: 16px;\n    background: #f8fafc;\n}\n\n.teamToolCallDetailsHeader {\n    display: flex;\n    align-items: center;\n    justify-content: space-between;\n    gap: 12px;\n    margin-bottom: 12px;\n}\n\n.teamToolCallDetailsTitle {\n    font-size: 14px;\n    font-weight: 600;\n    color: #0f172a;\n    display: flex;\n    align-items: center;\n    gap: 6px;\n}\n\n.teamToolCallDetailsClear {\n    background: rgba(148, 163, 184, 0.2);\n    border: 1px solid rgba(148, 163, 184, 0.4);\n    border-radius: 999px;\n    padding: 4px 10px;\n    font-size: 12px;\n    cursor: pointer;\n    color: #475569;\n    transition: all 0.2s ease;\n}\n\n.teamToolCallDetailsClear:hover {\n    background: rgba(148, 163, 184, 0.35);\n}\n\n.searchResultsList {\n    display: flex;\n    flex-direction: column;\n    gap: 28px;\n    text-align: left;\n}\n\n.searchResultsRaw {\n    text-align: left;\n    max-width: 650px;\n}\n\n.searchResultItem {\n    max-width: 650px;\n}\n\n.searchResultUrl {\n    font-size: 14px;\n    color: #202124;\n    margin-bottom: 4px;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n}\n\n.searchResultUrl a {\n    color: #202124;\n    text-decoration: none;\n}\n\n.searchResultTitle {\n    font-size: 20px !important;\n    font-weight: 400 !important;\n    margin: 0 0 4px 0 !important;\n    line-height: 1.3;\n}\n\n.searchResultTitle a {\n    color: #1a0dab;\n    text-decoration: none;\n}\n\n.searchResultTitle a:hover {\n    text-decoration: underline;\n}\n\n.searchResultSnippet {\n    font-size: 14px;\n    line-height: 1.58;\n    color: #4d5156;\n    margin: 0;\n}\n\n.noResults {\n    padding: 40px;\n    text-align: center;\n    color: #70757a;\n    font-style: italic;\n}\n\n.toolCallDetails {\n    text-align: left;\n    margin: 20px 0;\n}\n.citationDetails {\n    text-align: left;\n    margin: 20px 0;\n}\n\n.citationPreview {\n    margin-top: 16px;\n    border: 1px solid #ddd;\n    border-radius: 8px;\n    overflow: hidden;\n    background: #f8f8f8;\n}\n\n.citationIframe {\n    width: 100%;\n    height: 60vh;\n    min-height: 400px;\n    border: none;\n    display: block;\n}\n\n.citationMetadata {\n    background: #f8f8f8;\n    padding: 16px;\n    border-radius: 8px;\n    border: 1px solid #eee;\n    margin-bottom: 20px;\n}\n\n.citationMetadata p {\n    margin: 8px 0;\n    font-size: 0.9em;\n}\n\n.citationMetadata strong {\n    color: #333;\n}\n\n.citationMetadata a {\n    color: #007bff;\n    text-decoration: none;\n    word-break: break-all;\n}\n\n.citationMetadata a:hover {\n    text-decoration: underline;\n}\n\n.citationExcerpt {\n    background: #fff;\n    padding: 16px;\n    border-radius: 8px;\n    border: 1px solid #ddd;\n    max-height: 400px;\n    overflow-y: auto;\n}\n\n.citationExcerpt h4 {\n    margin: 0 0 12px 0;\n    color: #333;\n    font-size: 1em;\n}\n\n.citationHint {\n    font-size: 0.85em;\n    margin-top: 8px;\n    opacity: 0.7;\n}\n\n.toolCallDataContainer {\n    background: #f8f8f8;\n    padding: 12px;\n    border-radius: 6px;\n    border: 1px solid #eee;\n    margin-bottom: 15px;\n    max-height: 300px;\n    overflow-y: auto;\n}\n\n.toolCallData {\n    margin: 0;\n    font-size: 0.85em;\n    white-space: pre-wrap;\n    word-break: break-word;\n}\n\n.toolCallArgsList {\n    list-style: none;\n    padding: 0;\n    margin: 0;\n}\n\n.toolCallArgsList li {\n    margin-bottom: 5px;\n    font-size: 0.9em;\n}\n\n.toolCallModal .toolCallData {\n    background: #f8fafc;\n    color: #0f172a;\n}\n\n.ratingInput {\n    width: 100%;\n    min-height: 100px;\n    padding: 16px;\n    border: 2px solid rgba(125, 125, 125, 0.1);\n    border-radius: 12px;\n    margin-bottom: 18px;\n    resize: vertical;\n    background: rgba(125, 125, 125, 0.05);\n    color: #0b1220;\n    font-size: 14px;\n    line-height: 1.5;\n    transition: all 0.2s ease;\n}\n\n.ratingInput:focus {\n    border-color: #0084ff;\n    background: rgba(125, 125, 125, 0.08);\n    box-shadow: 0 0 0 4px rgba(0, 132, 255, 0.1);\n    outline: none;\n}\n\n.ratingInput[readonly] {\n    border: 1px solid rgba(125, 125, 125, 0.5);\n    background: rgba(125, 125, 125, 0.01);\n}\n\n.ratingActions {\n    display: flex;\n    justify-content: flex-end;\n    gap: 12px;\n}\n\n.ratingActions button {\n    padding: 12px 24px;\n    border: none;\n    border-radius: 8px;\n    cursor: pointer;\n    font-size: 14px;\n    font-weight: 500;\n    transition: all 0.2s ease;\n    min-width: 80px;\n}\n\n.ratingActions button:first-child {\n    background-color: rgba(125, 125, 125, 0.1);\n    color: #0b1220;\n    border: 1px solid rgba(125, 125, 125, 0.2);\n}\n\n.ratingActions button:first-child:hover:not(:disabled) {\n    background-color: rgba(125, 125, 125, 0.2);\n    color: #0b1220;\n}\n\n.ratingActions button:last-child {\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%);\n    color: #ffffff;\n}\n\n.ratingActions button:last-child:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0071d1 0%, #0052a3 100%);\n    transform: translateY(-1px);\n    box-shadow: 0 4px 12px rgba(0, 132, 255, 0.3);\n}\n\n.ratingActions button:last-child:disabled {\n    opacity: 0.5;\n    cursor: not-allowed;\n}\n\n.downloadButton {\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    gap: 8px;\n    padding: 12px 24px;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%);\n    color: #ffffff;\n    border: none;\n    border-radius: 8px;\n    cursor: pointer;\n    font-size: 14px;\n    font-weight: 600;\n    transition: all 0.2s ease;\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.2);\n}\n\n.downloadButton:hover {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px);\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.downloadButton:active {\n    transform: translateY(0);\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.2);\n}\n\n.downloadButton svg {\n    flex-shrink: 0;\n}\n\n.downloadButton span {\n    font-weight: 600;\n}\n\n/* New chat button styling - Matches the sleek chat message design */\n.chatButton {\n    display: inline-flex !important;\n    align-items: center !important;\n    justify-content: center !important;\n    gap: 8px !important;\n    padding: 12px 16px !important;\n    margin: 0 !important;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%) !important;\n    color: #ffffff !important;\n    border: none !important;\n    border-radius: 20px !important;\n    font-size: 13px !important;\n    font-weight: 600 !important;\n    line-height: 1.3 !important;\n    cursor: pointer !important;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94) !important;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3) !important;\n    backdrop-filter: blur(20px) !important;\n    -webkit-tap-highlight-color: transparent !important;\n    touch-action: manipulation !important;\n    user-select: none !important;\n    min-height: 40px !important;\n    min-width: 110px !important;\n    position: relative !important;\n    overflow: hidden !important;\n}\n\n.chatButton::before {\n    content: '';\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 100%;\n    height: 100%;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);\n    transition: left 0.5s ease;\n}\n\n.chatButton:hover::before {\n    left: 100%;\n}\n\n.chatButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(0, 132, 255, 0.4);\n}\n\n.chatButton:active {\n    transform: scale(0.98) translateY(-1px);\n    transition: transform 0.1s ease;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.chatButton:focus {\n    outline: none;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3), 0 0 0 3px rgba(0, 132, 255, 0.3);\n}\n\n.chatButton svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    opacity: 1;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.2));\n}\n\n.chatButton:hover svg {\n    transform: rotate(-90deg) scale(1.1);\n    filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.3));\n}\n\n.chatButtonText {\n    white-space: nowrap;\n    font-weight: 600;\n    transition: all 0.2s ease;\n    text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);\n    opacity: 1;\n}\n\n.chatButton:hover .chatButtonText {\n    transform: translateX(1px);\n}\n\n/* Use template button styling - matches the chatButton and saveButton */\n.useTemplateButton {\n    display: inline-flex !important;\n    align-items: center !important;\n    justify-content: center !important;\n    gap: 8px !important;\n    padding: 12px 16px !important;\n    margin: 0 !important;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%) !important;\n    color: #ffffff !important;\n    border: none !important;\n    border-radius: 20px !important;\n    font-size: 13px !important;\n    font-weight: 600 !important;\n    line-height: 1.3 !important;\n    cursor: pointer !important;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94) !important;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3) !important;\n    backdrop-filter: blur(20px) !important;\n    -webkit-tap-highlight-color: transparent !important;\n    touch-action: manipulation !important;\n    user-select: none !important;\n    min-height: 40px !important;\n    min-width: 110px !important;\n    position: relative !important;\n    overflow: hidden !important;\n}\n\n.useTemplateButton::before {\n    content: '';\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 100%;\n    height: 100%;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);\n    transition: left 0.5s ease;\n}\n\n.useTemplateButton:hover::before {\n    left: 100%;\n}\n\n.useTemplateButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(0, 132, 255, 0.4);\n}\n\n.useTemplateButton:active {\n    transform: scale(0.98) translateY(-1px);\n    transition: transform 0.1s ease;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.useTemplateButton:focus {\n    outline: none;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3), 0 0 0 3px rgba(0, 132, 255, 0.3);\n}\n\n.useTemplateButton svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    opacity: 1;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.2));\n}\n\n.useTemplateButton:hover svg {\n    transform: scale(1.1);\n    filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.3));\n}\n\n.useTemplateButton:hover .chatButtonText {\n    transform: translateX(1px);\n}\n\n.saveButtonContainer {\n    display: inline-block;\n    position: relative;\n}\n\n.saveMenu {\n    position: absolute;\n    top: 100%;\n    left: 0;\n    background: #fff;\n    border: 1px solid #ddd;\n    z-index: 10;\n    min-width: 120px;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);\n}\n\n.saveMenuItem {\n    display: block;\n    width: 100%;\n    padding: 8px 16px;\n    border: none;\n    background: none;\n    text-align: left;\n    cursor: pointer;\n    color: #111;\n}\n\n.saveMenuItem:hover {\n    background-color: #f0f0f0;\n}\n\n.saveMenuDivider {\n    height: 1px;\n    background-color: #ddd;\n    margin: 4px 0;\n}\n\n.chatFeedbackPanel {\n    width: min(260px, 100%);\n    padding: 6px 8px;\n    border-radius: 14px;\n    border: 1px solid #e5e7eb;\n    background: #ffffff;\n    box-shadow: 0 8px 18px rgba(15, 23, 42, 0.08);\n    display: flex;\n    flex-direction: column;\n    gap: 6px;\n}\n\n.chatFeedbackToggle {\n    display: flex;\n    align-items: center;\n    gap: 10px;\n    width: 100%;\n    padding: 6px 8px;\n    border-radius: 12px;\n    border: none;\n    background: transparent;\n    cursor: pointer;\n    transition: background 0.2s ease, transform 0.2s ease;\n    text-align: left;\n}\n\n.chatFeedbackToggle:hover {\n    background: #f5f5f5;\n    transform: translateY(-0.5px);\n}\n\n.chatFeedbackToggle:focus-visible {\n    outline: 2px solid #2563eb;\n    outline-offset: 2px;\n}\n\n.chatFeedbackIcon {\n    width: 34px;\n    height: 34px;\n    border-radius: 10px;\n    background: #eef2ff;\n    color: #6366f1;\n    display: grid;\n    place-items: center;\n    flex-shrink: 0;\n    transition: background 0.2s ease, color 0.2s ease;\n}\n\n.chatFeedbackToggle[data-enabled='true'] .chatFeedbackIcon {\n    background: #e0f2fe;\n    color: #0284c7;\n}\n\n.chatFeedbackToggle[data-enabled='false'] .chatFeedbackIcon {\n    background: #f3f4f6;\n    color: #94a3b8;\n}\n\n.chatFeedbackMeta {\n    flex: 1;\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n}\n\n.chatFeedbackLabel {\n    font-size: 0.85rem;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.chatFeedbackDescription {\n    font-size: 0.75rem;\n    color: #64748b;\n}\n\n.chatFeedbackIndicator {\n    font-size: 0.72rem;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    font-weight: 700;\n    padding: 2px 8px;\n    border-radius: 999px;\n}\n\n.chatFeedbackIndicatorActive {\n    color: #166534;\n    background: #dcfce7;\n    border: 1px solid #34d399;\n}\n\n.chatFeedbackIndicatorInactive {\n    color: #713f12;\n    background: #fef9c3;\n    border: 1px solid #facc15;\n}\n\n/* Pause/Resume button variant (reuses .chatButton base styles for DRY) */\n.pauseButton {\n    background: linear-gradient(135deg, #ffb347 0%, #ff8c42 100%) !important; /* Warm orange for \"active/running\" */\n}\n\n.pauseButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #ffc067 0%, #ff9e5f 100%) !important;\n}\n\n.pauseButton.pausing {\n    opacity: 0.6 !important;\n    cursor: wait !important;\n}\n\n.pauseButton.paused {\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%) !important; /* Green when paused (ready to resume) */\n}\n\n.pauseButton.paused:hover:not(:disabled) {\n    background: linear-gradient(135deg, #34d399 0%, #059669 100%) !important;\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(16, 185, 129, 0.35);\n}\n\n.pauseButton svg {\n    transition: transform 0.3s ease;\n}\n\n.pauseButton.paused svg {\n    transform: scale(1.1);\n}\n\n.pauseButton.pausing svg {\n    opacity: 0.8;\n}\n\n/* Voice call indicator bar */\n.voiceCallIndicatorBar {\n    grid-area: 🟦;\n    width: 100%;\n    padding: 12px 20px;\n    background: linear-gradient(135deg, rgba(34, 197, 94, 0.1) 0%, rgba(16, 185, 129, 0.1) 100%);\n    border-bottom: 1px solid rgba(34, 197, 94, 0.3);\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    backdrop-filter: blur(10px);\n}\n\n/* Voice call indicator */\n.voiceCallIndicator {\n    display: inline-flex;\n    align-items: center;\n    gap: 8px;\n    padding: 8px 16px;\n    background: linear-gradient(135deg, rgba(34, 197, 94, 0.2) 0%, rgba(16, 185, 129, 0.2) 100%);\n    border: 1px solid rgba(34, 197, 94, 0.4);\n    border-radius: 20px;\n    color: #10b981;\n    font-size: 13px;\n    font-weight: 600;\n    backdrop-filter: blur(10px);\n    box-shadow: 0 2px 8px rgba(34, 197, 94, 0.2);\n    position: relative;\n}\n\n.voiceCallIndicator svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    animation: voiceCallIconPulse 2s ease-in-out infinite;\n}\n\n.voiceCallIndicator span {\n    font-weight: 600;\n    text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);\n}\n\n/* Voice call pulse animation */\n.voiceCallPulse {\n    position: absolute;\n    right: 8px;\n    top: 50%;\n    transform: translateY(-50%);\n    width: 8px;\n    height: 8px;\n    background: #10b981;\n    border-radius: 50%;\n    animation: voiceCallPulse 1.5s ease-in-out infinite;\n}\n\n@keyframes voiceCallIconPulse {\n    0%,\n    100% {\n        transform: scale(1);\n        opacity: 1;\n    }\n    50% {\n        transform: scale(1.1);\n        opacity: 0.8;\n    }\n}\n\n@keyframes voiceCallPulse {\n    0%,\n    100% {\n        transform: translateY(-50%) scale(1);\n        opacity: 1;\n    }\n    50% {\n        transform: translateY(-50%) scale(1.3);\n        opacity: 0.6;\n    }\n}\n\n/* Voice call indicator in messages */\n.chatMessage .voiceCallIndicator {\n    margin-bottom: 8px;\n    padding: 6px 12px;\n    font-size: 12px;\n    border-radius: 16px;\n}\n\n.chatMessage .voiceCallIndicator svg {\n    width: 14px;\n    height: 14px;\n}\n\n/* Feedback status banner */\n.feedbackStatus {\n    position: fixed;\n    top: 20px;\n    right: 20px;\n    padding: 14px 22px;\n    border-radius: 14px;\n    box-shadow: 0 18px 32px rgba(15, 23, 42, 0.35);\n    z-index: 10000; /* [Fix z-index]: Ensure it's above the header (z-50) and other UI elements */\n    animation: feedbackStatusSlideIn 0.28s ease-out;\n    max-width: 340px;\n    word-wrap: break-word;\n    font-weight: 500;\n    backdrop-filter: blur(20px);\n    pointer-events: none;\n}\n\n.feedbackStatusSuccess {\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%);\n    color: #f8fafc;\n}\n\n.feedbackStatusError {\n    background: linear-gradient(135deg, #ef4444 0%, #dc2626 100%);\n    color: #fef2f2;\n}\n\n@keyframes feedbackStatusSlideIn {\n    from {\n        transform: translateX(100%);\n        opacity: 0;\n    }\n\n    to {\n        transform: translateX(0);\n        opacity: 1;\n    }\n}\n\n/* Mobile responsiveness */\n@media (max-width: 768px) {\n    .Chat {\n        --chat-message-min-width: 100px;\n        --chat-message-max-width: min(85%, 560px);\n        --chat-message-avatar-gap: 10px;\n    }\n\n    .actions {\n        margin: 12px 16px 0;\n        gap: 6px;\n    }\n\n    .chatMainFlow .chatMessages {\n        padding: 16px 12px;\n    }\n\n    .chatMainFlow .chatMessage {\n        margin-bottom: 16px;\n    }\n\n    .chatMainFlow .chatMessage .messageText {\n        padding: 12px 16px;\n        font-size: 14px;\n        border-radius: 18px;\n    }\n\n    .chatMainFlow .chatMessage .avatar {\n        width: 36px;\n        height: 36px;\n        margin: 0 0 4px;\n    }\n\n    /* [㊗️]\n    .chatMainFlow .chatMessage .avatar img {\n        width: 36px;\n        aspect-ratio: 1 / 1;\n    }\n    */\n\n    .chatMainFlow .chatInput {\n        padding: 16px 12px;\n        gap: 10px;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        font-size: 16px;\n        padding: 14px 18px;\n        border-radius: 22px;\n    }\n\n    .chatMainFlow .chatInput button {\n        width: 44px;\n        height: 44px;\n    }\n\n    .scrollToBottom {\n        width: 44px;\n        height: 44px;\n        font-size: 18px;\n        top: calc(100% - 160px);\n    }\n\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n\n    .ratingModalContent {\n        margin: 16px;\n        padding: 24px 20px;\n        border-radius: 16px;\n        max-height: 80vh;\n        overflow-y: auto;\n    }\n\n    .stars {\n        gap: 6px;\n        margin-bottom: 20px;\n    }\n\n    .stars span {\n        font-size: 32px;\n        padding: 8px;\n    }\n\n    .ratingActions {\n        flex-direction: column-reverse;\n        gap: 8px;\n    }\n\n    .ratingActions button {\n        width: 100%;\n        padding: 14px;\n        font-size: 16px;\n        border-radius: 10px;\n    }\n\n    .selfLearningModalHeader {\n        flex-direction: column;\n        align-items: flex-start;\n    }\n\n    .selfLearningAvatarGroup {\n        justify-content: flex-start;\n    }\n\n    .selfLearningMetaRow {\n        align-items: flex-start;\n    }\n}\n\n@media (max-width: 480px) {\n    .Chat {\n        --chat-message-min-width: 80px;\n        --chat-message-max-width: min(90%, 520px);\n        --chat-message-avatar-gap: 8px;\n    }\n\n    .actions {\n        margin: 8px 12px 0;\n        gap: 4px;\n    }\n\n    .chatMainFlow .chatMessages {\n        padding: 12px 8px;\n    }\n\n    .chatMainFlow .chatMessage .messageText {\n        padding: 10px 14px;\n        font-size: 14px;\n        border-radius: 16px;\n    }\n\n    .chatMainFlow .chatMessage .avatar {\n        width: 32px;\n        height: 32px;\n        margin: 0 0 4px;\n    }\n\n    /* [㊗️]\n    .chatMainFlow .chatMessage .avatar img {\n        width: 32px;\n        aspect-ratio: 1 / 1;\n    }\n    */\n\n    .chatMainFlow .chatInput {\n        padding: 12px 8px;\n        gap: 8px;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        padding: 12px 16px;\n        border-radius: 20px;\n        font-size: 16px;\n    }\n\n    .chatMainFlow .chatInput button {\n        width: 40px;\n        height: 40px;\n    }\n\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n\n    .scrollToBottom {\n        width: 40px;\n        height: 40px;\n        font-size: 16px;\n        top: calc(100% - 140px);\n    }\n\n    .ratingModal {\n        padding: 0;\n        align-items: flex-end;\n    }\n\n    .ratingModalContent {\n        margin: 0;\n        width: 100%;\n        border-radius: 20px 20px 0 0;\n        max-height: 70vh;\n        padding: 24px 16px 20px;\n    }\n}\n\n/* Reduced motion support */\n@media (prefers-reduced-motion: reduce) {\n    .chatMainFlow .chatMessage,\n    .scrollToBottom,\n    .ratingModal,\n    .ratingModalContent,\n    .feedbackStatus {\n        animation: none;\n    }\n\n    .chatMainFlow .chatMessage .messageText,\n    .chatMainFlow .chatInput textarea,\n    .chatMainFlow .chatInput button,\n    /* [㊗️] .chatMainFlow .chatMessage .avatar img, */\n    .chatButton {\n        transition: none;\n    }\n}\n\n/* High contrast mode support */\n@media (prefers-contrast: high) {\n    .chatMainFlow .chatMessage .messageText {\n        border: 2px solid currentColor;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        border-width: 3px;\n    }\n\n    .chatMainFlow .chatInput button {\n        border: 2px solid currentColor;\n    }\n}\n\n/* Email modal styles */\n.emailContainer {\n    margin-bottom: 20px;\n    display: flex;\n    flex-direction: column;\n    gap: 16px;\n}\n\n.emailModalHeader {\n    background: #f8fafc;\n}\n\n.emailHeaderText {\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n}\n\n.emailHeaderLabel {\n    font-size: 12px;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    color: #64748b;\n}\n\n.emailMetadata {\n    background: #ffffff;\n    border: 1px solid #e2e8f0;\n    border-radius: 12px;\n    padding: 18px 20px;\n    margin-bottom: 16px;\n    box-shadow: 0 1px 4px rgba(15, 23, 42, 0.08);\n}\n\n.emailField {\n    display: flex;\n    gap: 8px;\n    margin-bottom: 8px;\n    font-size: 14px;\n    line-height: 1.5;\n}\n\n.emailField:last-child {\n    margin-bottom: 0;\n}\n\n.emailField strong {\n    min-width: 60px;\n    color: #64748b;\n    font-weight: 600;\n}\n\n.emailRecipients {\n    color: #0f172a;\n    word-break: break-word;\n}\n\n.emailBody {\n    text-align: left;\n}\n\n.emailBody strong {\n    display: block;\n    margin-bottom: 12px;\n    color: #64748b;\n    font-weight: 600;\n    font-size: 14px;\n}\n\n.emailBodyContent {\n    background: #ffffff;\n    border: 1px solid #e2e8f0;\n    border-radius: 12px;\n    padding: 22px;\n    line-height: 1.6;\n    max-height: 400px;\n    overflow-y: auto;\n    box-shadow: 0 1px 4px rgba(15, 23, 42, 0.08);\n}\n\n.emailBodyContent p {\n    margin: 0 0 12px 0;\n}\n\n.emailBodyContent p:last-child {\n    margin-bottom: 0;\n}\n\n.emailStatus {\n    text-transform: capitalize;\n    color: #0f172a;\n    font-weight: 600;\n}\n\n/**\n * TODO: [🌉] DRY Markdown primitives styling\n */\n"]} */";
|
|
14845
|
-
var styles$4 = {"copiedToClipboardMessage":"Chat-module_copiedToClipboardMessage__apCPY","Chat":"Chat-module_Chat__j2eE5","chatMainFlow":"Chat-module_chatMainFlow__--8FE","chatBar":"Chat-module_chatBar__fLECN","TasksInProgress":"Chat-module_TasksInProgress__fQfei","actions":"Chat-module_actions__gTZ5T","portal":"Chat-module_portal__uTOT8","left":"Chat-module_left__7l5Mn","right":"Chat-module_right__ABZrW","actionsFaded":"Chat-module_actionsFaded__qqAVQ","actionsScrolling":"Chat-module_actionsScrolling__fIawW","chatButton":"Chat-module_chatButton__d9VgA","chatButtonText":"Chat-module_chatButtonText__RkGB-","useTemplateButton":"Chat-module_useTemplateButton__xcJNR","chatChildren":"Chat-module_chatChildren__flOPK","chatMessages":"Chat-module_chatMessages__J2u2N","hasActionsAndFirstMessageIsLong":"Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ","chatMessage":"Chat-module_chatMessage__nmLaZ","messageSlideIn":"Chat-module_messageSlideIn__soTy2","messageStack":"Chat-module_messageStack__n0xNL","isMe":"Chat-module_isMe__nBtaV","messageMeta":"Chat-module_messageMeta__eqCRu","messageTimestamp":"Chat-module_messageTimestamp__LDhRw","messageDuration":"Chat-module_messageDuration__Y9Kg9","participantLabel":"Chat-module_participantLabel__0Oo-3","isNotCompleteMessage":"Chat-module_isNotCompleteMessage__Hj2K7","NonCompleteMessageFiller":"Chat-module_NonCompleteMessageFiller__G5-Ve","messageText":"Chat-module_messageText__XgNyQ","loadingPulse":"Chat-module_loadingPulse__VomRm","ongoingToolCalls":"Chat-module_ongoingToolCalls__NZkQN","completedToolCalls":"Chat-module_completedToolCalls__vI1Qt","sourceCitations":"Chat-module_sourceCitations__oDtBX","ongoingToolCall":"Chat-module_ongoingToolCall__WT3Rc","completedToolCall":"Chat-module_completedToolCall__-q4Cs","toolCallOrigin":"Chat-module_toolCallOrigin__-Io6Y","toolCallDetails":"Chat-module_toolCallDetails__WUFlD","toolCallData":"Chat-module_toolCallData__UauCO","ongoingToolCallSpinner":"Chat-module_ongoingToolCallSpinner__7g-Ay","toolCallSpinner":"Chat-module_toolCallSpinner__LSiK6","ongoingToolCallName":"Chat-module_ongoingToolCallName__y59-0","typingIndicator":"Chat-module_typingIndicator__S-CT-","avatar":"Chat-module_avatar__gL6bm","typingBubble":"Chat-module_typingBubble__0Lb7B","typingDots":"Chat-module_typingDots__srOBB","typingDot":"Chat-module_typingDot__dnhKT","typingBounce":"Chat-module_typingBounce__1yp2v","ratingStar":"Chat-module_ratingStar__rRfqC","active":"Chat-module_active__lbYL-","copyButtonContainer":"Chat-module_copyButtonContainer__Rij0U","copyButton":"Chat-module_copyButton__DcxT5","copiedTooltip":"Chat-module_copiedTooltip__LH81j","copiedTooltipFadeIn":"Chat-module_copiedTooltipFadeIn__QekO1","copiedTooltipLeft":"Chat-module_copiedTooltipLeft__j-S-5","copiedTooltipRight":"Chat-module_copiedTooltipRight__R-2cE","chat-code-block":"Chat-module_chat-code-block__k8IyS","attachments":"Chat-module_attachments__m1Fts","attachment":"Chat-module_attachment__aE9hK","attachmentIcon":"Chat-module_attachmentIcon__BX3Cy","attachmentName":"Chat-module_attachmentName__aMx56","messageButtons":"Chat-module_messageButtons__WaOob","messageButton":"Chat-module_messageButton__mRnn-","rating":"Chat-module_rating__soc3M","chatInput":"Chat-module_chatInput__1Ecan","fullPageVisual":"Chat-module_fullPageVisual__zNAEy","dragOver":"Chat-module_dragOver__bkS-g","filePreviewContainer":"Chat-module_filePreviewContainer__R70hm","filePreview":"Chat-module_filePreview__kq2aX","fileIcon":"Chat-module_fileIcon__zoSKW","fileInfo":"Chat-module_fileInfo__wBLi0","fileName":"Chat-module_fileName__bBujo","fileSize":"Chat-module_fileSize__ivliq","removeFileButton":"Chat-module_removeFileButton__0gakR","inputContainer":"Chat-module_inputContainer__bPt99","attachmentButton":"Chat-module_attachmentButton__qLO47","voiceButton":"Chat-module_voiceButton__d2zlP","voiceButtonActive":"Chat-module_voiceButtonActive__Uoi3W","voiceRecordingPulse":"Chat-module_voiceRecordingPulse__y2wJ5","uploadProgress":"Chat-module_uploadProgress__jBTKe","uploadProgressBar":"Chat-module_uploadProgressBar__Gutnt","uploadProgressFill":"Chat-module_uploadProgressFill__EgubT","dragOverlay":"Chat-module_dragOverlay__SEGoS","dragOverlayContent":"Chat-module_dragOverlayContent__gb9kF","scrollToBottomContainer":"Chat-module_scrollToBottomContainer__5rXpK","scrollToBottomWrapper":"Chat-module_scrollToBottomWrapper__PCIu1","scrollButtonSlideIn":"Chat-module_scrollButtonSlideIn__XnImg","scrollToBottom":"Chat-module_scrollToBottom__nzxdZ","scrollToBottomBadge":"Chat-module_scrollToBottomBadge__N6T6-","ratingModal":"Chat-module_ratingModal__XVKYm","modalFadeIn":"Chat-module_modalFadeIn__RPc3w","ratingModalContent":"Chat-module_ratingModalContent__CCdq7","modalSlideIn":"Chat-module_modalSlideIn__XXtgN","stars":"Chat-module_stars__PCzNO","ratingModalStar":"Chat-module_ratingModalStar__XkbHr","toolCallModal":"Chat-module_toolCallModal__uNaIK","searchModalHeader":"Chat-module_searchModalHeader__Y8V-w","selfLearningModalHeader":"Chat-module_selfLearningModalHeader__PDIt1","selfLearningAvatarGroup":"Chat-module_selfLearningAvatarGroup__jgy2k","selfLearningAvatar":"Chat-module_selfLearningAvatar__NHXKm","selfLearningAvatarInitial":"Chat-module_selfLearningAvatarInitial__cuw19","selfLearningTeacher":"Chat-module_selfLearningTeacher__fGhX8","selfLearningHeaderText":"Chat-module_selfLearningHeaderText__LTwHL","selfLearningTitle":"Chat-module_selfLearningTitle__G0SAk","selfLearningMetaRow":"Chat-module_selfLearningMetaRow__z4Tg0","selfLearningMeta":"Chat-module_selfLearningMeta__58Pet","selfLearningMetaChip":"Chat-module_selfLearningMetaChip__0go88","selfLearningCommitments":"Chat-module_selfLearningCommitments__7ECnr","selfLearningCommitmentsLabel":"Chat-module_selfLearningCommitmentsLabel__OkDO1","selfLearningBookEditor":"Chat-module_selfLearningBookEditor__CZy5T","selfLearningEmpty":"Chat-module_selfLearningEmpty__vCKHL","modalCloseButton":"Chat-module_modalCloseButton__AAfWS","teamModalHeader":"Chat-module_teamModalHeader__SB5NL","teamHeaderParticipants":"Chat-module_teamHeaderParticipants__KS7aV","teamHeaderProfile":"Chat-module_teamHeaderProfile__xRlcS","teamHeaderProfileLink":"Chat-module_teamHeaderProfileLink__fYTfT","teamHeaderAvatar":"Chat-module_teamHeaderAvatar__vXcTp","teamHeaderName":"Chat-module_teamHeaderName__ygJR9","teamHeaderDivider":"Chat-module_teamHeaderDivider__cnRMd","searchModalIcon":"Chat-module_searchModalIcon__AR5KY","searchModalQuery":"Chat-module_searchModalQuery__5x-Ra","searchModalContent":"Chat-module_searchModalContent__mWLIE","teamChatContainer":"Chat-module_teamChatContainer__xdLtU","teamToolCallSection":"Chat-module_teamToolCallSection__FH9l7","teamToolCallGroup":"Chat-module_teamToolCallGroup__PO6rl","teamToolCallHeading":"Chat-module_teamToolCallHeading__G-JSj","teamToolCallChips":"Chat-module_teamToolCallChips__hRWqh","teamToolCallDetails":"Chat-module_teamToolCallDetails__ZaTtt","teamToolCallDetailsHeader":"Chat-module_teamToolCallDetailsHeader__FPIgI","teamToolCallDetailsTitle":"Chat-module_teamToolCallDetailsTitle__50SeL","teamToolCallDetailsClear":"Chat-module_teamToolCallDetailsClear__8yvu7","searchResultsList":"Chat-module_searchResultsList__xVDUZ","searchResultsRaw":"Chat-module_searchResultsRaw__i3GkD","searchResultItem":"Chat-module_searchResultItem__bTzq5","searchResultUrl":"Chat-module_searchResultUrl__SbrH4","searchResultTitle":"Chat-module_searchResultTitle__XS7zN","searchResultSnippet":"Chat-module_searchResultSnippet__Wsun2","noResults":"Chat-module_noResults__AAv6s","citationDetails":"Chat-module_citationDetails__2v2e1","citationPreview":"Chat-module_citationPreview__rGhCR","citationIframe":"Chat-module_citationIframe__ngzFY","citationMetadata":"Chat-module_citationMetadata__DPaCF","citationExcerpt":"Chat-module_citationExcerpt__VU4BS","citationHint":"Chat-module_citationHint__5AkLx","toolCallDataContainer":"Chat-module_toolCallDataContainer__rdEzC","toolCallArgsList":"Chat-module_toolCallArgsList__LD3xH","ratingInput":"Chat-module_ratingInput__z8Pv-","ratingActions":"Chat-module_ratingActions__nXcss","downloadButton":"Chat-module_downloadButton__8--bk","saveButtonContainer":"Chat-module_saveButtonContainer__lSNUJ","saveMenu":"Chat-module_saveMenu__-ph8y","saveMenuItem":"Chat-module_saveMenuItem__ISApL","saveMenuDivider":"Chat-module_saveMenuDivider__3ktTM","chatFeedbackPanel":"Chat-module_chatFeedbackPanel__gfOzk","chatFeedbackToggle":"Chat-module_chatFeedbackToggle__IeOM6","chatFeedbackIcon":"Chat-module_chatFeedbackIcon__XcB7A","chatFeedbackMeta":"Chat-module_chatFeedbackMeta__6t2be","chatFeedbackLabel":"Chat-module_chatFeedbackLabel__qQgH3","chatFeedbackDescription":"Chat-module_chatFeedbackDescription__I9bNg","chatFeedbackIndicator":"Chat-module_chatFeedbackIndicator__uHCr7","chatFeedbackIndicatorActive":"Chat-module_chatFeedbackIndicatorActive__bkw-w","chatFeedbackIndicatorInactive":"Chat-module_chatFeedbackIndicatorInactive__qMBkh","pauseButton":"Chat-module_pauseButton__eeu7K","pausing":"Chat-module_pausing__pTx8b","paused":"Chat-module_paused__j-pya","voiceCallIndicatorBar":"Chat-module_voiceCallIndicatorBar__N2sWN","voiceCallIndicator":"Chat-module_voiceCallIndicator__tsaaG","voiceCallIconPulse":"Chat-module_voiceCallIconPulse__zZbJn","voiceCallPulse":"Chat-module_voiceCallPulse__XcGU4","feedbackStatus":"Chat-module_feedbackStatus__-df7l","feedbackStatusSlideIn":"Chat-module_feedbackStatusSlideIn__n6bbB","feedbackStatusSuccess":"Chat-module_feedbackStatusSuccess__Ax2q0","feedbackStatusError":"Chat-module_feedbackStatusError__dnv6d","emailContainer":"Chat-module_emailContainer__TBxXZ","emailModalHeader":"Chat-module_emailModalHeader__Z4-CJ","emailHeaderText":"Chat-module_emailHeaderText__1dR6R","emailHeaderLabel":"Chat-module_emailHeaderLabel__7iAEH","emailMetadata":"Chat-module_emailMetadata__8XXGf","emailField":"Chat-module_emailField__pcb03","emailRecipients":"Chat-module_emailRecipients__YVVys","emailBody":"Chat-module_emailBody__OBt-w","emailBodyContent":"Chat-module_emailBodyContent__-vFo2","emailStatus":"Chat-module_emailStatus__T4UuL"};
|
|
16152
|
+
var css_248z$4 = "@font-face{font-family:OpenMojiBlack;src:url(https://s6.ptbk.io/fonts/OpenMoji-black-glyf.woff2) format(\"woff2\");unicode-range:u+23,u+2a,u+2d,u+30-39,u+a9,u+ae,u+200d,u+203c,u+2049,u+20e3,u+2117,u+2120,u+2122,u+2139,u+2194-2199,u+21a9,u+21aa,u+229c,u+231a,u+231b,u+2328,u+23cf,u+23e9-23f3,u+23f8-23fe,u+24c2,u+25a1,u+25aa-25ae,u+25b6,u+25c0,u+25c9,u+25d0,u+25d1,u+25e7-25ea,u+25ed,u+25ee,u+25fb-25fe,u+2600-2605,u+260e,u+2611,u+2614,u+2615,u+2618,u+261d,u+2620,u+2622,u+2623,u+2626,u+262a,u+262e,u+262f,u+2638-263a,u+2640,u+2642,u+2648-2653,u+265f,u+2660,u+2663,u+2665,u+2666,u+2668,u+267b,u+267e,u+267f,u+2691-2697,u+2699,u+269b,u+269c,u+26a0,u+26a1,u+26a7,u+26aa,u+26ab,u+26b0,u+26b1,u+26bd,u+26be,u+26c4,u+26c5,u+26c8,u+26ce,u+26cf,u+26d1,u+26d3,u+26d4,u+26e9,u+26ea,u+26f0-26f5,u+26f7-26fa,u+26fd,u+2702,u+2705,u+2708-270d,u+270f,u+2712,u+2714,u+2716,u+271d,u+2721,u+2728,u+2733,u+2734,u+2744,u+2747,u+274c,u+274e,u+2753-2755,u+2757,u+2763,u+2764,u+2795-2797,u+27a1,u+27b0,u+27bf,u+2934,u+2935,u+2b05-2b07,u+2b0c,u+2b0d,u+2b1b,u+2b1c,u+2b1f-2b24,u+2b2e,u+2b2f,u+2b50,u+2b55,u+2b58,u+2b8f,u+2bba-2bbc,u+2bc3,u+2bc4,u+2bea,u+2beb,u+3030,u+303d,u+3297,u+3299,u+e000-e009,u+e010,u+e011,u+e040-e06d,u+e080-e0b4,u+e0c0-e0cc,u+e0ff-e10d,u+e140-e14a,u+e150-e157,u+e181-e189,u+e1c0-e1c4,u+e1c6-e1d9,u+e200-e216,u+e240-e269,u+e280-e283,u+e2c0-e2c4,u+e2c6-e2da,u+e300-e303,u+e305-e30f,u+e312-e316,u+e318-e322,u+e324-e329,u+e32b,u+e340-e348,u+e380,u+e381,u+f000,u+f77a,u+f8ff,u+fe0f,u+1f004,u+1f0cf,u+1f10d-1f10f,u+1f12f,u+1f16d-1f171,u+1f17e,u+1f17f,u+1f18e,u+1f191-1f19a,u+1f1e6-1f1ff,u+1f201,u+1f202,u+1f21a,u+1f22f,u+1f232-1f23a,u+1f250,u+1f251,u+1f260-1f265,u+1f300-1f321,u+1f324-1f393,u+1f396,u+1f397,u+1f399-1f39b,u+1f39e-1f3f0,u+1f3f3-1f3f5,u+1f3f7-1f4fd,u+1f4ff-1f53d,u+1f549-1f54e,u+1f550-1f567,u+1f56f,u+1f570,u+1f573-1f57a,u+1f587,u+1f58a-1f58d,u+1f590,u+1f595,u+1f596,u+1f5a4,u+1f5a5,u+1f5a8,u+1f5b1,u+1f5b2,u+1f5bc,u+1f5c2-1f5c4,u+1f5d1-1f5d3,u+1f5dc-1f5de,u+1f5e1,u+1f5e3,u+1f5e8,u+1f5ef,u+1f5f3,u+1f5fa-1f64f,u+1f680-1f6c5,u+1f6cb-1f6d2,u+1f6d5-1f6d7,u+1f6dc-1f6e5,u+1f6e9,u+1f6eb,u+1f6ec,u+1f6f0,u+1f6f3-1f6fc,u+1f7e0-1f7eb,u+1f7f0,u+1f90c-1f93a,u+1f93c-1f945,u+1f947-1f9ff,u+1fa70-1fa7c,u+1fa80-1fa88,u+1fa90-1fabd,u+1fabf-1fac5,u+1face-1fadb,u+1fae0-1fae8,u+1faf0-1faf8,u+1fbc5-1fbc9,u+e0061-e0067,u+e0069,u+e006c-e0079,u+e007f}@font-face{font-display:swap;font-family:OpenMojiColor;src:url(https://s6.ptbk.io/fonts/OpenMoji-color-cbdt.woff2) format(\"woff2\");unicode-range:u+23,u+2a,u+2d,u+30-39,u+a9,u+ae,u+200d,u+203c,u+2049,u+20e3,u+2117,u+2120,u+2122,u+2139,u+2194-2199,u+21a9,u+21aa,u+229c,u+231a,u+231b,u+2328,u+23cf,u+23e9-23f3,u+23f8-23fe,u+24c2,u+25a1,u+25aa-25ae,u+25b6,u+25c0,u+25c9,u+25d0,u+25d1,u+25e7-25ea,u+25ed,u+25ee,u+25fb-25fe,u+2600-2605,u+260e,u+2611,u+2614,u+2615,u+2618,u+261d,u+2620,u+2622,u+2623,u+2626,u+262a,u+262e,u+262f,u+2638-263a,u+2640,u+2642,u+2648-2653,u+265f,u+2660,u+2663,u+2665,u+2666,u+2668,u+267b,u+267e,u+267f,u+2691-2697,u+2699,u+269b,u+269c,u+26a0,u+26a1,u+26a7,u+26aa,u+26ab,u+26b0,u+26b1,u+26bd,u+26be,u+26c4,u+26c5,u+26c8,u+26ce,u+26cf,u+26d1,u+26d3,u+26d4,u+26e9,u+26ea,u+26f0-26f5,u+26f7-26fa,u+26fd,u+2702,u+2705,u+2708-270d,u+270f,u+2712,u+2714,u+2716,u+271d,u+2721,u+2728,u+2733,u+2734,u+2744,u+2747,u+274c,u+274e,u+2753-2755,u+2757,u+2763,u+2764,u+2795-2797,u+27a1,u+27b0,u+27bf,u+2934,u+2935,u+2b05-2b07,u+2b0c,u+2b0d,u+2b1b,u+2b1c,u+2b1f-2b24,u+2b2e,u+2b2f,u+2b50,u+2b55,u+2b58,u+2b8f,u+2bba-2bbc,u+2bc3,u+2bc4,u+2bea,u+2beb,u+3030,u+303d,u+3297,u+3299,u+e000-e009,u+e010,u+e011,u+e040-e06d,u+e080-e0b4,u+e0c0-e0cc,u+e0ff-e10d,u+e140-e14a,u+e150-e157,u+e181-e189,u+e1c0-e1c4,u+e1c6-e1d9,u+e200-e216,u+e240-e269,u+e280-e283,u+e2c0-e2c4,u+e2c6-e2da,u+e300-e303,u+e305-e30f,u+e312-e316,u+e318-e322,u+e324-e329,u+e32b,u+e340-e348,u+e380,u+e381,u+f000,u+f77a,u+f8ff,u+fe0f,u+1f004,u+1f0cf,u+1f10d-1f10f,u+1f12f,u+1f16d-1f171,u+1f17e,u+1f17f,u+1f18e,u+1f191-1f19a,u+1f1e6-1f1ff,u+1f201,u+1f202,u+1f21a,u+1f22f,u+1f232-1f23a,u+1f250,u+1f251,u+1f260-1f265,u+1f300-1f321,u+1f324-1f393,u+1f396,u+1f397,u+1f399-1f39b,u+1f39e-1f3f0,u+1f3f3-1f3f5,u+1f3f7-1f4fd,u+1f4ff-1f53d,u+1f549-1f54e,u+1f550-1f567,u+1f56f,u+1f570,u+1f573-1f57a,u+1f587,u+1f58a-1f58d,u+1f590,u+1f595,u+1f596,u+1f5a4,u+1f5a5,u+1f5a8,u+1f5b1,u+1f5b2,u+1f5bc,u+1f5c2-1f5c4,u+1f5d1-1f5d3,u+1f5dc-1f5de,u+1f5e1,u+1f5e3,u+1f5e8,u+1f5ef,u+1f5f3,u+1f5fa-1f64f,u+1f680-1f6c5,u+1f6cb-1f6d2,u+1f6d5-1f6d7,u+1f6dc-1f6e5,u+1f6e9,u+1f6eb,u+1f6ec,u+1f6f0,u+1f6f3-1f6fc,u+1f7e0-1f7eb,u+1f7f0,u+1f90c-1f93a,u+1f93c-1f945,u+1f947-1f9ff,u+1fa70-1fa7c,u+1fa80-1fa88,u+1fa90-1fabd,u+1fabf-1fac5,u+1face-1fadb,u+1fae0-1fae8,u+1faf0-1faf8,u+1fbc5-1fbc9,u+e0061-e0067,u+e0069,u+e006c-e0079,u+e007f}.Chat-module_copiedToClipboardMessage__apCPY{background:#222;border-radius:8px;box-shadow:0 2px 12px rgba(0,0,0,.18);color:#fff;font-size:1.1em;left:50%;opacity:.97;padding:10px 24px;pointer-events:none;position:fixed;top:32px;transform:translateX(-50%);z-index:9999}.Chat-module_Chat__j2eE5{--chat-message-min-width:150px;--chat-message-max-width:min(70%,600px);--chat-message-avatar-gap:12px;display:flex;flex-direction:column;font-family:Arial,Helvetica,sans-serif,OpenMojiColor,OpenMojiBlack;height:100%;width:100%}.Chat-module_chatMainFlow__--8FE{display:grid;grid-template:\"🟦\" min-content \"💬\" 1fr \"📝\" min-content/1fr;height:100%;max-width:100vw;width:100%}.Chat-module_chatMainFlow__--8FE .Chat-module_chatBar__fLECN{background-color:#fff;border-bottom:1px solid rgba(15,23,36,.06);color:#0f1724;font-weight:500;grid-area:🟦;padding:16px 20px;text-align:center;width:100%}.Chat-module_TasksInProgress__fQfei{align-self:center;grid-area:🟦;height:min-content;justify-self:self-end;margin:8px 16px;width:auto}.Chat-module_actions__gTZ5T{align-items:center;align-self:self-start;display:flex;gap:8px;grid-area:💬;height:min-content;justify-self:self-end;margin:16px 20px 0;transition:opacity .2s ease;width:auto;will-change:opacity;z-index:40}.Chat-module_actions__gTZ5T.Chat-module_portal__uTOT8{margin:0}.Chat-module_actions__gTZ5T.Chat-module_left__7l5Mn{justify-self:self-start}.Chat-module_actions__gTZ5T.Chat-module_right__ABZrW{justify-self:self-end}.Chat-module_actionsFaded__qqAVQ{opacity:.3}.Chat-module_actionsScrolling__fIawW{opacity:.15;pointer-events:none}@media (max-width:900px){.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}}@media (max-width:600px){.Chat-module_actions__gTZ5T{gap:7px;margin:14px 18px 0}}.Chat-module_chatMainFlow__--8FE .Chat-module_chatChildren__flOPK{grid-area:💬;height:100%;width:100%;z-index:300}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{grid-area:💬;height:100%;overflow-x:hidden;overflow-y:auto;padding:24px 20px 16px;scroll-behavior:smooth;width:100%;z-index:10}.Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ{padding-top:80px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar{width:6px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-track{background:transparent}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-thumb{background:hsla(0,0%,49%,.2);border-radius:3px;transition:all .2s ease}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N::-webkit-scrollbar-thumb:hover{background:hsla(0,0%,49%,.3)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ{align-items:flex-end;animation:Chat-module_messageSlideIn__soTy2 .4s cubic-bezier(.25,.46,.45,.94);column-gap:var(--chat-message-avatar-gap);display:flex;flex-direction:row;margin-bottom:20px;max-width:100%;position:relative}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageStack__n0xNL{align-items:flex-start;display:flex;flex-direction:column;max-width:var(--chat-message-max-width)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageStack__n0xNL{align-items:flex-end}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageStack__n0xNL>*{max-width:100%}.Chat-module_messageMeta__eqCRu{align-items:center;color:#6b7280;display:flex;font-size:11px;gap:6px;line-height:1.4;margin:4px 6px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageMeta__eqCRu{align-self:flex-end;text-align:right}.Chat-module_messageTimestamp__LDhRw{letter-spacing:.02em}.Chat-module_messageDuration__Y9Kg9{opacity:.85}.Chat-module_participantLabel__0Oo-3{color:#64748b;font-size:12px;font-weight:600;margin:0 0 6px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_participantLabel__0Oo-3{text-align:right}@keyframes Chat-module_messageSlideIn__soTy2{0%{opacity:0;transform:translateY(20px) scale(.95)}to{opacity:1;transform:translateY(0) scale(1)}}.Chat-module_isNotCompleteMessage__Hj2K7{opacity:.7;position:relative}.Chat-module_NonCompleteMessageFiller__G5-Ve{color:transparent}.Chat-module_isNotCompleteMessage__Hj2K7 .Chat-module_messageText__XgNyQ:after{animation:Chat-module_loadingPulse__VomRm 1.5s ease-in-out infinite;background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.6),transparent);border-radius:2px;bottom:8px;content:\"\";height:4px;position:absolute;right:12px;width:20px}.Chat-module_completedToolCalls__vI1Qt,.Chat-module_ongoingToolCalls__NZkQN,.Chat-module_sourceCitations__oDtBX{border-top:1px solid hsla(0,0%,100%,.2);display:flex;flex-direction:row;flex-wrap:wrap;gap:8px;margin-top:8px;padding-top:8px}.Chat-module_completedToolCall__-q4Cs,.Chat-module_ongoingToolCall__WT3Rc{align-items:center;background:hsla(0,0%,100%,.15);border:1px solid hsla(0,0%,100%,.2);border-radius:12px;box-shadow:0 1px 3px rgba(0,0,0,.1);color:inherit;display:flex;font-family:OpenMojiColor,OpenMojiBlack,Arial,Helvetica,sans-serif;font-size:.8em;gap:8px;padding:4px 10px}.Chat-module_completedToolCall__-q4Cs{cursor:pointer;transition:all .2s ease}.Chat-module_completedToolCall__-q4Cs:hover{background:hsla(0,0%,100%,.3);box-shadow:0 2px 6px rgba(0,0,0,.15);transform:translateY(-1px)}.Chat-module_toolCallOrigin__-Io6Y{font-size:.85em;font-weight:500;opacity:.75;white-space:nowrap}.Chat-module_toolCallDetails__WUFlD{margin-bottom:24px}.Chat-module_toolCallDetails__WUFlD p{margin-bottom:8px}.Chat-module_toolCallData__UauCO{background:#f1f5f9;border:1px solid #e2e8f0;border-radius:8px;font-size:13px;max-height:300px;overflow-x:auto;padding:12px;word-break:break-all}.Chat-module_ongoingToolCallSpinner__7g-Ay{animation:Chat-module_toolCallSpinner__LSiK6 .8s linear infinite;border:2px solid hsla(0,0%,100%,.3);border-radius:50%;border-top-color:#fff;height:14px;width:14px}.Chat-module_ongoingToolCallName__y59-0{font-weight:500}@keyframes Chat-module_toolCallSpinner__LSiK6{to{transform:rotate(1turn)}}@keyframes Chat-module_loadingPulse__VomRm{0%,to{opacity:.3;transform:scaleX(.8)}50%{opacity:1;transform:scaleX(1.2)}}.Chat-module_typingIndicator__S-CT-{align-items:flex-end;animation:Chat-module_messageSlideIn__soTy2 .4s cubic-bezier(.25,.46,.45,.94);display:flex;margin-bottom:20px}.Chat-module_typingIndicator__S-CT- .Chat-module_avatar__gL6bm{flex-shrink:0;height:40px;margin:0 12px 4px;width:40px}.Chat-module_typingBubble__0Lb7B{backdrop-filter:blur(10px);border:1px solid hsla(0,0%,49%,.1);border-radius:20px;border-bottom-left-radius:6px;box-shadow:0 2px 8px rgba(0,0,0,.1);min-height:24px;padding:16px 20px}.Chat-module_typingBubble__0Lb7B,.Chat-module_typingDots__srOBB{align-items:center;display:flex;gap:4px}.Chat-module_typingDot__dnhKT{animation:Chat-module_typingBounce__1yp2v 1.4s ease-in-out infinite;background:linear-gradient(135deg,#6b7280,hsla(0,0%,49%,.6));border-radius:50%;box-shadow:0 2px 4px rgba(0,0,0,.1);height:8px;width:8px}.Chat-module_typingDot__dnhKT:first-child{animation-delay:-.32s}.Chat-module_typingDot__dnhKT:nth-child(2){animation-delay:-.16s}.Chat-module_typingDot__dnhKT:nth-child(3){animation-delay:0s}@keyframes Chat-module_typingBounce__1yp2v{0%,80%,to{opacity:.5;transform:scale(.8) translateY(0)}40%{opacity:1;transform:scale(1.2) translateY(-8px)}}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV{align-items:flex-end;flex-direction:row-reverse;justify-content:flex-start}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ.Chat-module_isMe__nBtaV .Chat-module_messageText__XgNyQ{border-bottom-right-radius:6px}.Chat-module_ratingStar__rRfqC{color:var(--star-inactive-color,#ccc);cursor:pointer;font-size:20px;transition:color .2s}.Chat-module_ratingStar__rRfqC.Chat-module_active__lbYL-{color:gold}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{aspect-ratio:1/1;flex-shrink:0;margin:0 0 4px;position:relative;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{word-wrap:break-word;--chat-heading-surface:hsla(0,0%,100%,.12);--chat-heading-accent:rgba(56,189,248,.85);--chat-heading-text-color:var(--message-text-color,#0f172a);--chat-heading-border-color:rgba(15,23,42,.18);--chat-heading-shadow-color:rgba(15,23,42,.65);backdrop-filter:blur(10px);background-color:var(--message-bg-color);border-radius:20px;box-shadow:0 2px 8px rgba(0,0,0,.1);color:var(--message-text-color);font-size:15px;line-height:1.5;margin-bottom:4px;max-width:100%;min-width:var(--chat-message-min-width);padding:14px 18px;position:relative;text-align:left;transition:all .2s ease}@supports (color:color-mix(in srgb,#000 50%,#fff 50%)){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{--chat-heading-surface:color-mix(in srgb,var(--message-bg-color,#fff) 70%,rgba(15,23,42,.08) 30%);--chat-heading-accent:color-mix(in srgb,var(--message-text-color,#0f172a) 65%,rgba(59,130,246,.9) 35%);--chat-heading-border-color:color-mix(in srgb,var(--message-bg-color,#fff) 60%,rgba(15,23,42,.12) 40%);--chat-heading-shadow-color:color-mix(in srgb,rgba(15,23,42,.65) 80%,rgba(96,165,250,.45) 20%);--chat-heading-text-color:var(--message-text-color,#0f172a)}}.Chat-module_copyButtonContainer__Rij0U{float:right;pointer-events:none;right:10px;top:8px;visibility:hidden;z-index:2}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ:hover .Chat-module_copyButtonContainer__Rij0U,.Chat-module_copyButtonContainer__Rij0U:focus-within{pointer-events:auto;visibility:visible}.Chat-module_messageControlGroup__gAo4k{align-items:center;display:flex;gap:6px;pointer-events:auto}.Chat-module_copyButton__DcxT5{align-items:center;background:hsla(0,0%,100%,.2);border:1px solid #ddd;border-radius:6px;box-shadow:0 1px 4px rgba(0,0,0,.07);cursor:pointer;display:flex;opacity:.7;padding:2px 5px;position:relative;transition:all .15s,box-shadow .15s,border .15s}.Chat-module_playButton__ZigYL{align-items:center;background:hsla(0,0%,100%,.25);border:1px solid hsla(0,0%,100%,.6);border-radius:12px;color:#111;cursor:pointer;display:inline-flex;height:36px;justify-content:center;padding:0;position:relative;transition:all .15s ease;width:36px}.Chat-module_playButtonIcon__wPErq{height:16px;width:16px}.Chat-module_playButtonSpinner__Ad33i{animation:Chat-module_messagePlaySpinner__ldlvy .9s linear infinite;border:2px solid rgba(0,0,0,.3);border-radius:50%;border-top-color:#111;height:16px;width:16px}.Chat-module_playButton__ZigYL:hover:not(:disabled){background:hsla(0,0%,100%,.4);border-color:hsla(0,0%,100%,.8)}.Chat-module_playButton__ZigYL:disabled{cursor:not-allowed;opacity:.6}@keyframes Chat-module_messagePlaySpinner__ldlvy{0%{transform:rotate(0deg)}to{transform:rotate(1turn)}}.Chat-module_copiedTooltip__LH81j{animation:Chat-module_copiedTooltipFadeIn__QekO1 .2s;background:#222;border-radius:8px;box-shadow:0 2px 12px rgba(0,0,0,.18);color:#fff;font-size:.98em;left:50%;margin-top:4px;max-width:220px;opacity:.97;overflow-wrap:break-word;padding:6px 16px;pointer-events:none;position:absolute;top:110%;transform:translateX(-50%);white-space:nowrap;word-break:break-word;z-index:100}.Chat-module_copiedTooltipLeft__j-S-5{left:0!important;transform:none!important}.Chat-module_copiedTooltipRight__R-2cE{left:auto!important;right:0!important;transform:none!important}@keyframes Chat-module_copiedTooltipFadeIn__QekO1{0%{opacity:0;transform:translateX(-50%) translateY(8px) scale(.97)}to{opacity:.97;transform:translateX(-50%) translateY(0) scale(1)}}.Chat-module_copyButton__DcxT5:focus,.Chat-module_copyButton__DcxT5:hover{border:1.5px solid #bbb;box-shadow:0 2px 8px rgba(0,132,255,.1);opacity:1}.Chat-module_copyButton__DcxT5 svg{display:block}.Chat-module_messageText__XgNyQ ul{list-style:disc;margin-left:20px}.Chat-module_messageText__XgNyQ ol{list-style:decimal;margin-left:20px}.Chat-module_messageText__XgNyQ blockquote,.Chat-module_messageText__XgNyQ img,.Chat-module_messageText__XgNyQ pre,.Chat-module_messageText__XgNyQ table{border-radius:8px;margin-bottom:10px;margin-top:10px}.Chat-module_messageText__XgNyQ pre{background:#000;color:#fff}.Chat-module_messageText__XgNyQ blockquote,.Chat-module_messageText__XgNyQ pre{border:none;box-shadow:none;display:block;font-size:inherit;line-height:inherit;padding:1em}.Chat-module_messageText__XgNyQ blockquote{background:#ffffffcc;color:#000}.Chat-module_messageText__XgNyQ code{background:#cccccc55;border:none;box-shadow:none;color:inherit;display:inline-block;font-size:inherit;line-height:inherit;margin:0;padding:0}.Chat-module_messageText__XgNyQ pre code{background-color:#000000cc;border-radius:8px}.Chat-module_messageText__XgNyQ .Chat-module_chat-code-block__k8IyS{background:#181c23;border-color:#23272f;box-shadow:0 2px 8px rgba(0,0,0,.12);color:#f8fafc;font-family:Fira Mono,Menlo,Consolas,Liberation Mono,monospace;font-size:14px;line-height:1.6;overflow-x:auto}.Chat-module_messageText__XgNyQ .Chat-module_chat-code-block__k8IyS code{background:none!important;border:none!important;box-shadow:none!important;color:inherit!important;display:block;font-family:inherit!important;font-size:inherit!important;overflow-x:auto;padding:0!important;white-space:pre;word-break:break-word}.Chat-module_messageText__XgNyQ table{background:#f8fafc;border-collapse:separate;border-spacing:0;box-shadow:0 2px 8px rgba(0,0,0,.08);color:#17223b;font-size:14px;margin:16px 0;width:100%}.Chat-module_messageText__XgNyQ td,.Chat-module_messageText__XgNyQ th{background:none;border-bottom:1px solid #d1dbe8;color:#17223b;padding:10px 16px;text-align:left}.Chat-module_messageText__XgNyQ th{background:linear-gradient(90deg,#eaf3fa 80%,#d1e3f8);border-bottom:2px solid #b5c7de;color:#17223b;font-weight:700}.Chat-module_messageText__XgNyQ tr:last-child td{border-bottom:none}.Chat-module_messageText__XgNyQ tr:nth-child(2n) td{background:#eaf3fa}.Chat-module_messageText__XgNyQ tr:hover td{background:#cbe0f7;transition:background .2s}.Chat-module_messageText__XgNyQ table{border-radius:12px;overflow:hidden}.Chat-module_messageText__XgNyQ td:first-child,.Chat-module_messageText__XgNyQ th:first-child{border-top-left-radius:12px}.Chat-module_messageText__XgNyQ td:last-child,.Chat-module_messageText__XgNyQ th:last-child{border-top-right-radius:12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ:hover{box-shadow:0 4px 16px rgba(0,0,0,.15);transform:translateY(-1px)}.Chat-module_attachments__m1Fts{border-top:1px solid hsla(0,0%,49%,.2);display:flex;flex-wrap:wrap;gap:8px;margin-top:10px;padding-top:10px}.Chat-module_attachment__aE9hK{align-items:center;background:hsla(0,0%,100%,.2);border:1px solid hsla(0,0%,49%,.3);border-radius:12px;color:inherit;display:flex;font-size:13px;gap:6px;max-width:200px;padding:6px 12px;text-decoration:none;transition:all .2s ease}.Chat-module_attachment__aE9hK:hover{background:hsla(0,0%,100%,.3);border-color:hsla(0,0%,49%,.5);transform:translateY(-1px)}.Chat-module_attachmentIcon__BX3Cy{font-size:14px;opacity:.8}.Chat-module_attachmentName__aMx56{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_messageButtons__WaOob{border-top:1px solid hsla(0,0%,49%,.83);display:flex;flex-wrap:wrap;gap:8px;margin-top:12px;padding-top:12px}.Chat-module_messageButton__mRnn-{-webkit-tap-highlight-color:transparent;align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.9);border-radius:16px;cursor:pointer;display:inline-flex;font-size:13px;font-weight:500;padding:8px 14px;touch-action:manipulation;transition:all .2s ease;user-select:none}.Chat-module_messageButton__mRnn-:hover{background:rgba(0,132,255,.1);border-color:rgba(0,132,255,.3);box-shadow:0 2px 8px rgba(0,132,255,.15);transform:translateY(-1px)}.Chat-module_messageButton__mRnn-:active{transform:scale(.98);transition:transform .1s ease}.Chat-module_messageButton__mRnn- p{line-height:inherit;margin:0;padding:0}.Chat-module_messageButton__mRnn- strong{font-weight:600}.Chat-module_messageButton__mRnn- em{font-style:italic}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M{align-items:center;backdrop-filter:blur(10px);background:rgba(0,0,0,.8);border-radius:12px;bottom:-8px;display:flex;gap:2px;min-width:24px;opacity:0;padding:4px 6px;position:absolute;right:8px;transform:translateY(4px);transition:all .3s cubic-bezier(.25,.46,.45,.94);z-index:1}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ:hover .Chat-module_rating__soc3M{opacity:1;transform:translateY(0)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M:hover{background:rgba(0,0,0,.9);box-shadow:0 4px 12px rgba(0,0,0,.3);padding:6px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M span{cursor:pointer;display:inline-block;font-size:16px;transition:transform .2s ease,color .2s ease}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_rating__soc3M:hover span{transform:scale(1.1)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{display:flex;flex-direction:column;gap:12px;grid-area:📝;padding:24px;position:relative;width:100%;z-index:10}.Chat-module_Chat__j2eE5.Chat-module_fullPageVisual__zNAEy .Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{backdrop-filter:blur(25px);background:linear-gradient(180deg,hsla(0,0%,100%,0),hsla(0,0%,100%,.8))}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan.Chat-module_dragOver__bkS-g{background:linear-gradient(0deg,rgba(0,132,255,.1) 0,rgba(0,132,255,.05))}.Chat-module_filePreviewContainer__R70hm{display:flex;flex-wrap:wrap;gap:8px;margin-bottom:8px}.Chat-module_filePreview__kq2aX{align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.2);border-radius:8px;display:flex;font-size:12px;gap:8px;padding:8px 12px;transition:all .2s ease}.Chat-module_filePreview__kq2aX:hover{background:hsla(0,0%,49%,.15);border-color:hsla(0,0%,49%,.3)}.Chat-module_fileIcon__zoSKW{font-size:14px;opacity:.7}.Chat-module_fileInfo__wBLi0{display:flex;flex-direction:column;gap:2px;min-width:0}.Chat-module_fileName__bBujo{color:#000;font-weight:500;max-width:150px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_fileSize__ivliq{color:#6b7280;font-size:11px}.Chat-module_removeFileButton__0gakR{align-items:center;background:rgba(255,0,0,.1);border:none;border-radius:50%;color:#f44;cursor:pointer;display:flex;flex-shrink:0;height:20px;justify-content:center;transition:all .2s ease;width:20px}.Chat-module_removeFileButton__0gakR:hover{background:rgba(255,0,0,.2);transform:scale(1.1)}.Chat-module_inputContainer__bPt99{align-items:flex-end;background:#fff;border:1px solid rgba(0,0,0,.08);border-radius:28px;box-shadow:0 4px 20px rgba(0,0,0,.06);display:flex;gap:8px;padding:8px 12px;transition:all .3s cubic-bezier(.25,.46,.45,.94)}.Chat-module_inputContainer__bPt99:focus-within{border-color:var(--brand-color);box-shadow:0 8px 32px rgba(0,132,255,.12);transform:translateY(-2px)}.Chat-module_inputContainer__bPt99 textarea::placeholder{color:#94a3b8;opacity:1}.Chat-module_attachmentButton__qLO47{align-items:center;background:transparent;border:none;border-radius:50%;color:#64748b;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;margin-bottom:2px;transition:all .2s ease;width:40px}.Chat-module_attachmentButton__qLO47:hover:not(:disabled){background:#f1f5f9;color:#0f172a}.Chat-module_attachmentButton__qLO47:disabled{cursor:not-allowed;opacity:.3}.Chat-module_voiceButton__d2zlP{align-items:center;background:transparent;border:none;border-radius:50%;color:#64748b;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;margin-bottom:2px;transition:all .2s ease;width:40px}.Chat-module_voiceButton__d2zlP:hover:not(:disabled){background:#f1f5f9;color:#0f172a}.Chat-module_voiceButtonActive__Uoi3W{animation:Chat-module_voiceRecordingPulse__y2wJ5 1.5s infinite;background:rgba(239,68,68,.1)!important;color:#ef4444!important}@keyframes Chat-module_voiceRecordingPulse__y2wJ5{0%{box-shadow:0 0 0 0 rgba(255,0,0,.4)}70%{box-shadow:0 0 0 10px rgba(255,0,0,0)}to{box-shadow:0 0 0 0 rgba(255,0,0,0)}}.Chat-module_uploadProgress__jBTKe{align-items:center;background:rgba(0,132,255,.1);border:1px solid rgba(0,132,255,.2);border-radius:8px;color:#0084ff;display:flex;font-size:13px;gap:12px;padding:8px 12px}.Chat-module_uploadProgressBar__Gutnt{background:rgba(0,132,255,.2);border-radius:2px;flex:1;height:4px;overflow:hidden}.Chat-module_uploadProgressFill__EgubT{animation:Chat-module_uploadProgress__jBTKe 1.5s ease-in-out infinite;background:linear-gradient(90deg,#0084ff,rgba(0,132,255,.8));border-radius:2px;height:100%}@keyframes Chat-module_uploadProgress__jBTKe{0%{transform:translateX(-100%)}50%{transform:translateX(0)}to{transform:translateX(100%)}}.Chat-module_dragOverlay__SEGoS{align-items:center;backdrop-filter:blur(10px);background:rgba(0,132,255,.1);border:2px dashed rgba(0,132,255,.5);border-radius:12px;bottom:0;display:flex;justify-content:center;left:0;pointer-events:none;position:absolute;right:0;top:0;z-index:20}.Chat-module_dragOverlayContent__gb9kF{align-items:center;color:#0084ff;display:flex;flex-direction:column;font-weight:600;gap:12px;text-align:center}.Chat-module_dragOverlayContent__gb9kF svg{opacity:.7}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{-webkit-tap-highlight-color:transparent;appearance:none;-webkit-appearance:none;background:transparent;border:none;color:#0f172a;flex:1;font-size:15px;line-height:1.5;margin:0;max-height:200px;min-width:100px;outline:none;padding:10px 12px;resize:none;touch-action:manipulation}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea:disabled{cursor:not-allowed;opacity:.6;transform:none}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea::placeholder{color:inherit;opacity:.7}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]{-webkit-tap-highlight-color:transparent;align-items:center;aspect-ratio:1/1;border:none;border-radius:50%!important;box-shadow:0 4px 12px rgba(0,132,255,.25);color:#fff;display:flex;height:40px;justify-content:center;margin:0 0 2px!important;min-height:unset!important;min-width:unset!important;overflow:visible;padding:0!important;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94);user-select:none;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]:hover{box-shadow:0 6px 16px rgba(0,132,255,.35);transform:scale(1.05)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action]:active{transform:scale(.95)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button[data-button-type=call-to-action] svg{height:20px;width:20px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button img{height:100%;object-fit:contain;width:50%}.Chat-module_scrollToBottomContainer__5rXpK{align-items:flex-start;display:flex;grid-area:📝;height:100%;justify-content:center;pointer-events:none;width:100%;z-index:20}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1{align-items:center;animation:Chat-module_scrollButtonSlideIn__XnImg .3s ease-out;display:inline-flex;justify-content:center;pointer-events:all;position:relative;transform:translate(-50%,-150%);transition:transform .3s cubic-bezier(.25,.46,.45,.94)}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottom__nzxdZ{-webkit-tap-highlight-color:transparent;align-items:center;background:rgba(0,0,0,.5);border:none;border-radius:50%;box-shadow:0 4px 16px rgba(0,0,0,.3);color:#fff;cursor:pointer;display:flex;font-weight:700;height:48px;justify-content:center;outline:none;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94);user-select:none;width:48px}@keyframes Chat-module_scrollButtonSlideIn__XnImg{0%{opacity:0;transform:translate(-50%,20px) scale(.8)}to{opacity:.9;transform:translate(-50%) scale(1)}}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1:hover{transform:translate(-50%,-160%) scale(1.05)}.Chat-module_scrollToBottomContainer__5rXpK .Chat-module_scrollToBottomWrapper__PCIu1:active{transform:translate(-50%,-160%) scale(.95);transition:transform .1s ease}.Chat-module_scrollToBottomBadge__N6T6-{align-items:center;background:#2563eb;border-radius:999px;box-shadow:0 2px 10px rgba(37,99,235,.4);color:#fff;display:inline-flex;font-size:10px;font-weight:600;justify-content:center;letter-spacing:.04em;min-width:32px;padding:2px 6px;pointer-events:none;position:absolute;right:-8px;text-transform:uppercase;top:-8px;white-space:nowrap}.Chat-module_ratingModal__XVKYm{align-items:center;animation:Chat-module_modalFadeIn__RPc3w .3s ease-out;backdrop-filter:blur(8px);background-color:rgba(0,0,0,.6);bottom:0;display:flex;justify-content:center;left:0;position:fixed;right:0;top:0;z-index:1000}@keyframes Chat-module_modalFadeIn__RPc3w{0%{backdrop-filter:blur(0);opacity:0}to{backdrop-filter:blur(8px);opacity:1}}.Chat-module_ratingModalContent__CCdq7{animation:Chat-module_modalSlideIn__XXtgN .3s cubic-bezier(.25,.46,.45,.94);backdrop-filter:blur(20px);background:#fff;border:1px solid hsla(0,0%,49%,.1);border-radius:16px;box-shadow:0 20px 60px rgba(0,0,0,.3);color:#0f1724;font-family:Arial,Helvetica,sans-serif,OpenMojiColor,OpenMojiBlack;max-width:480px;padding:32px;width:90%}@keyframes Chat-module_modalSlideIn__XXtgN{0%{opacity:0;transform:translateY(20px) scale(.95)}to{opacity:1;transform:translateY(0) scale(1)}}.Chat-module_ratingModalContent__CCdq7 h3{color:#000;font-size:20px;font-weight:600;margin:0 0 24px;text-align:center}.Chat-module_stars__PCzNO{display:flex;gap:8px;justify-content:center;margin-bottom:24px}.Chat-module_stars__PCzNO span{border-radius:8px;cursor:pointer;font-size:28px;padding:4px;transition:all .2s ease}.Chat-module_stars__PCzNO span:hover{background:rgba(255,215,0,.1);transform:scale(1.2)}.Chat-module_ratingModalStar__XkbHr{cursor:pointer;font-size:24px;transition:color .2s}.Chat-module_toolCallModal__uNaIK{max-width:800px;overflow:hidden;padding:0;position:relative}.Chat-module_toolCallHeader__ZaTZF{align-items:center;background:linear-gradient(135deg,#eef2ff,#f5f3ff 40%,#fefcc9);border-bottom:1px solid rgba(15,23,42,.08);display:flex;gap:16px;padding:24px 32px 16px}.Chat-module_toolCallIcon__UgRKj{font-size:32px}.Chat-module_toolCallHeaderMeta__iAL-U{display:flex;flex-direction:column;gap:4px}.Chat-module_toolCallLabel__q7EFf{color:#475467;font-size:11px;letter-spacing:.35em;margin:0;text-transform:uppercase}.Chat-module_toolCallTitle__jhL0A{color:#0f172a;font-size:24px;font-weight:600;margin:0}.Chat-module_toolCallSubtitle__a2GIO{color:#475467;font-size:13px;margin:0}.Chat-module_toolCallGrid__Nfr7N{background:#fefefe;display:grid;gap:16px;grid-template-columns:repeat(auto-fit,minmax(250px,1fr));padding:24px 32px 16px}.Chat-module_toolCallPanel__kQZE9{background:#fff;border:1px solid #e2e8f0;border-radius:16px;box-shadow:0 18px 40px rgba(15,23,42,.08);padding:18px}.Chat-module_toolCallPanelTitle__-7OtJ{color:#94a3b8;font-size:12px;font-weight:600;letter-spacing:.25em;margin:0 0 12px;text-transform:uppercase}.Chat-module_toolCallList__5co3s{display:flex;flex-direction:column;gap:12px;list-style:none;margin:0;padding:0}.Chat-module_toolCallItem__k7W5Z{display:flex;flex-direction:column;gap:4px}.Chat-module_toolCallItemLabel__d3C1E{color:#94a3b8;font-size:11px;letter-spacing:.25em;text-transform:uppercase}.Chat-module_toolCallItemValue__cStQ0{color:#0f172a;font-size:14px;line-height:1.4;white-space:pre-wrap;word-break:break-word}.Chat-module_toolCallSummary__wGZwg{color:#0f172a;font-size:15px;line-height:1.6;margin:0}.Chat-module_toolCallSummaryMeta__RXiSv{display:flex;flex-wrap:wrap;gap:8px;margin-top:14px}.Chat-module_toolCallSummaryMetaBadge__JBi-I{background:rgba(79,70,229,.15);border-radius:999px;color:#312e81;font-size:12px;font-weight:600;padding:5px 12px}.Chat-module_toolCallEmpty__bUz5v{color:#94a3b8;font-size:14px;margin:0}.Chat-module_toolCallIssues__evhYM{display:flex;flex-wrap:wrap;gap:8px;padding:0 32px 16px}.Chat-module_toolCallIssueBadge__Qbwdp{align-items:center;border-radius:999px;display:inline-flex;font-size:12px;font-weight:600;gap:4px;padding:6px 10px}.Chat-module_toolCallIssueError__LImeS{background:rgba(239,68,68,.12);color:#b91c1c}.Chat-module_toolCallIssueWarning__chQDf{background:rgba(234,179,8,.16);color:#a16207}.Chat-module_toolCallFooter__lJ24N{align-items:center;display:flex;flex-wrap:wrap;gap:12px;justify-content:space-between;padding:0 32px 24px}.Chat-module_toolCallTimestamp__-iWsu{color:#475467;font-size:13px}.Chat-module_toolCallRawDetails__CvFfs{color:#2563eb;font-size:13px}.Chat-module_toolCallRawSummary__-aLvG{color:#2563eb;cursor:pointer;font-weight:600}.Chat-module_toolCallRawColumns__8ummz{display:grid;gap:12px;grid-template-columns:repeat(auto-fit,minmax(220px,1fr));margin-top:12px}.Chat-module_toolCallRawColumnTitle__JQda7{color:#94a3b8;font-size:11px;letter-spacing:.25em;margin:0 0 8px;text-transform:uppercase}.Chat-module_memoryModalHeader__SpZJT{align-items:flex-start;background:linear-gradient(180deg,#fdfdfd,#f6f9ff);border-bottom:1px solid #e2e8f0;display:flex;gap:16px;padding:28px 32px 16px}.Chat-module_memoryModalIcon__eYbli{align-items:center;background:linear-gradient(135deg,#6366f1,#a855f7);border-radius:18px;box-shadow:0 18px 28px rgba(15,23,42,.18);color:#fff;display:inline-flex;font-size:24px;height:52px;justify-content:center;width:52px}.Chat-module_memoryModalHeaderText__S6MZG{flex:1}.Chat-module_memoryModalTitle__zzbaQ{color:#0f172a;font-size:22px;font-weight:600;margin:0}.Chat-module_memoryModalSubtitle__PStci{color:#475467;font-size:14px;margin:4px 0 0}.Chat-module_memoryModalCallTime__RmVI9{color:#94a3b8;font-size:13px;margin:4px 0 0}.Chat-module_memoryModalStatus__bmrfc{align-items:center;border-radius:999px;display:inline-flex;font-size:13px;font-weight:600;gap:6px;margin-left:auto;padding:6px 14px;white-space:nowrap}.Chat-module_memoryStatusDot__c0oVs{background:currentColor;border-radius:50%;display:inline-block;height:8px;width:8px}.Chat-module_memoryModalContent__8E7oj{background:#fff;display:flex;flex-direction:column;gap:16px;padding:24px 32px 32px}.Chat-module_memoryMessage__q0pNK{color:#0f172a;font-size:15px;font-weight:500;margin:0}.Chat-module_memoryMetaRow__Hc-Xp{align-items:center;color:#475467;display:flex;font-size:13px;justify-content:space-between}.Chat-module_memoryMetaLabel__LFx11{font-weight:600;letter-spacing:.08em;text-transform:uppercase}.Chat-module_memoryMetaValue__4bILO{color:#0f172a;font-weight:600}.Chat-module_memoryRetrieveSection__VbuS4,.Chat-module_memoryStoreSection__qQJPN{display:flex;flex-direction:column;gap:12px}.Chat-module_memoryCard__-kIDN{background:#f8fafc;border:1px solid #e5e7eb;border-radius:16px;box-shadow:0 10px 18px rgba(15,23,42,.08);padding:18px}.Chat-module_memoryCardContent__l9S4G{color:#0f172a;font-size:14px;line-height:1.6;overflow-wrap:break-word;white-space:pre-wrap}.Chat-module_memoryCardMeta__K66KE{align-items:center;color:#94a3b8;display:flex;font-size:12px;justify-content:space-between;margin-top:12px}.Chat-module_memoryScopeBadge__T2ACc{background:rgba(124,58,237,.12);border-radius:999px;color:#0f172a;font-size:11px;font-weight:700;padding:3px 10px}.Chat-module_memoryList__pUOg2{display:flex;flex-direction:column;gap:12px}.Chat-module_memoryEmptyState__fPrXA{background:#fef3c7;border:1px solid #fde68a;border-radius:12px;color:#92400e;font-weight:500;padding:16px}.Chat-module_memoryListFooter__Y0Q2Z{color:#475467;font-size:13px;text-align:center}.Chat-module_memoryStatusSuccess__4bhfo{background:rgba(16,185,129,.16);color:#047857}.Chat-module_memoryStatusWarning__AR6yh{background:rgba(234,179,8,.16);color:#a16207}.Chat-module_memoryStatusError__ZNAT6{background:rgba(239,68,68,.15);color:#b91c1c}.Chat-module_memoryStatusNeutral__Fa-4M{background:rgba(96,165,250,.15);color:#1d4ed8}.Chat-module_searchModalHeader__Y8V-w{align-items:center;background:#fdfdfd;border-bottom:1px solid #eee;display:flex;gap:16px;padding:24px 72px 24px 32px}.Chat-module_selfLearningModalHeader__PDIt1{align-items:center;background:linear-gradient(135deg,#f8fafc,#fef9c3 45%,#ecfccb);border-bottom:1px solid #e2e8f0;gap:18px}.Chat-module_selfLearningAvatarGroup__jgy2k{align-items:center;display:flex;gap:10px}.Chat-module_selfLearningAvatar__NHXKm{align-items:center;background:#e2e8f0;background-position:50%;background-size:cover;border-radius:50%;box-shadow:0 10px 20px rgba(15,23,42,.18);color:#0f172a;display:inline-flex;font-size:18px;font-weight:600;height:48px;justify-content:center;letter-spacing:.02em;width:48px}.Chat-module_selfLearningAvatarInitial__cuw19{line-height:1}.Chat-module_selfLearningTeacher__fGhX8{background:linear-gradient(135deg,#fde68a,#86efac)}.Chat-module_selfLearningHeaderText__LTwHL{display:flex;flex-direction:column;gap:6px;text-align:left}.Chat-module_selfLearningTitle__G0SAk{color:#0f172a;font-size:20px;font-weight:600;margin:0}.Chat-module_selfLearningMetaRow__z4Tg0{align-items:center;color:#64748b;display:flex;flex-wrap:wrap;font-size:13px;gap:10px;margin-bottom:12px}.Chat-module_selfLearningMeta__58Pet{color:#64748b;font-size:13px}.Chat-module_selfLearningMetaChip__0go88{background:#e2e8f0;border-radius:999px;color:#0f172a;font-size:12px;font-weight:600;padding:4px 10px}.Chat-module_selfLearningCommitments__7ECnr{background:#f8fafc;border:1px solid #e2e8f0;border-radius:16px;padding:16px}.Chat-module_selfLearningCommitmentsLabel__OkDO1{color:#94a3b8;font-size:12px;font-weight:600;letter-spacing:.08em;text-transform:uppercase}.Chat-module_selfLearningBookEditor__CZy5T{border-radius:12px;margin-top:12px;overflow:hidden}.Chat-module_selfLearningEmpty__vCKHL{color:#475569;font-size:13px;margin-top:12px}.Chat-module_modalCloseButton__AAfWS{align-items:center;background:#fff;border:1px solid #e2e8f0;border-radius:999px;box-shadow:0 6px 16px rgba(15,23,42,.08);color:#475569;cursor:pointer;display:inline-flex;height:36px;justify-content:center;position:absolute;right:16px;top:16px;transition:background .2s ease,color .2s ease,box-shadow .2s ease,transform .2s ease;width:36px;z-index:1}.Chat-module_modalCloseButton__AAfWS:hover{background:#f8fafc;box-shadow:0 10px 24px rgba(15,23,42,.12);color:#0f172a;transform:translateY(-1px)}.Chat-module_modalCloseButton__AAfWS:focus-visible{outline:2px solid #94a3b8;outline-offset:2px}.Chat-module_teamModalHeader__SB5NL{flex-wrap:wrap;gap:12px}.Chat-module_teamHeaderParticipants__KS7aV{align-items:center;display:flex;flex-wrap:wrap;gap:12px}.Chat-module_teamHeaderProfile__xRlcS{align-items:center;color:inherit;display:inline-flex;gap:12px;text-decoration:none}.Chat-module_teamHeaderProfileLink__fYTfT:hover{color:#0f172a}.Chat-module_teamHeaderAvatar__vXcTp{background-position:50%;background-repeat:no-repeat;background-size:cover;border:1px solid #e2e8f0;border-radius:999px;box-shadow:0 6px 14px rgba(15,23,42,.08);height:40px;width:40px}.Chat-module_teamHeaderName__ygJR9{color:#0f172a;font-size:18px;font-weight:600}.Chat-module_teamHeaderDivider__cnRMd{color:#94a3b8;font-size:14px;font-weight:600}.Chat-module_searchModalIcon__AR5KY{align-items:center;color:#475569;display:inline-flex;font-size:24px;justify-content:center}.Chat-module_searchModalQuery__5x-Ra{color:#202124!important;font-size:22px!important;font-weight:500!important;margin:0!important;text-align:left!important}.Chat-module_searchModalContent__mWLIE{background:#fff;max-height:60vh;overflow-y:auto;padding:24px 32px}.Chat-module_teamChatContainer__xdLtU{background:#f8fafc;border:1px solid #e5e7eb;border-radius:16px;margin-bottom:20px;overflow:hidden}.Chat-module_teamToolCallSection__FH9l7{display:flex;flex-direction:column;gap:16px}.Chat-module_teamToolCallGroup__PO6rl{display:flex;flex-direction:column;gap:8px}.Chat-module_teamToolCallHeading__G-JSj{color:#64748b;font-size:12px;font-weight:600;letter-spacing:.08em;text-transform:uppercase}.Chat-module_teamToolCallChips__hRWqh{display:flex;flex-wrap:wrap;gap:8px}.Chat-module_teamToolCallDetails__ZaTtt{background:#f8fafc;border:1px solid rgba(148,163,184,.4);border-radius:12px;padding:16px}.Chat-module_teamToolCallDetailsHeader__FPIgI{align-items:center;display:flex;gap:12px;justify-content:space-between;margin-bottom:12px}.Chat-module_teamToolCallDetailsTitle__50SeL{align-items:center;color:#0f172a;display:flex;font-size:14px;font-weight:600;gap:6px}.Chat-module_teamToolCallDetailsClear__8yvu7{background:rgba(148,163,184,.2);border:1px solid rgba(148,163,184,.4);border-radius:999px;color:#475569;cursor:pointer;font-size:12px;padding:4px 10px;transition:all .2s ease}.Chat-module_teamToolCallDetailsClear__8yvu7:hover{background:rgba(148,163,184,.35)}.Chat-module_searchResultsList__xVDUZ{display:flex;flex-direction:column;gap:28px;text-align:left}.Chat-module_searchResultsRaw__i3GkD{max-width:650px;text-align:left}.Chat-module_searchResultItem__bTzq5{max-width:650px}.Chat-module_searchResultUrl__SbrH4{color:#202124;font-size:14px;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.Chat-module_searchResultUrl__SbrH4 a{color:#202124;text-decoration:none}.Chat-module_searchResultTitle__XS7zN{font-size:20px!important;font-weight:400!important;line-height:1.3;margin:0 0 4px!important}.Chat-module_searchResultTitle__XS7zN a{color:#1a0dab;text-decoration:none}.Chat-module_searchResultTitle__XS7zN a:hover{text-decoration:underline}.Chat-module_searchResultSnippet__Wsun2{color:#4d5156;font-size:14px;line-height:1.58;margin:0}.Chat-module_noResults__AAv6s{color:#70757a;font-style:italic;padding:40px;text-align:center}.Chat-module_citationDetails__2v2e1,.Chat-module_toolCallDetails__WUFlD{margin:20px 0;text-align:left}.Chat-module_citationPreview__rGhCR{background:#f8f8f8;border:1px solid #ddd;border-radius:8px;margin-top:16px;overflow:hidden}.Chat-module_citationIframe__ngzFY{border:none;display:block;height:60vh;min-height:400px;width:100%}.Chat-module_citationMetadata__DPaCF{background:#f8f8f8;border:1px solid #eee;border-radius:8px;margin-bottom:20px;padding:16px}.Chat-module_citationMetadata__DPaCF p{font-size:.9em;margin:8px 0}.Chat-module_citationMetadata__DPaCF strong{color:#333}.Chat-module_citationMetadata__DPaCF a{color:#007bff;text-decoration:none;word-break:break-all}.Chat-module_citationMetadata__DPaCF a:hover{text-decoration:underline}.Chat-module_citationExcerpt__VU4BS{background:#fff;border:1px solid #ddd;border-radius:8px;max-height:400px;overflow-y:auto;padding:16px}.Chat-module_citationExcerpt__VU4BS h4{color:#333;font-size:1em;margin:0 0 12px}.Chat-module_citationHint__5AkLx{font-size:.85em;margin-top:8px;opacity:.7}.Chat-module_toolCallDataContainer__rdEzC{background:#f8f8f8;border:1px solid #eee;border-radius:6px;margin-bottom:15px;max-height:300px;overflow-y:auto;padding:12px}.Chat-module_toolCallData__UauCO{font-size:.85em;margin:0;white-space:pre-wrap;word-break:break-word}.Chat-module_toolCallArgsList__LD3xH{list-style:none;margin:0;padding:0}.Chat-module_toolCallArgsList__LD3xH li{font-size:.9em;margin-bottom:5px}.Chat-module_toolCallModal__uNaIK .Chat-module_toolCallData__UauCO{background:#f8fafc;color:#0f172a}.Chat-module_ratingInput__z8Pv-{background:hsla(0,0%,49%,.05);border:2px solid hsla(0,0%,49%,.1);border-radius:12px;color:#0b1220;font-size:14px;line-height:1.5;margin-bottom:18px;min-height:100px;padding:16px;resize:vertical;transition:all .2s ease;width:100%}.Chat-module_ratingInput__z8Pv-:focus{background:hsla(0,0%,49%,.08);border-color:#0084ff;box-shadow:0 0 0 4px rgba(0,132,255,.1);outline:none}.Chat-module_ratingInput__z8Pv-[readonly]{background:hsla(0,0%,49%,.01);border:1px solid hsla(0,0%,49%,.5)}.Chat-module_ratingActions__nXcss{display:flex;gap:12px;justify-content:flex-end}.Chat-module_ratingActions__nXcss button{border:none;border-radius:8px;cursor:pointer;font-size:14px;font-weight:500;min-width:80px;padding:12px 24px;transition:all .2s ease}.Chat-module_ratingActions__nXcss button:first-child{background-color:hsla(0,0%,49%,.1);border:1px solid hsla(0,0%,49%,.2);color:#0b1220}.Chat-module_ratingActions__nXcss button:first-child:hover:not(:disabled){background-color:hsla(0,0%,49%,.2);color:#0b1220}.Chat-module_ratingActions__nXcss button:last-child{background:linear-gradient(135deg,#0084ff,#06c);color:#fff}.Chat-module_ratingActions__nXcss button:last-child:hover:not(:disabled){background:linear-gradient(135deg,#0071d1,#0052a3);box-shadow:0 4px 12px rgba(0,132,255,.3);transform:translateY(-1px)}.Chat-module_ratingActions__nXcss button:last-child:disabled{cursor:not-allowed;opacity:.5}.Chat-module_downloadButton__8--bk{align-items:center;background:linear-gradient(135deg,#0084ff,#06c);border:none;border-radius:8px;box-shadow:0 2px 8px rgba(0,132,255,.2);color:#fff;cursor:pointer;display:inline-flex;font-size:14px;font-weight:600;gap:8px;justify-content:center;padding:12px 24px;transition:all .2s ease}.Chat-module_downloadButton__8--bk:hover{background:linear-gradient(135deg,#09f,#07d);box-shadow:0 4px 16px rgba(0,132,255,.3);transform:translateY(-2px)}.Chat-module_downloadButton__8--bk:active{box-shadow:0 2px 8px rgba(0,132,255,.2);transform:translateY(0)}.Chat-module_downloadButton__8--bk svg{flex-shrink:0}.Chat-module_downloadButton__8--bk span{font-weight:600}.Chat-module_chatButton__d9VgA{-webkit-tap-highlight-color:transparent!important;align-items:center!important;backdrop-filter:blur(20px)!important;background:linear-gradient(135deg,#0084ff,#06c)!important;border:none!important;border-radius:20px!important;box-shadow:0 4px 16px rgba(0,132,255,.3)!important;color:#fff!important;cursor:pointer!important;display:inline-flex!important;font-size:13px!important;font-weight:600!important;gap:8px!important;justify-content:center!important;line-height:1.3!important;margin:0!important;min-height:40px!important;min-width:110px!important;overflow:hidden!important;padding:12px 16px!important;position:relative!important;touch-action:manipulation!important;transition:all .3s cubic-bezier(.25,.46,.45,.94)!important;user-select:none!important}.Chat-module_chatButton__d9VgA:before{background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.2),transparent);content:\"\";height:100%;left:-100%;position:absolute;top:0;transition:left .5s ease;width:100%}.Chat-module_chatButton__d9VgA:hover:before{left:100%}.Chat-module_chatButton__d9VgA:hover:not(:disabled){background:linear-gradient(135deg,#09f,#07d);box-shadow:0 8px 24px rgba(0,132,255,.4);transform:translateY(-2px) scale(1.02)}.Chat-module_chatButton__d9VgA:active{box-shadow:0 4px 16px rgba(0,132,255,.3);transform:scale(.98) translateY(-1px);transition:transform .1s ease}.Chat-module_chatButton__d9VgA:focus{box-shadow:0 4px 16px rgba(0,132,255,.3),0 0 0 3px rgba(0,132,255,.3);outline:none}.Chat-module_chatButton__d9VgA svg{filter:drop-shadow(0 1px 2px rgba(0,0,0,.2));flex-shrink:0;height:16px;opacity:1;transition:all .3s cubic-bezier(.25,.46,.45,.94);width:16px}.Chat-module_chatButton__d9VgA:hover svg{filter:drop-shadow(0 2px 4px rgba(0,0,0,.3));transform:rotate(-90deg) scale(1.1)}.Chat-module_chatButtonText__RkGB-{font-weight:600;opacity:1;text-shadow:0 1px 2px rgba(0,0,0,.1);transition:all .2s ease;white-space:nowrap}.Chat-module_chatButton__d9VgA:hover .Chat-module_chatButtonText__RkGB-{transform:translateX(1px)}.Chat-module_useTemplateButton__xcJNR{-webkit-tap-highlight-color:transparent!important;align-items:center!important;backdrop-filter:blur(20px)!important;background:linear-gradient(135deg,#0084ff,#06c)!important;border:none!important;border-radius:20px!important;box-shadow:0 4px 16px rgba(0,132,255,.3)!important;color:#fff!important;cursor:pointer!important;display:inline-flex!important;font-size:13px!important;font-weight:600!important;gap:8px!important;justify-content:center!important;line-height:1.3!important;margin:0!important;min-height:40px!important;min-width:110px!important;overflow:hidden!important;padding:12px 16px!important;position:relative!important;touch-action:manipulation!important;transition:all .3s cubic-bezier(.25,.46,.45,.94)!important;user-select:none!important}.Chat-module_useTemplateButton__xcJNR:before{background:linear-gradient(90deg,transparent,hsla(0,0%,100%,.2),transparent);content:\"\";height:100%;left:-100%;position:absolute;top:0;transition:left .5s ease;width:100%}.Chat-module_useTemplateButton__xcJNR:hover:before{left:100%}.Chat-module_useTemplateButton__xcJNR:hover:not(:disabled){background:linear-gradient(135deg,#09f,#07d);box-shadow:0 8px 24px rgba(0,132,255,.4);transform:translateY(-2px) scale(1.02)}.Chat-module_useTemplateButton__xcJNR:active{box-shadow:0 4px 16px rgba(0,132,255,.3);transform:scale(.98) translateY(-1px);transition:transform .1s ease}.Chat-module_useTemplateButton__xcJNR:focus{box-shadow:0 4px 16px rgba(0,132,255,.3),0 0 0 3px rgba(0,132,255,.3);outline:none}.Chat-module_useTemplateButton__xcJNR svg{filter:drop-shadow(0 1px 2px rgba(0,0,0,.2));flex-shrink:0;height:16px;opacity:1;transition:all .3s cubic-bezier(.25,.46,.45,.94);width:16px}.Chat-module_useTemplateButton__xcJNR:hover svg{filter:drop-shadow(0 2px 4px rgba(0,0,0,.3));transform:scale(1.1)}.Chat-module_useTemplateButton__xcJNR:hover .Chat-module_chatButtonText__RkGB-{transform:translateX(1px)}.Chat-module_saveButtonContainer__lSNUJ{display:inline-block;position:relative}.Chat-module_saveMenu__-ph8y{background:#fff;border:1px solid #ddd;box-shadow:0 2px 8px rgba(0,0,0,.08);left:0;min-width:120px;position:absolute;top:100%;z-index:10}.Chat-module_saveMenuItem__ISApL{background:none;border:none;color:#111;cursor:pointer;display:block;padding:8px 16px;text-align:left;width:100%}.Chat-module_saveMenuItem__ISApL:hover{background-color:#f0f0f0}.Chat-module_saveMenuDivider__3ktTM{background-color:#ddd;height:1px;margin:4px 0}.Chat-module_chatFeedbackPanel__gfOzk{background:#fff;border:1px solid #e5e7eb;border-radius:14px;box-shadow:0 8px 18px rgba(15,23,42,.08);display:flex;flex-direction:column;gap:6px;padding:6px 8px;width:min(260px,100%)}.Chat-module_chatFeedbackToggle__IeOM6{align-items:center;background:transparent;border:none;border-radius:12px;cursor:pointer;display:flex;gap:10px;padding:6px 8px;text-align:left;transition:background .2s ease,transform .2s ease;width:100%}.Chat-module_chatFeedbackToggle__IeOM6:hover{background:#f5f5f5;transform:translateY(-.5px)}.Chat-module_chatFeedbackToggle__IeOM6:focus-visible{outline:2px solid #2563eb;outline-offset:2px}.Chat-module_chatFeedbackIcon__XcB7A{background:#eef2ff;border-radius:10px;color:#6366f1;display:grid;flex-shrink:0;height:34px;place-items:center;transition:background .2s ease,color .2s ease;width:34px}.Chat-module_chatFeedbackToggle__IeOM6[data-enabled=true] .Chat-module_chatFeedbackIcon__XcB7A{background:#e0f2fe;color:#0284c7}.Chat-module_chatFeedbackToggle__IeOM6[data-enabled=false] .Chat-module_chatFeedbackIcon__XcB7A{background:#f3f4f6;color:#94a3b8}.Chat-module_chatFeedbackMeta__6t2be{display:flex;flex:1;flex-direction:column;gap:2px}.Chat-module_chatFeedbackLabel__qQgH3{color:#0f172a;font-size:.85rem;font-weight:600}.Chat-module_chatFeedbackDescription__I9bNg{color:#64748b;font-size:.75rem}.Chat-module_chatFeedbackIndicator__uHCr7{border-radius:999px;font-size:.72rem;font-weight:700;letter-spacing:.08em;padding:2px 8px;text-transform:uppercase}.Chat-module_chatFeedbackIndicatorActive__bkw-w{background:#dcfce7;border:1px solid #34d399;color:#166534}.Chat-module_chatFeedbackIndicatorInactive__qMBkh{background:#fef9c3;border:1px solid #facc15;color:#713f12}.Chat-module_pauseButton__eeu7K{background:linear-gradient(135deg,#ffb347,#ff8c42)!important}.Chat-module_pauseButton__eeu7K:hover:not(:disabled){background:linear-gradient(135deg,#ffc067,#ff9e5f)!important}.Chat-module_pauseButton__eeu7K.Chat-module_pausing__pTx8b{cursor:wait!important;opacity:.6!important}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya{background:linear-gradient(135deg,#10b981,#059669)!important}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya:hover:not(:disabled){background:linear-gradient(135deg,#34d399,#059669)!important;box-shadow:0 8px 24px rgba(16,185,129,.35);transform:translateY(-2px) scale(1.02)}.Chat-module_pauseButton__eeu7K svg{transition:transform .3s ease}.Chat-module_pauseButton__eeu7K.Chat-module_paused__j-pya svg{transform:scale(1.1)}.Chat-module_pauseButton__eeu7K.Chat-module_pausing__pTx8b svg{opacity:.8}.Chat-module_voiceCallIndicatorBar__N2sWN{align-items:center;backdrop-filter:blur(10px);background:linear-gradient(135deg,rgba(34,197,94,.1),rgba(16,185,129,.1));border-bottom:1px solid rgba(34,197,94,.3);display:flex;grid-area:🟦;justify-content:center;padding:12px 20px;width:100%}.Chat-module_voiceCallIndicator__tsaaG{align-items:center;backdrop-filter:blur(10px);background:linear-gradient(135deg,rgba(34,197,94,.2),rgba(16,185,129,.2));border:1px solid rgba(34,197,94,.4);border-radius:20px;box-shadow:0 2px 8px rgba(34,197,94,.2);color:#10b981;display:inline-flex;font-size:13px;font-weight:600;gap:8px;padding:8px 16px;position:relative}.Chat-module_voiceCallIndicator__tsaaG svg{animation:Chat-module_voiceCallIconPulse__zZbJn 2s ease-in-out infinite;flex-shrink:0;height:16px;width:16px}.Chat-module_voiceCallIndicator__tsaaG span{font-weight:600;text-shadow:0 1px 2px rgba(0,0,0,.1)}.Chat-module_voiceCallPulse__XcGU4{animation:Chat-module_voiceCallPulse__XcGU4 1.5s ease-in-out infinite;background:#10b981;border-radius:50%;height:8px;position:absolute;right:8px;top:50%;transform:translateY(-50%);width:8px}@keyframes Chat-module_voiceCallIconPulse__zZbJn{0%,to{opacity:1;transform:scale(1)}50%{opacity:.8;transform:scale(1.1)}}@keyframes Chat-module_voiceCallPulse__XcGU4{0%,to{opacity:1;transform:translateY(-50%) scale(1)}50%{opacity:.6;transform:translateY(-50%) scale(1.3)}}.Chat-module_chatMessage__nmLaZ .Chat-module_voiceCallIndicator__tsaaG{border-radius:16px;font-size:12px;margin-bottom:8px;padding:6px 12px}.Chat-module_chatMessage__nmLaZ .Chat-module_voiceCallIndicator__tsaaG svg{height:14px;width:14px}.Chat-module_feedbackStatus__-df7l{word-wrap:break-word;animation:Chat-module_feedbackStatusSlideIn__n6bbB .28s ease-out;backdrop-filter:blur(20px);border-radius:14px;box-shadow:0 18px 32px rgba(15,23,42,.35);font-weight:500;max-width:340px;padding:14px 22px;pointer-events:none;position:fixed;right:20px;top:20px;z-index:10000}.Chat-module_feedbackStatusSuccess__Ax2q0{background:linear-gradient(135deg,#10b981,#059669);color:#f8fafc}.Chat-module_feedbackStatusError__dnv6d{background:linear-gradient(135deg,#ef4444,#dc2626);color:#fef2f2}@keyframes Chat-module_feedbackStatusSlideIn__n6bbB{0%{opacity:0;transform:translateX(100%)}to{opacity:1;transform:translateX(0)}}@media (max-width:768px){.Chat-module_Chat__j2eE5{--chat-message-min-width:100px;--chat-message-max-width:min(85%,560px);--chat-message-avatar-gap:10px}.Chat-module_actions__gTZ5T{gap:6px;margin:12px 16px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{padding:16px 12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ{margin-bottom:16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border-radius:18px;font-size:14px;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:36px;margin:0 0 4px;width:36px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{gap:10px;padding:16px 12px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-radius:22px;font-size:16px;padding:14px 18px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{height:44px;width:44px}.Chat-module_scrollToBottom__nzxdZ{font-size:18px;height:44px;top:calc(100% - 160px);width:44px}.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_ratingModalContent__CCdq7{border-radius:16px;margin:16px;max-height:80vh;overflow-y:auto;padding:24px 20px}.Chat-module_stars__PCzNO{gap:6px;margin-bottom:20px}.Chat-module_stars__PCzNO span{font-size:32px;padding:8px}.Chat-module_ratingActions__nXcss{flex-direction:column-reverse;gap:8px}.Chat-module_ratingActions__nXcss button{border-radius:10px;font-size:16px;padding:14px;width:100%}.Chat-module_selfLearningModalHeader__PDIt1{align-items:flex-start;flex-direction:column}.Chat-module_selfLearningAvatarGroup__jgy2k{justify-content:flex-start}.Chat-module_selfLearningMetaRow__z4Tg0{align-items:flex-start}}@media (max-width:480px){.Chat-module_Chat__j2eE5{--chat-message-min-width:80px;--chat-message-max-width:min(90%,520px);--chat-message-avatar-gap:8px}.Chat-module_actions__gTZ5T{gap:4px;margin:8px 12px 0}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessages__J2u2N{padding:12px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border-radius:16px;font-size:14px;padding:10px 14px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:32px;margin:0 0 4px;width:32px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan{gap:8px;padding:12px 8px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-radius:20px;font-size:16px;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{height:40px;width:40px}.Chat-module_chatButton__d9VgA{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_chatButton__d9VgA svg{height:18px!important;width:18px!important}.Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_useTemplateButton__xcJNR{border-radius:50%!important;gap:0!important;height:40px!important;margin:0!important;min-height:40px!important;min-width:40px!important;padding:0!important;width:40px!important}.Chat-module_useTemplateButton__xcJNR svg{height:18px!important;width:18px!important}.Chat-module_useTemplateButton__xcJNR .Chat-module_chatButtonText__RkGB-{display:none!important}.Chat-module_scrollToBottom__nzxdZ{font-size:16px;height:40px;top:calc(100% - 140px);width:40px}.Chat-module_ratingModal__XVKYm{align-items:flex-end;padding:0}.Chat-module_ratingModalContent__CCdq7{border-radius:20px 20px 0 0;margin:0;max-height:70vh;padding:24px 16px 20px;width:100%}}@media (prefers-reduced-motion:reduce){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ,.Chat-module_feedbackStatus__-df7l,.Chat-module_ratingModalContent__CCdq7,.Chat-module_ratingModal__XVKYm,.Chat-module_scrollToBottom__nzxdZ{animation:none}.Chat-module_chatButton__d9VgA,.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button,.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea,.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{transition:none}}@media (prefers-contrast:high){.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{border:2px solid}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea{border-width:3px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan button{border:2px solid}}.Chat-module_emailContainer__TBxXZ{display:flex;flex-direction:column;gap:16px;margin-bottom:20px}.Chat-module_emailModalHeader__Z4-CJ{background:#f8fafc}.Chat-module_emailHeaderText__1dR6R{display:flex;flex-direction:column;gap:2px}.Chat-module_emailHeaderLabel__7iAEH{color:#64748b;font-size:12px;letter-spacing:.08em;text-transform:uppercase}.Chat-module_emailMetadata__8XXGf{background:#fff;border:1px solid #e2e8f0;border-radius:12px;box-shadow:0 1px 4px rgba(15,23,42,.08);margin-bottom:16px;padding:18px 20px}.Chat-module_emailField__pcb03{display:flex;font-size:14px;gap:8px;line-height:1.5;margin-bottom:8px}.Chat-module_emailField__pcb03:last-child{margin-bottom:0}.Chat-module_emailField__pcb03 strong{color:#64748b;font-weight:600;min-width:60px}.Chat-module_emailRecipients__YVVys{color:#0f172a;word-break:break-word}.Chat-module_emailBody__OBt-w{text-align:left}.Chat-module_emailBody__OBt-w strong{color:#64748b;display:block;font-size:14px;font-weight:600;margin-bottom:12px}.Chat-module_emailBodyContent__-vFo2{background:#fff;border:1px solid #e2e8f0;border-radius:12px;box-shadow:0 1px 4px rgba(15,23,42,.08);line-height:1.6;max-height:400px;overflow-y:auto;padding:22px}.Chat-module_emailBodyContent__-vFo2 p{margin:0 0 12px}.Chat-module_emailBodyContent__-vFo2 p:last-child{margin-bottom:0}.Chat-module_emailStatus__T4UuL{color:#0f172a;font-weight:600;text-transform:capitalize}.Chat-module_imagePrompt__v7-Ho{backdrop-filter:blur(12px);background:hsla(0,0%,100%,.25);border:1px solid hsla(0,0%,49%,.35);border-radius:18px;box-shadow:inset 0 0 0 1px hsla(0,0%,100%,.25);display:flex;flex-direction:column;gap:8px;margin-bottom:16px;margin-top:16px;padding:12px}.Chat-module_imagePromptPreview__ve5dB{background:linear-gradient(180deg,rgba(15,118,255,.05),hsla(0,0%,100%,.4));border:1px dashed hsla(0,0%,49%,.3);border-radius:14px;min-height:160px;overflow:hidden;position:relative}.Chat-module_imagePromptImage__KjE3H{display:block;height:100%;object-fit:cover;width:100%}.Chat-module_imagePromptPlaceholder__b8cW6{align-items:center;display:flex;height:100%;justify-content:center;width:100%}.Chat-module_imagePromptSpinner__nat8r{animation:Chat-module_imagePromptSpinner__nat8r .9s linear infinite;border:3px solid hsla(0,0%,100%,.35);border-radius:50%;border-top-color:#2563eb;height:36px;width:36px}@keyframes Chat-module_imagePromptSpinner__nat8r{to{transform:rotate(1turn)}}.Chat-module_imagePromptMeta__YwltP{display:flex;flex-direction:column;gap:4px}.Chat-module_imagePromptPrompt__N1E--{color:#111;font-size:13px;font-weight:500;overflow-wrap:anywhere}.Chat-module_imagePromptStatus__0ROQF{color:#6b7280;font-size:11px}.Chat-module_imagePromptError__x1952{align-items:center;background:rgba(220,38,38,.08);color:#dc2626;display:flex;font-size:12px;inset:0;justify-content:center;padding:12px;position:absolute;text-align:center}\n/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Chat.module.css"],"names":[],"mappings":"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","file":"Chat.module.css","sourcesContent":["@font-face {\n    font-family: 'OpenMojiBlack';\n    src: url('https://s6.ptbk.io/fonts/OpenMoji-black-glyf.woff2') format('woff2'); /* <- TODO: [🐱‍🚀] Dynamically load from /servers.ts */\n    unicode-range: U+23, U+2A, U+2D, U+30-39, U+A9, U+AE, U+200D, U+203C, U+2049, U+20E3, U+2117, U+2120, U+2122, U+2139,\n        U+2194-2199, U+21A9, U+21AA, U+229C, U+231A, U+231B, U+2328, U+23CF, U+23E9-23F3, U+23F8-23FE, U+24C2, U+25A1,\n        U+25AA-25AE, U+25B6, U+25C0, U+25C9, U+25D0, U+25D1, U+25E7-25EA, U+25ED, U+25EE, U+25FB-25FE, U+2600-2605,\n        U+260E, U+2611, U+2614, U+2615, U+2618, U+261D, U+2620, U+2622, U+2623, U+2626, U+262A, U+262E, U+262F,\n        U+2638-263A, U+2640, U+2642, U+2648-2653, U+265F, U+2660, U+2663, U+2665, U+2666, U+2668, U+267B, U+267E, U+267F,\n        U+2691-2697, U+2699, U+269B, U+269C, U+26A0, U+26A1, U+26A7, U+26AA, U+26AB, U+26B0, U+26B1, U+26BD, U+26BE,\n        U+26C4, U+26C5, U+26C8, U+26CE, U+26CF, U+26D1, U+26D3, U+26D4, U+26E9, U+26EA, U+26F0-26F5, U+26F7-26FA, U+26FD,\n        U+2702, U+2705, U+2708-270D, U+270F, U+2712, U+2714, U+2716, U+271D, U+2721, U+2728, U+2733, U+2734, U+2744,\n        U+2747, U+274C, U+274E, U+2753-2755, U+2757, U+2763, U+2764, U+2795-2797, U+27A1, U+27B0, U+27BF, U+2934, U+2935,\n        U+2B05-2B07, U+2B0C, U+2B0D, U+2B1B, U+2B1C, U+2B1F-2B24, U+2B2E, U+2B2F, U+2B50, U+2B55, U+2B58, U+2B8F,\n        U+2BBA-2BBC, U+2BC3, U+2BC4, U+2BEA, U+2BEB, U+3030, U+303D, U+3297, U+3299, U+E000-E009, U+E010, U+E011,\n        U+E040-E06D, U+E080-E0B4, U+E0C0-E0CC, U+E0FF-E10D, U+E140-E14A, U+E150-E157, U+E181-E189, U+E1C0-E1C4,\n        U+E1C6-E1D9, U+E200-E216, U+E240-E269, U+E280-E283, U+E2C0-E2C4, U+E2C6-E2DA, U+E300-E303, U+E305-E30F,\n        U+E312-E316, U+E318-E322, U+E324-E329, U+E32B, U+E340-E348, U+E380, U+E381, U+F000, U+F77A, U+F8FF, U+FE0F,\n        U+1F004, U+1F0CF, U+1F10D-1F10F, U+1F12F, U+1F16D-1F171, U+1F17E, U+1F17F, U+1F18E, U+1F191-1F19A, U+1F1E6-1F1FF,\n        U+1F201, U+1F202, U+1F21A, U+1F22F, U+1F232-1F23A, U+1F250, U+1F251, U+1F260-1F265, U+1F300-1F321, U+1F324-1F393,\n        U+1F396, U+1F397, U+1F399-1F39B, U+1F39E-1F3F0, U+1F3F3-1F3F5, U+1F3F7-1F4FD, U+1F4FF-1F53D, U+1F549-1F54E,\n        U+1F550-1F567, U+1F56F, U+1F570, U+1F573-1F57A, U+1F587, U+1F58A-1F58D, U+1F590, U+1F595, U+1F596, U+1F5A4,\n        U+1F5A5, U+1F5A8, U+1F5B1, U+1F5B2, U+1F5BC, U+1F5C2-1F5C4, U+1F5D1-1F5D3, U+1F5DC-1F5DE, U+1F5E1, U+1F5E3,\n        U+1F5E8, U+1F5EF, U+1F5F3, U+1F5FA-1F64F, U+1F680-1F6C5, U+1F6CB-1F6D2, U+1F6D5-1F6D7, U+1F6DC-1F6E5, U+1F6E9,\n        U+1F6EB, U+1F6EC, U+1F6F0, U+1F6F3-1F6FC, U+1F7E0-1F7EB, U+1F7F0, U+1F90C-1F93A, U+1F93C-1F945, U+1F947-1F9FF,\n        U+1FA70-1FA7C, U+1FA80-1FA88, U+1FA90-1FABD, U+1FABF-1FAC5, U+1FACE-1FADB, U+1FAE0-1FAE8, U+1FAF0-1FAF8,\n        U+1FBC5-1FBC9, U+E0061-E0067, U+E0069, U+E006C-E0079, U+E007F;\n}\n\n@font-face {\n    font-family: 'OpenMojiColor';\n    src: url('https://s6.ptbk.io/fonts/OpenMoji-color-cbdt.woff2') format('woff2');\n    font-display: swap;\n    unicode-range: U+23, U+2A, U+2D, U+30-39, U+A9, U+AE, U+200D, U+203C, U+2049, U+20E3, U+2117, U+2120, U+2122, U+2139,\n        U+2194-2199, U+21A9, U+21AA, U+229C, U+231A, U+231B, U+2328, U+23CF, U+23E9-23F3, U+23F8-23FE, U+24C2, U+25A1,\n        U+25AA-25AE, U+25B6, U+25C0, U+25C9, U+25D0, U+25D1, U+25E7-25EA, U+25ED, U+25EE, U+25FB-25FE, U+2600-2605,\n        U+260E, U+2611, U+2614, U+2615, U+2618, U+261D, U+2620, U+2622, U+2623, U+2626, U+262A, U+262E, U+262F,\n        U+2638-263A, U+2640, U+2642, U+2648-2653, U+265F, U+2660, U+2663, U+2665, U+2666, U+2668, U+267B, U+267E, U+267F,\n        U+2691-2697, U+2699, U+269B, U+269C, U+26A0, U+26A1, U+26A7, U+26AA, U+26AB, U+26B0, U+26B1, U+26BD, U+26BE,\n        U+26C4, U+26C5, U+26C8, U+26CE, U+26CF, U+26D1, U+26D3, U+26D4, U+26E9, U+26EA, U+26F0-26F5, U+26F7-26FA, U+26FD,\n        U+2702, U+2705, U+2708-270D, U+270F, U+2712, U+2714, U+2716, U+271D, U+2721, U+2728, U+2733, U+2734, U+2744,\n        U+2747, U+274C, U+274E, U+2753-2755, U+2757, U+2763, U+2764, U+2795-2797, U+27A1, U+27B0, U+27BF, U+2934, U+2935,\n        U+2B05-2B07, U+2B0C, U+2B0D, U+2B1B, U+2B1C, U+2B1F-2B24, U+2B2E, U+2B2F, U+2B50, U+2B55, U+2B58, U+2B8F,\n        U+2BBA-2BBC, U+2BC3, U+2BC4, U+2BEA, U+2BEB, U+3030, U+303D, U+3297, U+3299, U+E000-E009, U+E010, U+E011,\n        U+E040-E06D, U+E080-E0B4, U+E0C0-E0CC, U+E0FF-E10D, U+E140-E14A, U+E150-E157, U+E181-E189, U+E1C0-E1C4,\n        U+E1C6-E1D9, U+E200-E216, U+E240-E269, U+E280-E283, U+E2C0-E2C4, U+E2C6-E2DA, U+E300-E303, U+E305-E30F,\n        U+E312-E316, U+E318-E322, U+E324-E329, U+E32B, U+E340-E348, U+E380, U+E381, U+F000, U+F77A, U+F8FF, U+FE0F,\n        U+1F004, U+1F0CF, U+1F10D-1F10F, U+1F12F, U+1F16D-1F171, U+1F17E, U+1F17F, U+1F18E, U+1F191-1F19A, U+1F1E6-1F1FF,\n        U+1F201, U+1F202, U+1F21A, U+1F22F, U+1F232-1F23A, U+1F250, U+1F251, U+1F260-1F265, U+1F300-1F321, U+1F324-1F393,\n        U+1F396, U+1F397, U+1F399-1F39B, U+1F39E-1F3F0, U+1F3F3-1F3F5, U+1F3F7-1F4FD, U+1F4FF-1F53D, U+1F549-1F54E,\n        U+1F550-1F567, U+1F56F, U+1F570, U+1F573-1F57A, U+1F587, U+1F58A-1F58D, U+1F590, U+1F595, U+1F596, U+1F5A4,\n        U+1F5A5, U+1F5A8, U+1F5B1, U+1F5B2, U+1F5BC, U+1F5C2-1F5C4, U+1F5D1-1F5D3, U+1F5DC-1F5DE, U+1F5E1, U+1F5E3,\n        U+1F5E8, U+1F5EF, U+1F5F3, U+1F5FA-1F64F, U+1F680-1F6C5, U+1F6CB-1F6D2, U+1F6D5-1F6D7, U+1F6DC-1F6E5, U+1F6E9,\n        U+1F6EB, U+1F6EC, U+1F6F0, U+1F6F3-1F6FC, U+1F7E0-1F7EB, U+1F7F0, U+1F90C-1F93A, U+1F93C-1F945, U+1F947-1F9FF,\n        U+1FA70-1FA7C, U+1FA80-1FA88, U+1FA90-1FABD, U+1FABF-1FAC5, U+1FACE-1FADB, U+1FAE0-1FAE8, U+1FAF0-1FAF8,\n        U+1FBC5-1FBC9, U+E0061-E0067, U+E0069, U+E006C-E0079, U+E007F;\n}\n\n.copiedToClipboardMessage {\n    position: fixed;\n    top: 32px;\n    left: 50%;\n    transform: translateX(-50%);\n    background: #222;\n    color: #fff;\n    padding: 10px 24px;\n    border-radius: 8px;\n    font-size: 1.1em;\n    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.18);\n    opacity: 0.97;\n    pointer-events: none;\n    z-index: 9999;\n}\n\n.Chat {\n    width: 100%;\n    height: 100%;\n    display: flex;\n    flex-direction: column;\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiColor', 'OpenMojiBlack';\n\n    --chat-message-min-width: 150px;\n    --chat-message-max-width: min(70%, 600px);\n    --chat-message-avatar-gap: 12px;\n    /* <- TODO: [🧠][🎱] Better, define other fonts */\n}\n\n.chatMainFlow {\n    width: 100%;\n    height: 100%;\n    max-width: 100vw;\n    display: grid;\n    grid-template:\n        '🟦' min-content\n        '💬' 1fr\n        '📝' min-content\n        / 1fr;\n}\n\n.chatMainFlow .chatBar {\n    grid-area: 🟦;\n    width: 100%;\n    padding: 16px 20px;\n    color: #0f1724;\n    background-color: #ffffff;\n    border-bottom: 1px solid rgba(15, 23, 36, 0.06);\n    text-align: center;\n    font-weight: 500;\n}\n\n.TasksInProgress {\n    grid-area: 🟦;\n    width: auto;\n    height: min-content;\n    align-self: center;\n    justify-self: self-end;\n    margin: 8px 16px;\n}\n\n.actions {\n    grid-area: 💬;\n    width: auto;\n    height: min-content;\n    z-index: 40;\n    align-self: self-start;\n    justify-self: self-end;\n    margin: 16px 20px 0;\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    transition: opacity 0.2s ease;\n    will-change: opacity;\n}\n\n.actions.portal {\n    margin: 0;\n}\n\n.actions.left {\n    justify-self: self-start;\n}\n\n.actions.right {\n    justify-self: self-end;\n}\n\n.actionsFaded {\n    opacity: 0.3;\n}\n\n.actionsScrolling {\n    opacity: 0.15;\n    pointer-events: none;\n}\n\n/* Large tablet and small desktop screens */\n@media (max-width: 900px) {\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n}\n\n/* Medium screens */\n@media (max-width: 600px) {\n    .actions {\n        margin: 14px 18px 0;\n        gap: 7px;\n    }\n}\n\n.chatMainFlow .chatChildren {\n    grid-area: 💬;\n    width: 100%;\n    height: 100%;\n    z-index: 300;\n}\n\n/* Chat messages area */\n.chatMainFlow .chatMessages {\n    grid-area: 💬;\n    width: 100%;\n    height: 100%;\n    z-index: 10;\n    padding: 24px 20px 16px;\n    overflow-y: auto;\n    overflow-x: hidden;\n    scroll-behavior: smooth;\n}\n\n.hasActionsAndFirstMessageIsLong {\n    padding-top: 80px;\n}\n\n/* Custom scrollbar styling */\n.chatMainFlow .chatMessages::-webkit-scrollbar {\n    width: 6px;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-track {\n    background: transparent;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-thumb {\n    background: rgba(125, 125, 125, 0.2);\n    border-radius: 3px;\n    transition: all 0.2s ease;\n}\n\n.chatMainFlow .chatMessages::-webkit-scrollbar-thumb:hover {\n    background: rgba(125, 125, 125, 0.3);\n}\n\n/* Individual chat message */\n.chatMainFlow .chatMessage {\n    display: flex;\n    margin-bottom: 20px;\n    align-items: flex-end;\n    flex-direction: row;\n    column-gap: var(--chat-message-avatar-gap);\n    position: relative;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    max-width: 100%;\n}\n\n.chatMainFlow .chatMessage .messageStack {\n    display: flex;\n    flex-direction: column;\n    align-items: flex-start;\n    max-width: var(--chat-message-max-width);\n}\n\n.chatMainFlow .chatMessage.isMe .messageStack {\n    align-items: flex-end;\n}\n\n.chatMainFlow .chatMessage .messageStack > * {\n    max-width: 100%;\n}\n\n.messageMeta {\n    display: flex;\n    align-items: center;\n    gap: 6px;\n    margin: 4px 6px 0;\n    font-size: 11px;\n    color: #6b7280;\n    line-height: 1.4;\n}\n\n.chatMainFlow .chatMessage.isMe .messageMeta {\n    align-self: flex-end;\n    text-align: right;\n}\n\n.messageTimestamp {\n    letter-spacing: 0.02em;\n}\n\n.messageDuration {\n    opacity: 0.85;\n}\n\n.participantLabel {\n    font-size: 12px;\n    font-weight: 600;\n    color: #64748b;\n    margin: 0 0 6px;\n}\n\n.chatMainFlow .chatMessage.isMe .participantLabel {\n    text-align: right;\n}\n\n@keyframes messageSlideIn {\n    from {\n        opacity: 0;\n        transform: translateY(20px) scale(0.95);\n    }\n    to {\n        opacity: 1;\n        transform: translateY(0) scale(1);\n    }\n}\n\n.isNotCompleteMessage {\n    /*/\n    outline: 1px dotted #ff0000 !important;\n    /**/\n\n    opacity: 0.7;\n    position: relative;\n}\n\n.NonCompleteMessageFiller {\n    color: transparent;\n}\n\n/* Enhanced loading states for messages */\n.isNotCompleteMessage .messageText::after {\n    content: '';\n    position: absolute;\n    bottom: 8px;\n    right: 12px;\n    width: 20px;\n    height: 4px;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.6), transparent);\n    border-radius: 2px;\n    animation: loadingPulse 1.5s ease-in-out infinite;\n}\n\n.ongoingToolCalls,\n.completedToolCalls {\n    display: flex;\n    flex-direction: row;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 8px;\n    padding-top: 8px;\n    border-top: 1px solid rgba(255, 255, 255, 0.2);\n}\n\n.sourceCitations {\n    display: flex;\n    flex-direction: row;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 8px;\n    padding-top: 8px;\n    border-top: 1px solid rgba(255, 255, 255, 0.2);\n}\n\n.ongoingToolCall,\n.completedToolCall {\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    font-size: 0.8em;\n    padding: 4px 10px;\n    background: rgba(255, 255, 255, 0.15);\n    border: 1px solid rgba(255, 255, 255, 0.2);\n    border-radius: 12px;\n    box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);\n    color: inherit;\n    font-family: 'OpenMojiColor', 'OpenMojiBlack', Arial, Helvetica, sans-serif;\n}\n\n.completedToolCall {\n    cursor: pointer;\n    transition: all 0.2s ease;\n}\n\n.completedToolCall:hover {\n    background: rgba(255, 255, 255, 0.3);\n    transform: translateY(-1px);\n    box-shadow: 0 2px 6px rgba(0, 0, 0, 0.15);\n}\n\n.toolCallOrigin {\n    font-size: 0.85em;\n    opacity: 0.75;\n    font-weight: 500;\n    white-space: nowrap;\n}\n\n.toolCallDetails {\n    margin-bottom: 24px;\n    text-align: left;\n}\n\n.toolCallDetails p {\n    margin-bottom: 8px;\n}\n\n.toolCallData {\n    background: #f1f5f9;\n    padding: 12px;\n    border-radius: 8px;\n    font-size: 13px;\n    overflow-x: auto;\n    white-space: pre-wrap;\n    word-break: break-all;\n    max-height: 300px;\n    border: 1px solid #e2e8f0;\n}\n\n.ongoingToolCallSpinner {\n    width: 14px;\n    height: 14px;\n    border: 2px solid rgba(255, 255, 255, 0.3);\n    border-top-color: #fff;\n    border-radius: 50%;\n    animation: toolCallSpinner 0.8s linear infinite;\n}\n\n.ongoingToolCallName {\n    font-weight: 500;\n}\n\n@keyframes toolCallSpinner {\n    to {\n        transform: rotate(360deg);\n    }\n}\n\n@keyframes loadingPulse {\n    0%,\n    100% {\n        opacity: 0.3;\n        transform: scaleX(0.8);\n    }\n    50% {\n        opacity: 1;\n        transform: scaleX(1.2);\n    }\n}\n\n/* Typing indicator for AI messages */\n.typingIndicator {\n    display: flex;\n    align-items: flex-end;\n    margin-bottom: 20px;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.typingIndicator .avatar {\n    width: 40px;\n    height: 40px;\n    margin: 0 12px 4px;\n    flex-shrink: 0;\n}\n\n/* [㊗️]\n.typingIndicator .avatar img {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    border-radius: 50%;\n    object-fit: cover;\n    background-color: #eef6fb;\n    border: 2px solid rgba(125, 125, 125, 0.1);\n}\n*/\n\n.typingBubble {\n    padding: 16px 20px;\n    border-radius: 20px;\n    border-bottom-left-radius: 6px;\n    border: 1px solid rgba(125, 125, 125, 0.1);\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);\n    backdrop-filter: blur(10px);\n    display: flex;\n    align-items: center;\n    gap: 4px;\n    min-height: 24px;\n}\n\n.typingDots {\n    display: flex;\n    gap: 4px;\n    align-items: center;\n}\n\n.typingDot {\n    width: 8px;\n    height: 8px;\n    border-radius: 50%;\n    background: linear-gradient(135deg, #6b7280 0%, rgba(125, 125, 125, 0.6) 100%);\n    animation: typingBounce 1.4s infinite ease-in-out;\n    box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);\n}\n\n.typingDot:nth-child(1) {\n    animation-delay: -0.32s;\n}\n\n.typingDot:nth-child(2) {\n    animation-delay: -0.16s;\n}\n\n.typingDot:nth-child(3) {\n    animation-delay: 0s;\n}\n\n@keyframes typingBounce {\n    0%,\n    80%,\n    100% {\n        transform: scale(0.8) translateY(0);\n        opacity: 0.5;\n    }\n    40% {\n        transform: scale(1.2) translateY(-8px);\n        opacity: 1;\n    }\n}\n\n.chatMainFlow .chatMessage.isMe {\n    align-items: flex-end;\n    flex-direction: row-reverse;\n    justify-content: flex-start;\n}\n\n.chatMainFlow .chatMessage.isMe .messageText {\n    border-bottom-right-radius: 6px;\n}\n\n.ratingStar {\n    cursor: pointer;\n    font-size: 20px;\n    transition: color 0.2s;\n    color: var(--star-inactive-color, #ccc);\n}\n\n.ratingStar.active {\n    color: #ffd700;\n}\n\n/* Sender Avatar */\n.chatMainFlow .chatMessage .avatar {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    margin: 0 0 4px;\n    flex-shrink: 0;\n    position: relative;\n}\n\n/* [㊗️]\n.chatMainFlow .chatMessage .avatar img {\n    width: 40px;\n    aspect-ratio: 1 / 1;\n    border-radius: 50%;\n    object-fit: cover;\n    background-color: var(--avatar-bg-color, #eef6fb);\n    border: 2px solid rgba(125, 125, 125, 0.1);\n    transition: transform 0.2s ease, box-shadow 0.2s ease;\n}\n\n\n.chatMainFlow .chatMessage .avatar img:hover {\n    transform: scale(1.05);\n    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\n}\n*/\n\n/* Message text bubble */\n.chatMainFlow .chatMessage .messageText {\n    background-color: var(--message-bg-color);\n    color: var(--message-text-color);\n    position: relative;\n    padding: 14px 18px;\n    border-radius: 20px;\n    max-width: 100%;\n    min-width: var(--chat-message-min-width);\n    text-align: left;\n    margin-bottom: 4px;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);\n    line-height: 1.5;\n    word-wrap: break-word;\n    transition: all 0.2s ease;\n    font-size: 15px;\n    backdrop-filter: blur(10px);\n    --chat-heading-surface: rgba(255, 255, 255, 0.12);\n    --chat-heading-accent: rgba(56, 189, 248, 0.85);\n    --chat-heading-text-color: var(--message-text-color, #0f172a);\n    --chat-heading-border-color: rgba(15, 23, 42, 0.18);\n    --chat-heading-shadow-color: rgba(15, 23, 42, 0.65);\n}\n\n@supports (color: color-mix(in srgb, #000 50%, #fff 50%)) {\n    .chatMainFlow .chatMessage .messageText {\n        --chat-heading-surface: color-mix(in srgb, var(--message-bg-color, #ffffff) 70%, rgba(15, 23, 42, 0.08) 30%);\n        --chat-heading-accent: color-mix(in srgb, var(--message-text-color, #0f172a) 65%, rgba(59, 130, 246, 0.9) 35%);\n        --chat-heading-border-color: color-mix(\n            in srgb,\n            var(--message-bg-color, #ffffff) 60%,\n            rgba(15, 23, 42, 0.12) 40%\n        );\n        --chat-heading-shadow-color: color-mix(in srgb, rgba(15, 23, 42, 0.65) 80%, rgba(96, 165, 250, 0.45) 20%);\n        --chat-heading-text-color: var(--message-text-color, #0f172a);\n    }\n}\n\n/* Copy button styles */\n.copyButtonContainer {\n    float: right;\n    top: 8px;\n    right: 10px;\n    z-index: 2;\n    pointer-events: none;\n\n    visibility: hidden;\n}\n.chatMainFlow .chatMessage .messageText:hover .copyButtonContainer,\n.copyButtonContainer:focus-within {\n    visibility: visible;\n    pointer-events: auto;\n}\n\n.messageControlGroup {\n    display: flex;\n    align-items: center;\n    gap: 6px;\n    pointer-events: auto;\n}\n\n.copyButton {\n    background: rgba(255, 255, 255, 0.2);\n    border: 1px solid #ddd;\n    border-radius: 6px;\n    padding: 2px 5px;\n    cursor: pointer;\n    box-shadow: 0 1px 4px rgba(0, 0, 0, 0.07);\n    transition: all 0.15s, box-shadow 0.15s, border 0.15s;\n    display: flex;\n    align-items: center;\n    opacity: 0.7;\n    position: relative;\n}\n\n.playButton {\n    width: 36px;\n    height: 36px;\n    border-radius: 12px;\n    border: 1px solid rgba(255, 255, 255, 0.6);\n    background: rgba(255, 255, 255, 0.25);\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    transition: all 0.15s ease;\n    color: #111;\n    position: relative;\n    padding: 0;\n}\n\n.playButtonIcon {\n    width: 16px;\n    height: 16px;\n}\n\n.playButtonSpinner {\n    width: 16px;\n    height: 16px;\n    border-radius: 50%;\n    border: 2px solid rgba(0, 0, 0, 0.3);\n    border-top-color: #111;\n    animation: messagePlaySpinner 0.9s linear infinite;\n}\n\n.playButton:hover:not(:disabled) {\n    background: rgba(255, 255, 255, 0.4);\n    border-color: rgba(255, 255, 255, 0.8);\n}\n\n.playButton:disabled {\n    cursor: not-allowed;\n    opacity: 0.6;\n}\n\n@keyframes messagePlaySpinner {\n    from {\n        transform: rotate(0deg);\n    }\n\n    to {\n        transform: rotate(360deg);\n    }\n}\n\n.copiedTooltip {\n    position: absolute;\n    left: 50%;\n    top: 110%;\n    transform: translateX(-50%);\n    background: #222;\n    color: #fff;\n    padding: 6px 16px;\n    border-radius: 8px;\n    font-size: 0.98em;\n    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.18);\n    opacity: 0.97;\n    pointer-events: none;\n    z-index: 100;\n    white-space: nowrap;\n    margin-top: 4px;\n    animation: copiedTooltipFadeIn 0.2s;\n    max-width: 220px;\n    overflow-wrap: break-word;\n    word-break: break-word;\n}\n\n.copiedTooltipLeft {\n    left: 0 !important;\n    transform: none !important;\n}\n\n.copiedTooltipRight {\n    left: auto !important;\n    right: 0 !important;\n    transform: none !important;\n}\n\n/* Removed right-aligned override to keep tooltip position stable */\n\n@keyframes copiedTooltipFadeIn {\n    from {\n        opacity: 0;\n        transform: translateX(-50%) translateY(8px) scale(0.97);\n    }\n    to {\n        opacity: 0.97;\n        transform: translateX(-50%) translateY(0) scale(1);\n    }\n}\n\n.copyButton:hover,\n.copyButton:focus {\n    border: 1.5px solid #bbb;\n    opacity: 1;\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.1);\n}\n\n.copyButton svg {\n    display: block;\n}\n\n.messageText ul {\n    list-style: disc;\n    margin-left: 20px;\n}\n\n.messageText ol {\n    list-style: decimal;\n    margin-left: 20px;\n}\n\n.messageText img,\n.messageText pre,\n.messageText blockquote,\n.messageText table {\n    margin-top: 10px;\n    margin-bottom: 10px;\n    border-radius: 8px;\n}\n\n.messageText pre {\n    display: block;\n    border: none;\n    box-shadow: none;\n    background: #000000ff;\n    font-size: inherit;\n    line-height: inherit;\n    color: #fff;\n    padding: 1em;\n}\n\n.messageText blockquote {\n    display: block;\n    border: none;\n    box-shadow: none;\n    background: #ffffffcc;\n    font-size: inherit;\n    line-height: inherit;\n    color: #000;\n    padding: 1em;\n}\n\n.messageText code {\n    display: inline-block;\n    margin: 0;\n    padding: 0;\n    border: none;\n    box-shadow: none;\n    background: #cccccc55;\n    font-size: inherit;\n    line-height: inherit;\n    color: inherit;\n}\n\n.messageText pre code {\n    background-color: #000000cc;\n    border-radius: 8px;\n}\n\n.messageText .chat-code-block {\n    background: #181c23;\n    color: #f8fafc;\n    font-size: 14px;\n    line-height: 1.6;\n    overflow-x: auto;\n    border-color: #23272f;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.12);\n    font-family: 'Fira Mono', 'Menlo', 'Consolas', 'Liberation Mono', monospace;\n}\n.messageText .chat-code-block code {\n    background: none !important;\n    color: inherit !important;\n    font-family: inherit !important;\n    font-size: inherit !important;\n    padding: 0 !important;\n    border: none !important;\n    box-shadow: none !important;\n    white-space: pre;\n    word-break: break-word;\n    overflow-x: auto;\n    display: block;\n}\n.messageText table {\n    width: 100%;\n    border-collapse: separate;\n    border-spacing: 0;\n    margin: 16px 0;\n    background: #f8fafc; /* Stronger light background for contrast */\n    border-radius: 12px;\n    overflow: hidden;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);\n    font-size: 14px;\n    color: #17223b; /* Dark text for contrast */\n}\n.messageText th,\n.messageText td {\n    padding: 10px 16px;\n    border-bottom: 1px solid #d1dbe8;\n    text-align: left;\n    color: #17223b; /* Ensure strong text color for all cells */\n    background: none;\n}\n.messageText th {\n    background: linear-gradient(90deg, #eaf3fa 80%, #d1e3f8 100%);\n    font-weight: 700;\n    color: #17223b; /* Strong header text */\n    border-bottom: 2px solid #b5c7de;\n}\n.messageText tr:last-child td {\n    border-bottom: none;\n}\n.messageText tr:nth-child(even) td {\n    background: #eaf3fa;\n}\n.messageText tr:hover td {\n    background: #cbe0f7;\n    transition: background 0.2s;\n}\n.messageText table {\n    border-radius: 12px;\n    overflow: hidden;\n}\n.messageText th:first-child,\n.messageText td:first-child {\n    border-top-left-radius: 12px;\n}\n.messageText th:last-child,\n.messageText td:last-child {\n    border-top-right-radius: 12px;\n}\n\n.chatMainFlow .chatMessage .messageText:hover {\n    box-shadow: 0 4px 16px rgba(0, 0, 0, 0.15);\n    transform: translateY(-1px);\n}\n\n/* Attachments styles */\n.attachments {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 10px;\n    padding-top: 10px;\n    border-top: 1px solid rgba(125, 125, 125, 0.2);\n}\n\n.attachment {\n    display: flex;\n    align-items: center;\n    gap: 6px;\n    padding: 6px 12px;\n    background: rgba(255, 255, 255, 0.2);\n    border: 1px solid rgba(125, 125, 125, 0.3);\n    border-radius: 12px;\n    font-size: 13px;\n    color: inherit;\n    text-decoration: none;\n    transition: all 0.2s ease;\n    max-width: 200px;\n}\n\n.attachment:hover {\n    background: rgba(255, 255, 255, 0.3);\n    border-color: rgba(125, 125, 125, 0.5);\n    transform: translateY(-1px);\n}\n\n.attachmentIcon {\n    font-size: 14px;\n    opacity: 0.8;\n}\n\n.attachmentName {\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n}\n\n/* Message buttons container */\n.messageButtons {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-top: 12px;\n    padding-top: 12px;\n    border-top: 1px solid rgba(125 125 125 / 0.83);\n}\n\n/* Individual message button */\n.messageButton {\n    display: inline-flex;\n    align-items: center;\n    padding: 8px 14px;\n    background: rgba(125, 125, 125, 0.1);\n    border: 1px solid rgba(125, 125, 125, 0.9);\n    border-radius: 16px;\n    font-size: 13px;\n    font-weight: 500;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    backdrop-filter: blur(10px);\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n}\n\n.messageButton:hover {\n    background: rgba(0, 132, 255, 0.1);\n    border-color: rgba(0, 132, 255, 0.3);\n    transform: translateY(-1px);\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.15);\n}\n\n.messageButton:active {\n    transform: scale(0.98);\n    transition: transform 0.1s ease;\n}\n\n/* Remove default markdown styles from button content */\n.messageButton p {\n    margin: 0;\n    padding: 0;\n    line-height: inherit;\n}\n\n.messageButton strong {\n    font-weight: 600;\n}\n\n.messageButton em {\n    font-style: italic;\n}\n\n/* Rating system */\n.chatMainFlow .chatMessage .rating {\n    position: absolute;\n    bottom: -8px;\n    right: 8px;\n    display: flex;\n    gap: 2px;\n    align-items: center;\n    min-width: 24px;\n    z-index: 1;\n    background: rgba(0, 0, 0, 0.8);\n    border-radius: 12px;\n    padding: 4px 6px;\n    backdrop-filter: blur(10px);\n    opacity: 0;\n    transform: translateY(4px);\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.chatMainFlow .chatMessage:hover .rating {\n    opacity: 1;\n    transform: translateY(0);\n}\n\n.chatMainFlow .chatMessage .rating:hover {\n    background: rgba(0, 0, 0, 0.9);\n    padding: 6px 8px;\n    box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);\n}\n\n.chatMainFlow .chatMessage .rating span {\n    transition: transform 0.2s ease, color 0.2s ease;\n    display: inline-block;\n    cursor: pointer;\n    font-size: 16px;\n}\n\n.chatMainFlow .chatMessage .rating:hover span {\n    transform: scale(1.1);\n}\n\n/* Chat input area */\n.chatMainFlow .chatInput {\n    z-index: 10;\n    grid-area: 📝;\n    width: 100%;\n    padding: 24px;\n\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n    position: relative;\n}\n\n.Chat.fullPageVisual .chatMainFlow .chatInput {\n    backdrop-filter: blur(25px);\n    background: linear-gradient(180deg, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.8) 100%);\n}\n\n/* File upload drag-and-drop styles */\n.chatMainFlow .chatInput.dragOver {\n    background: linear-gradient(to top, rgba(0, 132, 255, 0.1) 0%, rgba(0, 132, 255, 0.05) 100%);\n}\n\n/* File preview container */\n.filePreviewContainer {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n    margin-bottom: 8px;\n}\n\n/* Individual file preview */\n.filePreview {\n    display: flex;\n    align-items: center;\n    gap: 8px;\n    padding: 8px 12px;\n    background: rgba(125, 125, 125, 0.1);\n    border: 1px solid rgba(125, 125, 125, 0.2);\n    border-radius: 8px;\n    font-size: 12px;\n    backdrop-filter: blur(10px);\n    transition: all 0.2s ease;\n}\n\n.filePreview:hover {\n    background: rgba(125, 125, 125, 0.15);\n    border-color: rgba(125, 125, 125, 0.3);\n}\n\n.fileIcon {\n    font-size: 14px;\n    opacity: 0.7;\n}\n\n.fileInfo {\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n    min-width: 0;\n}\n\n.fileName {\n    font-weight: 500;\n    color: black;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    max-width: 150px;\n}\n\n.fileSize {\n    color: #6b7280;\n    font-size: 11px;\n}\n\n.removeFileButton {\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    width: 20px;\n    height: 20px;\n    border: none;\n    background: rgba(255, 0, 0, 0.1);\n    color: #ff4444;\n    border-radius: 50%;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n}\n\n.removeFileButton:hover {\n    background: rgba(255, 0, 0, 0.2);\n    transform: scale(1.1);\n}\n\n/* Input container for textarea and buttons */\n.inputContainer {\n    display: flex;\n    align-items: flex-end;\n    gap: 8px;\n    background: #ffffff;\n    border: 1px solid rgba(0, 0, 0, 0.08);\n    border-radius: 28px;\n    padding: 8px 12px;\n    box-shadow: 0 4px 20px rgba(0, 0, 0, 0.06);\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.inputContainer:focus-within {\n    border-color: var(--brand-color);\n    box-shadow: 0 8px 32px rgba(0, 132, 255, 0.12);\n    transform: translateY(-2px);\n}\n\n.inputContainer textarea::placeholder {\n    color: #94a3b8;\n    opacity: 1;\n}\n\n/* Attachment button */\n.attachmentButton {\n    width: 40px;\n    height: 40px;\n    border: none;\n    background: transparent;\n    color: #64748b;\n    border-radius: 50%;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n    margin-bottom: 2px;\n}\n\n.attachmentButton:hover:not(:disabled) {\n    background: #f1f5f9;\n    color: #0f172a;\n}\n\n.attachmentButton:disabled {\n    opacity: 0.3;\n    cursor: not-allowed;\n}\n\n/* Voice Button */\n.voiceButton {\n    width: 40px;\n    height: 40px;\n    border: none;\n    background: transparent;\n    color: #64748b;\n    border-radius: 50%;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    flex-shrink: 0;\n    margin-bottom: 2px;\n}\n\n.voiceButton:hover:not(:disabled) {\n    background: #f1f5f9;\n    color: #0f172a;\n}\n\n.voiceButtonActive {\n    background: rgba(239, 68, 68, 0.1) !important;\n    color: #ef4444 !important;\n    animation: voiceRecordingPulse 1.5s infinite;\n}\n\n@keyframes voiceRecordingPulse {\n    0% {\n        box-shadow: 0 0 0 0 rgba(255, 0, 0, 0.4);\n    }\n    70% {\n        box-shadow: 0 0 0 10px rgba(255, 0, 0, 0);\n    }\n    100% {\n        box-shadow: 0 0 0 0 rgba(255, 0, 0, 0);\n    }\n}\n\n/* Upload progress indicator */\n.uploadProgress {\n    display: flex;\n    align-items: center;\n    gap: 12px;\n    padding: 8px 12px;\n    background: rgba(0, 132, 255, 0.1);\n    border: 1px solid rgba(0, 132, 255, 0.2);\n    border-radius: 8px;\n    font-size: 13px;\n    color: #0084ff;\n}\n\n.uploadProgressBar {\n    flex: 1;\n    height: 4px;\n    background: rgba(0, 132, 255, 0.2);\n    border-radius: 2px;\n    overflow: hidden;\n}\n\n.uploadProgressFill {\n    height: 100%;\n    background: linear-gradient(90deg, #0084ff, rgba(0, 132, 255, 0.8));\n    border-radius: 2px;\n    animation: uploadProgress 1.5s ease-in-out infinite;\n}\n\n@keyframes uploadProgress {\n    0% {\n        transform: translateX(-100%);\n    }\n    50% {\n        transform: translateX(0%);\n    }\n    100% {\n        transform: translateX(100%);\n    }\n}\n\n/* Drag overlay */\n.dragOverlay {\n    position: absolute;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    background: rgba(0, 132, 255, 0.1);\n    border: 2px dashed rgba(0, 132, 255, 0.5);\n    border-radius: 12px;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    backdrop-filter: blur(10px);\n    z-index: 20;\n    pointer-events: none;\n}\n\n.dragOverlayContent {\n    display: flex;\n    flex-direction: column;\n    align-items: center;\n    gap: 12px;\n    color: #0084ff;\n    font-weight: 600;\n    text-align: center;\n}\n\n.dragOverlayContent svg {\n    opacity: 0.7;\n}\n\n/* Chat input field */\n.chatMainFlow .chatInput textarea {\n    flex: 1;\n    padding: 10px 12px;\n    margin: 0;\n    border: none;\n    outline: none;\n    background: transparent;\n    color: #0f172a;\n    min-width: 100px;\n    max-height: 200px;\n    font-size: 15px;\n    line-height: 1.5;\n    resize: none;\n    appearance: none;\n    -webkit-appearance: none;\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n}\n\n.chatMainFlow .chatInput textarea:focus {\n    /* Focus is handled by .inputContainer */\n}\n\n.chatMainFlow .chatInput textarea:disabled {\n    opacity: 0.6;\n    cursor: not-allowed;\n    transform: none;\n}\n\n.chatMainFlow .chatInput textarea::placeholder {\n    color: inherit;\n    opacity: 0.7;\n}\n\n/* Chat send button */\n.chatMainFlow .chatInput button[data-button-type='call-to-action'] {\n    width: 40px;\n    height: 40px;\n    margin: 0 0 2px 0 !important;\n    padding: 0 !important;\n\n    border: none;\n    color: #ffffff;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    border-radius: 50% !important;\n    aspect-ratio: 1 / 1;\n    min-width: unset !important;\n    min-height: unset !important;\n\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    box-shadow: 0 4px 12px rgba(0, 132, 255, 0.25);\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n    overflow: visible;\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action']:hover {\n    transform: scale(1.05);\n    box-shadow: 0 6px 16px rgba(0, 132, 255, 0.35);\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action']:active {\n    transform: scale(0.95);\n}\n\n.chatMainFlow .chatInput button[data-button-type='call-to-action'] svg {\n    width: 20px;\n    height: 20px;\n}\n\n.chatMainFlow .chatInput button img {\n    width: 50%;\n    height: 100%;\n    object-fit: contain;\n}\n\n/* Scroll to bottom button */\n\n.scrollToBottomContainer {\n    /*/\n    outline: 1px dotted red;\n    /**/\n\n    z-index: 20;\n    grid-area: 📝;\n    width: 100%;\n    height: 100%;\n    display: flex;\n    justify-content: center;\n    align-items: flex-start;\n\n    pointer-events: none;\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper {\n    pointer-events: all;\n    position: relative;\n    transform: translate(-50%, -150%);\n    display: inline-flex;\n    justify-content: center;\n    align-items: center;\n    animation: scrollButtonSlideIn 0.3s ease-out;\n    transition: transform 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n}\n\n.scrollToBottomContainer .scrollToBottom {\n    width: 48px;\n    height: 48px;\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    border: none;\n    outline: none;\n    background: rgba(0 0 0 / 0.5);\n    border-radius: 50%;\n    font-weight: bold;\n    color: white;\n    cursor: pointer;\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    box-shadow: 0 4px 16px rgba(0, 0, 0, 0.3);\n}\n\n@keyframes scrollButtonSlideIn {\n    from {\n        opacity: 0;\n        transform: translate(-50%, 20px) scale(0.8);\n    }\n    to {\n        opacity: 0.9;\n        transform: translate(-50%, 0) scale(1);\n    }\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper:hover {\n    transform: translate(-50%, -160%) scale(1.05);\n}\n\n.scrollToBottomContainer .scrollToBottomWrapper:active {\n    transform: translate(-50%, -160%) scale(0.95);\n    transition: transform 0.1s ease;\n}\n\n.scrollToBottomBadge {\n    position: absolute;\n    top: -8px;\n    right: -8px;\n    min-width: 32px;\n    padding: 2px 6px;\n    font-size: 10px;\n    font-weight: 600;\n    background: #2563eb;\n    color: #fff;\n    border-radius: 999px;\n    box-shadow: 0 2px 10px rgba(37, 99, 235, 0.4);\n    text-transform: uppercase;\n    letter-spacing: 0.04em;\n    white-space: nowrap;\n    pointer-events: none;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n}\n\n/* Rating modal */\n.ratingModal {\n    position: fixed;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n\n    background-color: rgba(0, 0, 0, 0.6);\n    backdrop-filter: blur(8px);\n\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    z-index: 1000;\n    animation: modalFadeIn 0.3s ease-out;\n}\n\n@keyframes modalFadeIn {\n    from {\n        opacity: 0;\n        backdrop-filter: blur(0px);\n    }\n    to {\n        opacity: 1;\n        backdrop-filter: blur(8px);\n    }\n}\n\n.ratingModalContent {\n    background: #ffffff;\n    color: #0f1724;\n    padding: 32px;\n    border-radius: 16px;\n    width: 90%;\n    max-width: 480px;\n    box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3);\n    border: 1px solid rgba(125, 125, 125, 0.1);\n    backdrop-filter: blur(20px);\n    animation: modalSlideIn 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiColor', 'OpenMojiBlack';\n    /* <- TODO: [🧠][🎱] Better, define other fonts */\n}\n\n@keyframes modalSlideIn {\n    from {\n        opacity: 0;\n        transform: translateY(20px) scale(0.95);\n    }\n    to {\n        opacity: 1;\n        transform: translateY(0) scale(1);\n    }\n}\n\n.ratingModalContent h3 {\n    margin: 0 0 24px 0;\n    text-align: center;\n    color: black;\n    font-size: 20px;\n    font-weight: 600;\n}\n\n.stars {\n    display: flex;\n    justify-content: center;\n    gap: 8px;\n    margin-bottom: 24px;\n}\n\n.stars span {\n    font-size: 28px;\n    cursor: pointer;\n    transition: all 0.2s ease;\n    padding: 4px;\n    border-radius: 8px;\n}\n\n.stars span:hover {\n    transform: scale(1.2);\n    background: rgba(255, 215, 0, 0.1);\n}\n\n.ratingModalStar {\n    cursor: pointer;\n    font-size: 24px;\n    transition: color 0.2s;\n}\n\n.toolCallModal {\n    max-width: 800px;\n    padding: 0;\n    overflow: hidden;\n    position: relative;\n}\n\n.toolCallHeader {\n    display: flex;\n    align-items: center;\n    gap: 16px;\n    padding: 24px 32px 16px;\n    background: linear-gradient(135deg, #eef2ff 0%, #f5f3ff 40%, #fefcc9 100%);\n    border-bottom: 1px solid rgba(15, 23, 42, 0.08);\n}\n\n.toolCallIcon {\n    font-size: 32px;\n}\n\n.toolCallHeaderMeta {\n    display: flex;\n    flex-direction: column;\n    gap: 4px;\n}\n\n.toolCallLabel {\n    font-size: 11px;\n    letter-spacing: 0.35em;\n    text-transform: uppercase;\n    color: #475467;\n    margin: 0;\n}\n\n.toolCallTitle {\n    margin: 0;\n    font-size: 24px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.toolCallSubtitle {\n    margin: 0;\n    font-size: 13px;\n    color: #475467;\n}\n\n.toolCallGrid {\n    display: grid;\n    grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));\n    gap: 16px;\n    padding: 24px 32px 16px;\n    background: #fefefe;\n}\n\n.toolCallPanel {\n    background: #ffffff;\n    border-radius: 16px;\n    border: 1px solid #e2e8f0;\n    padding: 18px;\n    box-shadow: 0 18px 40px rgba(15, 23, 42, 0.08);\n}\n\n.toolCallPanelTitle {\n    margin: 0 0 12px 0;\n    font-size: 12px;\n    font-weight: 600;\n    letter-spacing: 0.25em;\n    text-transform: uppercase;\n    color: #94a3b8;\n}\n\n.toolCallList {\n    list-style: none;\n    margin: 0;\n    padding: 0;\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n}\n\n.toolCallItem {\n    display: flex;\n    flex-direction: column;\n    gap: 4px;\n}\n\n.toolCallItemLabel {\n    font-size: 11px;\n    letter-spacing: 0.25em;\n    text-transform: uppercase;\n    color: #94a3b8;\n}\n\n.toolCallItemValue {\n    font-size: 14px;\n    color: #0f172a;\n    line-height: 1.4;\n    word-break: break-word;\n    white-space: pre-wrap;\n}\n\n.toolCallSummary {\n    margin: 0;\n    font-size: 15px;\n    color: #0f172a;\n    line-height: 1.6;\n}\n\n.toolCallSummaryMeta {\n    margin-top: 14px;\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n}\n\n.toolCallSummaryMetaBadge {\n    padding: 5px 12px;\n    border-radius: 999px;\n    background: rgba(79, 70, 229, 0.15);\n    color: #312e81;\n    font-size: 12px;\n    font-weight: 600;\n}\n\n.toolCallEmpty {\n    margin: 0;\n    color: #94a3b8;\n    font-size: 14px;\n}\n\n.toolCallIssues {\n    padding: 0 32px 16px;\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n}\n\n.toolCallIssueBadge {\n    display: inline-flex;\n    align-items: center;\n    gap: 4px;\n    font-size: 12px;\n    font-weight: 600;\n    padding: 6px 10px;\n    border-radius: 999px;\n}\n\n.toolCallIssueError {\n    background: rgba(239, 68, 68, 0.12);\n    color: #b91c1c;\n}\n\n.toolCallIssueWarning {\n    background: rgba(234, 179, 8, 0.16);\n    color: #a16207;\n}\n\n.toolCallFooter {\n    display: flex;\n    justify-content: space-between;\n    align-items: center;\n    gap: 12px;\n    padding: 0 32px 24px;\n    flex-wrap: wrap;\n}\n\n.toolCallTimestamp {\n    font-size: 13px;\n    color: #475467;\n}\n\n.toolCallRawDetails {\n    font-size: 13px;\n    color: #2563eb;\n}\n\n.toolCallRawSummary {\n    cursor: pointer;\n    font-weight: 600;\n    color: #2563eb;\n}\n\n.toolCallRawColumns {\n    margin-top: 12px;\n    display: grid;\n    grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));\n    gap: 12px;\n}\n\n.toolCallRawColumnTitle {\n    margin: 0 0 8px 0;\n    font-size: 11px;\n    letter-spacing: 0.25em;\n    text-transform: uppercase;\n    color: #94a3b8;\n}\n\n.memoryModalHeader {\n    padding: 28px 32px 16px;\n    display: flex;\n    align-items: flex-start;\n    gap: 16px;\n    background: linear-gradient(180deg, #fdfdfd 0%, #f6f9ff 100%);\n    border-bottom: 1px solid #e2e8f0;\n}\n\n.memoryModalIcon {\n    width: 52px;\n    height: 52px;\n    border-radius: 18px;\n    background: linear-gradient(135deg, #6366f1, #a855f7);\n    color: #fff;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    font-size: 24px;\n    box-shadow: 0 18px 28px rgba(15, 23, 42, 0.18);\n}\n\n.memoryModalHeaderText {\n    flex: 1;\n}\n\n.memoryModalTitle {\n    margin: 0;\n    font-size: 22px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.memoryModalSubtitle {\n    margin: 4px 0 0;\n    color: #475467;\n    font-size: 14px;\n}\n\n.memoryModalCallTime {\n    margin: 4px 0 0;\n    color: #94a3b8;\n    font-size: 13px;\n}\n\n.memoryModalStatus {\n    margin-left: auto;\n    border-radius: 999px;\n    padding: 6px 14px;\n    font-size: 13px;\n    font-weight: 600;\n    display: inline-flex;\n    align-items: center;\n    gap: 6px;\n    white-space: nowrap;\n}\n\n.memoryStatusDot {\n    width: 8px;\n    height: 8px;\n    border-radius: 50%;\n    background: currentColor;\n    display: inline-block;\n}\n\n.memoryModalContent {\n    padding: 24px 32px 32px;\n    background: #ffffff;\n    display: flex;\n    flex-direction: column;\n    gap: 16px;\n}\n\n.memoryMessage {\n    margin: 0;\n    color: #0f172a;\n    font-size: 15px;\n    font-weight: 500;\n}\n\n.memoryMetaRow {\n    display: flex;\n    justify-content: space-between;\n    align-items: center;\n    font-size: 13px;\n    color: #475467;\n}\n\n.memoryMetaLabel {\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    font-weight: 600;\n}\n\n.memoryMetaValue {\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.memoryStoreSection,\n.memoryRetrieveSection {\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n}\n\n.memoryCard {\n    border: 1px solid #e5e7eb;\n    border-radius: 16px;\n    padding: 18px;\n    background: #f8fafc;\n    box-shadow: 0 10px 18px rgba(15, 23, 42, 0.08);\n}\n\n.memoryCardContent {\n    color: #0f172a;\n    line-height: 1.6;\n    font-size: 14px;\n    white-space: pre-wrap;\n    overflow-wrap: break-word;\n}\n\n.memoryCardMeta {\n    margin-top: 12px;\n    display: flex;\n    justify-content: space-between;\n    align-items: center;\n    color: #94a3b8;\n    font-size: 12px;\n}\n\n.memoryScopeBadge {\n    padding: 3px 10px;\n    border-radius: 999px;\n    font-size: 11px;\n    font-weight: 700;\n    color: #0f172a;\n    background: rgba(124, 58, 237, 0.12);\n}\n\n.memoryList {\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n}\n\n.memoryEmptyState {\n    padding: 16px;\n    background: #fef3c7;\n    border: 1px solid #fde68a;\n    border-radius: 12px;\n    color: #92400e;\n    font-weight: 500;\n}\n\n.memoryListFooter {\n    font-size: 13px;\n    color: #475467;\n    text-align: center;\n}\n\n.memoryStatusSuccess {\n    background: rgba(16, 185, 129, 0.16);\n    color: #047857;\n}\n\n.memoryStatusWarning {\n    background: rgba(234, 179, 8, 0.16);\n    color: #a16207;\n}\n\n.memoryStatusError {\n    background: rgba(239, 68, 68, 0.15);\n    color: #b91c1c;\n}\n\n.memoryStatusNeutral {\n    background: rgba(96, 165, 250, 0.15);\n    color: #1d4ed8;\n}\n\n.searchModalHeader {\n    padding: 24px 32px;\n    padding-right: 72px;\n    border-bottom: 1px solid #eee;\n    display: flex;\n    align-items: center;\n    gap: 16px;\n    background: #fdfdfd;\n}\n\n.selfLearningModalHeader {\n    align-items: center;\n    gap: 18px;\n    background: linear-gradient(135deg, #f8fafc 0%, #fef9c3 45%, #ecfccb 100%);\n    border-bottom: 1px solid #e2e8f0;\n}\n\n.selfLearningAvatarGroup {\n    display: flex;\n    align-items: center;\n    gap: 10px;\n}\n\n.selfLearningAvatar {\n    width: 48px;\n    height: 48px;\n    border-radius: 50%;\n    background: #e2e8f0;\n    color: #0f172a;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    font-weight: 600;\n    font-size: 18px;\n    letter-spacing: 0.02em;\n    background-size: cover;\n    background-position: center;\n    box-shadow: 0 10px 20px rgba(15, 23, 42, 0.18);\n}\n\n.selfLearningAvatarInitial {\n    line-height: 1;\n}\n\n.selfLearningTeacher {\n    background: linear-gradient(135deg, #fde68a 0%, #86efac 100%);\n}\n\n.selfLearningHeaderText {\n    display: flex;\n    flex-direction: column;\n    gap: 6px;\n    text-align: left;\n}\n\n.selfLearningTitle {\n    margin: 0;\n    font-size: 20px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.selfLearningMetaRow {\n    display: flex;\n    flex-wrap: wrap;\n    align-items: center;\n    gap: 10px;\n    font-size: 13px;\n    color: #64748b;\n    margin-bottom: 12px;\n}\n\n.selfLearningMeta {\n    font-size: 13px;\n    color: #64748b;\n}\n\n.selfLearningMetaChip {\n    background: #e2e8f0;\n    color: #0f172a;\n    border-radius: 999px;\n    padding: 4px 10px;\n    font-size: 12px;\n    font-weight: 600;\n}\n\n.selfLearningCommitments {\n    background: #f8fafc;\n    border: 1px solid #e2e8f0;\n    border-radius: 16px;\n    padding: 16px;\n}\n\n.selfLearningCommitmentsLabel {\n    font-size: 12px;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    color: #94a3b8;\n    font-weight: 600;\n}\n\n.selfLearningBookEditor {\n    margin-top: 12px;\n    overflow: hidden;\n    border-radius: 12px;\n}\n\n.selfLearningEmpty {\n    margin-top: 12px;\n    font-size: 13px;\n    color: #475569;\n}\n\n.modalCloseButton {\n    position: absolute;\n    top: 16px;\n    right: 16px;\n    z-index: 1;\n    width: 36px;\n    height: 36px;\n    border-radius: 999px;\n    border: 1px solid #e2e8f0;\n    background: #fff;\n    color: #475569;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    cursor: pointer;\n    box-shadow: 0 6px 16px rgba(15, 23, 42, 0.08);\n    transition: background 0.2s ease, color 0.2s ease, box-shadow 0.2s ease, transform 0.2s ease;\n}\n\n.modalCloseButton:hover {\n    background: #f8fafc;\n    color: #0f172a;\n    box-shadow: 0 10px 24px rgba(15, 23, 42, 0.12);\n    transform: translateY(-1px);\n}\n\n.modalCloseButton:focus-visible {\n    outline: 2px solid #94a3b8;\n    outline-offset: 2px;\n}\n\n.teamModalHeader {\n    gap: 12px;\n    flex-wrap: wrap;\n}\n\n.teamHeaderParticipants {\n    display: flex;\n    align-items: center;\n    gap: 12px;\n    flex-wrap: wrap;\n}\n\n.teamHeaderProfile {\n    display: inline-flex;\n    align-items: center;\n    gap: 12px;\n    text-decoration: none;\n    color: inherit;\n}\n\n.teamHeaderProfileLink:hover {\n    color: #0f172a;\n}\n\n.teamHeaderAvatar {\n    width: 40px;\n    height: 40px;\n    border-radius: 999px;\n    border: 1px solid #e2e8f0;\n    background-size: cover;\n    background-position: center;\n    background-repeat: no-repeat;\n    box-shadow: 0 6px 14px rgba(15, 23, 42, 0.08);\n}\n\n.teamHeaderName {\n    font-size: 18px;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.teamHeaderDivider {\n    font-size: 14px;\n    font-weight: 600;\n    color: #94a3b8;\n}\n\n.searchModalIcon {\n    font-size: 24px;\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    color: #475569;\n}\n\n.searchModalQuery {\n    margin: 0 !important;\n    font-size: 22px !important;\n    font-weight: 500 !important;\n    color: #202124 !important;\n    text-align: left !important;\n}\n\n.searchModalContent {\n    padding: 24px 32px;\n    background: #fff;\n    max-height: 60vh;\n    overflow-y: auto;\n}\n\n.teamChatContainer {\n    border: 1px solid #e5e7eb;\n    border-radius: 16px;\n    overflow: hidden;\n    background: #f8fafc;\n    margin-bottom: 20px;\n}\n\n.teamToolCallSection {\n    display: flex;\n    flex-direction: column;\n    gap: 16px;\n}\n\n.teamToolCallGroup {\n    display: flex;\n    flex-direction: column;\n    gap: 8px;\n}\n\n.teamToolCallHeading {\n    font-size: 12px;\n    font-weight: 600;\n    text-transform: uppercase;\n    letter-spacing: 0.08em;\n    color: #64748b;\n}\n\n.teamToolCallChips {\n    display: flex;\n    flex-wrap: wrap;\n    gap: 8px;\n}\n\n.teamToolCallDetails {\n    border: 1px solid rgba(148, 163, 184, 0.4);\n    border-radius: 12px;\n    padding: 16px;\n    background: #f8fafc;\n}\n\n.teamToolCallDetailsHeader {\n    display: flex;\n    align-items: center;\n    justify-content: space-between;\n    gap: 12px;\n    margin-bottom: 12px;\n}\n\n.teamToolCallDetailsTitle {\n    font-size: 14px;\n    font-weight: 600;\n    color: #0f172a;\n    display: flex;\n    align-items: center;\n    gap: 6px;\n}\n\n.teamToolCallDetailsClear {\n    background: rgba(148, 163, 184, 0.2);\n    border: 1px solid rgba(148, 163, 184, 0.4);\n    border-radius: 999px;\n    padding: 4px 10px;\n    font-size: 12px;\n    cursor: pointer;\n    color: #475569;\n    transition: all 0.2s ease;\n}\n\n.teamToolCallDetailsClear:hover {\n    background: rgba(148, 163, 184, 0.35);\n}\n\n.searchResultsList {\n    display: flex;\n    flex-direction: column;\n    gap: 28px;\n    text-align: left;\n}\n\n.searchResultsRaw {\n    text-align: left;\n    max-width: 650px;\n}\n\n.searchResultItem {\n    max-width: 650px;\n}\n\n.searchResultUrl {\n    font-size: 14px;\n    color: #202124;\n    margin-bottom: 4px;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n}\n\n.searchResultUrl a {\n    color: #202124;\n    text-decoration: none;\n}\n\n.searchResultTitle {\n    font-size: 20px !important;\n    font-weight: 400 !important;\n    margin: 0 0 4px 0 !important;\n    line-height: 1.3;\n}\n\n.searchResultTitle a {\n    color: #1a0dab;\n    text-decoration: none;\n}\n\n.searchResultTitle a:hover {\n    text-decoration: underline;\n}\n\n.searchResultSnippet {\n    font-size: 14px;\n    line-height: 1.58;\n    color: #4d5156;\n    margin: 0;\n}\n\n.noResults {\n    padding: 40px;\n    text-align: center;\n    color: #70757a;\n    font-style: italic;\n}\n\n.toolCallDetails {\n    text-align: left;\n    margin: 20px 0;\n}\n.citationDetails {\n    text-align: left;\n    margin: 20px 0;\n}\n\n.citationPreview {\n    margin-top: 16px;\n    border: 1px solid #ddd;\n    border-radius: 8px;\n    overflow: hidden;\n    background: #f8f8f8;\n}\n\n.citationIframe {\n    width: 100%;\n    height: 60vh;\n    min-height: 400px;\n    border: none;\n    display: block;\n}\n\n.citationMetadata {\n    background: #f8f8f8;\n    padding: 16px;\n    border-radius: 8px;\n    border: 1px solid #eee;\n    margin-bottom: 20px;\n}\n\n.citationMetadata p {\n    margin: 8px 0;\n    font-size: 0.9em;\n}\n\n.citationMetadata strong {\n    color: #333;\n}\n\n.citationMetadata a {\n    color: #007bff;\n    text-decoration: none;\n    word-break: break-all;\n}\n\n.citationMetadata a:hover {\n    text-decoration: underline;\n}\n\n.citationExcerpt {\n    background: #fff;\n    padding: 16px;\n    border-radius: 8px;\n    border: 1px solid #ddd;\n    max-height: 400px;\n    overflow-y: auto;\n}\n\n.citationExcerpt h4 {\n    margin: 0 0 12px 0;\n    color: #333;\n    font-size: 1em;\n}\n\n.citationHint {\n    font-size: 0.85em;\n    margin-top: 8px;\n    opacity: 0.7;\n}\n\n.toolCallDataContainer {\n    background: #f8f8f8;\n    padding: 12px;\n    border-radius: 6px;\n    border: 1px solid #eee;\n    margin-bottom: 15px;\n    max-height: 300px;\n    overflow-y: auto;\n}\n\n.toolCallData {\n    margin: 0;\n    font-size: 0.85em;\n    white-space: pre-wrap;\n    word-break: break-word;\n}\n\n.toolCallArgsList {\n    list-style: none;\n    padding: 0;\n    margin: 0;\n}\n\n.toolCallArgsList li {\n    margin-bottom: 5px;\n    font-size: 0.9em;\n}\n\n.toolCallModal .toolCallData {\n    background: #f8fafc;\n    color: #0f172a;\n}\n\n.ratingInput {\n    width: 100%;\n    min-height: 100px;\n    padding: 16px;\n    border: 2px solid rgba(125, 125, 125, 0.1);\n    border-radius: 12px;\n    margin-bottom: 18px;\n    resize: vertical;\n    background: rgba(125, 125, 125, 0.05);\n    color: #0b1220;\n    font-size: 14px;\n    line-height: 1.5;\n    transition: all 0.2s ease;\n}\n\n.ratingInput:focus {\n    border-color: #0084ff;\n    background: rgba(125, 125, 125, 0.08);\n    box-shadow: 0 0 0 4px rgba(0, 132, 255, 0.1);\n    outline: none;\n}\n\n.ratingInput[readonly] {\n    border: 1px solid rgba(125, 125, 125, 0.5);\n    background: rgba(125, 125, 125, 0.01);\n}\n\n.ratingActions {\n    display: flex;\n    justify-content: flex-end;\n    gap: 12px;\n}\n\n.ratingActions button {\n    padding: 12px 24px;\n    border: none;\n    border-radius: 8px;\n    cursor: pointer;\n    font-size: 14px;\n    font-weight: 500;\n    transition: all 0.2s ease;\n    min-width: 80px;\n}\n\n.ratingActions button:first-child {\n    background-color: rgba(125, 125, 125, 0.1);\n    color: #0b1220;\n    border: 1px solid rgba(125, 125, 125, 0.2);\n}\n\n.ratingActions button:first-child:hover:not(:disabled) {\n    background-color: rgba(125, 125, 125, 0.2);\n    color: #0b1220;\n}\n\n.ratingActions button:last-child {\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%);\n    color: #ffffff;\n}\n\n.ratingActions button:last-child:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0071d1 0%, #0052a3 100%);\n    transform: translateY(-1px);\n    box-shadow: 0 4px 12px rgba(0, 132, 255, 0.3);\n}\n\n.ratingActions button:last-child:disabled {\n    opacity: 0.5;\n    cursor: not-allowed;\n}\n\n.downloadButton {\n    display: inline-flex;\n    align-items: center;\n    justify-content: center;\n    gap: 8px;\n    padding: 12px 24px;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%);\n    color: #ffffff;\n    border: none;\n    border-radius: 8px;\n    cursor: pointer;\n    font-size: 14px;\n    font-weight: 600;\n    transition: all 0.2s ease;\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.2);\n}\n\n.downloadButton:hover {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px);\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.downloadButton:active {\n    transform: translateY(0);\n    box-shadow: 0 2px 8px rgba(0, 132, 255, 0.2);\n}\n\n.downloadButton svg {\n    flex-shrink: 0;\n}\n\n.downloadButton span {\n    font-weight: 600;\n}\n\n/* New chat button styling - Matches the sleek chat message design */\n.chatButton {\n    display: inline-flex !important;\n    align-items: center !important;\n    justify-content: center !important;\n    gap: 8px !important;\n    padding: 12px 16px !important;\n    margin: 0 !important;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%) !important;\n    color: #ffffff !important;\n    border: none !important;\n    border-radius: 20px !important;\n    font-size: 13px !important;\n    font-weight: 600 !important;\n    line-height: 1.3 !important;\n    cursor: pointer !important;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94) !important;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3) !important;\n    backdrop-filter: blur(20px) !important;\n    -webkit-tap-highlight-color: transparent !important;\n    touch-action: manipulation !important;\n    user-select: none !important;\n    min-height: 40px !important;\n    min-width: 110px !important;\n    position: relative !important;\n    overflow: hidden !important;\n}\n\n.chatButton::before {\n    content: '';\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 100%;\n    height: 100%;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);\n    transition: left 0.5s ease;\n}\n\n.chatButton:hover::before {\n    left: 100%;\n}\n\n.chatButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(0, 132, 255, 0.4);\n}\n\n.chatButton:active {\n    transform: scale(0.98) translateY(-1px);\n    transition: transform 0.1s ease;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.chatButton:focus {\n    outline: none;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3), 0 0 0 3px rgba(0, 132, 255, 0.3);\n}\n\n.chatButton svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    opacity: 1;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.2));\n}\n\n.chatButton:hover svg {\n    transform: rotate(-90deg) scale(1.1);\n    filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.3));\n}\n\n.chatButtonText {\n    white-space: nowrap;\n    font-weight: 600;\n    transition: all 0.2s ease;\n    text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);\n    opacity: 1;\n}\n\n.chatButton:hover .chatButtonText {\n    transform: translateX(1px);\n}\n\n/* Use template button styling - matches the chatButton and saveButton */\n.useTemplateButton {\n    display: inline-flex !important;\n    align-items: center !important;\n    justify-content: center !important;\n    gap: 8px !important;\n    padding: 12px 16px !important;\n    margin: 0 !important;\n    background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%) !important;\n    color: #ffffff !important;\n    border: none !important;\n    border-radius: 20px !important;\n    font-size: 13px !important;\n    font-weight: 600 !important;\n    line-height: 1.3 !important;\n    cursor: pointer !important;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94) !important;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3) !important;\n    backdrop-filter: blur(20px) !important;\n    -webkit-tap-highlight-color: transparent !important;\n    touch-action: manipulation !important;\n    user-select: none !important;\n    min-height: 40px !important;\n    min-width: 110px !important;\n    position: relative !important;\n    overflow: hidden !important;\n}\n\n.useTemplateButton::before {\n    content: '';\n    position: absolute;\n    top: 0;\n    left: -100%;\n    width: 100%;\n    height: 100%;\n    background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);\n    transition: left 0.5s ease;\n}\n\n.useTemplateButton:hover::before {\n    left: 100%;\n}\n\n.useTemplateButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #0099ff 0%, #0077dd 100%);\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(0, 132, 255, 0.4);\n}\n\n.useTemplateButton:active {\n    transform: scale(0.98) translateY(-1px);\n    transition: transform 0.1s ease;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n}\n\n.useTemplateButton:focus {\n    outline: none;\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3), 0 0 0 3px rgba(0, 132, 255, 0.3);\n}\n\n.useTemplateButton svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    opacity: 1;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.2));\n}\n\n.useTemplateButton:hover svg {\n    transform: scale(1.1);\n    filter: drop-shadow(0 2px 4px rgba(0, 0, 0, 0.3));\n}\n\n.useTemplateButton:hover .chatButtonText {\n    transform: translateX(1px);\n}\n\n.saveButtonContainer {\n    display: inline-block;\n    position: relative;\n}\n\n.saveMenu {\n    position: absolute;\n    top: 100%;\n    left: 0;\n    background: #fff;\n    border: 1px solid #ddd;\n    z-index: 10;\n    min-width: 120px;\n    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);\n}\n\n.saveMenuItem {\n    display: block;\n    width: 100%;\n    padding: 8px 16px;\n    border: none;\n    background: none;\n    text-align: left;\n    cursor: pointer;\n    color: #111;\n}\n\n.saveMenuItem:hover {\n    background-color: #f0f0f0;\n}\n\n.saveMenuDivider {\n    height: 1px;\n    background-color: #ddd;\n    margin: 4px 0;\n}\n\n.chatFeedbackPanel {\n    width: min(260px, 100%);\n    padding: 6px 8px;\n    border-radius: 14px;\n    border: 1px solid #e5e7eb;\n    background: #ffffff;\n    box-shadow: 0 8px 18px rgba(15, 23, 42, 0.08);\n    display: flex;\n    flex-direction: column;\n    gap: 6px;\n}\n\n.chatFeedbackToggle {\n    display: flex;\n    align-items: center;\n    gap: 10px;\n    width: 100%;\n    padding: 6px 8px;\n    border-radius: 12px;\n    border: none;\n    background: transparent;\n    cursor: pointer;\n    transition: background 0.2s ease, transform 0.2s ease;\n    text-align: left;\n}\n\n.chatFeedbackToggle:hover {\n    background: #f5f5f5;\n    transform: translateY(-0.5px);\n}\n\n.chatFeedbackToggle:focus-visible {\n    outline: 2px solid #2563eb;\n    outline-offset: 2px;\n}\n\n.chatFeedbackIcon {\n    width: 34px;\n    height: 34px;\n    border-radius: 10px;\n    background: #eef2ff;\n    color: #6366f1;\n    display: grid;\n    place-items: center;\n    flex-shrink: 0;\n    transition: background 0.2s ease, color 0.2s ease;\n}\n\n.chatFeedbackToggle[data-enabled='true'] .chatFeedbackIcon {\n    background: #e0f2fe;\n    color: #0284c7;\n}\n\n.chatFeedbackToggle[data-enabled='false'] .chatFeedbackIcon {\n    background: #f3f4f6;\n    color: #94a3b8;\n}\n\n.chatFeedbackMeta {\n    flex: 1;\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n}\n\n.chatFeedbackLabel {\n    font-size: 0.85rem;\n    font-weight: 600;\n    color: #0f172a;\n}\n\n.chatFeedbackDescription {\n    font-size: 0.75rem;\n    color: #64748b;\n}\n\n.chatFeedbackIndicator {\n    font-size: 0.72rem;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    font-weight: 700;\n    padding: 2px 8px;\n    border-radius: 999px;\n}\n\n.chatFeedbackIndicatorActive {\n    color: #166534;\n    background: #dcfce7;\n    border: 1px solid #34d399;\n}\n\n.chatFeedbackIndicatorInactive {\n    color: #713f12;\n    background: #fef9c3;\n    border: 1px solid #facc15;\n}\n\n/* Pause/Resume button variant (reuses .chatButton base styles for DRY) */\n.pauseButton {\n    background: linear-gradient(135deg, #ffb347 0%, #ff8c42 100%) !important; /* Warm orange for \"active/running\" */\n}\n\n.pauseButton:hover:not(:disabled) {\n    background: linear-gradient(135deg, #ffc067 0%, #ff9e5f 100%) !important;\n}\n\n.pauseButton.pausing {\n    opacity: 0.6 !important;\n    cursor: wait !important;\n}\n\n.pauseButton.paused {\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%) !important; /* Green when paused (ready to resume) */\n}\n\n.pauseButton.paused:hover:not(:disabled) {\n    background: linear-gradient(135deg, #34d399 0%, #059669 100%) !important;\n    transform: translateY(-2px) scale(1.02);\n    box-shadow: 0 8px 24px rgba(16, 185, 129, 0.35);\n}\n\n.pauseButton svg {\n    transition: transform 0.3s ease;\n}\n\n.pauseButton.paused svg {\n    transform: scale(1.1);\n}\n\n.pauseButton.pausing svg {\n    opacity: 0.8;\n}\n\n/* Voice call indicator bar */\n.voiceCallIndicatorBar {\n    grid-area: 🟦;\n    width: 100%;\n    padding: 12px 20px;\n    background: linear-gradient(135deg, rgba(34, 197, 94, 0.1) 0%, rgba(16, 185, 129, 0.1) 100%);\n    border-bottom: 1px solid rgba(34, 197, 94, 0.3);\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    backdrop-filter: blur(10px);\n}\n\n/* Voice call indicator */\n.voiceCallIndicator {\n    display: inline-flex;\n    align-items: center;\n    gap: 8px;\n    padding: 8px 16px;\n    background: linear-gradient(135deg, rgba(34, 197, 94, 0.2) 0%, rgba(16, 185, 129, 0.2) 100%);\n    border: 1px solid rgba(34, 197, 94, 0.4);\n    border-radius: 20px;\n    color: #10b981;\n    font-size: 13px;\n    font-weight: 600;\n    backdrop-filter: blur(10px);\n    box-shadow: 0 2px 8px rgba(34, 197, 94, 0.2);\n    position: relative;\n}\n\n.voiceCallIndicator svg {\n    width: 16px;\n    height: 16px;\n    flex-shrink: 0;\n    animation: voiceCallIconPulse 2s ease-in-out infinite;\n}\n\n.voiceCallIndicator span {\n    font-weight: 600;\n    text-shadow: 0 1px 2px rgba(0, 0, 0, 0.1);\n}\n\n/* Voice call pulse animation */\n.voiceCallPulse {\n    position: absolute;\n    right: 8px;\n    top: 50%;\n    transform: translateY(-50%);\n    width: 8px;\n    height: 8px;\n    background: #10b981;\n    border-radius: 50%;\n    animation: voiceCallPulse 1.5s ease-in-out infinite;\n}\n\n@keyframes voiceCallIconPulse {\n    0%,\n    100% {\n        transform: scale(1);\n        opacity: 1;\n    }\n    50% {\n        transform: scale(1.1);\n        opacity: 0.8;\n    }\n}\n\n@keyframes voiceCallPulse {\n    0%,\n    100% {\n        transform: translateY(-50%) scale(1);\n        opacity: 1;\n    }\n    50% {\n        transform: translateY(-50%) scale(1.3);\n        opacity: 0.6;\n    }\n}\n\n/* Voice call indicator in messages */\n.chatMessage .voiceCallIndicator {\n    margin-bottom: 8px;\n    padding: 6px 12px;\n    font-size: 12px;\n    border-radius: 16px;\n}\n\n.chatMessage .voiceCallIndicator svg {\n    width: 14px;\n    height: 14px;\n}\n\n/* Feedback status banner */\n.feedbackStatus {\n    position: fixed;\n    top: 20px;\n    right: 20px;\n    padding: 14px 22px;\n    border-radius: 14px;\n    box-shadow: 0 18px 32px rgba(15, 23, 42, 0.35);\n    z-index: 10000; /* [Fix z-index]: Ensure it's above the header (z-50) and other UI elements */\n    animation: feedbackStatusSlideIn 0.28s ease-out;\n    max-width: 340px;\n    word-wrap: break-word;\n    font-weight: 500;\n    backdrop-filter: blur(20px);\n    pointer-events: none;\n}\n\n.feedbackStatusSuccess {\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%);\n    color: #f8fafc;\n}\n\n.feedbackStatusError {\n    background: linear-gradient(135deg, #ef4444 0%, #dc2626 100%);\n    color: #fef2f2;\n}\n\n@keyframes feedbackStatusSlideIn {\n    from {\n        transform: translateX(100%);\n        opacity: 0;\n    }\n\n    to {\n        transform: translateX(0);\n        opacity: 1;\n    }\n}\n\n/* Mobile responsiveness */\n@media (max-width: 768px) {\n    .Chat {\n        --chat-message-min-width: 100px;\n        --chat-message-max-width: min(85%, 560px);\n        --chat-message-avatar-gap: 10px;\n    }\n\n    .actions {\n        margin: 12px 16px 0;\n        gap: 6px;\n    }\n\n    .chatMainFlow .chatMessages {\n        padding: 16px 12px;\n    }\n\n    .chatMainFlow .chatMessage {\n        margin-bottom: 16px;\n    }\n\n    .chatMainFlow .chatMessage .messageText {\n        padding: 12px 16px;\n        font-size: 14px;\n        border-radius: 18px;\n    }\n\n    .chatMainFlow .chatMessage .avatar {\n        width: 36px;\n        height: 36px;\n        margin: 0 0 4px;\n    }\n\n    /* [㊗️]\n    .chatMainFlow .chatMessage .avatar img {\n        width: 36px;\n        aspect-ratio: 1 / 1;\n    }\n    */\n\n    .chatMainFlow .chatInput {\n        padding: 16px 12px;\n        gap: 10px;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        font-size: 16px;\n        padding: 14px 18px;\n        border-radius: 22px;\n    }\n\n    .chatMainFlow .chatInput button {\n        width: 44px;\n        height: 44px;\n    }\n\n    .scrollToBottom {\n        width: 44px;\n        height: 44px;\n        font-size: 18px;\n        top: calc(100% - 160px);\n    }\n\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n\n    .ratingModalContent {\n        margin: 16px;\n        padding: 24px 20px;\n        border-radius: 16px;\n        max-height: 80vh;\n        overflow-y: auto;\n    }\n\n    .stars {\n        gap: 6px;\n        margin-bottom: 20px;\n    }\n\n    .stars span {\n        font-size: 32px;\n        padding: 8px;\n    }\n\n    .ratingActions {\n        flex-direction: column-reverse;\n        gap: 8px;\n    }\n\n    .ratingActions button {\n        width: 100%;\n        padding: 14px;\n        font-size: 16px;\n        border-radius: 10px;\n    }\n\n    .selfLearningModalHeader {\n        flex-direction: column;\n        align-items: flex-start;\n    }\n\n    .selfLearningAvatarGroup {\n        justify-content: flex-start;\n    }\n\n    .selfLearningMetaRow {\n        align-items: flex-start;\n    }\n}\n\n@media (max-width: 480px) {\n    .Chat {\n        --chat-message-min-width: 80px;\n        --chat-message-max-width: min(90%, 520px);\n        --chat-message-avatar-gap: 8px;\n    }\n\n    .actions {\n        margin: 8px 12px 0;\n        gap: 4px;\n    }\n\n    .chatMainFlow .chatMessages {\n        padding: 12px 8px;\n    }\n\n    .chatMainFlow .chatMessage .messageText {\n        padding: 10px 14px;\n        font-size: 14px;\n        border-radius: 16px;\n    }\n\n    .chatMainFlow .chatMessage .avatar {\n        width: 32px;\n        height: 32px;\n        margin: 0 0 4px;\n    }\n\n    /* [㊗️]\n    .chatMainFlow .chatMessage .avatar img {\n        width: 32px;\n        aspect-ratio: 1 / 1;\n    }\n    */\n\n    .chatMainFlow .chatInput {\n        padding: 12px 8px;\n        gap: 8px;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        padding: 12px 16px;\n        border-radius: 20px;\n        font-size: 16px;\n    }\n\n    .chatMainFlow .chatInput button {\n        width: 40px;\n        height: 40px;\n    }\n\n    .chatButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .chatButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .chatButtonText {\n        display: none !important;\n    }\n\n    .useTemplateButton {\n        border-radius: 50% !important;\n        width: 40px !important;\n        height: 40px !important;\n        min-width: 40px !important;\n        min-height: 40px !important;\n        padding: 0 !important;\n        gap: 0 !important;\n        margin: 0 !important;\n    }\n\n    .useTemplateButton svg {\n        width: 18px !important;\n        height: 18px !important;\n    }\n\n    .useTemplateButton .chatButtonText {\n        display: none !important;\n    }\n\n    .scrollToBottom {\n        width: 40px;\n        height: 40px;\n        font-size: 16px;\n        top: calc(100% - 140px);\n    }\n\n    .ratingModal {\n        padding: 0;\n        align-items: flex-end;\n    }\n\n    .ratingModalContent {\n        margin: 0;\n        width: 100%;\n        border-radius: 20px 20px 0 0;\n        max-height: 70vh;\n        padding: 24px 16px 20px;\n    }\n}\n\n/* Reduced motion support */\n@media (prefers-reduced-motion: reduce) {\n    .chatMainFlow .chatMessage,\n    .scrollToBottom,\n    .ratingModal,\n    .ratingModalContent,\n    .feedbackStatus {\n        animation: none;\n    }\n\n    .chatMainFlow .chatMessage .messageText,\n    .chatMainFlow .chatInput textarea,\n    .chatMainFlow .chatInput button,\n    /* [㊗️] .chatMainFlow .chatMessage .avatar img, */\n    .chatButton {\n        transition: none;\n    }\n}\n\n/* High contrast mode support */\n@media (prefers-contrast: high) {\n    .chatMainFlow .chatMessage .messageText {\n        border: 2px solid currentColor;\n    }\n\n    .chatMainFlow .chatInput textarea {\n        border-width: 3px;\n    }\n\n    .chatMainFlow .chatInput button {\n        border: 2px solid currentColor;\n    }\n}\n\n/* Email modal styles */\n.emailContainer {\n    margin-bottom: 20px;\n    display: flex;\n    flex-direction: column;\n    gap: 16px;\n}\n\n.emailModalHeader {\n    background: #f8fafc;\n}\n\n.emailHeaderText {\n    display: flex;\n    flex-direction: column;\n    gap: 2px;\n}\n\n.emailHeaderLabel {\n    font-size: 12px;\n    letter-spacing: 0.08em;\n    text-transform: uppercase;\n    color: #64748b;\n}\n\n.emailMetadata {\n    background: #ffffff;\n    border: 1px solid #e2e8f0;\n    border-radius: 12px;\n    padding: 18px 20px;\n    margin-bottom: 16px;\n    box-shadow: 0 1px 4px rgba(15, 23, 42, 0.08);\n}\n\n.emailField {\n    display: flex;\n    gap: 8px;\n    margin-bottom: 8px;\n    font-size: 14px;\n    line-height: 1.5;\n}\n\n.emailField:last-child {\n    margin-bottom: 0;\n}\n\n.emailField strong {\n    min-width: 60px;\n    color: #64748b;\n    font-weight: 600;\n}\n\n.emailRecipients {\n    color: #0f172a;\n    word-break: break-word;\n}\n\n.emailBody {\n    text-align: left;\n}\n\n.emailBody strong {\n    display: block;\n    margin-bottom: 12px;\n    color: #64748b;\n    font-weight: 600;\n    font-size: 14px;\n}\n\n.emailBodyContent {\n    background: #ffffff;\n    border: 1px solid #e2e8f0;\n    border-radius: 12px;\n    padding: 22px;\n    line-height: 1.6;\n    max-height: 400px;\n    overflow-y: auto;\n    box-shadow: 0 1px 4px rgba(15, 23, 42, 0.08);\n}\n\n.emailBodyContent p {\n    margin: 0 0 12px 0;\n}\n\n.emailBodyContent p:last-child {\n    margin-bottom: 0;\n}\n\n.emailStatus {\n    text-transform: capitalize;\n    color: #0f172a;\n    font-weight: 600;\n}\n\n/**\n * TODO: [🌉] DRY Markdown primitives styling\n */\n/* Image prompt preview styles */\n.imagePrompt {\n    margin-top: 16px;\n    margin-bottom: 16px;\n    padding: 12px;\n    border-radius: 18px;\n    border: 1px solid rgba(125, 125, 125, 0.35);\n    background: rgba(255, 255, 255, 0.25);\n    backdrop-filter: blur(12px);\n    box-shadow: inset 0 0 0 1px rgba(255, 255, 255, 0.25);\n    display: flex;\n    flex-direction: column;\n    gap: 8px;\n}\n\n.imagePromptPreview {\n    position: relative;\n    border-radius: 14px;\n    overflow: hidden;\n    min-height: 160px;\n    border: 1px dashed rgba(125, 125, 125, 0.3);\n    background: linear-gradient(180deg, rgba(15, 118, 255, 0.05), rgba(255, 255, 255, 0.4));\n}\n\n.imagePromptImage {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n    display: block;\n}\n\n.imagePromptPlaceholder {\n    width: 100%;\n    height: 100%;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n}\n\n.imagePromptSpinner {\n    width: 36px;\n    height: 36px;\n    border-radius: 50%;\n    border: 3px solid rgba(255, 255, 255, 0.35);\n    border-top-color: #2563eb;\n    animation: imagePromptSpinner 0.9s linear infinite;\n}\n\n@keyframes imagePromptSpinner {\n    to {\n        transform: rotate(360deg);\n    }\n}\n\n.imagePromptMeta {\n    display: flex;\n    flex-direction: column;\n    gap: 4px;\n}\n\n.imagePromptPrompt {\n    font-size: 13px;\n    font-weight: 500;\n    color: #111;\n    overflow-wrap: anywhere;\n}\n\n.imagePromptStatus {\n    font-size: 11px;\n    color: #6b7280;\n}\n\n.imagePromptError {\n    position: absolute;\n    inset: 0;\n    display: flex;\n    align-items: center;\n    justify-content: center;\n    padding: 12px;\n    text-align: center;\n    font-size: 12px;\n    color: #dc2626;\n    background: rgba(220, 38, 38, 0.08);\n}\n"]} */";
|
|
16153
|
+
var styles$4 = {"copiedToClipboardMessage":"Chat-module_copiedToClipboardMessage__apCPY","Chat":"Chat-module_Chat__j2eE5","chatMainFlow":"Chat-module_chatMainFlow__--8FE","chatBar":"Chat-module_chatBar__fLECN","TasksInProgress":"Chat-module_TasksInProgress__fQfei","actions":"Chat-module_actions__gTZ5T","portal":"Chat-module_portal__uTOT8","left":"Chat-module_left__7l5Mn","right":"Chat-module_right__ABZrW","actionsFaded":"Chat-module_actionsFaded__qqAVQ","actionsScrolling":"Chat-module_actionsScrolling__fIawW","chatButton":"Chat-module_chatButton__d9VgA","chatButtonText":"Chat-module_chatButtonText__RkGB-","useTemplateButton":"Chat-module_useTemplateButton__xcJNR","chatChildren":"Chat-module_chatChildren__flOPK","chatMessages":"Chat-module_chatMessages__J2u2N","hasActionsAndFirstMessageIsLong":"Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ","chatMessage":"Chat-module_chatMessage__nmLaZ","messageSlideIn":"Chat-module_messageSlideIn__soTy2","messageStack":"Chat-module_messageStack__n0xNL","isMe":"Chat-module_isMe__nBtaV","messageMeta":"Chat-module_messageMeta__eqCRu","messageTimestamp":"Chat-module_messageTimestamp__LDhRw","messageDuration":"Chat-module_messageDuration__Y9Kg9","participantLabel":"Chat-module_participantLabel__0Oo-3","isNotCompleteMessage":"Chat-module_isNotCompleteMessage__Hj2K7","NonCompleteMessageFiller":"Chat-module_NonCompleteMessageFiller__G5-Ve","messageText":"Chat-module_messageText__XgNyQ","loadingPulse":"Chat-module_loadingPulse__VomRm","ongoingToolCalls":"Chat-module_ongoingToolCalls__NZkQN","completedToolCalls":"Chat-module_completedToolCalls__vI1Qt","sourceCitations":"Chat-module_sourceCitations__oDtBX","ongoingToolCall":"Chat-module_ongoingToolCall__WT3Rc","completedToolCall":"Chat-module_completedToolCall__-q4Cs","toolCallOrigin":"Chat-module_toolCallOrigin__-Io6Y","toolCallDetails":"Chat-module_toolCallDetails__WUFlD","toolCallData":"Chat-module_toolCallData__UauCO","ongoingToolCallSpinner":"Chat-module_ongoingToolCallSpinner__7g-Ay","toolCallSpinner":"Chat-module_toolCallSpinner__LSiK6","ongoingToolCallName":"Chat-module_ongoingToolCallName__y59-0","typingIndicator":"Chat-module_typingIndicator__S-CT-","avatar":"Chat-module_avatar__gL6bm","typingBubble":"Chat-module_typingBubble__0Lb7B","typingDots":"Chat-module_typingDots__srOBB","typingDot":"Chat-module_typingDot__dnhKT","typingBounce":"Chat-module_typingBounce__1yp2v","ratingStar":"Chat-module_ratingStar__rRfqC","active":"Chat-module_active__lbYL-","copyButtonContainer":"Chat-module_copyButtonContainer__Rij0U","messageControlGroup":"Chat-module_messageControlGroup__gAo4k","copyButton":"Chat-module_copyButton__DcxT5","playButton":"Chat-module_playButton__ZigYL","playButtonIcon":"Chat-module_playButtonIcon__wPErq","playButtonSpinner":"Chat-module_playButtonSpinner__Ad33i","messagePlaySpinner":"Chat-module_messagePlaySpinner__ldlvy","copiedTooltip":"Chat-module_copiedTooltip__LH81j","copiedTooltipFadeIn":"Chat-module_copiedTooltipFadeIn__QekO1","copiedTooltipLeft":"Chat-module_copiedTooltipLeft__j-S-5","copiedTooltipRight":"Chat-module_copiedTooltipRight__R-2cE","chat-code-block":"Chat-module_chat-code-block__k8IyS","attachments":"Chat-module_attachments__m1Fts","attachment":"Chat-module_attachment__aE9hK","attachmentIcon":"Chat-module_attachmentIcon__BX3Cy","attachmentName":"Chat-module_attachmentName__aMx56","messageButtons":"Chat-module_messageButtons__WaOob","messageButton":"Chat-module_messageButton__mRnn-","rating":"Chat-module_rating__soc3M","chatInput":"Chat-module_chatInput__1Ecan","fullPageVisual":"Chat-module_fullPageVisual__zNAEy","dragOver":"Chat-module_dragOver__bkS-g","filePreviewContainer":"Chat-module_filePreviewContainer__R70hm","filePreview":"Chat-module_filePreview__kq2aX","fileIcon":"Chat-module_fileIcon__zoSKW","fileInfo":"Chat-module_fileInfo__wBLi0","fileName":"Chat-module_fileName__bBujo","fileSize":"Chat-module_fileSize__ivliq","removeFileButton":"Chat-module_removeFileButton__0gakR","inputContainer":"Chat-module_inputContainer__bPt99","attachmentButton":"Chat-module_attachmentButton__qLO47","voiceButton":"Chat-module_voiceButton__d2zlP","voiceButtonActive":"Chat-module_voiceButtonActive__Uoi3W","voiceRecordingPulse":"Chat-module_voiceRecordingPulse__y2wJ5","uploadProgress":"Chat-module_uploadProgress__jBTKe","uploadProgressBar":"Chat-module_uploadProgressBar__Gutnt","uploadProgressFill":"Chat-module_uploadProgressFill__EgubT","dragOverlay":"Chat-module_dragOverlay__SEGoS","dragOverlayContent":"Chat-module_dragOverlayContent__gb9kF","scrollToBottomContainer":"Chat-module_scrollToBottomContainer__5rXpK","scrollToBottomWrapper":"Chat-module_scrollToBottomWrapper__PCIu1","scrollButtonSlideIn":"Chat-module_scrollButtonSlideIn__XnImg","scrollToBottom":"Chat-module_scrollToBottom__nzxdZ","scrollToBottomBadge":"Chat-module_scrollToBottomBadge__N6T6-","ratingModal":"Chat-module_ratingModal__XVKYm","modalFadeIn":"Chat-module_modalFadeIn__RPc3w","ratingModalContent":"Chat-module_ratingModalContent__CCdq7","modalSlideIn":"Chat-module_modalSlideIn__XXtgN","stars":"Chat-module_stars__PCzNO","ratingModalStar":"Chat-module_ratingModalStar__XkbHr","toolCallModal":"Chat-module_toolCallModal__uNaIK","toolCallHeader":"Chat-module_toolCallHeader__ZaTZF","toolCallIcon":"Chat-module_toolCallIcon__UgRKj","toolCallHeaderMeta":"Chat-module_toolCallHeaderMeta__iAL-U","toolCallLabel":"Chat-module_toolCallLabel__q7EFf","toolCallTitle":"Chat-module_toolCallTitle__jhL0A","toolCallSubtitle":"Chat-module_toolCallSubtitle__a2GIO","toolCallGrid":"Chat-module_toolCallGrid__Nfr7N","toolCallPanel":"Chat-module_toolCallPanel__kQZE9","toolCallPanelTitle":"Chat-module_toolCallPanelTitle__-7OtJ","toolCallList":"Chat-module_toolCallList__5co3s","toolCallItem":"Chat-module_toolCallItem__k7W5Z","toolCallItemLabel":"Chat-module_toolCallItemLabel__d3C1E","toolCallItemValue":"Chat-module_toolCallItemValue__cStQ0","toolCallSummary":"Chat-module_toolCallSummary__wGZwg","toolCallSummaryMeta":"Chat-module_toolCallSummaryMeta__RXiSv","toolCallSummaryMetaBadge":"Chat-module_toolCallSummaryMetaBadge__JBi-I","toolCallEmpty":"Chat-module_toolCallEmpty__bUz5v","toolCallIssues":"Chat-module_toolCallIssues__evhYM","toolCallIssueBadge":"Chat-module_toolCallIssueBadge__Qbwdp","toolCallIssueError":"Chat-module_toolCallIssueError__LImeS","toolCallIssueWarning":"Chat-module_toolCallIssueWarning__chQDf","toolCallFooter":"Chat-module_toolCallFooter__lJ24N","toolCallTimestamp":"Chat-module_toolCallTimestamp__-iWsu","toolCallRawDetails":"Chat-module_toolCallRawDetails__CvFfs","toolCallRawSummary":"Chat-module_toolCallRawSummary__-aLvG","toolCallRawColumns":"Chat-module_toolCallRawColumns__8ummz","toolCallRawColumnTitle":"Chat-module_toolCallRawColumnTitle__JQda7","memoryModalHeader":"Chat-module_memoryModalHeader__SpZJT","memoryModalIcon":"Chat-module_memoryModalIcon__eYbli","memoryModalHeaderText":"Chat-module_memoryModalHeaderText__S6MZG","memoryModalTitle":"Chat-module_memoryModalTitle__zzbaQ","memoryModalSubtitle":"Chat-module_memoryModalSubtitle__PStci","memoryModalCallTime":"Chat-module_memoryModalCallTime__RmVI9","memoryModalStatus":"Chat-module_memoryModalStatus__bmrfc","memoryStatusDot":"Chat-module_memoryStatusDot__c0oVs","memoryModalContent":"Chat-module_memoryModalContent__8E7oj","memoryMessage":"Chat-module_memoryMessage__q0pNK","memoryMetaRow":"Chat-module_memoryMetaRow__Hc-Xp","memoryMetaLabel":"Chat-module_memoryMetaLabel__LFx11","memoryMetaValue":"Chat-module_memoryMetaValue__4bILO","memoryStoreSection":"Chat-module_memoryStoreSection__qQJPN","memoryRetrieveSection":"Chat-module_memoryRetrieveSection__VbuS4","memoryCard":"Chat-module_memoryCard__-kIDN","memoryCardContent":"Chat-module_memoryCardContent__l9S4G","memoryCardMeta":"Chat-module_memoryCardMeta__K66KE","memoryScopeBadge":"Chat-module_memoryScopeBadge__T2ACc","memoryList":"Chat-module_memoryList__pUOg2","memoryEmptyState":"Chat-module_memoryEmptyState__fPrXA","memoryListFooter":"Chat-module_memoryListFooter__Y0Q2Z","memoryStatusSuccess":"Chat-module_memoryStatusSuccess__4bhfo","memoryStatusWarning":"Chat-module_memoryStatusWarning__AR6yh","memoryStatusError":"Chat-module_memoryStatusError__ZNAT6","memoryStatusNeutral":"Chat-module_memoryStatusNeutral__Fa-4M","searchModalHeader":"Chat-module_searchModalHeader__Y8V-w","selfLearningModalHeader":"Chat-module_selfLearningModalHeader__PDIt1","selfLearningAvatarGroup":"Chat-module_selfLearningAvatarGroup__jgy2k","selfLearningAvatar":"Chat-module_selfLearningAvatar__NHXKm","selfLearningAvatarInitial":"Chat-module_selfLearningAvatarInitial__cuw19","selfLearningTeacher":"Chat-module_selfLearningTeacher__fGhX8","selfLearningHeaderText":"Chat-module_selfLearningHeaderText__LTwHL","selfLearningTitle":"Chat-module_selfLearningTitle__G0SAk","selfLearningMetaRow":"Chat-module_selfLearningMetaRow__z4Tg0","selfLearningMeta":"Chat-module_selfLearningMeta__58Pet","selfLearningMetaChip":"Chat-module_selfLearningMetaChip__0go88","selfLearningCommitments":"Chat-module_selfLearningCommitments__7ECnr","selfLearningCommitmentsLabel":"Chat-module_selfLearningCommitmentsLabel__OkDO1","selfLearningBookEditor":"Chat-module_selfLearningBookEditor__CZy5T","selfLearningEmpty":"Chat-module_selfLearningEmpty__vCKHL","modalCloseButton":"Chat-module_modalCloseButton__AAfWS","teamModalHeader":"Chat-module_teamModalHeader__SB5NL","teamHeaderParticipants":"Chat-module_teamHeaderParticipants__KS7aV","teamHeaderProfile":"Chat-module_teamHeaderProfile__xRlcS","teamHeaderProfileLink":"Chat-module_teamHeaderProfileLink__fYTfT","teamHeaderAvatar":"Chat-module_teamHeaderAvatar__vXcTp","teamHeaderName":"Chat-module_teamHeaderName__ygJR9","teamHeaderDivider":"Chat-module_teamHeaderDivider__cnRMd","searchModalIcon":"Chat-module_searchModalIcon__AR5KY","searchModalQuery":"Chat-module_searchModalQuery__5x-Ra","searchModalContent":"Chat-module_searchModalContent__mWLIE","teamChatContainer":"Chat-module_teamChatContainer__xdLtU","teamToolCallSection":"Chat-module_teamToolCallSection__FH9l7","teamToolCallGroup":"Chat-module_teamToolCallGroup__PO6rl","teamToolCallHeading":"Chat-module_teamToolCallHeading__G-JSj","teamToolCallChips":"Chat-module_teamToolCallChips__hRWqh","teamToolCallDetails":"Chat-module_teamToolCallDetails__ZaTtt","teamToolCallDetailsHeader":"Chat-module_teamToolCallDetailsHeader__FPIgI","teamToolCallDetailsTitle":"Chat-module_teamToolCallDetailsTitle__50SeL","teamToolCallDetailsClear":"Chat-module_teamToolCallDetailsClear__8yvu7","searchResultsList":"Chat-module_searchResultsList__xVDUZ","searchResultsRaw":"Chat-module_searchResultsRaw__i3GkD","searchResultItem":"Chat-module_searchResultItem__bTzq5","searchResultUrl":"Chat-module_searchResultUrl__SbrH4","searchResultTitle":"Chat-module_searchResultTitle__XS7zN","searchResultSnippet":"Chat-module_searchResultSnippet__Wsun2","noResults":"Chat-module_noResults__AAv6s","citationDetails":"Chat-module_citationDetails__2v2e1","citationPreview":"Chat-module_citationPreview__rGhCR","citationIframe":"Chat-module_citationIframe__ngzFY","citationMetadata":"Chat-module_citationMetadata__DPaCF","citationExcerpt":"Chat-module_citationExcerpt__VU4BS","citationHint":"Chat-module_citationHint__5AkLx","toolCallDataContainer":"Chat-module_toolCallDataContainer__rdEzC","toolCallArgsList":"Chat-module_toolCallArgsList__LD3xH","ratingInput":"Chat-module_ratingInput__z8Pv-","ratingActions":"Chat-module_ratingActions__nXcss","downloadButton":"Chat-module_downloadButton__8--bk","saveButtonContainer":"Chat-module_saveButtonContainer__lSNUJ","saveMenu":"Chat-module_saveMenu__-ph8y","saveMenuItem":"Chat-module_saveMenuItem__ISApL","saveMenuDivider":"Chat-module_saveMenuDivider__3ktTM","chatFeedbackPanel":"Chat-module_chatFeedbackPanel__gfOzk","chatFeedbackToggle":"Chat-module_chatFeedbackToggle__IeOM6","chatFeedbackIcon":"Chat-module_chatFeedbackIcon__XcB7A","chatFeedbackMeta":"Chat-module_chatFeedbackMeta__6t2be","chatFeedbackLabel":"Chat-module_chatFeedbackLabel__qQgH3","chatFeedbackDescription":"Chat-module_chatFeedbackDescription__I9bNg","chatFeedbackIndicator":"Chat-module_chatFeedbackIndicator__uHCr7","chatFeedbackIndicatorActive":"Chat-module_chatFeedbackIndicatorActive__bkw-w","chatFeedbackIndicatorInactive":"Chat-module_chatFeedbackIndicatorInactive__qMBkh","pauseButton":"Chat-module_pauseButton__eeu7K","pausing":"Chat-module_pausing__pTx8b","paused":"Chat-module_paused__j-pya","voiceCallIndicatorBar":"Chat-module_voiceCallIndicatorBar__N2sWN","voiceCallIndicator":"Chat-module_voiceCallIndicator__tsaaG","voiceCallIconPulse":"Chat-module_voiceCallIconPulse__zZbJn","voiceCallPulse":"Chat-module_voiceCallPulse__XcGU4","feedbackStatus":"Chat-module_feedbackStatus__-df7l","feedbackStatusSlideIn":"Chat-module_feedbackStatusSlideIn__n6bbB","feedbackStatusSuccess":"Chat-module_feedbackStatusSuccess__Ax2q0","feedbackStatusError":"Chat-module_feedbackStatusError__dnv6d","emailContainer":"Chat-module_emailContainer__TBxXZ","emailModalHeader":"Chat-module_emailModalHeader__Z4-CJ","emailHeaderText":"Chat-module_emailHeaderText__1dR6R","emailHeaderLabel":"Chat-module_emailHeaderLabel__7iAEH","emailMetadata":"Chat-module_emailMetadata__8XXGf","emailField":"Chat-module_emailField__pcb03","emailRecipients":"Chat-module_emailRecipients__YVVys","emailBody":"Chat-module_emailBody__OBt-w","emailBodyContent":"Chat-module_emailBodyContent__-vFo2","emailStatus":"Chat-module_emailStatus__T4UuL","imagePrompt":"Chat-module_imagePrompt__v7-Ho","imagePromptPreview":"Chat-module_imagePromptPreview__ve5dB","imagePromptImage":"Chat-module_imagePromptImage__KjE3H","imagePromptPlaceholder":"Chat-module_imagePromptPlaceholder__b8cW6","imagePromptSpinner":"Chat-module_imagePromptSpinner__nat8r","imagePromptMeta":"Chat-module_imagePromptMeta__YwltP","imagePromptPrompt":"Chat-module_imagePromptPrompt__N1E--","imagePromptStatus":"Chat-module_imagePromptStatus__0ROQF","imagePromptError":"Chat-module_imagePromptError__x1952"};
|
|
14846
16154
|
styleInject(css_248z$4);
|
|
14847
16155
|
|
|
14848
16156
|
/**
|
|
@@ -15382,6 +16690,40 @@
|
|
|
15382
16690
|
return toolCall.name === ASSISTANT_PREPARATION_TOOL_CALL_NAME;
|
|
15383
16691
|
}
|
|
15384
16692
|
|
|
16693
|
+
/**
|
|
16694
|
+
* Builds a stable identity string for tool calls across partial updates.
|
|
16695
|
+
*
|
|
16696
|
+
* @param toolCall - Tool call entry to identify.
|
|
16697
|
+
* @returns Stable identity string for deduplication.
|
|
16698
|
+
*
|
|
16699
|
+
* @private function of <Chat/>
|
|
16700
|
+
*/
|
|
16701
|
+
function getToolCallIdentity(toolCall) {
|
|
16702
|
+
const rawToolCall = toolCall.rawToolCall;
|
|
16703
|
+
const rawId = (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.id) || (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.callId) || (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.call_id);
|
|
16704
|
+
if (rawId) {
|
|
16705
|
+
return `id:${rawId}`;
|
|
16706
|
+
}
|
|
16707
|
+
if (toolCall.createdAt) {
|
|
16708
|
+
return `time:${toolCall.createdAt}:${toolCall.name}`;
|
|
16709
|
+
}
|
|
16710
|
+
const argsKey = (() => {
|
|
16711
|
+
if (typeof toolCall.arguments === 'string') {
|
|
16712
|
+
return toolCall.arguments;
|
|
16713
|
+
}
|
|
16714
|
+
if (!toolCall.arguments) {
|
|
16715
|
+
return '';
|
|
16716
|
+
}
|
|
16717
|
+
try {
|
|
16718
|
+
return JSON.stringify(toolCall.arguments);
|
|
16719
|
+
}
|
|
16720
|
+
catch (_a) {
|
|
16721
|
+
return '';
|
|
16722
|
+
}
|
|
16723
|
+
})();
|
|
16724
|
+
return `fallback:${toolCall.name}:${argsKey}`;
|
|
16725
|
+
}
|
|
16726
|
+
|
|
15385
16727
|
var css_248z$3 = ".AvatarProfileTooltip-module_AvatarProfileTooltip__bZ020{background-color:transparent;border:none;max-width:90vw;padding:0;position:fixed;width:320px;z-index:1000}.AvatarProfileTooltip-module_arrow__wmM61{border-color:transparent transparent #fff;border-style:solid;border-width:0 9px 9px;filter:drop-shadow(0 -3px 4px rgba(15,23,42,.15));height:0;left:28px;position:absolute;top:-9px;width:0}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkF2YXRhclByb2ZpbGVUb29sdGlwLm1vZHVsZS5jc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEseURBSUksNEJBQTZCLENBQzdCLFdBQVksQ0FFWixjQUFlLENBSmYsU0FBVSxDQUZWLGNBQWUsQ0FLZixXQUFZLENBSlosWUFNSixDQUVBLDBDQVFJLHlDQUF5RCxDQUF6RCxrQkFBeUQsQ0FBekQsc0JBQXlELENBQ3pELGlEQUFzRCxDQU50RCxRQUFTLENBR1QsU0FBVSxDQUxWLGlCQUFrQixDQUlsQixRQUFTLENBSFQsT0FRSiIsImZpbGUiOiJBdmF0YXJQcm9maWxlVG9vbHRpcC5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLkF2YXRhclByb2ZpbGVUb29sdGlwIHtcbiAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgei1pbmRleDogMTAwMDtcbiAgICBwYWRkaW5nOiAwO1xuICAgIGJhY2tncm91bmQtY29sb3I6IHRyYW5zcGFyZW50O1xuICAgIGJvcmRlcjogbm9uZTtcbiAgICB3aWR0aDogMzIwcHg7XG4gICAgbWF4LXdpZHRoOiA5MHZ3O1xufVxuXG4uYXJyb3cge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB3aWR0aDogMDtcbiAgICBoZWlnaHQ6IDA7XG4gICAgYm9yZGVyLXN0eWxlOiBzb2xpZDtcbiAgICB0b3A6IC05cHg7XG4gICAgbGVmdDogMjhweDtcbiAgICBib3JkZXItd2lkdGg6IDAgOXB4IDlweCA5cHg7XG4gICAgYm9yZGVyLWNvbG9yOiB0cmFuc3BhcmVudCB0cmFuc3BhcmVudCAjZmZmZmZmIHRyYW5zcGFyZW50O1xuICAgIGZpbHRlcjogZHJvcC1zaGFkb3coMCAtM3B4IDRweCByZ2JhKDE1LCAyMywgNDIsIDAuMTUpKTtcbn1cbiJdfQ== */";
|
|
15386
16728
|
var styles$3 = {"AvatarProfileTooltip":"AvatarProfileTooltip-module_AvatarProfileTooltip__bZ020","arrow":"AvatarProfileTooltip-module_arrow__wmM61"};
|
|
15387
16729
|
styleInject(css_248z$3);
|
|
@@ -19893,6 +21235,7 @@
|
|
|
19893
21235
|
systemMessage: requirements.systemMessage + '\n\n' + exampleInteractionsContent,
|
|
19894
21236
|
};
|
|
19895
21237
|
}
|
|
21238
|
+
requirements = await applyPendingInlineKnowledgeSources(requirements, options === null || options === void 0 ? void 0 : options.inlineKnowledgeSourceUploader);
|
|
19896
21239
|
// Remove comment lines (lines starting with #) from the final system message
|
|
19897
21240
|
// while preserving the original content with comments in metadata
|
|
19898
21241
|
const cleanedSystemMessage = removeCommentsFromSystemMessage(requirements.systemMessage);
|
|
@@ -19902,16 +21245,64 @@
|
|
|
19902
21245
|
};
|
|
19903
21246
|
}
|
|
19904
21247
|
/**
|
|
19905
|
-
*
|
|
19906
|
-
*
|
|
19907
|
-
* @param urlOrPath - The URL or local path of the file to check
|
|
19908
|
-
* @returns A promise that resolves if the file is safe
|
|
21248
|
+
* @private Attempts to upload inline knowledge entries, falling back to legacy data URLs when the upload fails or is not configured.
|
|
19909
21249
|
*/
|
|
19910
|
-
async function
|
|
19911
|
-
|
|
19912
|
-
|
|
19913
|
-
if (
|
|
19914
|
-
|
|
21250
|
+
async function applyPendingInlineKnowledgeSources(requirements, uploader) {
|
|
21251
|
+
var _a;
|
|
21252
|
+
const inlineSources = extractInlineKnowledgeSources(requirements._metadata);
|
|
21253
|
+
if (inlineSources.length === 0) {
|
|
21254
|
+
return requirements;
|
|
21255
|
+
}
|
|
21256
|
+
const knowledgeSources = [...((_a = requirements.knowledgeSources) !== null && _a !== void 0 ? _a : [])];
|
|
21257
|
+
for (const inlineSource of inlineSources) {
|
|
21258
|
+
const url = uploader
|
|
21259
|
+
? await uploadInlineKnowledgeSourceWithFallback(inlineSource, uploader)
|
|
21260
|
+
: inlineKnowledgeSourceToDataUrl(inlineSource);
|
|
21261
|
+
knowledgeSources.push(url);
|
|
21262
|
+
}
|
|
21263
|
+
return {
|
|
21264
|
+
...requirements,
|
|
21265
|
+
knowledgeSources,
|
|
21266
|
+
_metadata: stripInlineKnowledgeMetadata(requirements._metadata),
|
|
21267
|
+
};
|
|
21268
|
+
}
|
|
21269
|
+
async function uploadInlineKnowledgeSourceWithFallback(inlineSource, uploader) {
|
|
21270
|
+
try {
|
|
21271
|
+
return await uploader(inlineSource);
|
|
21272
|
+
}
|
|
21273
|
+
catch (error) {
|
|
21274
|
+
console.error('[inline-knowledge] Failed to upload inline source', {
|
|
21275
|
+
filename: inlineSource.filename,
|
|
21276
|
+
error,
|
|
21277
|
+
});
|
|
21278
|
+
return inlineKnowledgeSourceToDataUrl(inlineSource);
|
|
21279
|
+
}
|
|
21280
|
+
}
|
|
21281
|
+
function extractInlineKnowledgeSources(metadata) {
|
|
21282
|
+
if (!metadata) {
|
|
21283
|
+
return [];
|
|
21284
|
+
}
|
|
21285
|
+
const value = metadata.inlineKnowledgeSources;
|
|
21286
|
+
return Array.isArray(value) ? value : [];
|
|
21287
|
+
}
|
|
21288
|
+
function stripInlineKnowledgeMetadata(metadata) {
|
|
21289
|
+
if (!metadata || !Object.prototype.hasOwnProperty.call(metadata, 'inlineKnowledgeSources')) {
|
|
21290
|
+
return metadata;
|
|
21291
|
+
}
|
|
21292
|
+
const { inlineKnowledgeSources: _unusedInlineKnowledgeSources, ...rest } = metadata;
|
|
21293
|
+
return Object.keys(rest).length > 0 ? rest : undefined;
|
|
21294
|
+
}
|
|
21295
|
+
/**
|
|
21296
|
+
* Mocked security check for imported files
|
|
21297
|
+
*
|
|
21298
|
+
* @param urlOrPath - The URL or local path of the file to check
|
|
21299
|
+
* @returns A promise that resolves if the file is safe
|
|
21300
|
+
*/
|
|
21301
|
+
async function mockedSecurityCheck(urlOrPath) {
|
|
21302
|
+
// TODO: Implement proper security checks
|
|
21303
|
+
await new Promise((resolve) => setTimeout(resolve, 10)); // Mock async delay
|
|
21304
|
+
if (urlOrPath.includes('malicious')) {
|
|
21305
|
+
throw new Error(`Security check failed for: ${urlOrPath}`);
|
|
19915
21306
|
}
|
|
19916
21307
|
}
|
|
19917
21308
|
/**
|
|
@@ -21106,6 +22497,28 @@
|
|
|
21106
22497
|
}));
|
|
21107
22498
|
}
|
|
21108
22499
|
|
|
22500
|
+
/**
|
|
22501
|
+
* Builds a tool invocation script that injects hidden runtime context into tool args.
|
|
22502
|
+
*
|
|
22503
|
+
* @private utility of OpenAI tool execution wrappers
|
|
22504
|
+
*/
|
|
22505
|
+
function buildToolInvocationScript(options) {
|
|
22506
|
+
const { functionName, functionArgsExpression } = options;
|
|
22507
|
+
return `
|
|
22508
|
+
const args = ${functionArgsExpression};
|
|
22509
|
+
const runtimeContextRaw =
|
|
22510
|
+
typeof ${TOOL_RUNTIME_CONTEXT_PARAMETER} === 'undefined'
|
|
22511
|
+
? undefined
|
|
22512
|
+
: ${TOOL_RUNTIME_CONTEXT_PARAMETER};
|
|
22513
|
+
|
|
22514
|
+
if (runtimeContextRaw !== undefined && args && typeof args === 'object' && !Array.isArray(args)) {
|
|
22515
|
+
args.${TOOL_RUNTIME_CONTEXT_ARGUMENT} = runtimeContextRaw;
|
|
22516
|
+
}
|
|
22517
|
+
|
|
22518
|
+
return await ${functionName}(args);
|
|
22519
|
+
`;
|
|
22520
|
+
}
|
|
22521
|
+
|
|
21109
22522
|
/**
|
|
21110
22523
|
* Parses an OpenAI error message to identify which parameter is unsupported
|
|
21111
22524
|
*
|
|
@@ -21459,10 +22872,10 @@
|
|
|
21459
22872
|
const scriptTool = scriptTools[0]; // <- TODO: [🧠] Which script tool to use?
|
|
21460
22873
|
functionResponse = await scriptTool.execute({
|
|
21461
22874
|
scriptLanguage: 'javascript',
|
|
21462
|
-
script:
|
|
21463
|
-
|
|
21464
|
-
|
|
21465
|
-
|
|
22875
|
+
script: buildToolInvocationScript({
|
|
22876
|
+
functionName,
|
|
22877
|
+
functionArgsExpression: functionArgs,
|
|
22878
|
+
}),
|
|
21466
22879
|
parameters: prompt.parameters,
|
|
21467
22880
|
});
|
|
21468
22881
|
}
|
|
@@ -23164,10 +24577,10 @@
|
|
|
23164
24577
|
try {
|
|
23165
24578
|
return await scriptTool.execute({
|
|
23166
24579
|
scriptLanguage: 'javascript',
|
|
23167
|
-
script:
|
|
23168
|
-
|
|
23169
|
-
|
|
23170
|
-
|
|
24580
|
+
script: buildToolInvocationScript({
|
|
24581
|
+
functionName,
|
|
24582
|
+
functionArgsExpression: JSON.stringify(functionArgs),
|
|
24583
|
+
}),
|
|
23171
24584
|
parameters: (_c = (_b = runContext === null || runContext === void 0 ? void 0 : runContext.context) === null || _b === void 0 ? void 0 : _b.parameters) !== null && _c !== void 0 ? _c : {},
|
|
23172
24585
|
});
|
|
23173
24586
|
}
|
|
@@ -23620,10 +25033,10 @@
|
|
|
23620
25033
|
const scriptTool = scriptTools[0]; // <- TODO: [🧠] Which script tool to use?
|
|
23621
25034
|
functionResponse = await scriptTool.execute({
|
|
23622
25035
|
scriptLanguage: 'javascript',
|
|
23623
|
-
script:
|
|
23624
|
-
|
|
23625
|
-
|
|
23626
|
-
|
|
25036
|
+
script: buildToolInvocationScript({
|
|
25037
|
+
functionName,
|
|
25038
|
+
functionArgsExpression: JSON.stringify(functionArgs),
|
|
25039
|
+
}),
|
|
23627
25040
|
parameters: prompt.parameters,
|
|
23628
25041
|
});
|
|
23629
25042
|
if (this.options.isVerbose) {
|
|
@@ -24964,38 +26377,18 @@
|
|
|
24964
26377
|
*/
|
|
24965
26378
|
|
|
24966
26379
|
/**
|
|
24967
|
-
*
|
|
26380
|
+
* Keep-alive helpers used for streaming chat responses.
|
|
24968
26381
|
*
|
|
24969
|
-
*
|
|
24970
|
-
*
|
|
26382
|
+
* These constants coordinate the signal sent by the Agents Server streaming
|
|
26383
|
+
* endpoint and the parser in the SDK so we can distinguish between
|
|
26384
|
+
* real content and occasional pings.
|
|
24971
26385
|
*
|
|
24972
|
-
* @private
|
|
26386
|
+
* @private internal streaming helper for Promptbook chat connections
|
|
26387
|
+
*/
|
|
26388
|
+
const CHAT_STREAM_KEEP_ALIVE_TOKEN = 'STREAM_KEEP_ALIVE';
|
|
26389
|
+
/**
|
|
26390
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
24973
26391
|
*/
|
|
24974
|
-
function getToolCallIdentity(toolCall) {
|
|
24975
|
-
const rawToolCall = toolCall.rawToolCall;
|
|
24976
|
-
const rawId = (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.id) || (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.callId) || (rawToolCall === null || rawToolCall === void 0 ? void 0 : rawToolCall.call_id);
|
|
24977
|
-
if (rawId) {
|
|
24978
|
-
return `id:${rawId}`;
|
|
24979
|
-
}
|
|
24980
|
-
if (toolCall.createdAt) {
|
|
24981
|
-
return `time:${toolCall.createdAt}:${toolCall.name}`;
|
|
24982
|
-
}
|
|
24983
|
-
const argsKey = (() => {
|
|
24984
|
-
if (typeof toolCall.arguments === 'string') {
|
|
24985
|
-
return toolCall.arguments;
|
|
24986
|
-
}
|
|
24987
|
-
if (!toolCall.arguments) {
|
|
24988
|
-
return '';
|
|
24989
|
-
}
|
|
24990
|
-
try {
|
|
24991
|
-
return JSON.stringify(toolCall.arguments);
|
|
24992
|
-
}
|
|
24993
|
-
catch (_a) {
|
|
24994
|
-
return '';
|
|
24995
|
-
}
|
|
24996
|
-
})();
|
|
24997
|
-
return `fallback:${toolCall.name}:${argsKey}`;
|
|
24998
|
-
}
|
|
24999
26392
|
|
|
25000
26393
|
/**
|
|
25001
26394
|
* Resolve a remote META IMAGE value into an absolute URL when possible.
|
|
@@ -25071,7 +26464,9 @@
|
|
|
25071
26464
|
static async connect(options) {
|
|
25072
26465
|
var _a, _b, _c;
|
|
25073
26466
|
const agentProfileUrl = `${options.agentUrl}/api/profile`;
|
|
25074
|
-
const profileResponse = await fetch(agentProfileUrl
|
|
26467
|
+
const profileResponse = await fetch(agentProfileUrl, {
|
|
26468
|
+
headers: attachClientVersionHeader(),
|
|
26469
|
+
});
|
|
25075
26470
|
// <- TODO: [🐱🚀] What about closed-source agents?
|
|
25076
26471
|
// <- TODO: [🐱🚀] Maybe use promptbookFetch
|
|
25077
26472
|
if (!profileResponse.ok) {
|
|
@@ -25172,6 +26567,7 @@
|
|
|
25172
26567
|
}
|
|
25173
26568
|
const response = await fetch(`${this.agentUrl}/api/voice`, {
|
|
25174
26569
|
method: 'POST',
|
|
26570
|
+
headers: attachClientVersionHeader(),
|
|
25175
26571
|
body: formData,
|
|
25176
26572
|
});
|
|
25177
26573
|
if (!response.ok) {
|
|
@@ -25201,13 +26597,14 @@
|
|
|
25201
26597
|
const chatPrompt = prompt;
|
|
25202
26598
|
const bookResponse = await fetch(`${this.agentUrl}/api/chat`, {
|
|
25203
26599
|
method: 'POST',
|
|
25204
|
-
headers: {
|
|
26600
|
+
headers: attachClientVersionHeader({
|
|
25205
26601
|
'Content-Type': 'application/json',
|
|
25206
|
-
},
|
|
26602
|
+
}),
|
|
25207
26603
|
body: JSON.stringify({
|
|
25208
26604
|
message: prompt.content,
|
|
25209
26605
|
thread: chatPrompt.thread,
|
|
25210
26606
|
attachments: chatPrompt.attachments,
|
|
26607
|
+
parameters: chatPrompt.parameters,
|
|
25211
26608
|
}),
|
|
25212
26609
|
});
|
|
25213
26610
|
// <- TODO: [🐱🚀] What about closed-source agents?
|
|
@@ -25289,6 +26686,9 @@
|
|
|
25289
26686
|
const lines = textChunk.split(/\r?\n/);
|
|
25290
26687
|
for (const line of lines) {
|
|
25291
26688
|
const trimmedLine = line.trim();
|
|
26689
|
+
if (trimmedLine === CHAT_STREAM_KEEP_ALIVE_TOKEN) {
|
|
26690
|
+
continue;
|
|
26691
|
+
}
|
|
25292
26692
|
let isToolCallLine = false;
|
|
25293
26693
|
if (trimmedLine.startsWith('{') && trimmedLine.endsWith('}')) {
|
|
25294
26694
|
try {
|
|
@@ -26374,49 +27774,6 @@
|
|
|
26374
27774
|
return name.startsWith(TEAM_TOOL_PREFIX);
|
|
26375
27775
|
}
|
|
26376
27776
|
|
|
26377
|
-
/**
|
|
26378
|
-
* Builds display-ready timestamp and duration labels for a chat message.
|
|
26379
|
-
*
|
|
26380
|
-
* @private utility of `<Chat/>` component
|
|
26381
|
-
*/
|
|
26382
|
-
function getChatMessageTimingDisplay(message) {
|
|
26383
|
-
if (!message.createdAt) {
|
|
26384
|
-
return null;
|
|
26385
|
-
}
|
|
26386
|
-
const timestamp = moment__default["default"](message.createdAt);
|
|
26387
|
-
if (!timestamp.isValid()) {
|
|
26388
|
-
return null;
|
|
26389
|
-
}
|
|
26390
|
-
const durationLabel = message.generationDurationMs === undefined
|
|
26391
|
-
? undefined
|
|
26392
|
-
: formatGenerationDurationLabel(message.generationDurationMs);
|
|
26393
|
-
return {
|
|
26394
|
-
timeLabel: timestamp.format('LT'),
|
|
26395
|
-
fullLabel: timestamp.format('YYYY-MM-DD HH:mm:ss'),
|
|
26396
|
-
durationLabel,
|
|
26397
|
-
};
|
|
26398
|
-
}
|
|
26399
|
-
/**
|
|
26400
|
-
* Formats a generation duration value into a compact label.
|
|
26401
|
-
*/
|
|
26402
|
-
function formatGenerationDurationLabel(durationMs) {
|
|
26403
|
-
if (!Number.isFinite(durationMs) || durationMs < 0) {
|
|
26404
|
-
return '0ms';
|
|
26405
|
-
}
|
|
26406
|
-
const duration = moment__default["default"].duration(durationMs);
|
|
26407
|
-
const totalSeconds = duration.asSeconds();
|
|
26408
|
-
if (totalSeconds < 1) {
|
|
26409
|
-
return `${Math.max(1, Math.round(duration.asMilliseconds()))}ms`;
|
|
26410
|
-
}
|
|
26411
|
-
if (totalSeconds < 60) {
|
|
26412
|
-
const precision = totalSeconds < 10 ? 1 : 0;
|
|
26413
|
-
return `${totalSeconds.toFixed(precision)}s`;
|
|
26414
|
-
}
|
|
26415
|
-
const minutes = Math.floor(duration.asMinutes());
|
|
26416
|
-
const seconds = Math.round(totalSeconds - minutes * 60);
|
|
26417
|
-
return `${minutes}m ${seconds}s`;
|
|
26418
|
-
}
|
|
26419
|
-
|
|
26420
27777
|
/**
|
|
26421
27778
|
* @@@
|
|
26422
27779
|
*
|
|
@@ -26644,130 +28001,6 @@
|
|
|
26644
28001
|
* <- TODO: But maybe split into multiple files later?
|
|
26645
28002
|
*/
|
|
26646
28003
|
|
|
26647
|
-
/**
|
|
26648
|
-
* Builds display text for a tool call chiplet.
|
|
26649
|
-
*
|
|
26650
|
-
* @param chipletInfo - Chiplet metadata for the tool call.
|
|
26651
|
-
*
|
|
26652
|
-
* @private utility of `<Chat/>`
|
|
26653
|
-
*/
|
|
26654
|
-
function buildToolCallChipText(chipletInfo) {
|
|
26655
|
-
return chipletInfo.text;
|
|
26656
|
-
}
|
|
26657
|
-
/**
|
|
26658
|
-
* Technical to user-friendly tool names and emojis
|
|
26659
|
-
*
|
|
26660
|
-
* @private utility of `<Chat/>` [🧠] Maybe public?
|
|
26661
|
-
*/
|
|
26662
|
-
const TOOL_TITLES = {
|
|
26663
|
-
[ASSISTANT_PREPARATION_TOOL_CALL_NAME]: { title: 'Preparing agent', emoji: '...' },
|
|
26664
|
-
'self-learning': { title: 'self-learning', emoji: '🧠' },
|
|
26665
|
-
web_search: { title: 'Searching the web', emoji: '🔎' },
|
|
26666
|
-
useSearchEngine: { title: 'Searching the web', emoji: '🔎' },
|
|
26667
|
-
search: { title: 'Searching the web', emoji: '🔎' },
|
|
26668
|
-
useBrowser: { title: 'Browsing the web', emoji: '🌐' },
|
|
26669
|
-
browse: { title: 'Browsing the web', emoji: '🌐' },
|
|
26670
|
-
fetch_url_content: { title: 'Fetching URL content', emoji: '🌐' },
|
|
26671
|
-
run_browser: { title: 'Running browser', emoji: '🌐' },
|
|
26672
|
-
get_current_time: { title: 'Checking time', emoji: '🕒' },
|
|
26673
|
-
useTime: { title: 'Checking time', emoji: '🕒' },
|
|
26674
|
-
send_email: { title: 'Sending email', emoji: '📧' },
|
|
26675
|
-
useEmail: { title: 'Sending email', emoji: '📧' },
|
|
26676
|
-
// Add more tools here as needed
|
|
26677
|
-
};
|
|
26678
|
-
/**
|
|
26679
|
-
* Gets the chiplet information including text and agent data (for team tools).
|
|
26680
|
-
*
|
|
26681
|
-
* @private [🧠] Maybe public?
|
|
26682
|
-
*/
|
|
26683
|
-
function getToolCallChipletInfo(toolCall) {
|
|
26684
|
-
const toolInfo = TOOL_TITLES[toolCall.name];
|
|
26685
|
-
const baseTitle = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title) || toolCall.name;
|
|
26686
|
-
const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
|
|
26687
|
-
const args = parseToolCallArguments(toolCall);
|
|
26688
|
-
const isTimeTool = toolCall.name === 'get_current_time' || toolCall.name === 'useTime';
|
|
26689
|
-
const isEmailTool = toolCall.name === 'send_email' || toolCall.name === 'useEmail';
|
|
26690
|
-
const resultRaw = parseToolCallResult(toolCall.result);
|
|
26691
|
-
const teamResult = parseTeamToolResult(resultRaw);
|
|
26692
|
-
if (teamResult === null || teamResult === void 0 ? void 0 : teamResult.teammate) {
|
|
26693
|
-
const label = teamResult.teammate.label || teamResult.teammate.url || baseTitle;
|
|
26694
|
-
const agentData = {
|
|
26695
|
-
url: teamResult.teammate.url,
|
|
26696
|
-
label,
|
|
26697
|
-
};
|
|
26698
|
-
return {
|
|
26699
|
-
text: label,
|
|
26700
|
-
agentData,
|
|
26701
|
-
};
|
|
26702
|
-
}
|
|
26703
|
-
if (isTimeTool) {
|
|
26704
|
-
const resultDate = getToolCallResultDate(resultRaw);
|
|
26705
|
-
if (resultDate) {
|
|
26706
|
-
return {
|
|
26707
|
-
text: `${emoji} ${resultDate.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' })}`,
|
|
26708
|
-
};
|
|
26709
|
-
}
|
|
26710
|
-
}
|
|
26711
|
-
if (isEmailTool) {
|
|
26712
|
-
if (args.subject) {
|
|
26713
|
-
return {
|
|
26714
|
-
text: `${emoji} ${args.subject}`,
|
|
26715
|
-
};
|
|
26716
|
-
}
|
|
26717
|
-
}
|
|
26718
|
-
if (args.query) {
|
|
26719
|
-
return {
|
|
26720
|
-
text: `${emoji} ${args.query}`,
|
|
26721
|
-
};
|
|
26722
|
-
}
|
|
26723
|
-
if (args.url) {
|
|
26724
|
-
try {
|
|
26725
|
-
const url = new URL(args.url);
|
|
26726
|
-
return {
|
|
26727
|
-
text: `${emoji} ${url.hostname}`,
|
|
26728
|
-
};
|
|
26729
|
-
}
|
|
26730
|
-
catch (e) {
|
|
26731
|
-
return {
|
|
26732
|
-
text: `${emoji} ${args.url}`,
|
|
26733
|
-
};
|
|
26734
|
-
}
|
|
26735
|
-
}
|
|
26736
|
-
return {
|
|
26737
|
-
text: `${emoji} ${baseTitle}`,
|
|
26738
|
-
};
|
|
26739
|
-
}
|
|
26740
|
-
|
|
26741
|
-
/**
|
|
26742
|
-
* Parses markdown buttons in the format [Button Text](?message=Message%20to%20send)
|
|
26743
|
-
* Returns both the content without buttons and the extracted buttons
|
|
26744
|
-
*
|
|
26745
|
-
* @param content The markdown content that may contain buttons
|
|
26746
|
-
* @returns Object with contentWithoutButtons and buttons array
|
|
26747
|
-
*
|
|
26748
|
-
* @public exported from `@promptbook/components`
|
|
26749
|
-
*/
|
|
26750
|
-
function parseMessageButtons(content) {
|
|
26751
|
-
const buttonRegex = /\[([^\]]+)\]\(\?message=([^)]+)\)/g;
|
|
26752
|
-
const buttons = [];
|
|
26753
|
-
let match;
|
|
26754
|
-
// Extract all buttons
|
|
26755
|
-
while ((match = buttonRegex.exec(content)) !== null) {
|
|
26756
|
-
const [, text, encodedMessage] = match;
|
|
26757
|
-
if (text && encodedMessage) {
|
|
26758
|
-
// Decode URL-encoded message
|
|
26759
|
-
const message = decodeURIComponent(encodedMessage);
|
|
26760
|
-
buttons.push({ text, message });
|
|
26761
|
-
}
|
|
26762
|
-
}
|
|
26763
|
-
// Remove buttons from content
|
|
26764
|
-
const contentWithoutButtons = content.replace(buttonRegex, '').trim();
|
|
26765
|
-
return {
|
|
26766
|
-
contentWithoutButtons,
|
|
26767
|
-
buttons,
|
|
26768
|
-
};
|
|
26769
|
-
}
|
|
26770
|
-
|
|
26771
28004
|
/**
|
|
26772
28005
|
* Collects tool calls and citations from TEAM tool call results, resolving nested teammate chains.
|
|
26773
28006
|
*
|
|
@@ -26900,15 +28133,361 @@
|
|
|
26900
28133
|
return `${origin.label}:${citation.id}:${citation.source}`;
|
|
26901
28134
|
}
|
|
26902
28135
|
|
|
28136
|
+
const MEMORY_CHIP_MAX_LENGTH = 48;
|
|
28137
|
+
const MEMORY_CHIP_TRUNCATE_LENGTH = 45;
|
|
26903
28138
|
/**
|
|
26904
|
-
*
|
|
26905
|
-
*
|
|
28139
|
+
* Builds display text for a tool call chiplet.
|
|
28140
|
+
*
|
|
28141
|
+
* @param chipletInfo - Chiplet metadata for the tool call.
|
|
28142
|
+
*
|
|
28143
|
+
* @private utility of `<Chat/>`
|
|
26906
28144
|
*/
|
|
26907
|
-
|
|
26908
|
-
|
|
26909
|
-
|
|
28145
|
+
function buildToolCallChipText(chipletInfo) {
|
|
28146
|
+
return chipletInfo.text;
|
|
28147
|
+
}
|
|
28148
|
+
/**
|
|
28149
|
+
* Technical to user-friendly tool names and emojis
|
|
28150
|
+
*
|
|
28151
|
+
* @private utility of `<Chat/>` [🧠] Maybe public?
|
|
28152
|
+
*/
|
|
28153
|
+
const TOOL_TITLES = {
|
|
28154
|
+
[ASSISTANT_PREPARATION_TOOL_CALL_NAME]: { title: 'Preparing agent', emoji: '...' },
|
|
28155
|
+
'self-learning': { title: 'self-learning', emoji: '🧠' },
|
|
28156
|
+
retrieve_user_memory: { title: 'Reading memory', emoji: '🧠' },
|
|
28157
|
+
store_user_memory: { title: 'Storing memory', emoji: '🧠' },
|
|
28158
|
+
web_search: { title: 'Searching the web', emoji: '🔎' },
|
|
28159
|
+
useSearchEngine: { title: 'Searching the web', emoji: '🔎' },
|
|
28160
|
+
search: { title: 'Searching the web', emoji: '🔎' },
|
|
28161
|
+
useBrowser: { title: 'Browsing the web', emoji: '🌐' },
|
|
28162
|
+
browse: { title: 'Browsing the web', emoji: '🌐' },
|
|
28163
|
+
fetch_url_content: { title: 'Fetching URL content', emoji: '🌐' },
|
|
28164
|
+
run_browser: { title: 'Running browser', emoji: '🌐' },
|
|
28165
|
+
get_current_time: { title: 'Checking time', emoji: '🕒' },
|
|
28166
|
+
useTime: { title: 'Checking time', emoji: '🕒' },
|
|
28167
|
+
send_email: { title: 'Sending email', emoji: '📧' },
|
|
28168
|
+
useEmail: { title: 'Sending email', emoji: '📧' },
|
|
28169
|
+
// Add more tools here as needed
|
|
28170
|
+
};
|
|
28171
|
+
/**
|
|
28172
|
+
* Gets the chiplet information including text and agent data (for team tools).
|
|
28173
|
+
*
|
|
28174
|
+
* @private [🧠] Maybe public?
|
|
28175
|
+
*/
|
|
28176
|
+
function getToolCallChipletInfo(toolCall) {
|
|
28177
|
+
const toolInfo = TOOL_TITLES[toolCall.name];
|
|
28178
|
+
const baseTitle = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title) || toolCall.name;
|
|
28179
|
+
const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
|
|
28180
|
+
const args = parseToolCallArguments(toolCall);
|
|
28181
|
+
const isTimeTool = toolCall.name === 'get_current_time' || toolCall.name === 'useTime';
|
|
28182
|
+
const isEmailTool = toolCall.name === 'send_email' || toolCall.name === 'useEmail';
|
|
28183
|
+
const isMemoryTool = toolCall.name === 'retrieve_user_memory' || toolCall.name === 'store_user_memory';
|
|
28184
|
+
const resultRaw = parseToolCallResult(toolCall.result);
|
|
28185
|
+
const teamResult = parseTeamToolResult(resultRaw);
|
|
28186
|
+
if (teamResult === null || teamResult === void 0 ? void 0 : teamResult.teammate) {
|
|
28187
|
+
const label = teamResult.teammate.label || teamResult.teammate.url || baseTitle;
|
|
28188
|
+
const agentData = {
|
|
28189
|
+
url: teamResult.teammate.url,
|
|
28190
|
+
label,
|
|
28191
|
+
};
|
|
28192
|
+
return {
|
|
28193
|
+
text: label,
|
|
28194
|
+
agentData,
|
|
28195
|
+
};
|
|
28196
|
+
}
|
|
28197
|
+
if (isTimeTool) {
|
|
28198
|
+
const resultDate = getToolCallResultDate(resultRaw);
|
|
28199
|
+
if (resultDate) {
|
|
28200
|
+
return {
|
|
28201
|
+
text: `${emoji} ${resultDate.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' })}`,
|
|
28202
|
+
};
|
|
28203
|
+
}
|
|
28204
|
+
}
|
|
28205
|
+
if (isEmailTool) {
|
|
28206
|
+
if (args.subject) {
|
|
28207
|
+
return {
|
|
28208
|
+
text: `${emoji} ${args.subject}`,
|
|
28209
|
+
};
|
|
28210
|
+
}
|
|
28211
|
+
}
|
|
28212
|
+
if (isMemoryTool) {
|
|
28213
|
+
const memoryPreview = getMemoryPreviewText(args, resultRaw);
|
|
28214
|
+
if (memoryPreview) {
|
|
28215
|
+
return {
|
|
28216
|
+
text: `${emoji} ${memoryPreview}`,
|
|
28217
|
+
};
|
|
28218
|
+
}
|
|
28219
|
+
}
|
|
28220
|
+
if (args.query) {
|
|
28221
|
+
return {
|
|
28222
|
+
text: `${emoji} ${args.query}`,
|
|
28223
|
+
};
|
|
28224
|
+
}
|
|
28225
|
+
if (args.url) {
|
|
28226
|
+
try {
|
|
28227
|
+
const url = new URL(args.url);
|
|
28228
|
+
return {
|
|
28229
|
+
text: `${emoji} ${url.hostname}`,
|
|
28230
|
+
};
|
|
28231
|
+
}
|
|
28232
|
+
catch (e) {
|
|
28233
|
+
return {
|
|
28234
|
+
text: `${emoji} ${args.url}`,
|
|
28235
|
+
};
|
|
28236
|
+
}
|
|
28237
|
+
}
|
|
28238
|
+
return {
|
|
28239
|
+
text: `${emoji} ${baseTitle}`,
|
|
28240
|
+
};
|
|
28241
|
+
}
|
|
28242
|
+
/**
|
|
28243
|
+
* Builds memory preview text for MEMORY commitment tool calls.
|
|
28244
|
+
*/
|
|
28245
|
+
function getMemoryPreviewText(args, resultRaw) {
|
|
28246
|
+
var _a, _b, _c;
|
|
28247
|
+
if (!resultRaw || typeof resultRaw !== 'object') {
|
|
28248
|
+
return typeof args.content === 'string' ? shortenMemoryPreview(args.content) : null;
|
|
28249
|
+
}
|
|
28250
|
+
const result = resultRaw;
|
|
28251
|
+
if (result.action === 'store') {
|
|
28252
|
+
const content = ((_a = result.memory) === null || _a === void 0 ? void 0 : _a.content) || (typeof args.content === 'string' ? args.content : '');
|
|
28253
|
+
return content ? shortenMemoryPreview(content) : null;
|
|
28254
|
+
}
|
|
28255
|
+
const firstMemoryContent = (_c = (_b = result.memories) === null || _b === void 0 ? void 0 : _b.find((memory) => typeof memory.content === 'string')) === null || _c === void 0 ? void 0 : _c.content;
|
|
28256
|
+
if (firstMemoryContent) {
|
|
28257
|
+
return shortenMemoryPreview(firstMemoryContent);
|
|
28258
|
+
}
|
|
28259
|
+
if (typeof result.query === 'string' && result.query.trim()) {
|
|
28260
|
+
return `No match for "${result.query.trim()}"`;
|
|
28261
|
+
}
|
|
28262
|
+
return null;
|
|
28263
|
+
}
|
|
28264
|
+
/**
|
|
28265
|
+
* Shortens long memory content for compact chip display.
|
|
28266
|
+
*/
|
|
28267
|
+
function shortenMemoryPreview(content) {
|
|
28268
|
+
const trimmed = content.trim().replace(/\s+/g, ' ');
|
|
28269
|
+
if (trimmed.length <= MEMORY_CHIP_MAX_LENGTH) {
|
|
28270
|
+
return trimmed;
|
|
28271
|
+
}
|
|
28272
|
+
return `${trimmed.slice(0, MEMORY_CHIP_TRUNCATE_LENGTH)}...`;
|
|
28273
|
+
}
|
|
28274
|
+
|
|
28275
|
+
const IMAGE_PROMPT_REGEX = /!\[([^\]]*)\]\(\?image-prompt=([^)]+)\)/gi;
|
|
28276
|
+
function decodePrompt(value) {
|
|
28277
|
+
try {
|
|
28278
|
+
return decodeURIComponent(value);
|
|
28279
|
+
}
|
|
28280
|
+
catch (_a) {
|
|
28281
|
+
return value;
|
|
28282
|
+
}
|
|
28283
|
+
}
|
|
28284
|
+
/**
|
|
28285
|
+
* Splits a chat message into text and image prompt segments.
|
|
28286
|
+
*
|
|
28287
|
+
* @param content - Raw markdown string produced by the agent.
|
|
28288
|
+
* @returns Ordered list of segments that preserves the original message flow.
|
|
28289
|
+
* @private internal helper of `<ChatMessageItem/>`
|
|
28290
|
+
*/
|
|
28291
|
+
function splitMessageContentByImagePrompts(content) {
|
|
28292
|
+
if (!content) {
|
|
28293
|
+
return [{ type: 'text', content }];
|
|
28294
|
+
}
|
|
28295
|
+
const segments = [];
|
|
28296
|
+
let lastIndex = 0;
|
|
28297
|
+
IMAGE_PROMPT_REGEX.lastIndex = 0;
|
|
28298
|
+
let match;
|
|
28299
|
+
while ((match = IMAGE_PROMPT_REGEX.exec(content)) !== null) {
|
|
28300
|
+
const [fullMatch, alt = '', rawPrompt = ''] = match;
|
|
28301
|
+
const start = match.index;
|
|
28302
|
+
if (start > lastIndex) {
|
|
28303
|
+
segments.push({
|
|
28304
|
+
type: 'text',
|
|
28305
|
+
content: content.slice(lastIndex, start),
|
|
28306
|
+
});
|
|
28307
|
+
}
|
|
28308
|
+
const decodedPrompt = decodePrompt(rawPrompt);
|
|
28309
|
+
const prompt = spaceTrim__default["default"](decodedPrompt) || decodedPrompt || 'Generated image';
|
|
28310
|
+
const decodedAlt = decodePrompt(alt);
|
|
28311
|
+
segments.push({
|
|
28312
|
+
type: 'image',
|
|
28313
|
+
alt: spaceTrim__default["default"](decodedAlt) || 'Generated image',
|
|
28314
|
+
prompt,
|
|
28315
|
+
});
|
|
28316
|
+
lastIndex = start + fullMatch.length;
|
|
28317
|
+
}
|
|
28318
|
+
if (lastIndex < content.length) {
|
|
28319
|
+
segments.push({
|
|
28320
|
+
type: 'text',
|
|
28321
|
+
content: content.slice(lastIndex),
|
|
28322
|
+
});
|
|
28323
|
+
}
|
|
28324
|
+
if (segments.length === 0) {
|
|
28325
|
+
return [{ type: 'text', content }];
|
|
28326
|
+
}
|
|
28327
|
+
return segments;
|
|
28328
|
+
}
|
|
28329
|
+
/**
|
|
28330
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
28331
|
+
*/
|
|
28332
|
+
|
|
28333
|
+
/**
|
|
28334
|
+
* Parses markdown buttons in the format [Button Text](?message=Message%20to%20send)
|
|
28335
|
+
* Returns both the content without buttons and the extracted buttons
|
|
28336
|
+
*
|
|
28337
|
+
* @param content The markdown content that may contain buttons
|
|
28338
|
+
* @returns Object with contentWithoutButtons and buttons array
|
|
28339
|
+
*
|
|
28340
|
+
* @public exported from `@promptbook/components`
|
|
28341
|
+
*/
|
|
28342
|
+
function parseMessageButtons(content) {
|
|
28343
|
+
const buttonRegex = /\[([^\]]+)\]\(\?message=([^)]+)\)/g;
|
|
28344
|
+
const buttons = [];
|
|
28345
|
+
let match;
|
|
28346
|
+
// Extract all buttons
|
|
28347
|
+
while ((match = buttonRegex.exec(content)) !== null) {
|
|
28348
|
+
const [, text, encodedMessage] = match;
|
|
28349
|
+
if (text && encodedMessage) {
|
|
28350
|
+
// Decode URL-encoded message
|
|
28351
|
+
const message = decodeURIComponent(encodedMessage);
|
|
28352
|
+
buttons.push({ text, message });
|
|
28353
|
+
}
|
|
28354
|
+
}
|
|
28355
|
+
// Remove buttons from content
|
|
28356
|
+
const contentWithoutButtons = content.replace(buttonRegex, '').trim();
|
|
28357
|
+
return {
|
|
28358
|
+
contentWithoutButtons,
|
|
28359
|
+
buttons,
|
|
28360
|
+
};
|
|
28361
|
+
}
|
|
28362
|
+
|
|
28363
|
+
/**
|
|
28364
|
+
* @deprecated use `isComplete` instead
|
|
28365
|
+
* @private util of `<Chat />`
|
|
28366
|
+
*/
|
|
28367
|
+
const LOADING_INTERACTIVE_IMAGE = 'Loading...';
|
|
28368
|
+
/**
|
|
28369
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
28370
|
+
*/
|
|
28371
|
+
|
|
28372
|
+
/**
|
|
28373
|
+
* Constructs a standardized filename for a generated image based on its prompt and parameters.
|
|
28374
|
+
*
|
|
28375
|
+
* @param params - The parameters for the image generation
|
|
28376
|
+
* @returns The constructed filename
|
|
28377
|
+
*
|
|
28378
|
+
* @private internal helper for image generation pipeline
|
|
28379
|
+
*/
|
|
28380
|
+
function constructImageFilename(params) {
|
|
28381
|
+
const { prompt, model, size, quality, style, attachments } = params;
|
|
28382
|
+
const promptTrimmed = spaceTrim$1.spaceTrim(prompt);
|
|
28383
|
+
return (promptTrimmed
|
|
28384
|
+
.toLowerCase()
|
|
28385
|
+
.replace(/[^a-z0-9]+/g, '-')
|
|
28386
|
+
.replace(/^-+|-+$/g, '') +
|
|
28387
|
+
// Note: Default model 'dall-e-3' is omitted for backward compatibility
|
|
28388
|
+
(model && model !== 'dall-e-3' ? `-${model.toLowerCase().replace(/[^a-z0-9]+/g, '-')}` : '') +
|
|
28389
|
+
(size === '1024x1024' || size === undefined ? '' : `-${size}`) +
|
|
28390
|
+
(quality === 'standard' || quality === undefined ? '' : `-${quality}`) +
|
|
28391
|
+
(style === 'vivid' || style === undefined ? '' : `-${style}`) +
|
|
28392
|
+
(attachments && attachments.length > 0 ? `-attach-${hashAttachments(attachments)}` : '') +
|
|
28393
|
+
'.png');
|
|
28394
|
+
}
|
|
28395
|
+
function hashAttachments(attachments) {
|
|
28396
|
+
const urls = attachments
|
|
28397
|
+
.map((a) => a.url)
|
|
28398
|
+
.sort()
|
|
28399
|
+
.join('|');
|
|
28400
|
+
let hash = 0;
|
|
28401
|
+
for (let i = 0; i < urls.length; i++) {
|
|
28402
|
+
const char = urls.charCodeAt(i);
|
|
28403
|
+
hash = (hash << 5) - hash + char;
|
|
28404
|
+
hash = hash & hash; // Convert to 32bit integer
|
|
28405
|
+
}
|
|
28406
|
+
return (hash >>> 0).toString(36);
|
|
28407
|
+
}
|
|
28408
|
+
|
|
28409
|
+
const generatedImageCache = new Map();
|
|
28410
|
+
const pendingImageFetches = new Map();
|
|
28411
|
+
async function fetchGeneratedImageUrl(filename) {
|
|
28412
|
+
const cached = generatedImageCache.get(filename);
|
|
28413
|
+
if (cached) {
|
|
28414
|
+
return cached;
|
|
28415
|
+
}
|
|
28416
|
+
if (pendingImageFetches.has(filename)) {
|
|
28417
|
+
return pendingImageFetches.get(filename);
|
|
28418
|
+
}
|
|
28419
|
+
const request = (async () => {
|
|
28420
|
+
const response = await fetch(`/api/images/${encodeURIComponent(filename)}?raw=true`, {
|
|
28421
|
+
headers: {
|
|
28422
|
+
'X-Promptbook-Client': 'chat',
|
|
28423
|
+
},
|
|
28424
|
+
});
|
|
28425
|
+
if (!response.ok) {
|
|
28426
|
+
const text = await response.text();
|
|
28427
|
+
throw new Error(text || `Image generation failed (${response.status})`);
|
|
28428
|
+
}
|
|
28429
|
+
const payload = await response.json();
|
|
28430
|
+
if (!(payload === null || payload === void 0 ? void 0 : payload.cdnUrl)) {
|
|
28431
|
+
throw new Error('Image generation API did not return a valid URL');
|
|
28432
|
+
}
|
|
28433
|
+
generatedImageCache.set(filename, payload.cdnUrl);
|
|
28434
|
+
return payload.cdnUrl;
|
|
28435
|
+
})();
|
|
28436
|
+
pendingImageFetches.set(filename, request);
|
|
28437
|
+
request.finally(() => pendingImageFetches.delete(filename));
|
|
28438
|
+
return request;
|
|
28439
|
+
}
|
|
28440
|
+
/**
|
|
28441
|
+
* Renders a preview area for the `` notation.
|
|
28442
|
+
*
|
|
28443
|
+
* @private internal component of `<ChatMessageItem/>`
|
|
26910
28444
|
*/
|
|
28445
|
+
function ImagePromptRenderer({ alt, prompt }) {
|
|
28446
|
+
const trimmedPrompt = react.useMemo(() => spaceTrim__default["default"](prompt), [prompt]);
|
|
28447
|
+
const filename = react.useMemo(() => constructImageFilename({
|
|
28448
|
+
prompt: trimmedPrompt,
|
|
28449
|
+
}), [trimmedPrompt]);
|
|
28450
|
+
const [status, setStatus] = react.useState('loading');
|
|
28451
|
+
const [imageUrl, setImageUrl] = react.useState(null);
|
|
28452
|
+
const [error, setError] = react.useState(null);
|
|
28453
|
+
react.useEffect(() => {
|
|
28454
|
+
let isMounted = true;
|
|
28455
|
+
setStatus('loading');
|
|
28456
|
+
setImageUrl(null);
|
|
28457
|
+
setError(null);
|
|
28458
|
+
void fetchGeneratedImageUrl(filename)
|
|
28459
|
+
.then((url) => {
|
|
28460
|
+
if (!isMounted) {
|
|
28461
|
+
return;
|
|
28462
|
+
}
|
|
28463
|
+
setImageUrl(url);
|
|
28464
|
+
setStatus('success');
|
|
28465
|
+
})
|
|
28466
|
+
.catch((fetchError) => {
|
|
28467
|
+
if (!isMounted) {
|
|
28468
|
+
return;
|
|
28469
|
+
}
|
|
28470
|
+
setStatus('error');
|
|
28471
|
+
setError((fetchError === null || fetchError === void 0 ? void 0 : fetchError.message) || 'Failed to generate image');
|
|
28472
|
+
});
|
|
28473
|
+
return () => {
|
|
28474
|
+
isMounted = false;
|
|
28475
|
+
};
|
|
28476
|
+
}, [filename]);
|
|
28477
|
+
const statusLabel = status === 'loading'
|
|
28478
|
+
? 'Generating image...'
|
|
28479
|
+
: status === 'error'
|
|
28480
|
+
? 'Failed to generate image'
|
|
28481
|
+
: 'Image generated';
|
|
28482
|
+
return (jsxRuntime.jsxs("div", { className: styles$4.imagePrompt, "aria-live": "polite", children: [jsxRuntime.jsxs("div", { className: styles$4.imagePromptPreview, children: [status === 'success' && imageUrl ? (jsxRuntime.jsx("img", { src: imageUrl, alt: alt, className: styles$4.imagePromptImage })) : (jsxRuntime.jsx("div", { className: styles$4.imagePromptPlaceholder, children: jsxRuntime.jsx("div", { className: styles$4.imagePromptSpinner }) })), status === 'error' && error && (jsxRuntime.jsx("div", { className: styles$4.imagePromptError, children: error }))] }), jsxRuntime.jsxs("div", { className: styles$4.imagePromptMeta, children: [jsxRuntime.jsx("span", { className: styles$4.imagePromptPrompt, children: trimmedPrompt || 'Generated image' }), jsxRuntime.jsx("span", { className: styles$4.imagePromptStatus, children: statusLabel })] })] }));
|
|
28483
|
+
}
|
|
26911
28484
|
|
|
28485
|
+
/**
|
|
28486
|
+
* Maximum characters allowed in a single ElevenLabs speech request.
|
|
28487
|
+
*
|
|
28488
|
+
* @private
|
|
28489
|
+
*/
|
|
28490
|
+
const MAX_MESSAGE_SPEECH_LENGTH = 4500;
|
|
26912
28491
|
/**
|
|
26913
28492
|
* Finds teammate metadata by tool name, falling back to the toolName field when needed.
|
|
26914
28493
|
*/
|
|
@@ -27126,16 +28705,11 @@
|
|
|
27126
28705
|
const color = Color.fromSafe((participant && participant.color) || (isMe ? USER_CHAT_COLOR : PROMPTBOOK_CHAT_COLOR));
|
|
27127
28706
|
const colorOfText = color.then(textColor);
|
|
27128
28707
|
const { contentWithoutButtons, buttons } = parseMessageButtons(message.content);
|
|
28708
|
+
const contentSegments = react.useMemo(() => splitMessageContentByImagePrompts(contentWithoutButtons), [contentWithoutButtons]);
|
|
27129
28709
|
const completedToolCalls = dedupeToolCalls((_a = (message.toolCalls || message.completedToolCalls)) === null || _a === void 0 ? void 0 : _a.filter((toolCall) => !isAssistantPreparationToolCall(toolCall)));
|
|
27130
28710
|
const teamToolCallSummary = react.useMemo(() => collectTeamToolCallSummary(completedToolCalls), [completedToolCalls]);
|
|
27131
28711
|
const transitiveToolCalls = teamToolCallSummary.toolCalls;
|
|
27132
28712
|
const transitiveCitations = teamToolCallSummary.citations;
|
|
27133
|
-
const ongoingToolCallGroups = react.useMemo(() => groupOngoingToolCalls(message.ongoingToolCalls, toolTitles, teammates), [message.ongoingToolCalls, toolTitles, teammates]);
|
|
27134
|
-
const completedToolCallCount = (_b = completedToolCalls === null || completedToolCalls === void 0 ? void 0 : completedToolCalls.length) !== null && _b !== void 0 ? _b : 0;
|
|
27135
|
-
const transitiveToolCallCount = transitiveToolCalls.length;
|
|
27136
|
-
const ongoingToolCallCount = ongoingToolCallGroups.length;
|
|
27137
|
-
const toolCallChipCount = completedToolCallCount + transitiveToolCallCount + ongoingToolCallCount;
|
|
27138
|
-
const shouldShowButtons = isLastMessage && buttons.length > 0 && onMessage;
|
|
27139
28713
|
// Extract citations from message content
|
|
27140
28714
|
const messageWithCitations = extractCitationsFromMessage(message);
|
|
27141
28715
|
const citations = messageWithCitations.citations || [];
|
|
@@ -27144,12 +28718,151 @@
|
|
|
27144
28718
|
const [copied, setCopied] = react.useState(false);
|
|
27145
28719
|
const [tooltipAlign, setTooltipAlign] = react.useState('center');
|
|
27146
28720
|
const copyTooltipRef = react.useRef(null);
|
|
28721
|
+
const contentWithoutButtonsRef = react.useRef(null);
|
|
28722
|
+
const audioRef = react.useRef(null);
|
|
28723
|
+
const [audioUrl, setAudioUrl] = react.useState(null);
|
|
28724
|
+
const [isAudioLoading, setIsAudioLoading] = react.useState(false);
|
|
28725
|
+
const [isAudioPlaying, setIsAudioPlaying] = react.useState(false);
|
|
28726
|
+
const [audioError, setAudioError] = react.useState(null);
|
|
28727
|
+
const ongoingToolCallGroups = react.useMemo(() => groupOngoingToolCalls(message.ongoingToolCalls, toolTitles, teammates), [message.ongoingToolCalls, toolTitles, teammates]);
|
|
28728
|
+
const completedToolCallCount = (_b = completedToolCalls === null || completedToolCalls === void 0 ? void 0 : completedToolCalls.length) !== null && _b !== void 0 ? _b : 0;
|
|
28729
|
+
const transitiveToolCallCount = transitiveToolCalls.length;
|
|
28730
|
+
const ongoingToolCallCount = ongoingToolCallGroups.length;
|
|
28731
|
+
const toolCallChipCount = completedToolCallCount + transitiveToolCallCount + ongoingToolCallCount;
|
|
28732
|
+
const shouldShowButtons = isLastMessage && buttons.length > 0 && onMessage;
|
|
28733
|
+
const playButtonTitle = audioError !== null && audioError !== void 0 ? audioError : (isAudioPlaying ? 'Pause message playback' : 'Read message aloud');
|
|
28734
|
+
const trimmedMessageContent = message.content.trim();
|
|
28735
|
+
const shouldShowPlayButton = trimmedMessageContent.length > 0;
|
|
28736
|
+
/**
|
|
28737
|
+
* Attaches playback listeners to keep the UI in sync with the audio element.
|
|
28738
|
+
*
|
|
28739
|
+
* @private
|
|
28740
|
+
*/
|
|
28741
|
+
const attachMessageAudioListeners = react.useCallback((element) => {
|
|
28742
|
+
element.onplay = () => {
|
|
28743
|
+
setIsAudioPlaying(true);
|
|
28744
|
+
};
|
|
28745
|
+
element.onpause = () => {
|
|
28746
|
+
setIsAudioPlaying(false);
|
|
28747
|
+
};
|
|
28748
|
+
element.onended = () => {
|
|
28749
|
+
setIsAudioPlaying(false);
|
|
28750
|
+
element.currentTime = 0;
|
|
28751
|
+
};
|
|
28752
|
+
}, []);
|
|
28753
|
+
/**
|
|
28754
|
+
* Derives the plain text that should be spoken, preferring the rendered node over raw markdown.
|
|
28755
|
+
*
|
|
28756
|
+
* @private
|
|
28757
|
+
*/
|
|
28758
|
+
const getMessageTextForSpeech = react.useCallback(() => {
|
|
28759
|
+
var _a, _b;
|
|
28760
|
+
const renderedText = (_b = (_a = contentWithoutButtonsRef.current) === null || _a === void 0 ? void 0 : _a.innerText) === null || _b === void 0 ? void 0 : _b.trim();
|
|
28761
|
+
if (renderedText) {
|
|
28762
|
+
return renderedText;
|
|
28763
|
+
}
|
|
28764
|
+
return trimmedMessageContent;
|
|
28765
|
+
}, [trimmedMessageContent]);
|
|
28766
|
+
/**
|
|
28767
|
+
* Fetches ElevenLabs speech audio (or replays cached audio) when the play button is pressed.
|
|
28768
|
+
*
|
|
28769
|
+
* @private
|
|
28770
|
+
*/
|
|
28771
|
+
const handlePlayMessage = react.useCallback(async () => {
|
|
28772
|
+
var _a;
|
|
28773
|
+
if (isAudioLoading) {
|
|
28774
|
+
return;
|
|
28775
|
+
}
|
|
28776
|
+
if (!shouldShowPlayButton) {
|
|
28777
|
+
setAudioError('Nothing to read aloud.');
|
|
28778
|
+
return;
|
|
28779
|
+
}
|
|
28780
|
+
const speechText = getMessageTextForSpeech();
|
|
28781
|
+
if (!speechText) {
|
|
28782
|
+
setAudioError('Nothing to read aloud.');
|
|
28783
|
+
return;
|
|
28784
|
+
}
|
|
28785
|
+
const payloadText = speechText.length > MAX_MESSAGE_SPEECH_LENGTH
|
|
28786
|
+
? speechText.slice(0, MAX_MESSAGE_SPEECH_LENGTH).trim()
|
|
28787
|
+
: speechText;
|
|
28788
|
+
if (!payloadText) {
|
|
28789
|
+
setAudioError('Nothing to read aloud.');
|
|
28790
|
+
return;
|
|
28791
|
+
}
|
|
28792
|
+
setAudioError(null);
|
|
28793
|
+
const playAudio = async (element) => {
|
|
28794
|
+
try {
|
|
28795
|
+
await element.play();
|
|
28796
|
+
}
|
|
28797
|
+
catch (playError) {
|
|
28798
|
+
setAudioError(playError instanceof Error ? playError.message : 'Browser blocked audio playback.');
|
|
28799
|
+
}
|
|
28800
|
+
};
|
|
28801
|
+
if (audioUrl) {
|
|
28802
|
+
const audio = (_a = audioRef.current) !== null && _a !== void 0 ? _a : new Audio(audioUrl);
|
|
28803
|
+
audioRef.current = audio;
|
|
28804
|
+
attachMessageAudioListeners(audio);
|
|
28805
|
+
if (audio.paused) {
|
|
28806
|
+
await playAudio(audio);
|
|
28807
|
+
}
|
|
28808
|
+
else {
|
|
28809
|
+
audio.pause();
|
|
28810
|
+
}
|
|
28811
|
+
return;
|
|
28812
|
+
}
|
|
28813
|
+
setIsAudioLoading(true);
|
|
28814
|
+
try {
|
|
28815
|
+
const response = await fetch('/api/elevenlabs/tts', {
|
|
28816
|
+
method: 'POST',
|
|
28817
|
+
headers: attachClientVersionHeader({
|
|
28818
|
+
'Content-Type': 'application/json',
|
|
28819
|
+
}),
|
|
28820
|
+
body: JSON.stringify({ text: payloadText }),
|
|
28821
|
+
});
|
|
28822
|
+
if (!response.ok) {
|
|
28823
|
+
const body = await response.text();
|
|
28824
|
+
throw new Error(body || 'Unable to request speech audio.');
|
|
28825
|
+
}
|
|
28826
|
+
const buffer = await response.arrayBuffer();
|
|
28827
|
+
const blob = new Blob([buffer], { type: 'audio/mpeg' });
|
|
28828
|
+
const url = URL.createObjectURL(blob);
|
|
28829
|
+
const audio = new Audio(url);
|
|
28830
|
+
audioRef.current = audio;
|
|
28831
|
+
attachMessageAudioListeners(audio);
|
|
28832
|
+
setAudioUrl((previousUrl) => {
|
|
28833
|
+
if (previousUrl) {
|
|
28834
|
+
URL.revokeObjectURL(previousUrl);
|
|
28835
|
+
}
|
|
28836
|
+
return url;
|
|
28837
|
+
});
|
|
28838
|
+
await playAudio(audio);
|
|
28839
|
+
}
|
|
28840
|
+
catch (error) {
|
|
28841
|
+
setAudioError(error instanceof Error ? error.message : 'Failed to generate speech.');
|
|
28842
|
+
}
|
|
28843
|
+
finally {
|
|
28844
|
+
setIsAudioLoading(false);
|
|
28845
|
+
}
|
|
28846
|
+
}, [attachMessageAudioListeners, audioUrl, getMessageTextForSpeech, isAudioLoading, shouldShowPlayButton]);
|
|
27147
28847
|
react.useEffect(() => {
|
|
27148
28848
|
if (!isExpanded) {
|
|
27149
28849
|
setLocalHoveredRating(0);
|
|
27150
28850
|
}
|
|
27151
28851
|
}, [isExpanded]);
|
|
27152
|
-
|
|
28852
|
+
react.useEffect(() => {
|
|
28853
|
+
return () => {
|
|
28854
|
+
var _a;
|
|
28855
|
+
(_a = audioRef.current) === null || _a === void 0 ? void 0 : _a.pause();
|
|
28856
|
+
audioRef.current = null;
|
|
28857
|
+
};
|
|
28858
|
+
}, []);
|
|
28859
|
+
react.useEffect(() => {
|
|
28860
|
+
return () => {
|
|
28861
|
+
if (audioUrl) {
|
|
28862
|
+
URL.revokeObjectURL(audioUrl);
|
|
28863
|
+
}
|
|
28864
|
+
};
|
|
28865
|
+
}, [audioUrl]);
|
|
27153
28866
|
react.useEffect(() => {
|
|
27154
28867
|
if (toolCallChipCount > toolCallChipCountRef.current) {
|
|
27155
28868
|
if (soundSystem) {
|
|
@@ -27180,54 +28893,64 @@
|
|
|
27180
28893
|
} }), isAvatarTooltipVisible && (participant === null || participant === void 0 ? void 0 : participant.agentSource) && avatarTooltipPosition && (jsxRuntime.jsx(AvatarProfileTooltip, { ref: tooltipRef, agentSource: participant.agentSource, position: avatarTooltipPosition }))] })), jsxRuntime.jsxs("div", { className: styles$4.messageStack, children: [shouldShowParticipantLabel && participantLabel && (jsxRuntime.jsx("div", { className: styles$4.participantLabel, children: participantLabel })), jsxRuntime.jsxs("div", { className: styles$4.messageText, style: {
|
|
27181
28894
|
'--message-bg-color': color.toHex(),
|
|
27182
28895
|
'--message-text-color': colorOfText.toHex(),
|
|
27183
|
-
}, children: [isCopyButtonEnabled && isComplete && (jsxRuntime.jsx("div", { className: styles$4.copyButtonContainer, children: jsxRuntime.jsxs("button", { className: styles$4.
|
|
27184
|
-
|
|
27185
|
-
|
|
27186
|
-
|
|
27187
|
-
|
|
27188
|
-
|
|
27189
|
-
|
|
27190
|
-
|
|
27191
|
-
|
|
27192
|
-
|
|
27193
|
-
|
|
27194
|
-
|
|
27195
|
-
clipboardItems['text/plain'] = new Blob([plain], {
|
|
27196
|
-
type: 'text/plain',
|
|
27197
|
-
});
|
|
27198
|
-
}
|
|
27199
|
-
await navigator.clipboard.write([new window.ClipboardItem(clipboardItems)]);
|
|
27200
|
-
setCopied(true);
|
|
27201
|
-
setTimeout(() => setCopied(false), 2000);
|
|
27202
|
-
// Tooltip positioning logic
|
|
27203
|
-
setTimeout(() => {
|
|
27204
|
-
const tooltip = copyTooltipRef.current;
|
|
27205
|
-
if (tooltip) {
|
|
27206
|
-
const rect = tooltip.getBoundingClientRect();
|
|
27207
|
-
if (rect.left < 8) {
|
|
27208
|
-
setTooltipAlign('left');
|
|
28896
|
+
}, children: [isCopyButtonEnabled && isComplete && (jsxRuntime.jsx("div", { className: styles$4.copyButtonContainer, children: jsxRuntime.jsxs("div", { className: styles$4.messageControlGroup, children: [shouldShowPlayButton && (jsxRuntime.jsx("button", { type: "button", className: styles$4.playButton, title: playButtonTitle, "aria-label": playButtonTitle, onClick: (event) => {
|
|
28897
|
+
event.stopPropagation();
|
|
28898
|
+
void handlePlayMessage();
|
|
28899
|
+
}, disabled: isAudioLoading, children: isAudioLoading ? (jsxRuntime.jsx("span", { className: styles$4.playButtonSpinner, "aria-hidden": "true" })) : isAudioPlaying ? (jsxRuntime.jsx(lucideReact.Pause, { className: styles$4.playButtonIcon, "aria-hidden": "true" })) : (jsxRuntime.jsx(lucideReact.Play, { className: styles$4.playButtonIcon, "aria-hidden": "true" })) })), jsxRuntime.jsxs("button", { className: styles$4.copyButton, title: "Copy message", onClick: async (e) => {
|
|
28900
|
+
e.stopPropagation();
|
|
28901
|
+
if (navigator.clipboard && window.ClipboardItem) {
|
|
28902
|
+
const clipboardItems = {};
|
|
28903
|
+
if (contentWithoutButtonsRef.current) {
|
|
28904
|
+
const html = contentWithoutButtonsRef.current.innerHTML;
|
|
28905
|
+
clipboardItems['text/html'] = new Blob([html], {
|
|
28906
|
+
type: 'text/html',
|
|
28907
|
+
});
|
|
27209
28908
|
}
|
|
27210
|
-
|
|
27211
|
-
|
|
28909
|
+
if (contentWithoutButtonsRef.current) {
|
|
28910
|
+
const plain = contentWithoutButtonsRef.current.innerText;
|
|
28911
|
+
clipboardItems['text/plain'] = new Blob([plain], {
|
|
28912
|
+
type: 'text/plain',
|
|
28913
|
+
});
|
|
27212
28914
|
}
|
|
27213
|
-
|
|
27214
|
-
|
|
28915
|
+
await navigator.clipboard.write([
|
|
28916
|
+
new window.ClipboardItem(clipboardItems),
|
|
28917
|
+
]);
|
|
28918
|
+
setCopied(true);
|
|
28919
|
+
setTimeout(() => setCopied(false), 2000);
|
|
28920
|
+
// Tooltip positioning logic
|
|
28921
|
+
setTimeout(() => {
|
|
28922
|
+
const tooltip = copyTooltipRef.current;
|
|
28923
|
+
if (tooltip) {
|
|
28924
|
+
const rect = tooltip.getBoundingClientRect();
|
|
28925
|
+
if (rect.left < 8) {
|
|
28926
|
+
setTooltipAlign('left');
|
|
28927
|
+
}
|
|
28928
|
+
else if (rect.right > window.innerWidth - 8) {
|
|
28929
|
+
setTooltipAlign('right');
|
|
28930
|
+
}
|
|
28931
|
+
else {
|
|
28932
|
+
setTooltipAlign('center');
|
|
28933
|
+
}
|
|
28934
|
+
}
|
|
28935
|
+
}, 10);
|
|
28936
|
+
if (typeof onCopy === 'function') {
|
|
28937
|
+
onCopy();
|
|
27215
28938
|
}
|
|
27216
28939
|
}
|
|
27217
|
-
|
|
27218
|
-
|
|
27219
|
-
|
|
27220
|
-
}
|
|
27221
|
-
|
|
27222
|
-
|
|
27223
|
-
|
|
27224
|
-
|
|
27225
|
-
|
|
27226
|
-
|
|
27227
|
-
|
|
27228
|
-
|
|
27229
|
-
|
|
27230
|
-
|
|
28940
|
+
else {
|
|
28941
|
+
throw new Error(`Your browser does not support copying to clipboard: navigator.clipboard && window.ClipboardItem.`);
|
|
28942
|
+
}
|
|
28943
|
+
}, children: [jsxRuntime.jsxs("svg", { width: "18", height: "18", viewBox: "0 0 24 24", fill: "none", children: [jsxRuntime.jsx("rect", { x: "7", y: "7", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" }), jsxRuntime.jsx("rect", { x: "3", y: "3", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" })] }), copied && (jsxRuntime.jsx("span", { ref: copyTooltipRef, className: styles$4.copiedTooltip +
|
|
28944
|
+
(tooltipAlign === 'left'
|
|
28945
|
+
? ' ' + styles$4.copiedTooltipLeft
|
|
28946
|
+
: tooltipAlign === 'right'
|
|
28947
|
+
? ' ' + styles$4.copiedTooltipRight
|
|
28948
|
+
: ''), children: "Copied!" }))] })] }) })), message.isVoiceCall && (jsxRuntime.jsx("div", { className: styles$4.voiceCallIndicator, children: jsxRuntime.jsx("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "currentColor", children: jsxRuntime.jsx("path", { d: "M6.62 10.79c1.44 2.83 3.76 5.14 6.59 6.59l2.2-2.2c.27-.27.67-.36 1.02-.24 1.12.37 2.33.57 3.57.57.55 0 1 .45 1 1V20c0 .55-.45 1-1 1-9.39 0-17-7.61-17-17 0-.55.45-1 1-1h3.5c.55 0 1 .45 1 1 0 1.25.2 2.45.57 3.57.11.35.03.74-.25 1.02l-2.2 2.2z" }) }) })), message.content === LOADING_INTERACTIVE_IMAGE ? (jsxRuntime.jsx(jsxRuntime.Fragment, {})) : (jsxRuntime.jsx("div", { ref: contentWithoutButtonsRef, children: contentSegments.map((segment, segmentIndex) => {
|
|
28949
|
+
if (segment.type === 'text') {
|
|
28950
|
+
return (jsxRuntime.jsx(MarkdownContent, { content: segment.content, onCreateAgent: onCreateAgent }, `text-${segmentIndex}`));
|
|
28951
|
+
}
|
|
28952
|
+
return (jsxRuntime.jsx(ImagePromptRenderer, { alt: segment.alt, prompt: segment.prompt }, `image-${segmentIndex}`));
|
|
28953
|
+
}) })), message.attachments && message.attachments.length > 0 && (jsxRuntime.jsx("div", { className: styles$4.attachments, children: message.attachments.map((attachment, index) => (jsxRuntime.jsxs("a", { href: attachment.url, target: "_blank", rel: "noopener noreferrer", className: styles$4.attachment, title: attachment.name, children: [jsxRuntime.jsx("span", { className: styles$4.attachmentIcon, children: "\uD83D\uDCCE" }), jsxRuntime.jsx("span", { className: styles$4.attachmentName, children: attachment.name })] }, index))) })), completedToolCalls && completedToolCalls.length > 0 && (jsxRuntime.jsxs("div", { className: styles$4.completedToolCalls, children: [completedToolCalls.map((toolCall, index) => {
|
|
27231
28954
|
const chipletInfo = getToolCallChipletInfo(toolCall);
|
|
27232
28955
|
const chipletText = buildToolCallChipText(chipletInfo);
|
|
27233
28956
|
const teamAgentData = resolveTeamAgentChipData(toolCall, teammates, chipletInfo);
|
|
@@ -27775,21 +29498,6 @@
|
|
|
27775
29498
|
onMessage: isSimulationComplete && chatProps.onMessage ? chatProps.onMessage : undefined }));
|
|
27776
29499
|
}
|
|
27777
29500
|
|
|
27778
|
-
/**
|
|
27779
|
-
* Renders a nice clock icon with a specific time.
|
|
27780
|
-
*
|
|
27781
|
-
* @private internal subcomponent of `<Chat>` component
|
|
27782
|
-
*/
|
|
27783
|
-
function ClockIcon({ date, size = 100 }) {
|
|
27784
|
-
const hours = date.getHours();
|
|
27785
|
-
const minutes = date.getMinutes();
|
|
27786
|
-
const seconds = date.getSeconds();
|
|
27787
|
-
const hourDeg = (hours % 12) * 30 + minutes * 0.5;
|
|
27788
|
-
const minuteDeg = minutes * 6;
|
|
27789
|
-
const secondDeg = seconds * 6;
|
|
27790
|
-
return (jsxRuntime.jsxs("svg", { width: size, height: size, viewBox: "0 0 100 100", style: { filter: 'drop-shadow(0 2px 4px rgba(0,0,0,0.1))' }, children: [jsxRuntime.jsx("circle", { cx: "50", cy: "50", r: "45", fill: "white", stroke: "#333", strokeWidth: "2" }), [...Array(12)].map((_, i) => (jsxRuntime.jsx("line", { x1: "50", y1: "12", x2: "50", y2: "18", transform: `rotate(${i * 30} 50 50)`, stroke: "#333", strokeWidth: "2" }, i))), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "25", transform: `rotate(${hourDeg} 50 50)`, stroke: "#333", strokeWidth: "4", strokeLinecap: "round" }), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "15", transform: `rotate(${minuteDeg} 50 50)`, stroke: "#666", strokeWidth: "3", strokeLinecap: "round" }), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "12", transform: `rotate(${secondDeg} 50 50)`, stroke: "#ff4444", strokeWidth: "1.5", strokeLinecap: "round" }), jsxRuntime.jsx("circle", { cx: "50", cy: "50", r: "2", fill: "#333" })] }));
|
|
27791
|
-
}
|
|
27792
|
-
|
|
27793
29501
|
/**
|
|
27794
29502
|
* Parses ISO timestamps into Date instances.
|
|
27795
29503
|
*
|
|
@@ -27878,6 +29586,21 @@
|
|
|
27878
29586
|
return (jsxRuntime.jsx("div", { className: classNames(styles$4.selfLearningAvatar, className), style: avatarStyle, role: "img", "aria-label": label, title: label, children: !avatarSrc && (children || jsxRuntime.jsx("span", { className: styles$4.selfLearningAvatarInitial, children: initial })) }));
|
|
27879
29587
|
}
|
|
27880
29588
|
|
|
29589
|
+
/**
|
|
29590
|
+
* Renders a nice clock icon with a specific time.
|
|
29591
|
+
*
|
|
29592
|
+
* @private internal subcomponent of `<Chat>` component
|
|
29593
|
+
*/
|
|
29594
|
+
function ClockIcon({ date, size = 100 }) {
|
|
29595
|
+
const hours = date.getHours();
|
|
29596
|
+
const minutes = date.getMinutes();
|
|
29597
|
+
const seconds = date.getSeconds();
|
|
29598
|
+
const hourDeg = (hours % 12) * 30 + minutes * 0.5;
|
|
29599
|
+
const minuteDeg = minutes * 6;
|
|
29600
|
+
const secondDeg = seconds * 6;
|
|
29601
|
+
return (jsxRuntime.jsxs("svg", { width: size, height: size, viewBox: "0 0 100 100", style: { filter: 'drop-shadow(0 2px 4px rgba(0,0,0,0.1))' }, children: [jsxRuntime.jsx("circle", { cx: "50", cy: "50", r: "45", fill: "white", stroke: "#333", strokeWidth: "2" }), [...Array(12)].map((_, i) => (jsxRuntime.jsx("line", { x1: "50", y1: "12", x2: "50", y2: "18", transform: `rotate(${i * 30} 50 50)`, stroke: "#333", strokeWidth: "2" }, i))), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "25", transform: `rotate(${hourDeg} 50 50)`, stroke: "#333", strokeWidth: "4", strokeLinecap: "round" }), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "15", transform: `rotate(${minuteDeg} 50 50)`, stroke: "#666", strokeWidth: "3", strokeLinecap: "round" }), jsxRuntime.jsx("line", { x1: "50", y1: "50", x2: "50", y2: "12", transform: `rotate(${secondDeg} 50 50)`, stroke: "#ff4444", strokeWidth: "1.5", strokeLinecap: "round" }), jsxRuntime.jsx("circle", { cx: "50", cy: "50", r: "2", fill: "#333" })] }));
|
|
29602
|
+
}
|
|
29603
|
+
|
|
27881
29604
|
/**
|
|
27882
29605
|
* Modal that renders rich tool call details for chat chiplets.
|
|
27883
29606
|
*
|
|
@@ -28035,6 +29758,199 @@
|
|
|
28035
29758
|
}
|
|
28036
29759
|
}, children: jsxRuntime.jsxs("div", { className: classNames(styles$4.ratingModalContent, styles$4.toolCallModal), children: [jsxRuntime.jsx("button", { type: "button", className: styles$4.modalCloseButton, onClick: onClose, "aria-label": "Close dialog", children: jsxRuntime.jsx(CloseIcon, {}) }), modalContent] }) }));
|
|
28037
29760
|
}
|
|
29761
|
+
/**
|
|
29762
|
+
* Maximum number of memory cards rendered inside the modal at once.
|
|
29763
|
+
*
|
|
29764
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29765
|
+
*/
|
|
29766
|
+
const MEMORY_DISPLAY_LIMIT = 3;
|
|
29767
|
+
/**
|
|
29768
|
+
* CSS classes mapped by memory status tone.
|
|
29769
|
+
*
|
|
29770
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29771
|
+
*/
|
|
29772
|
+
const MEMORY_STATUS_CLASS_BY_TONE = {
|
|
29773
|
+
success: styles$4.memoryStatusSuccess,
|
|
29774
|
+
warning: styles$4.memoryStatusWarning,
|
|
29775
|
+
error: styles$4.memoryStatusError,
|
|
29776
|
+
neutral: styles$4.memoryStatusNeutral,
|
|
29777
|
+
};
|
|
29778
|
+
/**
|
|
29779
|
+
* Renders a friendly memory summary screen when a MEMORY tool call is selected.
|
|
29780
|
+
*
|
|
29781
|
+
* @param options - View fragments required to render the memory modal.
|
|
29782
|
+
* @returns Memory-specific modal JSX or `null` when the tool is unrelated.
|
|
29783
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29784
|
+
*/
|
|
29785
|
+
function renderMemoryToolCall(options) {
|
|
29786
|
+
var _a;
|
|
29787
|
+
const { toolCall, args, resultRaw, toolCallDate } = options;
|
|
29788
|
+
if (toolCall.name !== 'retrieve_user_memory' && toolCall.name !== 'store_user_memory') {
|
|
29789
|
+
return null;
|
|
29790
|
+
}
|
|
29791
|
+
const isStoreAction = toolCall.name === 'store_user_memory';
|
|
29792
|
+
const memoryResult = buildMemoryToolResult(resultRaw, isStoreAction ? 'store' : 'retrieve');
|
|
29793
|
+
if (!memoryResult) {
|
|
29794
|
+
return null;
|
|
29795
|
+
}
|
|
29796
|
+
const heroTitle = isStoreAction ? 'Memory saved' : 'Memories retrieved';
|
|
29797
|
+
const heroSubtitle = isStoreAction
|
|
29798
|
+
? 'This detail is now stored so future chats will remember it.'
|
|
29799
|
+
: 'The agent pulled these facts from the memory vault.';
|
|
29800
|
+
const callTimeLabel = (_a = toolCallDate === null || toolCallDate === void 0 ? void 0 : toolCallDate.toLocaleString(undefined, {
|
|
29801
|
+
dateStyle: 'medium',
|
|
29802
|
+
timeStyle: 'short',
|
|
29803
|
+
})) !== null && _a !== void 0 ? _a : 'Call time unavailable';
|
|
29804
|
+
const statusInfo = buildMemoryStatusInfo(memoryResult.status, memoryResult.action);
|
|
29805
|
+
const statusClass = MEMORY_STATUS_CLASS_BY_TONE[statusInfo.tone] || styles$4.memoryStatusNeutral;
|
|
29806
|
+
return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { className: styles$4.memoryModalHeader, children: [jsxRuntime.jsx("div", { className: styles$4.memoryModalIcon, children: "\uD83E\uDDE0" }), jsxRuntime.jsxs("div", { className: styles$4.memoryModalHeaderText, children: [jsxRuntime.jsx("h3", { className: styles$4.memoryModalTitle, children: heroTitle }), jsxRuntime.jsx("p", { className: styles$4.memoryModalSubtitle, children: heroSubtitle }), jsxRuntime.jsxs("p", { className: styles$4.memoryModalCallTime, children: ["Call recorded ", callTimeLabel] })] }), jsxRuntime.jsxs("div", { className: classNames(styles$4.memoryModalStatus, statusClass), children: [jsxRuntime.jsx("span", { className: styles$4.memoryStatusDot }), statusInfo.label] })] }), jsxRuntime.jsxs("div", { className: styles$4.memoryModalContent, children: [memoryResult.message && jsxRuntime.jsx("p", { className: styles$4.memoryMessage, children: memoryResult.message }), isStoreAction
|
|
29807
|
+
? renderMemoryStoreSection({ memoryResult, args })
|
|
29808
|
+
: renderMemoryRetrieveSection({ memoryResult, args })] })] }));
|
|
29809
|
+
}
|
|
29810
|
+
/**
|
|
29811
|
+
* Renders the stored memory detail pane.
|
|
29812
|
+
*
|
|
29813
|
+
* @param options - Store action payload and arguments.
|
|
29814
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29815
|
+
*/
|
|
29816
|
+
function renderMemoryStoreSection(options) {
|
|
29817
|
+
var _a, _b, _c, _d, _e, _f, _g;
|
|
29818
|
+
const { memoryResult, args } = options;
|
|
29819
|
+
const storedScope = (_b = (_a = memoryResult.memory) === null || _a === void 0 ? void 0 : _a.isGlobal) !== null && _b !== void 0 ? _b : (typeof args.isGlobal === 'boolean' ? args.isGlobal : false);
|
|
29820
|
+
const scopeLabel = storedScope ? 'Global memory' : 'Personal memory';
|
|
29821
|
+
const scopeBadge = storedScope ? 'Global' : 'Personal';
|
|
29822
|
+
const storedContent = ((_d = (_c = memoryResult.memory) === null || _c === void 0 ? void 0 : _c.content) === null || _d === void 0 ? void 0 : _d.trim()) || (typeof args.content === 'string' ? args.content.trim() : '');
|
|
29823
|
+
const timestamp = (_f = formatMemoryTimestamp((_e = memoryResult.memory) === null || _e === void 0 ? void 0 : _e.updatedAt)) !== null && _f !== void 0 ? _f : formatMemoryTimestamp((_g = memoryResult.memory) === null || _g === void 0 ? void 0 : _g.createdAt);
|
|
29824
|
+
return (jsxRuntime.jsxs("div", { className: styles$4.memoryStoreSection, children: [jsxRuntime.jsxs("div", { className: styles$4.memoryMetaRow, children: [jsxRuntime.jsx("span", { className: styles$4.memoryMetaLabel, children: "Scope" }), jsxRuntime.jsx("span", { className: styles$4.memoryMetaValue, children: scopeLabel })] }), jsxRuntime.jsxs("div", { className: styles$4.memoryCard, children: [jsxRuntime.jsx("div", { className: styles$4.memoryCardContent, children: storedContent || 'No memory content was provided for this call.' }), jsxRuntime.jsxs("div", { className: styles$4.memoryCardMeta, children: [jsxRuntime.jsx("span", { className: styles$4.memoryScopeBadge, children: scopeBadge }), timestamp && jsxRuntime.jsx("span", { children: timestamp })] })] })] }));
|
|
29825
|
+
}
|
|
29826
|
+
/**
|
|
29827
|
+
* Renders the retrieved memories list.
|
|
29828
|
+
*
|
|
29829
|
+
* @param options - Retrieve action payload and arguments.
|
|
29830
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29831
|
+
*/
|
|
29832
|
+
function renderMemoryRetrieveSection(options) {
|
|
29833
|
+
var _a;
|
|
29834
|
+
const { memoryResult, args } = options;
|
|
29835
|
+
const queryLabel = ((_a = memoryResult.query) === null || _a === void 0 ? void 0 : _a.trim()) || (typeof args.query === 'string' ? args.query.trim() : '');
|
|
29836
|
+
const memories = (memoryResult.memories || [])
|
|
29837
|
+
.map((entry) => entry && normalizeMemoryRecord(entry))
|
|
29838
|
+
.filter((entry) => Boolean(entry && entry.content && entry.content.trim().length > 0));
|
|
29839
|
+
const displayedMemories = memories.slice(0, MEMORY_DISPLAY_LIMIT);
|
|
29840
|
+
const extraCount = memories.length - displayedMemories.length;
|
|
29841
|
+
return (jsxRuntime.jsxs("div", { className: styles$4.memoryRetrieveSection, children: [queryLabel && (jsxRuntime.jsxs("div", { className: styles$4.memoryMetaRow, children: [jsxRuntime.jsx("span", { className: styles$4.memoryMetaLabel, children: "Search" }), jsxRuntime.jsxs("span", { className: styles$4.memoryMetaValue, children: ["\u201C", queryLabel, "\u201D"] })] })), jsxRuntime.jsxs("div", { className: styles$4.memoryMetaRow, children: [jsxRuntime.jsx("span", { className: styles$4.memoryMetaLabel, children: "Matches" }), jsxRuntime.jsx("span", { className: styles$4.memoryMetaValue, children: memories.length })] }), memories.length === 0 ? (jsxRuntime.jsx("div", { className: styles$4.memoryEmptyState, children: memoryResult.message ||
|
|
29842
|
+
(queryLabel
|
|
29843
|
+
? `No memories match “${queryLabel}”.`
|
|
29844
|
+
: 'No memories were available for this conversation.') })) : (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx("div", { className: styles$4.memoryList, children: displayedMemories.map((memory, index) => {
|
|
29845
|
+
var _a;
|
|
29846
|
+
const timestamp = (_a = formatMemoryTimestamp(memory.updatedAt)) !== null && _a !== void 0 ? _a : formatMemoryTimestamp(memory.createdAt);
|
|
29847
|
+
return (jsxRuntime.jsxs("div", { className: styles$4.memoryCard, children: [jsxRuntime.jsx("div", { className: styles$4.memoryCardContent, children: memory.content }), jsxRuntime.jsxs("div", { className: styles$4.memoryCardMeta, children: [jsxRuntime.jsx("span", { className: styles$4.memoryScopeBadge, children: memory.isGlobal ? 'Global' : 'Personal' }), timestamp && jsxRuntime.jsx("span", { children: timestamp })] })] }, memory.id || `${memory.content}-${index}`));
|
|
29848
|
+
}) }), extraCount > 0 && (jsxRuntime.jsxs("div", { className: styles$4.memoryListFooter, children: [extraCount, " more ", extraCount === 1 ? 'memory' : 'memories', " available."] }))] }))] }));
|
|
29849
|
+
}
|
|
29850
|
+
/**
|
|
29851
|
+
* Transforms the raw tool result (or fallback data) into a normalized memory payload.
|
|
29852
|
+
*
|
|
29853
|
+
* @param raw - Raw data returned by the tool call.
|
|
29854
|
+
* @param fallbackAction - Action to assume when the payload does not declare one.
|
|
29855
|
+
* @returns Normalized memory details or `null` when no payload exists.
|
|
29856
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29857
|
+
*/
|
|
29858
|
+
function buildMemoryToolResult(raw, fallbackAction) {
|
|
29859
|
+
var _a, _b;
|
|
29860
|
+
if (raw && typeof raw === 'object') {
|
|
29861
|
+
const normalizedMemories = Array.isArray(raw.memories)
|
|
29862
|
+
? raw.memories
|
|
29863
|
+
.map((entry) => entry && normalizeMemoryRecord(entry))
|
|
29864
|
+
.filter((entry) => Boolean(entry))
|
|
29865
|
+
: [];
|
|
29866
|
+
const normalizedMemory = raw.memory ? (_a = normalizeMemoryRecord(raw.memory)) !== null && _a !== void 0 ? _a : undefined : undefined;
|
|
29867
|
+
return {
|
|
29868
|
+
action: raw.action === 'store' ? 'store' : fallbackAction,
|
|
29869
|
+
status: (_b = (typeof raw.status === 'string' ? raw.status : undefined)) !== null && _b !== void 0 ? _b : (fallbackAction === 'store' ? 'stored' : 'ok'),
|
|
29870
|
+
message: typeof raw.message === 'string' ? raw.message : undefined,
|
|
29871
|
+
query: typeof raw.query === 'string' ? raw.query : undefined,
|
|
29872
|
+
memory: normalizedMemory,
|
|
29873
|
+
memories: normalizedMemories.length > 0 ? normalizedMemories : undefined,
|
|
29874
|
+
};
|
|
29875
|
+
}
|
|
29876
|
+
if (typeof raw === 'string') {
|
|
29877
|
+
return {
|
|
29878
|
+
action: fallbackAction,
|
|
29879
|
+
status: 'error',
|
|
29880
|
+
message: raw,
|
|
29881
|
+
};
|
|
29882
|
+
}
|
|
29883
|
+
return {
|
|
29884
|
+
action: fallbackAction,
|
|
29885
|
+
status: fallbackAction === 'store' ? 'stored' : 'ok',
|
|
29886
|
+
};
|
|
29887
|
+
}
|
|
29888
|
+
/**
|
|
29889
|
+
* Normalizes a memory record payload.
|
|
29890
|
+
*
|
|
29891
|
+
* @param entry - Input record from the memory tool.
|
|
29892
|
+
* @returns Normalized record or `null` when the entry is invalid.
|
|
29893
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29894
|
+
*/
|
|
29895
|
+
function normalizeMemoryRecord(entry) {
|
|
29896
|
+
if (!entry || typeof entry !== 'object') {
|
|
29897
|
+
return null;
|
|
29898
|
+
}
|
|
29899
|
+
return {
|
|
29900
|
+
id: typeof entry.id === 'string' ? entry.id : undefined,
|
|
29901
|
+
content: typeof entry.content === 'string' ? entry.content.trim() : undefined,
|
|
29902
|
+
isGlobal: entry.isGlobal === true,
|
|
29903
|
+
createdAt: typeof entry.createdAt === 'string' ? entry.createdAt : undefined,
|
|
29904
|
+
updatedAt: typeof entry.updatedAt === 'string' ? entry.updatedAt : undefined,
|
|
29905
|
+
};
|
|
29906
|
+
}
|
|
29907
|
+
/**
|
|
29908
|
+
* Builds friendly status text and tone for memory actions.
|
|
29909
|
+
*
|
|
29910
|
+
* @param status - Raw status string returned by the tool.
|
|
29911
|
+
* @param action - Optional action to better describe neutral statuses.
|
|
29912
|
+
* @returns Label and tone for badge styling.
|
|
29913
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29914
|
+
*/
|
|
29915
|
+
function buildMemoryStatusInfo(status, action) {
|
|
29916
|
+
const normalized = (status || (action === 'store' ? 'stored' : 'ok')).toString().toLowerCase();
|
|
29917
|
+
if (normalized === 'stored') {
|
|
29918
|
+
return { label: 'Saved', tone: 'success' };
|
|
29919
|
+
}
|
|
29920
|
+
if (normalized === 'ok') {
|
|
29921
|
+
return { label: 'Loaded', tone: 'success' };
|
|
29922
|
+
}
|
|
29923
|
+
if (normalized === 'disabled') {
|
|
29924
|
+
return { label: 'Memory disabled', tone: 'warning' };
|
|
29925
|
+
}
|
|
29926
|
+
if (normalized === 'error') {
|
|
29927
|
+
return { label: 'Something went wrong', tone: 'error' };
|
|
29928
|
+
}
|
|
29929
|
+
return {
|
|
29930
|
+
label: action === 'store' ? 'Memory saved' : 'Memory retrieved',
|
|
29931
|
+
tone: 'neutral',
|
|
29932
|
+
};
|
|
29933
|
+
}
|
|
29934
|
+
/**
|
|
29935
|
+
* Formats ISO timestamps returned by memory records.
|
|
29936
|
+
*
|
|
29937
|
+
* @param value - Potential ISO timestamp string.
|
|
29938
|
+
* @returns Formatted label or `null` when the timestamp is invalid.
|
|
29939
|
+
* @private internal utility of `<ChatToolCallModal/>`
|
|
29940
|
+
*/
|
|
29941
|
+
function formatMemoryTimestamp(value) {
|
|
29942
|
+
if (!value) {
|
|
29943
|
+
return null;
|
|
29944
|
+
}
|
|
29945
|
+
const parsed = new Date(value);
|
|
29946
|
+
if (Number.isNaN(parsed.getTime())) {
|
|
29947
|
+
return null;
|
|
29948
|
+
}
|
|
29949
|
+
return parsed.toLocaleString(undefined, {
|
|
29950
|
+
dateStyle: 'medium',
|
|
29951
|
+
timeStyle: 'short',
|
|
29952
|
+
});
|
|
29953
|
+
}
|
|
28038
29954
|
/**
|
|
28039
29955
|
* Renders the detail view for a single tool call.
|
|
28040
29956
|
*
|
|
@@ -28045,6 +29961,15 @@
|
|
|
28045
29961
|
const resultRaw = parseToolCallResult(toolCall.result);
|
|
28046
29962
|
const args = parseToolCallArguments(toolCall);
|
|
28047
29963
|
const toolCallDate = getToolCallTimestamp(toolCall);
|
|
29964
|
+
const memoryView = renderMemoryToolCall({
|
|
29965
|
+
toolCall,
|
|
29966
|
+
args,
|
|
29967
|
+
resultRaw,
|
|
29968
|
+
toolCallDate,
|
|
29969
|
+
});
|
|
29970
|
+
if (memoryView) {
|
|
29971
|
+
return memoryView;
|
|
29972
|
+
}
|
|
28048
29973
|
const isSearch = toolCall.name === 'web_search' || toolCall.name === 'useSearchEngine' || toolCall.name === 'search';
|
|
28049
29974
|
const isTime = toolCall.name === 'get_current_time' || toolCall.name === 'useTime';
|
|
28050
29975
|
const isEmail = toolCall.name === 'send_email' || toolCall.name === 'useEmail';
|
|
@@ -28099,7 +30024,223 @@
|
|
|
28099
30024
|
const status = typeof (emailResult === null || emailResult === void 0 ? void 0 : emailResult.status) === 'string' ? emailResult.status : null;
|
|
28100
30025
|
return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { className: classNames(styles$4.searchModalHeader, styles$4.emailModalHeader), children: [jsxRuntime.jsx("span", { className: styles$4.searchModalIcon, children: jsxRuntime.jsx(EmailIcon, { size: 26 }) }), jsxRuntime.jsxs("div", { className: styles$4.emailHeaderText, children: [jsxRuntime.jsx("span", { className: styles$4.emailHeaderLabel, children: "Email" }), jsxRuntime.jsx("h3", { className: styles$4.searchModalQuery, children: subject })] })] }), jsxRuntime.jsxs("div", { className: styles$4.searchModalContent, children: [jsxRuntime.jsxs("div", { className: styles$4.emailContainer, children: [jsxRuntime.jsxs("div", { className: styles$4.emailMetadata, children: [jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "From:" }), jsxRuntime.jsx("span", { className: styles$4.emailRecipients, children: from })] }), jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "To:" }), jsxRuntime.jsx("span", { className: styles$4.emailRecipients, children: recipients.join(', ') })] }), ccRecipients.length > 0 && (jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "CC:" }), jsxRuntime.jsx("span", { className: styles$4.emailRecipients, children: ccRecipients.join(', ') })] })), jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "Subject:" }), jsxRuntime.jsx("span", { children: subject })] }), sentAt && (jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "Sent:" }), jsxRuntime.jsx("span", { children: sentAt })] })), status && (jsxRuntime.jsxs("div", { className: styles$4.emailField, children: [jsxRuntime.jsx("strong", { children: "Status:" }), jsxRuntime.jsx("span", { className: styles$4.emailStatus, children: status })] }))] }), jsxRuntime.jsxs("div", { className: styles$4.emailBody, children: [jsxRuntime.jsx("strong", { children: "Message:" }), jsxRuntime.jsx("div", { className: styles$4.emailBodyContent, children: jsxRuntime.jsx(MarkdownContent, { content: body }) })] })] }), jsxRuntime.jsxs("div", { className: styles$4.toolCallDetails, children: [jsxRuntime.jsx("p", { children: jsxRuntime.jsx("strong", { children: "Result:" }) }), jsxRuntime.jsx("div", { className: styles$4.toolCallDataContainer, children: jsxRuntime.jsx("pre", { className: styles$4.toolCallData, children: typeof resultRaw === 'object' ? JSON.stringify(resultRaw, null, 2) : String(resultRaw) }) })] })] })] }));
|
|
28101
30026
|
}
|
|
28102
|
-
|
|
30027
|
+
const chipletInfo = getToolCallChipletInfo(toolCall);
|
|
30028
|
+
const toolMetadata = TOOL_TITLES[toolCall.name];
|
|
30029
|
+
const headerEmoji = (toolMetadata === null || toolMetadata === void 0 ? void 0 : toolMetadata.emoji) || extractLeadingEmoji(chipletInfo.text) || '🛠️';
|
|
30030
|
+
const headerTitle = (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[toolCall.name]) || (toolMetadata === null || toolMetadata === void 0 ? void 0 : toolMetadata.title) || chipletInfo.text || toolCall.name;
|
|
30031
|
+
const callTimestamp = toolCallDate
|
|
30032
|
+
? toolCallDate.toLocaleString(undefined, { dateStyle: 'medium', timeStyle: 'short' })
|
|
30033
|
+
: 'Time unavailable';
|
|
30034
|
+
const relativeTimestamp = toolCallDate ? moment__default["default"](toolCallDate).fromNow() : null;
|
|
30035
|
+
const argumentEntries = buildArgumentEntries(args);
|
|
30036
|
+
const resultSummary = buildToolCallResultSummary(resultRaw);
|
|
30037
|
+
const resultKindLabel = describeResultKind(resultRaw);
|
|
30038
|
+
const resultCount = getResultItemCount(resultRaw);
|
|
30039
|
+
const toolCallIssues = normalizeToolCallIssues(toolCall);
|
|
30040
|
+
return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("header", { className: styles$4.toolCallHeader, children: [jsxRuntime.jsx("span", { className: styles$4.toolCallIcon, "aria-hidden": "true", children: headerEmoji }), jsxRuntime.jsxs("div", { className: styles$4.toolCallHeaderMeta, children: [jsxRuntime.jsx("p", { className: styles$4.toolCallLabel, children: "Tool call" }), jsxRuntime.jsx("h3", { className: styles$4.toolCallTitle, children: headerTitle }), jsxRuntime.jsxs("p", { className: styles$4.toolCallSubtitle, children: [relativeTimestamp ? `Executed ${relativeTimestamp}` : 'Executed just now', " \u00B7 ", callTimestamp] })] })] }), jsxRuntime.jsxs("div", { className: styles$4.toolCallGrid, children: [jsxRuntime.jsxs("section", { className: styles$4.toolCallPanel, children: [jsxRuntime.jsx("p", { className: styles$4.toolCallPanelTitle, children: "What the agent requested" }), argumentEntries.length > 0 ? (jsxRuntime.jsx("ul", { className: styles$4.toolCallList, children: argumentEntries.map((entry) => (jsxRuntime.jsxs("li", { className: styles$4.toolCallItem, children: [jsxRuntime.jsx("span", { className: styles$4.toolCallItemLabel, children: entry.label }), jsxRuntime.jsx("span", { className: styles$4.toolCallItemValue, children: entry.value })] }, entry.label))) })) : (jsxRuntime.jsx("p", { className: styles$4.toolCallEmpty, children: "No inputs were provided for this tool call." }))] }), jsxRuntime.jsxs("section", { className: styles$4.toolCallPanel, children: [jsxRuntime.jsx("p", { className: styles$4.toolCallPanelTitle, children: "What happened" }), resultSummary ? (jsxRuntime.jsx("p", { className: styles$4.toolCallSummary, children: resultSummary })) : (jsxRuntime.jsx("p", { className: styles$4.toolCallEmpty, children: "The tool responded, but no friendly summary is available." })), jsxRuntime.jsxs("div", { className: styles$4.toolCallSummaryMeta, children: [jsxRuntime.jsx("span", { className: styles$4.toolCallSummaryMetaBadge, children: resultKindLabel }), typeof resultCount === 'number' && (jsxRuntime.jsxs("span", { className: styles$4.toolCallSummaryMetaBadge, children: ["Returned ", resultCount, " ", resultCount === 1 ? 'item' : 'items'] }))] })] })] }), toolCallIssues.length > 0 && (jsxRuntime.jsx("div", { className: styles$4.toolCallIssues, children: toolCallIssues.map((issue, index) => (jsxRuntime.jsxs("span", { className: classNames(styles$4.toolCallIssueBadge, issue.type === 'warning' ? styles$4.toolCallIssueWarning : styles$4.toolCallIssueError), children: [jsxRuntime.jsx("strong", { children: issue.label }), ": ", issue.message] }, `${issue.type}-${index}`))) })), jsxRuntime.jsxs("footer", { className: styles$4.toolCallFooter, children: [jsxRuntime.jsx("span", { className: styles$4.toolCallTimestamp, children: toolCallDate ? `Recorded ${toolCallDate.toLocaleString()}` : 'Timestamp unavailable' }), jsxRuntime.jsxs("details", { className: styles$4.toolCallRawDetails, children: [jsxRuntime.jsx("summary", { className: styles$4.toolCallRawSummary, children: "View raw payload" }), jsxRuntime.jsxs("div", { className: styles$4.toolCallRawColumns, children: [jsxRuntime.jsxs("div", { className: styles$4.toolCallRawColumn, children: [jsxRuntime.jsx("h4", { className: styles$4.toolCallRawColumnTitle, children: "Arguments" }), jsxRuntime.jsx("div", { className: styles$4.toolCallDataContainer, children: argumentEntries.length > 0 ? (jsxRuntime.jsx("ul", { className: styles$4.toolCallArgsList, children: argumentEntries.map((entry) => (jsxRuntime.jsxs("li", { children: [jsxRuntime.jsxs("strong", { children: [entry.label, ":"] }), " ", entry.value] }, `raw-${entry.label}`))) })) : (jsxRuntime.jsx("pre", { className: styles$4.toolCallData, children: "No arguments provided." })) })] }), jsxRuntime.jsxs("div", { className: styles$4.toolCallRawColumn, children: [jsxRuntime.jsx("h4", { className: styles$4.toolCallRawColumnTitle, children: "Result" }), jsxRuntime.jsx("div", { className: styles$4.toolCallDataContainer, children: jsxRuntime.jsx("pre", { className: styles$4.toolCallData, children: typeof resultRaw === 'object'
|
|
30041
|
+
? JSON.stringify(resultRaw, null, 2)
|
|
30042
|
+
: String(resultRaw !== null && resultRaw !== void 0 ? resultRaw : 'No result') }) })] })] })] })] })] }));
|
|
30043
|
+
}
|
|
30044
|
+
/**
|
|
30045
|
+
* Builds a list of argument entries for the friendly summary view.
|
|
30046
|
+
*
|
|
30047
|
+
* @param args - Parsed tool call arguments.
|
|
30048
|
+
* @returns Array of display-ready argument entries.
|
|
30049
|
+
*/
|
|
30050
|
+
function buildArgumentEntries(args) {
|
|
30051
|
+
return Object.entries(args).map(([key, value]) => ({
|
|
30052
|
+
label: formatArgumentLabel(key),
|
|
30053
|
+
value: formatArgumentValue(value),
|
|
30054
|
+
}));
|
|
30055
|
+
}
|
|
30056
|
+
/**
|
|
30057
|
+
* Normalizes a tool call argument key into human-readable text.
|
|
30058
|
+
*
|
|
30059
|
+
* @param key - Raw argument key.
|
|
30060
|
+
* @returns Humanized label.
|
|
30061
|
+
*/
|
|
30062
|
+
function formatArgumentLabel(key) {
|
|
30063
|
+
const replaced = key.replace(/_/g, ' ').replace(/([a-z])([A-Z])/g, '$1 $2');
|
|
30064
|
+
return replaced.charAt(0).toUpperCase() + replaced.slice(1);
|
|
30065
|
+
}
|
|
30066
|
+
/**
|
|
30067
|
+
* Converts a value into a display-friendly string without exposing raw JSON.
|
|
30068
|
+
*
|
|
30069
|
+
* @param value - Arbitrary tool call argument value.
|
|
30070
|
+
* @returns Friendly string.
|
|
30071
|
+
*/
|
|
30072
|
+
function formatArgumentValue(value) {
|
|
30073
|
+
if (value === null || value === undefined) {
|
|
30074
|
+
return 'Not provided';
|
|
30075
|
+
}
|
|
30076
|
+
if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') {
|
|
30077
|
+
const text = String(value);
|
|
30078
|
+
return text === '' ? 'Empty string' : text;
|
|
30079
|
+
}
|
|
30080
|
+
if (Array.isArray(value)) {
|
|
30081
|
+
const items = value.map((entry) => formatArgumentValue(entry)).filter(Boolean);
|
|
30082
|
+
return items.length > 0 ? items.join(', ') : '[array]';
|
|
30083
|
+
}
|
|
30084
|
+
if (typeof value === 'object') {
|
|
30085
|
+
const entries = Object.entries(value)
|
|
30086
|
+
.map(([childKey, childValue]) => `${childKey}: ${formatArgumentValue(childValue)}`)
|
|
30087
|
+
.filter(Boolean);
|
|
30088
|
+
if (entries.length > 0) {
|
|
30089
|
+
const joined = entries.join('; ');
|
|
30090
|
+
return joined.length > 80 ? `${joined.slice(0, 80)}…` : joined;
|
|
30091
|
+
}
|
|
30092
|
+
const solo = JSON.stringify(value);
|
|
30093
|
+
return solo.length > 80 ? `${solo.slice(0, 80)}…` : solo;
|
|
30094
|
+
}
|
|
30095
|
+
return String(value);
|
|
30096
|
+
}
|
|
30097
|
+
/**
|
|
30098
|
+
* Extracts a short natural-language summary from the raw tool call result.
|
|
30099
|
+
*
|
|
30100
|
+
* @param resultRaw - Decoded tool call result.
|
|
30101
|
+
* @returns Friendly summary or `null` when nothing suitable is found.
|
|
30102
|
+
*/
|
|
30103
|
+
function buildToolCallResultSummary(resultRaw) {
|
|
30104
|
+
if (!resultRaw) {
|
|
30105
|
+
return null;
|
|
30106
|
+
}
|
|
30107
|
+
if (typeof resultRaw === 'string' && resultRaw.trim()) {
|
|
30108
|
+
return resultRaw.trim();
|
|
30109
|
+
}
|
|
30110
|
+
const candidate = findStringCandidate(resultRaw, [
|
|
30111
|
+
'summary',
|
|
30112
|
+
'text',
|
|
30113
|
+
'content',
|
|
30114
|
+
'description',
|
|
30115
|
+
'message',
|
|
30116
|
+
'result',
|
|
30117
|
+
]);
|
|
30118
|
+
if (candidate) {
|
|
30119
|
+
return candidate;
|
|
30120
|
+
}
|
|
30121
|
+
if (Array.isArray(resultRaw) && resultRaw.length > 0) {
|
|
30122
|
+
const firstEntry = resultRaw[0];
|
|
30123
|
+
if (typeof firstEntry === 'string' && firstEntry.trim()) {
|
|
30124
|
+
return firstEntry.trim();
|
|
30125
|
+
}
|
|
30126
|
+
const nested = findStringCandidate(firstEntry, ['title', 'snippet', 'summary']);
|
|
30127
|
+
if (nested) {
|
|
30128
|
+
return nested;
|
|
30129
|
+
}
|
|
30130
|
+
}
|
|
30131
|
+
return null;
|
|
30132
|
+
}
|
|
30133
|
+
/**
|
|
30134
|
+
* Searches for the first non-empty string field inside an object.
|
|
30135
|
+
*
|
|
30136
|
+
* @param value - Object to scan.
|
|
30137
|
+
* @param keys - Keys to try in order.
|
|
30138
|
+
* @returns First matching string or `null`.
|
|
30139
|
+
*/
|
|
30140
|
+
function findStringCandidate(value, keys) {
|
|
30141
|
+
if (!value || typeof value !== 'object') {
|
|
30142
|
+
return null;
|
|
30143
|
+
}
|
|
30144
|
+
for (const key of keys) {
|
|
30145
|
+
const candidate = value[key];
|
|
30146
|
+
if (typeof candidate === 'string' && candidate.trim()) {
|
|
30147
|
+
return candidate.trim();
|
|
30148
|
+
}
|
|
30149
|
+
}
|
|
30150
|
+
return null;
|
|
30151
|
+
}
|
|
30152
|
+
/**
|
|
30153
|
+
* Describes the kind of result that was returned to guide the summary badge row.
|
|
30154
|
+
*
|
|
30155
|
+
* @param resultRaw - Tool call result payload.
|
|
30156
|
+
* @returns Friendly label describing the result type.
|
|
30157
|
+
*/
|
|
30158
|
+
function describeResultKind(resultRaw) {
|
|
30159
|
+
if (resultRaw === null || resultRaw === undefined) {
|
|
30160
|
+
return 'No response';
|
|
30161
|
+
}
|
|
30162
|
+
if (typeof resultRaw === 'string') {
|
|
30163
|
+
return 'Text response';
|
|
30164
|
+
}
|
|
30165
|
+
if (typeof resultRaw === 'number') {
|
|
30166
|
+
return 'Numeric response';
|
|
30167
|
+
}
|
|
30168
|
+
if (Array.isArray(resultRaw)) {
|
|
30169
|
+
return 'List response';
|
|
30170
|
+
}
|
|
30171
|
+
if (typeof resultRaw === 'object') {
|
|
30172
|
+
return 'Structured response';
|
|
30173
|
+
}
|
|
30174
|
+
return 'Unknown response';
|
|
30175
|
+
}
|
|
30176
|
+
/**
|
|
30177
|
+
* Counts items returned by the tool call when the payload is iterable.
|
|
30178
|
+
*
|
|
30179
|
+
* @param resultRaw - Tool call result payload.
|
|
30180
|
+
* @returns Item count or `null` when the result is not a collection.
|
|
30181
|
+
*/
|
|
30182
|
+
function getResultItemCount(resultRaw) {
|
|
30183
|
+
if (Array.isArray(resultRaw)) {
|
|
30184
|
+
return resultRaw.length;
|
|
30185
|
+
}
|
|
30186
|
+
if (resultRaw && typeof resultRaw === 'object') {
|
|
30187
|
+
const candidates = ['results', 'items', 'data'];
|
|
30188
|
+
for (const key of candidates) {
|
|
30189
|
+
const candidate = resultRaw[key];
|
|
30190
|
+
if (Array.isArray(candidate)) {
|
|
30191
|
+
return candidate.length;
|
|
30192
|
+
}
|
|
30193
|
+
}
|
|
30194
|
+
}
|
|
30195
|
+
return null;
|
|
30196
|
+
}
|
|
30197
|
+
/**
|
|
30198
|
+
* Normalizes raw tool call errors and warnings for display badges.
|
|
30199
|
+
*
|
|
30200
|
+
* @param toolCall - Tool call payload to inspect.
|
|
30201
|
+
* @returns Array of structured issues.
|
|
30202
|
+
*/
|
|
30203
|
+
function normalizeToolCallIssues(toolCall) {
|
|
30204
|
+
const warnings = (toolCall.warnings || []).map((value) => ({
|
|
30205
|
+
type: 'warning',
|
|
30206
|
+
label: 'Warning',
|
|
30207
|
+
message: formatIssueValue(value),
|
|
30208
|
+
}));
|
|
30209
|
+
const errors = (toolCall.errors || []).map((value) => ({
|
|
30210
|
+
type: 'error',
|
|
30211
|
+
label: 'Error',
|
|
30212
|
+
message: formatIssueValue(value),
|
|
30213
|
+
}));
|
|
30214
|
+
return [...warnings, ...errors];
|
|
30215
|
+
}
|
|
30216
|
+
/**
|
|
30217
|
+
* Formats an error/warning payload into a single-line string.
|
|
30218
|
+
*
|
|
30219
|
+
* @param value - Raw issue payload.
|
|
30220
|
+
* @returns String suitable for badge display.
|
|
30221
|
+
*/
|
|
30222
|
+
function formatIssueValue(value) {
|
|
30223
|
+
if (typeof value === 'string' && value.trim()) {
|
|
30224
|
+
return value.trim();
|
|
30225
|
+
}
|
|
30226
|
+
if (typeof value === 'object' && value !== null) {
|
|
30227
|
+
const json = JSON.stringify(value);
|
|
30228
|
+
return json.length > 120 ? `${json.slice(0, 120)}…` : json;
|
|
30229
|
+
}
|
|
30230
|
+
return String(value !== null && value !== void 0 ? value : 'Unknown issue');
|
|
30231
|
+
}
|
|
30232
|
+
/**
|
|
30233
|
+
* Grabs the leading emoji (if present) from a chiplet label for fallback icons.
|
|
30234
|
+
*
|
|
30235
|
+
* @param text - Chiplet label text.
|
|
30236
|
+
* @returns First character or `null` when empty.
|
|
30237
|
+
*/
|
|
30238
|
+
function extractLeadingEmoji(text) {
|
|
30239
|
+
if (!text) {
|
|
30240
|
+
return null;
|
|
30241
|
+
}
|
|
30242
|
+
const trimmed = text.trim();
|
|
30243
|
+
return trimmed ? trimmed[0] : null;
|
|
28103
30244
|
}
|
|
28104
30245
|
|
|
28105
30246
|
/**
|
|
@@ -28409,6 +30550,16 @@
|
|
|
28409
30550
|
}
|
|
28410
30551
|
ChatPersistence.STORAGE_PREFIX = 'promptbook_chat_';
|
|
28411
30552
|
|
|
30553
|
+
/**
|
|
30554
|
+
* Default user-facing error text shown in chats when the generator fails.
|
|
30555
|
+
*
|
|
30556
|
+
* @private internal fallback for `LlmChat` so the same string can be reused across the stack without repetition.
|
|
30557
|
+
*/
|
|
30558
|
+
const DEFAULT_CHAT_FAIL_MESSAGE = 'Sorry, I encountered an error processing your message. Please, try again later.';
|
|
30559
|
+
/**
|
|
30560
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
30561
|
+
*/
|
|
30562
|
+
|
|
28412
30563
|
/**
|
|
28413
30564
|
* Builds a teammates lookup map from a list of teammate metadata entries.
|
|
28414
30565
|
*/
|
|
@@ -28481,7 +30632,8 @@
|
|
|
28481
30632
|
* @public exported from `@promptbook/components`
|
|
28482
30633
|
*/
|
|
28483
30634
|
function LlmChat(props) {
|
|
28484
|
-
const { llmTools, persistenceKey, onChange, onReset, onError, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, toolTitles, thinkingMessages, ...restProps } = props;
|
|
30635
|
+
const { llmTools, persistenceKey, onChange, onReset, onError, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, toolTitles, thinkingMessages, chatFailMessage, ...restProps } = props;
|
|
30636
|
+
const resolvedChatFailMessage = chatFailMessage || DEFAULT_CHAT_FAIL_MESSAGE;
|
|
28485
30637
|
// Internal state management
|
|
28486
30638
|
// DRY: Single factory for seeding initial messages (used on mount and after reset)
|
|
28487
30639
|
const buildInitialMessages = react.useCallback(() => initialMessages ? ([...initialMessages]) : ([]), [initialMessages]);
|
|
@@ -28721,7 +30873,7 @@
|
|
|
28721
30873
|
id: loadingMessage.id,
|
|
28722
30874
|
createdAt: $getCurrentDate(),
|
|
28723
30875
|
sender: llmParticipantName,
|
|
28724
|
-
content:
|
|
30876
|
+
content: resolvedChatFailMessage,
|
|
28725
30877
|
isComplete: true,
|
|
28726
30878
|
generationDurationMs: Date.now() - generationStartedAtMs,
|
|
28727
30879
|
};
|