@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 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-15';
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$9 = {"AvatarChip":"AvatarChip-module_AvatarChip__4sA0u","Avatar":"AvatarChip-module_Avatar__mN2sc","TemplateLabel":"AvatarChip-module_TemplateLabel__-7vVI"};
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$9.AvatarChip, className, isSelected ? styles$9.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$9.Avatar }), meta.fullname || agentName, isTemplate && jsx("span", { className: styles$9.TemplateLabel, children: "Template" })] }));
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$8 = {"scrim":"Modal-module_scrim__jKO-A","Modal":"Modal-module_Modal__k49dY","closeButton":"Modal-module_closeButton__dzf6l"};
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$8.scrim, onClick: onClose, children: jsxs("div", { className: classNames(styles$8.Modal, className), onClick: (e) => e.stopPropagation(), children: [jsx("button", { className: styles$8.closeButton, onClick: onClose, children: "\u00D7" }), children] }) }), document.body);
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$7 = {"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"};
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$6 = {"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"};
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$6.CodeBlockHeader, children: [jsx("span", { children: language }), jsxs("div", { className: styles$6.CodeBlockButtons, children: [jsx("button", { onClick: handleCopy, className: styles$6.CopyButton, title: "Copy to clipboard", children: copied ? 'Copied!' : 'Copy' }), jsx("button", { onClick: handleDownload, className: styles$6.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$6.CreateAgentButton, title: "Create agent from this book", children: "Create Agent" }))] })] })) : null;
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$6.CodeBlock, className), children: [header, jsx(BookEditor, { value: code, isReadonly: true, height: lines * 25 /* <- [🧠] A bit more than 19px to accommodate BookEditor lines */ })] }));
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$6.CodeBlock, className), children: [header, jsx(Editor, { height: `${height}px`, language: language || 'plaintext', value: code, theme: "vs-dark", options: {
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$5 = {"MarkdownContent":"MarkdownContent-module_MarkdownContent__2JuyX","chat-code-block":"MarkdownContent-module_chat-code-block__ZffFg","citation":"MarkdownContent-module_citation__11SMw"};
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$5.citation}">${content}</span>`;
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$5.MarkdownContent, className), dangerouslySetInnerHTML: {
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$4 = {"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"};
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$4.MenuHamburger, className), onClick: onClick, children: jsxs("div", { className: classNames(styles$4.MenuHamburgerInner, isOpen && styles$4.open), children: [jsx("div", { className: classNames(styles$4.bar, styles$4.bar1) }), jsx("div", { className: classNames(styles$4.bar, styles$4.bar2) }), jsx("div", { className: classNames(styles$4.bar, styles$4.bar3) })] }) }));
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$3 = {"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"};
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$3.dropdown, children: [jsx(HamburgerMenu, { className: classNames(styles$3.button, isOpen && styles$3.isOpen), isOpen: isOpen, onClick: () => setIsOpen(!isOpen) }), isOpen && (jsx("div", { className: styles$3.menu, children: actions.map(({ icon, name, onClick }) => (jsxs("button", { className: classNames(styles$3.button, styles$3.menuItem), onClick: () => {
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$7.bookEditorActionbar, children: jsx("button", { className: styles$7.button, onClick: onFullscreenClick, children: jsx(ExitFullscreenIcon, {}) }) }));
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$7.bookEditorActionbar, children: [actions.length >= 2 ? (jsx(Dropdown, { actions: actions })) : (actions.map(({ icon, name, onClick }) => (jsx("button", { className: styles$7.button, onClick: onClick, children: icon }, name)))), isAboutModalOpen && (jsx(Modal, { onClose: () => {
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$7.bookEditorContainer, instanceClass), onDrop: handleDrop, onPaste: handlePaste, onDragOver: handleDragOver, onDragEnter: handleDragEnter, onDragLeave: handleDragLeave, children: [(isUploadButtonShown ||
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$7.dropOverlay, children: "Drop files to upload" }), isSavedShown && (jsxs("div", { className: styles$7.savedNotification, children: [jsx(SaveIcon, {}), "Saved"] })), jsxs("div", { style: {
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$7.loading, children: "\uD83D\uDCD6" }) })] })] }));
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$7.BookEditor, isVerbose && styles$7.isVerbose, styles$7.bookEditorWrapper, isBorderRadiusDisabled && styles$7.isBorderRadiusDisabled, isFullscreen && styles$7.fullscreen, className), style: isFullscreen
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$2 = {"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"};
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$2.AvatarProfile, className), style: style, children: [jsx("img", { src: meta.image, alt: agentName || '', className: styles$2.Avatar }), jsxs("div", { className: styles$2.AgentInfo, children: [jsx("h2", { className: styles$2.AgentName, children: meta.fullname || agentName || 'Agent' }), jsx("p", { className: styles$2.AgentDescription, children: personaDescription }), agentSource !== undefined && (jsx("button", { className: styles$2.viewSourceButton, onClick: (event) => {
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.map((part) => part.trim()).filter(Boolean).join(' ');
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":"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","file":"Chat.module.css","sourcesContent":["@font-face {\n    font-family: 'OpenMojiBlack';\n    src: url('https://s6.ptbk.io/fonts/OpenMoji-black-glyf.woff2') format('woff2'); /* <- TODO: [🐱‍🚀] Dynamically load from /servers.ts */\n    unicode-range: U+23, U+2A, U+2D, U+30-39, U+A9, U+AE, U+200D, U+203C, U+2049, U+20E3, U+2117, U+2120, U+2122, U+2139,\n        U+2194-2199, U+21A9, U+21AA, U+229C, U+231A, U+231B, U+2328, U+23CF, U+23E9-23F3, U+23F8-23FE, U+24C2, U+25A1,\n        U+25AA-25AE, U+25B6, U+25C0, U+25C9, U+25D0, U+25D1, U+25E7-25EA, U+25ED, U+25EE, U+25FB-25FE, U+2600-2605,\n        U+260E, U+2611, U+2614, U+2615, U+2618, U+261D, U+2620, U+2622, U+2623, U+2626, U+262A, U+262E, U+262F,\n        U+2638-263A, U+2640, U+2642, U+2648-2653, U+265F, U+2660, U+2663, U+2665, U+2666, U+2668, U+267B, U+267E, U+267F,\n        U+2691-2697, U+2699, U+269B, U+269C, U+26A0, U+26A1, U+26A7, U+26AA, U+26AB, U+26B0, U+26B1, U+26BD, U+26BE,\n        U+26C4, U+26C5, U+26C8, U+26CE, U+26CF, U+26D1, U+26D3, U+26D4, U+26E9, U+26EA, U+26F0-26F5, U+26F7-26FA, U+26FD,\n        U+2702, U+2705, U+2708-270D, U+270F, U+2712, U+2714, U+2716, U+271D, U+2721, U+2728, U+2733, U+2734, U+2744,\n        U+2747, U+274C, U+274E, U+2753-2755, U+2757, U+2763, U+2764, U+2795-2797, U+27A1, U+27B0, U+27BF, U+2934, U+2935,\n        U+2B05-2B07, U+2B0C, U+2B0D, U+2B1B, U+2B1C, U+2B1F-2B24, U+2B2E, U+2B2F, U+2B50, U+2B55, U+2B58, U+2B8F,\n        U+2BBA-2BBC, U+2BC3, U+2BC4, U+2BEA, U+2BEB, U+3030, U+303D, U+3297, U+3299, U+E000-E009, U+E010, U+E011,\n        U+E040-E06D, U+E080-E0B4, U+E0C0-E0CC, U+E0FF-E10D, U+E140-E14A, U+E150-E157, U+E181-E189, U+E1C0-E1C4,\n        U+E1C6-E1D9, U+E200-E216, U+E240-E269, U+E280-E283, U+E2C0-E2C4, U+E2C6-E2DA, U+E300-E303, U+E305-E30F,\n        U+E312-E316, U+E318-E322, U+E324-E329, U+E32B, U+E340-E348, U+E380, U+E381, U+F000, U+F77A, U+F8FF, U+FE0F,\n        U+1F004, U+1F0CF, U+1F10D-1F10F, U+1F12F, U+1F16D-1F171, U+1F17E, U+1F17F, U+1F18E, U+1F191-1F19A, U+1F1E6-1F1FF,\n        U+1F201, U+1F202, U+1F21A, U+1F22F, U+1F232-1F23A, U+1F250, U+1F251, U+1F260-1F265, U+1F300-1F321, U+1F324-1F393,\n        U+1F396, U+1F397, U+1F399-1F39B, U+1F39E-1F3F0, U+1F3F3-1F3F5, U+1F3F7-1F4FD, U+1F4FF-1F53D, U+1F549-1F54E,\n        U+1F550-1F567, U+1F56F, U+1F570, U+1F573-1F57A, U+1F587, U+1F58A-1F58D, U+1F590, U+1F595, U+1F596, U+1F5A4,\n        U+1F5A5, U+1F5A8, U+1F5B1, U+1F5B2, U+1F5BC, U+1F5C2-1F5C4, U+1F5D1-1F5D3, U+1F5DC-1F5DE, U+1F5E1, U+1F5E3,\n        U+1F5E8, U+1F5EF, U+1F5F3, U+1F5FA-1F64F, U+1F680-1F6C5, U+1F6CB-1F6D2, U+1F6D5-1F6D7, U+1F6DC-1F6E5, U+1F6E9,\n        U+1F6EB, U+1F6EC, U+1F6F0, U+1F6F3-1F6FC, U+1F7E0-1F7EB, U+1F7F0, U+1F90C-1F93A, U+1F93C-1F945, U+1F947-1F9FF,\n        U+1FA70-1FA7C, U+1FA80-1FA88, U+1FA90-1FABD, U+1FABF-1FAC5, U+1FACE-1FADB, U+1FAE0-1FAE8, U+1FAF0-1FAF8,\n        U+1FBC5-1FBC9, U+E0061-E0067, U+E0069, U+E006C-E0079, U+E007F;\n}\n\n.copiedToClipboardMessage {\n    position: fixed;\n    top: 32px;\n    left: 50%;\n    transform: translateX(-50%);\n    background: #222;\n    color: #fff;\n    padding: 10px 24px;\n    border-radius: 8px;\n    font-size: 1.1em;\n    box-shadow: 0 2px 12px rgba(0, 0, 0, 0.18);\n    opacity: 0.97;\n    pointer-events: none;\n    z-index: 9999;\n}\n\n.Chat {\n    width: 100%;\n    height: 100%;\n    display: flex;\n    flex-direction: column;\n\n    font-family: Arial, Helvetica, sans-serif, 'OpenMojiBlack';\n    /* <- 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 chatStyles = {"copiedToClipboardMessage":"Chat-module_copiedToClipboardMessage__apCPY","Chat":"Chat-module_Chat__j2eE5","chatMainFlow":"Chat-module_chatMainFlow__--8FE","chatBar":"Chat-module_chatBar__fLECN","TasksInProgress":"Chat-module_TasksInProgress__fQfei","actions":"Chat-module_actions__gTZ5T","portal":"Chat-module_portal__uTOT8","left":"Chat-module_left__7l5Mn","right":"Chat-module_right__ABZrW","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","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","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"};
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(chatStyles.chatMessage, isMe && chatStyles.isMe, !isComplete && chatStyles.isNotCompleteMessage), onClick: () => {
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: chatStyles.avatar, onMouseEnter: handleMouseEnter, onMouseLeave: handleMouseLeave, onClick: showTooltip, children: [jsx("div", { style: {
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: chatStyles.messageText, style: {
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: chatStyles.copyButtonContainer, children: jsxs("button", { className: chatStyles.copyButton, title: "Copy message", onClick: async (e) => {
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: chatStyles.copiedTooltip +
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
- ? ' ' + chatStyles.copiedTooltipLeft
12225
+ ? ' ' + styles$2.copiedTooltipLeft
11742
12226
  : tooltipAlign === 'right'
11743
- ? ' ' + chatStyles.copiedTooltipRight
11744
- : ''), children: "Copied!" }))] }) })), message.isVoiceCall && (jsx("div", { className: chatStyles.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: chatStyles.attachments, children: message.attachments.map((attachment, index) => (jsxs("a", { href: attachment.url, target: "_blank", rel: "noopener noreferrer", className: chatStyles.attachment, title: attachment.name, children: [jsx("span", { className: chatStyles.attachmentIcon, children: "\uD83D\uDCCE" }), jsx("span", { className: chatStyles.attachmentName, children: attachment.name })] }, index))) })), completedToolCalls && completedToolCalls.length > 0 && (jsx("div", { className: chatStyles.completedToolCalls, children: completedToolCalls.map((toolCall, index) => {
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: chatStyles.completedToolCall, onClick: (event) => {
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: chatStyles.ongoingToolCalls, children: message.ongoingToolCalls.map((toolCall, index) => {
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 toolTitle = (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[toolCall.name]) || (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title);
11755
- const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
11756
- return (jsxs("div", { className: chatStyles.ongoingToolCall, children: [jsx("div", { className: chatStyles.ongoingToolCallSpinner }), jsx("span", { className: chatStyles.ongoingToolCallName, children: toolTitle
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: chatStyles.messageButtons, children: buttons.map((button, buttonIndex) => (jsx("button", { className: chatStyles.messageButton, onClick: (event) => {
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: chatStyles.rating, onMouseEnter: () => {
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(chatStyles.ratingStar, star <= (localHoveredRating || currentRating || 0) && chatStyles.active), style: {
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(chatStyles.ratingStar, currentRating && chatStyles.active), style: {
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
- * Renders a nice clock icon with a specific time.
12313
+ * Pause icon (two vertical bars)
11827
12314
  *
11828
- * @private internal subcomponent of `<Chat>` component
12315
+ * @public exported from `@promptbook/components`
11829
12316
  */
11830
- function ClockIcon({ date, size = 100 }) {
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
- * Renders a chat with messages and input for new messages
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
- * Note: There are multiple chat components:
11846
- * - `<Chat/>` renders chat as it is without any logic
11847
- * - `<LlmChat/>` connected to LLM Execution Tools of Promptbook
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
- * Use <WorkerChat/> or <SignalChat/> in most cases.
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
- function Chat(props) {
11854
- var _a;
11855
- const { title = 'Chat', messages, onChange, onMessage, onReset, onFeedback, onFileUpload, speechRecognition,
11856
- // isVoiceRecognitionButtonShown,
11857
- // voiceLanguage = 'en-US',
11858
- placeholderMessageContent, defaultMessage,
11859
- // tasksProgress,
11860
- children, className, style,
11861
- // voiceCallProps,
11862
- isAiTextHumanizedAndPromptbookified = true, isVoiceCalling = false, isFocusedOnLoad,
11863
- // isExperimental = false,
11864
- // TODO: [😅]> isSaveButtonEnabled = false,
11865
- // exportHeaderMarkdown,
11866
- participants = [], extraActions, actionsContainer, saveFormats, isSaveButtonEnabled = true, isCopyButtonEnabled = true, buttonColor: buttonColorRaw, onUseTemplate, onCreateAgent, toolTitles, visual, } = props;
11867
- const buttonColor = useMemo(() => Color.from(buttonColorRaw || '#0066cc'), [buttonColorRaw]);
11868
- // Use the auto-scroll hook
11869
- const { isAutoScrolling, chatMessagesRef, handleScroll, handleMessagesChange, scrollToBottom, isMobile: isMobileFromHook, } = useChatAutoScroll();
11870
- const textareaRef = useRef(null);
11871
- const buttonSendRef = useRef(null);
11872
- const fileInputRef = useRef(null);
11873
- const [ratingModalOpen, setRatingModalOpen] = useState(false);
11874
- const [toolCallModalOpen, setToolCallModalOpen] = useState(false);
11875
- const [selectedToolCall, setSelectedToolCall] = useState(null);
11876
- const [selectedMessage, setSelectedMessage] = useState(null);
11877
- const [messageRatings, setMessageRatings] = useState(new Map());
11878
- const [textRating, setTextRating] = useState('');
11879
- const [hoveredRating, setHoveredRating] = useState(0);
11880
- const [expandedMessageId, setExpandedMessageId] = useState(null);
11881
- // const [copiedToClipboard, setCopiedToClipboard] = useState(false);
11882
- // const [copiedMessageId, setCopiedMessageId] = useState<string | null>(null);
11883
- // const [isTyping, setIsTyping] = useState(false);
11884
- // const [inputValue, setInputValue] = useState('');
11885
- const [mode] = useState('LIGHT'); // Simplified light/dark mode
11886
- const [ratingConfirmation, setRatingConfirmation] = useState(null);
11887
- // File upload state
11888
- const [uploadedFiles, setUploadedFiles] = useState([]);
11889
- const [isDragOver, setIsDragOver] = useState(false);
11890
- const [isUploading, setIsUploading] = useState(false);
11891
- // Voice recognition state
11892
- const [speechRecognitionState, setSpeechRecognitionState] = useState('IDLE');
11893
- // Use mobile detection from the hook
11894
- const isMobile = isMobileFromHook;
11895
- useEffect(( /* Focus textarea on page load */) => {
11896
- if (!textareaRef.current) {
11897
- return;
11898
- }
11899
- // Note: By default, only auto-focus on desktop to prevent mobile keyboard from popping up
11900
- const isFocused = isFocusedOnLoad !== null && isFocusedOnLoad !== void 0 ? isFocusedOnLoad : !isMobile;
11901
- if (isFocused) {
11902
- textareaRef.current.focus();
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
- }, [textareaRef, isMobile, isFocusedOnLoad]);
11905
- // Voice recognition effects
11906
- useEffect(() => {
11907
- if (!speechRecognition) {
11908
- return;
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
- unsubscribe();
12445
+ setDisplayedMessages([]);
12446
+ setIsSimulationComplete(false);
12447
+ setResetNonce((nonce) => nonce + 1);
11938
12448
  };
11939
- }, [speechRecognition, onChange]);
11940
- const handleToggleVoiceInput = useCallback(() => {
11941
- if (!speechRecognition) {
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
- if (speechRecognition.state === 'IDLE' || speechRecognition.state === 'ERROR') {
11945
- speechRecognition.$start({ language: /* 'en-US' */ 'en' });
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
- else {
11948
- speechRecognition.$stop();
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
- }, [speechRecognition]);
11951
- // File upload handlers inspired by BookEditor
11952
- const handleFileUpload = useCallback(async (files) => {
11953
- if (!onFileUpload)
11954
- return;
11955
- setIsUploading(true);
11956
- const fileArray = Array.from(files);
11957
- try {
11958
- // Process files one by one as specified in requirements
11959
- const newUploadedFiles = [];
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
- setUploadedFiles((prev) => [...prev, ...newUploadedFiles]);
11969
- }
11970
- catch (error) {
11971
- console.error('File upload failed:', error);
11972
- alert('File upload failed. Please try again.');
11973
- }
11974
- finally {
11975
- setIsUploading(false);
11976
- }
11977
- }, [onFileUpload, onChange]);
11978
- const handleDrop = useCallback((event) => {
11979
- event.preventDefault();
11980
- setIsDragOver(false);
11981
- if (!onFileUpload)
11982
- return;
11983
- const files = event.dataTransfer.files;
11984
- if (files.length > 0) {
11985
- handleFileUpload(files);
11986
- }
11987
- }, [onFileUpload, handleFileUpload]);
11988
- const handleDragOver = useCallback((event) => {
11989
- event.preventDefault();
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
- ? chatStyles.actions + ' ' + chatStyles.left
12133
- : chatStyles.actions + ' ' + chatStyles.right;
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
- // Download logic
12147
- const [showSaveMenu, setShowSaveMenu] = useState(false);
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
- setShowSaveMenu((v) => !v);
13394
+ toggleSaveMenu();
12155
13395
  };
12156
13396
  window.addEventListener('keydown', handleKeyDown);
12157
13397
  return () => {
12158
13398
  window.removeEventListener('keydown', handleKeyDown);
12159
13399
  };
12160
- }, [setShowSaveMenu]);
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
- return (jsxs(Fragment, { children: [ratingConfirmation && jsx("div", { className: chatStyles.ratingConfirmation, children: ratingConfirmation }), jsx("div", { className: classNames(className, chatStyles.Chat, visual === 'STANDALONE' && chatStyles.standaloneVisual, visual === 'FULL_PAGE' && chatStyles.fullPageVisual, useChatCssClassName('Chat')), style, children: jsxs("div", { className: classNames(className, chatStyles.chatMainFlow, useChatCssClassName('chatMainFlow')), children: [children && jsx("div", { className: classNames(chatStyles.chatChildren), children: children }), !isAutoScrolling && (jsx("div", { className: chatStyles.scrollToBottomContainer, children: jsx("button", { "data-button-type": "custom", className: classNames(chatStyles.scrollToBottom, scrollToBottomCssClassName), onClick: scrollToBottom, children: jsx(ArrowIcon, { direction: "DOWN", size: 33 }) }) })), isVoiceCalling && (jsx("div", { className: chatStyles.voiceCallIndicatorBar, children: jsxs("div", { className: chatStyles.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: chatStyles.voiceCallPulse })] }) })), (() => {
12188
- const actionsContent = (jsxs("div", { className: classNames(actionsAlignmentClass, actionsContainer && chatStyles.portal), children: [onReset && postprocessedMessages.length !== 0 && (jsxs("button", { className: classNames(chatStyles.chatButton), onClick: () => {
12189
- if (!confirm(`Do you really want to reset the chat?`)) {
12190
- return;
12191
- }
12192
- onReset();
12193
- }, children: [jsx(ResetIcon, {}), jsx("span", { className: chatStyles.chatButtonText, children: "New chat" })] })), isSaveButtonEnabled && postprocessedMessages.length !== 0 && (jsxs("div", { className: chatStyles.saveButtonContainer, children: [jsxs("button", { className: classNames(chatStyles.chatButton), onClick: () => setShowSaveMenu((v) => !v), "aria-haspopup": "true", "aria-expanded": showSaveMenu, children: [jsx(SaveIcon, { size: 18 }), jsx("span", { className: chatStyles.chatButtonText, children: "Save" })] }), showSaveMenu && (jsx("div", { className: chatStyles.saveMenu, children: getChatSaveFormatDefinitions(saveFormats).map((formatDefinition) => (jsx("button", { className: chatStyles.saveMenuItem, onClick: () =>
12194
- // TODO: !!! Use here `$induceFileDownload`
12195
- handleDownload(formatDefinition.formatName), children: formatDefinition.label }, formatDefinition.formatName))) }))] })), onUseTemplate && (jsxs("button", { className: classNames(chatStyles.useTemplateButton), onClick: onUseTemplate, children: [jsx("span", { className: chatStyles.chatButtonText, children: "Use this template" }), jsx(TemplateIcon, { size: 16 })] })), extraActions] }));
12196
- if (actionsContainer) {
12197
- return createPortal(actionsContent, actionsContainer);
12198
- }
12199
- return actionsContent;
12200
- })(), jsxs("div", { className: classNames(chatStyles.chatMessages, useChatCssClassName('chatMessages'), (() => {
12201
- // Detect if actions are present
12202
- const hasActions = (!!onReset && postprocessedMessages.length !== 0) ||
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
- : {}), children: [uploadedFiles.length > 0 && (jsx("div", { className: chatStyles.filePreviewContainer, children: uploadedFiles.map((uploadedFile) => (jsxs("div", { className: chatStyles.filePreview, children: [jsx("div", { className: chatStyles.fileIcon, children: "\uD83D\uDCCE" }), jsxs("div", { className: chatStyles.fileInfo, children: [jsx("div", { className: chatStyles.fileName, children: uploadedFile.file.name }), jsxs("div", { className: chatStyles.fileSize, children: [(uploadedFile.file.size / 1024).toFixed(1), " KB"] })] }), jsx("button", { className: chatStyles.removeFileButton, onClick: () => removeUploadedFile(uploadedFile.id), title: "Remove file", children: jsx(CloseIcon /* !!! size={12}*/, {}) })] }, uploadedFile.id))) })), (() => {
12237
- var _a, _b, _c;
12238
- // Note: Find the isMe participant or fallback to default
12239
- const myColor = ((_a = participants.find((p) => p.isMe)) === null || _a === void 0 ? void 0 : _a.color) || USER_CHAT_COLOR;
12240
- const inputBgColor = Color.from(myColor).then(lighten(0.4)).then(grayscale(0.7));
12241
- const inputTextColor = inputBgColor.then(textColor);
12242
- return (jsxs("div", { className: chatStyles.inputContainer, style: {
12243
- // Use a high-contrast placeholder color for visibility
12244
- '--chat-placeholder-color': '#fff',
12245
- // <- TODO: Remove
12246
- '--input-bg-color': inputBgColor.toHex(),
12247
- '--input-text-color': inputTextColor.toHex(),
12248
- '--brand-color': buttonColor.toHex(),
12249
- }, children: [jsx("textarea", { ref: (element) => {
12250
- textareaRef.current = element;
12251
- }, onPaste: handlePaste, style: {
12252
- 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')
12253
- .length, 3) *
12254
- 25 +
12255
- 10,
12256
- }, defaultValue: defaultMessage, placeholder: placeholderMessageContent || 'Write a message...', onKeyDown: (event) => {
12257
- if (!onMessage) {
12258
- return;
12259
- }
12260
- if (event.shiftKey) {
12261
- return;
12262
- }
12263
- if (event.key !== 'Enter') {
12264
- return;
12265
- }
12266
- event.preventDefault();
12267
- /* not await */ handleSend();
12268
- }, onKeyUp: () => {
12269
- var _a;
12270
- if (!onChange) {
12271
- return;
12272
- }
12273
- onChange(((_a = textareaRef.current) === null || _a === void 0 ? void 0 : _a.value) || '');
12274
- } }), onFileUpload && (jsxs(Fragment, { children: [jsx("input", { ref: fileInputRef, type: "file", multiple: true, style: { display: 'none' }, onChange: handleFileInputChange }), jsx("button", { type: "button", style: {
12275
- backgroundColor: buttonColor.toHex(),
12276
- color: buttonColor.then(textColor).toHex(),
12277
- }, className: chatStyles.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' ||
12278
- speechRecognitionState === 'TRANSCRIBING', style: {
12279
- backgroundColor: (speechRecognitionState === 'RECORDING' ||
12280
- speechRecognitionState === 'TRANSCRIBING'
12281
- ? Color.from('#ff4444')
12282
- : buttonColor).toHex(),
12283
- color: (speechRecognitionState === 'RECORDING' ||
12284
- speechRecognitionState === 'TRANSCRIBING'
12285
- ? Color.from('#ffffff')
12286
- : buttonColor.then(textColor)).toHex(),
12287
- }, className: classNames(chatStyles.voiceButton, (isVoiceCalling ||
12288
- speechRecognitionState === 'RECORDING' ||
12289
- speechRecognitionState === 'TRANSCRIBING') &&
12290
- chatStyles.voiceButtonActive), onClick: (event) => {
12291
- event.preventDefault();
12292
- handleToggleVoiceInput();
12293
- }, title: speechRecognitionState === 'RECORDING'
12294
- ? 'Stop recording'
12295
- : speechRecognitionState === 'TRANSCRIBING'
12296
- ? 'Transcribing...'
12297
- : 'Start voice input', children: jsx(MicIcon, { size: 25 }) })), jsx("button", { "data-button-type": "call-to-action", style: {
12298
- backgroundColor: buttonColor.toHex(),
12299
- color: buttonColor.then(textColor).toHex(),
12300
- }, ref: buttonSendRef, onClick: (event) => {
12301
- if (!onMessage) {
12302
- return;
12303
- }
12304
- event.preventDefault();
12305
- /* not await */ handleSend();
12306
- }, children: jsx(SendIcon, { size: 25 }) })] }));
12307
- })(), isUploading && (jsxs("div", { className: chatStyles.uploadProgress, children: [jsx("div", { className: chatStyles.uploadProgressBar, children: jsx("div", { className: chatStyles.uploadProgressFill }) }), jsx("span", { children: "Uploading files..." })] })), isDragOver && onFileUpload && (jsx("div", { className: chatStyles.dragOverlay, children: jsxs("div", { className: chatStyles.dragOverlayContent, children: [jsx(AttachmentIcon, { size: 48 }), jsx("span", { children: "Drop files here to upload" })] }) }))] }))] }) }), toolCallModalOpen && selectedToolCall && (jsx("div", { className: chatStyles.ratingModal, onClick: (e) => {
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(chatStyles.ratingModalContent, chatStyles.toolCallModal), children: [(() => {
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: chatStyles.searchModalHeader, children: [jsx("span", { className: chatStyles.searchModalIcon, children: "\uD83D\uDD0E" }), jsx("h3", { className: chatStyles.searchModalQuery, children: args.query || args.searchText || 'Search Results' })] }), jsx("div", { className: chatStyles.searchModalContent, children: hasResults ? (jsx("div", { className: chatStyles.searchResultsList, children: results.map((item, i) => (jsxs("div", { className: chatStyles.searchResultItem, children: [jsx("div", { className: chatStyles.searchResultUrl, children: item.url && (jsx("a", { href: item.url, target: "_blank", rel: "noreferrer", children: item.url })) }), jsx("h4", { className: chatStyles.searchResultTitle, children: item.url ? (jsx("a", { href: item.url, target: "_blank", rel: "noreferrer", children: item.title || 'Untitled' })) : (item.title || 'Untitled') }), jsx("p", { className: chatStyles.searchResultSnippet, children: item.snippet || item.content || '' })] }, i))) })) : hasRawText ? (jsx(MarkdownContent, { className: chatStyles.searchResultsRaw, content: rawText })) : (jsx("div", { className: chatStyles.noResults, children: resultRaw
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: chatStyles.searchModalHeader, children: [jsx("span", { className: chatStyles.searchModalIcon, children: "\uD83D\uDD52" }), jsx("h3", { className: chatStyles.searchModalQuery, children: "Time at call" })] }), jsxs("div", { className: chatStyles.searchModalContent, children: [jsxs("div", { style: {
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: { fontSize: '0.9em', color: '#888', marginTop: '5px' }, children: ["(", relativeLabel, ")"] })), args.timezone && (jsxs("div", { style: { fontSize: '0.9em', color: '#888', marginTop: '5px' }, children: ["Timezone: ", args.timezone] }))] })] }), jsxs("div", { className: chatStyles.toolCallDetails, children: [jsx("p", { children: jsx("strong", { children: "Timestamp of call:" }) }), jsx("div", { className: chatStyles.toolCallDataContainer, children: jsx("pre", { className: chatStyles.toolCallData, children: toolCallDate ? toolCallDate.toLocaleString() : 'Unknown' }) })] })] })] }));
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: chatStyles.toolCallDetails, children: [jsx("p", { children: jsx("strong", { children: "Arguments:" }) }), jsx("div", { className: chatStyles.toolCallDataContainer, children: args && typeof args === 'object' ? (jsx("ul", { className: chatStyles.toolCallArgsList, children: Object.entries(args).map(([key, value]) => (jsxs("li", { children: [jsxs("strong", { children: [key, ":"] }), ' ', typeof value === 'object'
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: chatStyles.toolCallData, children: String(args) })) }), jsx("p", { children: jsx("strong", { children: "Result:" }) }), jsx("div", { className: chatStyles.toolCallDataContainer, children: jsx("pre", { className: chatStyles.toolCallData, children: typeof resultRaw === 'object'
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: chatStyles.ratingActions, children: jsx("button", { onClick: () => setToolCallModalOpen(false), children: "Close" }) })] }) })), ratingModalOpen && selectedMessage && (jsx("div", { className: chatStyles.ratingModal, onClick: (e) => {
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: chatStyles.ratingModalContent, children: [jsx("h3", { children: "Rate this response" }), jsx("div", { className: chatStyles.stars, children: [1, 2, 3, 4, 5].map((star) => (jsx("span", { onClick: () => setMessageRatings((previousRatings) => {
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: chatStyles.ratingModalStar, style: {
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: chatStyles.ratingInput }), "Expected answer:", jsx("textarea", { placeholder: selectedMessage.content || 'Expected answer (optional)', defaultValue: selectedMessage.expectedAnswer || selectedMessage.content, onChange: (e) => {
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: chatStyles.ratingInput }), "Note:", jsx("textarea", {
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: chatStyles.ratingInput }), jsxs("div", { className: chatStyles.ratingActions, children: [jsx("button", { onClick: () => setRatingModalOpen(false), children: "Cancel" }), jsx("button", { onClick: submitRating, children: "Submit" })] })] }) }))] }));
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
- // Notify about changes
12677
- if (onChange) {
12678
- onChange(buildInitialMessages(), participants);
13754
+ catch (error) {
13755
+ console.warn('Failed to clear chat messages from localStorage:', error);
12679
13756
  }
12680
- }, [persistenceKey, onReset, onChange, participants, buildInitialMessages]);
12681
- // Attach internal handler to external sendMessage (from useSendMessageToLlmChat) if provided
12682
- useEffect(() => {
12683
- if (sendMessage && sendMessage._attach) {
12684
- sendMessage._attach(handleMessage);
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
- }, [sendMessage, handleMessage]);
12687
- // Handle autoExecuteMessage
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
- }, [autoExecuteMessage, handleMessage]);
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
- * AgentChat component that provides chat functionality with LLM integration
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
- * - `<AgentChat/>` connected to LLM Execution Tools of Promptbook
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 MockedChat(props) {
12896
- const { delayConfig, messages: originalMessages, isResettable = true, isPausable = true, isSaveButtonEnabled = true, ...chatProps } = props;
12897
- // Helper to get random delay from config
12898
- function getDelay(val, fallback) {
12899
- if (Array.isArray(val) && val.length === 2) {
12900
- const [min, max] = val;
12901
- return Math.floor(Math.random() * (max - min + 1)) + min;
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
- if (typeof val === 'number')
12904
- return val;
12905
- return fallback;
12906
- }
12907
- const delays = {
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
- return () => {
12924
- setDisplayedMessages([]);
12925
- setIsSimulationComplete(false);
12926
- setResetNonce((nonce) => nonce + 1);
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
- }, [resetNonce, isResettable]);
12929
- // Helper: Wait while paused (entered only between messages, never mid-typing)
12930
- const waitIfPaused = async (isCancelledRef) => {
12931
- if (!pauseRequestedRef.current)
12932
- return;
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
- const resume = () => {
12951
- pauseRequestedRef.current = false;
12952
- if (playbackState !== 'RUNNING') {
12953
- // Actual state will become RUNNING after loop exits waitIfPaused
12954
- setPlaybackState('RUNNING');
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
- useEffect(() => {
12958
- let isCancelled = false;
12959
- const simulateChat = async () => {
12960
- // Reset state
12961
- setDisplayedMessages([]);
12962
- setIsSimulationComplete(false);
12963
- if (originalMessages.length === 0) {
12964
- setIsSimulationComplete(true);
12965
- return;
12966
- }
12967
- // Show intermediate messages immediately
12968
- const showIntermediateMessages = delays.showIntermediateMessages || 0;
12969
- if (showIntermediateMessages > 0) {
12970
- setDisplayedMessages(originalMessages.slice(0, showIntermediateMessages));
12971
- }
12972
- // Wait before first message (randomized)
12973
- await forTime(getDelay(delays.beforeFirstMessage, 1000));
12974
- if (isCancelled)
12975
- return;
12976
- for (let i = showIntermediateMessages; i < originalMessages.length; i++) {
12977
- // If a pause was requested earlier, we only pause between messages
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: currentMessage.id,
13039
- createdAt: currentMessage.createdAt,
13040
- sender: currentMessage.sender,
13041
- content: currentContent,
13902
+ id: loadingMessage.id,
13903
+ createdAt: new Date(),
13904
+ sender: llmParticipantName,
13905
+ content: chunk.content,
13042
13906
  isComplete: false,
13043
- expectedAnswer: currentMessage.expectedAnswer,
13044
- isVoiceCall: currentMessage.isVoiceCall,
13907
+ ongoingToolCalls: chunk.toolCalls,
13045
13908
  };
13046
- setDisplayedMessages((prev) => {
13047
- const newMessages = [...prev];
13048
- newMessages[newMessages.length - 1] = updatingMessage;
13049
- return newMessages;
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
- setIsSimulationComplete(true);
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
- return () => {
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
- else if (playbackState === 'PAUSED') {
13111
- resume();
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
- }, disabled: playbackState === 'PAUSING', children: [playbackState === 'RUNNING' && (jsxs(Fragment, { children: [jsx(PauseIcon, { size: 16 }), jsx("span", { className: chatStyles.chatButtonText, children: "Pause" })] })), playbackState === 'PAUSING' && (jsxs(Fragment, { children: [jsx(PauseIcon, { size: 16 }), jsx("span", { className: chatStyles.chatButtonText, children: "Pausing\u2026" })] })), playbackState === 'PAUSED' && (jsxs(Fragment, { children: [jsx(PlayIcon, { size: 16 }), jsx("span", { className: chatStyles.chatButtonText, children: "Resume" })] }))] })) : undefined;
13114
- // Use the internal Chat component with simulated messages
13115
- return (jsx(Chat, { ...chatProps, onReset: isResettable ? onReset : undefined, messages: displayedMessages, extraActions: extraActions, isSaveButtonEnabled: isSaveButtonEnabled, saveFormats: ['json', 'md', 'txt', 'html'],
13116
- // Disable input during simulation unless explicitly completed
13117
- onMessage: isSimulationComplete && chatProps.onMessage ? chatProps.onMessage : undefined }));
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 isHandled = false;
20930
- try {
20931
- const lines = textChunk.split('\n');
20932
- for (const line of lines) {
20933
- const trimmedLine = line.trim();
20934
- if (trimmedLine.startsWith('{') && trimmedLine.endsWith('}')) {
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
- isHandled = true;
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
- catch (error) {
20955
- // Ignore non-json chunks
21912
+ if (sawToolCalls) {
21913
+ if (!hasNonEmptyText) {
21914
+ continue;
21915
+ }
21916
+ const textChunkWithoutToolCalls = textLines.join('\n');
21917
+ content += textChunkWithoutToolCalls;
20956
21918
  }
20957
- if (isHandled) {
20958
- continue;
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,