@promptbook/components 0.105.0-15 → 0.105.0-16
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/esm/index.es.js +1983 -1022
- package/esm/index.es.js.map +1 -1
- package/esm/typings/src/_packages/types.index.d.ts +2 -0
- package/esm/typings/src/book-2.0/agent-source/AgentBasicInformation.d.ts +1 -1
- package/esm/typings/src/book-2.0/agent-source/parseTeamCommitment.d.ts +28 -0
- package/esm/typings/src/book-components/Chat/utils/toolCallParsing.d.ts +23 -0
- package/esm/typings/src/book-components/icons/SettingsIcon.d.ts +11 -0
- package/esm/typings/src/commitments/TEAM/TEAM.d.ts +45 -0
- package/esm/typings/src/commitments/index.d.ts +2 -1
- package/esm/typings/src/llm-providers/openai/OpenAiAssistantExecutionTools.d.ts +1 -0
- package/esm/typings/src/version.d.ts +1 -1
- package/package.json +1 -1
- package/umd/index.umd.js +1984 -1023
- package/umd/index.umd.js.map +1 -1
package/esm/index.es.js
CHANGED
|
@@ -36,7 +36,7 @@ const BOOK_LANGUAGE_VERSION = '2.0.0';
|
|
|
36
36
|
* @generated
|
|
37
37
|
* @see https://github.com/webgptorg/promptbook
|
|
38
38
|
*/
|
|
39
|
-
const PROMPTBOOK_ENGINE_VERSION = '0.105.0-
|
|
39
|
+
const PROMPTBOOK_ENGINE_VERSION = '0.105.0-16';
|
|
40
40
|
/**
|
|
41
41
|
* TODO: string_promptbook_version should be constrained to the all versions of Promptbook engine
|
|
42
42
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
@@ -86,7 +86,7 @@ function styleInject(css, ref) {
|
|
|
86
86
|
}
|
|
87
87
|
|
|
88
88
|
var css_248z$a = ".AvatarChip-module_AvatarChip__4sA0u{align-items:center;background:#e0e7ef;border:1px solid #b3c2d1;border-radius:16px;color:#2a3b4d;cursor:pointer;display:inline-flex;font-size:.95em;font-weight:500;margin:2px;padding:4px 12px}.AvatarChip-module_Avatar__mN2sc{border-radius:50%;height:24px;margin-right:8px;object-fit:cover;width:24px}.AvatarChip-module_TemplateLabel__-7vVI{background:#f0f4f8;border-radius:12px;color:#3a4752;font-size:.8em;margin-left:8px;margin-right:0;padding:2px 6px;text-transform:uppercase}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkF2YXRhckNoaXAubW9kdWxlLmNzcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxxQ0FFSSxrQkFBbUIsQ0FHbkIsa0JBQW1CLENBS25CLHdCQUF5QixDQU56QixrQkFBbUIsQ0FFbkIsYUFBYyxDQU1kLGNBQWUsQ0FYZixtQkFBb0IsQ0FPcEIsZUFBaUIsQ0FEakIsZUFBZ0IsQ0FFaEIsVUFBVyxDQU5YLGdCQVVKLENBRUEsaUNBR0ksaUJBQWtCLENBRGxCLFdBQVksQ0FHWixnQkFBaUIsQ0FEakIsZ0JBQWlCLENBSGpCLFVBS0osQ0FFQSx3Q0FDSSxrQkFBbUIsQ0FJbkIsa0JBQW1CLENBSG5CLGFBQWMsQ0FDZCxjQUFnQixDQUdoQixlQUFnQixDQUNoQixjQUFlLENBSGYsZUFBZ0IsQ0FJaEIsd0JBQ0oiLCJmaWxlIjoiQXZhdGFyQ2hpcC5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLkF2YXRhckNoaXAge1xuICAgIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgcGFkZGluZzogNHB4IDEycHg7XG4gICAgYm9yZGVyLXJhZGl1czogMTZweDtcbiAgICBiYWNrZ3JvdW5kOiAjZTBlN2VmO1xuICAgIGNvbG9yOiAjMmEzYjRkO1xuICAgIGZvbnQtd2VpZ2h0OiA1MDA7XG4gICAgZm9udC1zaXplOiAwLjk1ZW07XG4gICAgbWFyZ2luOiAycHg7XG4gICAgYm9yZGVyOiAxcHggc29saWQgI2IzYzJkMTtcblxuICAgIGN1cnNvcjogcG9pbnRlcjtcbn1cblxuLkF2YXRhciB7XG4gICAgd2lkdGg6IDI0cHg7XG4gICAgaGVpZ2h0OiAyNHB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcbiAgICBvYmplY3QtZml0OiBjb3ZlcjtcbiAgICBtYXJnaW4tcmlnaHQ6IDhweDtcbn1cblxuLlRlbXBsYXRlTGFiZWwge1xuICAgIGJhY2tncm91bmQ6ICNmMGY0Zjg7XG4gICAgY29sb3I6ICMzYTQ3NTI7XG4gICAgZm9udC1zaXplOiAwLjhlbTtcbiAgICBwYWRkaW5nOiAycHggNnB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDEycHg7XG4gICAgbWFyZ2luLWxlZnQ6IDhweDtcbiAgICBtYXJnaW4tcmlnaHQ6IDA7XG4gICAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbn1cbiJdfQ== */";
|
|
89
|
-
var styles$
|
|
89
|
+
var styles$a = {"AvatarChip":"AvatarChip-module_AvatarChip__4sA0u","Avatar":"AvatarChip-module_Avatar__mN2sc","TemplateLabel":"AvatarChip-module_TemplateLabel__-7vVI"};
|
|
90
90
|
styleInject(css_248z$a);
|
|
91
91
|
|
|
92
92
|
/**
|
|
@@ -97,7 +97,7 @@ styleInject(css_248z$a);
|
|
|
97
97
|
function AvatarChip(props) {
|
|
98
98
|
const { avatarBasicInformation, className, onSelect, isSelected, isTemplate } = props;
|
|
99
99
|
const { agentName, meta } = avatarBasicInformation;
|
|
100
|
-
return (jsxs("div", { className: classNames(styles$
|
|
100
|
+
return (jsxs("div", { className: classNames(styles$a.AvatarChip, className, isSelected ? styles$a.Selected : undefined), onClick: () => onSelect === null || onSelect === void 0 ? void 0 : onSelect(avatarBasicInformation), style: { cursor: onSelect ? 'pointer' : undefined }, children: [jsx("img", { src: meta.image, alt: agentName || '', className: styles$a.Avatar }), meta.fullname || agentName, isTemplate && jsx("span", { className: styles$a.TemplateLabel, children: "Template" })] }));
|
|
101
101
|
}
|
|
102
102
|
|
|
103
103
|
/**
|
|
@@ -4043,6 +4043,17 @@ function normalizeTo_snake_case(text) {
|
|
|
4043
4043
|
return normalizeTo_SCREAMING_CASE(text).toLowerCase();
|
|
4044
4044
|
}
|
|
4045
4045
|
|
|
4046
|
+
/**
|
|
4047
|
+
* Take every whitespace (space, new line, tab) and replace it with a single space
|
|
4048
|
+
*
|
|
4049
|
+
* Note: [🔂] This function is idempotent.
|
|
4050
|
+
*
|
|
4051
|
+
* @public exported from `@promptbook/utils`
|
|
4052
|
+
*/
|
|
4053
|
+
function normalizeWhitespaces(sentence) {
|
|
4054
|
+
return sentence.replace(/\s+/gs, ' ').trim();
|
|
4055
|
+
}
|
|
4056
|
+
|
|
4046
4057
|
/**
|
|
4047
4058
|
* Removes quotes and optional introduce text from a string
|
|
4048
4059
|
*
|
|
@@ -7417,6 +7428,432 @@ class StyleCommitmentDefinition extends BaseCommitmentDefinition {
|
|
|
7417
7428
|
* [💞] Ignore a discrepancy between file name and entity name
|
|
7418
7429
|
*/
|
|
7419
7430
|
|
|
7431
|
+
const urlRegex = /https?:\/\/[^\s]+/gi;
|
|
7432
|
+
const trailingPunctuationRegex = /[),.;!?]+$/;
|
|
7433
|
+
const clauseSeparators = ['.', '?', '!', ';', ','];
|
|
7434
|
+
const conjunctionSeparators = [' and ', ' or '];
|
|
7435
|
+
/**
|
|
7436
|
+
* Parses TEAM commitment content into teammates with instructions.
|
|
7437
|
+
*
|
|
7438
|
+
* @private
|
|
7439
|
+
*/
|
|
7440
|
+
function parseTeamCommitmentContent(content, options = {}) {
|
|
7441
|
+
const { strict = false } = options;
|
|
7442
|
+
const lines = content
|
|
7443
|
+
.split('\n')
|
|
7444
|
+
.map((line) => line.trim())
|
|
7445
|
+
.filter(Boolean);
|
|
7446
|
+
const teammates = [];
|
|
7447
|
+
const seenUrls = new Set();
|
|
7448
|
+
for (const line of lines) {
|
|
7449
|
+
const matches = Array.from(line.matchAll(urlRegex));
|
|
7450
|
+
if (matches.length === 0) {
|
|
7451
|
+
if (strict) {
|
|
7452
|
+
throw new Error(`TEAM commitment expects at least one agent URL, got: "${line}"`);
|
|
7453
|
+
}
|
|
7454
|
+
continue;
|
|
7455
|
+
}
|
|
7456
|
+
for (const [matchIndex, match] of matches.entries()) {
|
|
7457
|
+
const rawUrl = match[0] || '';
|
|
7458
|
+
const cleanedUrl = rawUrl.replace(trailingPunctuationRegex, '');
|
|
7459
|
+
if (!isValidAgentUrl(cleanedUrl)) {
|
|
7460
|
+
if (strict) {
|
|
7461
|
+
throw new Error(`Invalid agent URL in TEAM commitment: "${cleanedUrl}"`);
|
|
7462
|
+
}
|
|
7463
|
+
continue;
|
|
7464
|
+
}
|
|
7465
|
+
if (seenUrls.has(cleanedUrl)) {
|
|
7466
|
+
continue;
|
|
7467
|
+
}
|
|
7468
|
+
seenUrls.add(cleanedUrl);
|
|
7469
|
+
const instructionContext = extractInstructionContext(line, matches, matchIndex);
|
|
7470
|
+
const instructions = normalizeInstructionText(instructionContext);
|
|
7471
|
+
const label = createTeammateLabel(cleanedUrl);
|
|
7472
|
+
teammates.push({
|
|
7473
|
+
url: cleanedUrl,
|
|
7474
|
+
label,
|
|
7475
|
+
instructions,
|
|
7476
|
+
});
|
|
7477
|
+
}
|
|
7478
|
+
}
|
|
7479
|
+
return teammates;
|
|
7480
|
+
}
|
|
7481
|
+
function extractInstructionContext(line, matches, matchIndex) {
|
|
7482
|
+
var _a;
|
|
7483
|
+
const match = matches[matchIndex];
|
|
7484
|
+
if (!match || match.index === undefined) {
|
|
7485
|
+
return line.trim();
|
|
7486
|
+
}
|
|
7487
|
+
const rawUrl = match[0] || '';
|
|
7488
|
+
const matchStart = match.index;
|
|
7489
|
+
const matchEnd = matchStart + rawUrl.length;
|
|
7490
|
+
const previousMatch = matches[matchIndex - 1];
|
|
7491
|
+
const nextMatch = matches[matchIndex + 1];
|
|
7492
|
+
const previousEnd = previousMatch && previousMatch.index !== undefined ? previousMatch.index + (((_a = previousMatch[0]) === null || _a === void 0 ? void 0 : _a.length) || 0) : 0;
|
|
7493
|
+
const nextStart = nextMatch && nextMatch.index !== undefined ? nextMatch.index : line.length;
|
|
7494
|
+
const rawPrefix = line.slice(previousEnd, matchStart);
|
|
7495
|
+
const rawSuffix = line.slice(matchEnd, nextStart);
|
|
7496
|
+
const prefix = trimAfterLastDelimiter(rawPrefix);
|
|
7497
|
+
const suffix = trimBeforeLastDelimiter(rawSuffix);
|
|
7498
|
+
if (normalizeInstructionText(suffix)) {
|
|
7499
|
+
return suffix;
|
|
7500
|
+
}
|
|
7501
|
+
if (normalizeInstructionText(prefix)) {
|
|
7502
|
+
return prefix;
|
|
7503
|
+
}
|
|
7504
|
+
return `${prefix} ${suffix}`.trim();
|
|
7505
|
+
}
|
|
7506
|
+
function trimAfterLastDelimiter(text) {
|
|
7507
|
+
const match = findLastDelimiter(text);
|
|
7508
|
+
if (!match) {
|
|
7509
|
+
return text;
|
|
7510
|
+
}
|
|
7511
|
+
return text.slice(match.index + match.length);
|
|
7512
|
+
}
|
|
7513
|
+
function trimBeforeLastDelimiter(text) {
|
|
7514
|
+
const cleaned = text.replace(/^[,;:]\s*/g, '');
|
|
7515
|
+
const match = findLastDelimiter(cleaned);
|
|
7516
|
+
if (!match || match.index <= 0) {
|
|
7517
|
+
return cleaned;
|
|
7518
|
+
}
|
|
7519
|
+
return cleaned.slice(0, match.index);
|
|
7520
|
+
}
|
|
7521
|
+
function findLastDelimiter(text) {
|
|
7522
|
+
let bestIndex = -1;
|
|
7523
|
+
let bestLength = 0;
|
|
7524
|
+
for (const separator of clauseSeparators) {
|
|
7525
|
+
const index = text.lastIndexOf(separator);
|
|
7526
|
+
if (index > bestIndex) {
|
|
7527
|
+
bestIndex = index;
|
|
7528
|
+
bestLength = separator.length;
|
|
7529
|
+
}
|
|
7530
|
+
}
|
|
7531
|
+
const lowerText = text.toLowerCase();
|
|
7532
|
+
for (const separator of conjunctionSeparators) {
|
|
7533
|
+
const index = lowerText.lastIndexOf(separator);
|
|
7534
|
+
if (index > bestIndex) {
|
|
7535
|
+
bestIndex = index;
|
|
7536
|
+
bestLength = separator.length;
|
|
7537
|
+
}
|
|
7538
|
+
}
|
|
7539
|
+
if (bestIndex === -1) {
|
|
7540
|
+
return null;
|
|
7541
|
+
}
|
|
7542
|
+
return { index: bestIndex, length: bestLength };
|
|
7543
|
+
}
|
|
7544
|
+
function normalizeInstructionText(text) {
|
|
7545
|
+
if (!text) {
|
|
7546
|
+
return '';
|
|
7547
|
+
}
|
|
7548
|
+
const withoutUrls = text.replace(urlRegex, '');
|
|
7549
|
+
let normalized = normalizeWhitespaces(withoutUrls).trim();
|
|
7550
|
+
normalized = normalized.replace(/^[,;:]\s*/g, '');
|
|
7551
|
+
normalized = normalized.replace(/^(and|or|the|a|an)\s+/i, '');
|
|
7552
|
+
normalized = normalized.replace(/\s*[,;:]\s*$/g, '');
|
|
7553
|
+
normalized = normalized.replace(/\s+(and|or)\s*$/i, '');
|
|
7554
|
+
normalized = normalizeWhitespaces(normalized).trim();
|
|
7555
|
+
return normalized;
|
|
7556
|
+
}
|
|
7557
|
+
function createTeammateLabel(url) {
|
|
7558
|
+
try {
|
|
7559
|
+
const parsed = new URL(url);
|
|
7560
|
+
const pathParts = parsed.pathname.split('/').filter(Boolean);
|
|
7561
|
+
const lastPart = pathParts[pathParts.length - 1] || parsed.hostname;
|
|
7562
|
+
const decoded = decodeURIComponent(lastPart);
|
|
7563
|
+
const spaced = decoded.replace(/[-_]+/g, ' ').trim();
|
|
7564
|
+
if (!spaced) {
|
|
7565
|
+
return parsed.hostname;
|
|
7566
|
+
}
|
|
7567
|
+
return spaced
|
|
7568
|
+
.split(' ')
|
|
7569
|
+
.map((word) => word.charAt(0).toUpperCase() + word.slice(1))
|
|
7570
|
+
.join(' ');
|
|
7571
|
+
}
|
|
7572
|
+
catch (error) {
|
|
7573
|
+
return url;
|
|
7574
|
+
}
|
|
7575
|
+
}
|
|
7576
|
+
/**
|
|
7577
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
7578
|
+
*/
|
|
7579
|
+
|
|
7580
|
+
/**
|
|
7581
|
+
* The built-in `fetch' function with a lightweight error handling wrapper as default fetch function used in Promptbook scrapers
|
|
7582
|
+
*
|
|
7583
|
+
* @public exported from `@promptbook/core`
|
|
7584
|
+
*/
|
|
7585
|
+
const promptbookFetch = async (urlOrRequest, init) => {
|
|
7586
|
+
try {
|
|
7587
|
+
return await fetch(urlOrRequest, init);
|
|
7588
|
+
}
|
|
7589
|
+
catch (error) {
|
|
7590
|
+
assertsError(error);
|
|
7591
|
+
let url;
|
|
7592
|
+
if (typeof urlOrRequest === 'string') {
|
|
7593
|
+
url = urlOrRequest;
|
|
7594
|
+
}
|
|
7595
|
+
else if (urlOrRequest instanceof Request) {
|
|
7596
|
+
url = urlOrRequest.url;
|
|
7597
|
+
}
|
|
7598
|
+
throw new PromptbookFetchError(spaceTrim$2((block) => `
|
|
7599
|
+
Can not fetch "${url}"
|
|
7600
|
+
|
|
7601
|
+
Fetch error:
|
|
7602
|
+
${block(error.message)}
|
|
7603
|
+
|
|
7604
|
+
`));
|
|
7605
|
+
}
|
|
7606
|
+
};
|
|
7607
|
+
/**
|
|
7608
|
+
* TODO: [🧠] Maybe rename because it is not used only for scrapers but also in `$getCompiledBook`
|
|
7609
|
+
*/
|
|
7610
|
+
|
|
7611
|
+
const TEAM_TOOL_PREFIX = 'team_chat_';
|
|
7612
|
+
const teamToolFunctions = {};
|
|
7613
|
+
const teamToolTitles = {};
|
|
7614
|
+
/**
|
|
7615
|
+
* TEAM commitment definition
|
|
7616
|
+
*
|
|
7617
|
+
* The `TEAM` commitment defines teammates that the agent can consult via tools.
|
|
7618
|
+
*
|
|
7619
|
+
* Example usage in agent source:
|
|
7620
|
+
*
|
|
7621
|
+
* ```book
|
|
7622
|
+
* TEAM https://agents.ptbk.ik/agents/joe-green
|
|
7623
|
+
* TEAM You can talk with http://localhost:4440/agents/GMw67JN8TXxN7y to discuss the legal aspects.
|
|
7624
|
+
* ```
|
|
7625
|
+
*
|
|
7626
|
+
* @private [??] Maybe export the commitments through some package
|
|
7627
|
+
*/
|
|
7628
|
+
class TeamCommitmentDefinition extends BaseCommitmentDefinition {
|
|
7629
|
+
constructor() {
|
|
7630
|
+
super('TEAM');
|
|
7631
|
+
}
|
|
7632
|
+
/**
|
|
7633
|
+
* Short one-line description of TEAM.
|
|
7634
|
+
*/
|
|
7635
|
+
get description() {
|
|
7636
|
+
return 'Enable the agent to consult teammate agents via dedicated tools.';
|
|
7637
|
+
}
|
|
7638
|
+
/**
|
|
7639
|
+
* Icon for this commitment.
|
|
7640
|
+
*/
|
|
7641
|
+
get icon() {
|
|
7642
|
+
return '??';
|
|
7643
|
+
}
|
|
7644
|
+
/**
|
|
7645
|
+
* Markdown documentation for TEAM commitment.
|
|
7646
|
+
*/
|
|
7647
|
+
get documentation() {
|
|
7648
|
+
return spaceTrim$1(`
|
|
7649
|
+
# TEAM
|
|
7650
|
+
|
|
7651
|
+
Registers teammate agents that the current agent can consult via tools.
|
|
7652
|
+
|
|
7653
|
+
## Examples
|
|
7654
|
+
|
|
7655
|
+
\`\`\`book
|
|
7656
|
+
Legal Assistant
|
|
7657
|
+
|
|
7658
|
+
PERSONA An expert software developer
|
|
7659
|
+
TEAM You can talk with http://localhost:4440/agents/GMw67JN8TXxN7y to discuss the legal aspects.
|
|
7660
|
+
\`\`\`
|
|
7661
|
+
`);
|
|
7662
|
+
}
|
|
7663
|
+
applyToAgentModelRequirements(requirements, content) {
|
|
7664
|
+
var _a, _b;
|
|
7665
|
+
const trimmedContent = content.trim();
|
|
7666
|
+
if (!trimmedContent) {
|
|
7667
|
+
return requirements;
|
|
7668
|
+
}
|
|
7669
|
+
const teammates = parseTeamCommitmentContent(trimmedContent, { strict: true });
|
|
7670
|
+
if (teammates.length === 0) {
|
|
7671
|
+
return requirements;
|
|
7672
|
+
}
|
|
7673
|
+
const agentName = ((_a = requirements.metadata) === null || _a === void 0 ? void 0 : _a.agentName) || 'Agent';
|
|
7674
|
+
const teamEntries = teammates.map((teammate) => ({
|
|
7675
|
+
toolName: createTeamToolName(teammate.url),
|
|
7676
|
+
teammate,
|
|
7677
|
+
agentName,
|
|
7678
|
+
}));
|
|
7679
|
+
for (const entry of teamEntries) {
|
|
7680
|
+
registerTeamTool(entry);
|
|
7681
|
+
}
|
|
7682
|
+
const existingTools = requirements.tools || [];
|
|
7683
|
+
const updatedTools = [...existingTools];
|
|
7684
|
+
for (const entry of teamEntries) {
|
|
7685
|
+
if (updatedTools.some((tool) => tool.name === entry.toolName)) {
|
|
7686
|
+
continue;
|
|
7687
|
+
}
|
|
7688
|
+
const instructionSuffix = entry.teammate.instructions
|
|
7689
|
+
? `Use when: ${entry.teammate.instructions}`
|
|
7690
|
+
: 'Use when their expertise is needed.';
|
|
7691
|
+
updatedTools.push({
|
|
7692
|
+
name: entry.toolName,
|
|
7693
|
+
description: spaceTrim$1(`
|
|
7694
|
+
Consult teammate ${entry.teammate.label} (${entry.teammate.url}).
|
|
7695
|
+
${instructionSuffix}
|
|
7696
|
+
`),
|
|
7697
|
+
parameters: {
|
|
7698
|
+
type: 'object',
|
|
7699
|
+
properties: {
|
|
7700
|
+
message: {
|
|
7701
|
+
type: 'string',
|
|
7702
|
+
description: 'Question or request to send to the teammate.',
|
|
7703
|
+
},
|
|
7704
|
+
context: {
|
|
7705
|
+
type: 'string',
|
|
7706
|
+
description: 'Optional background context for the teammate.',
|
|
7707
|
+
},
|
|
7708
|
+
},
|
|
7709
|
+
required: ['message'],
|
|
7710
|
+
},
|
|
7711
|
+
});
|
|
7712
|
+
}
|
|
7713
|
+
const existingTeammates = ((_b = requirements.metadata) === null || _b === void 0 ? void 0 : _b.teammates) || [];
|
|
7714
|
+
const updatedTeammates = [...existingTeammates];
|
|
7715
|
+
for (const entry of teamEntries) {
|
|
7716
|
+
if (updatedTeammates.some((existing) => existing.url === entry.teammate.url)) {
|
|
7717
|
+
continue;
|
|
7718
|
+
}
|
|
7719
|
+
updatedTeammates.push({
|
|
7720
|
+
url: entry.teammate.url,
|
|
7721
|
+
label: entry.teammate.label,
|
|
7722
|
+
instructions: entry.teammate.instructions || undefined,
|
|
7723
|
+
toolName: entry.toolName,
|
|
7724
|
+
});
|
|
7725
|
+
}
|
|
7726
|
+
const teamSystemMessage = spaceTrim$1((block) => `
|
|
7727
|
+
Teammates:
|
|
7728
|
+
${block(teamEntries
|
|
7729
|
+
.map((entry) => {
|
|
7730
|
+
const whenToConsult = entry.teammate.instructions || 'Use when their expertise is needed.';
|
|
7731
|
+
return spaceTrim$1(() => `
|
|
7732
|
+
- ${entry.teammate.label} (${entry.teammate.url})
|
|
7733
|
+
- Tool: "${entry.toolName}"
|
|
7734
|
+
- When to consult: ${whenToConsult}
|
|
7735
|
+
`);
|
|
7736
|
+
})
|
|
7737
|
+
.join('\n'))}
|
|
7738
|
+
`);
|
|
7739
|
+
return this.appendToSystemMessage({
|
|
7740
|
+
...requirements,
|
|
7741
|
+
tools: updatedTools,
|
|
7742
|
+
metadata: {
|
|
7743
|
+
...requirements.metadata,
|
|
7744
|
+
teammates: updatedTeammates,
|
|
7745
|
+
},
|
|
7746
|
+
}, teamSystemMessage);
|
|
7747
|
+
}
|
|
7748
|
+
/**
|
|
7749
|
+
* Gets human-readable titles for tool functions provided by this commitment.
|
|
7750
|
+
*/
|
|
7751
|
+
getToolTitles() {
|
|
7752
|
+
return { ...teamToolTitles };
|
|
7753
|
+
}
|
|
7754
|
+
/**
|
|
7755
|
+
* Gets tool function implementations for teammate tools.
|
|
7756
|
+
*/
|
|
7757
|
+
getToolFunctions() {
|
|
7758
|
+
return { ...teamToolFunctions };
|
|
7759
|
+
}
|
|
7760
|
+
}
|
|
7761
|
+
function createTeamToolName(url) {
|
|
7762
|
+
const hash = computeHash(url).substring(0, 10);
|
|
7763
|
+
return `${TEAM_TOOL_PREFIX}${hash}`;
|
|
7764
|
+
}
|
|
7765
|
+
function registerTeamTool(entry) {
|
|
7766
|
+
teamToolFunctions[entry.toolName] = createTeamToolFunction(entry);
|
|
7767
|
+
teamToolTitles[entry.toolName] = `Consult ${entry.teammate.label}`;
|
|
7768
|
+
}
|
|
7769
|
+
function createTeamToolFunction(entry) {
|
|
7770
|
+
return async (args) => {
|
|
7771
|
+
const message = args.message || args.question || '';
|
|
7772
|
+
if (!message) {
|
|
7773
|
+
const result = {
|
|
7774
|
+
error: 'Message is required to contact teammate.',
|
|
7775
|
+
teammate: {
|
|
7776
|
+
url: entry.teammate.url,
|
|
7777
|
+
label: entry.teammate.label,
|
|
7778
|
+
instructions: entry.teammate.instructions,
|
|
7779
|
+
toolName: entry.toolName,
|
|
7780
|
+
},
|
|
7781
|
+
};
|
|
7782
|
+
return JSON.stringify(result);
|
|
7783
|
+
}
|
|
7784
|
+
const request = args.context ? `${message}\n\nContext:\n${args.context}` : message;
|
|
7785
|
+
let response = '';
|
|
7786
|
+
let error = null;
|
|
7787
|
+
try {
|
|
7788
|
+
response = await fetchTeammateResponse(entry.teammate.url, request);
|
|
7789
|
+
}
|
|
7790
|
+
catch (err) {
|
|
7791
|
+
error = err instanceof Error ? err.message : String(err);
|
|
7792
|
+
}
|
|
7793
|
+
const teammateReply = response || (error ? `Unable to reach teammate. Error: ${error}` : 'No response received.');
|
|
7794
|
+
const result = {
|
|
7795
|
+
teammate: {
|
|
7796
|
+
url: entry.teammate.url,
|
|
7797
|
+
label: entry.teammate.label,
|
|
7798
|
+
instructions: entry.teammate.instructions,
|
|
7799
|
+
toolName: entry.toolName,
|
|
7800
|
+
},
|
|
7801
|
+
request,
|
|
7802
|
+
response: teammateReply,
|
|
7803
|
+
error,
|
|
7804
|
+
conversation: [
|
|
7805
|
+
{
|
|
7806
|
+
sender: 'AGENT',
|
|
7807
|
+
name: entry.agentName,
|
|
7808
|
+
content: request,
|
|
7809
|
+
},
|
|
7810
|
+
{
|
|
7811
|
+
sender: 'TEAMMATE',
|
|
7812
|
+
name: entry.teammate.label,
|
|
7813
|
+
content: teammateReply,
|
|
7814
|
+
},
|
|
7815
|
+
],
|
|
7816
|
+
};
|
|
7817
|
+
return JSON.stringify(result);
|
|
7818
|
+
};
|
|
7819
|
+
}
|
|
7820
|
+
async function fetchTeammateResponse(agentUrl, message) {
|
|
7821
|
+
const url = `${agentUrl.replace(/\/$/, '')}/api/chat`;
|
|
7822
|
+
const response = await promptbookFetch(url, {
|
|
7823
|
+
method: 'POST',
|
|
7824
|
+
headers: {
|
|
7825
|
+
'Content-Type': 'application/json',
|
|
7826
|
+
},
|
|
7827
|
+
body: JSON.stringify({ message }),
|
|
7828
|
+
});
|
|
7829
|
+
if (!response.ok) {
|
|
7830
|
+
throw new Error(`Teammate request failed: ${response.status} ${response.statusText}`);
|
|
7831
|
+
}
|
|
7832
|
+
const rawText = await response.text();
|
|
7833
|
+
return stripToolCallLines(rawText).trim();
|
|
7834
|
+
}
|
|
7835
|
+
function stripToolCallLines(text) {
|
|
7836
|
+
const lines = text.replace(/\r\n/g, '\n').split('\n');
|
|
7837
|
+
return lines
|
|
7838
|
+
.filter((line) => {
|
|
7839
|
+
const trimmed = line.trim();
|
|
7840
|
+
if (!trimmed.startsWith('{') || !trimmed.endsWith('}')) {
|
|
7841
|
+
return true;
|
|
7842
|
+
}
|
|
7843
|
+
try {
|
|
7844
|
+
const parsed = JSON.parse(trimmed);
|
|
7845
|
+
return !('toolCalls' in parsed);
|
|
7846
|
+
}
|
|
7847
|
+
catch (_a) {
|
|
7848
|
+
return true;
|
|
7849
|
+
}
|
|
7850
|
+
})
|
|
7851
|
+
.join('\n');
|
|
7852
|
+
}
|
|
7853
|
+
/**
|
|
7854
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
7855
|
+
*/
|
|
7856
|
+
|
|
7420
7857
|
/**
|
|
7421
7858
|
* USE commitment definition
|
|
7422
7859
|
*
|
|
@@ -8411,6 +8848,7 @@ const COMMITMENT_REGISTRY = [
|
|
|
8411
8848
|
new DictionaryCommitmentDefinition(),
|
|
8412
8849
|
new OpenCommitmentDefinition(),
|
|
8413
8850
|
new ClosedCommitmentDefinition(),
|
|
8851
|
+
new TeamCommitmentDefinition(),
|
|
8414
8852
|
new UseBrowserCommitmentDefinition(),
|
|
8415
8853
|
new UseSearchEngineCommitmentDefinition(),
|
|
8416
8854
|
new UseTimeCommitmentDefinition(),
|
|
@@ -8862,6 +9300,18 @@ function parseAgentSource(agentSource) {
|
|
|
8862
9300
|
});
|
|
8863
9301
|
continue;
|
|
8864
9302
|
}
|
|
9303
|
+
if (commitment.type === 'TEAM') {
|
|
9304
|
+
const teammates = parseTeamCommitmentContent(commitment.content);
|
|
9305
|
+
for (const teammate of teammates) {
|
|
9306
|
+
capabilities.push({
|
|
9307
|
+
type: 'team',
|
|
9308
|
+
label: teammate.label,
|
|
9309
|
+
iconName: 'Users',
|
|
9310
|
+
agentUrl: teammate.url,
|
|
9311
|
+
});
|
|
9312
|
+
}
|
|
9313
|
+
continue;
|
|
9314
|
+
}
|
|
8865
9315
|
if (commitment.type === 'KNOWLEDGE') {
|
|
8866
9316
|
const content = spaceTrim$2(commitment.content).split('\n')[0] || '';
|
|
8867
9317
|
let label = content;
|
|
@@ -8982,7 +9432,7 @@ function AvatarChipFromSource(props) {
|
|
|
8982
9432
|
}
|
|
8983
9433
|
|
|
8984
9434
|
var css_248z$9 = ".Modal-module_scrim__jKO-A{align-items:center;background-color:rgba(0,0,0,.5);display:flex;height:100%;justify-content:center;left:0;position:fixed;top:0;width:100%;z-index:9000000}.Modal-module_Modal__k49dY{background:#fff;border-radius:8px;max-height:90%;max-width:90%;min-height:30vh;min-width:50vw;overflow:auto;padding:20px;position:relative}.Modal-module_closeButton__dzf6l{background:transparent;border:none;cursor:pointer;font-size:24px;position:absolute;right:10px;top:10px}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk1vZGFsLm1vZHVsZS5jc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsMkJBU0ksa0JBQW1CLENBSG5CLCtCQUFvQyxDQUNwQyxZQUFhLENBRmIsV0FBWSxDQUdaLHNCQUF1QixDQUx2QixNQUFPLENBRlAsY0FBZSxDQUNmLEtBQU0sQ0FFTixVQUFXLENBTVgsZUFDSixDQUVBLDJCQUNJLGVBQWlCLENBRWpCLGlCQUFrQixDQUdsQixjQUFlLENBRGYsYUFBYyxDQUdkLGVBQWdCLENBRGhCLGNBQWUsQ0FFZixhQUFjLENBUGQsWUFBYSxDQUViLGlCQU1KLENBRUEsaUNBSUksc0JBQXVCLENBQ3ZCLFdBQVksQ0FFWixjQUFlLENBRGYsY0FBZSxDQUxmLGlCQUFrQixDQUVsQixVQUFXLENBRFgsUUFNSiIsImZpbGUiOiJNb2RhbC5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLnNjcmltIHtcbiAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgdG9wOiAwO1xuICAgIGxlZnQ6IDA7XG4gICAgd2lkdGg6IDEwMCU7XG4gICAgaGVpZ2h0OiAxMDAlO1xuICAgIGJhY2tncm91bmQtY29sb3I6IHJnYmEoMCwgMCwgMCwgMC41KTtcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgei1pbmRleDogOTAwMDAwMDtcbn1cblxuLk1vZGFsIHtcbiAgICBiYWNrZ3JvdW5kOiB3aGl0ZTtcbiAgICBwYWRkaW5nOiAyMHB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDhweDtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgbWF4LXdpZHRoOiA5MCU7XG4gICAgbWF4LWhlaWdodDogOTAlO1xuICAgIG1pbi13aWR0aDogNTB2dztcbiAgICBtaW4taGVpZ2h0OiAzMHZoO1xuICAgIG92ZXJmbG93OiBhdXRvO1xufVxuXG4uY2xvc2VCdXR0b24ge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB0b3A6IDEwcHg7XG4gICAgcmlnaHQ6IDEwcHg7XG4gICAgYmFja2dyb3VuZDogdHJhbnNwYXJlbnQ7XG4gICAgYm9yZGVyOiBub25lO1xuICAgIGZvbnQtc2l6ZTogMjRweDtcbiAgICBjdXJzb3I6IHBvaW50ZXI7XG59XG4iXX0= */";
|
|
8985
|
-
var styles$
|
|
9435
|
+
var styles$9 = {"scrim":"Modal-module_scrim__jKO-A","Modal":"Modal-module_Modal__k49dY","closeButton":"Modal-module_closeButton__dzf6l"};
|
|
8986
9436
|
styleInject(css_248z$9);
|
|
8987
9437
|
|
|
8988
9438
|
/**
|
|
@@ -9001,7 +9451,7 @@ function Modal({ children, onClose, className, }) {
|
|
|
9001
9451
|
window.removeEventListener('keydown', handleKeyDown);
|
|
9002
9452
|
};
|
|
9003
9453
|
}, [onClose]);
|
|
9004
|
-
return createPortal(jsx("div", { className: styles$
|
|
9454
|
+
return createPortal(jsx("div", { className: styles$9.scrim, onClick: onClose, children: jsxs("div", { className: classNames(styles$9.Modal, className), onClick: (e) => e.stopPropagation(), children: [jsx("button", { className: styles$9.closeButton, onClick: onClose, children: "\u00D7" }), children] }) }), document.body);
|
|
9005
9455
|
}
|
|
9006
9456
|
|
|
9007
9457
|
/**
|
|
@@ -9087,7 +9537,7 @@ const DEFAULT_BOOK = padBook(validateBook(spaceTrim$2(`
|
|
|
9087
9537
|
// <- TODO: [🐱🚀] generateBookBoilerplate and deprecate `DEFAULT_BOOK`
|
|
9088
9538
|
|
|
9089
9539
|
var css_248z$8 = ".BookEditor-module_BookEditor__s-0PU{width:100%}.BookEditor-module_bookEditorContainer__wLMwM{box-sizing:border-box;height:100%;padding:10px 25px 0;position:relative;width:100%}.BookEditor-module_bookEditorContainer__wLMwM.BookEditor-module_isVerbose__VQ6iL{background-color:rgba(0,0,0,.05);outline:1px dotted rgba(0,0,0,.5)}.BookEditor-module_isVerbose__VQ6iL{outline:2px dotted #ff7526}.BookEditor-module_bookEditorWrapper__twppD{background-color:#fff;border:1px solid rgba(209,213,219,.8);border-radius:1rem;box-shadow:0 1px 2px 0 rgba(0,0,0,.05);overflow:hidden;padding-top:10px;transition:box-shadow .2s ease-in-out}.BookEditor-module_isVerbose__VQ6iL .BookEditor-module_bookEditorWrapper__twppD{overflow:visible}.BookEditor-module_bookEditorWrapper__twppD:hover{box-shadow:0 4px 6px -1px rgba(0,0,0,.1),0 2px 4px -1px rgba(0,0,0,.06)}.BookEditor-module_bookEditorWrapper__twppD.BookEditor-module_isBorderRadiusDisabled__h1I3v{border-radius:0}.BookEditor-module_dropOverlay__xWWoX{align-items:center;background-color:rgba(0,0,0,.5);bottom:0;color:#fff;display:flex;font-size:1.5rem;justify-content:center;left:0;pointer-events:none;position:absolute;right:0;top:0;z-index:100}.BookEditor-module_bookEditorActionbar__KW6dc{position:absolute;right:55px;top:10px;z-index:100}.BookEditor-module_fullscreen__rktsl{border:none;border-radius:0;bottom:0;box-shadow:none;height:100%;left:0;padding-top:50px;position:fixed;right:0;top:0;width:100%;z-index:9999}.BookEditor-module_button__hS390{align-items:center;background-color:#fff;border:1px solid #d1d5db;border-radius:.375rem;box-shadow:0 1px 2px 0 rgba(0,0,0,.05);color:#374151;cursor:pointer;display:inline-flex;gap:.5rem;padding:.5rem 1rem;transition:all .2s ease-in-out}.BookEditor-module_button__hS390:hover{background-color:#f9fafb;border-color:#b7bcce;box-shadow:0 1px 3px 0 rgba(0,0,0,.1),0 1px 2px 0 rgba(0,0,0,.06)}.BookEditor-module_savedNotification__OiX9L{align-items:center;animation:BookEditor-module_fadeOut__q8JnR 2s forwards;background-color:rgba(0,0,0,.7);border-radius:.5rem;color:#fff;display:flex;font-size:1.25rem;gap:.5rem;left:50%;padding:1rem 2rem;pointer-events:none;position:absolute;top:50%;transform:translate(-50%,-50%);z-index:1000}@keyframes BookEditor-module_fadeOut__q8JnR{0%{opacity:0}10%{opacity:1}70%{opacity:1}to{opacity:0}}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkJvb2tFZGl0b3IubW9kdWxlLmNzcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxxQ0FFSSxVQUNKLENBRUEsOENBS0kscUJBQXNCLENBSHRCLFdBQVksQ0FFWixtQkFBb0IsQ0FEcEIsaUJBQWtCLENBRmxCLFVBS0osQ0FFQSxpRkFFSSxnQ0FBcUMsQ0FDckMsaUNBRUosQ0FFQSxvQ0FDSSwwQkFDSixDQUVBLDRDQUtJLHFCQUF1QixDQUR2QixxQ0FBMEMsQ0FGMUMsa0JBQW1CLENBSW5CLHNDQUEyQyxDQUwzQyxlQUFnQixDQUVoQixnQkFBaUIsQ0FJakIscUNBQ0osQ0FDQSxnRkFFSSxnQkFFSixDQUVBLGtEQUNJLHVFQUNKLENBVUEsNEZBQ0ksZUFDSixDQUVBLHNDQVNJLGtCQUFtQixDQUhuQiwrQkFBb0MsQ0FEcEMsUUFBUyxDQUVULFVBQVksQ0FDWixZQUFhLENBR2IsZ0JBQWlCLENBRGpCLHNCQUF1QixDQVB2QixNQUFPLENBVVAsbUJBQW9CLENBWnBCLGlCQUFrQixDQUdsQixPQUFRLENBRlIsS0FBTSxDQVVOLFdBRUosQ0FRQSw4Q0FDSSxpQkFBa0IsQ0FFbEIsVUFBVyxDQURYLFFBQVMsQ0FFVCxXQUNKLENBRUEscUNBV0ksV0FBWSxDQUZaLGVBQWdCLENBSmhCLFFBQVMsQ0FPVCxlQUFnQixDQUxoQixXQUFZLENBSlosTUFBTyxDQU9QLGdCQUFpQixDQVRqQixjQUFlLENBR2YsT0FBUSxDQUZSLEtBQU0sQ0FJTixVQUFXLENBRVgsWUFLSixDQUVBLGlDQU9JLGtCQUFtQixDQU5uQixxQkFBc0IsQ0FDdEIsd0JBQXlCLENBQ3pCLHFCQUF1QixDQU92QixzQ0FBMkMsQ0FEM0MsYUFBYyxDQUpkLGNBQWUsQ0FDZixtQkFBb0IsQ0FFcEIsU0FBVyxDQUpYLGtCQUFvQixDQU9wQiw4QkFDSixDQUVBLHVDQUNJLHdCQUF5QixDQUN6QixvQkFBcUIsQ0FDckIsaUVBQ0osQ0FFQSw0Q0FhSSxrQkFBbUIsQ0FFbkIsc0RBQThCLENBVjlCLCtCQUFvQyxDQUdwQyxtQkFBcUIsQ0FGckIsVUFBWSxDQU1aLFlBQWEsQ0FIYixpQkFBa0IsQ0FLbEIsU0FBVyxDQVhYLFFBQVMsQ0FJVCxpQkFBa0IsQ0FJbEIsbUJBQW9CLENBVnBCLGlCQUFrQixDQUNsQixPQUFRLENBRVIsOEJBQWdDLENBTWhDLFlBTUosQ0FFQSw0Q0FDSSxHQUNJLFNBQ0osQ0FDQSxJQUNJLFNBQ0osQ0FDQSxJQUNJLFNBQ0osQ0FDQSxHQUNJLFNBQ0osQ0FDSiIsImZpbGUiOiJCb29rRWRpdG9yLm1vZHVsZS5jc3MiLCJzb3VyY2VzQ29udGVudCI6WyIuQm9va0VkaXRvciB7XG4gICAgLyogaGVpZ2h0OiA0NTBweDsgKi9cbiAgICB3aWR0aDogMTAwJTtcbn1cblxuLmJvb2tFZGl0b3JDb250YWluZXIge1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIGhlaWdodDogMTAwJTtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgcGFkZGluZzogMTBweCAyNXB4IDA7XG4gICAgYm94LXNpemluZzogYm9yZGVyLWJveDtcbn1cblxuLmJvb2tFZGl0b3JDb250YWluZXIuaXNWZXJib3NlIHtcbiAgICAvKiovXG4gICAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgwLCAwLCAwLCAwLjA1KTtcbiAgICBvdXRsaW5lOiAxcHggZG90dGVkIHJnYmEoMCwgMCwgMCwgMC41KTtcbiAgICAvKiovXG59XG5cbi5pc1ZlcmJvc2Uge1xuICAgIG91dGxpbmU6IDJweCBkb3R0ZWQgcmdiKDI1NSAxMTcgMzgpO1xufVxuXG4uYm9va0VkaXRvcldyYXBwZXIge1xuICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgYm9yZGVyLXJhZGl1czogMXJlbTtcbiAgICBwYWRkaW5nLXRvcDogMTBweDtcbiAgICBib3JkZXI6IDFweCBzb2xpZCByZ2JhKDIwOSwgMjEzLCAyMTksIDAuOCk7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogd2hpdGU7XG4gICAgYm94LXNoYWRvdzogMCAxcHggMnB4IDAgcmdiYSgwLCAwLCAwLCAwLjA1KTtcbiAgICB0cmFuc2l0aW9uOiBib3gtc2hhZG93IDAuMnMgZWFzZS1pbi1vdXQ7XG59XG4uaXNWZXJib3NlIC5ib29rRWRpdG9yV3JhcHBlciB7XG4gICAgLyoqL1xuICAgIG92ZXJmbG93OiB2aXNpYmxlO1xuICAgIC8qKi9cbn1cblxuLmJvb2tFZGl0b3JXcmFwcGVyOmhvdmVyIHtcbiAgICBib3gtc2hhZG93OiAwIDRweCA2cHggLTFweCByZ2JhKDAsIDAsIDAsIDAuMSksIDAgMnB4IDRweCAtMXB4IHJnYmEoMCwgMCwgMCwgMC4wNik7XG59XG5cbi5ib29rRWRpdG9yV3JhcHBlcjpmb2N1cy13aXRoaW4ge1xuICAgIC8qXG4gICAgb3V0bGluZTogMnB4IHNvbGlkIHRyYW5zcGFyZW50O1xuICAgIG91dGxpbmUtb2Zmc2V0OiAycHg7XG4gICAgYm94LXNoYWRvdzogMCAwIDAgM3B4IHJnYmEoOTksIDEwMiwgMjQxLCAwLjQpO1xuICAgICovXG59XG5cbi5ib29rRWRpdG9yV3JhcHBlci5pc0JvcmRlclJhZGl1c0Rpc2FibGVkIHtcbiAgICBib3JkZXItcmFkaXVzOiAwO1xufVxuXG4uZHJvcE92ZXJsYXkge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB0b3A6IDA7XG4gICAgbGVmdDogMDtcbiAgICByaWdodDogMDtcbiAgICBib3R0b206IDA7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogcmdiYSgwLCAwLCAwLCAwLjUpO1xuICAgIGNvbG9yOiB3aGl0ZTtcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4gICAgZm9udC1zaXplOiAxLjVyZW07XG4gICAgei1pbmRleDogMTAwO1xuICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xufVxuXG4vKlxuLmJvb2tFZGl0b3JDb250YWluZXIgOmdsb2JhbCgudmlldy1saW5lKSB7XG4gICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkICNlZWU7IC8qIDwtIE5vdGU6IGBQUk9NUFRCT09LX1NZTlRBWF9DT0xPUlMuTElORWAgKiAvXG59XG4qL1xuXG4uYm9va0VkaXRvckFjdGlvbmJhciB7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHRvcDogMTBweDtcbiAgICByaWdodDogNTVweDtcbiAgICB6LWluZGV4OiAxMDA7XG59XG5cbi5mdWxsc2NyZWVuIHtcbiAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgdG9wOiAwO1xuICAgIGxlZnQ6IDA7XG4gICAgcmlnaHQ6IDA7XG4gICAgYm90dG9tOiAwO1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIGhlaWdodDogMTAwJTtcbiAgICB6LWluZGV4OiA5OTk5O1xuICAgIGJvcmRlci1yYWRpdXM6IDA7XG4gICAgcGFkZGluZy10b3A6IDUwcHg7XG4gICAgYm9yZGVyOiBub25lO1xuICAgIGJveC1zaGFkb3c6IG5vbmU7XG59XG5cbi5idXR0b24ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICNmZmY7XG4gICAgYm9yZGVyOiAxcHggc29saWQgI2QxZDVkYjtcbiAgICBib3JkZXItcmFkaXVzOiAwLjM3NXJlbTtcbiAgICBwYWRkaW5nOiAwLjVyZW0gMXJlbTtcbiAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgZGlzcGxheTogaW5saW5lLWZsZXg7XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICBnYXA6IDAuNXJlbTtcbiAgICBjb2xvcjogIzM3NDE1MTtcbiAgICBib3gtc2hhZG93OiAwIDFweCAycHggMCByZ2JhKDAsIDAsIDAsIDAuMDUpO1xuICAgIHRyYW5zaXRpb246IGFsbCAwLjJzIGVhc2UtaW4tb3V0O1xufVxuXG4uYnV0dG9uOmhvdmVyIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZjlmYWZiO1xuICAgIGJvcmRlci1jb2xvcjogI2I3YmNjZTtcbiAgICBib3gtc2hhZG93OiAwIDFweCAzcHggMCByZ2JhKDAsIDAsIDAsIDAuMSksIDAgMXB4IDJweCAwIHJnYmEoMCwgMCwgMCwgMC4wNik7XG59XG5cbi5zYXZlZE5vdGlmaWNhdGlvbiB7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgIHRvcDogNTAlO1xuICAgIGxlZnQ6IDUwJTtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZSgtNTAlLCAtNTAlKTtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsIDAsIDAsIDAuNyk7XG4gICAgY29sb3I6IHdoaXRlO1xuICAgIHBhZGRpbmc6IDFyZW0gMnJlbTtcbiAgICBib3JkZXItcmFkaXVzOiAwLjVyZW07XG4gICAgZm9udC1zaXplOiAxLjI1cmVtO1xuICAgIHotaW5kZXg6IDEwMDA7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gICAgZGlzcGxheTogZmxleDtcbiAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICAgIGdhcDogMC41cmVtO1xuICAgIGFuaW1hdGlvbjogZmFkZU91dCAycyBmb3J3YXJkcztcbn1cblxuQGtleWZyYW1lcyBmYWRlT3V0IHtcbiAgICAwJSB7XG4gICAgICAgIG9wYWNpdHk6IDA7XG4gICAgfVxuICAgIDEwJSB7XG4gICAgICAgIG9wYWNpdHk6IDE7XG4gICAgfVxuICAgIDcwJSB7XG4gICAgICAgIG9wYWNpdHk6IDE7XG4gICAgfVxuICAgIDEwMCUge1xuICAgICAgICBvcGFjaXR5OiAwO1xuICAgIH1cbn1cbiJdfQ== */";
|
|
9090
|
-
var styles$
|
|
9540
|
+
var styles$8 = {"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"};
|
|
9091
9541
|
styleInject(css_248z$8);
|
|
9092
9542
|
|
|
9093
9543
|
/**
|
|
@@ -9240,7 +9690,7 @@ function downloadFile(content, filename, mimeType) {
|
|
|
9240
9690
|
}
|
|
9241
9691
|
|
|
9242
9692
|
var css_248z$7 = ".CodeBlock-module_CodeBlock__6K33Z{border:1px solid #444;border-radius:5px;margin:10px 0;overflow:hidden}.CodeBlock-module_CodeBlockHeader__tfOwl{align-items:center;background-color:#2d2d2d;border-bottom:1px solid #444;color:#ccc;display:flex;font-size:.8em;justify-content:space-between;padding:5px 10px}.CodeBlock-module_CodeBlockButtons__rz1VO{display:flex;gap:8px}.CodeBlock-module_CopyButton__M07tp,.CodeBlock-module_CreateAgentButton__kUEnp,.CodeBlock-module_DownloadButton__ZTrzQ{align-items:center;background:transparent;border:none;color:#aaa;cursor:pointer;display:flex;font-size:inherit;gap:5px;padding:0;transition:color .2s}.CodeBlock-module_CopyButton__M07tp:hover,.CodeBlock-module_CreateAgentButton__kUEnp:hover,.CodeBlock-module_DownloadButton__ZTrzQ:hover{color:#fff}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkNvZGVCbG9jay5tb2R1bGUuY3NzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLG1DQUlJLHFCQUFzQixDQUZ0QixpQkFBa0IsQ0FEbEIsYUFBYyxDQUVkLGVBRUosQ0FFQSx5Q0FRSSxrQkFBbUIsQ0FQbkIsd0JBQXlCLENBSXpCLDRCQUE2QixDQUg3QixVQUFXLENBSVgsWUFBYSxDQUZiLGNBQWdCLENBR2hCLDZCQUE4QixDQUo5QixnQkFNSixDQUVBLDBDQUNJLFlBQWEsQ0FDYixPQUNKLENBRUEsdUhBVUksa0JBQW1CLENBUG5CLHNCQUF1QixDQUN2QixXQUFZLENBQ1osVUFBVyxDQUNYLGNBQWUsQ0FHZixZQUFhLENBRmIsaUJBQWtCLENBSWxCLE9BQVEsQ0FIUixTQUFVLENBSVYsb0JBQ0osQ0FFQSx5SUFHSSxVQUNKIiwiZmlsZSI6IkNvZGVCbG9jay5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLkNvZGVCbG9jayB7XG4gICAgbWFyZ2luOiAxMHB4IDA7XG4gICAgYm9yZGVyLXJhZGl1czogNXB4O1xuICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgYm9yZGVyOiAxcHggc29saWQgIzQ0NDsgLyogRGFyayBtb2RlIGJvcmRlciAqL1xufVxuXG4uQ29kZUJsb2NrSGVhZGVyIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjMmQyZDJkO1xuICAgIGNvbG9yOiAjY2NjO1xuICAgIHBhZGRpbmc6IDVweCAxMHB4O1xuICAgIGZvbnQtc2l6ZTogMC44ZW07XG4gICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkICM0NDQ7XG4gICAgZGlzcGxheTogZmxleDtcbiAgICBqdXN0aWZ5LWNvbnRlbnQ6IHNwYWNlLWJldHdlZW47XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbn1cblxuLkNvZGVCbG9ja0J1dHRvbnMge1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgZ2FwOiA4cHg7XG59XG5cbi5Db3B5QnV0dG9uLFxuLkRvd25sb2FkQnV0dG9uLFxuLkNyZWF0ZUFnZW50QnV0dG9uIHtcbiAgICBiYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDtcbiAgICBib3JkZXI6IG5vbmU7XG4gICAgY29sb3I6ICNhYWE7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIGZvbnQtc2l6ZTogaW5oZXJpdDtcbiAgICBwYWRkaW5nOiAwO1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICBnYXA6IDVweDtcbiAgICB0cmFuc2l0aW9uOiBjb2xvciAwLjJzO1xufVxuXG4uQ29weUJ1dHRvbjpob3Zlcixcbi5Eb3dubG9hZEJ1dHRvbjpob3Zlcixcbi5DcmVhdGVBZ2VudEJ1dHRvbjpob3ZlciB7XG4gICAgY29sb3I6ICNmZmY7XG59XG4iXX0= */";
|
|
9243
|
-
var styles$
|
|
9693
|
+
var styles$7 = {"CodeBlock":"CodeBlock-module_CodeBlock__6K33Z","CodeBlockHeader":"CodeBlock-module_CodeBlockHeader__tfOwl","CodeBlockButtons":"CodeBlock-module_CodeBlockButtons__rz1VO","CopyButton":"CodeBlock-module_CopyButton__M07tp","DownloadButton":"CodeBlock-module_DownloadButton__ZTrzQ","CreateAgentButton":"CodeBlock-module_CreateAgentButton__kUEnp"};
|
|
9244
9694
|
styleInject(css_248z$7);
|
|
9245
9695
|
|
|
9246
9696
|
const LANGUAGE_EXTENSIONS = {
|
|
@@ -9288,11 +9738,11 @@ function CodeBlock({ code, language, className, onCreateAgent }) {
|
|
|
9288
9738
|
console.error('Failed to copy code to clipboard:', error);
|
|
9289
9739
|
}
|
|
9290
9740
|
};
|
|
9291
|
-
const header = language ? (jsxs("div", { className: styles$
|
|
9741
|
+
const header = language ? (jsxs("div", { className: styles$7.CodeBlockHeader, children: [jsx("span", { children: language }), jsxs("div", { className: styles$7.CodeBlockButtons, children: [jsx("button", { onClick: handleCopy, className: styles$7.CopyButton, title: "Copy to clipboard", children: copied ? 'Copied!' : 'Copy' }), jsx("button", { onClick: handleDownload, className: styles$7.DownloadButton, title: "Download code", children: "Download" }), (language === null || language === void 0 ? void 0 : language.trim().toLowerCase()) === 'book' && onCreateAgent && (jsx("button", { onClick: () => onCreateAgent(code), className: styles$7.CreateAgentButton, title: "Create agent from this book", children: "Create Agent" }))] })] })) : null;
|
|
9292
9742
|
if ((language === null || language === void 0 ? void 0 : language.trim().toLowerCase()) === 'book') {
|
|
9293
|
-
return (jsxs("div", { className: classNames(styles$
|
|
9743
|
+
return (jsxs("div", { className: classNames(styles$7.CodeBlock, className), children: [header, jsx(BookEditor, { value: code, isReadonly: true, height: lines * 25 /* <- [🧠] A bit more than 19px to accommodate BookEditor lines */ })] }));
|
|
9294
9744
|
}
|
|
9295
|
-
return (jsxs("div", { className: classNames(styles$
|
|
9745
|
+
return (jsxs("div", { className: classNames(styles$7.CodeBlock, className), children: [header, jsx(Editor, { height: `${height}px`, language: language || 'plaintext', value: code, theme: "vs-dark", options: {
|
|
9296
9746
|
readOnly: true,
|
|
9297
9747
|
minimap: { enabled: false },
|
|
9298
9748
|
scrollBeyondLastLine: false,
|
|
@@ -9317,7 +9767,7 @@ function CodeBlock({ code, language, className, onCreateAgent }) {
|
|
|
9317
9767
|
}
|
|
9318
9768
|
|
|
9319
9769
|
var css_248z$6 = ".MarkdownContent-module_MarkdownContent__2JuyX h1{font-size:2em}.MarkdownContent-module_MarkdownContent__2JuyX h2{font-size:1.75em}.MarkdownContent-module_MarkdownContent__2JuyX h3{font-size:1.5em}.MarkdownContent-module_MarkdownContent__2JuyX h4{font-size:1.25em}.MarkdownContent-module_MarkdownContent__2JuyX h5{font-size:1.1em}.MarkdownContent-module_MarkdownContent__2JuyX p{text-wrap:auto}.MarkdownContent-module_MarkdownContent__2JuyX ul{list-style:disc;margin-left:20px}.MarkdownContent-module_MarkdownContent__2JuyX ol{list-style:decimal;margin-left:20px}.MarkdownContent-module_MarkdownContent__2JuyX blockquote,.MarkdownContent-module_MarkdownContent__2JuyX img,.MarkdownContent-module_MarkdownContent__2JuyX pre,.MarkdownContent-module_MarkdownContent__2JuyX table{border-radius:8px;margin-bottom:10px;margin-top:10px}.MarkdownContent-module_MarkdownContent__2JuyX pre{background:#000;border:none;box-shadow:none;color:#fff;display:block;font-size:inherit;line-height:inherit;padding:1em}.MarkdownContent-module_MarkdownContent__2JuyX blockquote{background:#ffffffcc;border:none;box-shadow:none;color:#000;display:block;font-size:inherit;line-height:inherit;padding:1em}.MarkdownContent-module_MarkdownContent__2JuyX code{background:#cccccc55;border:none;box-shadow:none;color:inherit;display:inline-block;font-size:inherit;line-height:inherit;margin:0;padding:0}.MarkdownContent-module_MarkdownContent__2JuyX pre code{background-color:#000000cc;border-radius:8px}.MarkdownContent-module_MarkdownContent__2JuyX .MarkdownContent-module_chat-code-block__ZffFg{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}.MarkdownContent-module_MarkdownContent__2JuyX .MarkdownContent-module_chat-code-block__ZffFg 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}.MarkdownContent-module_MarkdownContent__2JuyX 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%}.MarkdownContent-module_MarkdownContent__2JuyX td,.MarkdownContent-module_MarkdownContent__2JuyX th{background:none;border-bottom:1px solid #d1dbe8;color:#17223b;padding:10px 16px;text-align:left}.MarkdownContent-module_MarkdownContent__2JuyX th{background:linear-gradient(90deg,#eaf3fa 80%,#d1e3f8);border-bottom:2px solid #b5c7de;color:#17223b;font-weight:700}.MarkdownContent-module_MarkdownContent__2JuyX tr:last-child td{border-bottom:none}.MarkdownContent-module_MarkdownContent__2JuyX tr:nth-child(2n) td{background:#eaf3fa}.MarkdownContent-module_MarkdownContent__2JuyX tr:hover td{background:#cbe0f7;transition:background .2s}.MarkdownContent-module_MarkdownContent__2JuyX table{border-radius:12px;overflow:hidden}.MarkdownContent-module_MarkdownContent__2JuyX td:first-child,.MarkdownContent-module_MarkdownContent__2JuyX th:first-child{border-top-left-radius:12px}.MarkdownContent-module_MarkdownContent__2JuyX td:last-child,.MarkdownContent-module_MarkdownContent__2JuyX th:last-child{border-top-right-radius:12px}.MarkdownContent-module_citation__11SMw{background:#e7f1ff;border-radius:4px;color:#007bff;cursor:default;display:inline-block;font-size:.7em;font-weight:700;margin:0 2px;padding:0 4px;user-select:none;vertical-align:super}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIk1hcmtkb3duQ29udGVudC5tb2R1bGUuY3NzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGtEQUNJLGFBQ0osQ0FFQSxrREFDSSxnQkFDSixDQUVBLGtEQUNJLGVBQ0osQ0FFQSxrREFDSSxnQkFDSixDQUVBLGtEQUNJLGVBQ0osQ0FFQSxpREFDSSxjQUNKLENBRUEsa0RBQ0ksZUFBZ0IsQ0FDaEIsZ0JBQ0osQ0FFQSxrREFDSSxrQkFBbUIsQ0FDbkIsZ0JBQ0osQ0FFQSxxTkFNSSxpQkFBa0IsQ0FEbEIsa0JBQW1CLENBRG5CLGVBR0osQ0FFQSxtREFJSSxlQUFxQixDQUZyQixXQUFZLENBQ1osZUFBZ0IsQ0FJaEIsVUFBVyxDQU5YLGFBQWMsQ0FJZCxpQkFBa0IsQ0FDbEIsbUJBQW9CLENBRXBCLFdBQ0osQ0FFQSwwREFJSSxvQkFBcUIsQ0FGckIsV0FBWSxDQUNaLGVBQWdCLENBSWhCLFVBQVcsQ0FOWCxhQUFjLENBSWQsaUJBQWtCLENBQ2xCLG1CQUFvQixDQUVwQixXQUNKLENBRUEsb0RBTUksb0JBQXFCLENBRnJCLFdBQVksQ0FDWixlQUFnQixDQUloQixhQUFjLENBUmQsb0JBQXFCLENBTXJCLGlCQUFrQixDQUNsQixtQkFBb0IsQ0FOcEIsUUFBUyxDQUNULFNBT0osQ0FFQSx3REFDSSwwQkFBMkIsQ0FDM0IsaUJBQ0osQ0FFQSw4RkFDSSxrQkFBbUIsQ0FLbkIsb0JBQXFCLENBQ3JCLG9DQUF5QyxDQUx6QyxhQUFjLENBTWQsOERBQTJFLENBTDNFLGNBQWUsQ0FDZixlQUFnQixDQUNoQixlQUlKLENBQ0EsbUdBQ0kseUJBQTJCLENBSzNCLHFCQUF1QixDQUN2Qix5QkFBMkIsQ0FMM0IsdUJBQXlCLENBU3pCLGFBQWMsQ0FSZCw2QkFBK0IsQ0FDL0IsMkJBQTZCLENBTTdCLGVBQWdCLENBTGhCLG1CQUFxQixDQUdyQixlQUFnQixDQUNoQixxQkFHSixDQUNBLHFEQUtJLGtCQUFtQixDQUhuQix3QkFBeUIsQ0FDekIsZ0JBQWlCLENBS2pCLG9DQUF5QyxDQUV6QyxhQUFjLENBRGQsY0FBZSxDQUxmLGFBQWMsQ0FIZCxVQVVKLENBQ0Esb0dBTUksZUFBZ0IsQ0FIaEIsK0JBQWdDLENBRWhDLGFBQWMsQ0FIZCxpQkFBa0IsQ0FFbEIsZUFHSixDQUNBLGtEQUNJLHFEQUE2RCxDQUc3RCwrQkFBZ0MsQ0FEaEMsYUFBYyxDQURkLGVBR0osQ0FDQSxnRUFDSSxrQkFDSixDQUNBLG1FQUNJLGtCQUNKLENBQ0EsMkRBQ0ksa0JBQW1CLENBQ25CLHlCQUNKLENBQ0EscURBQ0ksa0JBQW1CLENBQ25CLGVBQ0osQ0FDQSw0SEFFSSwyQkFDSixDQUNBLDBIQUVJLDRCQUNKLENBRUEsd0NBTUksa0JBQW1CLENBRW5CLGlCQUFrQixDQUhsQixhQUFjLENBS2QsY0FBZSxDQVRmLG9CQUFxQixDQUVyQixjQUFnQixDQUNoQixlQUFpQixDQUtqQixZQUFhLENBRmIsYUFBYyxDQUlkLGdCQUFpQixDQVRqQixvQkFVSiIsImZpbGUiOiJNYXJrZG93bkNvbnRlbnQubW9kdWxlLmNzcyIsInNvdXJjZXNDb250ZW50IjpbIi5NYXJrZG93bkNvbnRlbnQgaDEge1xuICAgIGZvbnQtc2l6ZTogMmVtO1xufVxuXG4uTWFya2Rvd25Db250ZW50IGgyIHtcbiAgICBmb250LXNpemU6IDEuNzVlbTtcbn1cblxuLk1hcmtkb3duQ29udGVudCBoMyB7XG4gICAgZm9udC1zaXplOiAxLjVlbTtcbn1cblxuLk1hcmtkb3duQ29udGVudCBoNCB7XG4gICAgZm9udC1zaXplOiAxLjI1ZW07XG59XG5cbi5NYXJrZG93bkNvbnRlbnQgaDUge1xuICAgIGZvbnQtc2l6ZTogMS4xZW07XG59XG5cbi5NYXJrZG93bkNvbnRlbnQgcCB7XG4gICAgdGV4dC13cmFwOiBhdXRvO1xufVxuXG4uTWFya2Rvd25Db250ZW50IHVsIHtcbiAgICBsaXN0LXN0eWxlOiBkaXNjO1xuICAgIG1hcmdpbi1sZWZ0OiAyMHB4O1xufVxuXG4uTWFya2Rvd25Db250ZW50IG9sIHtcbiAgICBsaXN0LXN0eWxlOiBkZWNpbWFsO1xuICAgIG1hcmdpbi1sZWZ0OiAyMHB4O1xufVxuXG4uTWFya2Rvd25Db250ZW50IGltZyxcbi5NYXJrZG93bkNvbnRlbnQgcHJlLFxuLk1hcmtkb3duQ29udGVudCBibG9ja3F1b3RlLFxuLk1hcmtkb3duQ29udGVudCB0YWJsZSB7XG4gICAgbWFyZ2luLXRvcDogMTBweDtcbiAgICBtYXJnaW4tYm90dG9tOiAxMHB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDhweDtcbn1cblxuLk1hcmtkb3duQ29udGVudCBwcmUge1xuICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgIGJvcmRlcjogbm9uZTtcbiAgICBib3gtc2hhZG93OiBub25lO1xuICAgIGJhY2tncm91bmQ6ICMwMDAwMDBmZjtcbiAgICBmb250LXNpemU6IGluaGVyaXQ7XG4gICAgbGluZS1oZWlnaHQ6IGluaGVyaXQ7XG4gICAgY29sb3I6ICNmZmY7XG4gICAgcGFkZGluZzogMWVtO1xufVxuXG4uTWFya2Rvd25Db250ZW50IGJsb2NrcXVvdGUge1xuICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgIGJvcmRlcjogbm9uZTtcbiAgICBib3gtc2hhZG93OiBub25lO1xuICAgIGJhY2tncm91bmQ6ICNmZmZmZmZjYztcbiAgICBmb250LXNpemU6IGluaGVyaXQ7XG4gICAgbGluZS1oZWlnaHQ6IGluaGVyaXQ7XG4gICAgY29sb3I6ICMwMDA7XG4gICAgcGFkZGluZzogMWVtO1xufVxuXG4uTWFya2Rvd25Db250ZW50IGNvZGUge1xuICAgIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgICBtYXJnaW46IDA7XG4gICAgcGFkZGluZzogMDtcbiAgICBib3JkZXI6IG5vbmU7XG4gICAgYm94LXNoYWRvdzogbm9uZTtcbiAgICBiYWNrZ3JvdW5kOiAjY2NjY2NjNTU7XG4gICAgZm9udC1zaXplOiBpbmhlcml0O1xuICAgIGxpbmUtaGVpZ2h0OiBpbmhlcml0O1xuICAgIGNvbG9yOiBpbmhlcml0O1xufVxuXG4uTWFya2Rvd25Db250ZW50IHByZSBjb2RlIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjMDAwMDAwY2M7XG4gICAgYm9yZGVyLXJhZGl1czogOHB4O1xufVxuXG4uTWFya2Rvd25Db250ZW50IC5jaGF0LWNvZGUtYmxvY2sge1xuICAgIGJhY2tncm91bmQ6ICMxODFjMjM7XG4gICAgY29sb3I6ICNmOGZhZmM7XG4gICAgZm9udC1zaXplOiAxNHB4O1xuICAgIGxpbmUtaGVpZ2h0OiAxLjY7XG4gICAgb3ZlcmZsb3cteDogYXV0bztcbiAgICBib3JkZXItY29sb3I6ICMyMzI3MmY7XG4gICAgYm94LXNoYWRvdzogMCAycHggOHB4IHJnYmEoMCwgMCwgMCwgMC4xMik7XG4gICAgZm9udC1mYW1pbHk6ICdGaXJhIE1vbm8nLCAnTWVubG8nLCAnQ29uc29sYXMnLCAnTGliZXJhdGlvbiBNb25vJywgbW9ub3NwYWNlO1xufVxuLk1hcmtkb3duQ29udGVudCAuY2hhdC1jb2RlLWJsb2NrIGNvZGUge1xuICAgIGJhY2tncm91bmQ6IG5vbmUgIWltcG9ydGFudDtcbiAgICBjb2xvcjogaW5oZXJpdCAhaW1wb3J0YW50O1xuICAgIGZvbnQtZmFtaWx5OiBpbmhlcml0ICFpbXBvcnRhbnQ7XG4gICAgZm9udC1zaXplOiBpbmhlcml0ICFpbXBvcnRhbnQ7XG4gICAgcGFkZGluZzogMCAhaW1wb3J0YW50O1xuICAgIGJvcmRlcjogbm9uZSAhaW1wb3J0YW50O1xuICAgIGJveC1zaGFkb3c6IG5vbmUgIWltcG9ydGFudDtcbiAgICB3aGl0ZS1zcGFjZTogcHJlO1xuICAgIHdvcmQtYnJlYWs6IGJyZWFrLXdvcmQ7XG4gICAgb3ZlcmZsb3cteDogYXV0bztcbiAgICBkaXNwbGF5OiBibG9jaztcbn1cbi5NYXJrZG93bkNvbnRlbnQgdGFibGUge1xuICAgIHdpZHRoOiAxMDAlO1xuICAgIGJvcmRlci1jb2xsYXBzZTogc2VwYXJhdGU7XG4gICAgYm9yZGVyLXNwYWNpbmc6IDA7XG4gICAgbWFyZ2luOiAxNnB4IDA7XG4gICAgYmFja2dyb3VuZDogI2Y4ZmFmYzsgLyogU3Ryb25nZXIgbGlnaHQgYmFja2dyb3VuZCBmb3IgY29udHJhc3QgKi9cbiAgICBib3JkZXItcmFkaXVzOiAxMnB4O1xuICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgYm94LXNoYWRvdzogMCAycHggOHB4IHJnYmEoMCwgMCwgMCwgMC4wOCk7XG4gICAgZm9udC1zaXplOiAxNHB4O1xuICAgIGNvbG9yOiAjMTcyMjNiOyAvKiBEYXJrIHRleHQgZm9yIGNvbnRyYXN0ICovXG59XG4uTWFya2Rvd25Db250ZW50IHRoLFxuLk1hcmtkb3duQ29udGVudCB0ZCB7XG4gICAgcGFkZGluZzogMTBweCAxNnB4O1xuICAgIGJvcmRlci1ib3R0b206IDFweCBzb2xpZCAjZDFkYmU4O1xuICAgIHRleHQtYWxpZ246IGxlZnQ7XG4gICAgY29sb3I6ICMxNzIyM2I7IC8qIEVuc3VyZSBzdHJvbmcgdGV4dCBjb2xvciBmb3IgYWxsIGNlbGxzICovXG4gICAgYmFja2dyb3VuZDogbm9uZTtcbn1cbi5NYXJrZG93bkNvbnRlbnQgdGgge1xuICAgIGJhY2tncm91bmQ6IGxpbmVhci1ncmFkaWVudCg5MGRlZywgI2VhZjNmYSA4MCUsICNkMWUzZjggMTAwJSk7XG4gICAgZm9udC13ZWlnaHQ6IDcwMDtcbiAgICBjb2xvcjogIzE3MjIzYjsgLyogU3Ryb25nIGhlYWRlciB0ZXh0ICovXG4gICAgYm9yZGVyLWJvdHRvbTogMnB4IHNvbGlkICNiNWM3ZGU7XG59XG4uTWFya2Rvd25Db250ZW50IHRyOmxhc3QtY2hpbGQgdGQge1xuICAgIGJvcmRlci1ib3R0b206IG5vbmU7XG59XG4uTWFya2Rvd25Db250ZW50IHRyOm50aC1jaGlsZChldmVuKSB0ZCB7XG4gICAgYmFja2dyb3VuZDogI2VhZjNmYTtcbn1cbi5NYXJrZG93bkNvbnRlbnQgdHI6aG92ZXIgdGQge1xuICAgIGJhY2tncm91bmQ6ICNjYmUwZjc7XG4gICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZCAwLjJzO1xufVxuLk1hcmtkb3duQ29udGVudCB0YWJsZSB7XG4gICAgYm9yZGVyLXJhZGl1czogMTJweDtcbiAgICBvdmVyZmxvdzogaGlkZGVuO1xufVxuLk1hcmtkb3duQ29udGVudCB0aDpmaXJzdC1jaGlsZCxcbi5NYXJrZG93bkNvbnRlbnQgdGQ6Zmlyc3QtY2hpbGQge1xuICAgIGJvcmRlci10b3AtbGVmdC1yYWRpdXM6IDEycHg7XG59XG4uTWFya2Rvd25Db250ZW50IHRoOmxhc3QtY2hpbGQsXG4uTWFya2Rvd25Db250ZW50IHRkOmxhc3QtY2hpbGQge1xuICAgIGJvcmRlci10b3AtcmlnaHQtcmFkaXVzOiAxMnB4O1xufVxuXG4uY2l0YXRpb24ge1xuICAgIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgICB2ZXJ0aWNhbC1hbGlnbjogc3VwZXI7XG4gICAgZm9udC1zaXplOiAwLjdlbTtcbiAgICBmb250LXdlaWdodDogYm9sZDtcbiAgICBjb2xvcjogIzAwN2JmZjtcbiAgICBiYWNrZ3JvdW5kOiAjZTdmMWZmO1xuICAgIHBhZGRpbmc6IDAgNHB4O1xuICAgIGJvcmRlci1yYWRpdXM6IDRweDtcbiAgICBtYXJnaW46IDAgMnB4O1xuICAgIGN1cnNvcjogZGVmYXVsdDtcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcbn1cblxuLyoqXG4gKiBUT0RPOiBb8J+MiV0gRFJZIE1hcmtkb3duIHByaW1pdGl2ZXMgc3R5bGluZ1xuICovXG4iXX0= */";
|
|
9320
|
-
var styles$
|
|
9770
|
+
var styles$6 = {"MarkdownContent":"MarkdownContent-module_MarkdownContent__2JuyX","chat-code-block":"MarkdownContent-module_chat-code-block__ZffFg","citation":"MarkdownContent-module_citation__11SMw"};
|
|
9321
9771
|
styleInject(css_248z$6);
|
|
9322
9772
|
|
|
9323
9773
|
/**
|
|
@@ -9362,7 +9812,7 @@ function createChatMarkdownConverter() {
|
|
|
9362
9812
|
type: 'lang',
|
|
9363
9813
|
regex: /【(.*?)†source】/g,
|
|
9364
9814
|
replace: (match, content) => {
|
|
9365
|
-
return `<span class="${styles$
|
|
9815
|
+
return `<span class="${styles$6.citation}">${content}</span>`;
|
|
9366
9816
|
},
|
|
9367
9817
|
}),
|
|
9368
9818
|
],
|
|
@@ -9502,7 +9952,7 @@ function MarkdownContent(props) {
|
|
|
9502
9952
|
rootsRef.current = [];
|
|
9503
9953
|
};
|
|
9504
9954
|
}, [htmlContent, onCreateAgent]);
|
|
9505
|
-
return (jsx("div", { ref: containerRef, className: classNames(styles$
|
|
9955
|
+
return (jsx("div", { ref: containerRef, className: classNames(styles$6.MarkdownContent, className), dangerouslySetInnerHTML: {
|
|
9506
9956
|
__html: htmlContent,
|
|
9507
9957
|
} }));
|
|
9508
9958
|
}
|
|
@@ -9757,7 +10207,7 @@ function FullscreenIcon() {
|
|
|
9757
10207
|
}
|
|
9758
10208
|
|
|
9759
10209
|
var css_248z$5 = ".HamburgerMenu-module_MenuHamburger__Jd-wk{align-items:center;cursor:pointer;display:flex;height:48px;justify-content:center;width:48px}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L{align-items:center;display:flex;height:24px;justify-content:center;position:relative;width:24px}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L .HamburgerMenu-module_bar__-HXPB{background-color:currentColor;border-radius:2px;display:block;height:2px;position:absolute;transition:all .3s cubic-bezier(.645,.045,.355,1);width:100%}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar1__5xeCB{transform:translateY(-7px)}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar2__Y-Ilp{opacity:1;transform:translateY(0)}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar3__m1m7T{transform:translateY(7px)}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L.HamburgerMenu-module_open__5O7zI .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar1__5xeCB{transform:translateY(0) rotate(45deg)}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L.HamburgerMenu-module_open__5O7zI .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar2__Y-Ilp{opacity:0;transform:translateX(-10px)}.HamburgerMenu-module_MenuHamburger__Jd-wk .HamburgerMenu-module_MenuHamburgerInner__Li47L.HamburgerMenu-module_open__5O7zI .HamburgerMenu-module_bar__-HXPB.HamburgerMenu-module_bar3__m1m7T{transform:translateY(0) rotate(-45deg)}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkhhbWJ1cmdlck1lbnUubW9kdWxlLmNzcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwyQ0FJSSxrQkFBbUIsQ0FIbkIsY0FBZSxDQUVmLFlBQWEsQ0FJYixXQUFZLENBRlosc0JBQXVCLENBQ3ZCLFVBRUosQ0FFQSwyRkFNSSxrQkFBbUIsQ0FGbkIsWUFBYSxDQUZiLFdBQVksQ0FHWixzQkFBdUIsQ0FGdkIsaUJBQWtCLENBRmxCLFVBTUosQ0FFQSw0SEFHSSw2QkFBOEIsQ0FFOUIsaUJBQWtCLENBRGxCLGFBQWMsQ0FIZCxVQUFXLENBTVgsaUJBQWtCLENBRGxCLGlEQUF5RCxDQUp6RCxVQU1KLENBRUEsNkpBQ0ksMEJBQ0osQ0FFQSw2SkFFSSxTQUFVLENBRFYsdUJBRUosQ0FFQSw2SkFDSSx5QkFDSixDQUVBLDhMQUNJLHFDQUNKLENBRUEsOExBQ0ksU0FBVSxDQUNWLDJCQUNKLENBRUEsOExBQ0ksc0NBQ0oiLCJmaWxlIjoiSGFtYnVyZ2VyTWVudS5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLk1lbnVIYW1idXJnZXIge1xuICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICAvKiA8LSBOb3RlOiBAc2VlIGh0dHA6Ly9lbGVrdHJvbm90ZGllbnN0LW51ZXJuYmVyZy5kZS9idWdzL2N1cnNvci5odG1sICovXG4gICAgZGlzcGxheTogZmxleDtcbiAgICBhbGlnbi1pdGVtczogY2VudGVyO1xuICAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIHdpZHRoOiA0OHB4O1xuICAgIGhlaWdodDogNDhweDtcbn1cblxuLk1lbnVIYW1idXJnZXIgLk1lbnVIYW1idXJnZXJJbm5lciB7XG4gICAgd2lkdGg6IDI0cHg7XG4gICAgaGVpZ2h0OiAyNHB4O1xuICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICBkaXNwbGF5OiBmbGV4O1xuICAgIGp1c3RpZnktY29udGVudDogY2VudGVyO1xuICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG59XG5cbi5NZW51SGFtYnVyZ2VyIC5NZW51SGFtYnVyZ2VySW5uZXIgLmJhciB7XG4gICAgaGVpZ2h0OiAycHg7XG4gICAgd2lkdGg6IDEwMCU7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogY3VycmVudENvbG9yO1xuICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgIGJvcmRlci1yYWRpdXM6IDJweDtcbiAgICB0cmFuc2l0aW9uOiBhbGwgMC4zcyBjdWJpYy1iZXppZXIoMC42NDUsIDAuMDQ1LCAwLjM1NSwgMSk7XG4gICAgcG9zaXRpb246IGFic29sdXRlO1xufVxuXG4uTWVudUhhbWJ1cmdlciAuTWVudUhhbWJ1cmdlcklubmVyIC5iYXIuYmFyMSB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKC03cHgpO1xufVxuXG4uTWVudUhhbWJ1cmdlciAuTWVudUhhbWJ1cmdlcklubmVyIC5iYXIuYmFyMiB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApO1xuICAgIG9wYWNpdHk6IDE7XG59XG5cbi5NZW51SGFtYnVyZ2VyIC5NZW51SGFtYnVyZ2VySW5uZXIgLmJhci5iYXIzIHtcbiAgICB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoN3B4KTtcbn1cblxuLk1lbnVIYW1idXJnZXIgLk1lbnVIYW1idXJnZXJJbm5lci5vcGVuIC5iYXIuYmFyMSB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApIHJvdGF0ZSg0NWRlZyk7XG59XG5cbi5NZW51SGFtYnVyZ2VyIC5NZW51SGFtYnVyZ2VySW5uZXIub3BlbiAuYmFyLmJhcjIge1xuICAgIG9wYWNpdHk6IDA7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKC0xMHB4KTtcbn1cblxuLk1lbnVIYW1idXJnZXIgLk1lbnVIYW1idXJnZXJJbm5lci5vcGVuIC5iYXIuYmFyMyB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApIHJvdGF0ZSgtNDVkZWcpO1xufVxuIl19 */";
|
|
9760
|
-
var styles$
|
|
10210
|
+
var styles$5 = {"MenuHamburger":"HamburgerMenu-module_MenuHamburger__Jd-wk","MenuHamburgerInner":"HamburgerMenu-module_MenuHamburgerInner__Li47L","bar":"HamburgerMenu-module_bar__-HXPB","bar1":"HamburgerMenu-module_bar1__5xeCB","bar2":"HamburgerMenu-module_bar2__Y-Ilp","bar3":"HamburgerMenu-module_bar3__m1m7T","open":"HamburgerMenu-module_open__5O7zI"};
|
|
9761
10211
|
styleInject(css_248z$5);
|
|
9762
10212
|
|
|
9763
10213
|
/**
|
|
@@ -9770,11 +10220,11 @@ styleInject(css_248z$5);
|
|
|
9770
10220
|
* @private Internal component
|
|
9771
10221
|
*/
|
|
9772
10222
|
function HamburgerMenu({ isOpen, onClick, className }) {
|
|
9773
|
-
return (jsx("div", { className: classNames(styles$
|
|
10223
|
+
return (jsx("div", { className: classNames(styles$5.MenuHamburger, className), onClick: onClick, children: jsxs("div", { className: classNames(styles$5.MenuHamburgerInner, isOpen && styles$5.open), children: [jsx("div", { className: classNames(styles$5.bar, styles$5.bar1) }), jsx("div", { className: classNames(styles$5.bar, styles$5.bar2) }), jsx("div", { className: classNames(styles$5.bar, styles$5.bar3) })] }) }));
|
|
9774
10224
|
}
|
|
9775
10225
|
|
|
9776
10226
|
var css_248z$4 = ".Dropdown-module_dropdown__ZShv1{display:inline-block;position:relative}.Dropdown-module_button__WzWhM{align-items:center;background-color:transparent;border:none;cursor:pointer;display:flex;font-size:16px;gap:10px;padding:10px;transition:background-color .2s ease-in-out}.Dropdown-module_button__WzWhM:hover{background-color:rgba(0,0,0,.05)}.Dropdown-module_button__WzWhM.Dropdown-module_isOpen__917OA .Dropdown-module_menuIcon__Mp4p8 line:first-child{transform:rotate(45deg) translate(5px,5px)}.Dropdown-module_button__WzWhM.Dropdown-module_isOpen__917OA .Dropdown-module_menuIcon__Mp4p8 line:nth-child(2){opacity:0}.Dropdown-module_button__WzWhM.Dropdown-module_isOpen__917OA .Dropdown-module_menuIcon__Mp4p8 line:nth-child(3){transform:rotate(-45deg) translate(5px,-5px)}.Dropdown-module_menu__-N5tr{background-color:#f9f9f9;border-radius:5px;box-shadow:0 8px 16px 0 rgba(0,0,0,.2);min-width:160px;overflow:hidden;position:absolute;right:0;top:100%;z-index:1}.Dropdown-module_menuItem__7FpAz{text-align:left;width:100%}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkRyb3Bkb3duLm1vZHVsZS5jc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsaUNBRUksb0JBQXFCLENBRHJCLGlCQUVKLENBRUEsK0JBT0ksa0JBQW1CLENBTm5CLDRCQUE2QixDQUM3QixXQUFZLENBQ1osY0FBZSxDQUdmLFlBQWEsQ0FEYixjQUFlLENBR2YsUUFBUyxDQUpULFlBQWEsQ0FLYiwyQ0FDSixDQUVBLHFDQUNJLGdDQUNKLENBRUEsK0dBQ0ksMENBQ0osQ0FFQSxnSEFDSSxTQUNKLENBRUEsZ0hBQ0ksNENBQ0osQ0FFQSw2QkFJSSx3QkFBeUIsQ0FJekIsaUJBQWtCLENBRmxCLHNDQUErQyxDQUQvQyxlQUFnQixDQUloQixlQUFnQixDQVJoQixpQkFBa0IsQ0FFbEIsT0FBUSxDQURSLFFBQVMsQ0FLVCxTQUdKLENBRUEsaUNBRUksZUFBZ0IsQ0FEaEIsVUFFSiIsImZpbGUiOiJEcm9wZG93bi5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLmRyb3Bkb3duIHtcbiAgICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xufVxuXG4uYnV0dG9uIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiB0cmFuc3BhcmVudDtcbiAgICBib3JkZXI6IG5vbmU7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIHBhZGRpbmc6IDEwcHg7XG4gICAgZm9udC1zaXplOiAxNnB4O1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICBnYXA6IDEwcHg7XG4gICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZC1jb2xvciAwLjJzIGVhc2UtaW4tb3V0O1xufVxuXG4uYnV0dG9uOmhvdmVyIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiByZ2JhKDAsIDAsIDAsIDAuMDUpO1xufVxuXG4uYnV0dG9uLmlzT3BlbiAubWVudUljb24gbGluZTpudGgtY2hpbGQoMSkge1xuICAgIHRyYW5zZm9ybTogcm90YXRlKDQ1ZGVnKSB0cmFuc2xhdGUoNXB4LCA1cHgpO1xufVxuXG4uYnV0dG9uLmlzT3BlbiAubWVudUljb24gbGluZTpudGgtY2hpbGQoMikge1xuICAgIG9wYWNpdHk6IDA7XG59XG5cbi5idXR0b24uaXNPcGVuIC5tZW51SWNvbiBsaW5lOm50aC1jaGlsZCgzKSB7XG4gICAgdHJhbnNmb3JtOiByb3RhdGUoLTQ1ZGVnKSB0cmFuc2xhdGUoNXB4LCAtNXB4KTtcbn1cblxuLm1lbnUge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB0b3A6IDEwMCU7XG4gICAgcmlnaHQ6IDA7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI2Y5ZjlmOTtcbiAgICBtaW4td2lkdGg6IDE2MHB4O1xuICAgIGJveC1zaGFkb3c6IDBweCA4cHggMTZweCAwcHggcmdiYSgwLCAwLCAwLCAwLjIpO1xuICAgIHotaW5kZXg6IDE7XG4gICAgYm9yZGVyLXJhZGl1czogNXB4O1xuICAgIG92ZXJmbG93OiBoaWRkZW47XG59XG5cbi5tZW51SXRlbSB7XG4gICAgd2lkdGg6IDEwMCU7XG4gICAgdGV4dC1hbGlnbjogbGVmdDtcbn1cbiJdfQ== */";
|
|
9777
|
-
var styles$
|
|
10227
|
+
var styles$4 = {"dropdown":"Dropdown-module_dropdown__ZShv1","button":"Dropdown-module_button__WzWhM","isOpen":"Dropdown-module_isOpen__917OA","menuIcon":"Dropdown-module_menuIcon__Mp4p8","menu":"Dropdown-module_menu__-N5tr","menuItem":"Dropdown-module_menuItem__7FpAz"};
|
|
9778
10228
|
styleInject(css_248z$4);
|
|
9779
10229
|
|
|
9780
10230
|
/**
|
|
@@ -9788,7 +10238,7 @@ styleInject(css_248z$4);
|
|
|
9788
10238
|
*/
|
|
9789
10239
|
function Dropdown({ actions }) {
|
|
9790
10240
|
const [isOpen, setIsOpen] = useState(false);
|
|
9791
|
-
return (jsxs("div", { className: styles$
|
|
10241
|
+
return (jsxs("div", { className: styles$4.dropdown, children: [jsx(HamburgerMenu, { className: classNames(styles$4.button, isOpen && styles$4.isOpen), isOpen: isOpen, onClick: () => setIsOpen(!isOpen) }), isOpen && (jsx("div", { className: styles$4.menu, children: actions.map(({ icon, name, onClick }) => (jsxs("button", { className: classNames(styles$4.button, styles$4.menuItem), onClick: () => {
|
|
9792
10242
|
onClick();
|
|
9793
10243
|
setIsOpen(false);
|
|
9794
10244
|
}, children: [icon, jsx("span", { children: name })] }, name))) }))] }));
|
|
@@ -9880,14 +10330,14 @@ function BookEditorActionbar(props) {
|
|
|
9880
10330
|
};
|
|
9881
10331
|
}, [menuHoisting, actions, isFullscreen]);
|
|
9882
10332
|
if (isFullscreen) {
|
|
9883
|
-
return (jsx("div", { className: styles$
|
|
10333
|
+
return (jsx("div", { className: styles$8.bookEditorActionbar, children: jsx("button", { className: styles$8.button, onClick: onFullscreenClick, children: jsx(ExitFullscreenIcon, {}) }) }));
|
|
9884
10334
|
}
|
|
9885
10335
|
if (menuHoisting && !isFullscreen) {
|
|
9886
10336
|
return (jsx(Fragment, { children: isAboutModalOpen && (jsx(Modal, { onClose: () => {
|
|
9887
10337
|
setIsAboutModalOpen(false);
|
|
9888
10338
|
}, children: jsx(AboutPromptbookInformation, {}) })) }));
|
|
9889
10339
|
}
|
|
9890
|
-
return (jsxs("div", { className: styles$
|
|
10340
|
+
return (jsxs("div", { className: styles$8.bookEditorActionbar, children: [actions.length >= 2 ? (jsx(Dropdown, { actions: actions })) : (actions.map(({ icon, name, onClick }) => (jsx("button", { className: styles$8.button, onClick: onClick, children: icon }, name)))), isAboutModalOpen && (jsx(Modal, { onClose: () => {
|
|
9891
10341
|
setIsAboutModalOpen(false);
|
|
9892
10342
|
}, children: jsx(AboutPromptbookInformation, {}) }))] }));
|
|
9893
10343
|
}
|
|
@@ -10333,7 +10783,7 @@ function BookEditorMonaco(props) {
|
|
|
10333
10783
|
event.preventDefault();
|
|
10334
10784
|
setIsDragOver(false);
|
|
10335
10785
|
}, []);
|
|
10336
|
-
return (jsxs("div", { className: classNames(styles$
|
|
10786
|
+
return (jsxs("div", { className: classNames(styles$8.bookEditorContainer, instanceClass), onDrop: handleDrop, onPaste: handlePaste, onDragOver: handleDragOver, onDragEnter: handleDragEnter, onDragLeave: handleDragLeave, children: [(isUploadButtonShown ||
|
|
10337
10787
|
isCameraButtonShown ||
|
|
10338
10788
|
isDownloadButtonShown ||
|
|
10339
10789
|
isAboutButtonShown ||
|
|
@@ -10346,7 +10796,7 @@ function BookEditorMonaco(props) {
|
|
|
10346
10796
|
onFullscreenClick,
|
|
10347
10797
|
onUploadDocument: handleUploadDocument,
|
|
10348
10798
|
onTakePhoto: handleTakePhoto,
|
|
10349
|
-
isFullscreen })), jsx("input", { type: "file", ref: fileUploadInputRef, style: { display: 'none' }, onChange: handleFileInputChange, multiple: true }), jsx("input", { type: "file", ref: cameraInputRef, style: { display: 'none' }, accept: "image/*", capture: "environment", onChange: handleFileInputChange }), isDragOver && jsx("div", { className: styles$
|
|
10799
|
+
isFullscreen })), jsx("input", { type: "file", ref: fileUploadInputRef, style: { display: 'none' }, onChange: handleFileInputChange, multiple: true }), jsx("input", { type: "file", ref: cameraInputRef, style: { display: 'none' }, accept: "image/*", capture: "environment", onChange: handleFileInputChange }), isDragOver && jsx("div", { className: styles$8.dropOverlay, children: "Drop files to upload" }), isSavedShown && (jsxs("div", { className: styles$8.savedNotification, children: [jsx(SaveIcon, {}), "Saved"] })), jsxs("div", { style: {
|
|
10350
10800
|
position: 'relative',
|
|
10351
10801
|
flex: 1,
|
|
10352
10802
|
height: '100%',
|
|
@@ -10411,7 +10861,7 @@ function BookEditorMonaco(props) {
|
|
|
10411
10861
|
arrowSize: 0,
|
|
10412
10862
|
useShadows: false,
|
|
10413
10863
|
},
|
|
10414
|
-
}, loading: jsx("div", { className: styles$
|
|
10864
|
+
}, loading: jsx("div", { className: styles$8.loading, children: "\uD83D\uDCD6" }) })] })] }));
|
|
10415
10865
|
}
|
|
10416
10866
|
|
|
10417
10867
|
/**
|
|
@@ -10433,7 +10883,7 @@ function BookEditor(props) {
|
|
|
10433
10883
|
const handleFullscreenToggle = () => {
|
|
10434
10884
|
setIsFullscreen(!isFullscreen);
|
|
10435
10885
|
};
|
|
10436
|
-
const editorContent = (jsx("div", { "data-book-component": "BookEditor", className: classNames(styles$
|
|
10886
|
+
const editorContent = (jsx("div", { "data-book-component": "BookEditor", className: classNames(styles$8.BookEditor, isVerbose && styles$8.isVerbose, styles$8.bookEditorWrapper, isBorderRadiusDisabled && styles$8.isBorderRadiusDisabled, isFullscreen && styles$8.fullscreen, className), style: isFullscreen
|
|
10437
10887
|
? style
|
|
10438
10888
|
: {
|
|
10439
10889
|
...(style || {}),
|
|
@@ -10449,7 +10899,7 @@ function BookEditor(props) {
|
|
|
10449
10899
|
}
|
|
10450
10900
|
|
|
10451
10901
|
var css_248z$3 = ".AvatarProfile-module_AvatarProfile__CUARY{align-items:flex-start;background:#f0f4f8;border:1px solid #e0e7ef;border-radius:8px;display:flex;padding:16px}.AvatarProfile-module_Avatar__-iSnq{border-radius:50%;flex-shrink:0;height:64px;margin-right:16px;object-fit:cover;width:64px}.AvatarProfile-module_AgentInfo__W0TR6{display:flex;flex-direction:column}.AvatarProfile-module_AgentName__9v6Qk{color:#2a3b4d;font-size:1.25em;font-weight:600;margin:0}.AvatarProfile-module_AgentTitle__nnAQM{color:#5a6b7d;font-size:1em;font-weight:500;margin:4px 0 0}.AvatarProfile-module_AgentDescription__ExidJ{color:#5a6b7d;font-size:.9em;margin:8px 0 0}.AvatarProfile-module_viewSourceButton__oi0dE{background-color:#fff;border:1px solid #e0e7ef;border-radius:6px;color:#2a3b4d;cursor:pointer;font-size:.9em;font-weight:500;margin-top:12px;padding:8px 12px;transition:background-color .2s}.AvatarProfile-module_viewSourceButton__oi0dE:hover{background-color:#f0f4f8}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkF2YXRhclByb2ZpbGUubW9kdWxlLmNzcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwyQ0FFSSxzQkFBdUIsQ0FHdkIsa0JBQW1CLENBQ25CLHdCQUF5QixDQUZ6QixpQkFBa0IsQ0FIbEIsWUFBYSxDQUViLFlBSUosQ0FFQSxvQ0FHSSxpQkFBa0IsQ0FHbEIsYUFBYyxDQUpkLFdBQVksQ0FHWixpQkFBa0IsQ0FEbEIsZ0JBQWlCLENBSGpCLFVBTUosQ0FFQSx1Q0FDSSxZQUFhLENBQ2IscUJBQ0osQ0FFQSx1Q0FJSSxhQUFjLENBRmQsZ0JBQWlCLENBQ2pCLGVBQWdCLENBRmhCLFFBSUosQ0FFQSx3Q0FJSSxhQUFjLENBRmQsYUFBYyxDQUNkLGVBQWdCLENBRmhCLGNBSUosQ0FFQSw4Q0FHSSxhQUFjLENBRGQsY0FBZ0IsQ0FEaEIsY0FHSixDQUVBLDhDQUtJLHFCQUFzQixDQUZ0Qix3QkFBeUIsQ0FDekIsaUJBQWtCLENBRWxCLGFBQWMsQ0FHZCxjQUFlLENBRmYsY0FBZ0IsQ0FDaEIsZUFBZ0IsQ0FQaEIsZUFBZ0IsQ0FDaEIsZ0JBQWlCLENBUWpCLCtCQUNKLENBRUEsb0RBQ0ksd0JBQ0oiLCJmaWxlIjoiQXZhdGFyUHJvZmlsZS5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLkF2YXRhclByb2ZpbGUge1xuICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgYWxpZ24taXRlbXM6IGZsZXgtc3RhcnQ7XG4gICAgcGFkZGluZzogMTZweDtcbiAgICBib3JkZXItcmFkaXVzOiA4cHg7XG4gICAgYmFja2dyb3VuZDogI2YwZjRmODtcbiAgICBib3JkZXI6IDFweCBzb2xpZCAjZTBlN2VmO1xufVxuXG4uQXZhdGFyIHtcbiAgICB3aWR0aDogNjRweDtcbiAgICBoZWlnaHQ6IDY0cHg7XG4gICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgIG9iamVjdC1maXQ6IGNvdmVyO1xuICAgIG1hcmdpbi1yaWdodDogMTZweDtcbiAgICBmbGV4LXNocmluazogMDtcbn1cblxuLkFnZW50SW5mbyB7XG4gICAgZGlzcGxheTogZmxleDtcbiAgICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xufVxuXG4uQWdlbnROYW1lIHtcbiAgICBtYXJnaW46IDA7XG4gICAgZm9udC1zaXplOiAxLjI1ZW07XG4gICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICBjb2xvcjogIzJhM2I0ZDtcbn1cblxuLkFnZW50VGl0bGUge1xuICAgIG1hcmdpbjogNHB4IDAgMCAwO1xuICAgIGZvbnQtc2l6ZTogMWVtO1xuICAgIGZvbnQtd2VpZ2h0OiA1MDA7XG4gICAgY29sb3I6ICM1YTZiN2Q7XG59XG5cbi5BZ2VudERlc2NyaXB0aW9uIHtcbiAgICBtYXJnaW46IDhweCAwIDAgMDtcbiAgICBmb250LXNpemU6IDAuOWVtO1xuICAgIGNvbG9yOiAjNWE2YjdkO1xufVxuXG4udmlld1NvdXJjZUJ1dHRvbiB7XG4gICAgbWFyZ2luLXRvcDogMTJweDtcbiAgICBwYWRkaW5nOiA4cHggMTJweDtcbiAgICBib3JkZXI6IDFweCBzb2xpZCAjZTBlN2VmO1xuICAgIGJvcmRlci1yYWRpdXM6IDZweDtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAjZmZmO1xuICAgIGNvbG9yOiAjMmEzYjRkO1xuICAgIGZvbnQtc2l6ZTogMC45ZW07XG4gICAgZm9udC13ZWlnaHQ6IDUwMDtcbiAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZC1jb2xvciAwLjJzO1xufVxuXG4udmlld1NvdXJjZUJ1dHRvbjpob3ZlciB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogI2YwZjRmODtcbn1cbiJdfQ== */";
|
|
10452
|
-
var styles$
|
|
10902
|
+
var styles$3 = {"AvatarProfile":"AvatarProfile-module_AvatarProfile__CUARY","Avatar":"AvatarProfile-module_Avatar__-iSnq","AgentInfo":"AvatarProfile-module_AgentInfo__W0TR6","AgentName":"AvatarProfile-module_AgentName__9v6Qk","AgentTitle":"AvatarProfile-module_AgentTitle__nnAQM","AgentDescription":"AvatarProfile-module_AgentDescription__ExidJ","viewSourceButton":"AvatarProfile-module_viewSourceButton__oi0dE"};
|
|
10453
10903
|
styleInject(css_248z$3);
|
|
10454
10904
|
|
|
10455
10905
|
/**
|
|
@@ -10461,7 +10911,7 @@ function AvatarProfile(props) {
|
|
|
10461
10911
|
const { agent, agentSource, className, style } = props;
|
|
10462
10912
|
const { agentName, personaDescription, meta } = agent;
|
|
10463
10913
|
const [isBookEditorVisible, setIsBookEditorVisible] = useState(false);
|
|
10464
|
-
return (jsxs(Fragment, { children: [jsxs("div", { className: classNames(styles$
|
|
10914
|
+
return (jsxs(Fragment, { children: [jsxs("div", { className: classNames(styles$3.AvatarProfile, className), style: style, children: [jsx("img", { src: meta.image, alt: agentName || '', className: styles$3.Avatar }), jsxs("div", { className: styles$3.AgentInfo, children: [jsx("h2", { className: styles$3.AgentName, children: meta.fullname || agentName || 'Agent' }), jsx("p", { className: styles$3.AgentDescription, children: personaDescription }), agentSource !== undefined && (jsx("button", { className: styles$3.viewSourceButton, onClick: (event) => {
|
|
10465
10915
|
event.stopPropagation();
|
|
10466
10916
|
setIsBookEditorVisible(true);
|
|
10467
10917
|
}, children: "View Source" }))] })] }), isBookEditorVisible && (jsx(Modal, { onClose: () => {
|
|
@@ -10813,6 +11263,16 @@ const ResetIcon = () => (jsx("svg", { width: "16", height: "16", viewBox: "0 0 2
|
|
|
10813
11263
|
*/
|
|
10814
11264
|
const SendIcon = ({ size }) => (jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", children: jsx("path", { d: "M2.01 21L23 12 2.01 3 2 10l15 2-15 2z" }) }));
|
|
10815
11265
|
|
|
11266
|
+
/**
|
|
11267
|
+
* Renders a settings (gear) icon.
|
|
11268
|
+
*
|
|
11269
|
+
* @param props - SVG properties augmented with an optional `size`
|
|
11270
|
+
* @private internal subcomponent used by various components
|
|
11271
|
+
*/
|
|
11272
|
+
function SettingsIcon({ size = 18, color = 'currentColor', ...props }) {
|
|
11273
|
+
return (jsxs("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "none", stroke: color, strokeWidth: "1.5", strokeLinecap: "round", strokeLinejoin: "round", ...props, children: [jsx("circle", { cx: "12", cy: "12", r: "3" }), jsx("path", { d: "M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 1 1-2.83 2.83l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-4 0v-.09a1.65 1.65 0 0 0-1-1.51 1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 1 1-2.83-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1 0-4h.09a1.65 1.65 0 0 0 1.51-1 1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 1 1 2.83-2.83l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 4 0v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 1 1 2.83 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 0 4h-.09a1.65 1.65 0 0 0-1.51 1z" })] }));
|
|
11274
|
+
}
|
|
11275
|
+
|
|
10816
11276
|
/**
|
|
10817
11277
|
* Shows simple template icon
|
|
10818
11278
|
*
|
|
@@ -11299,7 +11759,10 @@ function parseSearchResultsFromText(text) {
|
|
|
11299
11759
|
current = null;
|
|
11300
11760
|
return;
|
|
11301
11761
|
}
|
|
11302
|
-
const snippet = current.snippetParts
|
|
11762
|
+
const snippet = current.snippetParts
|
|
11763
|
+
.map((part) => part.trim())
|
|
11764
|
+
.filter(Boolean)
|
|
11765
|
+
.join(' ');
|
|
11303
11766
|
results.push({
|
|
11304
11767
|
title: current.title,
|
|
11305
11768
|
url: current.url,
|
|
@@ -11421,6 +11884,21 @@ function extractSearchResults(resultRaw) {
|
|
|
11421
11884
|
}
|
|
11422
11885
|
return { results: [], rawText: null };
|
|
11423
11886
|
}
|
|
11887
|
+
/**
|
|
11888
|
+
* @@@
|
|
11889
|
+
*
|
|
11890
|
+
* @private utility of `<Chat/>` component
|
|
11891
|
+
*/
|
|
11892
|
+
function parseTeamToolResult(resultRaw) {
|
|
11893
|
+
if (!resultRaw || typeof resultRaw !== 'object') {
|
|
11894
|
+
return null;
|
|
11895
|
+
}
|
|
11896
|
+
const teammate = resultRaw.teammate;
|
|
11897
|
+
if (!teammate || typeof teammate !== 'object') {
|
|
11898
|
+
return null;
|
|
11899
|
+
}
|
|
11900
|
+
return resultRaw;
|
|
11901
|
+
}
|
|
11424
11902
|
/**
|
|
11425
11903
|
* @@@
|
|
11426
11904
|
*
|
|
@@ -11460,8 +11938,8 @@ function getToolCallResultDate(result) {
|
|
|
11460
11938
|
* <- TODO: But maybe split into multiple files later?
|
|
11461
11939
|
*/
|
|
11462
11940
|
|
|
11463
|
-
var css_248z$2 = "@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{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;width:auto;z-index:200}.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}@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_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);display:flex;flex-direction:row;margin-bottom:20px;max-width:100%;position:relative}@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{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-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_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 12px 4px;position:relative;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{word-wrap:break-word;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:min(70%,600px);padding:14px 18px;position:relative;text-align:left;transition:all .2s ease}.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 h1{font-size:2em}.Chat-module_messageText__XgNyQ h2{font-size:1.75em}.Chat-module_messageText__XgNyQ h3{font-size:1.5em}.Chat-module_messageText__XgNyQ h4{font-size:1.25em}.Chat-module_messageText__XgNyQ h5{font-size:1.1em}.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_scrollToBottom__nzxdZ{-webkit-tap-highlight-color:transparent;align-items:center;animation:Chat-module_scrollButtonSlideIn__XnImg .3s ease-out;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;pointer-events:all;touch-action:manipulation;transform:translate(-50%,-150%);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_scrollToBottom__nzxdZ:hover{transform:translate(-50%,-160%) scale(1.05)}.Chat-module_scrollToBottom__nzxdZ:active{transform:translate(-50%,-160%) scale(.95);transition:transform .1s ease}.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}.Chat-module_searchModalHeader__Y8V-w{align-items:center;background:#fdfdfd;border-bottom:1px solid #eee;display:flex;gap:16px;padding:24px 32px}.Chat-module_searchModalIcon__AR5KY{font-size:24px}.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_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_toolCallDetails__WUFlD{margin:20px 0;text-align:left}.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_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_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_ratingConfirmation__n16vb{word-wrap:break-word;animation:Chat-module_confirmationSlideIn__5U-wz .3s ease-out;backdrop-filter:blur(20px);background:linear-gradient(135deg,#10b981,#059669);border-radius:12px;box-shadow:0 8px 32px rgba(16,185,129,.3);color:#fff;font-weight:500;max-width:300px;padding:16px 20px;position:fixed;right:20px;top:20px;z-index:1001}@keyframes Chat-module_confirmationSlideIn__5U-wz{0%{opacity:0;transform:translateX(100%)}to{opacity:1;transform:translateX(0)}}@media (max-width:768px){.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;max-width:85%;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:36px;margin:0 10px 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%}}@media (max-width:480px){.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;max-width:90%;padding:10px 14px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:32px;margin:0 8px 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_ratingConfirmation__n16vb,.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}}\n/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Chat.module.css"],"names":[],"mappings":"AAAA,WACI,yBAA4B,CAC5B,2EAA8E,CAC9E,irEAuBJ,CAEA,6CAKI,eAAgB,CAGhB,iBAAkB,CAElB,qCAA0C,CAJ1C,UAAW,CAGX,eAAgB,CANhB,QAAS,CAQT,WAAa,CAJb,iBAAkB,CAKlB,mBAAoB,CAXpB,cAAe,CACf,QAAS,CAET,0BAA2B,CAS3B,YACJ,CAEA,yBAGI,YAAa,CACb,qBAAsB,CAEtB,oDAA0D,CAJ1D,WAAY,CADZ,UAOJ,CAEA,iCAII,YAAa,CACb,4DAIS,CAPT,WAAY,CACZ,eAAgB,CAFhB,UASJ,CAEA,6DAKI,qBAAyB,CACzB,0CAA+C,CAF/C,aAAc,CAId,eAAgB,CAPhB,YAAa,CAEb,iBAAkB,CAIlB,iBAAkB,CALlB,UAOJ,CAEA,oCAII,iBAAkB,CAHlB,YAAa,CAEb,kBAAmB,CAEnB,qBAAsB,CACtB,eAAgB,CAJhB,UAKJ,CAEA,4BASI,kBAAmB,CAJnB,qBAAsB,CAGtB,YAAa,CAEb,OAAQ,CATR,YAAa,CAEb,kBAAmB,CAGnB,qBAAsB,CACtB,kBAAmB,CALnB,UAAW,CAEX,WAOJ,CAEA,sDACI,QACJ,CAEA,oDACI,uBACJ,CAEA,qDACI,qBACJ,CAGA,yBACI,+BACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,mCAEI,qBAAuB,CADvB,oBAEJ,CAEA,mCACI,sBACJ,CAEA,sCACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,0CAEI,qBAAuB,CADvB,oBAEJ,CAEA,yEACI,sBACJ,CACJ,CAGA,yBACI,4BAEI,OAAQ,CADR,kBAEJ,CACJ,CAEA,kEACI,YAAa,CAEb,WAAY,CADZ,UAAW,CAEX,WACJ,CAGA,kEACI,YAAa,CAEb,WAAY,CAIZ,iBAAkB,CADlB,eAAgB,CADhB,sBAAuB,CAGvB,sBAAuB,CANvB,UAAW,CAEX,UAKJ,CAGA,qFACI,SACJ,CAEA,2FACI,sBACJ,CAEA,2FACI,4BAAoC,CACpC,iBAAkB,CAClB,uBACJ,CAEA,iGACI,4BACJ,CAGA,iEAGI,oBAAqB,CAGrB,6EAAmE,CALnE,YAAa,CAGb,kBAAmB,CAFnB,kBAAmB,CAKnB,cAAe,CAFf,iBAGJ,CAEA,6CACI,GACI,SAAU,CACV,qCACJ,CACA,GACI,SAAU,CACV,gCACJ,CACJ,CAEA,yCAKI,UAAY,CACZ,iBACJ,CAEA,6CACI,iBACJ,CAGA,+EASI,mEAAiD,CAFjD,4EAAsF,CACtF,iBAAkB,CALlB,UAAW,CAFX,UAAW,CAKX,UAAW,CAJX,iBAAkB,CAElB,UAAW,CACX,UAKJ,CAEA,4EAQI,uCAA8C,CAN9C,YAAa,CACb,kBAAmB,CACnB,cAAe,CACf,OAAQ,CACR,cAAe,CACf,eAEJ,CAEA,0EAGI,kBAAmB,CAInB,8BAAqC,CACrC,mCAA0C,CAC1C,kBAAmB,CACnB,mCAAwC,CACxC,aAAc,CATd,YAAa,CAGb,cAAgB,CADhB,OAAQ,CAER,gBAMJ,CAEA,sCACI,cAAe,CACf,uBACJ,CAEA,4CACI,6BAAoC,CAEpC,oCAAyC,CADzC,0BAEJ,CAEA,oCACI,kBAEJ,CAEA,sCACI,iBACJ,CAEA,iCACI,kBAAmB,CAQnB,wBAAyB,CANzB,iBAAkB,CAClB,cAAe,CAIf,gBAAiB,CAHjB,eAAgB,CAHhB,YAAa,CAKb,oBAGJ,CAEA,2CAMI,gEAA+C,CAF/C,mCAAsB,CACtB,iBAAkB,CADlB,qBAAsB,CAFtB,WAAY,CADZ,UAMJ,CAEA,wCACI,eACJ,CAEA,8CACI,GACI,uBACJ,CACJ,CAEA,2CACI,MAEI,UAAY,CACZ,oBACJ,CACA,IACI,SAAU,CACV,qBACJ,CACJ,CAGA,oCAEI,oBAAqB,CAErB,6EAAmE,CAHnE,YAAa,CAEb,kBAEJ,CAEA,+DAII,aAAc,CAFd,WAAY,CACZ,iBAAkB,CAFlB,UAIJ,CAaA,iCAMI,0BAA2B,CAF3B,kCAA0C,CAF1C,kBAAmB,CACnB,6BAA8B,CAE9B,mCAAwC,CAKxC,eAAgB,CAThB,iBAUJ,CAEA,gEALI,kBAAmB,CADnB,YAAa,CAEb,OAQJ,CAEA,8BAKI,mEAAiD,CADjD,4DAA8E,CAD9E,iBAAkB,CAGlB,mCAAwC,CAJxC,UAAW,CADX,SAMJ,CAEA,0CACI,qBACJ,CAEA,2CACI,qBACJ,CAEA,2CACI,kBACJ,CAEA,2CACI,UAII,UAAY,CADZ,iCAEJ,CACA,IAEI,SAAU,CADV,qCAEJ,CACJ,CAEA,yFACI,oBAAqB,CACrB,0BAA2B,CAC3B,0BACJ,CAEA,yHACI,8BACJ,CAEA,+BAII,qCAAuC,CAHvC,cAAe,CACf,cAAe,CACf,oBAEJ,CAEA,yDACI,UACJ,CAGA,4FAEI,gBAAmB,CAEnB,aAAc,CADd,iBAAkB,CAElB,iBAAkB,CAJlB,UAKJ,CAqBA,iGAWI,oBAAqB,CAGrB,0BAA2B,CAb3B,wCAAyC,CAIzC,kBAAmB,CAInB,mCAAwC,CAPxC,+BAAgC,CAWhC,cAAe,CAHf,eAAgB,CAFhB,iBAAkB,CAFlB,wBAA0B,CAF1B,iBAAkB,CADlB,iBAAkB,CAIlB,eAAgB,CAKhB,uBAGJ,CAGA,wCAKI,kBAAmB,CAJnB,WAAY,CAKZ,wBAAyB,CACzB,mBAAoB,CAJpB,UAAW,CADX,OAAQ,CAOR,iBAAkB,CALlB,SAMJ,CACA,oMAGI,mBAAoB,CADpB,kBAEJ,CAEA,+BASI,kBAAmB,CARnB,6BAAoC,CACpC,qBAAsB,CACtB,iBAAkB,CAGlB,oCAAyC,CADzC,cAAe,CAGf,YAAa,CAEb,UAAY,CANZ,eAAgB,CAOhB,iBAAkB,CAJlB,+CAKJ,CAEA,kCAgBI,oDAAmC,CAXnC,eAAgB,CAGhB,iBAAkB,CAElB,qCAA0C,CAJ1C,UAAW,CAGX,eAAiB,CAPjB,QAAS,CAaT,cAAe,CAEf,eAAgB,CANhB,WAAa,CAOb,wBAAyB,CAXzB,gBAAiB,CAKjB,mBAAoB,CAXpB,iBAAkB,CAElB,QAAS,CACT,0BAA2B,CAU3B,kBAAmB,CAKnB,qBAAsB,CANtB,WAOJ,CAEA,sCACI,gBAAkB,CAClB,wBACJ,CAEA,uCACI,mBAAqB,CACrB,iBAAmB,CACnB,wBACJ,CAIA,kDACI,GACI,SAAU,CACV,qDACJ,CACA,GACI,WAAa,CACb,iDACJ,CACJ,CAEA,0EAEI,uBAAwB,CAExB,uCAA4C,CAD5C,SAEJ,CAEA,mCACI,aACJ,CAEA,mCACI,aACJ,CAEA,mCACI,gBACJ,CAEA,mCACI,eACJ,CAEA,mCACI,gBACJ,CAEA,mCACI,eACJ,CAEA,mCACI,eAAgB,CAChB,gBACJ,CAEA,mCACI,kBAAmB,CACnB,gBACJ,CAEA,yJAMI,iBAAkB,CADlB,kBAAmB,CADnB,eAGJ,CAEA,oCAII,eAAqB,CAGrB,UAEJ,CAEA,+EATI,WAAY,CACZ,eAAgB,CAFhB,aAAc,CAId,iBAAkB,CAClB,mBAAoB,CAEpB,WAYJ,CATA,2CAII,oBAAqB,CAGrB,UAEJ,CAEA,qCAMI,oBAAqB,CAFrB,WAAY,CACZ,eAAgB,CAIhB,aAAc,CARd,oBAAqB,CAMrB,iBAAkB,CAClB,mBAAoB,CANpB,QAAS,CACT,SAOJ,CAEA,yCACI,0BAA2B,CAC3B,iBACJ,CAEA,oEACI,kBAAmB,CAKnB,oBAAqB,CACrB,oCAAyC,CALzC,aAAc,CAMd,8DAA2E,CAL3E,cAAe,CACf,eAAgB,CAChB,eAIJ,CACA,yEACI,yBAA2B,CAK3B,qBAAuB,CACvB,yBAA2B,CAL3B,uBAAyB,CASzB,aAAc,CARd,6BAA+B,CAC/B,2BAA6B,CAM7B,eAAgB,CALhB,mBAAqB,CAGrB,eAAgB,CAChB,qBAGJ,CACA,sCAKI,kBAAmB,CAHnB,wBAAyB,CACzB,gBAAiB,CAKjB,oCAAyC,CAEzC,aAAc,CADd,cAAe,CALf,aAAc,CAHd,UAUJ,CACA,sEAMI,eAAgB,CAHhB,+BAAgC,CAEhC,aAAc,CAHd,iBAAkB,CAElB,eAGJ,CACA,mCACI,qDAA6D,CAG7D,+BAAgC,CADhC,aAAc,CADd,eAGJ,CACA,iDACI,kBACJ,CACA,oDACI,kBACJ,CACA,4CACI,kBAAmB,CACnB,yBACJ,CACA,sCACI,kBAAmB,CACnB,eACJ,CACA,8FAEI,2BACJ,CACA,4FAEI,4BACJ,CAEA,uGACI,qCAA0C,CAC1C,0BACJ,CAGA,gCAMI,sCAA8C,CAL9C,YAAa,CACb,cAAe,CACf,OAAQ,CACR,eAAgB,CAChB,gBAEJ,CAEA,+BAEI,kBAAmB,CAGnB,6BAAoC,CACpC,kCAA0C,CAC1C,kBAAmB,CAEnB,aAAc,CARd,YAAa,CAOb,cAAe,CALf,OAAQ,CASR,eAAgB,CARhB,gBAAiB,CAMjB,oBAAqB,CACrB,uBAEJ,CAEA,qCACI,6BAAoC,CACpC,8BAAsC,CACtC,0BACJ,CAEA,mCACI,cAAe,CACf,UACJ,CAEA,mCAEI,eAAgB,CAChB,sBAAuB,CAFvB,kBAGJ,CAGA,mCAMI,uCAA8C,CAL9C,YAAa,CACb,cAAe,CACf,OAAQ,CACR,eAAgB,CAChB,gBAEJ,CAGA,kCAYI,uCAAwC,CAVxC,kBAAmB,CASnB,0BAA2B,CAP3B,4BAAoC,CACpC,kCAA0C,CAC1C,kBAAmB,CAGnB,cAAe,CARf,mBAAoB,CAMpB,cAAe,CACf,eAAgB,CALhB,gBAAiB,CAUjB,yBAA0B,CAH1B,uBAAyB,CAIzB,gBACJ,CAEA,wCACI,6BAAkC,CAClC,+BAAoC,CAEpC,wCAA6C,CAD7C,0BAEJ,CAEA,yCACI,oBAAsB,CACtB,6BACJ,CAGA,oCAGI,mBAAoB,CAFpB,QAAS,CACT,SAEJ,CAEA,yCACI,eACJ,CAEA,qCACI,iBACJ,CAGA,4FAMI,kBAAmB,CAMnB,0BAA2B,CAH3B,yBAA8B,CAC9B,kBAAmB,CARnB,WAAY,CAEZ,YAAa,CACb,OAAQ,CAER,cAAe,CAMf,SAAU,CAFV,eAAgB,CAVhB,iBAAkB,CAElB,SAAU,CAWV,yBAA0B,CAC1B,gDAAyD,CAPzD,SAQJ,CAEA,kGACI,SAAU,CACV,uBACJ,CAEA,kGACI,yBAA8B,CAE9B,oCAAyC,CADzC,eAEJ,CAEA,iGAGI,cAAe,CADf,oBAAqB,CAErB,cAAe,CAHf,4CAIJ,CAEA,uGACI,oBACJ,CAGA,+DAMI,YAAa,CACb,qBAAsB,CACtB,QAAS,CANT,YAAa,CAEb,YAAa,CAKb,iBAAkB,CANlB,UAAW,CAFX,UASJ,CAEA,0HACI,0BAA2B,CAC3B,uEACJ,CAGA,2FACI,yEACJ,CAGA,yCACI,YAAa,CACb,cAAe,CACf,OAAQ,CACR,iBACJ,CAGA,gCAEI,kBAAmB,CAOnB,0BAA2B,CAJ3B,4BAAoC,CACpC,kCAA0C,CAC1C,iBAAkB,CANlB,YAAa,CAOb,cAAe,CALf,OAAQ,CACR,gBAAiB,CAMjB,uBACJ,CAEA,sCACI,6BAAqC,CACrC,8BACJ,CAEA,6BACI,cAAe,CACf,UACJ,CAEA,6BACI,YAAa,CACb,qBAAsB,CACtB,OAAQ,CACR,WACJ,CAEA,6BAEI,UAAY,CADZ,eAAgB,CAKhB,eAAgB,CAFhB,eAAgB,CAChB,sBAAuB,CAFvB,kBAIJ,CAEA,6BACI,aAAc,CACd,cACJ,CAEA,qCAEI,kBAAmB,CAKnB,2BAAgC,CADhC,WAAY,CAGZ,iBAAkB,CADlB,UAAc,CAEd,cAAe,CATf,YAAa,CAWb,aAAc,CAPd,WAAY,CAFZ,sBAAuB,CAQvB,uBAAyB,CAPzB,UASJ,CAEA,2CACI,2BAAgC,CAChC,oBACJ,CAGA,mCAEI,oBAAqB,CAErB,eAAmB,CACnB,gCAAqC,CACrC,kBAAmB,CAEnB,qCAA0C,CAP1C,YAAa,CAEb,OAAQ,CAIR,gBAAiB,CAEjB,gDACJ,CAEA,gDACI,+BAAgC,CAChC,yCAA8C,CAC9C,0BACJ,CAEA,yDACI,aAAc,CACd,SACJ,CAGA,qCAQI,kBAAmB,CAJnB,sBAAuB,CADvB,WAAY,CAGZ,iBAAkB,CADlB,aAAc,CAKd,cAAe,CAHf,YAAa,CAKb,aAAc,CAVd,WAAY,CAOZ,sBAAuB,CAIvB,iBAAkB,CAFlB,uBAAyB,CAVzB,UAaJ,CAEA,0DACI,kBAAmB,CACnB,aACJ,CAEA,8CAEI,kBAAmB,CADnB,UAEJ,CAGA,gCAQI,kBAAmB,CAJnB,sBAAuB,CADvB,WAAY,CAGZ,iBAAkB,CADlB,aAAc,CAKd,cAAe,CAHf,YAAa,CAKb,aAAc,CAVd,WAAY,CAOZ,sBAAuB,CAIvB,iBAAkB,CAFlB,uBAAyB,CAVzB,UAaJ,CAEA,qDACI,kBAAmB,CACnB,aACJ,CAEA,sCAGI,8DAA4C,CAF5C,uCAA6C,CAC7C,uBAEJ,CAEA,kDACI,GACI,mCACJ,CACA,IACI,qCACJ,CACA,GACI,kCACJ,CACJ,CAGA,mCAEI,kBAAmB,CAGnB,6BAAkC,CAClC,mCAAwC,CACxC,iBAAkB,CAElB,aAAc,CARd,YAAa,CAOb,cAAe,CALf,QAAS,CACT,gBAMJ,CAEA,sCAGI,6BAAkC,CAClC,iBAAkB,CAHlB,MAAO,CACP,UAAW,CAGX,eACJ,CAEA,uCAII,qEAAmD,CAFnD,4DAAmE,CACnE,iBAAkB,CAFlB,WAIJ,CAEA,6CACI,GACI,2BACJ,CACA,IACI,uBACJ,CACA,GACI,0BACJ,CACJ,CAGA,gCAUI,kBAAmB,CAEnB,0BAA2B,CAN3B,6BAAkC,CAClC,oCAAyC,CACzC,kBAAmB,CAHnB,QAAS,CAIT,YAAa,CAEb,sBAAuB,CARvB,MAAO,CAWP,mBAAoB,CAbpB,iBAAkB,CAGlB,OAAQ,CAFR,KAAM,CAWN,UAEJ,CAEA,uCAGI,kBAAmB,CAEnB,aAAc,CAJd,YAAa,CACb,qBAAsB,CAItB,eAAgB,CAFhB,QAAS,CAGT,iBACJ,CAEA,2CACI,UACJ,CAGA,wEAeI,uCAAwC,CAFxC,eAAgB,CAChB,uBAAwB,CARxB,sBAAuB,CAFvB,WAAY,CAGZ,aAAc,CANd,MAAO,CASP,cAAe,CACf,eAAgB,CARhB,QAAS,CAMT,gBAAiB,CADjB,eAAgB,CAHhB,YAAa,CAHb,iBAAkB,CAUlB,WAAY,CAIZ,yBACJ,CAMA,iFAEI,kBAAmB,CADnB,UAAY,CAEZ,cACJ,CAEA,qFACI,aAAc,CACd,UACJ,CAGA,uGAkBI,uCAAwC,CATxC,kBAAmB,CAGnB,gBAAmB,CANnB,WAAY,CAKZ,2BAA6B,CAM7B,yCAA8C,CAV9C,UAAc,CACd,YAAa,CANb,WAAY,CAQZ,sBAAuB,CAPvB,wBAA4B,CAW5B,0BAA4B,CAD5B,yBAA2B,CAQ3B,gBAAiB,CAjBjB,mBAAqB,CAerB,yBAA0B,CAH1B,gDAAyD,CAIzD,gBAAiB,CAnBjB,UAqBJ,CAEA,6GAEI,yCAA8C,CAD9C,qBAEJ,CAEA,8GACI,oBACJ,CAEA,2GAEI,WAAY,CADZ,UAEJ,CAEA,0EAEI,WAAY,CACZ,kBAAmB,CAFnB,SAGJ,CAIA,4CAWI,sBAAuB,CAFvB,YAAa,CAHb,YAAa,CAEb,WAAY,CAEZ,sBAAuB,CAGvB,mBAAoB,CANpB,UAAW,CAFX,UASJ,CAEA,+EAiBI,uCAAwC,CATxC,kBAAmB,CAcnB,6DAA4C,CAV5C,yBAA0B,CAD1B,yBAA6B,CAF7B,WAAY,CAIZ,iBAAkB,CAQlB,oCAAyC,CANzC,UAAY,CACZ,cAAe,CAVf,YAAa,CAQb,eAAiB,CATjB,WAAY,CAEZ,sBAAuB,CAGvB,YAAa,CATb,kBAAmB,CAiBnB,yBAA0B,CAf1B,+BAAiC,CAiBjC,gDAAyD,CADzD,gBAAiB,CAfjB,UAmBJ,CAEA,kDACI,GACI,SAAU,CACV,wCACJ,CACA,GACI,UAAY,CACZ,kCACJ,CACJ,CAEA,yCACI,2CACJ,CAEA,0CACI,0CAA6C,CAC7C,6BACJ,CAGA,gCAYI,kBAAmB,CAEnB,qDAAoC,CANpC,yBAA0B,CAD1B,+BAAoC,CAFpC,QAAS,CAKT,YAAa,CACb,sBAAuB,CARvB,MAAO,CAFP,cAAe,CAGf,OAAQ,CAFR,KAAM,CAWN,YAEJ,CAEA,0CACI,GAEI,uBAA0B,CAD1B,SAEJ,CACA,GAEI,yBAA0B,CAD1B,SAEJ,CACJ,CAEA,uCAUI,2EAAiE,CADjE,0BAA2B,CAR3B,eAAmB,CAOnB,kCAA0C,CAJ1C,kBAAmB,CAGnB,qCAA0C,CAL1C,aAAc,CAUd,oDAA0D,CAN1D,eAAgB,CAHhB,YAAa,CAEb,SASJ,CAEA,2CACI,GACI,SAAU,CACV,qCACJ,CACA,GACI,SAAU,CACV,gCACJ,CACJ,CAEA,0CAGI,UAAY,CACZ,cAAe,CACf,eAAgB,CAJhB,eAAkB,CAClB,iBAIJ,CAEA,0BACI,YAAa,CAEb,OAAQ,CADR,sBAAuB,CAEvB,kBACJ,CAEA,+BAKI,iBAAkB,CAHlB,cAAe,CADf,cAAe,CAGf,WAAY,CADZ,uBAGJ,CAEA,qCAEI,6BAAkC,CADlC,oBAEJ,CAEA,oCACI,cAAe,CACf,cAAe,CACf,oBACJ,CAEA,kCACI,eAAgB,CAEhB,eAAgB,CADhB,SAEJ,CAEA,sCAII,kBAAmB,CAEnB,kBAAmB,CAJnB,4BAA6B,CAC7B,YAAa,CAEb,QAAS,CAJT,iBAMJ,CAEA,oCACI,cACJ,CAEA,qCAII,uBAAyB,CAFzB,wBAA0B,CAC1B,yBAA2B,CAF3B,kBAAoB,CAIpB,yBACJ,CAEA,uCAEI,eAAgB,CAChB,eAAgB,CAChB,eAAgB,CAHhB,iBAIJ,CAEA,sCACI,YAAa,CACb,qBAAsB,CACtB,QAAS,CACT,eACJ,CAEA,qCAEI,eAAgB,CADhB,eAEJ,CAEA,qCACI,eACJ,CAEA,oCAEI,aAAc,CADd,cAAe,CAEf,iBAAkB,CAElB,eAAgB,CAChB,sBAAuB,CAFvB,kBAGJ,CAEA,sCACI,aAAc,CACd,oBACJ,CAEA,sCACI,wBAA0B,CAC1B,yBAA2B,CAE3B,eAAgB,CADhB,wBAEJ,CAEA,wCACI,aAAc,CACd,oBACJ,CAEA,8CACI,yBACJ,CAEA,wCAGI,aAAc,CAFd,cAAe,CACf,gBAAiB,CAEjB,QACJ,CAEA,8BAGI,aAAc,CACd,iBAAkB,CAHlB,YAAa,CACb,iBAGJ,CAEA,oCAEI,aAAc,CADd,eAEJ,CAEA,0CACI,kBAAmB,CAGnB,qBAAsB,CADtB,iBAAkB,CAElB,kBAAmB,CACnB,gBAAiB,CACjB,eAAgB,CALhB,YAMJ,CAEA,iCAEI,eAAiB,CADjB,QAAS,CAET,oBAAqB,CACrB,qBACJ,CAEA,qCACI,eAAgB,CAEhB,QAAS,CADT,SAEJ,CAEA,wCAEI,cAAgB,CADhB,iBAEJ,CAEA,mEACI,kBAAmB,CACnB,aACJ,CAEA,gCAQI,6BAAqC,CAJrC,kCAA0C,CAC1C,kBAAmB,CAInB,aAAc,CACd,cAAe,CACf,eAAgB,CALhB,kBAAmB,CAJnB,gBAAiB,CACjB,YAAa,CAIb,eAAgB,CAKhB,uBAAyB,CAXzB,UAYJ,CAEA,sCAEI,6BAAqC,CADrC,oBAAqB,CAErB,uCAA4C,CAC5C,YACJ,CAEA,0CAEI,6BAAqC,CADrC,kCAEJ,CAEA,kCACI,YAAa,CAEb,QAAS,CADT,wBAEJ,CAEA,yCAEI,WAAY,CACZ,iBAAkB,CAClB,cAAe,CACf,cAAe,CACf,eAAgB,CAEhB,cAAe,CAPf,iBAAkB,CAMlB,uBAEJ,CAEA,qDACI,kCAA0C,CAE1C,kCAA0C,CAD1C,aAEJ,CAEA,0EACI,kCAA0C,CAC1C,aACJ,CAEA,oDACI,+CAA6D,CAC7D,UACJ,CAEA,yEACI,kDAA6D,CAE7D,wCAA6C,CAD7C,0BAEJ,CAEA,6DAEI,kBAAmB,CADnB,UAEJ,CAGA,+BAkBI,iDAAmD,CAhBnD,4BAA8B,CAe9B,oCAAsC,CAVtC,yDAAwE,CAExE,qBAAuB,CACvB,4BAA8B,CAM9B,kDAAwD,CARxD,oBAAyB,CAMzB,wBAA0B,CAb1B,6BAA+B,CAU/B,wBAA0B,CAC1B,yBAA2B,CAR3B,iBAAmB,CADnB,gCAAkC,CAUlC,yBAA2B,CAP3B,kBAAoB,CAepB,yBAA2B,CAC3B,yBAA2B,CAE3B,yBAA2B,CAnB3B,2BAA6B,CAkB7B,2BAA6B,CAJ7B,mCAAqC,CAJrC,0DAAoE,CAKpE,0BAKJ,CAEA,sCAOI,4EAAsF,CANtF,UAAW,CAKX,WAAY,CAFZ,UAAW,CAFX,iBAAkB,CAClB,KAAM,CAKN,wBAA0B,CAH1B,UAIJ,CAEA,4CACI,SACJ,CAEA,oDACI,4CAA6D,CAE7D,wCAA6C,CAD7C,sCAEJ,CAEA,sCAGI,wCAA6C,CAF7C,qCAAuC,CACvC,6BAEJ,CAEA,qCAEI,qEAA+E,CAD/E,YAEJ,CAEA,mCAMI,4CAAiD,CAHjD,aAAc,CADd,WAAY,CAEZ,SAAU,CACV,gDAAyD,CAJzD,UAMJ,CAEA,yCAEI,4CAAiD,CADjD,mCAEJ,CAEA,mCAEI,eAAgB,CAGhB,SAAU,CADV,oCAAyC,CADzC,uBAAyB,CAFzB,kBAKJ,CAEA,wEACI,yBACJ,CAGA,sCAkBI,iDAAmD,CAhBnD,4BAA8B,CAe9B,oCAAsC,CAVtC,yDAAwE,CAExE,qBAAuB,CACvB,4BAA8B,CAM9B,kDAAwD,CARxD,oBAAyB,CAMzB,wBAA0B,CAb1B,6BAA+B,CAU/B,wBAA0B,CAC1B,yBAA2B,CAR3B,iBAAmB,CADnB,gCAAkC,CAUlC,yBAA2B,CAP3B,kBAAoB,CAepB,yBAA2B,CAC3B,yBAA2B,CAE3B,yBAA2B,CAnB3B,2BAA6B,CAkB7B,2BAA6B,CAJ7B,mCAAqC,CAJrC,0DAAoE,CAKpE,0BAKJ,CAEA,6CAOI,4EAAsF,CANtF,UAAW,CAKX,WAAY,CAFZ,UAAW,CAFX,iBAAkB,CAClB,KAAM,CAKN,wBAA0B,CAH1B,UAIJ,CAEA,mDACI,SACJ,CAEA,2DACI,4CAA6D,CAE7D,wCAA6C,CAD7C,sCAEJ,CAEA,6CAGI,wCAA6C,CAF7C,qCAAuC,CACvC,6BAEJ,CAEA,4CAEI,qEAA+E,CAD/E,YAEJ,CAEA,0CAMI,4CAAiD,CAHjD,aAAc,CADd,WAAY,CAEZ,SAAU,CACV,gDAAyD,CAJzD,UAMJ,CAEA,gDAEI,4CAAiD,CADjD,oBAEJ,CAEA,+EACI,yBACJ,CAEA,wCACI,oBAAqB,CACrB,iBACJ,CAEA,6BAII,eAAgB,CAChB,qBAAsB,CAGtB,oCAAyC,CALzC,MAAO,CAIP,eAAgB,CANhB,iBAAkB,CAClB,QAAS,CAIT,UAGJ,CAEA,iCAKI,eAAgB,CADhB,WAAY,CAIZ,UAAW,CADX,cAAe,CANf,aAAc,CAEd,gBAAiB,CAGjB,eAAgB,CAJhB,UAOJ,CAEA,uCACI,wBACJ,CAGA,gCACI,4DACJ,CAEA,qDACI,4DACJ,CAEA,2DAEI,qBAAuB,CADvB,oBAEJ,CAEA,0DACI,4DACJ,CAEA,+EACI,4DAAwE,CAExE,0CAA+C,CAD/C,sCAEJ,CAEA,oCACI,6BACJ,CAEA,8DACI,oBACJ,CAEA,+DACI,UACJ,CAGA,0CAQI,kBAAmB,CACnB,0BAA2B,CAL3B,yEAA4F,CAC5F,0CAA+C,CAC/C,YAAa,CALb,YAAa,CAMb,sBAAuB,CAJvB,iBAAkB,CADlB,UAQJ,CAGA,uCAEI,kBAAmB,CASnB,0BAA2B,CAN3B,yEAA4F,CAC5F,mCAAwC,CACxC,kBAAmB,CAKnB,uCAA4C,CAJ5C,aAAc,CAPd,mBAAoB,CAQpB,cAAe,CACf,eAAgB,CAPhB,OAAQ,CACR,gBAAiB,CASjB,iBACJ,CAEA,2CAII,uEAAqD,CADrD,aAAc,CADd,WAAY,CADZ,UAIJ,CAEA,4CACI,eAAgB,CAChB,oCACJ,CAGA,mCASI,qEAAmD,CAFnD,kBAAmB,CACnB,iBAAkB,CAFlB,UAAW,CALX,iBAAkB,CAClB,SAAU,CACV,OAAQ,CACR,0BAA2B,CAC3B,SAKJ,CAEA,iDACI,MAGI,SAAU,CADV,kBAEJ,CACA,IAEI,UAAY,CADZ,oBAEJ,CACJ,CAEA,6CACI,MAGI,SAAU,CADV,mCAEJ,CACA,IAEI,UAAY,CADZ,qCAEJ,CACJ,CAGA,uEAII,kBAAmB,CADnB,cAAe,CAFf,iBAAkB,CAClB,gBAGJ,CAEA,2EAEI,WAAY,CADZ,UAEJ,CAGA,uCAYI,oBAAqB,CAFrB,6DAA4C,CAI5C,0BAA2B,CAV3B,kDAA6D,CAG7D,kBAAmB,CACnB,yCAA8C,CAH9C,UAAY,CAQZ,eAAgB,CAFhB,eAAgB,CALhB,iBAAkB,CALlB,cAAe,CAEf,UAAW,CADX,QAAS,CAOT,YAMJ,CAEA,kDACI,GAEI,SAAU,CADV,0BAEJ,CACA,GAEI,SAAU,CADV,uBAEJ,CACJ,CAGA,yBACI,4BAEI,OAAQ,CADR,kBAEJ,CAEA,kEACI,iBACJ,CAEA,iEACI,kBACJ,CAEA,iGAII,kBAAmB,CADnB,cAAe,CAFf,aAAc,CACd,iBAGJ,CAEA,4FAEI,WAAY,CACZ,iBAAkB,CAFlB,UAGJ,CASA,+DAEI,QAAS,CADT,iBAEJ,CAEA,wEAGI,kBAAmB,CAFnB,cAAe,CACf,iBAEJ,CAEA,sEAEI,WAAY,CADZ,UAEJ,CAEA,mCAGI,cAAe,CADf,WAAY,CAEZ,sBAAuB,CAHvB,UAIJ,CAEA,+BACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,mCAEI,qBAAuB,CADvB,oBAEJ,CAEA,mCACI,sBACJ,CAEA,sCACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,0CAEI,qBAAuB,CADvB,oBAEJ,CAEA,yEACI,sBACJ,CAEA,uCAGI,kBAAmB,CAFnB,WAAY,CAGZ,eAAgB,CAChB,eAAgB,CAHhB,iBAIJ,CAEA,0BACI,OAAQ,CACR,kBACJ,CAEA,+BACI,cAAe,CACf,WACJ,CAEA,kCACI,6BAA8B,CAC9B,OACJ,CAEA,yCAII,kBAAmB,CADnB,cAAe,CADf,YAAa,CADb,UAIJ,CACJ,CAEA,yBACI,4BAEI,OAAQ,CADR,iBAEJ,CAEA,kEACI,gBACJ,CAEA,iGAII,kBAAmB,CADnB,cAAe,CAFf,aAAc,CACd,iBAGJ,CAEA,4FAEI,WAAY,CACZ,gBAAiB,CAFjB,UAGJ,CASA,+DAEI,OAAQ,CADR,gBAEJ,CAEA,wEAEI,kBAAmB,CACnB,cAAe,CAFf,iBAGJ,CAEA,sEAEI,WAAY,CADZ,UAEJ,CAEA,+BACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,mCAEI,qBAAuB,CADvB,oBAEJ,CAEA,mCACI,sBACJ,CAEA,sCACI,2BAA6B,CAM7B,eAAiB,CAJjB,qBAAuB,CAKvB,kBAAoB,CAHpB,yBAA2B,CAD3B,wBAA0B,CAE1B,mBAAqB,CAJrB,oBAOJ,CAEA,0CAEI,qBAAuB,CADvB,oBAEJ,CAEA,yEACI,sBACJ,CAEA,mCAGI,cAAe,CADf,WAAY,CAEZ,sBAAuB,CAHvB,UAIJ,CAEA,gCAEI,oBAAqB,CADrB,SAEJ,CAEA,uCAGI,2BAA4B,CAF5B,QAAS,CAGT,eAAgB,CAChB,sBAAuB,CAHvB,UAIJ,CACJ,CAGA,uCACI,kNAKI,cACJ,CAEA,8QAKI,eACJ,CACJ,CAGA,+BACI,iGACI,gBACJ,CAEA,wEACI,gBACJ,CAEA,sEACI,gBACJ,CACJ","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    /* <- 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: 200;\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}\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/* 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/* 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    position: relative;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    max-width: 100%;\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.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}\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.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 12px 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: min(70%, 600px);\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}\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 h1 {\n    font-size: 2em;\n}\n\n.messageText h2 {\n    font-size: 1.75em;\n}\n\n.messageText h3 {\n    font-size: 1.5em;\n}\n\n.messageText h4 {\n    font-size: 1.25em;\n}\n\n.messageText h5 {\n    font-size: 1.1em;\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 .scrollToBottom {\n    pointer-events: all;\n\n    transform: translate(-50%, -150%);\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    animation: scrollButtonSlideIn 0.3s ease-out;\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.scrollToBottom:hover {\n    transform: translate(-50%, -160%) scale(1.05);\n}\n\n.scrollToBottom:active {\n    transform: translate(-50%, -160%) scale(0.95);\n    transition: transform 0.1s ease;\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}\n\n.searchModalHeader {\n    padding: 24px 32px;\n    border-bottom: 1px solid #eee;\n    display: flex;\n    align-items: center;\n    gap: 16px;\n    background: #fdfdfd;\n}\n\n.searchModalIcon {\n    font-size: 24px;\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.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\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/* 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/* 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/* Rating confirmation */\n.ratingConfirmation {\n    position: fixed;\n    top: 20px;\n    right: 20px;\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%);\n    color: white;\n    padding: 16px 20px;\n    border-radius: 12px;\n    box-shadow: 0 8px 32px rgba(16, 185, 129, 0.3);\n    z-index: 1001;\n    animation: confirmationSlideIn 0.3s ease-out;\n    max-width: 300px;\n    word-wrap: break-word;\n    font-weight: 500;\n    backdrop-filter: blur(20px);\n}\n\n@keyframes confirmationSlideIn {\n    from {\n        transform: translateX(100%);\n        opacity: 0;\n    }\n    to {\n        transform: translateX(0);\n        opacity: 1;\n    }\n}\n\n/* Mobile responsiveness */\n@media (max-width: 768px) {\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        max-width: 85%;\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 10px 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\n@media (max-width: 480px) {\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        max-width: 90%;\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 8px 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    .ratingConfirmation {\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/**\n * TODO: [🌉] DRY Markdown primitives styling\n */\n"]} */";
|
|
11464
|
-
var
|
|
11941
|
+
var css_248z$2 = "@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{--message-min-width:min(240px,100%);display:flex;flex-direction:column;font-family:Arial,Helvetica,sans-serif,OpenMojiBlack;height:100%;overflow:hidden;position:relative;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_emojiEffects__aEAUg{overflow:hidden;z-index:250}.Chat-module_emojiEffect__V7GZ4,.Chat-module_emojiEffects__aEAUg{inset:0;pointer-events:none;position:absolute}.Chat-module_confettiPiece__T6aFe{animation:Chat-module_confettiFall__iSvDL var(--duration) cubic-bezier(.19,1,.22,1) forwards;animation-delay:var(--delay);background:var(--color);border-radius:2px;height:calc(var(--size)*.6);left:var(--x);opacity:.95;position:absolute;top:-10%;transform:translate3d(0,-10%,0) rotate(var(--rotation));width:var(--size);will-change:transform,opacity}.Chat-module_heartParticle__UJzAM{animation:Chat-module_heartFloat__PZoBQ var(--duration) ease-out forwards;animation-delay:var(--delay);bottom:-10%;filter:drop-shadow(0 4px 10px rgba(0,0,0,.2));font-size:var(--size);left:var(--x);opacity:0;position:absolute;will-change:transform,opacity}@keyframes Chat-module_confettiFall__iSvDL{0%{opacity:1;transform:translate3d(0,-10%,0) rotate(var(--rotation))}to{opacity:0;transform:translate3d(var(--drift),110%,0) rotate(calc(var(--rotation) + 2turn))}}@keyframes Chat-module_heartFloat__PZoBQ{0%{opacity:0;transform:translateZ(0) scale(.6)}15%{opacity:1}to{opacity:0;transform:translate3d(var(--drift),-120%,0) scale(1.15)}}.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;width:auto;z-index:200}.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_hoistedMenus__Jub9l{align-items:flex-end;display:flex;flex-direction:column;gap:8px;position:absolute;right:20px;top:16px;z-index:220}.Chat-module_hoistedMenu__i6jtt{min-width:180px;position:static}@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_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);display:flex;flex-direction:row;margin-bottom:20px;max-width:100%;position:relative}@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{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-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_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 12px 4px;position:relative;width:40px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_messageText__XgNyQ{word-wrap:break-word;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:min(70%,600px);min-width:var(--message-min-width);padding:14px 18px;position:relative;text-align:left;transition:all .2s ease}.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 h1{font-size:2em}.Chat-module_messageText__XgNyQ h2{font-size:1.75em}.Chat-module_messageText__XgNyQ h3{font-size:1.5em}.Chat-module_messageText__XgNyQ h4{font-size:1.25em}.Chat-module_messageText__XgNyQ h5{font-size:1.1em}.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_scrollToBottom__nzxdZ{-webkit-tap-highlight-color:transparent;align-items:center;animation:Chat-module_scrollButtonSlideIn__XnImg .3s ease-out;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;pointer-events:all;touch-action:manipulation;transform:translate(-50%,-150%);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_scrollToBottom__nzxdZ:hover{transform:translate(-50%,-160%) scale(1.05)}.Chat-module_scrollToBottom__nzxdZ:active{transform:translate(-50%,-160%) scale(.95);transition:transform .1s ease}.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}.Chat-module_searchModalHeader__Y8V-w{align-items:center;background:#fdfdfd;border-bottom:1px solid #eee;display:flex;gap:16px;padding:24px 32px}.Chat-module_searchModalIcon__AR5KY{font-size:24px}.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_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_toolCallDetails__WUFlD{margin:20px 0;text-align:left}.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_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_settingsMenuItem__bvX1P{align-items:center;display:flex;gap:12px;justify-content:space-between}.Chat-module_settingsToggleState__W5DML{background:#e5e7eb;border-radius:999px;color:#374151;font-size:11px;font-weight:600;letter-spacing:.03em;min-width:38px;padding:2px 8px;text-transform:uppercase}.Chat-module_settingsToggleStateActive__W0qDk{background:#dbeafe;color:#1d4ed8}.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_ratingConfirmation__n16vb{word-wrap:break-word;animation:Chat-module_confirmationSlideIn__5U-wz .3s ease-out;backdrop-filter:blur(20px);background:linear-gradient(135deg,#10b981,#059669);border-radius:12px;box-shadow:0 8px 32px rgba(16,185,129,.3);color:#fff;font-weight:500;max-width:300px;padding:16px 20px;position:fixed;right:20px;top:20px;z-index:1001}@keyframes Chat-module_confirmationSlideIn__5U-wz{0%{opacity:0;transform:translateX(100%)}to{opacity:1;transform:translateX(0)}}@media (max-width:768px){.Chat-module_Chat__j2eE5{--message-min-width:min(200px,90%)}.Chat-module_hoistedMenus__Jub9l{align-items:stretch;left:12px;right:12px;top:12px}.Chat-module_hoistedMenu__i6jtt{min-width:0;width:100%}.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;max-width:85%;padding:12px 16px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:36px;margin:0 10px 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%}}@media (max-width:480px){.Chat-module_Chat__j2eE5{--message-min-width:min(180px,90%)}.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;max-width:90%;padding:10px 14px}.Chat-module_chatMainFlow__--8FE .Chat-module_chatMessage__nmLaZ .Chat-module_avatar__gL6bm{height:32px;margin:0 8px 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_ratingConfirmation__n16vb,.Chat-module_ratingModalContent__CCdq7,.Chat-module_ratingModal__XVKYm,.Chat-module_scrollToBottom__nzxdZ{animation:none}.Chat-module_emojiEffects__aEAUg{display: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}}\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    position: relative;\n    overflow: hidden;\n    --message-min-width: min(240px, 100%);\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiBlack';\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.emojiEffects {\n    position: absolute;\n    inset: 0;\n    pointer-events: none;\n    overflow: hidden;\n    z-index: 250;\n}\n\n.emojiEffect {\n    position: absolute;\n    inset: 0;\n    pointer-events: none;\n}\n\n.confettiPiece {\n    position: absolute;\n    top: -10%;\n    left: var(--x);\n    width: var(--size);\n    height: calc(var(--size) * 0.6);\n    background: var(--color);\n    opacity: 0.95;\n    border-radius: 2px;\n    transform: translate3d(0, -10%, 0) rotate(var(--rotation));\n    animation: confettiFall var(--duration) cubic-bezier(0.19, 1, 0.22, 1) forwards;\n    animation-delay: var(--delay);\n    will-change: transform, opacity;\n}\n\n.heartParticle {\n    position: absolute;\n    bottom: -10%;\n    left: var(--x);\n    font-size: var(--size);\n    opacity: 0;\n    animation: heartFloat var(--duration) ease-out forwards;\n    animation-delay: var(--delay);\n    filter: drop-shadow(0 4px 10px rgba(0, 0, 0, 0.2));\n    will-change: transform, opacity;\n}\n\n@keyframes confettiFall {\n    0% {\n        transform: translate3d(0, -10%, 0) rotate(var(--rotation));\n        opacity: 1;\n    }\n    100% {\n        transform: translate3d(var(--drift), 110%, 0) rotate(calc(var(--rotation) + 720deg));\n        opacity: 0;\n    }\n}\n\n@keyframes heartFloat {\n    0% {\n        transform: translate3d(0, 0, 0) scale(0.6);\n        opacity: 0;\n    }\n    15% {\n        opacity: 1;\n    }\n    100% {\n        transform: translate3d(var(--drift), -120%, 0) scale(1.15);\n        opacity: 0;\n    }\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: 200;\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}\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.hoistedMenus {\n    position: absolute;\n    top: 16px;\n    right: 20px;\n    z-index: 220;\n    display: flex;\n    flex-direction: column;\n    gap: 8px;\n    align-items: flex-end;\n}\n\n.hoistedMenu {\n    position: static;\n    min-width: 180px;\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/* 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    position: relative;\n    animation: messageSlideIn 0.4s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    max-width: 100%;\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.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}\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.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 12px 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: min(70%, 600px);\n    min-width: var(--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}\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 h1 {\n    font-size: 2em;\n}\n\n.messageText h2 {\n    font-size: 1.75em;\n}\n\n.messageText h3 {\n    font-size: 1.5em;\n}\n\n.messageText h4 {\n    font-size: 1.25em;\n}\n\n.messageText h5 {\n    font-size: 1.1em;\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 .scrollToBottom {\n    pointer-events: all;\n\n    transform: translate(-50%, -150%);\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    animation: scrollButtonSlideIn 0.3s ease-out;\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.scrollToBottom:hover {\n    transform: translate(-50%, -160%) scale(1.05);\n}\n\n.scrollToBottom:active {\n    transform: translate(-50%, -160%) scale(0.95);\n    transition: transform 0.1s ease;\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}\n\n.searchModalHeader {\n    padding: 24px 32px;\n    border-bottom: 1px solid #eee;\n    display: flex;\n    align-items: center;\n    gap: 16px;\n    background: #fdfdfd;\n}\n\n.searchModalIcon {\n    font-size: 24px;\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.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\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/* 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.settingsMenuItem {\n    display: flex;\n    align-items: center;\n    justify-content: space-between;\n    gap: 12px;\n}\n\n.settingsToggleState {\n    min-width: 38px;\n    padding: 2px 8px;\n    border-radius: 999px;\n    font-size: 11px;\n    font-weight: 600;\n    letter-spacing: 0.03em;\n    text-transform: uppercase;\n    background: #e5e7eb;\n    color: #374151;\n}\n\n.settingsToggleStateActive {\n    background: #dbeafe;\n    color: #1d4ed8;\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/* Rating confirmation */\n.ratingConfirmation {\n    position: fixed;\n    top: 20px;\n    right: 20px;\n    background: linear-gradient(135deg, #10b981 0%, #059669 100%);\n    color: white;\n    padding: 16px 20px;\n    border-radius: 12px;\n    box-shadow: 0 8px 32px rgba(16, 185, 129, 0.3);\n    z-index: 1001;\n    animation: confirmationSlideIn 0.3s ease-out;\n    max-width: 300px;\n    word-wrap: break-word;\n    font-weight: 500;\n    backdrop-filter: blur(20px);\n}\n\n@keyframes confirmationSlideIn {\n    from {\n        transform: translateX(100%);\n        opacity: 0;\n    }\n    to {\n        transform: translateX(0);\n        opacity: 1;\n    }\n}\n\n/* Mobile responsiveness */\n@media (max-width: 768px) {\n    .Chat {\n        --message-min-width: min(200px, 90%);\n    }\n\n    .hoistedMenus {\n        top: 12px;\n        left: 12px;\n        right: 12px;\n        align-items: stretch;\n    }\n\n    .hoistedMenu {\n        width: 100%;\n        min-width: 0;\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        max-width: 85%;\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 10px 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\n@media (max-width: 480px) {\n    .Chat {\n        --message-min-width: min(180px, 90%);\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        max-width: 90%;\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 8px 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    .ratingConfirmation {\n        animation: none;\n    }\n\n    .emojiEffects {\n        display: 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/**\n * TODO: [🌉] DRY Markdown primitives styling\n */\n"]} */";
|
|
11942
|
+
var styles$2 = {"copiedToClipboardMessage":"Chat-module_copiedToClipboardMessage__apCPY","Chat":"Chat-module_Chat__j2eE5","chatMainFlow":"Chat-module_chatMainFlow__--8FE","emojiEffects":"Chat-module_emojiEffects__aEAUg","emojiEffect":"Chat-module_emojiEffect__V7GZ4","confettiPiece":"Chat-module_confettiPiece__T6aFe","confettiFall":"Chat-module_confettiFall__iSvDL","heartParticle":"Chat-module_heartParticle__UJzAM","heartFloat":"Chat-module_heartFloat__PZoBQ","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","hoistedMenus":"Chat-module_hoistedMenus__Jub9l","hoistedMenu":"Chat-module_hoistedMenu__i6jtt","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","chatMessage":"Chat-module_chatMessage__nmLaZ","messageSlideIn":"Chat-module_messageSlideIn__soTy2","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","ongoingToolCall":"Chat-module_ongoingToolCall__WT3Rc","completedToolCall":"Chat-module_completedToolCall__-q4Cs","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","isMe":"Chat-module_isMe__nBtaV","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","scrollToBottom":"Chat-module_scrollToBottom__nzxdZ","scrollButtonSlideIn":"Chat-module_scrollButtonSlideIn__XnImg","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","searchModalIcon":"Chat-module_searchModalIcon__AR5KY","searchModalQuery":"Chat-module_searchModalQuery__5x-Ra","searchModalContent":"Chat-module_searchModalContent__mWLIE","teamChatContainer":"Chat-module_teamChatContainer__xdLtU","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","toolCallDataContainer":"Chat-module_toolCallDataContainer__rdEzC","toolCallArgsList":"Chat-module_toolCallArgsList__LD3xH","ratingInput":"Chat-module_ratingInput__z8Pv-","ratingActions":"Chat-module_ratingActions__nXcss","saveButtonContainer":"Chat-module_saveButtonContainer__lSNUJ","saveMenu":"Chat-module_saveMenu__-ph8y","saveMenuItem":"Chat-module_saveMenuItem__ISApL","settingsMenuItem":"Chat-module_settingsMenuItem__bvX1P","settingsToggleState":"Chat-module_settingsToggleState__W5DML","settingsToggleStateActive":"Chat-module_settingsToggleStateActive__W0qDk","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","ratingConfirmation":"Chat-module_ratingConfirmation__n16vb","confirmationSlideIn":"Chat-module_confirmationSlideIn__5U-wz"};
|
|
11465
11943
|
styleInject(css_248z$2);
|
|
11466
11944
|
|
|
11467
11945
|
var css_248z$1 = ".AvatarProfileTooltip-module_AvatarProfileTooltip__bZ020{background-color:#fff;border:1px solid #ccc;border-radius:10px;box-shadow:0 4px 8px rgba(0,0,0,.1);max-width:300px;padding:20px;position:fixed;z-index:1000}.AvatarProfileTooltip-module_arrow__wmM61{border-color:transparent transparent #fff;border-style:solid;border-width:0 10px 10px;filter:drop-shadow(0 -1px 1px rgba(0,0,0,.1));height:0;left:20px;position:absolute;top:-10px;width:0}\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkF2YXRhclByb2ZpbGVUb29sdGlwLm1vZHVsZS5jc3MiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEseURBSUkscUJBQXVCLENBQ3ZCLHFCQUFzQixDQUN0QixrQkFBbUIsQ0FDbkIsbUNBQXdDLENBQ3hDLGVBQWdCLENBTGhCLFlBQWEsQ0FGYixjQUFlLENBQ2YsWUFRSixDQUVBLDBDQVFJLHlDQUF1RCxDQUF2RCxrQkFBdUQsQ0FBdkQsd0JBQXVELENBQ3ZELDZDQUFrRCxDQU5sRCxRQUFTLENBR1QsU0FBVSxDQUxWLGlCQUFrQixDQUlsQixTQUFVLENBSFYsT0FRSiIsImZpbGUiOiJBdmF0YXJQcm9maWxlVG9vbHRpcC5tb2R1bGUuY3NzIiwic291cmNlc0NvbnRlbnQiOlsiLkF2YXRhclByb2ZpbGVUb29sdGlwIHtcbiAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgei1pbmRleDogMTAwMDtcbiAgICBwYWRkaW5nOiAyMHB4O1xuICAgIGJhY2tncm91bmQtY29sb3I6IHdoaXRlO1xuICAgIGJvcmRlcjogMXB4IHNvbGlkICNjY2M7XG4gICAgYm9yZGVyLXJhZGl1czogMTBweDtcbiAgICBib3gtc2hhZG93OiAwIDRweCA4cHggcmdiYSgwLCAwLCAwLCAwLjEpO1xuICAgIG1heC13aWR0aDogMzAwcHg7XG4gICAgLyogW/Cfp6BdIFdoYXQgaXMgdGhlIGJlc3QgbWF4LXdpZHRoIGZvciB0aGUgdG9vbHRpcCAqL1xufVxuXG4uYXJyb3cge1xuICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICB3aWR0aDogMDtcbiAgICBoZWlnaHQ6IDA7XG4gICAgYm9yZGVyLXN0eWxlOiBzb2xpZDtcbiAgICB0b3A6IC0xMHB4O1xuICAgIGxlZnQ6IDIwcHg7XG4gICAgYm9yZGVyLXdpZHRoOiAwIDEwcHggMTBweCAxMHB4O1xuICAgIGJvcmRlci1jb2xvcjogdHJhbnNwYXJlbnQgdHJhbnNwYXJlbnQgd2hpdGUgdHJhbnNwYXJlbnQ7XG4gICAgZmlsdGVyOiBkcm9wLXNoYWRvdygwIC0xcHggMXB4IHJnYmEoMCwgMCwgMCwgMC4xKSk7XG59XG4iXX0= */";
|
|
@@ -11509,8 +11987,14 @@ function getToolCallChipletText(toolCall) {
|
|
|
11509
11987
|
const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
|
|
11510
11988
|
const args = parseToolCallArguments(toolCall);
|
|
11511
11989
|
const isTimeTool = toolCall.name === 'get_current_time' || toolCall.name === 'useTime';
|
|
11990
|
+
const resultRaw = parseToolCallResult(toolCall.result);
|
|
11991
|
+
const teamResult = parseTeamToolResult(resultRaw);
|
|
11992
|
+
if (teamResult === null || teamResult === void 0 ? void 0 : teamResult.teammate) {
|
|
11993
|
+
const label = teamResult.teammate.label || teamResult.teammate.url || baseTitle;
|
|
11994
|
+
const teamEmoji = '🤝';
|
|
11995
|
+
return `${teamEmoji} ${label}`.trim();
|
|
11996
|
+
}
|
|
11512
11997
|
if (isTimeTool) {
|
|
11513
|
-
const resultRaw = parseToolCallResult(toolCall.result);
|
|
11514
11998
|
const resultDate = getToolCallResultDate(resultRaw);
|
|
11515
11999
|
if (resultDate) {
|
|
11516
12000
|
return `${emoji} ${resultDate.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' })}`;
|
|
@@ -11674,7 +12158,7 @@ const ChatMessageItem = memo(
|
|
|
11674
12158
|
}
|
|
11675
12159
|
}, [isExpanded]);
|
|
11676
12160
|
const contentWithoutButtonsRef = useRef(null);
|
|
11677
|
-
return (jsxs("div", { className: classNames(
|
|
12161
|
+
return (jsxs("div", { className: classNames(styles$2.chatMessage, isMe && styles$2.isMe, !isComplete && styles$2.isNotCompleteMessage), onClick: () => {
|
|
11678
12162
|
console.group('💬', message.content);
|
|
11679
12163
|
console.info('message', message);
|
|
11680
12164
|
console.info('participant', participant);
|
|
@@ -11682,7 +12166,7 @@ const ChatMessageItem = memo(
|
|
|
11682
12166
|
console.info('participant avatarSrc', avatarSrc);
|
|
11683
12167
|
console.info('participant color', { color, colorOfText });
|
|
11684
12168
|
console.groupEnd();
|
|
11685
|
-
}, children: [avatarSrc && (jsxs("div", { ref: avatarRef, className:
|
|
12169
|
+
}, children: [avatarSrc && (jsxs("div", { ref: avatarRef, className: styles$2.avatar, onMouseEnter: handleMouseEnter, onMouseLeave: handleMouseLeave, onClick: showTooltip, children: [jsx("div", { style: {
|
|
11686
12170
|
width: AVATAR_SIZE,
|
|
11687
12171
|
height: AVATAR_SIZE,
|
|
11688
12172
|
aspectRatio: '1 / 1',
|
|
@@ -11693,10 +12177,10 @@ const ChatMessageItem = memo(
|
|
|
11693
12177
|
borderRadius: '50%',
|
|
11694
12178
|
backgroundPosition: '50% 20%',
|
|
11695
12179
|
'--avatar-bg-color': color.toHex(), // <- TODO: Maybe remove these deprecated CSS variables
|
|
11696
|
-
} }), isAvatarTooltipVisible && (participant === null || participant === void 0 ? void 0 : participant.agentSource) && avatarTooltipPosition && (jsx(AvatarProfileTooltip, { ref: tooltipRef, agentSource: participant.agentSource, position: avatarTooltipPosition }))] })), jsxs("div", { className:
|
|
12180
|
+
} }), isAvatarTooltipVisible && (participant === null || participant === void 0 ? void 0 : participant.agentSource) && avatarTooltipPosition && (jsx(AvatarProfileTooltip, { ref: tooltipRef, agentSource: participant.agentSource, position: avatarTooltipPosition }))] })), jsxs("div", { className: styles$2.messageText, style: {
|
|
11697
12181
|
'--message-bg-color': color.toHex(),
|
|
11698
12182
|
'--message-text-color': colorOfText.toHex(),
|
|
11699
|
-
}, children: [isCopyButtonEnabled && isComplete && (jsx("div", { className:
|
|
12183
|
+
}, children: [isCopyButtonEnabled && isComplete && (jsx("div", { className: styles$2.copyButtonContainer, children: jsxs("button", { className: styles$2.copyButton, title: "Copy message", onClick: async (e) => {
|
|
11700
12184
|
e.stopPropagation();
|
|
11701
12185
|
if (navigator.clipboard && window.ClipboardItem) {
|
|
11702
12186
|
const clipboardItems = {};
|
|
@@ -11736,39 +12220,42 @@ const ChatMessageItem = memo(
|
|
|
11736
12220
|
else {
|
|
11737
12221
|
throw new Error(`Your browser does not support copying to clipboard: navigator.clipboard && window.ClipboardItem.`);
|
|
11738
12222
|
}
|
|
11739
|
-
}, children: [jsxs("svg", { width: "18", height: "18", viewBox: "0 0 24 24", fill: "none", children: [jsx("rect", { x: "7", y: "7", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" }), jsx("rect", { x: "3", y: "3", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" })] }), copied && (jsx("span", { ref: copyTooltipRef, className:
|
|
12223
|
+
}, children: [jsxs("svg", { width: "18", height: "18", viewBox: "0 0 24 24", fill: "none", children: [jsx("rect", { x: "7", y: "7", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" }), jsx("rect", { x: "3", y: "3", width: "10", height: "14", rx: "2", fill: "#fff", stroke: "#bbb", strokeWidth: "1.5" })] }), copied && (jsx("span", { ref: copyTooltipRef, className: styles$2.copiedTooltip +
|
|
11740
12224
|
(tooltipAlign === 'left'
|
|
11741
|
-
? ' ' +
|
|
12225
|
+
? ' ' + styles$2.copiedTooltipLeft
|
|
11742
12226
|
: tooltipAlign === 'right'
|
|
11743
|
-
? ' ' +
|
|
11744
|
-
: ''), children: "Copied!" }))] }) })), message.isVoiceCall && (jsx("div", { className:
|
|
12227
|
+
? ' ' + styles$2.copiedTooltipRight
|
|
12228
|
+
: ''), children: "Copied!" }))] }) })), message.isVoiceCall && (jsx("div", { className: styles$2.voiceCallIndicator, children: jsx("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "currentColor", children: 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 ? (jsx(Fragment, {})) : (jsx("div", { ref: contentWithoutButtonsRef, children: jsx(MarkdownContent, { content: contentWithoutButtons, onCreateAgent: onCreateAgent }) })), message.attachments && message.attachments.length > 0 && (jsx("div", { className: styles$2.attachments, children: message.attachments.map((attachment, index) => (jsxs("a", { href: attachment.url, target: "_blank", rel: "noopener noreferrer", className: styles$2.attachment, title: attachment.name, children: [jsx("span", { className: styles$2.attachmentIcon, children: "\uD83D\uDCCE" }), jsx("span", { className: styles$2.attachmentName, children: attachment.name })] }, index))) })), completedToolCalls && completedToolCalls.length > 0 && (jsx("div", { className: styles$2.completedToolCalls, children: completedToolCalls.map((toolCall, index) => {
|
|
11745
12229
|
const chipletText = getToolCallChipletText(toolCall);
|
|
11746
|
-
return (jsxs("button", { className:
|
|
12230
|
+
return (jsxs("button", { className: styles$2.completedToolCall, onClick: (event) => {
|
|
11747
12231
|
event.stopPropagation();
|
|
11748
12232
|
if (onToolCallClick) {
|
|
11749
12233
|
onToolCallClick(toolCall);
|
|
11750
12234
|
}
|
|
11751
12235
|
}, children: ["[", chipletText, "]"] }, index));
|
|
11752
|
-
}) })), !isComplete && message.ongoingToolCalls && message.ongoingToolCalls.length > 0 && (jsx("div", { className:
|
|
12236
|
+
}) })), !isComplete && message.ongoingToolCalls && message.ongoingToolCalls.length > 0 && (jsx("div", { className: styles$2.ongoingToolCalls, children: message.ongoingToolCalls.map((toolCall, index) => {
|
|
11753
12237
|
const toolInfo = TOOL_TITLES[toolCall.name];
|
|
11754
|
-
const
|
|
11755
|
-
const
|
|
11756
|
-
|
|
12238
|
+
const isTeamTool = toolCall.name.startsWith('team_chat_');
|
|
12239
|
+
const toolTitle = (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[toolCall.name]) ||
|
|
12240
|
+
(toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title) ||
|
|
12241
|
+
(isTeamTool ? 'Consulting teammate' : undefined);
|
|
12242
|
+
const emoji = isTeamTool ? '🤝' : (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
|
|
12243
|
+
return (jsxs("div", { className: styles$2.ongoingToolCall, children: [jsx("div", { className: styles$2.ongoingToolCallSpinner }), jsx("span", { className: styles$2.ongoingToolCallName, children: toolTitle
|
|
11757
12244
|
? `${emoji} ${toolTitle}...`
|
|
11758
12245
|
: `${emoji} Executing ${toolCall.name}...` })] }, index));
|
|
11759
|
-
}) })), shouldShowButtons && (jsx("div", { className:
|
|
12246
|
+
}) })), shouldShowButtons && (jsx("div", { className: styles$2.messageButtons, children: buttons.map((button, buttonIndex) => (jsx("button", { className: styles$2.messageButton, "data-sound": "send", onClick: (event) => {
|
|
11760
12247
|
event.stopPropagation();
|
|
11761
12248
|
if (onMessage) {
|
|
11762
12249
|
onMessage(button.message);
|
|
11763
12250
|
}
|
|
11764
|
-
}, children: jsx(MarkdownContent, { content: button.text }) }, buttonIndex))) })), isFeedbackEnabled && isComplete && (jsx("div", { className:
|
|
12251
|
+
}, children: jsx(MarkdownContent, { content: button.text }) }, buttonIndex))) })), isFeedbackEnabled && isComplete && (jsx("div", { className: styles$2.rating, onMouseEnter: () => {
|
|
11765
12252
|
setExpandedMessageId(message.id || message.content /* <-[💃] */);
|
|
11766
12253
|
}, onMouseLeave: () => {
|
|
11767
12254
|
setExpandedMessageId(null);
|
|
11768
12255
|
setLocalHoveredRating(0);
|
|
11769
|
-
}, children: isExpanded ? ([1, 2, 3, 4, 5].map((star) => (jsx("span", { onClick: () => handleRating(message, star), onMouseEnter: () => setLocalHoveredRating(star), className: classNames(
|
|
12256
|
+
}, children: isExpanded ? ([1, 2, 3, 4, 5].map((star) => (jsx("span", { onClick: () => handleRating(message, star), onMouseEnter: () => setLocalHoveredRating(star), className: classNames(styles$2.ratingStar, star <= (localHoveredRating || currentRating || 0) && styles$2.active), style: {
|
|
11770
12257
|
'--star-inactive-color': mode === 'LIGHT' ? '#ccc' : '#555',
|
|
11771
|
-
}, children: "\u2B50" }, star)))) : (jsx("span", { onClick: () => handleRating(message, currentRating || 1), className: classNames(
|
|
12258
|
+
}, children: "\u2B50" }, star)))) : (jsx("span", { onClick: () => handleRating(message, currentRating || 1), className: classNames(styles$2.ratingStar, currentRating && styles$2.active), style: {
|
|
11772
12259
|
'--star-inactive-color': mode === 'LIGHT' ? '#888' : '#666',
|
|
11773
12260
|
}, children: "\u2B50" })) }))] })] }));
|
|
11774
12261
|
}, (prev, next) => {
|
|
@@ -11823,170 +12310,775 @@ const ChatMessageItem = memo(
|
|
|
11823
12310
|
ChatMessageItem.displayName = 'ChatMessageItem';
|
|
11824
12311
|
|
|
11825
12312
|
/**
|
|
11826
|
-
*
|
|
12313
|
+
* Pause icon (two vertical bars)
|
|
11827
12314
|
*
|
|
11828
|
-
* @
|
|
12315
|
+
* @public exported from `@promptbook/components`
|
|
11829
12316
|
*/
|
|
11830
|
-
|
|
11831
|
-
const hours = date.getHours();
|
|
11832
|
-
const minutes = date.getMinutes();
|
|
11833
|
-
const seconds = date.getSeconds();
|
|
11834
|
-
const hourDeg = (hours % 12) * 30 + minutes * 0.5;
|
|
11835
|
-
const minuteDeg = minutes * 6;
|
|
11836
|
-
const secondDeg = seconds * 6;
|
|
11837
|
-
return (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: [jsx("circle", { cx: "50", cy: "50", r: "45", fill: "white", stroke: "#333", strokeWidth: "2" }), [...Array(12)].map((_, i) => (jsx("line", { x1: "50", y1: "12", x2: "50", y2: "18", transform: `rotate(${i * 30} 50 50)`, stroke: "#333", strokeWidth: "2" }, i))), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "25", transform: `rotate(${hourDeg} 50 50)`, stroke: "#333", strokeWidth: "4", strokeLinecap: "round" }), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "15", transform: `rotate(${minuteDeg} 50 50)`, stroke: "#666", strokeWidth: "3", strokeLinecap: "round" }), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "12", transform: `rotate(${secondDeg} 50 50)`, stroke: "#ff4444", strokeWidth: "1.5", strokeLinecap: "round" }), jsx("circle", { cx: "50", cy: "50", r: "2", fill: "#333" })] }));
|
|
11838
|
-
}
|
|
12317
|
+
const PauseIcon = ({ size = 16 }) => (jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", role: "img", "aria-label": "Pause", children: jsx("path", { d: "M6 4h4v16H6V4zm8 0h4v16h-4V4z" }) }));
|
|
11839
12318
|
|
|
11840
12319
|
/**
|
|
11841
|
-
*
|
|
11842
|
-
*
|
|
11843
|
-
* Note: 🔇 This component does NOT have speak functionality, it just allows to trigger voice recognition
|
|
12320
|
+
* Play icon (triangle) used for resume action
|
|
11844
12321
|
*
|
|
11845
|
-
*
|
|
11846
|
-
|
|
11847
|
-
|
|
12322
|
+
* @public exported from `@promptbook/components`
|
|
12323
|
+
*/
|
|
12324
|
+
const PlayIcon = ({ size = 16 }) => (jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", role: "img", "aria-label": "Play", children: jsx("path", { d: "M8 5v14l11-7L8 5z" }) }));
|
|
12325
|
+
|
|
12326
|
+
/**
|
|
12327
|
+
* Normal flow of messages in the `MockedChat` component
|
|
11848
12328
|
*
|
|
11849
|
-
*
|
|
12329
|
+
* @public exported from `@promptbook/components`
|
|
12330
|
+
*/
|
|
12331
|
+
const NORMAL_FLOW = {
|
|
12332
|
+
beforeFirstMessage: 1000,
|
|
12333
|
+
thinkingBetweenMessages: 2000,
|
|
12334
|
+
waitAfterWord: 100,
|
|
12335
|
+
extraWordDelay: 50,
|
|
12336
|
+
longPauseChance: 0.2,
|
|
12337
|
+
longPauseDuration: [1200, 3500],
|
|
12338
|
+
};
|
|
12339
|
+
/**
|
|
12340
|
+
* Fast flow of messages in the `MockedChat` component
|
|
11850
12341
|
*
|
|
11851
12342
|
* @public exported from `@promptbook/components`
|
|
11852
12343
|
*/
|
|
11853
|
-
|
|
11854
|
-
|
|
11855
|
-
|
|
11856
|
-
|
|
11857
|
-
|
|
11858
|
-
|
|
11859
|
-
|
|
11860
|
-
|
|
11861
|
-
|
|
11862
|
-
|
|
11863
|
-
|
|
11864
|
-
|
|
11865
|
-
|
|
11866
|
-
|
|
11867
|
-
|
|
11868
|
-
|
|
11869
|
-
|
|
11870
|
-
|
|
11871
|
-
|
|
11872
|
-
|
|
11873
|
-
|
|
11874
|
-
|
|
11875
|
-
|
|
11876
|
-
|
|
11877
|
-
|
|
11878
|
-
|
|
11879
|
-
|
|
11880
|
-
|
|
11881
|
-
|
|
11882
|
-
|
|
11883
|
-
|
|
11884
|
-
|
|
11885
|
-
|
|
11886
|
-
|
|
11887
|
-
|
|
11888
|
-
|
|
11889
|
-
|
|
11890
|
-
|
|
11891
|
-
|
|
11892
|
-
|
|
11893
|
-
|
|
11894
|
-
|
|
11895
|
-
|
|
11896
|
-
|
|
11897
|
-
|
|
11898
|
-
|
|
11899
|
-
|
|
11900
|
-
|
|
11901
|
-
|
|
11902
|
-
|
|
12344
|
+
const FAST_FLOW = {
|
|
12345
|
+
beforeFirstMessage: 300,
|
|
12346
|
+
thinkingBetweenMessages: 500,
|
|
12347
|
+
waitAfterWord: 20,
|
|
12348
|
+
extraWordDelay: 10,
|
|
12349
|
+
longPauseChance: 0.1,
|
|
12350
|
+
longPauseDuration: [400, 800],
|
|
12351
|
+
};
|
|
12352
|
+
/**
|
|
12353
|
+
* Slow flow of messages in the `MockedChat` component
|
|
12354
|
+
*
|
|
12355
|
+
* @public exported from `@promptbook/components`
|
|
12356
|
+
*/
|
|
12357
|
+
const SLOW_FLOW = {
|
|
12358
|
+
beforeFirstMessage: 2000,
|
|
12359
|
+
thinkingBetweenMessages: 4000,
|
|
12360
|
+
waitAfterWord: 300,
|
|
12361
|
+
extraWordDelay: 150,
|
|
12362
|
+
longPauseChance: 0.3,
|
|
12363
|
+
longPauseDuration: [2500, 5000],
|
|
12364
|
+
};
|
|
12365
|
+
/**
|
|
12366
|
+
* Blocky flow of messages in the `MockedChat` component
|
|
12367
|
+
*
|
|
12368
|
+
* Messages appear in blocks rather than word-by-word
|
|
12369
|
+
*
|
|
12370
|
+
* @public exported from `@promptbook/components`
|
|
12371
|
+
*/
|
|
12372
|
+
const BLOCKY_FLOW = {
|
|
12373
|
+
beforeFirstMessage: 1000,
|
|
12374
|
+
thinkingBetweenMessages: 2000,
|
|
12375
|
+
waitAfterWord: 0,
|
|
12376
|
+
extraWordDelay: 0,
|
|
12377
|
+
longPauseChance: 0.2,
|
|
12378
|
+
longPauseDuration: [1200, 3500],
|
|
12379
|
+
blocky: true,
|
|
12380
|
+
};
|
|
12381
|
+
/**
|
|
12382
|
+
* Random flow of messages in the `MockedChat` component
|
|
12383
|
+
*
|
|
12384
|
+
* @public exported from `@promptbook/components`
|
|
12385
|
+
*/
|
|
12386
|
+
const RANDOM_FLOW = {
|
|
12387
|
+
beforeFirstMessage: [500, 2000],
|
|
12388
|
+
thinkingBetweenMessages: [500, 3500],
|
|
12389
|
+
waitAfterWord: [10, 200],
|
|
12390
|
+
extraWordDelay: [0, 100],
|
|
12391
|
+
longPauseChance: 0.25,
|
|
12392
|
+
longPauseDuration: [1200, 5000],
|
|
12393
|
+
};
|
|
12394
|
+
/**
|
|
12395
|
+
* All predefined delay configurations for the `MockedChat` component
|
|
12396
|
+
*
|
|
12397
|
+
* @public exported from `@promptbook/components`
|
|
12398
|
+
*/
|
|
12399
|
+
const MOCKED_CHAT_DELAY_CONFIGS = {
|
|
12400
|
+
NORMAL_FLOW,
|
|
12401
|
+
FAST_FLOW,
|
|
12402
|
+
SLOW_FLOW,
|
|
12403
|
+
BLOCKY_FLOW,
|
|
12404
|
+
RANDOM_FLOW,
|
|
12405
|
+
};
|
|
12406
|
+
/**
|
|
12407
|
+
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
12408
|
+
*/
|
|
12409
|
+
|
|
12410
|
+
/**
|
|
12411
|
+
* MockedChat component that shows the same chat as Chat but emulates ongoing discussion
|
|
12412
|
+
* with realistic typing delays and thinking pauses.
|
|
12413
|
+
*
|
|
12414
|
+
* @public exported from `@promptbook/components`
|
|
12415
|
+
*/
|
|
12416
|
+
function MockedChat(props) {
|
|
12417
|
+
const { delayConfig, messages: originalMessages, isResettable = true, isPausable = true, isSaveButtonEnabled = true, ...chatProps } = props;
|
|
12418
|
+
// Helper to get random delay from config
|
|
12419
|
+
function getDelay(val, fallback) {
|
|
12420
|
+
if (Array.isArray(val) && val.length === 2) {
|
|
12421
|
+
const [min, max] = val;
|
|
12422
|
+
return Math.floor(Math.random() * (max - min + 1)) + min;
|
|
11903
12423
|
}
|
|
11904
|
-
|
|
11905
|
-
|
|
11906
|
-
|
|
11907
|
-
|
|
11908
|
-
|
|
12424
|
+
if (typeof val === 'number')
|
|
12425
|
+
return val;
|
|
12426
|
+
return fallback;
|
|
12427
|
+
}
|
|
12428
|
+
const delays = {
|
|
12429
|
+
...MOCKED_CHAT_DELAY_CONFIGS.NORMAL_FLOW,
|
|
12430
|
+
...delayConfig,
|
|
12431
|
+
};
|
|
12432
|
+
const [displayedMessages, setDisplayedMessages] = useState([]);
|
|
12433
|
+
const [isSimulationComplete, setIsSimulationComplete] = useState(false);
|
|
12434
|
+
// Playback state machine
|
|
12435
|
+
// RUNNING -> (user clicks Pause) -> PAUSING (finish current message) -> PAUSED
|
|
12436
|
+
// PAUSED -> (user clicks Resume) -> RUNNING
|
|
12437
|
+
const [playbackState, setPlaybackState] = useState('RUNNING');
|
|
12438
|
+
const pauseRequestedRef = useRef(false);
|
|
12439
|
+
const [resetNonce, setResetNonce] = useState(0);
|
|
12440
|
+
const onReset = useMemo(() => {
|
|
12441
|
+
if (!isResettable) {
|
|
12442
|
+
return undefined;
|
|
11909
12443
|
}
|
|
11910
|
-
const unsubscribe = speechRecognition.subscribe((event) => {
|
|
11911
|
-
if (event.type === 'START') {
|
|
11912
|
-
setSpeechRecognitionState('RECORDING');
|
|
11913
|
-
}
|
|
11914
|
-
else if (event.type === 'RESULT') {
|
|
11915
|
-
// [🧠] Note: This logic assumes that interim results are being updated.
|
|
11916
|
-
// For OpenAiSpeechRecognition, it's just one final result.
|
|
11917
|
-
if (textareaRef.current) {
|
|
11918
|
-
const textarea = textareaRef.current;
|
|
11919
|
-
const currentValue = textarea.value;
|
|
11920
|
-
// Append the transcribed text with a space if needed
|
|
11921
|
-
const separator = currentValue && !currentValue.endsWith(' ') && !currentValue.endsWith('\n') ? ' ' : '';
|
|
11922
|
-
textarea.value += separator + event.text;
|
|
11923
|
-
if (onChange) {
|
|
11924
|
-
onChange(textarea.value);
|
|
11925
|
-
}
|
|
11926
|
-
}
|
|
11927
|
-
}
|
|
11928
|
-
else if (event.type === 'ERROR') {
|
|
11929
|
-
setSpeechRecognitionState('ERROR');
|
|
11930
|
-
alert(`Speech recognition error: ${event.message}`);
|
|
11931
|
-
}
|
|
11932
|
-
else if (event.type === 'STOP') {
|
|
11933
|
-
setSpeechRecognitionState('IDLE');
|
|
11934
|
-
}
|
|
11935
|
-
});
|
|
11936
12444
|
return () => {
|
|
11937
|
-
|
|
12445
|
+
setDisplayedMessages([]);
|
|
12446
|
+
setIsSimulationComplete(false);
|
|
12447
|
+
setResetNonce((nonce) => nonce + 1);
|
|
11938
12448
|
};
|
|
11939
|
-
}, [
|
|
11940
|
-
|
|
11941
|
-
|
|
12449
|
+
}, [resetNonce, isResettable]);
|
|
12450
|
+
// Helper: Wait while paused (entered only between messages, never mid-typing)
|
|
12451
|
+
const waitIfPaused = async (isCancelledRef) => {
|
|
12452
|
+
if (!pauseRequestedRef.current)
|
|
11942
12453
|
return;
|
|
12454
|
+
setPlaybackState('PAUSED');
|
|
12455
|
+
// Busy wait with small sleeps until resume
|
|
12456
|
+
while (pauseRequestedRef.current) {
|
|
12457
|
+
if (isCancelledRef())
|
|
12458
|
+
return;
|
|
12459
|
+
await forTime(100);
|
|
11943
12460
|
}
|
|
11944
|
-
|
|
11945
|
-
|
|
12461
|
+
// Resumed
|
|
12462
|
+
setPlaybackState('RUNNING');
|
|
12463
|
+
};
|
|
12464
|
+
const requestPause = () => {
|
|
12465
|
+
if (playbackState === 'RUNNING') {
|
|
12466
|
+
pauseRequestedRef.current = true;
|
|
12467
|
+
// Will flip to PAUSING when current message completes
|
|
12468
|
+
setPlaybackState('PAUSING');
|
|
11946
12469
|
}
|
|
11947
|
-
|
|
11948
|
-
|
|
12470
|
+
};
|
|
12471
|
+
const resume = () => {
|
|
12472
|
+
pauseRequestedRef.current = false;
|
|
12473
|
+
if (playbackState !== 'RUNNING') {
|
|
12474
|
+
// Actual state will become RUNNING after loop exits waitIfPaused
|
|
12475
|
+
setPlaybackState('RUNNING');
|
|
11949
12476
|
}
|
|
11950
|
-
}
|
|
11951
|
-
|
|
11952
|
-
|
|
11953
|
-
|
|
11954
|
-
|
|
11955
|
-
|
|
11956
|
-
|
|
11957
|
-
|
|
11958
|
-
|
|
11959
|
-
|
|
11960
|
-
for (const file of fileArray) {
|
|
11961
|
-
const content = await onFileUpload(file);
|
|
11962
|
-
newUploadedFiles.push({
|
|
11963
|
-
id: Math.random().toString(36).substring(2),
|
|
11964
|
-
file,
|
|
11965
|
-
content,
|
|
11966
|
-
});
|
|
12477
|
+
};
|
|
12478
|
+
useEffect(() => {
|
|
12479
|
+
let isCancelled = false;
|
|
12480
|
+
const simulateChat = async () => {
|
|
12481
|
+
// Reset state
|
|
12482
|
+
setDisplayedMessages([]);
|
|
12483
|
+
setIsSimulationComplete(false);
|
|
12484
|
+
if (originalMessages.length === 0) {
|
|
12485
|
+
setIsSimulationComplete(true);
|
|
12486
|
+
return;
|
|
11967
12487
|
}
|
|
11968
|
-
|
|
11969
|
-
|
|
11970
|
-
|
|
11971
|
-
|
|
11972
|
-
|
|
11973
|
-
|
|
11974
|
-
|
|
11975
|
-
|
|
11976
|
-
|
|
11977
|
-
|
|
11978
|
-
|
|
11979
|
-
|
|
11980
|
-
|
|
11981
|
-
|
|
11982
|
-
|
|
11983
|
-
|
|
11984
|
-
|
|
11985
|
-
|
|
11986
|
-
|
|
11987
|
-
|
|
11988
|
-
|
|
11989
|
-
|
|
12488
|
+
// Show intermediate messages immediately
|
|
12489
|
+
const showIntermediateMessages = delays.showIntermediateMessages || 0;
|
|
12490
|
+
if (showIntermediateMessages > 0) {
|
|
12491
|
+
setDisplayedMessages(originalMessages.slice(0, showIntermediateMessages));
|
|
12492
|
+
}
|
|
12493
|
+
// Wait before first message (randomized)
|
|
12494
|
+
await forTime(getDelay(delays.beforeFirstMessage, 1000));
|
|
12495
|
+
if (isCancelled)
|
|
12496
|
+
return;
|
|
12497
|
+
for (let i = showIntermediateMessages; i < originalMessages.length; i++) {
|
|
12498
|
+
// If a pause was requested earlier, we only pause between messages
|
|
12499
|
+
if (pauseRequestedRef.current) {
|
|
12500
|
+
await waitIfPaused(() => isCancelled);
|
|
12501
|
+
if (isCancelled)
|
|
12502
|
+
return;
|
|
12503
|
+
}
|
|
12504
|
+
if (isCancelled)
|
|
12505
|
+
return;
|
|
12506
|
+
const currentMessage = originalMessages[i];
|
|
12507
|
+
if (!currentMessage)
|
|
12508
|
+
continue;
|
|
12509
|
+
// Add thinking delay between messages (except for the first one)
|
|
12510
|
+
if (i > 0) {
|
|
12511
|
+
// Sometimes do a longer pause (agent switch or random)
|
|
12512
|
+
let didLongPause = false;
|
|
12513
|
+
if (delays.longPauseChance &&
|
|
12514
|
+
Math.random() < delays.longPauseChance &&
|
|
12515
|
+
i > 0 &&
|
|
12516
|
+
originalMessages[i].sender !== originalMessages[i - 1].sender) {
|
|
12517
|
+
await forTime(getDelay(delays.longPauseDuration, 2000));
|
|
12518
|
+
didLongPause = true;
|
|
12519
|
+
if (isCancelled)
|
|
12520
|
+
return;
|
|
12521
|
+
}
|
|
12522
|
+
// Otherwise normal thinking delay
|
|
12523
|
+
if (!didLongPause) {
|
|
12524
|
+
await forTime(getDelay(delays.thinkingBetweenMessages, 2000));
|
|
12525
|
+
if (isCancelled)
|
|
12526
|
+
return;
|
|
12527
|
+
}
|
|
12528
|
+
// Pause check (still between messages)
|
|
12529
|
+
if (pauseRequestedRef.current) {
|
|
12530
|
+
await waitIfPaused(() => isCancelled);
|
|
12531
|
+
if (isCancelled)
|
|
12532
|
+
return;
|
|
12533
|
+
}
|
|
12534
|
+
}
|
|
12535
|
+
// Show incomplete message first (for typing effect)
|
|
12536
|
+
const incompleteMessage = {
|
|
12537
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
12538
|
+
id: currentMessage.id,
|
|
12539
|
+
createdAt: currentMessage.createdAt,
|
|
12540
|
+
sender: currentMessage.sender,
|
|
12541
|
+
content: '',
|
|
12542
|
+
isComplete: false,
|
|
12543
|
+
expectedAnswer: currentMessage.expectedAnswer,
|
|
12544
|
+
isVoiceCall: currentMessage.isVoiceCall,
|
|
12545
|
+
};
|
|
12546
|
+
setDisplayedMessages((prev) => [...prev, incompleteMessage]);
|
|
12547
|
+
// Split message content into words
|
|
12548
|
+
const words = currentMessage.content.split(' ');
|
|
12549
|
+
let currentContent = '';
|
|
12550
|
+
// Type each word with delay (randomized)
|
|
12551
|
+
for (let wordIndex = 0; wordIndex < words.length; wordIndex++) {
|
|
12552
|
+
if (isCancelled)
|
|
12553
|
+
return;
|
|
12554
|
+
const word = words[wordIndex];
|
|
12555
|
+
currentContent += (wordIndex > 0 ? ' ' : '') + word;
|
|
12556
|
+
// Update the message with current content
|
|
12557
|
+
const updatingMessage = {
|
|
12558
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
12559
|
+
id: currentMessage.id,
|
|
12560
|
+
createdAt: currentMessage.createdAt,
|
|
12561
|
+
sender: currentMessage.sender,
|
|
12562
|
+
content: currentContent,
|
|
12563
|
+
isComplete: false,
|
|
12564
|
+
expectedAnswer: currentMessage.expectedAnswer,
|
|
12565
|
+
isVoiceCall: currentMessage.isVoiceCall,
|
|
12566
|
+
};
|
|
12567
|
+
setDisplayedMessages((prev) => {
|
|
12568
|
+
const newMessages = [...prev];
|
|
12569
|
+
newMessages[newMessages.length - 1] = updatingMessage;
|
|
12570
|
+
return newMessages;
|
|
12571
|
+
});
|
|
12572
|
+
// Wait after word with extra delay (randomized)
|
|
12573
|
+
await forTime(getDelay(delays.waitAfterWord, 100) + getDelay(delays.extraWordDelay, 50));
|
|
12574
|
+
if (isCancelled)
|
|
12575
|
+
return;
|
|
12576
|
+
}
|
|
12577
|
+
// Mark message as complete
|
|
12578
|
+
const completeMessage = {
|
|
12579
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
12580
|
+
id: currentMessage.id,
|
|
12581
|
+
createdAt: currentMessage.createdAt,
|
|
12582
|
+
sender: currentMessage.sender,
|
|
12583
|
+
content: currentMessage.content,
|
|
12584
|
+
isComplete: true,
|
|
12585
|
+
expectedAnswer: currentMessage.expectedAnswer,
|
|
12586
|
+
isVoiceCall: currentMessage.isVoiceCall,
|
|
12587
|
+
};
|
|
12588
|
+
setDisplayedMessages((prev) => {
|
|
12589
|
+
const newMessages = [...prev];
|
|
12590
|
+
newMessages[newMessages.length - 1] = completeMessage;
|
|
12591
|
+
return newMessages;
|
|
12592
|
+
});
|
|
12593
|
+
// Small pause after completing the message
|
|
12594
|
+
await forTime(200);
|
|
12595
|
+
if (isCancelled)
|
|
12596
|
+
return;
|
|
12597
|
+
// Transition PAUSING -> PAUSED (after finishing current message)
|
|
12598
|
+
if (pauseRequestedRef.current && playbackState === 'PAUSING') ;
|
|
12599
|
+
}
|
|
12600
|
+
setIsSimulationComplete(true);
|
|
12601
|
+
};
|
|
12602
|
+
simulateChat().catch((error) => {
|
|
12603
|
+
if (!isCancelled) {
|
|
12604
|
+
console.error('Error in MockedChat simulation:', error);
|
|
12605
|
+
// Fallback to showing all messages immediately
|
|
12606
|
+
setDisplayedMessages(originalMessages);
|
|
12607
|
+
setIsSimulationComplete(true);
|
|
12608
|
+
}
|
|
12609
|
+
});
|
|
12610
|
+
return () => {
|
|
12611
|
+
isCancelled = true;
|
|
12612
|
+
};
|
|
12613
|
+
}, [
|
|
12614
|
+
originalMessages,
|
|
12615
|
+
delays.beforeFirstMessage,
|
|
12616
|
+
delays.thinkingBetweenMessages,
|
|
12617
|
+
delays.waitAfterWord,
|
|
12618
|
+
delays.extraWordDelay,
|
|
12619
|
+
resetNonce,
|
|
12620
|
+
]);
|
|
12621
|
+
// Build extra actions (Pause / Resume)
|
|
12622
|
+
const showPauseButton = isPausable && !isSimulationComplete;
|
|
12623
|
+
const extraActions = showPauseButton ? (jsxs("button", { className: `${styles$2.chatButton} ${styles$2.pauseButton} ${playbackState === 'PAUSING' ? styles$2.pausing : playbackState === 'PAUSED' ? styles$2.paused : ''}`, "aria-label": playbackState === 'RUNNING'
|
|
12624
|
+
? 'Pause simulation'
|
|
12625
|
+
: playbackState === 'PAUSING'
|
|
12626
|
+
? 'Pausing simulation'
|
|
12627
|
+
: 'Resume simulation', onClick: () => {
|
|
12628
|
+
if (playbackState === 'RUNNING') {
|
|
12629
|
+
requestPause();
|
|
12630
|
+
}
|
|
12631
|
+
else if (playbackState === 'PAUSED') {
|
|
12632
|
+
resume();
|
|
12633
|
+
}
|
|
12634
|
+
}, disabled: playbackState === 'PAUSING', children: [playbackState === 'RUNNING' && (jsxs(Fragment, { children: [jsx(PauseIcon, { size: 16 }), jsx("span", { className: styles$2.chatButtonText, children: "Pause" })] })), playbackState === 'PAUSING' && (jsxs(Fragment, { children: [jsx(PauseIcon, { size: 16 }), jsx("span", { className: styles$2.chatButtonText, children: "Pausing\u2026" })] })), playbackState === 'PAUSED' && (jsxs(Fragment, { children: [jsx(PlayIcon, { size: 16 }), jsx("span", { className: styles$2.chatButtonText, children: "Resume" })] }))] })) : undefined;
|
|
12635
|
+
// Use the internal Chat component with simulated messages
|
|
12636
|
+
return (jsx(Chat, { ...chatProps, onReset: isResettable ? onReset : undefined, messages: displayedMessages, extraActions: extraActions, isSaveButtonEnabled: isSaveButtonEnabled, saveFormats: ['json', 'md', 'txt', 'html'],
|
|
12637
|
+
// Disable input during simulation unless explicitly completed
|
|
12638
|
+
onMessage: isSimulationComplete && chatProps.onMessage ? chatProps.onMessage : undefined }));
|
|
12639
|
+
}
|
|
12640
|
+
|
|
12641
|
+
/**
|
|
12642
|
+
* Renders a nice clock icon with a specific time.
|
|
12643
|
+
*
|
|
12644
|
+
* @private internal subcomponent of `<Chat>` component
|
|
12645
|
+
*/
|
|
12646
|
+
function ClockIcon({ date, size = 100 }) {
|
|
12647
|
+
const hours = date.getHours();
|
|
12648
|
+
const minutes = date.getMinutes();
|
|
12649
|
+
const seconds = date.getSeconds();
|
|
12650
|
+
const hourDeg = (hours % 12) * 30 + minutes * 0.5;
|
|
12651
|
+
const minuteDeg = minutes * 6;
|
|
12652
|
+
const secondDeg = seconds * 6;
|
|
12653
|
+
return (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: [jsx("circle", { cx: "50", cy: "50", r: "45", fill: "white", stroke: "#333", strokeWidth: "2" }), [...Array(12)].map((_, i) => (jsx("line", { x1: "50", y1: "12", x2: "50", y2: "18", transform: `rotate(${i * 30} 50 50)`, stroke: "#333", strokeWidth: "2" }, i))), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "25", transform: `rotate(${hourDeg} 50 50)`, stroke: "#333", strokeWidth: "4", strokeLinecap: "round" }), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "15", transform: `rotate(${minuteDeg} 50 50)`, stroke: "#666", strokeWidth: "3", strokeLinecap: "round" }), jsx("line", { x1: "50", y1: "50", x2: "50", y2: "12", transform: `rotate(${secondDeg} 50 50)`, stroke: "#ff4444", strokeWidth: "1.5", strokeLinecap: "round" }), jsx("circle", { cx: "50", cy: "50", r: "2", fill: "#333" })] }));
|
|
12654
|
+
}
|
|
12655
|
+
|
|
12656
|
+
const CONFETTI_TRIGGER_EMOJI = '🎉';
|
|
12657
|
+
const HEART_TRIGGER_EMOJIS = [
|
|
12658
|
+
'❤️',
|
|
12659
|
+
'❤',
|
|
12660
|
+
'💙',
|
|
12661
|
+
'💚',
|
|
12662
|
+
'💛',
|
|
12663
|
+
'💜',
|
|
12664
|
+
'🧡',
|
|
12665
|
+
'🤍',
|
|
12666
|
+
'🤎',
|
|
12667
|
+
'🖤',
|
|
12668
|
+
'💖',
|
|
12669
|
+
'💗',
|
|
12670
|
+
'💓',
|
|
12671
|
+
'💞',
|
|
12672
|
+
'💕',
|
|
12673
|
+
'💘',
|
|
12674
|
+
'💝',
|
|
12675
|
+
'💟',
|
|
12676
|
+
'❣️',
|
|
12677
|
+
'💔',
|
|
12678
|
+
'❤️🔥',
|
|
12679
|
+
'❤️🩹',
|
|
12680
|
+
];
|
|
12681
|
+
const HEART_PARTICLE_EMOJIS = ['❤️', '💖', '💗', '💜', '💙', '💚', '💛', '🧡', '🤍', '🤎', '🖤'];
|
|
12682
|
+
const CONFETTI_COLORS = ['#f97316', '#facc15', '#34d399', '#60a5fa', '#f472b6', '#a78bfa'];
|
|
12683
|
+
const EMOJI_EFFECT_CONFIG = {
|
|
12684
|
+
confetti: {
|
|
12685
|
+
particleCount: 24,
|
|
12686
|
+
durationRangeMs: [900, 1500],
|
|
12687
|
+
delayRangeMs: [0, 250],
|
|
12688
|
+
removalBufferMs: 200,
|
|
12689
|
+
},
|
|
12690
|
+
hearts: {
|
|
12691
|
+
particleCount: 14,
|
|
12692
|
+
durationRangeMs: [1600, 2400],
|
|
12693
|
+
delayRangeMs: [0, 400],
|
|
12694
|
+
removalBufferMs: 300,
|
|
12695
|
+
},
|
|
12696
|
+
};
|
|
12697
|
+
const SOUND_PREFERENCE_KEY = 'promptbook-chat-sounds-enabled';
|
|
12698
|
+
const TYPING_SOUND_INTERVAL_MS = 1200;
|
|
12699
|
+
const escapeRegExp = (value) => value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
12700
|
+
const HEART_EMOJI_REGEX = new RegExp(HEART_TRIGGER_EMOJIS.map(escapeRegExp).join('|'), 'u');
|
|
12701
|
+
const randomBetween = (min, max) => Math.random() * (max - min) + min;
|
|
12702
|
+
const randomPick = (items) => items[Math.floor(Math.random() * items.length)];
|
|
12703
|
+
const buildConfettiParticles = (effectId) => {
|
|
12704
|
+
const config = EMOJI_EFFECT_CONFIG.confetti;
|
|
12705
|
+
return Array.from({ length: config.particleCount }, (_, index) => {
|
|
12706
|
+
const size = Math.round(randomBetween(6, 12));
|
|
12707
|
+
return {
|
|
12708
|
+
id: `${effectId}-confetti-${index}`,
|
|
12709
|
+
style: {
|
|
12710
|
+
'--x': `${Math.round(randomBetween(5, 95))}%`,
|
|
12711
|
+
'--size': `${size}px`,
|
|
12712
|
+
'--delay': `${Math.round(randomBetween(...config.delayRangeMs))}ms`,
|
|
12713
|
+
'--duration': `${Math.round(randomBetween(...config.durationRangeMs))}ms`,
|
|
12714
|
+
'--drift': `${Math.round(randomBetween(-60, 60))}px`,
|
|
12715
|
+
'--rotation': `${Math.round(randomBetween(0, 360))}deg`,
|
|
12716
|
+
'--color': randomPick(CONFETTI_COLORS),
|
|
12717
|
+
},
|
|
12718
|
+
};
|
|
12719
|
+
});
|
|
12720
|
+
};
|
|
12721
|
+
const buildHeartParticles = (effectId) => {
|
|
12722
|
+
const config = EMOJI_EFFECT_CONFIG.hearts;
|
|
12723
|
+
return Array.from({ length: config.particleCount }, (_, index) => {
|
|
12724
|
+
return {
|
|
12725
|
+
id: `${effectId}-heart-${index}`,
|
|
12726
|
+
symbol: randomPick(HEART_PARTICLE_EMOJIS),
|
|
12727
|
+
style: {
|
|
12728
|
+
'--x': `${Math.round(randomBetween(10, 90))}%`,
|
|
12729
|
+
'--size': `${Math.round(randomBetween(16, 28))}px`,
|
|
12730
|
+
'--delay': `${Math.round(randomBetween(...config.delayRangeMs))}ms`,
|
|
12731
|
+
'--duration': `${Math.round(randomBetween(...config.durationRangeMs))}ms`,
|
|
12732
|
+
'--drift': `${Math.round(randomBetween(-40, 40))}px`,
|
|
12733
|
+
},
|
|
12734
|
+
};
|
|
12735
|
+
});
|
|
12736
|
+
};
|
|
12737
|
+
/**
|
|
12738
|
+
* Renders a chat with messages and input for new messages
|
|
12739
|
+
*
|
|
12740
|
+
* Note: 🔇 This component does NOT have speak functionality, it just allows to trigger voice recognition
|
|
12741
|
+
*
|
|
12742
|
+
* Note: There are multiple chat components:
|
|
12743
|
+
* - `<Chat/>` renders chat as it is without any logic
|
|
12744
|
+
* - `<LlmChat/>` connected to LLM Execution Tools of Promptbook
|
|
12745
|
+
*
|
|
12746
|
+
* Use <WorkerChat/> or <SignalChat/> in most cases.
|
|
12747
|
+
*
|
|
12748
|
+
* @public exported from `@promptbook/components`
|
|
12749
|
+
*/
|
|
12750
|
+
function Chat(props) {
|
|
12751
|
+
var _a;
|
|
12752
|
+
const { title = 'Chat', messages, onChange, onMessage, onReset, onFeedback, onFileUpload, speechRecognition,
|
|
12753
|
+
// isVoiceRecognitionButtonShown,
|
|
12754
|
+
// voiceLanguage = 'en-US',
|
|
12755
|
+
placeholderMessageContent, defaultMessage,
|
|
12756
|
+
// tasksProgress,
|
|
12757
|
+
children, className, style,
|
|
12758
|
+
// voiceCallProps,
|
|
12759
|
+
isAiTextHumanizedAndPromptbookified = true, isVoiceCalling = false, isFocusedOnLoad,
|
|
12760
|
+
// isExperimental = false,
|
|
12761
|
+
// TODO: [😅]> isSaveButtonEnabled = false,
|
|
12762
|
+
// exportHeaderMarkdown,
|
|
12763
|
+
participants = [], extraActions, actionsContainer, saveFormats, isSaveButtonEnabled = true, isCopyButtonEnabled = true, buttonColor: buttonColorRaw, onUseTemplate, onCreateAgent, toolTitles, visual, } = props;
|
|
12764
|
+
const buttonColor = useMemo(() => Color.from(buttonColorRaw || '#0066cc'), [buttonColorRaw]);
|
|
12765
|
+
const participantsByName = useMemo(() => new Map(participants.map((participant) => [participant.name, participant])), [participants]);
|
|
12766
|
+
// Use the auto-scroll hook
|
|
12767
|
+
const { isAutoScrolling, chatMessagesRef, handleScroll, handleMessagesChange, scrollToBottom, isMobile: isMobileFromHook, } = useChatAutoScroll();
|
|
12768
|
+
const menuHoisting = useMenuHoisting();
|
|
12769
|
+
const textareaRef = useRef(null);
|
|
12770
|
+
const buttonSendRef = useRef(null);
|
|
12771
|
+
const fileInputRef = useRef(null);
|
|
12772
|
+
const [ratingModalOpen, setRatingModalOpen] = useState(false);
|
|
12773
|
+
const [toolCallModalOpen, setToolCallModalOpen] = useState(false);
|
|
12774
|
+
const [selectedToolCall, setSelectedToolCall] = useState(null);
|
|
12775
|
+
const [selectedMessage, setSelectedMessage] = useState(null);
|
|
12776
|
+
const [messageRatings, setMessageRatings] = useState(new Map());
|
|
12777
|
+
const [textRating, setTextRating] = useState('');
|
|
12778
|
+
const [hoveredRating, setHoveredRating] = useState(0);
|
|
12779
|
+
const [expandedMessageId, setExpandedMessageId] = useState(null);
|
|
12780
|
+
// const [copiedToClipboard, setCopiedToClipboard] = useState(false);
|
|
12781
|
+
// const [copiedMessageId, setCopiedMessageId] = useState<string | null>(null);
|
|
12782
|
+
// const [isTyping, setIsTyping] = useState(false);
|
|
12783
|
+
// const [inputValue, setInputValue] = useState('');
|
|
12784
|
+
const [mode] = useState('LIGHT'); // Simplified light/dark mode
|
|
12785
|
+
const [ratingConfirmation, setRatingConfirmation] = useState(null);
|
|
12786
|
+
const [emojiEffects, setEmojiEffects] = useState([]);
|
|
12787
|
+
const [areSoundsEnabled, setAreSoundsEnabled] = useState(true);
|
|
12788
|
+
// File upload state
|
|
12789
|
+
const [uploadedFiles, setUploadedFiles] = useState([]);
|
|
12790
|
+
const [isDragOver, setIsDragOver] = useState(false);
|
|
12791
|
+
const [isUploading, setIsUploading] = useState(false);
|
|
12792
|
+
// Voice recognition state
|
|
12793
|
+
const [speechRecognitionState, setSpeechRecognitionState] = useState('IDLE');
|
|
12794
|
+
// Use mobile detection from the hook
|
|
12795
|
+
const isMobile = isMobileFromHook;
|
|
12796
|
+
const emojiEffectIdRef = useRef(0);
|
|
12797
|
+
const emojiEffectTimeoutsRef = useRef([]);
|
|
12798
|
+
const emojiMessageEffectRef = useRef(new Map());
|
|
12799
|
+
const audioContextRef = useRef(null);
|
|
12800
|
+
const typingSoundIntervalRef = useRef(null);
|
|
12801
|
+
const messageSoundStateRef = useRef(new Map());
|
|
12802
|
+
const hasMessageSoundInitRef = useRef(false);
|
|
12803
|
+
const hasUserInteractedRef = useRef(false);
|
|
12804
|
+
useEffect(() => {
|
|
12805
|
+
return () => {
|
|
12806
|
+
emojiEffectTimeoutsRef.current.forEach((timeoutId) => window.clearTimeout(timeoutId));
|
|
12807
|
+
emojiEffectTimeoutsRef.current = [];
|
|
12808
|
+
};
|
|
12809
|
+
}, []);
|
|
12810
|
+
useEffect(() => {
|
|
12811
|
+
try {
|
|
12812
|
+
const stored = localStorage.getItem(SOUND_PREFERENCE_KEY);
|
|
12813
|
+
if (stored === 'true') {
|
|
12814
|
+
setAreSoundsEnabled(true);
|
|
12815
|
+
}
|
|
12816
|
+
if (stored === 'false') {
|
|
12817
|
+
setAreSoundsEnabled(false);
|
|
12818
|
+
}
|
|
12819
|
+
}
|
|
12820
|
+
catch (error) {
|
|
12821
|
+
console.warn('Failed to load chat sound preference:', error);
|
|
12822
|
+
}
|
|
12823
|
+
}, []);
|
|
12824
|
+
useEffect(() => {
|
|
12825
|
+
try {
|
|
12826
|
+
localStorage.setItem(SOUND_PREFERENCE_KEY, String(areSoundsEnabled));
|
|
12827
|
+
}
|
|
12828
|
+
catch (error) {
|
|
12829
|
+
console.warn('Failed to save chat sound preference:', error);
|
|
12830
|
+
}
|
|
12831
|
+
}, [areSoundsEnabled]);
|
|
12832
|
+
useEffect(() => {
|
|
12833
|
+
var _a;
|
|
12834
|
+
if (!areSoundsEnabled && ((_a = audioContextRef.current) === null || _a === void 0 ? void 0 : _a.state) === 'running') {
|
|
12835
|
+
void audioContextRef.current.suspend();
|
|
12836
|
+
}
|
|
12837
|
+
}, [areSoundsEnabled]);
|
|
12838
|
+
const addEmojiEffect = useCallback((kind) => {
|
|
12839
|
+
const effectId = `emoji-${emojiEffectIdRef.current++}`;
|
|
12840
|
+
const particles = kind === 'confetti' ? buildConfettiParticles(effectId) : buildHeartParticles(effectId);
|
|
12841
|
+
const config = EMOJI_EFFECT_CONFIG[kind];
|
|
12842
|
+
const removalDelay = config.durationRangeMs[1] + config.delayRangeMs[1] + config.removalBufferMs;
|
|
12843
|
+
setEmojiEffects((previous) => [...previous, { id: effectId, kind, particles }]);
|
|
12844
|
+
const timeoutId = window.setTimeout(() => {
|
|
12845
|
+
setEmojiEffects((previous) => previous.filter((effect) => effect.id !== effectId));
|
|
12846
|
+
}, removalDelay);
|
|
12847
|
+
emojiEffectTimeoutsRef.current.push(timeoutId);
|
|
12848
|
+
}, []);
|
|
12849
|
+
const isMessageFromMe = useCallback((message) => {
|
|
12850
|
+
const participant = participantsByName.get(message.sender);
|
|
12851
|
+
if (participant === null || participant === void 0 ? void 0 : participant.isMe) {
|
|
12852
|
+
return true;
|
|
12853
|
+
}
|
|
12854
|
+
return message.sender === 'USER';
|
|
12855
|
+
}, [participantsByName]);
|
|
12856
|
+
const getMessageKey = useCallback((message, index) => {
|
|
12857
|
+
return message.id || `${index}-${message.sender}`;
|
|
12858
|
+
}, []);
|
|
12859
|
+
const ensureAudioContext = useCallback((force = false) => {
|
|
12860
|
+
if (!areSoundsEnabled) {
|
|
12861
|
+
return null;
|
|
12862
|
+
}
|
|
12863
|
+
if (!force && !hasUserInteractedRef.current) {
|
|
12864
|
+
return null;
|
|
12865
|
+
}
|
|
12866
|
+
if (typeof window === 'undefined') {
|
|
12867
|
+
return null;
|
|
12868
|
+
}
|
|
12869
|
+
if (!audioContextRef.current) {
|
|
12870
|
+
const AudioContextCtor = window.AudioContext ||
|
|
12871
|
+
window.webkitAudioContext;
|
|
12872
|
+
if (!AudioContextCtor) {
|
|
12873
|
+
return null;
|
|
12874
|
+
}
|
|
12875
|
+
audioContextRef.current = new AudioContextCtor();
|
|
12876
|
+
}
|
|
12877
|
+
if (audioContextRef.current.state === 'suspended') {
|
|
12878
|
+
void audioContextRef.current.resume();
|
|
12879
|
+
}
|
|
12880
|
+
return audioContextRef.current;
|
|
12881
|
+
}, [areSoundsEnabled]);
|
|
12882
|
+
const markUserInteracted = useCallback(() => {
|
|
12883
|
+
if (!hasUserInteractedRef.current) {
|
|
12884
|
+
hasUserInteractedRef.current = true;
|
|
12885
|
+
}
|
|
12886
|
+
ensureAudioContext(true);
|
|
12887
|
+
}, [ensureAudioContext]);
|
|
12888
|
+
const playTone = useCallback((config) => {
|
|
12889
|
+
const audioContext = ensureAudioContext();
|
|
12890
|
+
if (!audioContext) {
|
|
12891
|
+
return;
|
|
12892
|
+
}
|
|
12893
|
+
const startTime = audioContext.currentTime + (config.startOffsetMs || 0) / 1000;
|
|
12894
|
+
const durationSec = config.durationMs / 1000;
|
|
12895
|
+
const oscillator = audioContext.createOscillator();
|
|
12896
|
+
const gainNode = audioContext.createGain();
|
|
12897
|
+
oscillator.type = config.type || 'sine';
|
|
12898
|
+
oscillator.frequency.setValueAtTime(config.frequency, startTime);
|
|
12899
|
+
if (config.frequencyEnd) {
|
|
12900
|
+
oscillator.frequency.exponentialRampToValueAtTime(config.frequencyEnd, startTime + durationSec);
|
|
12901
|
+
}
|
|
12902
|
+
gainNode.gain.setValueAtTime(0.0001, startTime);
|
|
12903
|
+
gainNode.gain.exponentialRampToValueAtTime(config.volume, startTime + Math.min(0.02, durationSec / 3));
|
|
12904
|
+
gainNode.gain.exponentialRampToValueAtTime(0.0001, startTime + durationSec);
|
|
12905
|
+
oscillator.connect(gainNode);
|
|
12906
|
+
gainNode.connect(audioContext.destination);
|
|
12907
|
+
oscillator.start(startTime);
|
|
12908
|
+
oscillator.stop(startTime + durationSec + 0.05);
|
|
12909
|
+
}, [ensureAudioContext]);
|
|
12910
|
+
const playSound = useCallback((sound) => {
|
|
12911
|
+
if (!areSoundsEnabled) {
|
|
12912
|
+
return;
|
|
12913
|
+
}
|
|
12914
|
+
switch (sound) {
|
|
12915
|
+
case 'ding':
|
|
12916
|
+
playTone({
|
|
12917
|
+
frequency: 880,
|
|
12918
|
+
frequencyEnd: 1320,
|
|
12919
|
+
durationMs: 160,
|
|
12920
|
+
volume: 0.05,
|
|
12921
|
+
type: 'sine',
|
|
12922
|
+
});
|
|
12923
|
+
playTone({
|
|
12924
|
+
frequency: 660,
|
|
12925
|
+
frequencyEnd: 990,
|
|
12926
|
+
durationMs: 140,
|
|
12927
|
+
volume: 0.04,
|
|
12928
|
+
type: 'sine',
|
|
12929
|
+
startOffsetMs: 70,
|
|
12930
|
+
});
|
|
12931
|
+
break;
|
|
12932
|
+
case 'whoosh':
|
|
12933
|
+
playTone({
|
|
12934
|
+
frequency: 520,
|
|
12935
|
+
frequencyEnd: 180,
|
|
12936
|
+
durationMs: 220,
|
|
12937
|
+
volume: 0.04,
|
|
12938
|
+
type: 'triangle',
|
|
12939
|
+
});
|
|
12940
|
+
break;
|
|
12941
|
+
case 'tap':
|
|
12942
|
+
playTone({
|
|
12943
|
+
frequency: 900,
|
|
12944
|
+
frequencyEnd: 600,
|
|
12945
|
+
durationMs: 40,
|
|
12946
|
+
volume: 0.03,
|
|
12947
|
+
type: 'square',
|
|
12948
|
+
});
|
|
12949
|
+
break;
|
|
12950
|
+
case 'typing':
|
|
12951
|
+
playTone({
|
|
12952
|
+
frequency: 520,
|
|
12953
|
+
frequencyEnd: 420,
|
|
12954
|
+
durationMs: 35,
|
|
12955
|
+
volume: 0.02,
|
|
12956
|
+
type: 'triangle',
|
|
12957
|
+
});
|
|
12958
|
+
break;
|
|
12959
|
+
}
|
|
12960
|
+
}, [areSoundsEnabled, playTone]);
|
|
12961
|
+
const handleChatButtonClickCapture = useCallback((event) => {
|
|
12962
|
+
const target = event.target;
|
|
12963
|
+
if (!target) {
|
|
12964
|
+
return;
|
|
12965
|
+
}
|
|
12966
|
+
const button = target.closest('button');
|
|
12967
|
+
if (!button || button.disabled) {
|
|
12968
|
+
return;
|
|
12969
|
+
}
|
|
12970
|
+
if (button.getAttribute('data-sound') === 'send') {
|
|
12971
|
+
return;
|
|
12972
|
+
}
|
|
12973
|
+
markUserInteracted();
|
|
12974
|
+
playSound('tap');
|
|
12975
|
+
}, [markUserInteracted, playSound]);
|
|
12976
|
+
const handleChatUserInteractionCapture = useCallback(() => {
|
|
12977
|
+
markUserInteracted();
|
|
12978
|
+
}, [markUserInteracted]);
|
|
12979
|
+
useEffect(( /* Focus textarea on page load */) => {
|
|
12980
|
+
if (!textareaRef.current) {
|
|
12981
|
+
return;
|
|
12982
|
+
}
|
|
12983
|
+
// Note: By default, only auto-focus on desktop to prevent mobile keyboard from popping up
|
|
12984
|
+
const isFocused = isFocusedOnLoad !== null && isFocusedOnLoad !== void 0 ? isFocusedOnLoad : !isMobile;
|
|
12985
|
+
if (isFocused) {
|
|
12986
|
+
textareaRef.current.focus();
|
|
12987
|
+
}
|
|
12988
|
+
}, [textareaRef, isMobile, isFocusedOnLoad]);
|
|
12989
|
+
// Voice recognition effects
|
|
12990
|
+
useEffect(() => {
|
|
12991
|
+
if (!speechRecognition) {
|
|
12992
|
+
return;
|
|
12993
|
+
}
|
|
12994
|
+
const unsubscribe = speechRecognition.subscribe((event) => {
|
|
12995
|
+
if (event.type === 'START') {
|
|
12996
|
+
setSpeechRecognitionState('RECORDING');
|
|
12997
|
+
}
|
|
12998
|
+
else if (event.type === 'RESULT') {
|
|
12999
|
+
// [🧠] Note: This logic assumes that interim results are being updated.
|
|
13000
|
+
// For OpenAiSpeechRecognition, it's just one final result.
|
|
13001
|
+
if (textareaRef.current) {
|
|
13002
|
+
const textarea = textareaRef.current;
|
|
13003
|
+
const currentValue = textarea.value;
|
|
13004
|
+
// Append the transcribed text with a space if needed
|
|
13005
|
+
const separator = currentValue && !currentValue.endsWith(' ') && !currentValue.endsWith('\n') ? ' ' : '';
|
|
13006
|
+
textarea.value += separator + event.text;
|
|
13007
|
+
if (onChange) {
|
|
13008
|
+
onChange(textarea.value);
|
|
13009
|
+
}
|
|
13010
|
+
}
|
|
13011
|
+
}
|
|
13012
|
+
else if (event.type === 'ERROR') {
|
|
13013
|
+
setSpeechRecognitionState('ERROR');
|
|
13014
|
+
alert(`Speech recognition error: ${event.message}`);
|
|
13015
|
+
}
|
|
13016
|
+
else if (event.type === 'STOP') {
|
|
13017
|
+
setSpeechRecognitionState('IDLE');
|
|
13018
|
+
}
|
|
13019
|
+
});
|
|
13020
|
+
return () => {
|
|
13021
|
+
unsubscribe();
|
|
13022
|
+
};
|
|
13023
|
+
}, [speechRecognition, onChange]);
|
|
13024
|
+
const handleToggleVoiceInput = useCallback(() => {
|
|
13025
|
+
if (!speechRecognition) {
|
|
13026
|
+
return;
|
|
13027
|
+
}
|
|
13028
|
+
if (speechRecognition.state === 'IDLE' || speechRecognition.state === 'ERROR') {
|
|
13029
|
+
speechRecognition.$start({ language: /* 'en-US' */ 'en' });
|
|
13030
|
+
}
|
|
13031
|
+
else {
|
|
13032
|
+
speechRecognition.$stop();
|
|
13033
|
+
}
|
|
13034
|
+
}, [speechRecognition]);
|
|
13035
|
+
const sendMessageWithSound = useCallback(async (messageContent, attachments) => {
|
|
13036
|
+
if (!onMessage) {
|
|
13037
|
+
throw new Error(`Can not find onMessage callback`);
|
|
13038
|
+
}
|
|
13039
|
+
markUserInteracted();
|
|
13040
|
+
playSound('whoosh');
|
|
13041
|
+
await sendMessageWithSound(messageContent, attachments);
|
|
13042
|
+
}, [onMessage, markUserInteracted, playSound]);
|
|
13043
|
+
// File upload handlers inspired by BookEditor
|
|
13044
|
+
const handleFileUpload = useCallback(async (files) => {
|
|
13045
|
+
if (!onFileUpload)
|
|
13046
|
+
return;
|
|
13047
|
+
setIsUploading(true);
|
|
13048
|
+
const fileArray = Array.from(files);
|
|
13049
|
+
try {
|
|
13050
|
+
// Process files one by one as specified in requirements
|
|
13051
|
+
const newUploadedFiles = [];
|
|
13052
|
+
for (const file of fileArray) {
|
|
13053
|
+
const content = await onFileUpload(file);
|
|
13054
|
+
newUploadedFiles.push({
|
|
13055
|
+
id: Math.random().toString(36).substring(2),
|
|
13056
|
+
file,
|
|
13057
|
+
content,
|
|
13058
|
+
});
|
|
13059
|
+
}
|
|
13060
|
+
setUploadedFiles((prev) => [...prev, ...newUploadedFiles]);
|
|
13061
|
+
}
|
|
13062
|
+
catch (error) {
|
|
13063
|
+
console.error('File upload failed:', error);
|
|
13064
|
+
alert('File upload failed. Please try again.');
|
|
13065
|
+
}
|
|
13066
|
+
finally {
|
|
13067
|
+
setIsUploading(false);
|
|
13068
|
+
}
|
|
13069
|
+
}, [onFileUpload, onChange]);
|
|
13070
|
+
const handleDrop = useCallback((event) => {
|
|
13071
|
+
event.preventDefault();
|
|
13072
|
+
setIsDragOver(false);
|
|
13073
|
+
if (!onFileUpload)
|
|
13074
|
+
return;
|
|
13075
|
+
const files = event.dataTransfer.files;
|
|
13076
|
+
if (files.length > 0) {
|
|
13077
|
+
handleFileUpload(files);
|
|
13078
|
+
}
|
|
13079
|
+
}, [onFileUpload, handleFileUpload]);
|
|
13080
|
+
const handleDragOver = useCallback((event) => {
|
|
13081
|
+
event.preventDefault();
|
|
11990
13082
|
setIsDragOver(true);
|
|
11991
13083
|
}, []);
|
|
11992
13084
|
const handleDragLeave = useCallback((event) => {
|
|
@@ -12063,7 +13155,7 @@ function Chat(props) {
|
|
|
12063
13155
|
textareaElement.focus();
|
|
12064
13156
|
}
|
|
12065
13157
|
}
|
|
12066
|
-
}, [onMessage, uploadedFiles]);
|
|
13158
|
+
}, [onMessage, uploadedFiles, sendMessageWithSound]);
|
|
12067
13159
|
const useChatCssClassName = (suffix) => `chat-${suffix}`;
|
|
12068
13160
|
const scrollToBottomCssClassName = useChatCssClassName('scrollToBottom');
|
|
12069
13161
|
const handleRating = useCallback(async (message, newRating) => {
|
|
@@ -12129,8 +13221,8 @@ function Chat(props) {
|
|
|
12129
13221
|
// Determine alignment for actions (Reset button) based on the first message
|
|
12130
13222
|
const firstMessageFromUser = ((_a = messages[0]) === null || _a === void 0 ? void 0 : _a.sender) === 'USER';
|
|
12131
13223
|
const actionsAlignmentClass = firstMessageFromUser
|
|
12132
|
-
?
|
|
12133
|
-
:
|
|
13224
|
+
? styles$2.actions + ' ' + styles$2.left
|
|
13225
|
+
: styles$2.actions + ' ' + styles$2.right;
|
|
12134
13226
|
const postprocessedMessages = useMemo(() => {
|
|
12135
13227
|
if (!isAiTextHumanizedAndPromptbookified) {
|
|
12136
13228
|
return messages;
|
|
@@ -12139,25 +13231,173 @@ function Chat(props) {
|
|
|
12139
13231
|
return { ...message, content: promptbookifyAiText(humanizeAiText(message.content)) };
|
|
12140
13232
|
});
|
|
12141
13233
|
}, [messages, isAiTextHumanizedAndPromptbookified]);
|
|
13234
|
+
// Download logic
|
|
13235
|
+
const [showSaveMenu, setShowSaveMenu] = useState(false);
|
|
13236
|
+
const [showSettingsMenu, setShowSettingsMenu] = useState(false);
|
|
13237
|
+
const saveMenuDefinitions = useMemo(() => getChatSaveFormatDefinitions(saveFormats), [saveFormats]);
|
|
13238
|
+
const toggleSaveMenu = useCallback(() => {
|
|
13239
|
+
setShowSettingsMenu(false);
|
|
13240
|
+
setShowSaveMenu((value) => !value);
|
|
13241
|
+
}, [setShowSettingsMenu, setShowSaveMenu]);
|
|
13242
|
+
const toggleSettingsMenu = useCallback(() => {
|
|
13243
|
+
setShowSaveMenu(false);
|
|
13244
|
+
setShowSettingsMenu((value) => !value);
|
|
13245
|
+
}, [setShowSaveMenu, setShowSettingsMenu]);
|
|
13246
|
+
const handleResetClick = useCallback(() => {
|
|
13247
|
+
if (!onReset) {
|
|
13248
|
+
return;
|
|
13249
|
+
}
|
|
13250
|
+
if (!confirm(`Do you really want to reset the chat?`)) {
|
|
13251
|
+
return;
|
|
13252
|
+
}
|
|
13253
|
+
onReset();
|
|
13254
|
+
}, [onReset]);
|
|
13255
|
+
const shouldShowReset = !!onReset && postprocessedMessages.length !== 0;
|
|
13256
|
+
const shouldShowSave = isSaveButtonEnabled && postprocessedMessages.length !== 0;
|
|
13257
|
+
const shouldHoistActions = !!menuHoisting && !actionsContainer && visual === 'FULL_PAGE';
|
|
13258
|
+
const menuPanelClassName = classNames(styles$2.saveMenu, shouldHoistActions && styles$2.hoistedMenu);
|
|
13259
|
+
const hoistedMenuItems = useMemo(() => {
|
|
13260
|
+
if (!shouldHoistActions) {
|
|
13261
|
+
return [];
|
|
13262
|
+
}
|
|
13263
|
+
const items = [];
|
|
13264
|
+
if (shouldShowReset) {
|
|
13265
|
+
items.push({
|
|
13266
|
+
key: 'chat-reset',
|
|
13267
|
+
icon: jsx(ResetIcon, {}),
|
|
13268
|
+
name: 'New chat',
|
|
13269
|
+
onClick: handleResetClick,
|
|
13270
|
+
});
|
|
13271
|
+
}
|
|
13272
|
+
if (shouldShowSave) {
|
|
13273
|
+
items.push({
|
|
13274
|
+
key: 'chat-save',
|
|
13275
|
+
icon: jsx(SaveIcon, { size: 18 }),
|
|
13276
|
+
name: 'Save',
|
|
13277
|
+
onClick: toggleSaveMenu,
|
|
13278
|
+
isActive: showSaveMenu,
|
|
13279
|
+
});
|
|
13280
|
+
}
|
|
13281
|
+
items.push({
|
|
13282
|
+
key: 'chat-settings',
|
|
13283
|
+
icon: jsx(SettingsIcon, { size: 18 }),
|
|
13284
|
+
name: 'Settings',
|
|
13285
|
+
onClick: toggleSettingsMenu,
|
|
13286
|
+
isActive: showSettingsMenu,
|
|
13287
|
+
});
|
|
13288
|
+
if (onUseTemplate) {
|
|
13289
|
+
items.push({
|
|
13290
|
+
key: 'chat-template',
|
|
13291
|
+
icon: jsx(TemplateIcon, { size: 16 }),
|
|
13292
|
+
name: 'Use this template',
|
|
13293
|
+
onClick: onUseTemplate,
|
|
13294
|
+
});
|
|
13295
|
+
}
|
|
13296
|
+
return items;
|
|
13297
|
+
}, [
|
|
13298
|
+
shouldHoistActions,
|
|
13299
|
+
shouldShowReset,
|
|
13300
|
+
handleResetClick,
|
|
13301
|
+
shouldShowSave,
|
|
13302
|
+
toggleSaveMenu,
|
|
13303
|
+
showSaveMenu,
|
|
13304
|
+
toggleSettingsMenu,
|
|
13305
|
+
showSettingsMenu,
|
|
13306
|
+
onUseTemplate,
|
|
13307
|
+
]);
|
|
13308
|
+
useEffect(() => {
|
|
13309
|
+
if (!menuHoisting || !shouldHoistActions) {
|
|
13310
|
+
return;
|
|
13311
|
+
}
|
|
13312
|
+
menuHoisting.setMenu(hoistedMenuItems);
|
|
13313
|
+
return () => {
|
|
13314
|
+
menuHoisting.setMenu([]);
|
|
13315
|
+
};
|
|
13316
|
+
}, [menuHoisting, shouldHoistActions, hoistedMenuItems]);
|
|
12142
13317
|
// Trigger auto-scroll when messages change
|
|
12143
13318
|
useEffect(() => {
|
|
12144
13319
|
handleMessagesChange();
|
|
12145
13320
|
}, [postprocessedMessages, handleMessagesChange]);
|
|
12146
|
-
|
|
12147
|
-
|
|
13321
|
+
useEffect(() => {
|
|
13322
|
+
messages.forEach((message, index) => {
|
|
13323
|
+
const key = String(getMessageKey(message, index));
|
|
13324
|
+
const record = emojiMessageEffectRef.current.get(key) || { confetti: false, hearts: false };
|
|
13325
|
+
const content = message.content || '';
|
|
13326
|
+
if (!record.confetti && content.includes(CONFETTI_TRIGGER_EMOJI)) {
|
|
13327
|
+
addEmojiEffect('confetti');
|
|
13328
|
+
record.confetti = true;
|
|
13329
|
+
}
|
|
13330
|
+
if (!record.hearts && HEART_EMOJI_REGEX.test(content)) {
|
|
13331
|
+
addEmojiEffect('hearts');
|
|
13332
|
+
record.hearts = true;
|
|
13333
|
+
}
|
|
13334
|
+
emojiMessageEffectRef.current.set(key, record);
|
|
13335
|
+
});
|
|
13336
|
+
}, [messages, addEmojiEffect, getMessageKey]);
|
|
13337
|
+
useEffect(() => {
|
|
13338
|
+
const nextMessageState = new Map();
|
|
13339
|
+
let shouldPlayDing = false;
|
|
13340
|
+
messages.forEach((message, index) => {
|
|
13341
|
+
var _a;
|
|
13342
|
+
const key = String(getMessageKey(message, index));
|
|
13343
|
+
const isComplete = (_a = message.isComplete) !== null && _a !== void 0 ? _a : true;
|
|
13344
|
+
nextMessageState.set(key, isComplete);
|
|
13345
|
+
if (!hasMessageSoundInitRef.current || isMessageFromMe(message)) {
|
|
13346
|
+
return;
|
|
13347
|
+
}
|
|
13348
|
+
const wasComplete = messageSoundStateRef.current.get(key);
|
|
13349
|
+
if (wasComplete === undefined && isComplete) {
|
|
13350
|
+
shouldPlayDing = true;
|
|
13351
|
+
}
|
|
13352
|
+
else if (wasComplete === false && isComplete) {
|
|
13353
|
+
shouldPlayDing = true;
|
|
13354
|
+
}
|
|
13355
|
+
});
|
|
13356
|
+
messageSoundStateRef.current = nextMessageState;
|
|
13357
|
+
if (!hasMessageSoundInitRef.current) {
|
|
13358
|
+
hasMessageSoundInitRef.current = true;
|
|
13359
|
+
return;
|
|
13360
|
+
}
|
|
13361
|
+
if (shouldPlayDing) {
|
|
13362
|
+
playSound('ding');
|
|
13363
|
+
}
|
|
13364
|
+
}, [messages, getMessageKey, isMessageFromMe, playSound]);
|
|
13365
|
+
const hasPendingAgentMessage = useMemo(() => messages.some((message) => { var _a; return !isMessageFromMe(message) && ((_a = message.isComplete) !== null && _a !== void 0 ? _a : true) === false; }), [messages, isMessageFromMe]);
|
|
13366
|
+
useEffect(() => {
|
|
13367
|
+
if (!areSoundsEnabled || !hasPendingAgentMessage) {
|
|
13368
|
+
if (typingSoundIntervalRef.current) {
|
|
13369
|
+
window.clearInterval(typingSoundIntervalRef.current);
|
|
13370
|
+
typingSoundIntervalRef.current = null;
|
|
13371
|
+
}
|
|
13372
|
+
return;
|
|
13373
|
+
}
|
|
13374
|
+
if (typingSoundIntervalRef.current) {
|
|
13375
|
+
return;
|
|
13376
|
+
}
|
|
13377
|
+
playSound('typing');
|
|
13378
|
+
typingSoundIntervalRef.current = window.setInterval(() => {
|
|
13379
|
+
playSound('typing');
|
|
13380
|
+
}, TYPING_SOUND_INTERVAL_MS);
|
|
13381
|
+
return () => {
|
|
13382
|
+
if (typingSoundIntervalRef.current) {
|
|
13383
|
+
window.clearInterval(typingSoundIntervalRef.current);
|
|
13384
|
+
typingSoundIntervalRef.current = null;
|
|
13385
|
+
}
|
|
13386
|
+
};
|
|
13387
|
+
}, [areSoundsEnabled, hasPendingAgentMessage, playSound]);
|
|
12148
13388
|
useEffect(() => {
|
|
12149
13389
|
const handleKeyDown = (event) => {
|
|
12150
13390
|
if (!(event.ctrlKey || event.metaKey) || event.key !== 's') {
|
|
12151
13391
|
return;
|
|
12152
13392
|
}
|
|
12153
13393
|
event.preventDefault();
|
|
12154
|
-
|
|
13394
|
+
toggleSaveMenu();
|
|
12155
13395
|
};
|
|
12156
13396
|
window.addEventListener('keydown', handleKeyDown);
|
|
12157
13397
|
return () => {
|
|
12158
13398
|
window.removeEventListener('keydown', handleKeyDown);
|
|
12159
13399
|
};
|
|
12160
|
-
}, [
|
|
13400
|
+
}, [toggleSaveMenu]);
|
|
12161
13401
|
const handleDownload = useCallback(async (format) => {
|
|
12162
13402
|
const formatDefinition = getChatSaveFormatDefinitions([format])[0];
|
|
12163
13403
|
if (!formatDefinition)
|
|
@@ -12184,143 +13424,199 @@ function Chat(props) {
|
|
|
12184
13424
|
const isFeedbackEnabled = !!onFeedback;
|
|
12185
13425
|
// Handler for copy button
|
|
12186
13426
|
const handleCopy = () => { };
|
|
12187
|
-
|
|
12188
|
-
|
|
12189
|
-
|
|
12190
|
-
|
|
12191
|
-
|
|
12192
|
-
|
|
12193
|
-
|
|
12194
|
-
|
|
12195
|
-
|
|
12196
|
-
|
|
12197
|
-
|
|
12198
|
-
|
|
12199
|
-
|
|
12200
|
-
|
|
12201
|
-
|
|
12202
|
-
|
|
12203
|
-
(isSaveButtonEnabled && postprocessedMessages.length !== 0) ||
|
|
12204
|
-
!!onUseTemplate ||
|
|
12205
|
-
!!extraActions;
|
|
12206
|
-
if (!hasActions) {
|
|
12207
|
-
return false;
|
|
12208
|
-
}
|
|
12209
|
-
// Detect if first message is long
|
|
12210
|
-
const firstMsg = postprocessedMessages[0];
|
|
12211
|
-
const firstMsgContent = (firstMsg === null || firstMsg === void 0 ? void 0 : firstMsg.content) || '';
|
|
12212
|
-
const firstMsgLines = firstMsgContent.split('\n').length; // <- TODO: Maybe use official counting functions here
|
|
12213
|
-
const firstMsgChars = firstMsgContent.length;
|
|
12214
|
-
const isFirstLong = firstMsgLines > 5 || firstMsgChars > 50;
|
|
12215
|
-
if (!isFirstLong) {
|
|
12216
|
-
return false;
|
|
12217
|
-
}
|
|
12218
|
-
return true;
|
|
12219
|
-
})() && chatStyles.hasActionsAndFirstMessageIsLong), ref: chatMessagesRef, onScroll: handleScroll, children: [postprocessedMessages.map((message, i) => {
|
|
12220
|
-
const participant = participants.find((participant) => participant.name === message.sender);
|
|
12221
|
-
const isLastMessage = i === postprocessedMessages.length - 1;
|
|
12222
|
-
const isExpanded = expandedMessageId === message.id;
|
|
12223
|
-
const currentRating = messageRatings.get(message.id || message.content /* <-[💃] */) || 0;
|
|
12224
|
-
return (jsx(ChatMessageItem, { message: message, participant: participant, participants: participants, isLastMessage: isLastMessage, onMessage: onMessage, setExpandedMessageId: setExpandedMessageId, isExpanded: isExpanded, currentRating: currentRating, handleRating: handleRating, mode: mode, isCopyButtonEnabled: isCopyButtonEnabled, isFeedbackEnabled: isFeedbackEnabled, onCopy: handleCopy, onCreateAgent: onCreateAgent, toolTitles: toolTitles, onToolCallClick: (toolCall) => {
|
|
12225
|
-
setSelectedToolCall(toolCall);
|
|
12226
|
-
setToolCallModalOpen(true);
|
|
12227
|
-
} }, message.id));
|
|
12228
|
-
}), jsx("div", {
|
|
12229
|
-
// Note: Extra space at bottom for input area
|
|
12230
|
-
style: { height: 100 } })] }), onMessage && (jsxs("div", { className: classNames(chatStyles.chatInput, useChatCssClassName('chatInput'), isDragOver && chatStyles.dragOver), ...(onFileUpload
|
|
12231
|
-
? {
|
|
12232
|
-
onDrop: handleDrop,
|
|
12233
|
-
onDragOver: handleDragOver,
|
|
12234
|
-
onDragLeave: handleDragLeave,
|
|
13427
|
+
const saveMenuContent = (jsx("div", { className: menuPanelClassName, children: saveMenuDefinitions.map((formatDefinition) => (jsx("button", { className: styles$2.saveMenuItem, onClick: () =>
|
|
13428
|
+
// TODO: !!! Use here `$induceFileDownload`
|
|
13429
|
+
handleDownload(formatDefinition.formatName), children: formatDefinition.label }, formatDefinition.formatName))) }));
|
|
13430
|
+
const settingsMenuContent = (jsx("div", { className: menuPanelClassName, children: jsxs("button", { className: classNames(styles$2.saveMenuItem, styles$2.settingsMenuItem), "aria-pressed": areSoundsEnabled, onClick: () => setAreSoundsEnabled((value) => !value), children: [jsx("span", { children: "Sounds" }), jsx("span", { className: classNames(styles$2.settingsToggleState, areSoundsEnabled && styles$2.settingsToggleStateActive), children: areSoundsEnabled ? 'On' : 'Off' })] }) }));
|
|
13431
|
+
return (jsxs(Fragment, { children: [ratingConfirmation && jsx("div", { className: styles$2.ratingConfirmation, children: ratingConfirmation }), jsxs("div", { className: classNames(className, styles$2.Chat, visual === 'STANDALONE' && styles$2.standaloneVisual, visual === 'FULL_PAGE' && styles$2.fullPageVisual, useChatCssClassName('Chat')), onClickCapture: handleChatButtonClickCapture, onPointerDownCapture: handleChatUserInteractionCapture, onKeyDownCapture: handleChatUserInteractionCapture, style, children: [jsx("div", { className: styles$2.emojiEffects, "aria-hidden": "true", children: emojiEffects.map((effect) => (jsx("div", { className: styles$2.emojiEffect, children: effect.particles.map((particle) => (jsx("span", { className: effect.kind === 'confetti' ? styles$2.confettiPiece : styles$2.heartParticle, style: particle.style, children: effect.kind === 'hearts' ? particle.symbol : null }, particle.id))) }, effect.id))) }), jsxs("div", { className: classNames(className, styles$2.chatMainFlow, useChatCssClassName('chatMainFlow')), children: [children && jsx("div", { className: classNames(styles$2.chatChildren), children: children }), !isAutoScrolling && (jsx("div", { className: styles$2.scrollToBottomContainer, children: jsx("button", { "data-button-type": "custom", className: classNames(styles$2.scrollToBottom, scrollToBottomCssClassName), onClick: scrollToBottom, children: jsx(ArrowIcon, { direction: "DOWN", size: 33 }) }) })), isVoiceCalling && (jsx("div", { className: styles$2.voiceCallIndicatorBar, children: jsxs("div", { className: styles$2.voiceCallIndicator, children: [jsx("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "currentColor", children: 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" }) }), jsx("span", { children: "Voice call active" }), jsx("div", { className: styles$2.voiceCallPulse })] }) })), (() => {
|
|
13432
|
+
const actionsHandlers = actionsContainer
|
|
13433
|
+
? {
|
|
13434
|
+
onClickCapture: handleChatButtonClickCapture,
|
|
13435
|
+
onPointerDownCapture: handleChatUserInteractionCapture,
|
|
13436
|
+
onKeyDownCapture: handleChatUserInteractionCapture,
|
|
13437
|
+
}
|
|
13438
|
+
: undefined;
|
|
13439
|
+
const hasInlineActions = !shouldHoistActions || !!extraActions;
|
|
13440
|
+
const actionsContent = hasInlineActions ? (jsxs("div", { className: classNames(actionsAlignmentClass, actionsContainer && styles$2.portal), ...actionsHandlers, children: [!shouldHoistActions && shouldShowReset && (jsxs("button", { className: classNames(styles$2.chatButton), onClick: handleResetClick, children: [jsx(ResetIcon, {}), jsx("span", { className: styles$2.chatButtonText, children: "New chat" })] })), !shouldHoistActions && shouldShowSave && (jsxs("div", { className: styles$2.saveButtonContainer, children: [jsxs("button", { className: classNames(styles$2.chatButton), onClick: toggleSaveMenu, "aria-haspopup": "true", "aria-expanded": showSaveMenu, children: [jsx(SaveIcon, { size: 18 }), jsx("span", { className: styles$2.chatButtonText, children: "Save" })] }), showSaveMenu && saveMenuContent] })), !shouldHoistActions && (jsxs("div", { className: styles$2.saveButtonContainer, children: [jsxs("button", { className: classNames(styles$2.chatButton), onClick: toggleSettingsMenu, "aria-haspopup": "true", "aria-expanded": showSettingsMenu, children: [jsx(SettingsIcon, { size: 18 }), jsx("span", { className: styles$2.chatButtonText, children: "Settings" })] }), showSettingsMenu && settingsMenuContent] })), !shouldHoistActions && onUseTemplate && (jsxs("button", { className: classNames(styles$2.useTemplateButton), onClick: onUseTemplate, children: [jsx("span", { className: styles$2.chatButtonText, children: "Use this template" }), jsx(TemplateIcon, { size: 16 })] })), extraActions] })) : null;
|
|
13441
|
+
if (actionsContainer) {
|
|
13442
|
+
return actionsContent ? createPortal(actionsContent, actionsContainer) : null;
|
|
12235
13443
|
}
|
|
12236
|
-
|
|
12237
|
-
|
|
12238
|
-
//
|
|
12239
|
-
const
|
|
12240
|
-
|
|
12241
|
-
|
|
12242
|
-
|
|
12243
|
-
|
|
12244
|
-
|
|
12245
|
-
|
|
12246
|
-
|
|
12247
|
-
|
|
12248
|
-
|
|
12249
|
-
|
|
12250
|
-
|
|
12251
|
-
|
|
12252
|
-
|
|
12253
|
-
|
|
12254
|
-
|
|
12255
|
-
|
|
12256
|
-
|
|
12257
|
-
|
|
12258
|
-
|
|
12259
|
-
|
|
12260
|
-
|
|
12261
|
-
|
|
12262
|
-
|
|
12263
|
-
|
|
12264
|
-
|
|
12265
|
-
|
|
12266
|
-
|
|
12267
|
-
|
|
12268
|
-
|
|
12269
|
-
|
|
12270
|
-
|
|
12271
|
-
|
|
12272
|
-
|
|
12273
|
-
|
|
12274
|
-
|
|
12275
|
-
|
|
12276
|
-
|
|
12277
|
-
|
|
12278
|
-
|
|
12279
|
-
|
|
12280
|
-
|
|
12281
|
-
|
|
12282
|
-
|
|
12283
|
-
|
|
12284
|
-
|
|
12285
|
-
|
|
12286
|
-
|
|
12287
|
-
|
|
12288
|
-
|
|
12289
|
-
|
|
12290
|
-
|
|
12291
|
-
|
|
12292
|
-
|
|
12293
|
-
|
|
12294
|
-
|
|
12295
|
-
|
|
12296
|
-
|
|
12297
|
-
|
|
12298
|
-
|
|
12299
|
-
|
|
12300
|
-
|
|
12301
|
-
|
|
12302
|
-
|
|
12303
|
-
|
|
12304
|
-
|
|
12305
|
-
|
|
12306
|
-
|
|
12307
|
-
|
|
13444
|
+
return actionsContent;
|
|
13445
|
+
})(), shouldHoistActions && (showSaveMenu || showSettingsMenu) && (jsxs("div", { className: styles$2.hoistedMenus, children: [showSaveMenu && saveMenuContent, showSettingsMenu && settingsMenuContent] })), jsxs("div", { className: classNames(styles$2.chatMessages, useChatCssClassName('chatMessages'), (() => {
|
|
13446
|
+
// Detect if actions are present
|
|
13447
|
+
const hasActions = (!!onReset && postprocessedMessages.length !== 0) ||
|
|
13448
|
+
(isSaveButtonEnabled && postprocessedMessages.length !== 0) ||
|
|
13449
|
+
!!onUseTemplate ||
|
|
13450
|
+
!!extraActions;
|
|
13451
|
+
if (!hasActions) {
|
|
13452
|
+
return false;
|
|
13453
|
+
}
|
|
13454
|
+
// Detect if first message is long
|
|
13455
|
+
const firstMsg = postprocessedMessages[0];
|
|
13456
|
+
const firstMsgContent = (firstMsg === null || firstMsg === void 0 ? void 0 : firstMsg.content) || '';
|
|
13457
|
+
const firstMsgLines = firstMsgContent.split('\n').length; // <- TODO: Maybe use official counting functions here
|
|
13458
|
+
const firstMsgChars = firstMsgContent.length;
|
|
13459
|
+
const isFirstLong = firstMsgLines > 5 || firstMsgChars > 50;
|
|
13460
|
+
if (!isFirstLong) {
|
|
13461
|
+
return false;
|
|
13462
|
+
}
|
|
13463
|
+
return true;
|
|
13464
|
+
})() && styles$2.hasActionsAndFirstMessageIsLong), ref: chatMessagesRef, onScroll: handleScroll, children: [postprocessedMessages.map((message, i) => {
|
|
13465
|
+
const participant = participants.find((participant) => participant.name === message.sender);
|
|
13466
|
+
const isLastMessage = i === postprocessedMessages.length - 1;
|
|
13467
|
+
const isExpanded = expandedMessageId === message.id;
|
|
13468
|
+
const currentRating = messageRatings.get(message.id || message.content /* <-[💃] */) || 0;
|
|
13469
|
+
return (jsx(ChatMessageItem, { message: message, participant: participant, participants: participants, isLastMessage: isLastMessage, onMessage: onMessage ? sendMessageWithSound : undefined, setExpandedMessageId: setExpandedMessageId, isExpanded: isExpanded, currentRating: currentRating, handleRating: handleRating, mode: mode, isCopyButtonEnabled: isCopyButtonEnabled, isFeedbackEnabled: isFeedbackEnabled, onCopy: handleCopy, onCreateAgent: onCreateAgent, toolTitles: toolTitles, onToolCallClick: (toolCall) => {
|
|
13470
|
+
setSelectedToolCall(toolCall);
|
|
13471
|
+
setToolCallModalOpen(true);
|
|
13472
|
+
} }, message.id));
|
|
13473
|
+
}), jsx("div", {
|
|
13474
|
+
// Note: Extra space at bottom for input area
|
|
13475
|
+
style: { height: 100 } })] }), onMessage && (jsxs("div", { className: classNames(styles$2.chatInput, useChatCssClassName('chatInput'), isDragOver && styles$2.dragOver), ...(onFileUpload
|
|
13476
|
+
? {
|
|
13477
|
+
onDrop: handleDrop,
|
|
13478
|
+
onDragOver: handleDragOver,
|
|
13479
|
+
onDragLeave: handleDragLeave,
|
|
13480
|
+
}
|
|
13481
|
+
: {}), children: [uploadedFiles.length > 0 && (jsx("div", { className: styles$2.filePreviewContainer, children: uploadedFiles.map((uploadedFile) => (jsxs("div", { className: styles$2.filePreview, children: [jsx("div", { className: styles$2.fileIcon, children: "\uD83D\uDCCE" }), jsxs("div", { className: styles$2.fileInfo, children: [jsx("div", { className: styles$2.fileName, children: uploadedFile.file.name }), jsxs("div", { className: styles$2.fileSize, children: [(uploadedFile.file.size / 1024).toFixed(1), " KB"] })] }), jsx("button", { className: styles$2.removeFileButton, onClick: () => removeUploadedFile(uploadedFile.id), title: "Remove file", children: jsx(CloseIcon /* !!! size={12}*/, {}) })] }, uploadedFile.id))) })), (() => {
|
|
13482
|
+
var _a, _b, _c;
|
|
13483
|
+
// Note: Find the isMe participant or fallback to default
|
|
13484
|
+
const myColor = ((_a = participants.find((p) => p.isMe)) === null || _a === void 0 ? void 0 : _a.color) || USER_CHAT_COLOR;
|
|
13485
|
+
const inputBgColor = Color.from(myColor).then(lighten(0.4)).then(grayscale(0.7));
|
|
13486
|
+
const inputTextColor = inputBgColor.then(textColor);
|
|
13487
|
+
return (jsxs("div", { className: styles$2.inputContainer, style: {
|
|
13488
|
+
// Use a high-contrast placeholder color for visibility
|
|
13489
|
+
'--chat-placeholder-color': '#fff',
|
|
13490
|
+
// <- TODO: Remove
|
|
13491
|
+
'--input-bg-color': inputBgColor.toHex(),
|
|
13492
|
+
'--input-text-color': inputTextColor.toHex(),
|
|
13493
|
+
'--brand-color': buttonColor.toHex(),
|
|
13494
|
+
}, children: [jsx("textarea", { ref: (element) => {
|
|
13495
|
+
textareaRef.current = element;
|
|
13496
|
+
}, onPaste: handlePaste, style: {
|
|
13497
|
+
height: Math.max(countLines(((_b = textareaRef.current) === null || _b === void 0 ? void 0 : _b.value) || defaultMessage || ''), (((_c = textareaRef.current) === null || _c === void 0 ? void 0 : _c.value) || defaultMessage || '').split('\n')
|
|
13498
|
+
.length, 3) *
|
|
13499
|
+
25 +
|
|
13500
|
+
10,
|
|
13501
|
+
}, defaultValue: defaultMessage, placeholder: placeholderMessageContent || 'Write a message...', onKeyDown: (event) => {
|
|
13502
|
+
if (!onMessage) {
|
|
13503
|
+
return;
|
|
13504
|
+
}
|
|
13505
|
+
if (event.shiftKey) {
|
|
13506
|
+
return;
|
|
13507
|
+
}
|
|
13508
|
+
if (event.key !== 'Enter') {
|
|
13509
|
+
return;
|
|
13510
|
+
}
|
|
13511
|
+
event.preventDefault();
|
|
13512
|
+
/* not await */ handleSend();
|
|
13513
|
+
}, onKeyUp: () => {
|
|
13514
|
+
var _a;
|
|
13515
|
+
if (!onChange) {
|
|
13516
|
+
return;
|
|
13517
|
+
}
|
|
13518
|
+
onChange(((_a = textareaRef.current) === null || _a === void 0 ? void 0 : _a.value) || '');
|
|
13519
|
+
} }), onFileUpload && (jsxs(Fragment, { children: [jsx("input", { ref: fileInputRef, type: "file", multiple: true, style: { display: 'none' }, onChange: handleFileInputChange }), jsx("button", { type: "button", style: {
|
|
13520
|
+
backgroundColor: buttonColor.toHex(),
|
|
13521
|
+
color: buttonColor.then(textColor).toHex(),
|
|
13522
|
+
}, className: styles$2.attachmentButton, onClick: () => { var _a; return (_a = fileInputRef.current) === null || _a === void 0 ? void 0 : _a.click(); }, disabled: isUploading, title: "Attach file", children: jsx(AttachmentIcon, { size: 20 }) })] })), speechRecognition && (jsx("button", { "data-button-type": "voice", disabled: speechRecognitionState === 'STARTING' ||
|
|
13523
|
+
speechRecognitionState === 'TRANSCRIBING', style: {
|
|
13524
|
+
backgroundColor: (speechRecognitionState === 'RECORDING' ||
|
|
13525
|
+
speechRecognitionState === 'TRANSCRIBING'
|
|
13526
|
+
? Color.from('#ff4444')
|
|
13527
|
+
: buttonColor).toHex(),
|
|
13528
|
+
color: (speechRecognitionState === 'RECORDING' ||
|
|
13529
|
+
speechRecognitionState === 'TRANSCRIBING'
|
|
13530
|
+
? Color.from('#ffffff')
|
|
13531
|
+
: buttonColor.then(textColor)).toHex(),
|
|
13532
|
+
}, className: classNames(styles$2.voiceButton, (isVoiceCalling ||
|
|
13533
|
+
speechRecognitionState === 'RECORDING' ||
|
|
13534
|
+
speechRecognitionState === 'TRANSCRIBING') &&
|
|
13535
|
+
styles$2.voiceButtonActive), onClick: (event) => {
|
|
13536
|
+
event.preventDefault();
|
|
13537
|
+
handleToggleVoiceInput();
|
|
13538
|
+
}, title: speechRecognitionState === 'RECORDING'
|
|
13539
|
+
? 'Stop recording'
|
|
13540
|
+
: speechRecognitionState === 'TRANSCRIBING'
|
|
13541
|
+
? 'Transcribing...'
|
|
13542
|
+
: 'Start voice input', children: jsx(MicIcon, { size: 25 }) })), jsx("button", { "data-button-type": "call-to-action", "data-sound": "send", style: {
|
|
13543
|
+
backgroundColor: buttonColor.toHex(),
|
|
13544
|
+
color: buttonColor.then(textColor).toHex(),
|
|
13545
|
+
}, ref: buttonSendRef, onClick: (event) => {
|
|
13546
|
+
if (!onMessage) {
|
|
13547
|
+
return;
|
|
13548
|
+
}
|
|
13549
|
+
event.preventDefault();
|
|
13550
|
+
/* not await */ handleSend();
|
|
13551
|
+
}, children: jsx(SendIcon, { size: 25 }) })] }));
|
|
13552
|
+
})(), isUploading && (jsxs("div", { className: styles$2.uploadProgress, children: [jsx("div", { className: styles$2.uploadProgressBar, children: jsx("div", { className: styles$2.uploadProgressFill }) }), jsx("span", { children: "Uploading files..." })] })), isDragOver && onFileUpload && (jsx("div", { className: styles$2.dragOverlay, children: jsxs("div", { className: styles$2.dragOverlayContent, children: [jsx(AttachmentIcon, { size: 48 }), jsx("span", { children: "Drop files here to upload" })] }) }))] }))] })] }), toolCallModalOpen && selectedToolCall && (jsx("div", { className: styles$2.ratingModal, onClick: (e) => {
|
|
12308
13553
|
if (e.target === e.currentTarget) {
|
|
12309
13554
|
setToolCallModalOpen(false);
|
|
12310
13555
|
}
|
|
12311
|
-
}, children: jsxs("div", { className: classNames(
|
|
13556
|
+
}, children: jsxs("div", { className: classNames(styles$2.ratingModalContent, styles$2.toolCallModal), children: [(() => {
|
|
13557
|
+
var _a, _b;
|
|
12312
13558
|
const isSearch = selectedToolCall.name === 'web_search' ||
|
|
12313
13559
|
selectedToolCall.name === 'useSearchEngine' ||
|
|
12314
13560
|
selectedToolCall.name === 'search';
|
|
12315
13561
|
const isTime = selectedToolCall.name === 'get_current_time' || selectedToolCall.name === 'useTime';
|
|
12316
13562
|
const args = parseToolCallArguments(selectedToolCall);
|
|
12317
13563
|
const resultRaw = parseToolCallResult(selectedToolCall.result);
|
|
13564
|
+
const teamResult = parseTeamToolResult(resultRaw);
|
|
12318
13565
|
const toolCallDate = getToolCallTimestamp(selectedToolCall);
|
|
12319
13566
|
const { results, rawText } = extractSearchResults(resultRaw);
|
|
12320
13567
|
const hasResults = results.length > 0;
|
|
12321
13568
|
const hasRawText = !hasResults && !!rawText && rawText.trim().length > 0;
|
|
13569
|
+
if (teamResult === null || teamResult === void 0 ? void 0 : teamResult.teammate) {
|
|
13570
|
+
const teammateLabel = teamResult.teammate.label || teamResult.teammate.url || 'Teammate';
|
|
13571
|
+
const agentLabel = ((_b = (_a = teamResult.conversation) === null || _a === void 0 ? void 0 : _a.find((entry) => entry.sender === 'AGENT' || entry.role === 'AGENT')) === null || _b === void 0 ? void 0 : _b.name) || 'Agent';
|
|
13572
|
+
const baseTime = toolCallDate ? toolCallDate.getTime() : Date.now();
|
|
13573
|
+
const messages = (teamResult.conversation || [])
|
|
13574
|
+
.filter((entry) => entry && entry.content)
|
|
13575
|
+
.map((entry, index) => ({
|
|
13576
|
+
id: `team-${index}`,
|
|
13577
|
+
createdAt: new Date(baseTime + index * 1000),
|
|
13578
|
+
sender: entry.sender === 'TEAMMATE' || entry.role === 'TEAMMATE'
|
|
13579
|
+
? 'TEAMMATE'
|
|
13580
|
+
: 'AGENT',
|
|
13581
|
+
content: entry.content || '',
|
|
13582
|
+
isComplete: true,
|
|
13583
|
+
}));
|
|
13584
|
+
if (messages.length === 0) {
|
|
13585
|
+
if (teamResult.request) {
|
|
13586
|
+
messages.push({
|
|
13587
|
+
id: 'team-request',
|
|
13588
|
+
createdAt: new Date(baseTime),
|
|
13589
|
+
sender: 'AGENT',
|
|
13590
|
+
content: teamResult.request,
|
|
13591
|
+
isComplete: true,
|
|
13592
|
+
});
|
|
13593
|
+
}
|
|
13594
|
+
if (teamResult.response) {
|
|
13595
|
+
messages.push({
|
|
13596
|
+
id: 'team-response',
|
|
13597
|
+
createdAt: new Date(baseTime + 1000),
|
|
13598
|
+
sender: 'TEAMMATE',
|
|
13599
|
+
content: teamResult.response,
|
|
13600
|
+
isComplete: true,
|
|
13601
|
+
});
|
|
13602
|
+
}
|
|
13603
|
+
}
|
|
13604
|
+
const participants = [
|
|
13605
|
+
{
|
|
13606
|
+
name: 'AGENT',
|
|
13607
|
+
fullname: agentLabel,
|
|
13608
|
+
color: '#64748b',
|
|
13609
|
+
},
|
|
13610
|
+
{
|
|
13611
|
+
name: 'TEAMMATE',
|
|
13612
|
+
fullname: teammateLabel,
|
|
13613
|
+
color: '#0ea5e9',
|
|
13614
|
+
},
|
|
13615
|
+
];
|
|
13616
|
+
return (jsxs(Fragment, { children: [jsxs("div", { className: styles$2.searchModalHeader, children: [jsx("span", { className: styles$2.searchModalIcon, children: "\uD83E\uDD1D" }), jsx("h3", { className: styles$2.searchModalQuery, children: teammateLabel })] }), jsxs("div", { className: styles$2.searchModalContent, children: [messages.length > 0 ? (jsx("div", { className: styles$2.teamChatContainer, children: jsx(MockedChat, { title: `Chat with ${teammateLabel}`, messages: messages, participants: participants, isResettable: false, isPausable: false, isSaveButtonEnabled: false, isCopyButtonEnabled: false, visual: "STANDALONE" }) })) : (jsx("div", { className: styles$2.noResults, children: "No teammate conversation available." })), jsxs("div", { className: styles$2.toolCallDetails, children: [jsx("p", { children: jsx("strong", { children: "Teammate:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: jsx("pre", { className: styles$2.toolCallData, children: teamResult.teammate.url || teammateLabel }) }), teamResult.teammate.instructions && (jsxs(Fragment, { children: [jsx("p", { children: jsx("strong", { children: "When to consult:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: jsx("pre", { className: styles$2.toolCallData, children: teamResult.teammate.instructions }) })] })), teamResult.error && (jsxs(Fragment, { children: [jsx("p", { children: jsx("strong", { children: "Error:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: jsx("pre", { className: styles$2.toolCallData, children: teamResult.error }) })] }))] })] })] }));
|
|
13617
|
+
}
|
|
12322
13618
|
if (isSearch) {
|
|
12323
|
-
return (jsxs(Fragment, { children: [jsxs("div", { className:
|
|
13619
|
+
return (jsxs(Fragment, { children: [jsxs("div", { className: styles$2.searchModalHeader, children: [jsx("span", { className: styles$2.searchModalIcon, children: "\uD83D\uDD0E" }), jsx("h3", { className: styles$2.searchModalQuery, children: args.query || args.searchText || 'Search Results' })] }), jsx("div", { className: styles$2.searchModalContent, children: hasResults ? (jsx("div", { className: styles$2.searchResultsList, children: results.map((item, i) => (jsxs("div", { className: styles$2.searchResultItem, children: [jsx("div", { className: styles$2.searchResultUrl, children: item.url && (jsx("a", { href: item.url, target: "_blank", rel: "noreferrer", children: item.url })) }), jsx("h4", { className: styles$2.searchResultTitle, children: item.url ? (jsx("a", { href: item.url, target: "_blank", rel: "noreferrer", children: item.title || 'Untitled' })) : (item.title || 'Untitled') }), jsx("p", { className: styles$2.searchResultSnippet, children: item.snippet || item.content || '' })] }, i))) })) : hasRawText ? (jsx(MarkdownContent, { className: styles$2.searchResultsRaw, content: rawText })) : (jsx("div", { className: styles$2.noResults, children: resultRaw
|
|
12324
13620
|
? 'No search results found.'
|
|
12325
13621
|
: 'Search results are not available.' })) })] }));
|
|
12326
13622
|
}
|
|
@@ -12329,37 +13625,45 @@ function Chat(props) {
|
|
|
12329
13625
|
const displayDate = timeResultDate || toolCallDate;
|
|
12330
13626
|
const isValidDate = !!displayDate && !isNaN(displayDate.getTime());
|
|
12331
13627
|
const relativeLabel = toolCallDate ? `called ${moment(toolCallDate).fromNow()}` : null;
|
|
12332
|
-
return (jsxs(Fragment, { children: [jsxs("div", { className:
|
|
13628
|
+
return (jsxs(Fragment, { children: [jsxs("div", { className: styles$2.searchModalHeader, children: [jsx("span", { className: styles$2.searchModalIcon, children: "\uD83D\uDD52" }), jsx("h3", { className: styles$2.searchModalQuery, children: "Time at call" })] }), jsxs("div", { className: styles$2.searchModalContent, children: [jsxs("div", { style: {
|
|
12333
13629
|
display: 'flex',
|
|
12334
13630
|
flexDirection: 'column',
|
|
12335
13631
|
alignItems: 'center',
|
|
12336
13632
|
gap: '20px',
|
|
12337
13633
|
padding: '20px',
|
|
12338
|
-
}, children: [isValidDate && displayDate && jsx(ClockIcon, { date: displayDate, size: 150 }), jsxs("div", { style: { textAlign: 'center' }, children: [jsx("div", { style: { fontSize: '2em', fontWeight: 'bold' }, children: isValidDate && displayDate
|
|
13634
|
+
}, children: [isValidDate && displayDate && (jsx(ClockIcon, { date: displayDate, size: 150 })), jsxs("div", { style: { textAlign: 'center' }, children: [jsx("div", { style: { fontSize: '2em', fontWeight: 'bold' }, children: isValidDate && displayDate
|
|
12339
13635
|
? displayDate.toLocaleTimeString([], {
|
|
12340
13636
|
hour: '2-digit',
|
|
12341
13637
|
minute: '2-digit',
|
|
12342
13638
|
})
|
|
12343
13639
|
: 'Unknown time' }), jsx("div", { style: { color: '#666' }, children: isValidDate && displayDate
|
|
12344
13640
|
? displayDate.toLocaleDateString()
|
|
12345
|
-
: 'Unknown date' }), relativeLabel && (jsxs("div", { style: {
|
|
13641
|
+
: 'Unknown date' }), relativeLabel && (jsxs("div", { style: {
|
|
13642
|
+
fontSize: '0.9em',
|
|
13643
|
+
color: '#888',
|
|
13644
|
+
marginTop: '5px',
|
|
13645
|
+
}, children: ["(", relativeLabel, ")"] })), args.timezone && (jsxs("div", { style: {
|
|
13646
|
+
fontSize: '0.9em',
|
|
13647
|
+
color: '#888',
|
|
13648
|
+
marginTop: '5px',
|
|
13649
|
+
}, children: ["Timezone: ", args.timezone] }))] })] }), jsxs("div", { className: styles$2.toolCallDetails, children: [jsx("p", { children: jsx("strong", { children: "Timestamp of call:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: jsx("pre", { className: styles$2.toolCallData, children: toolCallDate ? toolCallDate.toLocaleString() : 'Unknown' }) })] })] })] }));
|
|
12346
13650
|
}
|
|
12347
13651
|
// Fallback for other tools
|
|
12348
|
-
return (jsxs(Fragment, { children: [jsxs("h3", { children: ["Tool Call: ", (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[selectedToolCall.name]) || selectedToolCall.name] }), jsxs("div", { className:
|
|
13652
|
+
return (jsxs(Fragment, { children: [jsxs("h3", { children: ["Tool Call: ", (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[selectedToolCall.name]) || selectedToolCall.name] }), jsxs("div", { className: styles$2.toolCallDetails, children: [jsx("p", { children: jsx("strong", { children: "Arguments:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: args && typeof args === 'object' ? (jsx("ul", { className: styles$2.toolCallArgsList, children: Object.entries(args).map(([key, value]) => (jsxs("li", { children: [jsxs("strong", { children: [key, ":"] }), ' ', typeof value === 'object'
|
|
12349
13653
|
? JSON.stringify(value)
|
|
12350
|
-
: String(value)] }, key))) })) : (jsx("pre", { className:
|
|
13654
|
+
: String(value)] }, key))) })) : (jsx("pre", { className: styles$2.toolCallData, children: String(args) })) }), jsx("p", { children: jsx("strong", { children: "Result:" }) }), jsx("div", { className: styles$2.toolCallDataContainer, children: jsx("pre", { className: styles$2.toolCallData, children: typeof resultRaw === 'object'
|
|
12351
13655
|
? JSON.stringify(resultRaw, null, 4)
|
|
12352
13656
|
: String(resultRaw) }) })] })] }));
|
|
12353
|
-
})(), jsx("div", { className:
|
|
13657
|
+
})(), jsx("div", { className: styles$2.ratingActions, children: jsx("button", { onClick: () => setToolCallModalOpen(false), children: "Close" }) })] }) })), ratingModalOpen && selectedMessage && (jsx("div", { className: styles$2.ratingModal, onClick: (e) => {
|
|
12354
13658
|
// Close modal when clicking backdrop on mobile
|
|
12355
13659
|
if (e.target === e.currentTarget && isMobile) {
|
|
12356
13660
|
setRatingModalOpen(false);
|
|
12357
13661
|
}
|
|
12358
|
-
}, children: jsxs("div", { className:
|
|
13662
|
+
}, children: jsxs("div", { className: styles$2.ratingModalContent, children: [jsx("h3", { children: "Rate this response" }), jsx("div", { className: styles$2.stars, children: [1, 2, 3, 4, 5].map((star) => (jsx("span", { onClick: () => setMessageRatings((previousRatings) => {
|
|
12359
13663
|
const nextRatings = new Map(previousRatings);
|
|
12360
13664
|
nextRatings.set(selectedMessage.id || selectedMessage.content /* <-[💃] */, star);
|
|
12361
13665
|
return nextRatings;
|
|
12362
|
-
}), onMouseEnter: () => setHoveredRating(star), onMouseLeave: () => setHoveredRating(0), className:
|
|
13666
|
+
}), onMouseEnter: () => setHoveredRating(star), onMouseLeave: () => setHoveredRating(0), className: styles$2.ratingModalStar, style: {
|
|
12363
13667
|
color: star <=
|
|
12364
13668
|
(hoveredRating ||
|
|
12365
13669
|
messageRatings.get(selectedMessage.id || selectedMessage.content /* <-[💃] */) ||
|
|
@@ -12384,13 +13688,13 @@ function Chat(props) {
|
|
|
12384
13688
|
}
|
|
12385
13689
|
}
|
|
12386
13690
|
return '';
|
|
12387
|
-
})(), className:
|
|
13691
|
+
})(), className: styles$2.ratingInput }), "Expected answer:", jsx("textarea", { placeholder: selectedMessage.content || 'Expected answer (optional)', defaultValue: selectedMessage.expectedAnswer || selectedMessage.content, onChange: (e) => {
|
|
12388
13692
|
if (selectedMessage) {
|
|
12389
13693
|
setSelectedMessage({ ...selectedMessage, expectedAnswer: e.target.value });
|
|
12390
13694
|
}
|
|
12391
|
-
}, className:
|
|
13695
|
+
}, className: styles$2.ratingInput }), "Note:", jsx("textarea", {
|
|
12392
13696
|
// Note: This is correctly mapped to `textRating` not a `note` which is universal column in lot of other tables and means the internal note
|
|
12393
|
-
placeholder: "Add a note (optional)", defaultValue: textRating, onChange: (e) => setTextRating(e.target.value), className:
|
|
13697
|
+
placeholder: "Add a note (optional)", defaultValue: textRating, onChange: (e) => setTextRating(e.target.value), className: styles$2.ratingInput }), jsxs("div", { className: styles$2.ratingActions, children: [jsx("button", { onClick: () => setRatingModalOpen(false), children: "Cancel" }), jsx("button", { onClick: submitRating, children: "Submit" })] })] }) }))] }));
|
|
12394
13698
|
}
|
|
12395
13699
|
|
|
12396
13700
|
/**
|
|
@@ -12440,681 +13744,352 @@ class ChatPersistence {
|
|
|
12440
13744
|
}
|
|
12441
13745
|
}
|
|
12442
13746
|
/**
|
|
12443
|
-
* Clear messages from localStorage for the given key
|
|
12444
|
-
*/
|
|
12445
|
-
static clearMessages(persistenceKey) {
|
|
12446
|
-
try {
|
|
12447
|
-
const storageKey = this.STORAGE_PREFIX + persistenceKey;
|
|
12448
|
-
localStorage.removeItem(storageKey);
|
|
12449
|
-
}
|
|
12450
|
-
catch (error) {
|
|
12451
|
-
console.warn('Failed to clear chat messages from localStorage:', error);
|
|
12452
|
-
}
|
|
12453
|
-
}
|
|
12454
|
-
/**
|
|
12455
|
-
* Check if localStorage is available
|
|
12456
|
-
*/
|
|
12457
|
-
static isAvailable() {
|
|
12458
|
-
try {
|
|
12459
|
-
const testKey = '__promptbook_storage_test__';
|
|
12460
|
-
localStorage.setItem(testKey, 'test');
|
|
12461
|
-
localStorage.removeItem(testKey);
|
|
12462
|
-
return true;
|
|
12463
|
-
}
|
|
12464
|
-
catch (_a) {
|
|
12465
|
-
return false;
|
|
12466
|
-
}
|
|
12467
|
-
}
|
|
12468
|
-
}
|
|
12469
|
-
ChatPersistence.STORAGE_PREFIX = 'promptbook_chat_';
|
|
12470
|
-
|
|
12471
|
-
/**
|
|
12472
|
-
* LlmChat component that provides chat functionality with LLM integration
|
|
12473
|
-
*
|
|
12474
|
-
* This component internally manages messages, participants, and task progress,
|
|
12475
|
-
* and uses the provided LLM tools to generate responses via `LlmExecutionTools.callChatModel`.
|
|
12476
|
-
*
|
|
12477
|
-
* Note: There are multiple chat components:
|
|
12478
|
-
* - `<Chat/>` renders chat as it is without any logic
|
|
12479
|
-
* - `<LlmChat/>` connected to LLM Execution Tools of Promptbook
|
|
12480
|
-
*
|
|
12481
|
-
* @public exported from `@promptbook/components`
|
|
12482
|
-
*/
|
|
12483
|
-
function LlmChat(props) {
|
|
12484
|
-
const { llmTools, persistenceKey, onChange, onReset, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, toolTitles, ...restProps } = props;
|
|
12485
|
-
// Internal state management
|
|
12486
|
-
// DRY: Single factory for seeding initial messages (used on mount and after reset)
|
|
12487
|
-
const buildInitialMessages = useCallback(() => initialMessages ? ([...initialMessages]) : ([]), [initialMessages]);
|
|
12488
|
-
const [messages, setMessages] = useState(() => buildInitialMessages());
|
|
12489
|
-
const [tasksProgress, setTasksProgress] = useState([]);
|
|
12490
|
-
const [isVoiceCalling] = useState(false);
|
|
12491
|
-
// Refs to keep latest state for long-lived handlers
|
|
12492
|
-
const messagesRef = useRef([]);
|
|
12493
|
-
const participantsRef = useRef([]);
|
|
12494
|
-
/**
|
|
12495
|
-
* Tracks whether the user (or system via persistence restoration) has interacted.
|
|
12496
|
-
* We do NOT persist purely initialMessages until the user sends something.
|
|
12497
|
-
*/
|
|
12498
|
-
const hasUserInteractedRef = useRef(false);
|
|
12499
|
-
// Load persisted messages on component mount
|
|
12500
|
-
useEffect(() => {
|
|
12501
|
-
if (persistenceKey && ChatPersistence.isAvailable()) {
|
|
12502
|
-
const persistedMessages = ChatPersistence.loadMessages(persistenceKey);
|
|
12503
|
-
if (persistedMessages.length > 0) {
|
|
12504
|
-
setMessages(persistedMessages);
|
|
12505
|
-
hasUserInteractedRef.current = true; // Persisted conversation exists; allow saving next changes
|
|
12506
|
-
// Notify about loaded messages
|
|
12507
|
-
if (onChange) {
|
|
12508
|
-
onChange(persistedMessages, participants);
|
|
12509
|
-
}
|
|
12510
|
-
}
|
|
12511
|
-
}
|
|
12512
|
-
}, [persistenceKey]); // Only depend on persistenceKey, not participants or onChange to avoid infinite loops
|
|
12513
|
-
// Save messages to localStorage whenever messages change (and persistence is enabled)
|
|
12514
|
-
useEffect(() => {
|
|
12515
|
-
if (persistenceKey && ChatPersistence.isAvailable() && messages.length > 0 && hasUserInteractedRef.current) {
|
|
12516
|
-
ChatPersistence.saveMessages(persistenceKey, messages);
|
|
12517
|
-
}
|
|
12518
|
-
}, [messages, persistenceKey]);
|
|
12519
|
-
// Generate participants from llmTools
|
|
12520
|
-
const participants = useMemo(() => props.participants || [
|
|
12521
|
-
{
|
|
12522
|
-
name: userParticipantName,
|
|
12523
|
-
fullname: 'You',
|
|
12524
|
-
isMe: true,
|
|
12525
|
-
color: '#1D4ED8',
|
|
12526
|
-
},
|
|
12527
|
-
// Use the profile from llmTools if available, otherwise fallback to default
|
|
12528
|
-
llmTools.profile || {
|
|
12529
|
-
name: llmParticipantName,
|
|
12530
|
-
fullname: llmTools.title || 'AI Assistant',
|
|
12531
|
-
color: '#10b981',
|
|
12532
|
-
},
|
|
12533
|
-
], [llmTools.profile, llmTools.title]);
|
|
12534
|
-
// Keep refs in sync for usage inside long-lived callbacks
|
|
12535
|
-
useEffect(() => {
|
|
12536
|
-
messagesRef.current = messages;
|
|
12537
|
-
}, [messages]);
|
|
12538
|
-
useEffect(() => {
|
|
12539
|
-
participantsRef.current = participants;
|
|
12540
|
-
}, [participants]);
|
|
12541
|
-
// Handle user messages and LLM responses
|
|
12542
|
-
const handleMessage = useCallback(async (messageContent, attachments = []) => {
|
|
12543
|
-
hasUserInteractedRef.current = true;
|
|
12544
|
-
// Add user message
|
|
12545
|
-
const userMessage = {
|
|
12546
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12547
|
-
id: `user_${Date.now()}`,
|
|
12548
|
-
createdAt: new Date(),
|
|
12549
|
-
sender: userParticipantName,
|
|
12550
|
-
content: messageContent,
|
|
12551
|
-
isComplete: true,
|
|
12552
|
-
attachments,
|
|
12553
|
-
};
|
|
12554
|
-
const newMessages = [...messages, userMessage];
|
|
12555
|
-
setMessages(newMessages);
|
|
12556
|
-
// Notify about changes
|
|
12557
|
-
if (onChange) {
|
|
12558
|
-
onChange(newMessages, participants);
|
|
12559
|
-
}
|
|
12560
|
-
// Add loading message for assistant
|
|
12561
|
-
const loadingMessage = {
|
|
12562
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12563
|
-
id: `assistant_${Date.now()}`,
|
|
12564
|
-
createdAt: new Date(),
|
|
12565
|
-
sender: llmParticipantName,
|
|
12566
|
-
content: 'Thinking...',
|
|
12567
|
-
isComplete: false,
|
|
12568
|
-
};
|
|
12569
|
-
const messagesWithLoading = [...newMessages, loadingMessage];
|
|
12570
|
-
setMessages(messagesWithLoading);
|
|
12571
|
-
// Notify about changes
|
|
12572
|
-
if (onChange) {
|
|
12573
|
-
onChange(newMessages, participants);
|
|
12574
|
-
}
|
|
12575
|
-
// Add task progress for LLM call
|
|
12576
|
-
const taskId = `llm_call_${Date.now()}`;
|
|
12577
|
-
setTasksProgress([{ id: taskId, name: 'Generating response...', progress: 0 }]);
|
|
12578
|
-
try {
|
|
12579
|
-
// Build thread: use props.thread if provided, otherwise use current messages
|
|
12580
|
-
const thread = props.thread ? [...props.thread] : [...newMessages];
|
|
12581
|
-
const prompt = {
|
|
12582
|
-
title: 'User Message',
|
|
12583
|
-
content: messageContent,
|
|
12584
|
-
parameters: {},
|
|
12585
|
-
modelRequirements: {
|
|
12586
|
-
modelVariant: 'CHAT',
|
|
12587
|
-
},
|
|
12588
|
-
thread,
|
|
12589
|
-
attachments,
|
|
12590
|
-
};
|
|
12591
|
-
// Update task progress
|
|
12592
|
-
setTasksProgress([{ id: taskId, name: 'Generating response...', progress: 50 }]);
|
|
12593
|
-
let result;
|
|
12594
|
-
if (llmTools.callChatModelStream) {
|
|
12595
|
-
result = await llmTools.callChatModelStream(prompt, (chunk) => {
|
|
12596
|
-
const assistantMessage = {
|
|
12597
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12598
|
-
id: loadingMessage.id,
|
|
12599
|
-
createdAt: new Date(),
|
|
12600
|
-
sender: llmParticipantName,
|
|
12601
|
-
content: chunk.content,
|
|
12602
|
-
isComplete: false,
|
|
12603
|
-
ongoingToolCalls: chunk.toolCalls,
|
|
12604
|
-
};
|
|
12605
|
-
const currentMessages = [...newMessages, assistantMessage];
|
|
12606
|
-
setMessages(currentMessages);
|
|
12607
|
-
if (onChange) {
|
|
12608
|
-
onChange(currentMessages, participants);
|
|
12609
|
-
}
|
|
12610
|
-
});
|
|
12611
|
-
}
|
|
12612
|
-
else if (llmTools.callChatModel) {
|
|
12613
|
-
result = await llmTools.callChatModel(prompt);
|
|
12614
|
-
}
|
|
12615
|
-
else {
|
|
12616
|
-
throw new Error('LLM tools do not support chat model calls');
|
|
12617
|
-
}
|
|
12618
|
-
// Update task progress to complete
|
|
12619
|
-
setTasksProgress([{ id: taskId, name: 'Response generated', progress: 100 }]);
|
|
12620
|
-
// Replace loading message with actual response
|
|
12621
|
-
const assistantMessage = {
|
|
12622
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12623
|
-
id: loadingMessage.id,
|
|
12624
|
-
createdAt: new Date(),
|
|
12625
|
-
sender: llmParticipantName,
|
|
12626
|
-
content: result.content,
|
|
12627
|
-
isComplete: true,
|
|
12628
|
-
toolCalls: result.toolCalls,
|
|
12629
|
-
completedToolCalls: result.toolCalls,
|
|
12630
|
-
};
|
|
12631
|
-
const finalMessages = [...newMessages, assistantMessage];
|
|
12632
|
-
setMessages(finalMessages);
|
|
12633
|
-
// Clear task progress after a short delay
|
|
12634
|
-
setTimeout(() => {
|
|
12635
|
-
setTasksProgress([]);
|
|
12636
|
-
}, 1000);
|
|
12637
|
-
// Notify about changes
|
|
12638
|
-
if (onChange) {
|
|
12639
|
-
onChange(finalMessages, participants);
|
|
12640
|
-
}
|
|
12641
|
-
}
|
|
12642
|
-
catch (error) {
|
|
12643
|
-
console.error('Error calling LLM:', error);
|
|
12644
|
-
// Replace loading message with error message
|
|
12645
|
-
const errorMessage = {
|
|
12646
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12647
|
-
id: loadingMessage.id,
|
|
12648
|
-
createdAt: new Date(),
|
|
12649
|
-
sender: llmParticipantName,
|
|
12650
|
-
content: `Sorry, I encountered an error: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
12651
|
-
isComplete: true,
|
|
12652
|
-
};
|
|
12653
|
-
const finalMessages = [...newMessages, errorMessage];
|
|
12654
|
-
setMessages(finalMessages);
|
|
12655
|
-
// Clear task progress
|
|
12656
|
-
setTasksProgress([]);
|
|
12657
|
-
// Notify about changes
|
|
12658
|
-
if (onChange) {
|
|
12659
|
-
onChange(finalMessages, participants);
|
|
12660
|
-
}
|
|
12661
|
-
}
|
|
12662
|
-
}, [messages, llmTools, onChange, participants]);
|
|
12663
|
-
// Handle chat reset
|
|
12664
|
-
const handleReset = useCallback(async () => {
|
|
12665
|
-
// Re-seed with initialMessages instead of empty array
|
|
12666
|
-
setMessages(buildInitialMessages());
|
|
12667
|
-
setTasksProgress([]);
|
|
12668
|
-
hasUserInteractedRef.current = false;
|
|
12669
|
-
// Clear persisted messages if persistence is enabled
|
|
12670
|
-
if (persistenceKey && ChatPersistence.isAvailable()) {
|
|
12671
|
-
ChatPersistence.clearMessages(persistenceKey);
|
|
12672
|
-
}
|
|
12673
|
-
if (onReset) {
|
|
12674
|
-
await onReset();
|
|
13747
|
+
* Clear messages from localStorage for the given key
|
|
13748
|
+
*/
|
|
13749
|
+
static clearMessages(persistenceKey) {
|
|
13750
|
+
try {
|
|
13751
|
+
const storageKey = this.STORAGE_PREFIX + persistenceKey;
|
|
13752
|
+
localStorage.removeItem(storageKey);
|
|
12675
13753
|
}
|
|
12676
|
-
|
|
12677
|
-
|
|
12678
|
-
onChange(buildInitialMessages(), participants);
|
|
13754
|
+
catch (error) {
|
|
13755
|
+
console.warn('Failed to clear chat messages from localStorage:', error);
|
|
12679
13756
|
}
|
|
12680
|
-
}
|
|
12681
|
-
|
|
12682
|
-
|
|
12683
|
-
|
|
12684
|
-
|
|
13757
|
+
}
|
|
13758
|
+
/**
|
|
13759
|
+
* Check if localStorage is available
|
|
13760
|
+
*/
|
|
13761
|
+
static isAvailable() {
|
|
13762
|
+
try {
|
|
13763
|
+
const testKey = '__promptbook_storage_test__';
|
|
13764
|
+
localStorage.setItem(testKey, 'test');
|
|
13765
|
+
localStorage.removeItem(testKey);
|
|
13766
|
+
return true;
|
|
12685
13767
|
}
|
|
12686
|
-
|
|
12687
|
-
|
|
12688
|
-
const hasAutoExecutedRef = useRef(false);
|
|
12689
|
-
useEffect(() => {
|
|
12690
|
-
if (autoExecuteMessage && !hasAutoExecutedRef.current) {
|
|
12691
|
-
hasAutoExecutedRef.current = true;
|
|
12692
|
-
handleMessage(autoExecuteMessage);
|
|
13768
|
+
catch (_a) {
|
|
13769
|
+
return false;
|
|
12693
13770
|
}
|
|
12694
|
-
}
|
|
12695
|
-
return (jsx(Fragment, { children: jsx(Chat, { ...restProps, messages, onReset, tasksProgress, participants, buttonColor, toolTitles, onMessage: handleMessage, onReset: handleReset, isVoiceCalling: isVoiceCalling }) }));
|
|
13771
|
+
}
|
|
12696
13772
|
}
|
|
13773
|
+
ChatPersistence.STORAGE_PREFIX = 'promptbook_chat_';
|
|
12697
13774
|
|
|
12698
13775
|
/**
|
|
12699
|
-
*
|
|
13776
|
+
* LlmChat component that provides chat functionality with LLM integration
|
|
12700
13777
|
*
|
|
12701
13778
|
* This component internally manages messages, participants, and task progress,
|
|
12702
13779
|
* and uses the provided LLM tools to generate responses via `LlmExecutionTools.callChatModel`.
|
|
12703
13780
|
*
|
|
12704
13781
|
* Note: There are multiple chat components:
|
|
12705
13782
|
* - `<Chat/>` renders chat as it is without any logic
|
|
12706
|
-
* - `<
|
|
12707
|
-
*
|
|
12708
|
-
* @public exported from `@promptbook/components`
|
|
12709
|
-
*/
|
|
12710
|
-
function AgentChat(props) {
|
|
12711
|
-
const { agent, title, persistenceKey, onChange, sendMessage, toolTitles, ...restProps } = props;
|
|
12712
|
-
const brandColor = Color.fromSafe(agent.meta.color || PROMPTBOOK_COLOR).then(saturate(-0.2));
|
|
12713
|
-
return (jsx(Fragment, { children: jsx(LlmChat, { title: title || `Chat with ${agent.meta.fullname || agent.agentName || 'Agent'}`, persistenceKey: persistenceKey || `agent-chat-${agent.agentName}`, userParticipantName: "USER", llmParticipantName: "AGENT" // <- TODO: [🧠] Maybe dynamic agent id
|
|
12714
|
-
, initialMessages: [
|
|
12715
|
-
{
|
|
12716
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
12717
|
-
sender: 'AGENT',
|
|
12718
|
-
content: agent.initialMessage ||
|
|
12719
|
-
spaceTrim$2(`
|
|
12720
|
-
|
|
12721
|
-
Hello! I am ${agent.meta.fullname || agent.agentName || 'an AI Agent'}.
|
|
12722
|
-
|
|
12723
|
-
[Hello](?message=Hello, can you tell me about yourself?)
|
|
12724
|
-
`),
|
|
12725
|
-
},
|
|
12726
|
-
], participants: [
|
|
12727
|
-
{
|
|
12728
|
-
name: 'AGENT',
|
|
12729
|
-
fullname: agent.meta.fullname || agent.agentName || 'Agent',
|
|
12730
|
-
avatarSrc: agent.meta.image,
|
|
12731
|
-
color: brandColor,
|
|
12732
|
-
isMe: false,
|
|
12733
|
-
agentSource: asUpdatableSubject(agent.agentSource).getValue() /* <- TODO: [🐱🚀] asValue */,
|
|
12734
|
-
},
|
|
12735
|
-
{
|
|
12736
|
-
name: 'USER',
|
|
12737
|
-
fullname: 'User',
|
|
12738
|
-
color: '#115EB6',
|
|
12739
|
-
isMe: true,
|
|
12740
|
-
},
|
|
12741
|
-
], buttonColor: brandColor, llmTools: agent, onChange, sendMessage, toolTitles, ...restProps }) }));
|
|
12742
|
-
}
|
|
12743
|
-
|
|
12744
|
-
/**
|
|
12745
|
-
* Hook to create a sendMessage function for an <LlmChat/> component WITHOUT needing any React Context.
|
|
12746
|
-
*
|
|
12747
|
-
* Usage pattern:
|
|
12748
|
-
* ```tsx
|
|
12749
|
-
* const sendMessage = useSendMessageToLlmChat();
|
|
12750
|
-
* return (
|
|
12751
|
-
* <>
|
|
12752
|
-
* <button onClick={() => sendMessage('Hello!')}>Hello</button>
|
|
12753
|
-
* <LlmChat llmTools={llmTools} sendMessage={sendMessage} />
|
|
12754
|
-
* </>
|
|
12755
|
-
* );
|
|
12756
|
-
* ```
|
|
12757
|
-
*
|
|
12758
|
-
* - No provider wrapping needed.
|
|
12759
|
-
* - Safe to call before the <LlmChat/> mounts (messages will be queued).
|
|
12760
|
-
* - Keeps DRY by letting <LlmChat/> reuse its internal `handleMessage` logic.
|
|
12761
|
-
*
|
|
12762
|
-
* @public exported from `@promptbook/components`
|
|
12763
|
-
*/
|
|
12764
|
-
function useSendMessageToLlmChat() {
|
|
12765
|
-
const ref = useRef(null);
|
|
12766
|
-
if (!ref.current) {
|
|
12767
|
-
let handler = null;
|
|
12768
|
-
const queue = [];
|
|
12769
|
-
const sendMessage = (message) => {
|
|
12770
|
-
if (handler) {
|
|
12771
|
-
// Fire and forget
|
|
12772
|
-
void handler(message);
|
|
12773
|
-
}
|
|
12774
|
-
else {
|
|
12775
|
-
queue.push(message);
|
|
12776
|
-
}
|
|
12777
|
-
};
|
|
12778
|
-
sendMessage._attach = (attachedHandler) => {
|
|
12779
|
-
handler = attachedHandler;
|
|
12780
|
-
// Flush queued messages
|
|
12781
|
-
while (queue.length > 0) {
|
|
12782
|
-
const next = queue.shift();
|
|
12783
|
-
void handler(next);
|
|
12784
|
-
}
|
|
12785
|
-
};
|
|
12786
|
-
ref.current = sendMessage;
|
|
12787
|
-
}
|
|
12788
|
-
return ref.current;
|
|
12789
|
-
}
|
|
12790
|
-
|
|
12791
|
-
/**
|
|
12792
|
-
* Normal flow of messages in the `MockedChat` component
|
|
12793
|
-
*
|
|
12794
|
-
* @public exported from `@promptbook/components`
|
|
12795
|
-
*/
|
|
12796
|
-
const NORMAL_FLOW = {
|
|
12797
|
-
beforeFirstMessage: 1000,
|
|
12798
|
-
thinkingBetweenMessages: 2000,
|
|
12799
|
-
waitAfterWord: 100,
|
|
12800
|
-
extraWordDelay: 50,
|
|
12801
|
-
longPauseChance: 0.2,
|
|
12802
|
-
longPauseDuration: [1200, 3500],
|
|
12803
|
-
};
|
|
12804
|
-
/**
|
|
12805
|
-
* Fast flow of messages in the `MockedChat` component
|
|
12806
|
-
*
|
|
12807
|
-
* @public exported from `@promptbook/components`
|
|
12808
|
-
*/
|
|
12809
|
-
const FAST_FLOW = {
|
|
12810
|
-
beforeFirstMessage: 300,
|
|
12811
|
-
thinkingBetweenMessages: 500,
|
|
12812
|
-
waitAfterWord: 20,
|
|
12813
|
-
extraWordDelay: 10,
|
|
12814
|
-
longPauseChance: 0.1,
|
|
12815
|
-
longPauseDuration: [400, 800],
|
|
12816
|
-
};
|
|
12817
|
-
/**
|
|
12818
|
-
* Slow flow of messages in the `MockedChat` component
|
|
12819
|
-
*
|
|
12820
|
-
* @public exported from `@promptbook/components`
|
|
12821
|
-
*/
|
|
12822
|
-
const SLOW_FLOW = {
|
|
12823
|
-
beforeFirstMessage: 2000,
|
|
12824
|
-
thinkingBetweenMessages: 4000,
|
|
12825
|
-
waitAfterWord: 300,
|
|
12826
|
-
extraWordDelay: 150,
|
|
12827
|
-
longPauseChance: 0.3,
|
|
12828
|
-
longPauseDuration: [2500, 5000],
|
|
12829
|
-
};
|
|
12830
|
-
/**
|
|
12831
|
-
* Blocky flow of messages in the `MockedChat` component
|
|
12832
|
-
*
|
|
12833
|
-
* Messages appear in blocks rather than word-by-word
|
|
12834
|
-
*
|
|
12835
|
-
* @public exported from `@promptbook/components`
|
|
12836
|
-
*/
|
|
12837
|
-
const BLOCKY_FLOW = {
|
|
12838
|
-
beforeFirstMessage: 1000,
|
|
12839
|
-
thinkingBetweenMessages: 2000,
|
|
12840
|
-
waitAfterWord: 0,
|
|
12841
|
-
extraWordDelay: 0,
|
|
12842
|
-
longPauseChance: 0.2,
|
|
12843
|
-
longPauseDuration: [1200, 3500],
|
|
12844
|
-
blocky: true,
|
|
12845
|
-
};
|
|
12846
|
-
/**
|
|
12847
|
-
* Random flow of messages in the `MockedChat` component
|
|
12848
|
-
*
|
|
12849
|
-
* @public exported from `@promptbook/components`
|
|
12850
|
-
*/
|
|
12851
|
-
const RANDOM_FLOW = {
|
|
12852
|
-
beforeFirstMessage: [500, 2000],
|
|
12853
|
-
thinkingBetweenMessages: [500, 3500],
|
|
12854
|
-
waitAfterWord: [10, 200],
|
|
12855
|
-
extraWordDelay: [0, 100],
|
|
12856
|
-
longPauseChance: 0.25,
|
|
12857
|
-
longPauseDuration: [1200, 5000],
|
|
12858
|
-
};
|
|
12859
|
-
/**
|
|
12860
|
-
* All predefined delay configurations for the `MockedChat` component
|
|
12861
|
-
*
|
|
12862
|
-
* @public exported from `@promptbook/components`
|
|
12863
|
-
*/
|
|
12864
|
-
const MOCKED_CHAT_DELAY_CONFIGS = {
|
|
12865
|
-
NORMAL_FLOW,
|
|
12866
|
-
FAST_FLOW,
|
|
12867
|
-
SLOW_FLOW,
|
|
12868
|
-
BLOCKY_FLOW,
|
|
12869
|
-
RANDOM_FLOW,
|
|
12870
|
-
};
|
|
12871
|
-
/**
|
|
12872
|
-
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
12873
|
-
*/
|
|
12874
|
-
|
|
12875
|
-
/**
|
|
12876
|
-
* Pause icon (two vertical bars)
|
|
12877
|
-
*
|
|
12878
|
-
* @public exported from `@promptbook/components`
|
|
12879
|
-
*/
|
|
12880
|
-
const PauseIcon = ({ size = 16 }) => (jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", role: "img", "aria-label": "Pause", children: jsx("path", { d: "M6 4h4v16H6V4zm8 0h4v16h-4V4z" }) }));
|
|
12881
|
-
|
|
12882
|
-
/**
|
|
12883
|
-
* Play icon (triangle) used for resume action
|
|
12884
|
-
*
|
|
12885
|
-
* @public exported from `@promptbook/components`
|
|
12886
|
-
*/
|
|
12887
|
-
const PlayIcon = ({ size = 16 }) => (jsx("svg", { width: size, height: size, viewBox: "0 0 24 24", fill: "currentColor", role: "img", "aria-label": "Play", children: jsx("path", { d: "M8 5v14l11-7L8 5z" }) }));
|
|
12888
|
-
|
|
12889
|
-
/**
|
|
12890
|
-
* MockedChat component that shows the same chat as Chat but emulates ongoing discussion
|
|
12891
|
-
* with realistic typing delays and thinking pauses.
|
|
13783
|
+
* - `<LlmChat/>` connected to LLM Execution Tools of Promptbook
|
|
12892
13784
|
*
|
|
12893
13785
|
* @public exported from `@promptbook/components`
|
|
12894
13786
|
*/
|
|
12895
|
-
function
|
|
12896
|
-
const {
|
|
12897
|
-
//
|
|
12898
|
-
|
|
12899
|
-
|
|
12900
|
-
|
|
12901
|
-
|
|
13787
|
+
function LlmChat(props) {
|
|
13788
|
+
const { llmTools, persistenceKey, onChange, onReset, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, toolTitles, ...restProps } = props;
|
|
13789
|
+
// Internal state management
|
|
13790
|
+
// DRY: Single factory for seeding initial messages (used on mount and after reset)
|
|
13791
|
+
const buildInitialMessages = useCallback(() => initialMessages ? ([...initialMessages]) : ([]), [initialMessages]);
|
|
13792
|
+
const [messages, setMessages] = useState(() => buildInitialMessages());
|
|
13793
|
+
const [tasksProgress, setTasksProgress] = useState([]);
|
|
13794
|
+
const [isVoiceCalling] = useState(false);
|
|
13795
|
+
// Refs to keep latest state for long-lived handlers
|
|
13796
|
+
const messagesRef = useRef([]);
|
|
13797
|
+
const participantsRef = useRef([]);
|
|
13798
|
+
/**
|
|
13799
|
+
* Tracks whether the user (or system via persistence restoration) has interacted.
|
|
13800
|
+
* We do NOT persist purely initialMessages until the user sends something.
|
|
13801
|
+
*/
|
|
13802
|
+
const hasUserInteractedRef = useRef(false);
|
|
13803
|
+
// Load persisted messages on component mount
|
|
13804
|
+
useEffect(() => {
|
|
13805
|
+
if (persistenceKey && ChatPersistence.isAvailable()) {
|
|
13806
|
+
const persistedMessages = ChatPersistence.loadMessages(persistenceKey);
|
|
13807
|
+
if (persistedMessages.length > 0) {
|
|
13808
|
+
setMessages(persistedMessages);
|
|
13809
|
+
hasUserInteractedRef.current = true; // Persisted conversation exists; allow saving next changes
|
|
13810
|
+
// Notify about loaded messages
|
|
13811
|
+
if (onChange) {
|
|
13812
|
+
onChange(persistedMessages, participants);
|
|
13813
|
+
}
|
|
13814
|
+
}
|
|
12902
13815
|
}
|
|
12903
|
-
|
|
12904
|
-
|
|
12905
|
-
|
|
12906
|
-
|
|
12907
|
-
|
|
12908
|
-
...MOCKED_CHAT_DELAY_CONFIGS.NORMAL_FLOW,
|
|
12909
|
-
...delayConfig,
|
|
12910
|
-
};
|
|
12911
|
-
const [displayedMessages, setDisplayedMessages] = useState([]);
|
|
12912
|
-
const [isSimulationComplete, setIsSimulationComplete] = useState(false);
|
|
12913
|
-
// Playback state machine
|
|
12914
|
-
// RUNNING -> (user clicks Pause) -> PAUSING (finish current message) -> PAUSED
|
|
12915
|
-
// PAUSED -> (user clicks Resume) -> RUNNING
|
|
12916
|
-
const [playbackState, setPlaybackState] = useState('RUNNING');
|
|
12917
|
-
const pauseRequestedRef = useRef(false);
|
|
12918
|
-
const [resetNonce, setResetNonce] = useState(0);
|
|
12919
|
-
const onReset = useMemo(() => {
|
|
12920
|
-
if (!isResettable) {
|
|
12921
|
-
return undefined;
|
|
13816
|
+
}, [persistenceKey]); // Only depend on persistenceKey, not participants or onChange to avoid infinite loops
|
|
13817
|
+
// Save messages to localStorage whenever messages change (and persistence is enabled)
|
|
13818
|
+
useEffect(() => {
|
|
13819
|
+
if (persistenceKey && ChatPersistence.isAvailable() && messages.length > 0 && hasUserInteractedRef.current) {
|
|
13820
|
+
ChatPersistence.saveMessages(persistenceKey, messages);
|
|
12922
13821
|
}
|
|
12923
|
-
|
|
12924
|
-
|
|
12925
|
-
|
|
12926
|
-
|
|
13822
|
+
}, [messages, persistenceKey]);
|
|
13823
|
+
// Generate participants from llmTools
|
|
13824
|
+
const participants = useMemo(() => props.participants || [
|
|
13825
|
+
{
|
|
13826
|
+
name: userParticipantName,
|
|
13827
|
+
fullname: 'You',
|
|
13828
|
+
isMe: true,
|
|
13829
|
+
color: '#1D4ED8',
|
|
13830
|
+
},
|
|
13831
|
+
// Use the profile from llmTools if available, otherwise fallback to default
|
|
13832
|
+
llmTools.profile || {
|
|
13833
|
+
name: llmParticipantName,
|
|
13834
|
+
fullname: llmTools.title || 'AI Assistant',
|
|
13835
|
+
color: '#10b981',
|
|
13836
|
+
},
|
|
13837
|
+
], [llmTools.profile, llmTools.title]);
|
|
13838
|
+
// Keep refs in sync for usage inside long-lived callbacks
|
|
13839
|
+
useEffect(() => {
|
|
13840
|
+
messagesRef.current = messages;
|
|
13841
|
+
}, [messages]);
|
|
13842
|
+
useEffect(() => {
|
|
13843
|
+
participantsRef.current = participants;
|
|
13844
|
+
}, [participants]);
|
|
13845
|
+
// Handle user messages and LLM responses
|
|
13846
|
+
const handleMessage = useCallback(async (messageContent, attachments = []) => {
|
|
13847
|
+
hasUserInteractedRef.current = true;
|
|
13848
|
+
// Add user message
|
|
13849
|
+
const userMessage = {
|
|
13850
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
13851
|
+
id: `user_${Date.now()}`,
|
|
13852
|
+
createdAt: new Date(),
|
|
13853
|
+
sender: userParticipantName,
|
|
13854
|
+
content: messageContent,
|
|
13855
|
+
isComplete: true,
|
|
13856
|
+
attachments,
|
|
12927
13857
|
};
|
|
12928
|
-
|
|
12929
|
-
|
|
12930
|
-
|
|
12931
|
-
if (
|
|
12932
|
-
|
|
12933
|
-
setPlaybackState('PAUSED');
|
|
12934
|
-
// Busy wait with small sleeps until resume
|
|
12935
|
-
while (pauseRequestedRef.current) {
|
|
12936
|
-
if (isCancelledRef())
|
|
12937
|
-
return;
|
|
12938
|
-
await forTime(100);
|
|
12939
|
-
}
|
|
12940
|
-
// Resumed
|
|
12941
|
-
setPlaybackState('RUNNING');
|
|
12942
|
-
};
|
|
12943
|
-
const requestPause = () => {
|
|
12944
|
-
if (playbackState === 'RUNNING') {
|
|
12945
|
-
pauseRequestedRef.current = true;
|
|
12946
|
-
// Will flip to PAUSING when current message completes
|
|
12947
|
-
setPlaybackState('PAUSING');
|
|
13858
|
+
const newMessages = [...messages, userMessage];
|
|
13859
|
+
setMessages(newMessages);
|
|
13860
|
+
// Notify about changes
|
|
13861
|
+
if (onChange) {
|
|
13862
|
+
onChange(newMessages, participants);
|
|
12948
13863
|
}
|
|
12949
|
-
|
|
12950
|
-
|
|
12951
|
-
|
|
12952
|
-
|
|
12953
|
-
|
|
12954
|
-
|
|
13864
|
+
// Add loading message for assistant
|
|
13865
|
+
const loadingMessage = {
|
|
13866
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
13867
|
+
id: `assistant_${Date.now()}`,
|
|
13868
|
+
createdAt: new Date(),
|
|
13869
|
+
sender: llmParticipantName,
|
|
13870
|
+
content: 'Thinking...',
|
|
13871
|
+
isComplete: false,
|
|
13872
|
+
};
|
|
13873
|
+
const messagesWithLoading = [...newMessages, loadingMessage];
|
|
13874
|
+
setMessages(messagesWithLoading);
|
|
13875
|
+
// Notify about changes
|
|
13876
|
+
if (onChange) {
|
|
13877
|
+
onChange(newMessages, participants);
|
|
12955
13878
|
}
|
|
12956
|
-
|
|
12957
|
-
|
|
12958
|
-
|
|
12959
|
-
|
|
12960
|
-
//
|
|
12961
|
-
|
|
12962
|
-
|
|
12963
|
-
|
|
12964
|
-
|
|
12965
|
-
|
|
12966
|
-
|
|
12967
|
-
|
|
12968
|
-
|
|
12969
|
-
|
|
12970
|
-
|
|
12971
|
-
}
|
|
12972
|
-
//
|
|
12973
|
-
|
|
12974
|
-
|
|
12975
|
-
|
|
12976
|
-
|
|
12977
|
-
|
|
12978
|
-
if (pauseRequestedRef.current) {
|
|
12979
|
-
await waitIfPaused(() => isCancelled);
|
|
12980
|
-
if (isCancelled)
|
|
12981
|
-
return;
|
|
12982
|
-
}
|
|
12983
|
-
if (isCancelled)
|
|
12984
|
-
return;
|
|
12985
|
-
const currentMessage = originalMessages[i];
|
|
12986
|
-
if (!currentMessage)
|
|
12987
|
-
continue;
|
|
12988
|
-
// Add thinking delay between messages (except for the first one)
|
|
12989
|
-
if (i > 0) {
|
|
12990
|
-
// Sometimes do a longer pause (agent switch or random)
|
|
12991
|
-
let didLongPause = false;
|
|
12992
|
-
if (delays.longPauseChance &&
|
|
12993
|
-
Math.random() < delays.longPauseChance &&
|
|
12994
|
-
i > 0 &&
|
|
12995
|
-
originalMessages[i].sender !== originalMessages[i - 1].sender) {
|
|
12996
|
-
await forTime(getDelay(delays.longPauseDuration, 2000));
|
|
12997
|
-
didLongPause = true;
|
|
12998
|
-
if (isCancelled)
|
|
12999
|
-
return;
|
|
13000
|
-
}
|
|
13001
|
-
// Otherwise normal thinking delay
|
|
13002
|
-
if (!didLongPause) {
|
|
13003
|
-
await forTime(getDelay(delays.thinkingBetweenMessages, 2000));
|
|
13004
|
-
if (isCancelled)
|
|
13005
|
-
return;
|
|
13006
|
-
}
|
|
13007
|
-
// Pause check (still between messages)
|
|
13008
|
-
if (pauseRequestedRef.current) {
|
|
13009
|
-
await waitIfPaused(() => isCancelled);
|
|
13010
|
-
if (isCancelled)
|
|
13011
|
-
return;
|
|
13012
|
-
}
|
|
13013
|
-
}
|
|
13014
|
-
// Show incomplete message first (for typing effect)
|
|
13015
|
-
const incompleteMessage = {
|
|
13016
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
13017
|
-
id: currentMessage.id,
|
|
13018
|
-
createdAt: currentMessage.createdAt,
|
|
13019
|
-
sender: currentMessage.sender,
|
|
13020
|
-
content: '',
|
|
13021
|
-
isComplete: false,
|
|
13022
|
-
expectedAnswer: currentMessage.expectedAnswer,
|
|
13023
|
-
isVoiceCall: currentMessage.isVoiceCall,
|
|
13024
|
-
};
|
|
13025
|
-
setDisplayedMessages((prev) => [...prev, incompleteMessage]);
|
|
13026
|
-
// Split message content into words
|
|
13027
|
-
const words = currentMessage.content.split(' ');
|
|
13028
|
-
let currentContent = '';
|
|
13029
|
-
// Type each word with delay (randomized)
|
|
13030
|
-
for (let wordIndex = 0; wordIndex < words.length; wordIndex++) {
|
|
13031
|
-
if (isCancelled)
|
|
13032
|
-
return;
|
|
13033
|
-
const word = words[wordIndex];
|
|
13034
|
-
currentContent += (wordIndex > 0 ? ' ' : '') + word;
|
|
13035
|
-
// Update the message with current content
|
|
13036
|
-
const updatingMessage = {
|
|
13879
|
+
// Add task progress for LLM call
|
|
13880
|
+
const taskId = `llm_call_${Date.now()}`;
|
|
13881
|
+
setTasksProgress([{ id: taskId, name: 'Generating response...', progress: 0 }]);
|
|
13882
|
+
try {
|
|
13883
|
+
// Build thread: use props.thread if provided, otherwise use current messages
|
|
13884
|
+
const thread = props.thread ? [...props.thread] : [...newMessages];
|
|
13885
|
+
const prompt = {
|
|
13886
|
+
title: 'User Message',
|
|
13887
|
+
content: messageContent,
|
|
13888
|
+
parameters: {},
|
|
13889
|
+
modelRequirements: {
|
|
13890
|
+
modelVariant: 'CHAT',
|
|
13891
|
+
},
|
|
13892
|
+
thread,
|
|
13893
|
+
attachments,
|
|
13894
|
+
};
|
|
13895
|
+
// Update task progress
|
|
13896
|
+
setTasksProgress([{ id: taskId, name: 'Generating response...', progress: 50 }]);
|
|
13897
|
+
let result;
|
|
13898
|
+
if (llmTools.callChatModelStream) {
|
|
13899
|
+
result = await llmTools.callChatModelStream(prompt, (chunk) => {
|
|
13900
|
+
const assistantMessage = {
|
|
13037
13901
|
// channel: 'PROMPTBOOK_CHAT',
|
|
13038
|
-
id:
|
|
13039
|
-
createdAt:
|
|
13040
|
-
sender:
|
|
13041
|
-
content:
|
|
13902
|
+
id: loadingMessage.id,
|
|
13903
|
+
createdAt: new Date(),
|
|
13904
|
+
sender: llmParticipantName,
|
|
13905
|
+
content: chunk.content,
|
|
13042
13906
|
isComplete: false,
|
|
13043
|
-
|
|
13044
|
-
isVoiceCall: currentMessage.isVoiceCall,
|
|
13907
|
+
ongoingToolCalls: chunk.toolCalls,
|
|
13045
13908
|
};
|
|
13046
|
-
|
|
13047
|
-
|
|
13048
|
-
|
|
13049
|
-
|
|
13050
|
-
}
|
|
13051
|
-
// Wait after word with extra delay (randomized)
|
|
13052
|
-
await forTime(getDelay(delays.waitAfterWord, 100) + getDelay(delays.extraWordDelay, 50));
|
|
13053
|
-
if (isCancelled)
|
|
13054
|
-
return;
|
|
13055
|
-
}
|
|
13056
|
-
// Mark message as complete
|
|
13057
|
-
const completeMessage = {
|
|
13058
|
-
// channel: 'PROMPTBOOK_CHAT',
|
|
13059
|
-
id: currentMessage.id,
|
|
13060
|
-
createdAt: currentMessage.createdAt,
|
|
13061
|
-
sender: currentMessage.sender,
|
|
13062
|
-
content: currentMessage.content,
|
|
13063
|
-
isComplete: true,
|
|
13064
|
-
expectedAnswer: currentMessage.expectedAnswer,
|
|
13065
|
-
isVoiceCall: currentMessage.isVoiceCall,
|
|
13066
|
-
};
|
|
13067
|
-
setDisplayedMessages((prev) => {
|
|
13068
|
-
const newMessages = [...prev];
|
|
13069
|
-
newMessages[newMessages.length - 1] = completeMessage;
|
|
13070
|
-
return newMessages;
|
|
13909
|
+
const currentMessages = [...newMessages, assistantMessage];
|
|
13910
|
+
setMessages(currentMessages);
|
|
13911
|
+
if (onChange) {
|
|
13912
|
+
onChange(currentMessages, participants);
|
|
13913
|
+
}
|
|
13071
13914
|
});
|
|
13072
|
-
// Small pause after completing the message
|
|
13073
|
-
await forTime(200);
|
|
13074
|
-
if (isCancelled)
|
|
13075
|
-
return;
|
|
13076
|
-
// Transition PAUSING -> PAUSED (after finishing current message)
|
|
13077
|
-
if (pauseRequestedRef.current && playbackState === 'PAUSING') ;
|
|
13078
13915
|
}
|
|
13079
|
-
|
|
13080
|
-
|
|
13081
|
-
simulateChat().catch((error) => {
|
|
13082
|
-
if (!isCancelled) {
|
|
13083
|
-
console.error('Error in MockedChat simulation:', error);
|
|
13084
|
-
// Fallback to showing all messages immediately
|
|
13085
|
-
setDisplayedMessages(originalMessages);
|
|
13086
|
-
setIsSimulationComplete(true);
|
|
13916
|
+
else if (llmTools.callChatModel) {
|
|
13917
|
+
result = await llmTools.callChatModel(prompt);
|
|
13087
13918
|
}
|
|
13088
|
-
|
|
13089
|
-
|
|
13090
|
-
isCancelled = true;
|
|
13091
|
-
};
|
|
13092
|
-
}, [
|
|
13093
|
-
originalMessages,
|
|
13094
|
-
delays.beforeFirstMessage,
|
|
13095
|
-
delays.thinkingBetweenMessages,
|
|
13096
|
-
delays.waitAfterWord,
|
|
13097
|
-
delays.extraWordDelay,
|
|
13098
|
-
resetNonce,
|
|
13099
|
-
]);
|
|
13100
|
-
// Build extra actions (Pause / Resume)
|
|
13101
|
-
const showPauseButton = isPausable && !isSimulationComplete;
|
|
13102
|
-
const extraActions = showPauseButton ? (jsxs("button", { className: `${chatStyles.chatButton} ${chatStyles.pauseButton} ${playbackState === 'PAUSING' ? chatStyles.pausing : playbackState === 'PAUSED' ? chatStyles.paused : ''}`, "aria-label": playbackState === 'RUNNING'
|
|
13103
|
-
? 'Pause simulation'
|
|
13104
|
-
: playbackState === 'PAUSING'
|
|
13105
|
-
? 'Pausing simulation'
|
|
13106
|
-
: 'Resume simulation', onClick: () => {
|
|
13107
|
-
if (playbackState === 'RUNNING') {
|
|
13108
|
-
requestPause();
|
|
13919
|
+
else {
|
|
13920
|
+
throw new Error('LLM tools do not support chat model calls');
|
|
13109
13921
|
}
|
|
13110
|
-
|
|
13111
|
-
|
|
13922
|
+
// Update task progress to complete
|
|
13923
|
+
setTasksProgress([{ id: taskId, name: 'Response generated', progress: 100 }]);
|
|
13924
|
+
// Replace loading message with actual response
|
|
13925
|
+
const assistantMessage = {
|
|
13926
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
13927
|
+
id: loadingMessage.id,
|
|
13928
|
+
createdAt: new Date(),
|
|
13929
|
+
sender: llmParticipantName,
|
|
13930
|
+
content: result.content,
|
|
13931
|
+
isComplete: true,
|
|
13932
|
+
toolCalls: result.toolCalls,
|
|
13933
|
+
completedToolCalls: result.toolCalls,
|
|
13934
|
+
};
|
|
13935
|
+
const finalMessages = [...newMessages, assistantMessage];
|
|
13936
|
+
setMessages(finalMessages);
|
|
13937
|
+
// Clear task progress after a short delay
|
|
13938
|
+
setTimeout(() => {
|
|
13939
|
+
setTasksProgress([]);
|
|
13940
|
+
}, 1000);
|
|
13941
|
+
// Notify about changes
|
|
13942
|
+
if (onChange) {
|
|
13943
|
+
onChange(finalMessages, participants);
|
|
13112
13944
|
}
|
|
13113
|
-
}
|
|
13114
|
-
|
|
13115
|
-
|
|
13116
|
-
|
|
13117
|
-
|
|
13945
|
+
}
|
|
13946
|
+
catch (error) {
|
|
13947
|
+
console.error('Error calling LLM:', error);
|
|
13948
|
+
// Replace loading message with error message
|
|
13949
|
+
const errorMessage = {
|
|
13950
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
13951
|
+
id: loadingMessage.id,
|
|
13952
|
+
createdAt: new Date(),
|
|
13953
|
+
sender: llmParticipantName,
|
|
13954
|
+
content: `Sorry, I encountered an error: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
13955
|
+
isComplete: true,
|
|
13956
|
+
};
|
|
13957
|
+
const finalMessages = [...newMessages, errorMessage];
|
|
13958
|
+
setMessages(finalMessages);
|
|
13959
|
+
// Clear task progress
|
|
13960
|
+
setTasksProgress([]);
|
|
13961
|
+
// Notify about changes
|
|
13962
|
+
if (onChange) {
|
|
13963
|
+
onChange(finalMessages, participants);
|
|
13964
|
+
}
|
|
13965
|
+
}
|
|
13966
|
+
}, [messages, llmTools, onChange, participants]);
|
|
13967
|
+
// Handle chat reset
|
|
13968
|
+
const handleReset = useCallback(async () => {
|
|
13969
|
+
// Re-seed with initialMessages instead of empty array
|
|
13970
|
+
setMessages(buildInitialMessages());
|
|
13971
|
+
setTasksProgress([]);
|
|
13972
|
+
hasUserInteractedRef.current = false;
|
|
13973
|
+
// Clear persisted messages if persistence is enabled
|
|
13974
|
+
if (persistenceKey && ChatPersistence.isAvailable()) {
|
|
13975
|
+
ChatPersistence.clearMessages(persistenceKey);
|
|
13976
|
+
}
|
|
13977
|
+
if (onReset) {
|
|
13978
|
+
await onReset();
|
|
13979
|
+
}
|
|
13980
|
+
// Notify about changes
|
|
13981
|
+
if (onChange) {
|
|
13982
|
+
onChange(buildInitialMessages(), participants);
|
|
13983
|
+
}
|
|
13984
|
+
}, [persistenceKey, onReset, onChange, participants, buildInitialMessages]);
|
|
13985
|
+
// Attach internal handler to external sendMessage (from useSendMessageToLlmChat) if provided
|
|
13986
|
+
useEffect(() => {
|
|
13987
|
+
if (sendMessage && sendMessage._attach) {
|
|
13988
|
+
sendMessage._attach(handleMessage);
|
|
13989
|
+
}
|
|
13990
|
+
}, [sendMessage, handleMessage]);
|
|
13991
|
+
// Handle autoExecuteMessage
|
|
13992
|
+
const hasAutoExecutedRef = useRef(false);
|
|
13993
|
+
useEffect(() => {
|
|
13994
|
+
if (autoExecuteMessage && !hasAutoExecutedRef.current) {
|
|
13995
|
+
hasAutoExecutedRef.current = true;
|
|
13996
|
+
handleMessage(autoExecuteMessage);
|
|
13997
|
+
}
|
|
13998
|
+
}, [autoExecuteMessage, handleMessage]);
|
|
13999
|
+
return (jsx(Fragment, { children: jsx(Chat, { ...restProps, messages, onReset, tasksProgress, participants, buttonColor, toolTitles, onMessage: handleMessage, onReset: handleReset, isVoiceCalling: isVoiceCalling }) }));
|
|
14000
|
+
}
|
|
14001
|
+
|
|
14002
|
+
/**
|
|
14003
|
+
* AgentChat component that provides chat functionality with LLM integration
|
|
14004
|
+
*
|
|
14005
|
+
* This component internally manages messages, participants, and task progress,
|
|
14006
|
+
* and uses the provided LLM tools to generate responses via `LlmExecutionTools.callChatModel`.
|
|
14007
|
+
*
|
|
14008
|
+
* Note: There are multiple chat components:
|
|
14009
|
+
* - `<Chat/>` renders chat as it is without any logic
|
|
14010
|
+
* - `<AgentChat/>` connected to LLM Execution Tools of Promptbook
|
|
14011
|
+
*
|
|
14012
|
+
* @public exported from `@promptbook/components`
|
|
14013
|
+
*/
|
|
14014
|
+
function AgentChat(props) {
|
|
14015
|
+
const { agent, title, persistenceKey, onChange, sendMessage, toolTitles, ...restProps } = props;
|
|
14016
|
+
const brandColor = Color.fromSafe(agent.meta.color || PROMPTBOOK_COLOR).then(saturate(-0.2));
|
|
14017
|
+
return (jsx(Fragment, { children: jsx(LlmChat, { title: title || `Chat with ${agent.meta.fullname || agent.agentName || 'Agent'}`, persistenceKey: persistenceKey || `agent-chat-${agent.agentName}`, userParticipantName: "USER", llmParticipantName: "AGENT" // <- TODO: [🧠] Maybe dynamic agent id
|
|
14018
|
+
, initialMessages: [
|
|
14019
|
+
{
|
|
14020
|
+
// channel: 'PROMPTBOOK_CHAT',
|
|
14021
|
+
sender: 'AGENT',
|
|
14022
|
+
content: agent.initialMessage ||
|
|
14023
|
+
spaceTrim$2(`
|
|
14024
|
+
|
|
14025
|
+
Hello! I am ${agent.meta.fullname || agent.agentName || 'an AI Agent'}.
|
|
14026
|
+
|
|
14027
|
+
[Hello](?message=Hello, can you tell me about yourself?)
|
|
14028
|
+
`),
|
|
14029
|
+
},
|
|
14030
|
+
], participants: [
|
|
14031
|
+
{
|
|
14032
|
+
name: 'AGENT',
|
|
14033
|
+
fullname: agent.meta.fullname || agent.agentName || 'Agent',
|
|
14034
|
+
avatarSrc: agent.meta.image,
|
|
14035
|
+
color: brandColor,
|
|
14036
|
+
isMe: false,
|
|
14037
|
+
agentSource: asUpdatableSubject(agent.agentSource).getValue() /* <- TODO: [🐱🚀] asValue */,
|
|
14038
|
+
},
|
|
14039
|
+
{
|
|
14040
|
+
name: 'USER',
|
|
14041
|
+
fullname: 'User',
|
|
14042
|
+
color: '#115EB6',
|
|
14043
|
+
isMe: true,
|
|
14044
|
+
},
|
|
14045
|
+
], buttonColor: brandColor, llmTools: agent, onChange, sendMessage, toolTitles, ...restProps }) }));
|
|
14046
|
+
}
|
|
14047
|
+
|
|
14048
|
+
/**
|
|
14049
|
+
* Hook to create a sendMessage function for an <LlmChat/> component WITHOUT needing any React Context.
|
|
14050
|
+
*
|
|
14051
|
+
* Usage pattern:
|
|
14052
|
+
* ```tsx
|
|
14053
|
+
* const sendMessage = useSendMessageToLlmChat();
|
|
14054
|
+
* return (
|
|
14055
|
+
* <>
|
|
14056
|
+
* <button onClick={() => sendMessage('Hello!')}>Hello</button>
|
|
14057
|
+
* <LlmChat llmTools={llmTools} sendMessage={sendMessage} />
|
|
14058
|
+
* </>
|
|
14059
|
+
* );
|
|
14060
|
+
* ```
|
|
14061
|
+
*
|
|
14062
|
+
* - No provider wrapping needed.
|
|
14063
|
+
* - Safe to call before the <LlmChat/> mounts (messages will be queued).
|
|
14064
|
+
* - Keeps DRY by letting <LlmChat/> reuse its internal `handleMessage` logic.
|
|
14065
|
+
*
|
|
14066
|
+
* @public exported from `@promptbook/components`
|
|
14067
|
+
*/
|
|
14068
|
+
function useSendMessageToLlmChat() {
|
|
14069
|
+
const ref = useRef(null);
|
|
14070
|
+
if (!ref.current) {
|
|
14071
|
+
let handler = null;
|
|
14072
|
+
const queue = [];
|
|
14073
|
+
const sendMessage = (message) => {
|
|
14074
|
+
if (handler) {
|
|
14075
|
+
// Fire and forget
|
|
14076
|
+
void handler(message);
|
|
14077
|
+
}
|
|
14078
|
+
else {
|
|
14079
|
+
queue.push(message);
|
|
14080
|
+
}
|
|
14081
|
+
};
|
|
14082
|
+
sendMessage._attach = (attachedHandler) => {
|
|
14083
|
+
handler = attachedHandler;
|
|
14084
|
+
// Flush queued messages
|
|
14085
|
+
while (queue.length > 0) {
|
|
14086
|
+
const next = queue.shift();
|
|
14087
|
+
void handler(next);
|
|
14088
|
+
}
|
|
14089
|
+
};
|
|
14090
|
+
ref.current = sendMessage;
|
|
14091
|
+
}
|
|
14092
|
+
return ref.current;
|
|
13118
14093
|
}
|
|
13119
14094
|
|
|
13120
14095
|
/**
|
|
@@ -14844,37 +15819,6 @@ function mimeTypeToExtension(value) {
|
|
|
14844
15819
|
return extension(value) || null;
|
|
14845
15820
|
}
|
|
14846
15821
|
|
|
14847
|
-
/**
|
|
14848
|
-
* The built-in `fetch' function with a lightweight error handling wrapper as default fetch function used in Promptbook scrapers
|
|
14849
|
-
*
|
|
14850
|
-
* @public exported from `@promptbook/core`
|
|
14851
|
-
*/
|
|
14852
|
-
const promptbookFetch = async (urlOrRequest, init) => {
|
|
14853
|
-
try {
|
|
14854
|
-
return await fetch(urlOrRequest, init);
|
|
14855
|
-
}
|
|
14856
|
-
catch (error) {
|
|
14857
|
-
assertsError(error);
|
|
14858
|
-
let url;
|
|
14859
|
-
if (typeof urlOrRequest === 'string') {
|
|
14860
|
-
url = urlOrRequest;
|
|
14861
|
-
}
|
|
14862
|
-
else if (urlOrRequest instanceof Request) {
|
|
14863
|
-
url = urlOrRequest.url;
|
|
14864
|
-
}
|
|
14865
|
-
throw new PromptbookFetchError(spaceTrim$2((block) => `
|
|
14866
|
-
Can not fetch "${url}"
|
|
14867
|
-
|
|
14868
|
-
Fetch error:
|
|
14869
|
-
${block(error.message)}
|
|
14870
|
-
|
|
14871
|
-
`));
|
|
14872
|
-
}
|
|
14873
|
-
};
|
|
14874
|
-
/**
|
|
14875
|
-
* TODO: [🧠] Maybe rename because it is not used only for scrapers but also in `$getCompiledBook`
|
|
14876
|
-
*/
|
|
14877
|
-
|
|
14878
15822
|
/**
|
|
14879
15823
|
* Factory function that creates a handler for processing knowledge sources.
|
|
14880
15824
|
* Provides standardized processing of different types of knowledge sources
|
|
@@ -20180,6 +21124,7 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
|
|
|
20180
21124
|
*/
|
|
20181
21125
|
const DISCRIMINANT = 'OPEN_AI_ASSISTANT_V1';
|
|
20182
21126
|
/**
|
|
21127
|
+
* TODO: !!!!! [✨🥚] Knowledge should work both with and without scrapers
|
|
20183
21128
|
* TODO: [🙎] In `OpenAiAssistantExecutionTools` Allow to create abstract assistants with `isCreatingNewAssistantsAllowed`
|
|
20184
21129
|
* TODO: [🧠][🧙♂️] Maybe there can be some wizard for those who want to use just OpenAI
|
|
20185
21130
|
* TODO: Maybe make custom OpenAiError
|
|
@@ -20926,12 +21871,15 @@ class RemoteAgent extends Agent {
|
|
|
20926
21871
|
doneReading = !!done;
|
|
20927
21872
|
if (value) {
|
|
20928
21873
|
const textChunk = decoder.decode(value, { stream: true });
|
|
20929
|
-
let
|
|
20930
|
-
|
|
20931
|
-
|
|
20932
|
-
|
|
20933
|
-
|
|
20934
|
-
|
|
21874
|
+
let sawToolCalls = false;
|
|
21875
|
+
let hasNonEmptyText = false;
|
|
21876
|
+
const textLines = [];
|
|
21877
|
+
const lines = textChunk.split('\n');
|
|
21878
|
+
for (const line of lines) {
|
|
21879
|
+
const trimmedLine = line.trim();
|
|
21880
|
+
let isToolCallLine = false;
|
|
21881
|
+
if (trimmedLine.startsWith('{') && trimmedLine.endsWith('}')) {
|
|
21882
|
+
try {
|
|
20935
21883
|
const chunk = JSON.parse(trimmedLine);
|
|
20936
21884
|
if (chunk.toolCalls) {
|
|
20937
21885
|
const normalizedToolCalls = chunk.toolCalls.map(normalizeToolCall);
|
|
@@ -20946,19 +21894,32 @@ class RemoteAgent extends Agent {
|
|
|
20946
21894
|
rawResponse: {},
|
|
20947
21895
|
toolCalls: normalizedToolCalls,
|
|
20948
21896
|
});
|
|
20949
|
-
|
|
21897
|
+
sawToolCalls = true;
|
|
21898
|
+
isToolCallLine = true;
|
|
20950
21899
|
}
|
|
20951
21900
|
}
|
|
21901
|
+
catch (error) {
|
|
21902
|
+
// Ignore non-json lines
|
|
21903
|
+
}
|
|
21904
|
+
}
|
|
21905
|
+
if (!isToolCallLine) {
|
|
21906
|
+
textLines.push(line);
|
|
21907
|
+
if (line.length > 0) {
|
|
21908
|
+
hasNonEmptyText = true;
|
|
21909
|
+
}
|
|
20952
21910
|
}
|
|
20953
21911
|
}
|
|
20954
|
-
|
|
20955
|
-
|
|
21912
|
+
if (sawToolCalls) {
|
|
21913
|
+
if (!hasNonEmptyText) {
|
|
21914
|
+
continue;
|
|
21915
|
+
}
|
|
21916
|
+
const textChunkWithoutToolCalls = textLines.join('\n');
|
|
21917
|
+
content += textChunkWithoutToolCalls;
|
|
20956
21918
|
}
|
|
20957
|
-
|
|
20958
|
-
|
|
21919
|
+
else {
|
|
21920
|
+
// console.debug('RemoteAgent chunk:', textChunk);
|
|
21921
|
+
content += textChunk;
|
|
20959
21922
|
}
|
|
20960
|
-
// console.debug('RemoteAgent chunk:', textChunk);
|
|
20961
|
-
content += textChunk;
|
|
20962
21923
|
onProgress({
|
|
20963
21924
|
content,
|
|
20964
21925
|
modelName: this.modelName,
|