@promptbook/components 0.105.0-1 → 0.105.0-11

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.
Files changed (71) hide show
  1. package/README.md +36 -77
  2. package/esm/index.es.js +2463 -483
  3. package/esm/index.es.js.map +1 -1
  4. package/esm/typings/servers.d.ts +6 -0
  5. package/esm/typings/src/_packages/browser.index.d.ts +2 -0
  6. package/esm/typings/src/_packages/core.index.d.ts +4 -0
  7. package/esm/typings/src/_packages/types.index.d.ts +16 -0
  8. package/esm/typings/src/_packages/utils.index.d.ts +2 -0
  9. package/esm/typings/src/book-2.0/agent-source/AgentBasicInformation.d.ts +15 -3
  10. package/esm/typings/src/book-2.0/agent-source/AgentModelRequirements.d.ts +11 -1
  11. package/esm/typings/src/book-2.0/agent-source/communication-samples.test.d.ts +1 -0
  12. package/esm/typings/src/book-2.0/agent-source/createAgentModelRequirementsWithCommitments.blocks.test.d.ts +1 -0
  13. package/esm/typings/src/book-2.0/agent-source/createAgentModelRequirementsWithCommitments.import.test.d.ts +1 -0
  14. package/esm/typings/src/book-2.0/agent-source/parseAgentSource.import.test.d.ts +1 -0
  15. package/esm/typings/src/book-2.0/agent-source/parseAgentSourceWithCommitments.blocks.test.d.ts +1 -0
  16. package/esm/typings/src/book-components/Chat/AgentChat/AgentChatProps.d.ts +5 -0
  17. package/esm/typings/src/book-components/Chat/Chat/ChatMessageItem.d.ts +15 -1
  18. package/esm/typings/src/book-components/Chat/Chat/ChatProps.d.ts +26 -11
  19. package/esm/typings/src/book-components/Chat/LlmChat/LlmChatProps.d.ts +5 -0
  20. package/esm/typings/src/book-components/Chat/types/ChatMessage.d.ts +43 -0
  21. package/esm/typings/src/book-components/Chat/utils/getToolCallChipletText.d.ts +22 -0
  22. package/esm/typings/src/commitments/NOTE/NOTE.d.ts +2 -2
  23. package/esm/typings/src/commitments/USE_BROWSER/USE_BROWSER.d.ts +4 -0
  24. package/esm/typings/src/commitments/USE_IMAGE_GENERATOR/USE_IMAGE_GENERATOR.d.ts +46 -0
  25. package/esm/typings/src/commitments/USE_IMAGE_GENERATOR/USE_IMAGE_GENERATOR.test.d.ts +1 -0
  26. package/esm/typings/src/commitments/USE_SEARCH_ENGINE/USE_SEARCH_ENGINE.d.ts +10 -0
  27. package/esm/typings/src/commitments/USE_SEARCH_ENGINE/USE_SEARCH_ENGINE.test.d.ts +1 -0
  28. package/esm/typings/src/commitments/USE_TIME/USE_TIME.d.ts +44 -0
  29. package/esm/typings/src/commitments/USE_TIME/USE_TIME.test.d.ts +1 -0
  30. package/esm/typings/src/commitments/_base/BaseCommitmentDefinition.d.ts +14 -0
  31. package/esm/typings/src/commitments/_base/CommitmentDefinition.d.ts +14 -0
  32. package/esm/typings/src/commitments/index.d.ts +18 -2
  33. package/esm/typings/src/config.d.ts +1 -0
  34. package/esm/typings/src/execution/LlmExecutionTools.d.ts +3 -1
  35. package/esm/typings/src/import-plugins/$fileImportPlugins.d.ts +7 -0
  36. package/esm/typings/src/import-plugins/AgentFileImportPlugin.d.ts +7 -0
  37. package/esm/typings/src/import-plugins/FileImportPlugin.d.ts +24 -0
  38. package/esm/typings/src/import-plugins/JsonFileImportPlugin.d.ts +7 -0
  39. package/esm/typings/src/import-plugins/TextFileImportPlugin.d.ts +7 -0
  40. package/esm/typings/src/llm-providers/_common/utils/cache/cacheLlmTools.d.ts +2 -1
  41. package/esm/typings/src/llm-providers/_common/utils/count-total-usage/countUsage.d.ts +2 -2
  42. package/esm/typings/src/llm-providers/agent/Agent.d.ts +14 -2
  43. package/esm/typings/src/llm-providers/agent/AgentLlmExecutionTools.d.ts +3 -1
  44. package/esm/typings/src/llm-providers/agent/AgentOptions.d.ts +7 -0
  45. package/esm/typings/src/llm-providers/agent/RemoteAgent.d.ts +1 -0
  46. package/esm/typings/src/llm-providers/agent/RemoteAgentOptions.d.ts +1 -1
  47. package/esm/typings/src/llm-providers/openai/OpenAiAssistantExecutionTools.d.ts +10 -0
  48. package/esm/typings/src/llm-providers/openai/OpenAiCompatibleExecutionTools.d.ts +7 -0
  49. package/esm/typings/src/llm-providers/remote/RemoteLlmExecutionTools.d.ts +1 -1
  50. package/esm/typings/src/scripting/javascript/JavascriptExecutionToolsOptions.d.ts +6 -1
  51. package/esm/typings/src/search-engines/SearchEngine.d.ts +1 -1
  52. package/esm/typings/src/search-engines/_index.d.ts +6 -0
  53. package/esm/typings/src/search-engines/bing/BingSearchEngine.d.ts +1 -1
  54. package/esm/typings/src/search-engines/dummy/DummySearchEngine.d.ts +1 -1
  55. package/esm/typings/src/search-engines/google/GoogleSearchEngine.d.ts +18 -0
  56. package/esm/typings/src/search-engines/serp/SerpSearchEngine.d.ts +15 -0
  57. package/esm/typings/src/speech-recognition/BrowserSpeechRecognition.d.ts +21 -0
  58. package/esm/typings/src/speech-recognition/OpenAiSpeechRecognition.d.ts +32 -0
  59. package/esm/typings/src/types/ModelRequirements.d.ts +6 -12
  60. package/esm/typings/src/types/SpeechRecognition.d.ts +58 -0
  61. package/esm/typings/src/types/typeAliases.d.ts +4 -0
  62. package/esm/typings/src/utils/execCommand/$execCommandNormalizeOptions.d.ts +2 -3
  63. package/esm/typings/src/utils/execCommand/ExecCommandOptions.d.ts +7 -1
  64. package/esm/typings/src/utils/misc/linguisticHash.d.ts +6 -0
  65. package/esm/typings/src/utils/misc/linguisticHash.test.d.ts +1 -0
  66. package/esm/typings/src/utils/organization/keepImported.d.ts +9 -0
  67. package/esm/typings/src/utils/organization/keepTypeImported.d.ts +0 -1
  68. package/esm/typings/src/version.d.ts +1 -1
  69. package/package.json +1 -1
  70. package/umd/index.umd.js +2466 -486
  71. package/umd/index.umd.js.map +1 -1
package/esm/index.es.js CHANGED
@@ -13,10 +13,10 @@ import { createRoot } from 'react-dom/client';
13
13
  import { Converter } from 'showdown';
14
14
  import { BehaviorSubject, Subject } from 'rxjs';
15
15
  import { forTime } from 'waitasecond';
16
+ import colors from 'colors';
16
17
  import sha256 from 'crypto-js/sha256';
17
18
  import { lookup, extension } from 'mime-types';
18
19
  import { parse, unparse } from 'papaparse';
19
- import colors from 'colors';
20
20
  import Bottleneck from 'bottleneck';
21
21
  import OpenAI from 'openai';
22
22
  import QRCode from 'qrcode';
@@ -35,7 +35,7 @@ const BOOK_LANGUAGE_VERSION = '2.0.0';
35
35
  * @generated
36
36
  * @see https://github.com/webgptorg/promptbook
37
37
  */
38
- const PROMPTBOOK_ENGINE_VERSION = '0.105.0-1';
38
+ const PROMPTBOOK_ENGINE_VERSION = '0.105.0-11';
39
39
  /**
40
40
  * TODO: string_promptbook_version should be constrained to the all versions of Promptbook engine
41
41
  * Note: [💞] Ignore a discrepancy between file name and entity name
@@ -1106,6 +1106,7 @@ const PROMPTBOOK_SYNTAX_COLORS = {
1106
1106
  SEPARATOR: Color.fromHex('#cccccc'),
1107
1107
  COMMITMENT: Color.fromHex('#DA0F78'),
1108
1108
  PARAMETER: Color.fromHex('#8e44ad'),
1109
+ CODE_BLOCK: Color.fromHex('#7700ffff'),
1109
1110
  };
1110
1111
  // <- TODO: [🧠][🈵] Using `Color` here increases the package size approx 3kb, maybe remove it
1111
1112
  /**
@@ -3211,6 +3212,698 @@ function capitalize(word) {
3211
3212
  return word.substring(0, 1).toUpperCase() + word.substring(1);
3212
3213
  }
3213
3214
 
3215
+ /**
3216
+ * Creates human-readable hash
3217
+ *
3218
+ * @public exported from `@promptbook/utils`
3219
+ */
3220
+ async function linguisticHash(input) {
3221
+ const hash = computeHash(input);
3222
+ // Use parts of the hash to select words
3223
+ // SHA256 is 64 hex characters
3224
+ // We use different slices of the hash to ensure variety even with small changes in input
3225
+ const part1 = parseInt(hash.substring(0, 10), 16);
3226
+ const part2 = parseInt(hash.substring(10, 20), 16);
3227
+ const part3 = parseInt(hash.substring(20, 30), 16);
3228
+ const adjective = ADJECTIVES[part1 % ADJECTIVES.length];
3229
+ const noun = NOUNS[part2 % NOUNS.length];
3230
+ const verb = VERBS[part3 % VERBS.length];
3231
+ return `${capitalize(adjective)} ${noun.toLowerCase()} ${verb.toLowerCase()}`;
3232
+ }
3233
+ const ADJECTIVES = [
3234
+ 'red',
3235
+ 'blue',
3236
+ 'green',
3237
+ 'yellow',
3238
+ 'quick',
3239
+ 'slow',
3240
+ 'bright',
3241
+ 'dark',
3242
+ 'happy',
3243
+ 'sad',
3244
+ 'brave',
3245
+ 'calm',
3246
+ 'clever',
3247
+ 'eager',
3248
+ 'fancy',
3249
+ 'grand',
3250
+ 'jolly',
3251
+ 'kind',
3252
+ 'lucky',
3253
+ 'nice',
3254
+ 'proud',
3255
+ 'silly',
3256
+ 'wise',
3257
+ 'young',
3258
+ 'old',
3259
+ 'big',
3260
+ 'small',
3261
+ 'fast',
3262
+ 'shiny',
3263
+ 'wild',
3264
+ 'silent',
3265
+ 'loud',
3266
+ 'soft',
3267
+ 'hard',
3268
+ 'warm',
3269
+ 'cold',
3270
+ 'sweet',
3271
+ 'sour',
3272
+ 'bitter',
3273
+ 'salty',
3274
+ 'rich',
3275
+ 'poor',
3276
+ 'heavy',
3277
+ 'light',
3278
+ 'strong',
3279
+ 'weak',
3280
+ 'smooth',
3281
+ 'rough',
3282
+ 'clean',
3283
+ 'dirty',
3284
+ 'fresh',
3285
+ 'stale',
3286
+ 'sharp',
3287
+ 'blunt',
3288
+ 'thick',
3289
+ 'thin',
3290
+ 'wide',
3291
+ 'narrow',
3292
+ 'deep',
3293
+ 'shallow',
3294
+ 'mighty',
3295
+ 'gentle',
3296
+ 'fierce',
3297
+ 'vibrant',
3298
+ 'dusty',
3299
+ 'golden',
3300
+ 'silver',
3301
+ 'frozen',
3302
+ 'burning',
3303
+ 'ancient',
3304
+ 'modern',
3305
+ 'hidden',
3306
+ 'lost',
3307
+ 'found',
3308
+ 'magic',
3309
+ 'mystic',
3310
+ 'cosmic',
3311
+ 'stellar',
3312
+ 'lunar',
3313
+ 'solar',
3314
+ 'misty',
3315
+ 'foggy',
3316
+ 'stormy',
3317
+ 'sunny',
3318
+ 'windy',
3319
+ 'quiet',
3320
+ 'noisy',
3321
+ 'peaceful',
3322
+ 'busy',
3323
+ 'empty',
3324
+ 'full',
3325
+ 'round',
3326
+ 'square',
3327
+ 'flat',
3328
+ 'curved',
3329
+ 'tiny',
3330
+ 'huge',
3331
+ 'giant',
3332
+ 'little',
3333
+ 'short',
3334
+ 'long',
3335
+ 'near',
3336
+ 'distant',
3337
+ 'inner',
3338
+ 'outer',
3339
+ 'patient',
3340
+ 'steady',
3341
+ 'noble',
3342
+ 'pure',
3343
+ 'graceful',
3344
+ 'honest',
3345
+ 'simple',
3346
+ 'complex',
3347
+ 'active',
3348
+ 'passive',
3349
+ 'vivid',
3350
+ 'pale',
3351
+ 'loyal',
3352
+ 'true',
3353
+ 'false',
3354
+ 'fair',
3355
+ 'clear',
3356
+ 'murky',
3357
+ 'vast',
3358
+ 'slick',
3359
+ 'slippery',
3360
+ 'sticky',
3361
+ 'dull',
3362
+ 'keen',
3363
+ 'broad',
3364
+ 'slim',
3365
+ 'slender',
3366
+ 'fat',
3367
+ 'lean',
3368
+ 'stiff',
3369
+ 'flexible',
3370
+ 'rigid',
3371
+ 'elastic',
3372
+ 'tough',
3373
+ 'brittle',
3374
+ 'fragile',
3375
+ 'solid',
3376
+ 'liquid',
3377
+ 'gaseous',
3378
+ 'airy',
3379
+ 'weighty',
3380
+ 'buoyant',
3381
+ 'dense',
3382
+ 'sparse',
3383
+ 'hollow',
3384
+ 'stuffed',
3385
+ 'crowded',
3386
+ 'lonely',
3387
+ 'social',
3388
+ 'private',
3389
+ 'public',
3390
+ 'secret',
3391
+ 'famous',
3392
+ 'certain',
3393
+ 'vague',
3394
+ 'plain',
3395
+ 'easy',
3396
+ 'tame',
3397
+ 'mild',
3398
+ 'hot',
3399
+ 'cool',
3400
+ 'dry',
3401
+ 'wet',
3402
+ 'damp',
3403
+ 'moist',
3404
+ 'soaked',
3405
+ 'parched',
3406
+ 'hungry',
3407
+ 'thirsty',
3408
+ 'sleepy',
3409
+ 'awake',
3410
+ 'tired',
3411
+ 'lazy',
3412
+ 'idle',
3413
+ 'swift',
3414
+ 'rapid',
3415
+ 'unstable',
3416
+ 'shaky',
3417
+ 'firm',
3418
+ 'bold',
3419
+ 'timid',
3420
+ 'brave',
3421
+ 'cowardly',
3422
+ 'smart',
3423
+ 'dumb',
3424
+ 'foolish',
3425
+ 'mean',
3426
+ 'rude',
3427
+ 'tasty',
3428
+ 'bland',
3429
+ 'arctic',
3430
+ 'tropical',
3431
+ 'deserted',
3432
+ 'urban',
3433
+ 'rural',
3434
+ 'local',
3435
+ 'global',
3436
+ 'digital',
3437
+ 'analog',
3438
+ 'virtual',
3439
+ 'real',
3440
+ 'fake',
3441
+ 'natural',
3442
+ 'artificial',
3443
+ 'living',
3444
+ 'dead',
3445
+ 'broken',
3446
+ 'fixed',
3447
+ 'new',
3448
+ 'worn',
3449
+ 'neat',
3450
+ 'messy',
3451
+ 'brave',
3452
+ 'fearful',
3453
+ 'proud',
3454
+ 'humble',
3455
+ 'greedy',
3456
+ 'generous',
3457
+ 'calm',
3458
+ 'angry',
3459
+ 'happy',
3460
+ 'sad',
3461
+ 'excited',
3462
+ 'bored',
3463
+ 'strange',
3464
+ 'normal',
3465
+ 'odd',
3466
+ 'even',
3467
+ 'rare',
3468
+ 'common',
3469
+ 'unique',
3470
+ 'plain',
3471
+ 'fancy',
3472
+ 'basic',
3473
+ 'prime',
3474
+ 'super',
3475
+ 'mega',
3476
+ 'ultra',
3477
+ 'micro',
3478
+ 'nano',
3479
+ 'macro',
3480
+ 'mini',
3481
+ ];
3482
+ const NOUNS = [
3483
+ 'apple',
3484
+ 'sky',
3485
+ 'tree',
3486
+ 'fox',
3487
+ 'cat',
3488
+ 'bird',
3489
+ 'dog',
3490
+ 'river',
3491
+ 'mountain',
3492
+ 'forest',
3493
+ 'ocean',
3494
+ 'star',
3495
+ 'moon',
3496
+ 'sun',
3497
+ 'cloud',
3498
+ 'flower',
3499
+ 'leaf',
3500
+ 'stone',
3501
+ 'wind',
3502
+ 'rain',
3503
+ 'fire',
3504
+ 'ice',
3505
+ 'book',
3506
+ 'dream',
3507
+ 'song',
3508
+ 'road',
3509
+ 'gate',
3510
+ 'key',
3511
+ 'lamp',
3512
+ 'map',
3513
+ 'house',
3514
+ 'city',
3515
+ 'bridge',
3516
+ 'field',
3517
+ 'garden',
3518
+ 'lake',
3519
+ 'beach',
3520
+ 'island',
3521
+ 'valley',
3522
+ 'desert',
3523
+ 'world',
3524
+ 'spirit',
3525
+ 'heart',
3526
+ 'mind',
3527
+ 'soul',
3528
+ 'life',
3529
+ 'time',
3530
+ 'space',
3531
+ 'light',
3532
+ 'shadow',
3533
+ 'sound',
3534
+ 'music',
3535
+ 'voice',
3536
+ 'word',
3537
+ 'page',
3538
+ 'story',
3539
+ 'pearl',
3540
+ 'gold',
3541
+ 'silver',
3542
+ 'crystal',
3543
+ 'diamond',
3544
+ 'emerald',
3545
+ 'ruby',
3546
+ 'path',
3547
+ 'trail',
3548
+ 'peak',
3549
+ 'shore',
3550
+ 'wave',
3551
+ 'tide',
3552
+ 'flame',
3553
+ 'spark',
3554
+ 'beam',
3555
+ 'ray',
3556
+ 'seed',
3557
+ 'root',
3558
+ 'branch',
3559
+ 'bloom',
3560
+ 'thorn',
3561
+ 'bark',
3562
+ 'shell',
3563
+ 'feather',
3564
+ 'wing',
3565
+ 'claw',
3566
+ 'paw',
3567
+ 'nest',
3568
+ 'cave',
3569
+ 'grove',
3570
+ 'tower',
3571
+ 'castle',
3572
+ 'crown',
3573
+ 'sword',
3574
+ 'shield',
3575
+ 'coin',
3576
+ 'gem',
3577
+ 'ring',
3578
+ 'bell',
3579
+ 'clock',
3580
+ 'compass',
3581
+ 'anchor',
3582
+ 'torch',
3583
+ 'flute',
3584
+ 'harp',
3585
+ 'drum',
3586
+ 'lens',
3587
+ 'glass',
3588
+ 'sand',
3589
+ 'dust',
3590
+ 'mist',
3591
+ 'dew',
3592
+ 'dawn',
3593
+ 'dusk',
3594
+ 'night',
3595
+ 'day',
3596
+ 'year',
3597
+ 'age',
3598
+ 'bolt',
3599
+ 'drop',
3600
+ 'storm',
3601
+ 'snow',
3602
+ 'hail',
3603
+ 'fog',
3604
+ 'smoke',
3605
+ 'vapor',
3606
+ 'gas',
3607
+ 'fluid',
3608
+ 'liquid',
3609
+ 'solid',
3610
+ 'metal',
3611
+ 'rock',
3612
+ 'dirt',
3613
+ 'clay',
3614
+ 'sand',
3615
+ 'salt',
3616
+ 'sugar',
3617
+ 'wood',
3618
+ 'bone',
3619
+ 'skin',
3620
+ 'flesh',
3621
+ 'blood',
3622
+ 'cell',
3623
+ 'atom',
3624
+ 'pulse',
3625
+ 'beat',
3626
+ 'breath',
3627
+ 'sigh',
3628
+ 'name',
3629
+ 'echo',
3630
+ 'image',
3631
+ 'vision',
3632
+ 'thought',
3633
+ 'idea',
3634
+ 'plan',
3635
+ 'goal',
3636
+ 'wish',
3637
+ 'hope',
3638
+ 'fear',
3639
+ 'joy',
3640
+ 'love',
3641
+ 'hate',
3642
+ 'will',
3643
+ 'power',
3644
+ 'force',
3645
+ 'energy',
3646
+ 'motion',
3647
+ 'speed',
3648
+ 'place',
3649
+ 'point',
3650
+ 'line',
3651
+ 'shape',
3652
+ 'form',
3653
+ 'size',
3654
+ 'mass',
3655
+ 'weight',
3656
+ 'heat',
3657
+ 'cold',
3658
+ 'color',
3659
+ 'tone',
3660
+ 'pitch',
3661
+ 'rhythm',
3662
+ 'vibe',
3663
+ 'mood',
3664
+ 'state',
3665
+ 'way',
3666
+ 'step',
3667
+ 'move',
3668
+ 'turn',
3669
+ 'fall',
3670
+ 'rise',
3671
+ 'jump',
3672
+ 'leap',
3673
+ 'run',
3674
+ 'walk',
3675
+ 'rest',
3676
+ 'stay',
3677
+ 'trip',
3678
+ 'quest',
3679
+ 'task',
3680
+ 'work',
3681
+ 'job',
3682
+ 'play',
3683
+ 'game',
3684
+ 'sport',
3685
+ 'art',
3686
+ 'craft',
3687
+ 'tool',
3688
+ 'ship',
3689
+ 'boat',
3690
+ 'car',
3691
+ 'bike',
3692
+ 'train',
3693
+ 'plane',
3694
+ 'hub',
3695
+ 'base',
3696
+ 'core',
3697
+ 'node',
3698
+ 'link',
3699
+ 'net',
3700
+ 'web',
3701
+ 'box',
3702
+ 'bag',
3703
+ 'jar',
3704
+ 'cup',
3705
+ 'bowl',
3706
+ 'plate',
3707
+ 'dish',
3708
+ 'spoon',
3709
+ 'fork',
3710
+ 'knife',
3711
+ 'pan',
3712
+ 'pot',
3713
+ 'bed',
3714
+ 'desk',
3715
+ 'chair',
3716
+ 'door',
3717
+ 'wall',
3718
+ 'roof',
3719
+ 'floor',
3720
+ ];
3721
+ const VERBS = [
3722
+ 'jumping',
3723
+ 'dancing',
3724
+ 'flying',
3725
+ 'running',
3726
+ 'singing',
3727
+ 'shining',
3728
+ 'growing',
3729
+ 'flowing',
3730
+ 'falling',
3731
+ 'rising',
3732
+ 'sleeping',
3733
+ 'walking',
3734
+ 'talking',
3735
+ 'thinking',
3736
+ 'dreaming',
3737
+ 'looking',
3738
+ 'feeling',
3739
+ 'smiling',
3740
+ 'laughing',
3741
+ 'playing',
3742
+ 'working',
3743
+ 'resting',
3744
+ 'moving',
3745
+ 'staying',
3746
+ 'beaming',
3747
+ 'glowing',
3748
+ 'sparkling',
3749
+ 'waiting',
3750
+ 'waking',
3751
+ 'drifting',
3752
+ 'spinning',
3753
+ 'gliding',
3754
+ 'soaring',
3755
+ 'floating',
3756
+ 'whispering',
3757
+ 'calling',
3758
+ 'seeking',
3759
+ 'finding',
3760
+ 'giving',
3761
+ 'taking',
3762
+ 'weaving',
3763
+ 'building',
3764
+ 'creating',
3765
+ 'burning',
3766
+ 'freezing',
3767
+ 'melting',
3768
+ 'breathing',
3769
+ 'pulsing',
3770
+ 'beating',
3771
+ 'living',
3772
+ 'learning',
3773
+ 'knowing',
3774
+ 'hidden',
3775
+ 'shown',
3776
+ 'broken',
3777
+ 'mended',
3778
+ 'lost',
3779
+ 'found',
3780
+ 'starting',
3781
+ 'ending',
3782
+ 'climbing',
3783
+ 'diving',
3784
+ 'swimming',
3785
+ 'sailing',
3786
+ 'rolling',
3787
+ 'shaking',
3788
+ 'turning',
3789
+ 'shifting',
3790
+ 'changing',
3791
+ 'fading',
3792
+ 'dying',
3793
+ 'born',
3794
+ 'humming',
3795
+ 'crying',
3796
+ 'racing',
3797
+ 'creeping',
3798
+ 'hiding',
3799
+ 'watching',
3800
+ 'hearing',
3801
+ 'sensing',
3802
+ 'longing',
3803
+ 'hoping',
3804
+ 'loving',
3805
+ 'fearing',
3806
+ 'wondering',
3807
+ 'wandering',
3808
+ 'traveling',
3809
+ 'crossing',
3810
+ 'meeting',
3811
+ 'parting',
3812
+ 'keeping',
3813
+ 'sharing',
3814
+ 'sparking',
3815
+ 'flaming',
3816
+ 'healing',
3817
+ 'solving',
3818
+ 'opening',
3819
+ 'closing',
3820
+ 'lifting',
3821
+ 'pulling',
3822
+ 'pushing',
3823
+ 'holding',
3824
+ 'tossing',
3825
+ 'throwing',
3826
+ 'catching',
3827
+ 'fixing',
3828
+ 'making',
3829
+ 'doing',
3830
+ 'seeing',
3831
+ 'tasting',
3832
+ 'smelling',
3833
+ 'touching',
3834
+ 'pacing',
3835
+ 'hurrying',
3836
+ 'pausing',
3837
+ 'going',
3838
+ 'coming',
3839
+ 'leaving',
3840
+ 'acting',
3841
+ 'being',
3842
+ 'seeming',
3843
+ 'shrinking',
3844
+ 'widening',
3845
+ 'narrowing',
3846
+ 'heating',
3847
+ 'cooling',
3848
+ 'drying',
3849
+ 'wetting',
3850
+ 'filling',
3851
+ 'filling',
3852
+ 'emptying',
3853
+ 'letting',
3854
+ 'gaining',
3855
+ 'winning',
3856
+ 'failing',
3857
+ 'trying',
3858
+ 'using',
3859
+ 'getting',
3860
+ 'showing',
3861
+ 'hiding',
3862
+ 'breaking',
3863
+ 'fixing',
3864
+ 'saving',
3865
+ 'spending',
3866
+ 'buying',
3867
+ 'selling',
3868
+ 'paying',
3869
+ 'costing',
3870
+ 'reaching',
3871
+ 'missing',
3872
+ 'hitting',
3873
+ 'striking',
3874
+ 'leading',
3875
+ 'following',
3876
+ 'helping',
3877
+ 'serving',
3878
+ 'teaching',
3879
+ 'training',
3880
+ 'coding',
3881
+ 'writing',
3882
+ 'reading',
3883
+ 'drawing',
3884
+ 'painting',
3885
+ 'crafting',
3886
+ 'shaping',
3887
+ 'forming',
3888
+ 'joining',
3889
+ 'splitting',
3890
+ 'sharing',
3891
+ 'bonding',
3892
+ 'healing',
3893
+ 'harming',
3894
+ 'protecting',
3895
+ 'fighting',
3896
+ 'defending',
3897
+ 'attacking',
3898
+ 'escaping',
3899
+ 'catching',
3900
+ 'trapping',
3901
+ 'freeing',
3902
+ 'binding',
3903
+ 'weaving',
3904
+ 'spinning',
3905
+ ];
3906
+
3214
3907
  /**
3215
3908
  * Normalizes a text string to SCREAMING_CASE (all uppercase with underscores).
3216
3909
  *
@@ -3310,6 +4003,95 @@ function normalizeTo_snake_case(text) {
3310
4003
  return normalizeTo_SCREAMING_CASE(text).toLowerCase();
3311
4004
  }
3312
4005
 
4006
+ /**
4007
+ * Removes quotes and optional introduce text from a string
4008
+ *
4009
+ * Tip: This is very useful for post-processing of the result of the LLM model
4010
+ * Note: This function trims the text and removes whole introduce sentence if it is present
4011
+ * Note: There are two similar functions:
4012
+ * - `removeQuotes` which removes only bounding quotes
4013
+ * - `unwrapResult` which removes whole introduce sentence
4014
+ *
4015
+ * @param text optionally quoted text
4016
+ * @returns text without quotes
4017
+ * @public exported from `@promptbook/utils`
4018
+ */
4019
+ function unwrapResult(text, options) {
4020
+ const { isTrimmed = true, isIntroduceSentenceRemoved = true } = options || {};
4021
+ let trimmedText = text;
4022
+ // Remove leading and trailing spaces and newlines
4023
+ if (isTrimmed) {
4024
+ trimmedText = spaceTrim$1(trimmedText);
4025
+ }
4026
+ let processedText = trimmedText;
4027
+ // Check for markdown code block
4028
+ const codeBlockRegex = /^```[a-z]*\n([\s\S]*?)\n```\s*$/;
4029
+ const codeBlockMatch = processedText.match(codeBlockRegex);
4030
+ if (codeBlockMatch && codeBlockMatch[1] !== undefined) {
4031
+ // Check if there's only one code block
4032
+ const codeBlockCount = (processedText.match(/```/g) || []).length / 2;
4033
+ if (codeBlockCount === 1) {
4034
+ return unwrapResult(codeBlockMatch[1], { isTrimmed: false, isIntroduceSentenceRemoved: false });
4035
+ }
4036
+ }
4037
+ if (isIntroduceSentenceRemoved) {
4038
+ const introduceSentenceRegex = /^[a-zěščřžýáíéúů:\s]*:\s*/i;
4039
+ if (introduceSentenceRegex.test(text)) {
4040
+ // Remove the introduce sentence and quotes by replacing it with an empty string
4041
+ processedText = processedText.replace(introduceSentenceRegex, '');
4042
+ }
4043
+ processedText = spaceTrim$1(processedText);
4044
+ // Check again for code block after removing introduce sentence
4045
+ const codeBlockMatch2 = processedText.match(codeBlockRegex);
4046
+ if (codeBlockMatch2 && codeBlockMatch2[1] !== undefined) {
4047
+ const codeBlockCount = (processedText.match(/```/g) || []).length / 2;
4048
+ if (codeBlockCount === 1) {
4049
+ return unwrapResult(codeBlockMatch2[1], { isTrimmed: false, isIntroduceSentenceRemoved: false });
4050
+ }
4051
+ }
4052
+ }
4053
+ if (processedText.length < 3) {
4054
+ return trimmedText;
4055
+ }
4056
+ if (processedText.includes('\n')) {
4057
+ return trimmedText;
4058
+ }
4059
+ // Remove the quotes by extracting the substring without the first and last characters
4060
+ const unquotedText = processedText.slice(1, -1);
4061
+ // Check if the text starts and ends with quotes
4062
+ if ([
4063
+ ['"', '"'],
4064
+ ["'", "'"],
4065
+ ['`', '`'],
4066
+ ['*', '*'],
4067
+ ['_', '_'],
4068
+ ['„', '“'],
4069
+ ['«', '»'] /* <- QUOTES to config */,
4070
+ ].some(([startQuote, endQuote]) => {
4071
+ if (!processedText.startsWith(startQuote)) {
4072
+ return false;
4073
+ }
4074
+ if (!processedText.endsWith(endQuote)) {
4075
+ return false;
4076
+ }
4077
+ if (unquotedText.includes(startQuote) && !unquotedText.includes(endQuote)) {
4078
+ return false;
4079
+ }
4080
+ if (!unquotedText.includes(startQuote) && unquotedText.includes(endQuote)) {
4081
+ return false;
4082
+ }
4083
+ return true;
4084
+ })) {
4085
+ return unwrapResult(unquotedText, { isTrimmed: false, isIntroduceSentenceRemoved: false });
4086
+ }
4087
+ else {
4088
+ return processedText;
4089
+ }
4090
+ }
4091
+ /**
4092
+ * TODO: [🧠] Should this also unwrap the (parenthesis)
4093
+ */
4094
+
3313
4095
  /**
3314
4096
  * Parses the task and returns the list of all parameter names
3315
4097
  *
@@ -3714,6 +4496,22 @@ class BaseCommitmentDefinition {
3714
4496
  return this.appendToSystemMessage(requirements, commentSection);
3715
4497
  }
3716
4498
  }
4499
+ /**
4500
+ * Gets tool function implementations provided by this commitment
4501
+ *
4502
+ * When the `applyToAgentModelRequirements` adds tools to the requirements, this method should return the corresponding function definitions.
4503
+ */
4504
+ getToolFunctions() {
4505
+ return {};
4506
+ }
4507
+ /**
4508
+ * Gets human-readable titles for tool functions provided by this commitment
4509
+ *
4510
+ * This is used in the UI to show a user-friendly name instead of the technical function name.
4511
+ */
4512
+ getToolTitles() {
4513
+ return {};
4514
+ }
3717
4515
  }
3718
4516
 
3719
4517
  /**
@@ -4346,79 +5144,6 @@ class FromCommitmentDefinition extends BaseCommitmentDefinition {
4346
5144
  * Note: [💞] Ignore a discrepancy between file name and entity name
4347
5145
  */
4348
5146
 
4349
- /**
4350
- * IMPORT commitment definition
4351
- *
4352
- * The IMPORT commitment tells the agent to import content from another agent at the current location.
4353
- *
4354
- * Example usage in agent source:
4355
- *
4356
- * ```book
4357
- * IMPORT https://s6.ptbk.io/benjamin-white
4358
- * ```
4359
- *
4360
- * @private [🪔] Maybe export the commitments through some package
4361
- */
4362
- class ImportCommitmentDefinition extends BaseCommitmentDefinition {
4363
- constructor(type = 'IMPORT') {
4364
- super(type);
4365
- }
4366
- /**
4367
- * Short one-line description of IMPORT.
4368
- */
4369
- get description() {
4370
- return 'Import content from another agent.';
4371
- }
4372
- /**
4373
- * Icon for this commitment.
4374
- */
4375
- get icon() {
4376
- return '📥';
4377
- }
4378
- /**
4379
- * Markdown documentation for IMPORT commitment.
4380
- */
4381
- get documentation() {
4382
- return spaceTrim$1(`
4383
- # ${this.type}
4384
-
4385
- Imports content from another agent at the location of the commitment.
4386
-
4387
- ## Examples
4388
-
4389
- \`\`\`book
4390
- My AI Agent
4391
-
4392
- IMPORT https://s6.ptbk.io/benjamin-white
4393
- RULE Speak only in English.
4394
- \`\`\`
4395
- `);
4396
- }
4397
- applyToAgentModelRequirements(requirements, content) {
4398
- const trimmedContent = content.trim();
4399
- if (!trimmedContent) {
4400
- return requirements;
4401
- }
4402
- if (!isValidAgentUrl(trimmedContent)) {
4403
- throw new Error(spaceTrim$1((block) => `
4404
- Invalid agent URL in IMPORT commitment: "${trimmedContent}"
4405
-
4406
- \`\`\`book
4407
- ${block(content)}
4408
- \`\`\`
4409
- `));
4410
- }
4411
- const importedAgentUrl = trimmedContent;
4412
- return {
4413
- ...requirements,
4414
- importedAgentUrls: [...(requirements.importedAgentUrls || []), importedAgentUrl],
4415
- };
4416
- }
4417
- }
4418
- /**
4419
- * Note: [💞] Ignore a discrepancy between file name and entity name
4420
- */
4421
-
4422
5147
  /**
4423
5148
  * GOAL commitment definition
4424
5149
  *
@@ -4519,6 +5244,87 @@ class GoalCommitmentDefinition extends BaseCommitmentDefinition {
4519
5244
  * Note: [💞] Ignore a discrepancy between file name and entity name
4520
5245
  */
4521
5246
 
5247
+ /**
5248
+ * IMPORT commitment definition
5249
+ *
5250
+ * The IMPORT commitment tells the agent to import content from another agent at the current location.
5251
+ *
5252
+ * Example usage in agent source:
5253
+ *
5254
+ * ```book
5255
+ * IMPORT https://s6.ptbk.io/benjamin-white
5256
+ * ```
5257
+ *
5258
+ * @private [🪔] Maybe export the commitments through some package
5259
+ */
5260
+ class ImportCommitmentDefinition extends BaseCommitmentDefinition {
5261
+ constructor(type = 'IMPORT') {
5262
+ super(type);
5263
+ }
5264
+ /**
5265
+ * Short one-line description of IMPORT.
5266
+ */
5267
+ get description() {
5268
+ return 'Import content from another agent or a generic text file.';
5269
+ }
5270
+ /**
5271
+ * Icon for this commitment.
5272
+ */
5273
+ get icon() {
5274
+ return '📥';
5275
+ }
5276
+ /**
5277
+ * Markdown documentation for IMPORT commitment.
5278
+ */
5279
+ get documentation() {
5280
+ return spaceTrim$1(`
5281
+ # ${this.type}
5282
+
5283
+ Imports content from another agent or a generic text file at the location of the commitment.
5284
+
5285
+ ## Examples
5286
+
5287
+ \`\`\`book
5288
+ My AI Agent
5289
+
5290
+ IMPORT https://s6.ptbk.io/benjamin-white
5291
+ IMPORT https://example.com/some-text-file.txt
5292
+ IMPORT ./path/to/local-file.json
5293
+ RULE Speak only in English.
5294
+ \`\`\`
5295
+ `);
5296
+ }
5297
+ applyToAgentModelRequirements(requirements, content) {
5298
+ const trimmedContent = content.trim();
5299
+ if (!trimmedContent) {
5300
+ return requirements;
5301
+ }
5302
+ if (isValidAgentUrl(trimmedContent)) {
5303
+ const importedAgentUrl = trimmedContent;
5304
+ return {
5305
+ ...requirements,
5306
+ importedAgentUrls: [...(requirements.importedAgentUrls || []), importedAgentUrl],
5307
+ };
5308
+ }
5309
+ if (isValidUrl(trimmedContent) || isValidFilePath(trimmedContent)) {
5310
+ return {
5311
+ ...requirements,
5312
+ importedFileUrls: [...(requirements.importedFileUrls || []), trimmedContent],
5313
+ };
5314
+ }
5315
+ throw new Error(spaceTrim$1((block) => `
5316
+ Invalid agent URL or file path in IMPORT commitment: "${trimmedContent}"
5317
+
5318
+ \`\`\`book
5319
+ ${block(content)}
5320
+ \`\`\`
5321
+ `));
5322
+ }
5323
+ }
5324
+ /**
5325
+ * Note: [💞] Ignore a discrepancy between file name and entity name
5326
+ */
5327
+
4522
5328
  /**
4523
5329
  * KNOWLEDGE commitment definition
4524
5330
  *
@@ -4938,7 +5744,13 @@ class InitialMessageCommitmentDefinition extends BaseCommitmentDefinition {
4938
5744
  `);
4939
5745
  }
4940
5746
  applyToAgentModelRequirements(requirements, content) {
4941
- return requirements;
5747
+ // INITIAL MESSAGE is for UI display purposes and for conversation history construction.
5748
+ const newSample = { question: null, answer: content };
5749
+ const newSamples = [...(requirements.samples || []), newSample];
5750
+ return {
5751
+ ...requirements,
5752
+ samples: newSamples,
5753
+ };
4942
5754
  }
4943
5755
  }
4944
5756
 
@@ -5970,27 +6782,16 @@ class NoteCommitmentDefinition extends BaseCommitmentDefinition {
5970
6782
  `);
5971
6783
  }
5972
6784
  applyToAgentModelRequirements(requirements, content) {
5973
- var _a;
5974
6785
  // The NOTE commitment makes no changes to the system message or model requirements
5975
6786
  // It only stores the note content in metadata for documentation purposes
5976
- const trimmedContent = content.trim();
5977
- if (!trimmedContent) {
6787
+ const trimmedContent = spaceTrim$1(content);
6788
+ if (trimmedContent === '') {
5978
6789
  return requirements;
5979
6790
  }
5980
- // Get existing note content from metadata
5981
- const existingNoteContent = ((_a = requirements.metadata) === null || _a === void 0 ? void 0 : _a.NOTE) || '';
5982
- // Merge the new content with existing note content
5983
- // When multiple NOTE commitments exist, they are aggregated together
5984
- const mergedNoteContent = existingNoteContent ? `${existingNoteContent}\n${trimmedContent}` : trimmedContent;
5985
- // Store the merged note content in metadata for debugging and inspection
5986
- const updatedMetadata = {
5987
- ...requirements.metadata,
5988
- NOTE: mergedNoteContent,
5989
- };
5990
- // Return requirements with updated metadata but no changes to system message
6791
+ // Return requirements with updated notes but no changes to system message
5991
6792
  return {
5992
6793
  ...requirements,
5993
- metadata: updatedMetadata,
6794
+ notes: [...(requirements.notes || []), trimmedContent],
5994
6795
  };
5995
6796
  }
5996
6797
  }
@@ -6774,6 +7575,14 @@ class UseBrowserCommitmentDefinition extends BaseCommitmentDefinition {
6774
7575
  \`\`\`
6775
7576
  `);
6776
7577
  }
7578
+ /**
7579
+ * Gets human-readable titles for tool functions provided by this commitment.
7580
+ */
7581
+ getToolTitles() {
7582
+ return {
7583
+ web_browser: 'Web browser',
7584
+ };
7585
+ }
6777
7586
  applyToAgentModelRequirements(requirements, content) {
6778
7587
  // Get existing tools array or create new one
6779
7588
  const existingTools = requirements.tools || [];
@@ -6802,14 +7611,17 @@ class UseBrowserCommitmentDefinition extends BaseCommitmentDefinition {
6802
7611
  },
6803
7612
  ]);
6804
7613
  // Return requirements with updated tools and metadata
6805
- return {
7614
+ return this.appendToSystemMessage({
6806
7615
  ...requirements,
6807
7616
  tools: updatedTools,
6808
7617
  metadata: {
6809
7618
  ...requirements.metadata,
6810
7619
  useBrowser: true,
6811
7620
  },
6812
- };
7621
+ }, spaceTrim$1(`
7622
+ You have access to the web browser. Use it to access specific websites or browse the internet.
7623
+ When you need to know some information from a specific website, use the tool provided to you.
7624
+ `));
6813
7625
  }
6814
7626
  }
6815
7627
  /**
@@ -6817,82 +7629,267 @@ class UseBrowserCommitmentDefinition extends BaseCommitmentDefinition {
6817
7629
  */
6818
7630
 
6819
7631
  /**
6820
- * USE MCP commitment definition
6821
- *
6822
- * The `USE MCP` commitment allows to specify an MCP server URL which the agent will connect to
6823
- * for retrieving additional instructions and actions.
7632
+ * USE IMAGE GENERATOR commitment definition
6824
7633
  *
6825
- * The content following `USE MCP` is the URL of the MCP server.
7634
+ * The `USE IMAGE GENERATOR` commitment indicates that the agent should utilize an image generation tool
7635
+ * to create images based on text prompts.
6826
7636
  *
6827
7637
  * Example usage in agent source:
6828
7638
  *
6829
7639
  * ```book
6830
- * USE MCP http://mcp-server-url.com
7640
+ * USE IMAGE GENERATOR
7641
+ * USE IMAGE GENERATOR Create realistic images of nature
6831
7642
  * ```
6832
7643
  *
6833
7644
  * @private [🪔] Maybe export the commitments through some package
6834
7645
  */
6835
- class UseMcpCommitmentDefinition extends BaseCommitmentDefinition {
6836
- constructor() {
6837
- super('USE MCP', ['MCP']);
7646
+ class UseImageGeneratorCommitmentDefinition extends BaseCommitmentDefinition {
7647
+ constructor(type = 'USE IMAGE GENERATOR') {
7648
+ super(type, ['USE IMAGE GENERATION', 'IMAGE GENERATOR', 'IMAGE GENERATION', 'USE IMAGE']);
6838
7649
  }
6839
7650
  /**
6840
- * Short one-line description of USE MCP.
7651
+ * Short one-line description of USE IMAGE GENERATOR.
6841
7652
  */
6842
7653
  get description() {
6843
- return 'Connects the agent to an external MCP server for additional capabilities.';
7654
+ return 'Enable the agent to use an image generation tool for creating images from text prompts.';
6844
7655
  }
6845
7656
  /**
6846
7657
  * Icon for this commitment.
6847
7658
  */
6848
7659
  get icon() {
6849
- return '🔌';
7660
+ return '🖼️';
6850
7661
  }
6851
7662
  /**
6852
- * Markdown documentation for USE MCP commitment.
7663
+ * Markdown documentation for USE IMAGE GENERATOR commitment.
6853
7664
  */
6854
7665
  get documentation() {
6855
7666
  return spaceTrim$1(`
6856
- # USE MCP
7667
+ # USE IMAGE GENERATOR
6857
7668
 
6858
- Connects the agent to an external Model Context Protocol (MCP) server.
7669
+ Enables the agent to use an image generation tool to create images based on text prompts.
6859
7670
 
6860
7671
  ## Key aspects
6861
7672
 
6862
- - The content following \`USE MCP\` must be a valid URL
6863
- - Multiple MCP servers can be connected by using multiple \`USE MCP\` commitments
6864
- - The agent will have access to tools and resources provided by the MCP server
7673
+ - The content following \`USE IMAGE GENERATOR\` is an arbitrary text that the agent should know (e.g. style instructions or safety guidelines).
7674
+ - The actual image generation is handled by the agent runtime using LLM execution tools.
7675
+ - Allows the agent to generate visual content based on user requests.
7676
+ - Returns the URL of the generated image.
6865
7677
 
6866
- ## Example
7678
+ ## Examples
6867
7679
 
6868
7680
  \`\`\`book
6869
- Company Lawyer
7681
+ Visual Artist
6870
7682
 
6871
- PERSONA You are a company lawyer.
6872
- USE MCP http://legal-db.example.com
7683
+ PERSONA You are a creative visual artist who can generate images.
7684
+ USE IMAGE GENERATOR
7685
+ RULE Always describe the generated image to the user.
7686
+ \`\`\`
7687
+
7688
+ \`\`\`book
7689
+ Interior Designer
7690
+
7691
+ PERSONA You are an interior designer who helps users visualize their space.
7692
+ USE IMAGE GENERATOR Professional interior design renders.
7693
+ ACTION Generate a preview of the designed room.
6873
7694
  \`\`\`
6874
7695
  `);
6875
7696
  }
6876
7697
  applyToAgentModelRequirements(requirements, content) {
6877
- const mcpServerUrl = content.trim();
6878
- if (!mcpServerUrl) {
6879
- return requirements;
6880
- }
6881
- const existingMcpServers = requirements.mcpServers || [];
6882
- // Avoid duplicates
6883
- if (existingMcpServers.includes(mcpServerUrl)) {
6884
- return requirements;
6885
- }
6886
- return {
7698
+ // Get existing tools array or create new one
7699
+ const existingTools = requirements.tools || [];
7700
+ // Add 'generate_image' to tools if not already present
7701
+ const updatedTools = existingTools.some((tool) => tool.name === 'generate_image')
7702
+ ? existingTools
7703
+ : [
7704
+ ...existingTools,
7705
+ {
7706
+ name: 'generate_image',
7707
+ description: spaceTrim$1(`
7708
+ Generate an image from a text prompt.
7709
+ Use this tool when the user asks to create, draw, or generate an image.
7710
+ ${!content ? '' : `Style instructions / guidelines: ${content}`}
7711
+ `),
7712
+ parameters: {
7713
+ type: 'object',
7714
+ properties: {
7715
+ prompt: {
7716
+ type: 'string',
7717
+ description: 'The detailed description of the image to generate',
7718
+ },
7719
+ },
7720
+ required: ['prompt'],
7721
+ },
7722
+ },
7723
+ ];
7724
+ // Return requirements with updated tools and metadata
7725
+ return this.appendToSystemMessage({
6887
7726
  ...requirements,
6888
- mcpServers: [...existingMcpServers, mcpServerUrl],
6889
- };
7727
+ tools: updatedTools,
7728
+ metadata: {
7729
+ ...requirements.metadata,
7730
+ useImageGenerator: content || true,
7731
+ },
7732
+ }, spaceTrim$1(`
7733
+ You have access to an image generator. Use it to create images based on user requests.
7734
+ When you generate an image, you will receive a URL of the generated image.
7735
+ `));
7736
+ }
7737
+ /**
7738
+ * Gets human-readable titles for tool functions provided by this commitment.
7739
+ */
7740
+ getToolTitles() {
7741
+ return {
7742
+ generate_image: 'Generate image',
7743
+ };
7744
+ }
7745
+ /**
7746
+ * Gets the `generate_image` tool function implementation.
7747
+ */
7748
+ getToolFunctions() {
7749
+ return {
7750
+ async generate_image(args, ...extra) {
7751
+ console.log('!!!! [Tool] generate_image called', { args });
7752
+ const { prompt } = args;
7753
+ if (!prompt) {
7754
+ throw new Error('Image prompt is required');
7755
+ }
7756
+ const { llmTools } = extra[0] || {};
7757
+ if (!llmTools || !llmTools.callImageGenerationModel) {
7758
+ throw new Error('Image generation is not supported by the current model provider');
7759
+ }
7760
+ const result = await llmTools.callImageGenerationModel({
7761
+ content: prompt,
7762
+ modelName: 'dall-e-3', // Defaulting to dall-e-3, but this could be configurable
7763
+ });
7764
+ return result.content;
7765
+ },
7766
+ };
6890
7767
  }
6891
7768
  }
6892
7769
  /**
6893
7770
  * Note: [💞] Ignore a discrepancy between file name and entity name
6894
7771
  */
6895
7772
 
7773
+ /**
7774
+ * USE MCP commitment definition
7775
+ *
7776
+ * The `USE MCP` commitment allows to specify an MCP server URL which the agent will connect to
7777
+ * for retrieving additional instructions and actions.
7778
+ *
7779
+ * The content following `USE MCP` is the URL of the MCP server.
7780
+ *
7781
+ * Example usage in agent source:
7782
+ *
7783
+ * ```book
7784
+ * USE MCP http://mcp-server-url.com
7785
+ * ```
7786
+ *
7787
+ * @private [🪔] Maybe export the commitments through some package
7788
+ */
7789
+ class UseMcpCommitmentDefinition extends BaseCommitmentDefinition {
7790
+ constructor() {
7791
+ super('USE MCP', ['MCP']);
7792
+ }
7793
+ /**
7794
+ * Short one-line description of USE MCP.
7795
+ */
7796
+ get description() {
7797
+ return 'Connects the agent to an external MCP server for additional capabilities.';
7798
+ }
7799
+ /**
7800
+ * Icon for this commitment.
7801
+ */
7802
+ get icon() {
7803
+ return '🔌';
7804
+ }
7805
+ /**
7806
+ * Markdown documentation for USE MCP commitment.
7807
+ */
7808
+ get documentation() {
7809
+ return spaceTrim$1(`
7810
+ # USE MCP
7811
+
7812
+ Connects the agent to an external Model Context Protocol (MCP) server.
7813
+
7814
+ ## Key aspects
7815
+
7816
+ - The content following \`USE MCP\` must be a valid URL
7817
+ - Multiple MCP servers can be connected by using multiple \`USE MCP\` commitments
7818
+ - The agent will have access to tools and resources provided by the MCP server
7819
+
7820
+ ## Example
7821
+
7822
+ \`\`\`book
7823
+ Company Lawyer
7824
+
7825
+ PERSONA You are a company lawyer.
7826
+ USE MCP http://legal-db.example.com
7827
+ \`\`\`
7828
+ `);
7829
+ }
7830
+ applyToAgentModelRequirements(requirements, content) {
7831
+ const mcpServerUrl = content.trim();
7832
+ if (!mcpServerUrl) {
7833
+ return requirements;
7834
+ }
7835
+ const existingMcpServers = requirements.mcpServers || [];
7836
+ // Avoid duplicates
7837
+ if (existingMcpServers.includes(mcpServerUrl)) {
7838
+ return requirements;
7839
+ }
7840
+ return {
7841
+ ...requirements,
7842
+ mcpServers: [...existingMcpServers, mcpServerUrl],
7843
+ };
7844
+ }
7845
+ }
7846
+ /**
7847
+ * Note: [💞] Ignore a discrepancy between file name and entity name
7848
+ */
7849
+
7850
+ /**
7851
+ * A search engine implementation that uses the SerpApi to fetch Google search results.
7852
+ *
7853
+ * @private <- TODO: !!!! Export via some package
7854
+ */
7855
+ class SerpSearchEngine {
7856
+ get title() {
7857
+ return 'SerpApi Search Engine';
7858
+ }
7859
+ get description() {
7860
+ return 'Search engine that uses SerpApi to fetch Google search results';
7861
+ }
7862
+ checkConfiguration() {
7863
+ if (!process.env.SERP_API_KEY) {
7864
+ throw new Error('SERP_API_KEY is not configured');
7865
+ }
7866
+ }
7867
+ async search(query, options = {}) {
7868
+ const apiKey = process.env.SERP_API_KEY;
7869
+ if (!apiKey) {
7870
+ throw new Error('SERP_API_KEY is not configured');
7871
+ }
7872
+ const url = new URL('https://serpapi.com/search');
7873
+ url.searchParams.set('api_key', apiKey);
7874
+ url.searchParams.set('engine', 'google');
7875
+ url.searchParams.set('q', query);
7876
+ for (const [key, value] of Object.entries(options)) {
7877
+ url.searchParams.set(key, String(value));
7878
+ }
7879
+ const response = await fetch(url.toString());
7880
+ if (!response.ok) {
7881
+ const body = await response.text();
7882
+ throw new Error(`SerpApi failed with status ${response.status}: ${response.statusText}\n${body}`);
7883
+ }
7884
+ const data = (await response.json());
7885
+ return (data.organic_results || []).map((item) => ({
7886
+ title: item.title,
7887
+ url: item.link,
7888
+ snippet: item.snippet || '',
7889
+ }));
7890
+ }
7891
+ }
7892
+
6896
7893
  /**
6897
7894
  * USE SEARCH ENGINE commitment definition
6898
7895
  *
@@ -6912,7 +7909,7 @@ class UseMcpCommitmentDefinition extends BaseCommitmentDefinition {
6912
7909
  */
6913
7910
  class UseSearchEngineCommitmentDefinition extends BaseCommitmentDefinition {
6914
7911
  constructor() {
6915
- super('USE SEARCH ENGINE', ['SEARCH ENGINE', 'SEARCH']);
7912
+ super('USE SEARCH ENGINE', ['USE SEARCH']);
6916
7913
  }
6917
7914
  /**
6918
7915
  * Short one-line description of USE SEARCH ENGINE.
@@ -6969,18 +7966,13 @@ class UseSearchEngineCommitmentDefinition extends BaseCommitmentDefinition {
6969
7966
  ? existingTools
6970
7967
  : [
6971
7968
  ...existingTools,
6972
- { type: 'web_search' },
6973
- // <- Note: [🔰] This is just using simple native search tool by OpenAI @see https://platform.openai.com/docs/guides/tools-web-search
6974
- // In future we will use proper MCP search tool:
6975
- /*
6976
-
6977
7969
  {
6978
7970
  name: 'web_search',
6979
- description: spaceTrim(`
6980
- Search the internet for information.
6981
- Use this tool when you need to find up-to-date information or facts that you don't know.
6982
- ${!content ? '' : `Search scope / instructions: ${content}`}
6983
- `),
7971
+ description: spaceTrim$1(`
7972
+ Search the internet for information.
7973
+ Use this tool when you need to find up-to-date information or facts that you don't know.
7974
+ ${!content ? '' : `Search scope / instructions: ${content}`}
7975
+ `),
6984
7976
  parameters: {
6985
7977
  type: 'object',
6986
7978
  properties: {
@@ -6988,20 +7980,229 @@ class UseSearchEngineCommitmentDefinition extends BaseCommitmentDefinition {
6988
7980
  type: 'string',
6989
7981
  description: 'The search query',
6990
7982
  },
7983
+ location: {
7984
+ type: 'string',
7985
+ description: 'The location for the search (e.g., "Austin, Texas, United States" or "Prague, Czechia")',
7986
+ },
7987
+ gl: {
7988
+ type: 'string',
7989
+ description: 'The country code (e.g., "us" for United States, "cz" for Czechia)',
7990
+ },
7991
+ hl: {
7992
+ type: 'string',
7993
+ description: 'The language code (e.g., "en" for English, "cs" for Czech)',
7994
+ },
7995
+ num: {
7996
+ type: 'integer',
7997
+ description: 'Number of results to return',
7998
+ },
7999
+ engine: {
8000
+ type: 'string',
8001
+ description: 'The search engine to use (e.g., "google", "bing", "yahoo", "baidu")',
8002
+ },
8003
+ google_domain: {
8004
+ type: 'string',
8005
+ description: 'The Google domain to use (e.g., "google.com", "google.cz")',
8006
+ },
6991
8007
  },
6992
8008
  required: ['query'],
6993
8009
  },
6994
8010
  },
6995
- */
6996
8011
  ];
6997
8012
  // Return requirements with updated tools and metadata
6998
- return {
8013
+ return this.appendToSystemMessage({
6999
8014
  ...requirements,
7000
8015
  tools: updatedTools,
7001
8016
  metadata: {
7002
8017
  ...requirements.metadata,
7003
8018
  useSearchEngine: content || true,
7004
8019
  },
8020
+ }, spaceTrim$1(`
8021
+ Tools:
8022
+ You have access to the web search engine via the tool "web_search".
8023
+ Use it to find up-to-date information or facts that you don't know.
8024
+ When you need to know some information from the internet, use the tool provided to you.
8025
+ Do not make up information when you can search for it.
8026
+ Do not tell the user you cannot search for information, YOU CAN.
8027
+ `));
8028
+ }
8029
+ /**
8030
+ * Gets human-readable titles for tool functions provided by this commitment.
8031
+ */
8032
+ getToolTitles() {
8033
+ return {
8034
+ web_search: 'Web search',
8035
+ };
8036
+ }
8037
+ /**
8038
+ * Gets the `web_search` tool function implementation.
8039
+ */
8040
+ getToolFunctions() {
8041
+ return {
8042
+ async web_search(args) {
8043
+ console.log('!!!! [Tool] web_search called', { args });
8044
+ const { query, ...options } = args;
8045
+ if (!query) {
8046
+ throw new Error('Search query is required');
8047
+ }
8048
+ const searchEngine = new SerpSearchEngine();
8049
+ const results = await searchEngine.search(query, options);
8050
+ return spaceTrim$1((block) => `
8051
+ Search results for "${query}"${Object.keys(options).length === 0 ? '' : ` with options ${JSON.stringify(options)}`}:
8052
+
8053
+ ${block(results
8054
+ .map((result) => spaceTrim$1(`
8055
+ - **${result.title}**
8056
+ ${result.url}
8057
+ ${result.snippet}
8058
+ `))
8059
+ .join('\n\n'))}
8060
+ `);
8061
+ },
8062
+ };
8063
+ }
8064
+ }
8065
+ /**
8066
+ * Note: [💞] Ignore a discrepancy between file name and entity name
8067
+ */
8068
+
8069
+ /**
8070
+ * USE TIME commitment definition
8071
+ *
8072
+ * The `USE TIME` commitment indicates that the agent should be able to determine the current date and time.
8073
+ *
8074
+ * Example usage in agent source:
8075
+ *
8076
+ * ```book
8077
+ * USE TIME
8078
+ * ```
8079
+ *
8080
+ * @private [🪔] Maybe export the commitments through some package
8081
+ */
8082
+ class UseTimeCommitmentDefinition extends BaseCommitmentDefinition {
8083
+ constructor() {
8084
+ super('USE TIME', ['CURRENT TIME', 'TIME', 'DATE']);
8085
+ }
8086
+ /**
8087
+ * Short one-line description of USE TIME.
8088
+ */
8089
+ get description() {
8090
+ return 'Enable the agent to determine the current date and time.';
8091
+ }
8092
+ /**
8093
+ * Icon for this commitment.
8094
+ */
8095
+ get icon() {
8096
+ return '🕒';
8097
+ }
8098
+ /**
8099
+ * Markdown documentation for USE TIME commitment.
8100
+ */
8101
+ get documentation() {
8102
+ return spaceTrim$1(`
8103
+ # USE TIME
8104
+
8105
+ Enables the agent to determine the current date and time.
8106
+
8107
+ ## Key aspects
8108
+
8109
+ - This tool won't receive any input.
8110
+ - It outputs the current date and time as an ISO 8601 string.
8111
+ - Allows the agent to answer questions about the current time or date.
8112
+
8113
+ ## Examples
8114
+
8115
+ \`\`\`book
8116
+ Time-aware Assistant
8117
+
8118
+ PERSONA You are a helpful assistant who knows the current time.
8119
+ USE TIME
8120
+ \`\`\`
8121
+ `);
8122
+ }
8123
+ applyToAgentModelRequirements(requirements, content) {
8124
+ // Get existing tools array or create new one
8125
+ const existingTools = requirements.tools || [];
8126
+ // Add 'get_current_time' to tools if not already present
8127
+ const updatedTools = existingTools.some((tool) => tool.name === 'get_current_time')
8128
+ ? existingTools
8129
+ : [
8130
+ ...existingTools,
8131
+ {
8132
+ name: 'get_current_time',
8133
+ description: 'Get the current date and time in ISO 8601 format.',
8134
+ parameters: {
8135
+ type: 'object',
8136
+ properties: {
8137
+ timezone: {
8138
+ type: 'string',
8139
+ description: 'Optional timezone name (e.g. "Europe/Prague", "UTC", "America/New_York").',
8140
+ },
8141
+ },
8142
+ required: [],
8143
+ },
8144
+ },
8145
+ // <- TODO: !!!! define the function in LLM tools
8146
+ ];
8147
+ // Return requirements with updated tools and metadata
8148
+ return this.appendToSystemMessage({
8149
+ ...requirements,
8150
+ tools: updatedTools,
8151
+ metadata: {
8152
+ ...requirements.metadata,
8153
+ },
8154
+ }, spaceTrim$1(`
8155
+ Tool:
8156
+ You have access to the current date and time via the tool "get_current_time".
8157
+ Use it to answer questions about the current date and time.
8158
+ When you need to know the current date or time, use the tool provided to you.
8159
+ Do not make up the current date or time; always use the tool to get accurate information.
8160
+ `));
8161
+ }
8162
+ /**
8163
+ * Gets human-readable titles for tool functions provided by this commitment.
8164
+ */
8165
+ getToolTitles() {
8166
+ return {
8167
+ get_current_time: 'Get current time',
8168
+ };
8169
+ }
8170
+ /**
8171
+ * Gets the `get_current_time` tool function implementation.
8172
+ */
8173
+ getToolFunctions() {
8174
+ return {
8175
+ async get_current_time(args) {
8176
+ var _a;
8177
+ console.log('!!!! [Tool] get_current_time called', { args });
8178
+ const { timezone } = args;
8179
+ if (!timezone) {
8180
+ return new Date().toISOString();
8181
+ }
8182
+ try {
8183
+ // Note: Returning ISO 8601 string but in the requested timezone
8184
+ const formatter = new Intl.DateTimeFormat('en-CA', {
8185
+ timeZone: timezone,
8186
+ year: 'numeric',
8187
+ month: '2-digit',
8188
+ day: '2-digit',
8189
+ hour: '2-digit',
8190
+ minute: '2-digit',
8191
+ second: '2-digit',
8192
+ hour12: false,
8193
+ timeZoneName: 'shortOffset',
8194
+ });
8195
+ const parts = formatter.formatToParts(new Date());
8196
+ const part = (type) => { var _a; return (_a = parts.find((p) => p.type === type)) === null || _a === void 0 ? void 0 : _a.value; };
8197
+ // en-CA format is YYYY-MM-DD
8198
+ const isoString = `${part('year')}-${part('month')}-${part('day')}T${part('hour')}:${part('minute')}:${part('second')}${(_a = part('timeZoneName')) === null || _a === void 0 ? void 0 : _a.replace('GMT', '')}`;
8199
+ return isoString;
8200
+ }
8201
+ catch (error) {
8202
+ // Fallback to UTC if timezone is invalid
8203
+ return new Date().toISOString();
8204
+ }
8205
+ },
7005
8206
  };
7006
8207
  }
7007
8208
  }
@@ -7076,7 +8277,6 @@ class NotYetImplementedCommitmentDefinition extends BaseCommitmentDefinition {
7076
8277
  }
7077
8278
  }
7078
8279
 
7079
- // Import all commitment definition classes
7080
8280
  /**
7081
8281
  * Registry of all available commitment definitions
7082
8282
  * This array contains instances of all commitment definitions
@@ -7093,10 +8293,10 @@ const COMMITMENT_REGISTRY = [
7093
8293
  new MemoryCommitmentDefinition('MEMORIES'),
7094
8294
  new StyleCommitmentDefinition('STYLE'),
7095
8295
  new StyleCommitmentDefinition('STYLES'),
7096
- new RuleCommitmentDefinition('RULE'),
7097
8296
  new RuleCommitmentDefinition('RULES'),
7098
- new LanguageCommitmentDefinition('LANGUAGE'),
8297
+ new RuleCommitmentDefinition('RULE'),
7099
8298
  new LanguageCommitmentDefinition('LANGUAGES'),
8299
+ new LanguageCommitmentDefinition('LANGUAGE'),
7100
8300
  new SampleCommitmentDefinition('SAMPLE'),
7101
8301
  new SampleCommitmentDefinition('EXAMPLE'),
7102
8302
  new FormatCommitmentDefinition('FORMAT'),
@@ -7118,6 +8318,7 @@ const COMMITMENT_REGISTRY = [
7118
8318
  new NoteCommitmentDefinition('NOTES'),
7119
8319
  new NoteCommitmentDefinition('COMMENT'),
7120
8320
  new NoteCommitmentDefinition('NONCE'),
8321
+ new NoteCommitmentDefinition('TODO'),
7121
8322
  new GoalCommitmentDefinition('GOAL'),
7122
8323
  new GoalCommitmentDefinition('GOALS'),
7123
8324
  new InitialMessageCommitmentDefinition(),
@@ -7136,6 +8337,16 @@ const COMMITMENT_REGISTRY = [
7136
8337
  new ClosedCommitmentDefinition(),
7137
8338
  new UseBrowserCommitmentDefinition(),
7138
8339
  new UseSearchEngineCommitmentDefinition(),
8340
+ new UseTimeCommitmentDefinition(),
8341
+ new UseImageGeneratorCommitmentDefinition('USE IMAGE GENERATOR'),
8342
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
8343
+ new UseImageGeneratorCommitmentDefinition('USE IMAGE GENERATION'),
8344
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
8345
+ new UseImageGeneratorCommitmentDefinition('IMAGE GENERATOR'),
8346
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
8347
+ new UseImageGeneratorCommitmentDefinition('IMAGE GENERATION'),
8348
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
8349
+ new UseImageGeneratorCommitmentDefinition('USE IMAGE'),
7139
8350
  new UseMcpCommitmentDefinition(),
7140
8351
  new UseCommitmentDefinition(),
7141
8352
  // Not yet implemented commitments (using placeholder)
@@ -7145,6 +8356,7 @@ const COMMITMENT_REGISTRY = [
7145
8356
  new NotYetImplementedCommitmentDefinition('AVOID'),
7146
8357
  new NotYetImplementedCommitmentDefinition('AVOIDANCE'),
7147
8358
  new NotYetImplementedCommitmentDefinition('CONTEXT'),
8359
+ // <- TODO: Prompt: Leverage aliases instead of duplicating commitment definitions
7148
8360
  ];
7149
8361
  /**
7150
8362
  * Gets a commitment definition by its type
@@ -7165,6 +8377,21 @@ function getCommitmentDefinition(type) {
7165
8377
  function getAllCommitmentDefinitions() {
7166
8378
  return $deepFreeze([...COMMITMENT_REGISTRY]);
7167
8379
  }
8380
+ /**
8381
+ * Gets all tool titles provided by all commitments
8382
+ *
8383
+ * @public exported from `@promptbook/core`
8384
+ */
8385
+ function getAllCommitmentsToolTitles() {
8386
+ const allToolTitles = {};
8387
+ for (const commitmentDefinition of getAllCommitmentDefinitions()) {
8388
+ const toolTitles = commitmentDefinition.getToolTitles();
8389
+ for (const [funcName, title] of Object.entries(toolTitles)) {
8390
+ allToolTitles[funcName] = title;
8391
+ }
8392
+ }
8393
+ return allToolTitles;
8394
+ }
7168
8395
  /**
7169
8396
  * TODO: [🧠] Maybe create through standardized $register
7170
8397
  * Note: [💞] Ignore a discrepancy between file name and entity name
@@ -7248,11 +8475,37 @@ function parseAgentSourceWithCommitments(agentSource) {
7248
8475
  let currentCommitment = null;
7249
8476
  // Process lines starting from after the agent name line
7250
8477
  const startIndex = agentNameLineIndex >= 0 ? agentNameLineIndex + 1 : 0;
8478
+ let isInsideCodeBlock = false;
7251
8479
  for (let i = startIndex; i < lines.length; i++) {
7252
8480
  const line = lines[i];
7253
8481
  if (line === undefined) {
7254
8482
  continue;
7255
8483
  }
8484
+ const trimmedLine = line.trim();
8485
+ // Check if this line starts or ends a code block
8486
+ if (trimmedLine.startsWith('```')) {
8487
+ isInsideCodeBlock = !isInsideCodeBlock;
8488
+ if (currentCommitment) {
8489
+ // If we are inside a commitment, the code block is part of it
8490
+ currentCommitment.contentLines.push(line);
8491
+ }
8492
+ else {
8493
+ // If we are not inside a commitment, the code block is non-commitment
8494
+ nonCommitmentLines.push(line);
8495
+ }
8496
+ continue;
8497
+ }
8498
+ if (isInsideCodeBlock) {
8499
+ if (currentCommitment) {
8500
+ // If we are inside a commitment and a code block, the line is part of the commitment
8501
+ currentCommitment.contentLines.push(line);
8502
+ }
8503
+ else {
8504
+ // If we are inside a code block but not a commitment, the line is non-commitment
8505
+ nonCommitmentLines.push(line);
8506
+ }
8507
+ continue;
8508
+ }
7256
8509
  // Check if this line starts a new commitment
7257
8510
  let foundNewCommitment = false;
7258
8511
  for (const definition of COMMITMENT_REGISTRY) {
@@ -7426,7 +8679,24 @@ function parseAgentSource(agentSource) {
7426
8679
  const meta = {};
7427
8680
  const links = [];
7428
8681
  const capabilities = [];
8682
+ const samples = [];
8683
+ let pendingUserMessage = null;
7429
8684
  for (const commitment of parseResult.commitments) {
8685
+ if (commitment.type === 'INITIAL MESSAGE') {
8686
+ samples.push({ question: null, answer: commitment.content });
8687
+ continue;
8688
+ }
8689
+ if (commitment.type === 'USER MESSAGE') {
8690
+ pendingUserMessage = commitment.content;
8691
+ continue;
8692
+ }
8693
+ if (commitment.type === 'AGENT MESSAGE') {
8694
+ if (pendingUserMessage !== null) {
8695
+ samples.push({ question: pendingUserMessage, answer: commitment.content });
8696
+ pendingUserMessage = null;
8697
+ }
8698
+ continue;
8699
+ }
7430
8700
  if (commitment.type === 'USE BROWSER') {
7431
8701
  capabilities.push({
7432
8702
  type: 'browser',
@@ -7438,11 +8708,84 @@ function parseAgentSource(agentSource) {
7438
8708
  if (commitment.type === 'USE SEARCH ENGINE') {
7439
8709
  capabilities.push({
7440
8710
  type: 'search-engine',
7441
- label: 'Search Internet',
8711
+ label: 'Internet',
7442
8712
  iconName: 'Search',
7443
8713
  });
7444
8714
  continue;
7445
8715
  }
8716
+ if (commitment.type === 'USE SEARCH') {
8717
+ capabilities.push({
8718
+ type: 'search-engine',
8719
+ label: 'Internet',
8720
+ iconName: 'Search',
8721
+ });
8722
+ continue;
8723
+ }
8724
+ if (commitment.type === 'USE TIME') {
8725
+ capabilities.push({
8726
+ type: 'time',
8727
+ label: 'Time',
8728
+ iconName: 'Clock',
8729
+ });
8730
+ continue;
8731
+ }
8732
+ if (commitment.type === 'USE IMAGE GENERATOR') {
8733
+ capabilities.push({
8734
+ type: 'image-generator',
8735
+ label: 'Image Generator',
8736
+ iconName: 'Image',
8737
+ });
8738
+ continue;
8739
+ }
8740
+ if (commitment.type === 'FROM') {
8741
+ const content = spaceTrim$2(commitment.content).split('\n')[0] || '';
8742
+ if (content === 'Adam' || content === '' /* <- Note: Adam is implicit */) {
8743
+ continue;
8744
+ }
8745
+ let label = content;
8746
+ let iconName = 'SquareArrowOutUpRight'; // Inheritance remote
8747
+ if (content.startsWith('./') || content.startsWith('../') || content.startsWith('/')) {
8748
+ label = content.split('/').pop() || content;
8749
+ iconName = 'SquareArrowUpRight'; // Inheritance local
8750
+ }
8751
+ if (content === 'VOID') {
8752
+ label = 'VOID';
8753
+ iconName = 'ShieldAlert'; // [🧠] Or some other icon for VOID
8754
+ }
8755
+ capabilities.push({
8756
+ type: 'inheritance',
8757
+ label,
8758
+ iconName,
8759
+ agentUrl: content,
8760
+ });
8761
+ continue;
8762
+ }
8763
+ if (commitment.type === 'IMPORT') {
8764
+ const content = spaceTrim$2(commitment.content).split('\n')[0] || '';
8765
+ let label = content;
8766
+ let iconName = 'ExternalLink'; // Import remote
8767
+ try {
8768
+ if (content.startsWith('http://') || content.startsWith('https://')) {
8769
+ const url = new URL(content);
8770
+ label = url.hostname.replace(/^www\./, '') + '.../' + url.pathname.split('/').pop();
8771
+ iconName = 'ExternalLink';
8772
+ }
8773
+ else if (content.startsWith('./') || content.startsWith('../') || content.startsWith('/')) {
8774
+ label = content.split('/').pop() || content;
8775
+ iconName = 'Link'; // Import local
8776
+ }
8777
+ }
8778
+ catch (e) {
8779
+ // Invalid URL or path, keep default label
8780
+ }
8781
+ capabilities.push({
8782
+ type: 'import',
8783
+ label,
8784
+ iconName,
8785
+ agentUrl: content,
8786
+ });
8787
+ continue;
8788
+ }
7446
8789
  if (commitment.type === 'KNOWLEDGE') {
7447
8790
  const content = spaceTrim$2(commitment.content).split('\n')[0] || '';
7448
8791
  let label = content;
@@ -7527,6 +8870,7 @@ function parseAgentSource(agentSource) {
7527
8870
  links,
7528
8871
  parameters,
7529
8872
  capabilities,
8873
+ samples,
7530
8874
  };
7531
8875
  }
7532
8876
  /**
@@ -8596,13 +9940,17 @@ function BookEditorMonaco(props) {
8596
9940
  // Register a new language
8597
9941
  monaco.languages.register({ id: BOOK_LANGUAGE_ID });
8598
9942
  const commitmentTypes = [...new Set(getAllCommitmentDefinitions().map(({ type }) => type))];
8599
- const commitmentRegex = new RegExp(`^(${commitmentTypes.map((type) => (type === 'META' ? 'META\\s+\\w+' : type)).join('|')})`);
9943
+ const commitmentRegex = new RegExp(`^\\s*(${commitmentTypes
9944
+ .sort((a, b) => b.length - a.length) // [1] Prefer longer commitments to avoid partial matching (e.g. LANGUAGES vs LANGUAGE)
9945
+ .map((type) => (type === 'META' ? 'META\\s+\\w+' : type.replace(/\s+/, '\\s+')))
9946
+ .join('|')})(?=\\s|$)`);
8600
9947
  // Note: Using a broad character set for Latin and Cyrillic to support international characters in parameters.
8601
9948
  // Monarch tokenizer does not support Unicode property escapes like \p{L}.
8602
9949
  const parameterRegex = /@([a-zA-Z0-9_á-žÁ-Žč-řČ-Řš-žŠ-Žа-яА-ЯёЁ]+)/;
8603
9950
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
8604
9951
  const bookRules = [
8605
9952
  [/^---[-]*$/, ''],
9953
+ [/^```.*$/, 'code-block', '@codeblock'],
8606
9954
  [parameterRegex, 'parameter'],
8607
9955
  [/\{[^}]+\}/, 'parameter'],
8608
9956
  [commitmentRegex, 'commitment'],
@@ -8614,10 +9962,15 @@ function BookEditorMonaco(props) {
8614
9962
  root: [
8615
9963
  [/^\s*$/, 'empty'],
8616
9964
  [/^-*$/, 'line'],
9965
+ [/^```.*$/, 'code-block', '@codeblock'],
8617
9966
  [/^.*$/, 'title', '@body'],
8618
9967
  [commitmentRegex, 'commitment'],
8619
9968
  ],
8620
9969
  body: bookRules,
9970
+ codeblock: [
9971
+ [/^```.*$/, 'code-block', '@pop'],
9972
+ [/^.*$/, 'code-block'],
9973
+ ],
8621
9974
  },
8622
9975
  });
8623
9976
  // Register a completion item provider for the language
@@ -8659,6 +10012,10 @@ function BookEditorMonaco(props) {
8659
10012
  foreground: PROMPTBOOK_SYNTAX_COLORS.PARAMETER.toHex(),
8660
10013
  fontStyle: `italic`,
8661
10014
  },
10015
+ {
10016
+ token: 'code-block',
10017
+ foreground: PROMPTBOOK_SYNTAX_COLORS.CODE_BLOCK.toHex(),
10018
+ },
8662
10019
  ],
8663
10020
  colors: {
8664
10021
  'editor.scrollbarSlider.background': '#E0E0E0',
@@ -8715,12 +10072,35 @@ function BookEditorMonaco(props) {
8715
10072
  .${instanceClass} .monaco-editor .transparent-text {
8716
10073
  color: transparent !important;
8717
10074
  }
10075
+
10076
+ .${instanceClass} .monaco-editor .code-block-box {
10077
+ background-color: #f5f5f566;
10078
+ border-left: 1px solid ${PROMPTBOOK_SYNTAX_COLORS.CODE_BLOCK.toHex()};
10079
+ border-right: 1px solid ${PROMPTBOOK_SYNTAX_COLORS.CODE_BLOCK.toHex()};
10080
+ padding-left: ${Math.round(8 * zoomLevel)}px;
10081
+ padding-right: ${Math.round(8 * zoomLevel)}px;
10082
+ }
10083
+
10084
+ .${instanceClass} .monaco-editor .code-block-top {
10085
+ border-top: 1px solid ${PROMPTBOOK_SYNTAX_COLORS.CODE_BLOCK.toHex()};
10086
+ border-top-left-radius: ${Math.round(10 * zoomLevel)}px;
10087
+ border-top-right-radius: ${Math.round(10 * zoomLevel)}px;
10088
+ overflow: hidden;
10089
+ }
10090
+
10091
+ .${instanceClass} .monaco-editor .code-block-bottom {
10092
+ border-bottom: 1px solid ${PROMPTBOOK_SYNTAX_COLORS.CODE_BLOCK.toHex()};
10093
+ border-bottom-left-radius: ${Math.round(10 * zoomLevel)}px;
10094
+ border-bottom-right-radius: ${Math.round(10 * zoomLevel)}px;
10095
+ overflow: hidden;
10096
+ }
8718
10097
  `;
8719
10098
  return () => {
8720
10099
  // Note: Style is not removed on purpose to avoid flickering during development with fast refresh
8721
10100
  };
8722
10101
  }, [scaledLineHeight, scaledContentPaddingLeft, scaledVerticalLineLeft]);
8723
10102
  const decorationIdsRef = useRef([]);
10103
+ const codeBlockDecorationIdsRef = useRef([]);
8724
10104
  useEffect(() => {
8725
10105
  if (!editor || !monaco) {
8726
10106
  return;
@@ -8749,6 +10129,39 @@ function BookEditorMonaco(props) {
8749
10129
  });
8750
10130
  }
8751
10131
  decorationIdsRef.current = editor.deltaDecorations(decorationIdsRef.current, newDecorations);
10132
+ // Add decorations for code blocks
10133
+ const lines = text.split('\n');
10134
+ const codeBlockDecorations = [];
10135
+ let inCodeBlock = false;
10136
+ let codeBlockStartLine = 0;
10137
+ for (let i = 0; i < lines.length; i++) {
10138
+ const line = lines[i];
10139
+ if (line === null || line === void 0 ? void 0 : line.trim().startsWith('```')) {
10140
+ if (!inCodeBlock) {
10141
+ // Starting a code block
10142
+ inCodeBlock = true;
10143
+ codeBlockStartLine = i + 1; // 1-based line number
10144
+ }
10145
+ else {
10146
+ // Ending a code block
10147
+ inCodeBlock = false;
10148
+ const endLine = i + 1; // 1-based line number
10149
+ // Add decorations for each line in the code block
10150
+ for (let j = codeBlockStartLine; j <= endLine; j++) {
10151
+ const isFirst = j === codeBlockStartLine;
10152
+ const isLast = j === endLine;
10153
+ codeBlockDecorations.push({
10154
+ range: new monaco.Range(j, 1, j, 1),
10155
+ options: {
10156
+ isWholeLine: true,
10157
+ className: `code-block-box${isFirst ? ' code-block-top' : ''}${isLast ? ' code-block-bottom' : ''}`,
10158
+ },
10159
+ });
10160
+ }
10161
+ }
10162
+ }
10163
+ }
10164
+ codeBlockDecorationIdsRef.current = editor.deltaDecorations(codeBlockDecorationIdsRef.current, codeBlockDecorations);
8752
10165
  };
8753
10166
  updateDecorations();
8754
10167
  const changeListener = editor.onDidChangeModelContent(() => {
@@ -9762,8 +11175,8 @@ function getChatSaveFormatDefinitions(formatNames) {
9762
11175
  return CHAT_SAVE_FORMATS.filter((saveFormatDefinition) => formatNames.includes(saveFormatDefinition.formatName));
9763
11176
  }
9764
11177
 
9765
- 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}.Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ{margin-top:55px}@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%,49%,.6),transparent);border-radius:2px;bottom:8px;content:\"\";height:4px;position:absolute;right:12px;width:20px}@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{backdrop-filter:blur(20px);border-top:1px solid hsla(0,0%,49%,.1);display:flex;flex-direction:column;gap:12px;grid-area:📝;padding:20px;position:relative;width:100%;z-index:10}.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));border-top:2px solid rgba(0,132,255,.3)}.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:center;display:flex;gap:12px}.Chat-module_inputContainer__bPt99 textarea::placeholder{color:inherit;opacity:.7}.Chat-module_attachmentButton__qLO47{align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.05);border:1px solid hsla(0,0%,49%,.2);border-radius:50%;color:#6b7280;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;transition:all .2s ease;width:40px}.Chat-module_attachmentButton__qLO47:hover:not(:disabled){background:rgba(0,132,255,.1);border-color:rgba(0,132,255,.3);color:#0084ff;transform:scale(1.05)}.Chat-module_attachmentButton__qLO47:disabled{cursor:not-allowed;opacity:.5;transform:none}.Chat-module_voiceButton__d2zlP{align-items:center;backdrop-filter:blur(10px);background:hsla(0,0%,49%,.05);border:1px solid hsla(0,0%,49%,.2);border-radius:50%;color:#6b7280;cursor:pointer;display:flex;flex-shrink:0;height:40px;justify-content:center;transition:all .2s ease;width:40px}.Chat-module_voiceButton__d2zlP:hover{background:rgba(0,132,255,.1);border-color:rgba(0,132,255,.3);color:#0084ff;transform:scale(1.05)}.Chat-module_voiceButtonActive__Uoi3W{animation:Chat-module_voiceRecordingPulse__y2wJ5 1.5s infinite;background:rgba(255,0,0,.1)!important;border-color:rgba(255,0,0,.3)!important;color:red!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;backdrop-filter:blur(20px);background:hsla(0,0%,49%,.05);border-radius:25px;color:#000;flex:1;font-size:15px;line-height:1.4;max-height:120px;min-width:200px;outline:none;padding:16px 20px;resize:none;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94)}.Chat-module_chatMainFlow__--8FE .Chat-module_chatInput__1Ecan textarea:focus{background:hsla(0,0%,49%,.08);border-color:#0084ff;box-shadow:0 0 0 4px rgba(0,132,255,.1),0 8px 32px rgba(0,132,255,.15);transform:translateY(-2px)}.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{-webkit-tap-highlight-color:transparent;align-items:center;aspect-ratio:1/1;border:none;border-radius:50%!important;box-shadow:0 4px 16px rgba(0,132,255,.3);color:#fff;display:flex;height:48px;justify-content:center;margin:0!important;min-height:unset!important;min-width:unset!important;overflow:visible;padding:0!important;padding-left:unset;padding-right:unset;touch-action:manipulation;transition:all .3s cubic-bezier(.25,.46,.45,.94);user-select:none;width:48px}.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_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/* Add top margin to first message if actions are present and first message is long */\n.hasActionsAndFirstMessageIsLong {\n    margin-top: 55px;\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(125, 125, 125, 0.6), transparent);\n    border-radius: 2px;\n    animation: loadingPulse 1.5s ease-in-out infinite;\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: 20px;\n    backdrop-filter: blur(20px);\n    border-top: 1px solid rgba(125, 125, 125, 0.1);\n    display: flex;\n    flex-direction: column;\n    gap: 12px;\n    position: relative;\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    border-top: 2px solid rgba(0, 132, 255, 0.3);\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: center;\n    gap: 12px;\n}\n\n.inputContainer textarea::placeholder {\n    color: inherit;\n    opacity: 0.7;\n}\n\n/* Attachment button */\n.attachmentButton {\n    width: 40px;\n    height: 40px;\n    border: 1px solid rgba(125, 125, 125, 0.2);\n    background: rgba(125, 125, 125, 0.05);\n    color: #6b7280;\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    backdrop-filter: blur(10px);\n    flex-shrink: 0;\n}\n\n.attachmentButton:hover:not(:disabled) {\n    background: rgba(0, 132, 255, 0.1);\n    border-color: rgba(0, 132, 255, 0.3);\n    color: #0084ff;\n    transform: scale(1.05);\n}\n\n.attachmentButton:disabled {\n    opacity: 0.5;\n    cursor: not-allowed;\n    transform: none;\n}\n\n/* Voice Button */\n.voiceButton {\n    width: 40px;\n    height: 40px;\n    border: 1px solid rgba(125, 125, 125, 0.2);\n    background: rgba(125, 125, 125, 0.05);\n    color: #6b7280;\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    backdrop-filter: blur(10px);\n    flex-shrink: 0;\n}\n\n.voiceButton:hover {\n    background: rgba(0, 132, 255, 0.1);\n    border-color: rgba(0, 132, 255, 0.3);\n    color: #0084ff;\n    transform: scale(1.05);\n}\n\n.voiceButtonActive {\n    background: rgba(255, 0, 0, 0.1) !important;\n    border-color: rgba(255, 0, 0, 0.3) !important;\n    color: #ff0000 !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: 16px 20px;\n\n    outline: none;\n    border-radius: 25px;\n    background: rgba(125, 125, 125, 0.05);\n    backdrop-filter: blur(20px);\n    color: black;\n    min-width: 200px;\n    max-height: 120px;\n    font-size: 15px;\n    line-height: 1.4;\n    resize: none;\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\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    border-color: #0084ff;\n    background: rgba(125, 125, 125, 0.08);\n    box-shadow: 0 0 0 4px rgba(0, 132, 255, 0.1), 0 8px 32px rgba(0, 132, 255, 0.15);\n    transform: translateY(-2px);\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 {\n    width: 48px;\n    height: 48px;\n    margin: 0 !important;\n    padding: 0 !important;\n\n    border: none;\n    /* background: linear-gradient(135deg, #0084ff 0%, #0066cc 100%); */\n    /* <- Note: Background is set in the inline style */\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    padding-left: unset;\n    padding-right: unset;\n\n    transition: all 0.3s cubic-bezier(0.25, 0.46, 0.45, 0.94);\n    box-shadow: 0 4px 16px rgba(0, 132, 255, 0.3);\n    -webkit-tap-highlight-color: transparent;\n    touch-action: manipulation;\n    user-select: none;\n    overflow: visible;\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.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"]} */";
9766
- 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","hasActionsAndFirstMessageIsLong":"Chat-module_hasActionsAndFirstMessageIsLong__5jgoZ","isNotCompleteMessage":"Chat-module_isNotCompleteMessage__Hj2K7","NonCompleteMessageFiller":"Chat-module_NonCompleteMessageFiller__G5-Ve","messageText":"Chat-module_messageText__XgNyQ","loadingPulse":"Chat-module_loadingPulse__VomRm","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","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","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"};
11178
+ 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_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.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"]} */";
11179
+ 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","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"};
9767
11180
  styleInject(css_248z$2);
9768
11181
 
9769
11182
  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= */";
@@ -9782,6 +11195,53 @@ const AvatarProfileTooltip = forwardRef(function AvatarProfileTooltip({ agentSou
9782
11195
  }, children: [jsx("div", { className: styles$1.arrow }), jsx(AvatarProfileFromSource, { agentSource: agentSource })] }), document.body);
9783
11196
  });
9784
11197
 
11198
+ /**
11199
+ * Utility to format tool call information for user-friendly display.
11200
+ */
11201
+ /**
11202
+ * Technical to user-friendly tool names and emojis
11203
+ *
11204
+ * @private [🧠] Maybe public?
11205
+ */
11206
+ const TOOL_TITLES = {
11207
+ web_search: { title: 'Searching the web', emoji: '🔎' },
11208
+ useSearchEngine: { title: 'Searching the web', emoji: '🔎' },
11209
+ search: { title: 'Searching the web', emoji: '🔎' },
11210
+ useBrowser: { title: 'Browsing the web', emoji: '🌐' },
11211
+ browse: { title: 'Browsing the web', emoji: '🌐' },
11212
+ // Add more tools here as needed
11213
+ };
11214
+ /**
11215
+ * Gets the user-friendly text for a tool call chiplet.
11216
+ *
11217
+ * @private [🧠] Maybe public?
11218
+ */
11219
+ function getToolCallChipletText(toolCall) {
11220
+ const toolInfo = TOOL_TITLES[toolCall.name];
11221
+ const baseTitle = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title) || toolCall.name;
11222
+ const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
11223
+ let args = {};
11224
+ try {
11225
+ args = typeof toolCall.arguments === 'string' ? JSON.parse(toolCall.arguments) : toolCall.arguments || {};
11226
+ }
11227
+ catch (e) {
11228
+ // Ignore parse errors
11229
+ }
11230
+ if (args.query) {
11231
+ return `${emoji} ${args.query}`;
11232
+ }
11233
+ if (args.url) {
11234
+ try {
11235
+ const url = new URL(args.url);
11236
+ return `${emoji} ${url.hostname}`;
11237
+ }
11238
+ catch (e) {
11239
+ return `${emoji} ${args.url}`;
11240
+ }
11241
+ }
11242
+ return `${emoji} ${baseTitle}`;
11243
+ }
11244
+
9785
11245
  /**
9786
11246
  * Parses markdown buttons in the format [Button Text](?message=Message%20to%20send)
9787
11247
  * Returns both the content without buttons and the extracted buttons
@@ -9833,7 +11293,7 @@ const AVATAR_SIZE = 40;
9833
11293
  *
9834
11294
  * @private internal subcomponent of `<Chat>` component
9835
11295
  */
9836
- const ChatMessageItem = memo(({ message, participant, participants, isLastMessage, onMessage, setExpandedMessageId, isExpanded, currentRating, handleRating, mode, isCopyButtonEnabled, isFeedbackEnabled, onCopy, onCreateAgent, }) => {
11296
+ const ChatMessageItem = memo(({ message, participant, participants, isLastMessage, onMessage, setExpandedMessageId, isExpanded, currentRating, handleRating, mode, isCopyButtonEnabled, isFeedbackEnabled, onCopy, onCreateAgent, toolTitles, onToolCallClick, }) => {
9837
11297
  const avatarSrc = (participant === null || participant === void 0 ? void 0 : participant.avatarSrc) || null;
9838
11298
  const [isAvatarTooltipVisible, setIsAvatarTooltipVisible] = useState(false);
9839
11299
  const [avatarTooltipPosition, setAvatarTooltipPosition] = useState(null);
@@ -9984,7 +11444,20 @@ const ChatMessageItem = memo(({ message, participant, participants, isLastMessag
9984
11444
  ? ' ' + chatStyles.copiedTooltipLeft
9985
11445
  : tooltipAlign === 'right'
9986
11446
  ? ' ' + chatStyles.copiedTooltipRight
9987
- : ''), 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))) })), !message.isComplete && jsx("span", { className: chatStyles.NonCompleteMessageFiller, children: '_'.repeat(70) }), shouldShowButtons && (jsx("div", { className: chatStyles.messageButtons, children: buttons.map((button, buttonIndex) => (jsx("button", { className: chatStyles.messageButton, onClick: (event) => {
11447
+ : ''), 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))) })), message.completedToolCalls && message.completedToolCalls.length > 0 && (jsx("div", { className: chatStyles.completedToolCalls, children: message.completedToolCalls.map((toolCall, index) => {
11448
+ const chipletText = getToolCallChipletText(toolCall);
11449
+ return (jsxs("button", { className: chatStyles.completedToolCall, onClick: (event) => {
11450
+ event.stopPropagation();
11451
+ if (onToolCallClick) {
11452
+ onToolCallClick(toolCall);
11453
+ }
11454
+ }, children: ["[", chipletText, "]"] }, index));
11455
+ }) })), !message.isComplete && (jsx("div", { className: chatStyles.ongoingToolCalls, children: message.ongoingToolCalls && message.ongoingToolCalls.length > 0 ? (message.ongoingToolCalls.map((toolCall, index) => {
11456
+ const toolInfo = TOOL_TITLES[toolCall.name];
11457
+ const toolTitle = (toolTitles === null || toolTitles === void 0 ? void 0 : toolTitles[toolCall.name]) || (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.title);
11458
+ const emoji = (toolInfo === null || toolInfo === void 0 ? void 0 : toolInfo.emoji) || '🛠️';
11459
+ return (jsxs("div", { className: chatStyles.ongoingToolCall, children: [jsx("div", { className: chatStyles.ongoingToolCallSpinner }), jsx("span", { className: chatStyles.ongoingToolCallName, children: toolTitle ? `${emoji} ${toolTitle}...` : `${emoji} Executing ${toolCall.name}...` })] }, index));
11460
+ })) : (jsx("span", { className: chatStyles.NonCompleteMessageFiller, children: '_'.repeat(70) })) })), shouldShowButtons && (jsx("div", { className: chatStyles.messageButtons, children: buttons.map((button, buttonIndex) => (jsx("button", { className: chatStyles.messageButton, onClick: (event) => {
9988
11461
  event.stopPropagation();
9989
11462
  if (onMessage) {
9990
11463
  onMessage(button.message);
@@ -10040,6 +11513,12 @@ const ChatMessageItem = memo(({ message, participant, participants, isLastMessag
10040
11513
  if (prev.handleRating !== next.handleRating) {
10041
11514
  return false;
10042
11515
  }
11516
+ if (prev.toolTitles !== next.toolTitles) {
11517
+ return false;
11518
+ }
11519
+ if (prev.onToolCallClick !== next.onToolCallClick) {
11520
+ return false;
11521
+ }
10043
11522
  return prev.mode === next.mode;
10044
11523
  });
10045
11524
  ChatMessageItem.displayName = 'ChatMessageItem';
@@ -10059,7 +11538,7 @@ ChatMessageItem.displayName = 'ChatMessageItem';
10059
11538
  */
10060
11539
  function Chat(props) {
10061
11540
  var _a;
10062
- const { title = 'Chat', messages, onChange, onMessage, onReset, onFeedback, onFileUpload, onVoiceInput,
11541
+ const { title = 'Chat', messages, onChange, onMessage, onReset, onFeedback, onFileUpload, speechRecognition,
10063
11542
  // isVoiceRecognitionButtonShown,
10064
11543
  // voiceLanguage = 'en-US',
10065
11544
  placeholderMessageContent, defaultMessage,
@@ -10070,7 +11549,7 @@ function Chat(props) {
10070
11549
  // isExperimental = false,
10071
11550
  // TODO: [😅]> isSaveButtonEnabled = false,
10072
11551
  // exportHeaderMarkdown,
10073
- participants = [], extraActions, actionsContainer, saveFormats, isSaveButtonEnabled = true, isCopyButtonEnabled = true, buttonColor: buttonColorRaw, onUseTemplate, onCreateAgent, } = props;
11552
+ participants = [], extraActions, actionsContainer, saveFormats, isSaveButtonEnabled = true, isCopyButtonEnabled = true, buttonColor: buttonColorRaw, onUseTemplate, onCreateAgent, toolTitles, visual, } = props;
10074
11553
  const buttonColor = useMemo(() => Color.from(buttonColorRaw || '#0066cc'), [buttonColorRaw]);
10075
11554
  // Use the auto-scroll hook
10076
11555
  const { isAutoScrolling, chatMessagesRef, handleScroll, handleMessagesChange, scrollToBottom, isMobile: isMobileFromHook, } = useChatAutoScroll();
@@ -10078,6 +11557,8 @@ function Chat(props) {
10078
11557
  const buttonSendRef = useRef(null);
10079
11558
  const fileInputRef = useRef(null);
10080
11559
  const [ratingModalOpen, setRatingModalOpen] = useState(false);
11560
+ const [toolCallModalOpen, setToolCallModalOpen] = useState(false);
11561
+ const [selectedToolCall, setSelectedToolCall] = useState(null);
10081
11562
  const [selectedMessage, setSelectedMessage] = useState(null);
10082
11563
  const [messageRatings, setMessageRatings] = useState(new Map());
10083
11564
  const [textRating, setTextRating] = useState('');
@@ -10093,6 +11574,9 @@ function Chat(props) {
10093
11574
  const [uploadedFiles, setUploadedFiles] = useState([]);
10094
11575
  const [isDragOver, setIsDragOver] = useState(false);
10095
11576
  const [isUploading, setIsUploading] = useState(false);
11577
+ // Voice recognition state
11578
+ const [speechRecognitionState, setSpeechRecognitionState] = useState('IDLE');
11579
+ const [speechRecognitionText, setSpeechRecognitionText] = useState('');
10096
11580
  // Use mobile detection from the hook
10097
11581
  const isMobile = isMobileFromHook;
10098
11582
  useEffect(( /* Focus textarea on page load */) => {
@@ -10104,7 +11588,61 @@ function Chat(props) {
10104
11588
  if (isFocused) {
10105
11589
  textareaRef.current.focus();
10106
11590
  }
10107
- }, [textareaRef, isMobile]);
11591
+ }, [textareaRef, isMobile, isFocusedOnLoad]);
11592
+ // Voice recognition effects
11593
+ useEffect(() => {
11594
+ if (!speechRecognition) {
11595
+ return;
11596
+ }
11597
+ const unsubscribe = speechRecognition.subscribe((event) => {
11598
+ if (event.type === 'START') {
11599
+ setSpeechRecognitionState('RECORDING');
11600
+ setSpeechRecognitionText('');
11601
+ }
11602
+ else if (event.type === 'RESULT') {
11603
+ setSpeechRecognitionText(event.text);
11604
+ if (textareaRef.current) {
11605
+ const textarea = textareaRef.current;
11606
+ const currentValue = textarea.value;
11607
+ const lastResult = speechRecognitionText;
11608
+ // If the current value ends with the last interim result, replace it
11609
+ if (lastResult && currentValue.endsWith(lastResult)) {
11610
+ textarea.value = currentValue.slice(0, -lastResult.length) + event.text;
11611
+ }
11612
+ else {
11613
+ // Otherwise just append with a space if needed
11614
+ const separator = currentValue && !currentValue.endsWith(' ') ? ' ' : '';
11615
+ textarea.value += separator + event.text;
11616
+ }
11617
+ if (onChange) {
11618
+ onChange(textarea.value);
11619
+ }
11620
+ }
11621
+ }
11622
+ else if (event.type === 'ERROR') {
11623
+ setSpeechRecognitionState('ERROR');
11624
+ alert(`Speech recognition error: ${event.message}`);
11625
+ }
11626
+ else if (event.type === 'STOP') {
11627
+ setSpeechRecognitionState('IDLE');
11628
+ setSpeechRecognitionText('');
11629
+ }
11630
+ });
11631
+ return () => {
11632
+ unsubscribe();
11633
+ };
11634
+ }, [speechRecognition, onChange, speechRecognitionText]);
11635
+ const handleToggleVoiceInput = useCallback(() => {
11636
+ if (!speechRecognition) {
11637
+ return;
11638
+ }
11639
+ if (speechRecognition.state === 'IDLE' || speechRecognition.state === 'ERROR') {
11640
+ speechRecognition.$start({ language: /* 'en-US' */ 'en' });
11641
+ }
11642
+ else {
11643
+ speechRecognition.$stop();
11644
+ }
11645
+ }, [speechRecognition]);
10108
11646
  // File upload handlers inspired by BookEditor
10109
11647
  const handleFileUpload = useCallback(async (files) => {
10110
11648
  if (!onFileUpload)
@@ -10332,7 +11870,7 @@ function Chat(props) {
10332
11870
  const isFeedbackEnabled = !!onFeedback;
10333
11871
  // Handler for copy button
10334
11872
  const handleCopy = () => { };
10335
- return (jsxs(Fragment, { children: [ratingConfirmation && jsx("div", { className: chatStyles.ratingConfirmation, children: ratingConfirmation }), jsx("div", { className: classNames(className, chatStyles.Chat, 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 })] }) })), (() => {
11873
+ 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 })] }) })), (() => {
10336
11874
  const actionsContent = (jsxs("div", { className: classNames(actionsAlignmentClass, actionsContainer && chatStyles.portal), children: [onReset && postprocessedMessages.length !== 0 && (jsxs("button", { className: classNames(chatStyles.chatButton), onClick: () => {
10337
11875
  if (!confirm(`Do you really want to reset the chat?`)) {
10338
11876
  return;
@@ -10369,7 +11907,10 @@ function Chat(props) {
10369
11907
  const isLastMessage = i === postprocessedMessages.length - 1;
10370
11908
  const isExpanded = expandedMessageId === message.id;
10371
11909
  const currentRating = messageRatings.get(message.id || message.content /* <-[💃] */) || 0;
10372
- 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 }, message.id));
11910
+ 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) => {
11911
+ setSelectedToolCall(toolCall);
11912
+ setToolCallModalOpen(true);
11913
+ } }, message.id));
10373
11914
  }), jsx("div", {
10374
11915
  // Note: Extra space at bottom for input area
10375
11916
  style: { height: 100 } })] }), onMessage && (jsxs("div", { className: classNames(chatStyles.chatInput, useChatCssClassName('chatInput'), isDragOver && chatStyles.dragOver), ...(onFileUpload
@@ -10390,6 +11931,7 @@ function Chat(props) {
10390
11931
  // <- TODO: Remove
10391
11932
  '--input-bg-color': inputBgColor.toHex(),
10392
11933
  '--input-text-color': inputTextColor.toHex(),
11934
+ '--brand-color': buttonColor.toHex(),
10393
11935
  }, children: [jsx("textarea", { ref: (element) => {
10394
11936
  textareaRef.current = element;
10395
11937
  }, style: {
@@ -10418,13 +11960,27 @@ function Chat(props) {
10418
11960
  } }), onFileUpload && (jsxs(Fragment, { children: [jsx("input", { ref: fileInputRef, type: "file", multiple: true, style: { display: 'none' }, onChange: handleFileInputChange }), jsx("button", { type: "button", style: {
10419
11961
  backgroundColor: buttonColor.toHex(),
10420
11962
  color: buttonColor.then(textColor).toHex(),
10421
- }, 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 }) })] })), onVoiceInput && (jsx("button", { "data-button-type": "voice", style: {
10422
- backgroundColor: buttonColor.toHex(),
10423
- color: buttonColor.then(textColor).toHex(),
10424
- }, className: classNames(chatStyles.voiceButton, isVoiceCalling && chatStyles.voiceButtonActive), onClick: (event) => {
11963
+ }, 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' ||
11964
+ speechRecognitionState === 'TRANSCRIBING', style: {
11965
+ backgroundColor: (speechRecognitionState === 'RECORDING' ||
11966
+ speechRecognitionState === 'TRANSCRIBING'
11967
+ ? Color.from('#ff4444')
11968
+ : buttonColor).toHex(),
11969
+ color: (speechRecognitionState === 'RECORDING' ||
11970
+ speechRecognitionState === 'TRANSCRIBING'
11971
+ ? Color.from('#ffffff')
11972
+ : buttonColor.then(textColor)).toHex(),
11973
+ }, className: classNames(chatStyles.voiceButton, (isVoiceCalling ||
11974
+ speechRecognitionState === 'RECORDING' ||
11975
+ speechRecognitionState === 'TRANSCRIBING') &&
11976
+ chatStyles.voiceButtonActive), onClick: (event) => {
10425
11977
  event.preventDefault();
10426
- onVoiceInput();
10427
- }, title: isVoiceCalling ? 'Stop voice call' : 'Start voice call', children: jsx(MicIcon, { size: 25 }) })), jsx("button", { "data-button-type": "call-to-action", style: {
11978
+ handleToggleVoiceInput();
11979
+ }, title: speechRecognitionState === 'RECORDING'
11980
+ ? 'Stop recording'
11981
+ : speechRecognitionState === 'TRANSCRIBING'
11982
+ ? 'Transcribing...'
11983
+ : 'Start voice input', children: jsx(MicIcon, { size: 25 }) })), jsx("button", { "data-button-type": "call-to-action", style: {
10428
11984
  backgroundColor: buttonColor.toHex(),
10429
11985
  color: buttonColor.then(textColor).toHex(),
10430
11986
  }, ref: buttonSendRef, onClick: (event) => {
@@ -10434,7 +11990,33 @@ function Chat(props) {
10434
11990
  event.preventDefault();
10435
11991
  /* not await */ handleSend();
10436
11992
  }, children: jsx(SendIcon, { size: 25 }) })] }));
10437
- })(), 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" })] }) }))] }))] }) }), ratingModalOpen && selectedMessage && (jsx("div", { className: chatStyles.ratingModal, onClick: (e) => {
11993
+ })(), 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) => {
11994
+ if (e.target === e.currentTarget) {
11995
+ setToolCallModalOpen(false);
11996
+ }
11997
+ }, children: jsxs("div", { className: classNames(chatStyles.ratingModalContent, chatStyles.toolCallModal), children: [(() => {
11998
+ const isSearch = selectedToolCall.name === 'web_search' ||
11999
+ selectedToolCall.name === 'useSearchEngine' ||
12000
+ selectedToolCall.name === 'search';
12001
+ const args = typeof selectedToolCall.arguments === 'string'
12002
+ ? JSON.parse(selectedToolCall.arguments)
12003
+ : selectedToolCall.arguments || {};
12004
+ const resultRaw = selectedToolCall.result;
12005
+ const results = Array.isArray(resultRaw)
12006
+ ? resultRaw
12007
+ : resultRaw && typeof resultRaw === 'object' && Array.isArray(resultRaw.results)
12008
+ ? resultRaw.results
12009
+ : [];
12010
+ if (isSearch) {
12011
+ 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 || 'Search Results' })] }), jsx("div", { className: chatStyles.searchModalContent, children: results.length > 0 ? (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))) })) : (jsx("div", { className: chatStyles.noResults, children: resultRaw ? 'No search results found.' : 'Executing search...' })) })] }));
12012
+ }
12013
+ // Fallback for other tools
12014
+ 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'
12015
+ ? JSON.stringify(value)
12016
+ : 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'
12017
+ ? JSON.stringify(resultRaw, null, 4)
12018
+ : String(resultRaw) }) })] })] }));
12019
+ })(), jsx("div", { className: chatStyles.ratingActions, children: jsx("button", { onClick: () => setToolCallModalOpen(false), children: "Close" }) })] }) })), ratingModalOpen && selectedMessage && (jsx("div", { className: chatStyles.ratingModal, onClick: (e) => {
10438
12020
  // Close modal when clicking backdrop on mobile
10439
12021
  if (e.target === e.currentTarget && isMobile) {
10440
12022
  setRatingModalOpen(false);
@@ -10565,20 +12147,16 @@ ChatPersistence.STORAGE_PREFIX = 'promptbook_chat_';
10565
12147
  * @public exported from `@promptbook/components`
10566
12148
  */
10567
12149
  function LlmChat(props) {
10568
- const { llmTools, persistenceKey, onChange, onReset, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, ...restProps } = props;
12150
+ const { llmTools, persistenceKey, onChange, onReset, initialMessages, sendMessage, userParticipantName = 'USER', llmParticipantName = 'ASSISTANT', autoExecuteMessage, buttonColor, toolTitles, ...restProps } = props;
10569
12151
  // Internal state management
10570
12152
  // DRY: Single factory for seeding initial messages (used on mount and after reset)
10571
12153
  const buildInitialMessages = useCallback(() => initialMessages ? ([...initialMessages]) : ([]), [initialMessages]);
10572
12154
  const [messages, setMessages] = useState(() => buildInitialMessages());
10573
12155
  const [tasksProgress, setTasksProgress] = useState([]);
10574
- const [isVoiceCalling, setIsVoiceCalling] = useState(false);
10575
- const mediaRecorderRef = useRef(null);
10576
- // Refs to keep latest state for long-lived voice handlers
12156
+ const [isVoiceCalling] = useState(false);
12157
+ // Refs to keep latest state for long-lived handlers
10577
12158
  const messagesRef = useRef([]);
10578
12159
  const participantsRef = useRef([]);
10579
- const isProcessingVoiceChunkRef = useRef(false);
10580
- const allVoiceChunksRef = useRef([]);
10581
- const utteranceTimeoutRef = useRef(null);
10582
12160
  /**
10583
12161
  * Tracks whether the user (or system via persistence restoration) has interacted.
10584
12162
  * We do NOT persist purely initialMessages until the user sends something.
@@ -10607,164 +12185,25 @@ function LlmChat(props) {
10607
12185
  // Generate participants from llmTools
10608
12186
  const participants = useMemo(() => props.participants || [
10609
12187
  {
10610
- name: userParticipantName,
10611
- fullname: 'You',
10612
- isMe: true,
10613
- color: '#1D4ED8',
10614
- },
10615
- // Use the profile from llmTools if available, otherwise fallback to default
10616
- llmTools.profile || {
10617
- name: llmParticipantName,
10618
- fullname: llmTools.title || 'AI Assistant',
10619
- color: '#10b981',
10620
- },
10621
- ], [llmTools.profile, llmTools.title]);
10622
- // Keep refs in sync for usage inside long-lived callbacks
10623
- useEffect(() => {
10624
- messagesRef.current = messages;
10625
- }, [messages]);
10626
- useEffect(() => {
10627
- participantsRef.current = participants;
10628
- }, [participants]);
10629
- // Handle voice input in a fluent, multi-message way:
10630
- // - While the call is active we keep recording audio.
10631
- // - When the user is silent for ~2 seconds, we treat it as the end of an utterance
10632
- // and send one combined audio blob to the agent (one agent turn per utterance).
10633
- const processCurrentUtterance = useCallback(async () => {
10634
- var _a;
10635
- if (!llmTools.callVoiceChatModel) {
10636
- return;
10637
- }
10638
- if (isProcessingVoiceChunkRef.current) {
10639
- return;
10640
- }
10641
- const chunks = allVoiceChunksRef.current;
10642
- if (!chunks.length) {
10643
- return;
10644
- }
10645
- isProcessingVoiceChunkRef.current = true;
10646
- allVoiceChunksRef.current = [];
10647
- const blob = new Blob(chunks, {
10648
- type: ((_a = chunks[0]) === null || _a === void 0 ? void 0 : _a.type) || 'audio/webm',
10649
- });
10650
- const taskId = `voice_call_${Date.now()}`;
10651
- setTasksProgress([{ id: taskId, name: 'Processing voice...', progress: 50 }]);
10652
- try {
10653
- const thread = props.thread ? [...props.thread] : [...messagesRef.current];
10654
- const result = await llmTools.callVoiceChatModel(blob, {
10655
- title: 'Voice Message',
10656
- content: '',
10657
- parameters: {},
10658
- modelRequirements: { modelVariant: 'CHAT' },
10659
- thread,
10660
- });
10661
- setTasksProgress([{ id: taskId, name: 'Playing response...', progress: 100 }]);
10662
- const now = Date.now();
10663
- const userMessage = {
10664
- // channel: 'PROMPTBOOK_CHAT',
10665
- id: `user_${now}`,
10666
- createdAt: new Date(),
10667
- sender: userParticipantName,
10668
- content: (result.userMessage || '(Voice message)'),
10669
- isComplete: true,
10670
- isVoiceCall: true,
10671
- };
10672
- const agentMessage = {
10673
- // channel: 'PROMPTBOOK_CHAT',
10674
- id: `agent_${now}`,
10675
- createdAt: new Date(),
10676
- sender: llmParticipantName,
10677
- content: (result.agentMessage || result.text),
10678
- isComplete: true,
10679
- isVoiceCall: true,
10680
- };
10681
- setMessages((prevMessages) => {
10682
- const newMessages = [...prevMessages, userMessage, agentMessage];
10683
- messagesRef.current = newMessages;
10684
- if (onChange) {
10685
- onChange(newMessages, participantsRef.current);
10686
- }
10687
- return newMessages;
10688
- });
10689
- // Play audio
10690
- const audioUrl = URL.createObjectURL(result.audio);
10691
- const audioEl = new Audio(audioUrl);
10692
- audioEl.play();
10693
- setTimeout(() => setTasksProgress([]), 1000);
10694
- }
10695
- catch (error) {
10696
- console.error('Error calling Voice LLM:', error);
10697
- setTasksProgress([]);
10698
- }
10699
- finally {
10700
- isProcessingVoiceChunkRef.current = false;
10701
- }
10702
- }, [llmTools, onChange, userParticipantName, llmParticipantName, props.thread]);
10703
- const handleVoiceInput = useCallback(async () => {
10704
- if (!llmTools.callVoiceChatModel) {
10705
- return;
10706
- }
10707
- if (isVoiceCalling) {
10708
- // Stop recording and end call
10709
- const recorder = mediaRecorderRef.current;
10710
- if (recorder) {
10711
- recorder.stop();
10712
- }
10713
- mediaRecorderRef.current = null;
10714
- setIsVoiceCalling(false);
10715
- // Process any remaining audio as the final utterance
10716
- if (utteranceTimeoutRef.current) {
10717
- clearTimeout(utteranceTimeoutRef.current);
10718
- utteranceTimeoutRef.current = null;
10719
- }
10720
- void processCurrentUtterance();
10721
- }
10722
- else {
10723
- // Start recording and keep listening for utterances separated by ~2s of silence
10724
- try {
10725
- const stream = await navigator.mediaDevices.getUserMedia({ audio: true });
10726
- const mediaRecorder = new MediaRecorder(stream);
10727
- mediaRecorderRef.current = mediaRecorder;
10728
- allVoiceChunksRef.current = [];
10729
- isProcessingVoiceChunkRef.current = false;
10730
- if (utteranceTimeoutRef.current) {
10731
- clearTimeout(utteranceTimeoutRef.current);
10732
- utteranceTimeoutRef.current = null;
10733
- }
10734
- mediaRecorder.ondataavailable = (event) => {
10735
- const chunk = event.data;
10736
- if (!chunk || chunk.size === 0) {
10737
- return;
10738
- }
10739
- // Accumulate chunks for the current utterance
10740
- allVoiceChunksRef.current.push(chunk);
10741
- // Reset the silence timer; if there is no new audio for 2s,
10742
- // treat it as the end of the current utterance.
10743
- if (utteranceTimeoutRef.current) {
10744
- clearTimeout(utteranceTimeoutRef.current);
10745
- }
10746
- utteranceTimeoutRef.current = setTimeout(() => {
10747
- void processCurrentUtterance();
10748
- }, 2000);
10749
- };
10750
- mediaRecorder.onstop = () => {
10751
- mediaRecorder.stream.getTracks().forEach((track) => track.stop());
10752
- if (utteranceTimeoutRef.current) {
10753
- clearTimeout(utteranceTimeoutRef.current);
10754
- utteranceTimeoutRef.current = null;
10755
- }
10756
- // Process any remaining audio when the recorder stops
10757
- void processCurrentUtterance();
10758
- };
10759
- // Use timeslices so ondataavailable is fired continuously
10760
- mediaRecorder.start(500);
10761
- setIsVoiceCalling(true);
10762
- }
10763
- catch (err) {
10764
- console.error('Error accessing microphone:', err);
10765
- }
10766
- }
10767
- }, [isVoiceCalling, llmTools, processCurrentUtterance]);
12188
+ name: userParticipantName,
12189
+ fullname: 'You',
12190
+ isMe: true,
12191
+ color: '#1D4ED8',
12192
+ },
12193
+ // Use the profile from llmTools if available, otherwise fallback to default
12194
+ llmTools.profile || {
12195
+ name: llmParticipantName,
12196
+ fullname: llmTools.title || 'AI Assistant',
12197
+ color: '#10b981',
12198
+ },
12199
+ ], [llmTools.profile, llmTools.title]);
12200
+ // Keep refs in sync for usage inside long-lived callbacks
12201
+ useEffect(() => {
12202
+ messagesRef.current = messages;
12203
+ }, [messages]);
12204
+ useEffect(() => {
12205
+ participantsRef.current = participants;
12206
+ }, [participants]);
10768
12207
  // Handle user messages and LLM responses
10769
12208
  const handleMessage = useCallback(async (messageContent, attachments = []) => {
10770
12209
  hasUserInteractedRef.current = true;
@@ -10827,6 +12266,7 @@ function LlmChat(props) {
10827
12266
  sender: llmParticipantName,
10828
12267
  content: chunk.content,
10829
12268
  isComplete: false,
12269
+ ongoingToolCalls: chunk.toolCalls,
10830
12270
  };
10831
12271
  const currentMessages = [...newMessages, assistantMessage];
10832
12272
  setMessages(currentMessages);
@@ -10851,6 +12291,7 @@ function LlmChat(props) {
10851
12291
  sender: llmParticipantName,
10852
12292
  content: result.content,
10853
12293
  isComplete: true,
12294
+ completedToolCalls: result.toolCalls,
10854
12295
  };
10855
12296
  const finalMessages = [...newMessages, assistantMessage];
10856
12297
  setMessages(finalMessages);
@@ -10916,7 +12357,7 @@ function LlmChat(props) {
10916
12357
  handleMessage(autoExecuteMessage);
10917
12358
  }
10918
12359
  }, [autoExecuteMessage, handleMessage]);
10919
- return (jsx(Fragment, { children: jsx(Chat, { ...restProps, messages, onReset, tasksProgress, participants, buttonColor, onMessage: handleMessage, onReset: handleReset, onVoiceInput: llmTools.callVoiceChatModel ? handleVoiceInput : undefined, isVoiceCalling: isVoiceCalling }) }));
12360
+ return (jsx(Fragment, { children: jsx(Chat, { ...restProps, messages, onReset, tasksProgress, participants, buttonColor, toolTitles, onMessage: handleMessage, onReset: handleReset, isVoiceCalling: isVoiceCalling }) }));
10920
12361
  }
10921
12362
 
10922
12363
  /**
@@ -10932,8 +12373,8 @@ function LlmChat(props) {
10932
12373
  * @public exported from `@promptbook/components`
10933
12374
  */
10934
12375
  function AgentChat(props) {
10935
- const { agent, title, persistenceKey, onChange, sendMessage, ...restProps } = props;
10936
- const brandColor = Color.fromSafe(agent.meta.color || PROMPTBOOK_COLOR).then(saturate(-0.5));
12376
+ const { agent, title, persistenceKey, onChange, sendMessage, toolTitles, ...restProps } = props;
12377
+ const brandColor = Color.fromSafe(agent.meta.color || PROMPTBOOK_COLOR).then(saturate(-0.2));
10937
12378
  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
10938
12379
  , initialMessages: [
10939
12380
  {
@@ -10962,7 +12403,7 @@ function AgentChat(props) {
10962
12403
  color: '#115EB6',
10963
12404
  isMe: true,
10964
12405
  },
10965
- ], buttonColor: brandColor, llmTools: agent, onChange, sendMessage, ...restProps }) }));
12406
+ ], buttonColor: brandColor, llmTools: agent, onChange, sendMessage, toolTitles, ...restProps }) }));
10966
12407
  }
10967
12408
 
10968
12409
  /**
@@ -12716,74 +14157,90 @@ function addUsage(...usageItems) {
12716
14157
  * in real-time through an observable.
12717
14158
  *
12718
14159
  * @param llmTools - The LLM tools to be intercepted and tracked
12719
- * @returns An augmented version of the tools that includes usage tracking capabilities
14160
+ * @returns Full proxy of the tools with added usage tracking capabilities
12720
14161
  * @public exported from `@promptbook/core`
12721
14162
  */
12722
14163
  function countUsage(llmTools) {
12723
14164
  let totalUsage = ZERO_USAGE;
12724
14165
  const spending = new Subject();
12725
- const proxyTools = {
12726
- get title() {
12727
- return `${llmTools.title} (+usage)`;
12728
- // <- TODO: [🧈] Maybe standartize the suffix when wrapping `LlmExecutionTools` up
12729
- // <- TODO: [🧈][🧠] Does it make sense to suffix "(+usage)"?
12730
- },
12731
- get description() {
12732
- return `${llmTools.description} (+usage)`;
12733
- // <- TODO: [🧈] Maybe standartize the suffix when wrapping `LlmExecutionTools` up
12734
- // <- TODO: [🧈][🧠] Does it make sense to suffix "(+usage)"?
12735
- },
12736
- checkConfiguration() {
12737
- return /* not await */ llmTools.checkConfiguration();
12738
- },
12739
- listModels() {
12740
- return /* not await */ llmTools.listModels();
12741
- },
12742
- spending() {
12743
- return spending.asObservable();
12744
- },
12745
- getTotalUsage() {
12746
- // <- Note: [🥫] Not using getter `get totalUsage` but `getTotalUsage` to allow this object to be proxied
12747
- return totalUsage;
14166
+ // Create a Proxy to intercept all property access and ensure full proxying of all properties
14167
+ const proxyTools = new Proxy(llmTools, {
14168
+ get(target, prop, receiver) {
14169
+ // Handle title property
14170
+ if (prop === 'title') {
14171
+ return `${target.title} (+usage)`;
14172
+ // <- TODO: [🧈] Maybe standartize the suffix when wrapping `LlmExecutionTools` up
14173
+ // <- TODO: [🧈][🧠] Does it make sense to suffix "(+usage)"?
14174
+ }
14175
+ // Handle description property
14176
+ if (prop === 'description') {
14177
+ return `${target.description} (+usage)`;
14178
+ // <- TODO: [🧈] Maybe standartize the suffix when wrapping `LlmExecutionTools` up
14179
+ // <- TODO: [🧈][🧠] Does it make sense to suffix "(+usage)"?
14180
+ }
14181
+ // Handle spending method (new method added by this wrapper)
14182
+ if (prop === 'spending') {
14183
+ return () => {
14184
+ return spending.asObservable();
14185
+ };
14186
+ }
14187
+ // Handle getTotalUsage method (new method added by this wrapper)
14188
+ if (prop === 'getTotalUsage') {
14189
+ // <- Note: [🥫] Not using getter `get totalUsage` but `getTotalUsage` to allow this object to be proxied
14190
+ return () => {
14191
+ return totalUsage;
14192
+ };
14193
+ }
14194
+ // Handle callChatModel method with usage counting
14195
+ if (prop === 'callChatModel' && target.callChatModel !== undefined) {
14196
+ return async (prompt) => {
14197
+ // console.info('[🚕] callChatModel through countTotalUsage');
14198
+ const promptResult = await target.callChatModel(prompt);
14199
+ totalUsage = addUsage(totalUsage, promptResult.usage);
14200
+ spending.next(promptResult.usage);
14201
+ return promptResult;
14202
+ };
14203
+ }
14204
+ // Handle callCompletionModel method with usage counting
14205
+ if (prop === 'callCompletionModel' && target.callCompletionModel !== undefined) {
14206
+ return async (prompt) => {
14207
+ // console.info('[🚕] callCompletionModel through countTotalUsage');
14208
+ const promptResult = await target.callCompletionModel(prompt);
14209
+ totalUsage = addUsage(totalUsage, promptResult.usage);
14210
+ spending.next(promptResult.usage);
14211
+ return promptResult;
14212
+ };
14213
+ }
14214
+ // Handle callEmbeddingModel method with usage counting
14215
+ if (prop === 'callEmbeddingModel' && target.callEmbeddingModel !== undefined) {
14216
+ return async (prompt) => {
14217
+ // console.info('[🚕] callEmbeddingModel through countTotalUsage');
14218
+ const promptResult = await target.callEmbeddingModel(prompt);
14219
+ totalUsage = addUsage(totalUsage, promptResult.usage);
14220
+ spending.next(promptResult.usage);
14221
+ return promptResult;
14222
+ };
14223
+ }
14224
+ // Handle callImageGenerationModel method with usage counting
14225
+ if (prop === 'callImageGenerationModel' && target.callImageGenerationModel !== undefined) {
14226
+ return async (prompt) => {
14227
+ // console.info('[🚕] callImageGenerationModel through countTotalUsage');
14228
+ const promptResult = await target.callImageGenerationModel(prompt);
14229
+ totalUsage = addUsage(totalUsage, promptResult.usage);
14230
+ spending.next(promptResult.usage);
14231
+ return promptResult;
14232
+ };
14233
+ }
14234
+ // <- Note: [🤖]
14235
+ // For all other properties and methods, delegate to the original target
14236
+ const value = Reflect.get(target, prop, receiver);
14237
+ // If it's a function, bind it to the target to preserve context
14238
+ if (typeof value === 'function') {
14239
+ return value.bind(target);
14240
+ }
14241
+ return value;
12748
14242
  },
12749
- };
12750
- if (llmTools.callChatModel !== undefined) {
12751
- proxyTools.callChatModel = async (prompt) => {
12752
- // console.info('[🚕] callChatModel through countTotalUsage');
12753
- const promptResult = await llmTools.callChatModel(prompt);
12754
- totalUsage = addUsage(totalUsage, promptResult.usage);
12755
- spending.next(promptResult.usage);
12756
- return promptResult;
12757
- };
12758
- }
12759
- if (llmTools.callCompletionModel !== undefined) {
12760
- proxyTools.callCompletionModel = async (prompt) => {
12761
- // console.info('[🚕] callCompletionModel through countTotalUsage');
12762
- const promptResult = await llmTools.callCompletionModel(prompt);
12763
- totalUsage = addUsage(totalUsage, promptResult.usage);
12764
- spending.next(promptResult.usage);
12765
- return promptResult;
12766
- };
12767
- }
12768
- if (llmTools.callEmbeddingModel !== undefined) {
12769
- proxyTools.callEmbeddingModel = async (prompt) => {
12770
- // console.info('[🚕] callEmbeddingModel through countTotalUsage');
12771
- const promptResult = await llmTools.callEmbeddingModel(prompt);
12772
- totalUsage = addUsage(totalUsage, promptResult.usage);
12773
- spending.next(promptResult.usage);
12774
- return promptResult;
12775
- };
12776
- }
12777
- if (llmTools.callImageGenerationModel !== undefined) {
12778
- proxyTools.callImageGenerationModel = async (prompt) => {
12779
- // console.info('[🚕] callImageGenerationModel through countTotalUsage');
12780
- const promptResult = await llmTools.callImageGenerationModel(prompt);
12781
- totalUsage = addUsage(totalUsage, promptResult.usage);
12782
- spending.next(promptResult.usage);
12783
- return promptResult;
12784
- };
12785
- }
12786
- // <- Note: [🤖]
14243
+ });
12787
14244
  return proxyTools;
12788
14245
  }
12789
14246
  /**
@@ -15563,6 +17020,97 @@ function createBasicAgentModelRequirements(agentName) {
15563
17020
  * TODO: [🐤] Deduplicate `AgentModelRequirements` and `ModelRequirements` model requirements
15564
17021
  */
15565
17022
 
17023
+ /**
17024
+ * Plugin for importing agent books *(`.book` files)*
17025
+ *
17026
+ * @private [🥝] Maybe export the import plugins through some package
17027
+ */
17028
+ const AgentFileImportPlugin = {
17029
+ name: 'agent-file-import-plugin',
17030
+ canImport(mimeType) {
17031
+ // [🧠] Should we have a specific MIME type for agent books?
17032
+ // For now, let's assume it's identified by .book extension or certain MIME types if provided
17033
+ return mimeType === 'text/x-promptbook' || mimeType === 'application/x-promptbook';
17034
+ },
17035
+ import(content) {
17036
+ const parseResult = parseAgentSourceWithCommitments(content);
17037
+ // Bring only the agent corpus (non-commitment lines and relevant commitments)
17038
+ // Stripping the agent name (which is usually the first line)
17039
+ const corpus = parseResult.nonCommitmentLines
17040
+ .filter((line, index) => index > 0 || !parseResult.agentName)
17041
+ .join('\n')
17042
+ .trim();
17043
+ // Also include relevant commitments that make up the "corpus" of the agent
17044
+ // For example PERSONA, RULE, KNOWLEDGE
17045
+ const relevantCommitments = parseResult.commitments
17046
+ .filter((c) => ['PERSONA', 'RULE', 'KNOWLEDGE'].includes(c.type))
17047
+ .map((c) => `${c.type} ${c.content}`)
17048
+ .join('\n\n');
17049
+ return spaceTrim$1((block) => `
17050
+ ${block(relevantCommitments)}
17051
+
17052
+ ${block(corpus)}
17053
+ `).trim();
17054
+ },
17055
+ };
17056
+
17057
+ /**
17058
+ * Plugin for importing JSON files
17059
+ *
17060
+ * @private [🥝] Maybe export the import plugins through some package
17061
+ */
17062
+ const JsonFileImportPlugin = {
17063
+ name: 'json-file-import-plugin',
17064
+ canImport(mimeType) {
17065
+ return mimeType === 'application/json' || mimeType.endsWith('+json');
17066
+ },
17067
+ import(content) {
17068
+ try {
17069
+ const json = JSON.parse(content);
17070
+ const formattedJson = JSON.stringify(json, null, 4);
17071
+ return `\`\`\`json\n${formattedJson}\n\`\`\``;
17072
+ }
17073
+ catch (error) {
17074
+ // If JSON is invalid, still import it but maybe not as pretty JSON
17075
+ return `\`\`\`json\n${content}\n\`\`\``;
17076
+ }
17077
+ },
17078
+ };
17079
+
17080
+ /**
17081
+ * Plugin for importing generic text files
17082
+ *
17083
+ * @private [🥝] Maybe export the import plugins through some package
17084
+ */
17085
+ const TextFileImportPlugin = {
17086
+ name: 'text-file-import-plugin',
17087
+ canImport(mimeType) {
17088
+ return (mimeType === 'text/plain' ||
17089
+ mimeType === 'text/markdown' ||
17090
+ mimeType === 'text/x-typescript' ||
17091
+ mimeType === 'text/javascript' ||
17092
+ mimeType === 'text/css' ||
17093
+ mimeType === 'text/html' ||
17094
+ mimeType.startsWith('text/'));
17095
+ },
17096
+ import(content, mimeType) {
17097
+ const extension = mimeTypeToExtension(mimeType);
17098
+ const codeBlockType = extension || 'txt';
17099
+ return `\`\`\`${codeBlockType}\n${content}\n\`\`\``;
17100
+ },
17101
+ };
17102
+
17103
+ /**
17104
+ * All available file import plugins
17105
+ *
17106
+ * @private [🥝] Maybe export the import plugins through some package
17107
+ */
17108
+ const $fileImportPlugins = [
17109
+ AgentFileImportPlugin,
17110
+ JsonFileImportPlugin,
17111
+ TextFileImportPlugin,
17112
+ ];
17113
+
15566
17114
  /**
15567
17115
  * Removes comment lines (lines starting with #) from a system message
15568
17116
  * This is used to clean up the final system message before sending it to the AI model
@@ -15594,6 +17142,7 @@ function removeCommentsFromSystemMessage(systemMessage) {
15594
17142
  * @public exported from `@promptbook/core`
15595
17143
  */
15596
17144
  async function createAgentModelRequirementsWithCommitments(agentSource, modelName) {
17145
+ var _a;
15597
17146
  // Parse the agent source to extract commitments
15598
17147
  const parseResult = parseAgentSourceWithCommitments(agentSource);
15599
17148
  // Apply DELETE filtering: remove prior commitments tagged by parameters targeted by DELETE/CANCEL/DISCARD/REMOVE
@@ -15660,6 +17209,64 @@ async function createAgentModelRequirementsWithCommitments(agentSource, modelNam
15660
17209
  }
15661
17210
  }
15662
17211
  }
17212
+ // Handle IMPORT commitments for generic files
17213
+ // Note: This logic could be moved to ImportCommitmentDefinition, but it needs to be asynchronous
17214
+ if (requirements.importedFileUrls && requirements.importedFileUrls.length > 0) {
17215
+ for (const fileUrl of requirements.importedFileUrls) {
17216
+ try {
17217
+ // 1. Mocked security check
17218
+ await mockedSecurityCheck(fileUrl);
17219
+ // 2. Fetch file content
17220
+ let content;
17221
+ let mimeType = null;
17222
+ if (isValidUrl(fileUrl)) {
17223
+ const response = await promptbookFetch(fileUrl);
17224
+ if (!response.ok) {
17225
+ throw new Error(`Failed to fetch ${fileUrl}: ${response.statusText}`);
17226
+ }
17227
+ content = await response.text();
17228
+ mimeType = response.headers.get('Content-Type');
17229
+ /*
17230
+ TODO: !!!! Commented out this case because we need to work in Browser-compatible mode in many packages, use passed `fs` instead
17231
+ } else if (isValidFilePath(fileUrl)) {
17232
+ // [x🟢x] This code is expected to run in Node environment if local files are used
17233
+ const fs = await import('fs/promises');
17234
+ content = await fs.readFile(fileUrl, 'utf-8');
17235
+ const extension = getFileExtension(fileUrl);
17236
+ mimeType = extensionToMimeType(extension as string);
17237
+ */
17238
+ }
17239
+ else {
17240
+ throw new Error(`Invalid file URL or path: ${fileUrl}`);
17241
+ }
17242
+ if (!mimeType) {
17243
+ mimeType = 'text/plain';
17244
+ }
17245
+ // Remove charset from mime type
17246
+ mimeType = mimeType.split(';')[0].trim();
17247
+ // 3. Prevent importing binary files (mocked check)
17248
+ if (isBinaryMimeType(mimeType)) {
17249
+ throw new Error(`Importing binary files is not allowed: ${mimeType}`);
17250
+ }
17251
+ // 4. Find appropriate plugin
17252
+ const plugin = $fileImportPlugins.find((p) => p.canImport(mimeType));
17253
+ if (!plugin) {
17254
+ throw new Error(`No import plugin found for MIME type: ${mimeType}`);
17255
+ }
17256
+ // 5. Process content
17257
+ const importedContent = await plugin.import(content, mimeType);
17258
+ // 6. Append to system message
17259
+ requirements = {
17260
+ ...requirements,
17261
+ systemMessage: requirements.systemMessage + '\n\n' + importedContent,
17262
+ };
17263
+ }
17264
+ catch (error) {
17265
+ console.warn(`Failed to import file ${fileUrl}:`, error);
17266
+ // Continue with other imports even if one fails
17267
+ }
17268
+ }
17269
+ }
15663
17270
  // Handle MCP servers (extract from original agent source)
15664
17271
  const mcpServers = extractMcpServers(agentSource);
15665
17272
  if (mcpServers.length > 0) {
@@ -15669,9 +17276,11 @@ async function createAgentModelRequirementsWithCommitments(agentSource, modelNam
15669
17276
  };
15670
17277
  }
15671
17278
  // Add non-commitment lines to system message if they exist
17279
+ // Note: Filtering out horizontal lines (---) as requested
15672
17280
  const nonCommitmentContent = parseResult.nonCommitmentLines
15673
17281
  .filter((line, index) => index > 0 || !parseResult.agentName) // Skip first line if it's the agent name
15674
17282
  .filter((line) => line.trim()) // Remove empty lines
17283
+ .filter((line) => !/^[\s]*[-_*][\s]*[-_*][\s]*[-_*][\s]*[-_*]*[\s]*$/.test(line)) // Remove horizontal lines
15675
17284
  .join('\n')
15676
17285
  .trim();
15677
17286
  if (nonCommitmentContent) {
@@ -15680,6 +17289,26 @@ async function createAgentModelRequirementsWithCommitments(agentSource, modelNam
15680
17289
  systemMessage: requirements.systemMessage + '\n\n' + nonCommitmentContent,
15681
17290
  };
15682
17291
  }
17292
+ // Add example interactions to the system message
17293
+ const examples = [];
17294
+ // 1. Initial message as an example agent response
17295
+ const initialMessage = (_a = parseResult.commitments.find((c) => c.type === 'INITIAL MESSAGE')) === null || _a === void 0 ? void 0 : _a.content;
17296
+ if (initialMessage) {
17297
+ examples.push(`Agent: ${initialMessage}`);
17298
+ }
17299
+ // 2. User and Agent message pairs
17300
+ if (requirements.samples && requirements.samples.length > 0) {
17301
+ for (const sample of requirements.samples) {
17302
+ examples.push(`User: ${sample.question}\nAgent: ${sample.answer}`);
17303
+ }
17304
+ }
17305
+ if (examples.length > 0) {
17306
+ const exampleInteractionsContent = `Example interaction:\n\n${examples.join('\n\n')}`;
17307
+ requirements = {
17308
+ ...requirements,
17309
+ systemMessage: requirements.systemMessage + '\n\n' + exampleInteractionsContent,
17310
+ };
17311
+ }
15683
17312
  // Remove comment lines (lines starting with #) from the final system message
15684
17313
  // while preserving the original content with comments in metadata
15685
17314
  const cleanedSystemMessage = removeCommentsFromSystemMessage(requirements.systemMessage);
@@ -15688,6 +17317,36 @@ async function createAgentModelRequirementsWithCommitments(agentSource, modelNam
15688
17317
  systemMessage: cleanedSystemMessage,
15689
17318
  };
15690
17319
  }
17320
+ /**
17321
+ * Mocked security check for imported files
17322
+ *
17323
+ * @param urlOrPath - The URL or local path of the file to check
17324
+ * @returns A promise that resolves if the file is safe
17325
+ */
17326
+ async function mockedSecurityCheck(urlOrPath) {
17327
+ // TODO: Implement proper security checks
17328
+ await new Promise((resolve) => setTimeout(resolve, 10)); // Mock async delay
17329
+ if (urlOrPath.includes('malicious')) {
17330
+ throw new Error(`Security check failed for: ${urlOrPath}`);
17331
+ }
17332
+ }
17333
+ /**
17334
+ * Checks if the given MIME type belongs to a binary file
17335
+ *
17336
+ * @param mimeType - The MIME type to check
17337
+ * @returns True if it's a binary MIME type
17338
+ */
17339
+ function isBinaryMimeType(mimeType) {
17340
+ const binaryPrefixes = [
17341
+ 'image/',
17342
+ 'video/',
17343
+ 'audio/',
17344
+ 'application/octet-stream',
17345
+ 'application/pdf',
17346
+ 'application/zip',
17347
+ ];
17348
+ return binaryPrefixes.some((prefix) => mimeType.startsWith(prefix));
17349
+ }
15691
17350
 
15692
17351
  /**
15693
17352
  * Creates model requirements for an agent based on its source
@@ -16635,20 +18294,29 @@ class OpenAiCompatibleExecutionTools {
16635
18294
  });
16636
18295
  return availableModels;
16637
18296
  }
18297
+ /**
18298
+ * Calls OpenAI compatible API to use a chat model.
18299
+ */
16638
18300
  /**
16639
18301
  * Calls OpenAI compatible API to use a chat model.
16640
18302
  */
16641
18303
  async callChatModel(prompt) {
18304
+ return this.callChatModelStream(prompt, () => { });
18305
+ }
18306
+ /**
18307
+ * Calls OpenAI compatible API to use a chat model with streaming.
18308
+ */
18309
+ async callChatModelStream(prompt, onProgress) {
16642
18310
  // Deep clone prompt and modelRequirements to avoid mutation across calls
16643
18311
  const clonedPrompt = JSON.parse(JSON.stringify(prompt));
16644
18312
  // Use local Set for retried parameters to ensure independence and thread safety
16645
18313
  const retriedUnsupportedParameters = new Set();
16646
- return this.callChatModelWithRetry(clonedPrompt, clonedPrompt.modelRequirements, [], retriedUnsupportedParameters);
18314
+ return this.callChatModelWithRetry(clonedPrompt, clonedPrompt.modelRequirements, [], retriedUnsupportedParameters, onProgress);
16647
18315
  }
16648
18316
  /**
16649
18317
  * Internal method that handles parameter retry for chat model calls
16650
18318
  */
16651
- async callChatModelWithRetry(prompt, currentModelRequirements, attemptStack = [], retriedUnsupportedParameters = new Set()) {
18319
+ async callChatModelWithRetry(prompt, currentModelRequirements, attemptStack = [], retriedUnsupportedParameters = new Set(), onProgress) {
16652
18320
  var _a;
16653
18321
  if (this.options.isVerbose) {
16654
18322
  console.info(`💬 ${this.title} callChatModel call`, { prompt, currentModelRequirements });
@@ -16754,6 +18422,23 @@ class OpenAiCompatibleExecutionTools {
16754
18422
  const usage = this.computeUsage(content || '', responseMessage.content || '', rawResponse);
16755
18423
  totalUsage = addUsage(totalUsage, usage);
16756
18424
  if (responseMessage.tool_calls && responseMessage.tool_calls.length > 0) {
18425
+ if (onProgress) {
18426
+ onProgress({
18427
+ content: responseMessage.content || '',
18428
+ modelName: rawResponse.model || modelName,
18429
+ timing: { start, complete: $getCurrentDate() },
18430
+ usage: totalUsage,
18431
+ toolCalls: responseMessage.tool_calls.map((toolCall) => ({
18432
+ name: toolCall.function.name,
18433
+ arguments: toolCall.function.arguments,
18434
+ result: '',
18435
+ rawToolCall: toolCall,
18436
+ })),
18437
+ rawPromptContent,
18438
+ rawRequest,
18439
+ rawResponse,
18440
+ });
18441
+ }
16757
18442
  await forEachAsync(responseMessage.tool_calls, {}, async (toolCall) => {
16758
18443
  const functionName = toolCall.function.name;
16759
18444
  const functionArgs = toolCall.function.arguments;
@@ -16775,7 +18460,7 @@ class OpenAiCompatibleExecutionTools {
16775
18460
  const args = ${functionArgs};
16776
18461
  return await ${functionName}(args);
16777
18462
  `,
16778
- parameters: {}, // <- TODO: [🧠] What parameters to pass?
18463
+ parameters: prompt.parameters,
16779
18464
  });
16780
18465
  }
16781
18466
  catch (error) {
@@ -16881,7 +18566,7 @@ class OpenAiCompatibleExecutionTools {
16881
18566
  });
16882
18567
  // Remove the unsupported parameter and retry
16883
18568
  const modifiedModelRequirements = removeUnsupportedModelRequirement(currentModelRequirements, unsupportedParameter);
16884
- return this.callChatModelWithRetry(prompt, modifiedModelRequirements, attemptStack, retriedUnsupportedParameters);
18569
+ return this.callChatModelWithRetry(prompt, modifiedModelRequirements, attemptStack, retriedUnsupportedParameters, onProgress);
16885
18570
  }
16886
18571
  }
16887
18572
  throw new PipelineExecutionError(`Tool calling loop did not return a result from ${this.title}`);
@@ -17514,7 +19199,7 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17514
19199
  * Calls OpenAI API to use a chat model with streaming.
17515
19200
  */
17516
19201
  async callChatModelStream(prompt, onProgress) {
17517
- var _a, _b, _c;
19202
+ var _a, _b, _c, _d;
17518
19203
  if (this.options.isVerbose) {
17519
19204
  console.info('💬 OpenAI callChatModel call', { prompt });
17520
19205
  }
@@ -17569,6 +19254,157 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17569
19254
  }
17570
19255
  // Always add the current user message
17571
19256
  threadMessages.push({ role: 'user', content: rawPromptContent });
19257
+ // Check if tools are being used - if so, use non-streaming mode
19258
+ const hasTools = modelRequirements.tools !== undefined && modelRequirements.tools.length > 0;
19259
+ const start = $getCurrentDate();
19260
+ let complete;
19261
+ // [🐱‍🚀] When tools are present, we need to use the non-streaming Runs API
19262
+ // because streaming doesn't support tool execution flow properly
19263
+ if (hasTools) {
19264
+ onProgress({
19265
+ content: '',
19266
+ modelName: 'assistant',
19267
+ timing: { start, complete: $getCurrentDate() },
19268
+ usage: UNCERTAIN_USAGE,
19269
+ rawPromptContent,
19270
+ rawRequest: null,
19271
+ rawResponse: null,
19272
+ });
19273
+ const rawRequest = {
19274
+ assistant_id: this.assistantId,
19275
+ thread: {
19276
+ messages: threadMessages,
19277
+ },
19278
+ tools: mapToolsToOpenAi(modelRequirements.tools),
19279
+ };
19280
+ if (this.options.isVerbose) {
19281
+ console.info(colors.bgWhite('rawRequest (non-streaming with tools)'), JSON.stringify(rawRequest, null, 4));
19282
+ }
19283
+ // Create thread and run
19284
+ const threadAndRun = await client.beta.threads.createAndRun(rawRequest);
19285
+ let run = threadAndRun;
19286
+ // Poll until run completes or requires action
19287
+ while (run.status === 'queued' || run.status === 'in_progress' || run.status === 'requires_action') {
19288
+ if (run.status === 'requires_action' && ((_a = run.required_action) === null || _a === void 0 ? void 0 : _a.type) === 'submit_tool_outputs') {
19289
+ // Execute tools
19290
+ const toolCalls = run.required_action.submit_tool_outputs.tool_calls;
19291
+ const toolOutputs = [];
19292
+ for (const toolCall of toolCalls) {
19293
+ if (toolCall.type === 'function') {
19294
+ const functionName = toolCall.function.name;
19295
+ const functionArgs = JSON.parse(toolCall.function.arguments);
19296
+ onProgress({
19297
+ content: '',
19298
+ modelName: 'assistant',
19299
+ timing: { start, complete: $getCurrentDate() },
19300
+ usage: UNCERTAIN_USAGE,
19301
+ rawPromptContent,
19302
+ rawRequest: null,
19303
+ rawResponse: null,
19304
+ toolCalls: [
19305
+ {
19306
+ name: functionName,
19307
+ arguments: toolCall.function.arguments,
19308
+ result: '',
19309
+ rawToolCall: toolCall,
19310
+ },
19311
+ ],
19312
+ });
19313
+ if (this.options.isVerbose) {
19314
+ console.info(`🔧 Executing tool: ${functionName}`, functionArgs);
19315
+ }
19316
+ // Get execution tools for script execution
19317
+ const executionTools = this.options
19318
+ .executionTools;
19319
+ if (!executionTools || !executionTools.script) {
19320
+ throw new PipelineExecutionError(`Model requested tool '${functionName}' but no executionTools.script were provided in OpenAiAssistantExecutionTools options`);
19321
+ }
19322
+ // TODO: [DRY] Use some common tool caller (similar to OpenAiCompatibleExecutionTools)
19323
+ const scriptTools = Array.isArray(executionTools.script)
19324
+ ? executionTools.script
19325
+ : [executionTools.script];
19326
+ let functionResponse;
19327
+ try {
19328
+ const scriptTool = scriptTools[0]; // <- TODO: [🧠] Which script tool to use?
19329
+ functionResponse = await scriptTool.execute({
19330
+ scriptLanguage: 'javascript',
19331
+ script: `
19332
+ const args = ${JSON.stringify(functionArgs)};
19333
+ return await ${functionName}(args);
19334
+ `,
19335
+ parameters: prompt.parameters,
19336
+ });
19337
+ if (this.options.isVerbose) {
19338
+ console.info(`✅ Tool ${functionName} executed:`, functionResponse);
19339
+ }
19340
+ }
19341
+ catch (error) {
19342
+ assertsError(error);
19343
+ functionResponse = spaceTrim$2((block) => `
19344
+
19345
+ The invoked tool \`${functionName}\` failed with error:
19346
+
19347
+ \`\`\`json
19348
+ ${block(JSON.stringify(serializeError(error), null, 4))}
19349
+ \`\`\`
19350
+
19351
+ `);
19352
+ console.error(colors.bgRed(`❌ Error executing tool ${functionName}:`));
19353
+ console.error(error);
19354
+ }
19355
+ toolOutputs.push({
19356
+ tool_call_id: toolCall.id,
19357
+ output: functionResponse,
19358
+ });
19359
+ }
19360
+ }
19361
+ // Submit tool outputs
19362
+ run = await client.beta.threads.runs.submitToolOutputs(run.thread_id, run.id, {
19363
+ tool_outputs: toolOutputs,
19364
+ });
19365
+ }
19366
+ else {
19367
+ // Wait a bit before polling again
19368
+ await new Promise((resolve) => setTimeout(resolve, 500));
19369
+ run = await client.beta.threads.runs.retrieve(run.thread_id, run.id);
19370
+ }
19371
+ }
19372
+ if (run.status !== 'completed') {
19373
+ throw new PipelineExecutionError(`Assistant run failed with status: ${run.status}`);
19374
+ }
19375
+ // Get messages from the thread
19376
+ const messages = await client.beta.threads.messages.list(run.thread_id);
19377
+ const assistantMessages = messages.data.filter((msg) => msg.role === 'assistant');
19378
+ if (assistantMessages.length === 0) {
19379
+ throw new PipelineExecutionError('No assistant messages found after run completion');
19380
+ }
19381
+ const lastMessage = assistantMessages[0];
19382
+ const textContent = lastMessage.content.find((c) => c.type === 'text');
19383
+ if (!textContent || textContent.type !== 'text') {
19384
+ throw new PipelineExecutionError('No text content in assistant response');
19385
+ }
19386
+ complete = $getCurrentDate();
19387
+ const resultContent = textContent.text.value;
19388
+ const usage = UNCERTAIN_USAGE;
19389
+ // Progress callback with final result
19390
+ const finalChunk = {
19391
+ content: resultContent,
19392
+ modelName: 'assistant',
19393
+ timing: { start, complete },
19394
+ usage,
19395
+ rawPromptContent,
19396
+ rawRequest,
19397
+ rawResponse: { run, messages: messages.data },
19398
+ };
19399
+ onProgress(finalChunk);
19400
+ return exportJson({
19401
+ name: 'promptResult',
19402
+ message: `Result of \`OpenAiAssistantExecutionTools.callChatModelStream\` (with tools)`,
19403
+ order: [],
19404
+ value: finalChunk,
19405
+ });
19406
+ }
19407
+ // Streaming mode (without tools)
17572
19408
  const rawRequest = {
17573
19409
  // TODO: [👨‍👨‍👧‍👧] ...modelSettings,
17574
19410
  // TODO: [👨‍👨‍👧‍👧][🧠] What about system message for assistants, does it make sense - combination of OpenAI assistants with Promptbook Personas
@@ -17579,10 +19415,8 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17579
19415
  tools: modelRequirements.tools === undefined ? undefined : mapToolsToOpenAi(modelRequirements.tools),
17580
19416
  // <- TODO: Add user identification here> user: this.options.user,
17581
19417
  };
17582
- const start = $getCurrentDate();
17583
- let complete;
17584
19418
  if (this.options.isVerbose) {
17585
- console.info(colors.bgWhite('rawRequest'), JSON.stringify(rawRequest, null, 4));
19419
+ console.info(colors.bgWhite('rawRequest (streaming)'), JSON.stringify(rawRequest, null, 4));
17586
19420
  }
17587
19421
  const stream = await client.beta.threads.createAndRunStream(rawRequest);
17588
19422
  stream.on('connect', () => {
@@ -17618,6 +19452,15 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17618
19452
  console.info('messageDone', message);
17619
19453
  }
17620
19454
  });
19455
+ // TODO: [🐱‍🚀] Handle tool calls in assistants
19456
+ // Note: OpenAI Assistant streaming with tool calls requires special handling.
19457
+ // The stream will pause when a tool call is needed, and we need to:
19458
+ // 1. Wait for the run to reach 'requires_action' status
19459
+ // 2. Execute the tool calls
19460
+ // 3. Submit tool outputs via a separate API call (not on the stream)
19461
+ // 4. Continue the run
19462
+ // This requires switching to non-streaming mode or using the Runs API directly.
19463
+ // For now, tools with assistants should use the non-streaming chat completions API instead.
17621
19464
  const rawResponse = await stream.finalMessages();
17622
19465
  if (this.options.isVerbose) {
17623
19466
  console.info(colors.bgWhite('rawResponse'), JSON.stringify(rawResponse, null, 4));
@@ -17628,10 +19471,10 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17628
19471
  if (rawResponse[0].content.length !== 1) {
17629
19472
  throw new PipelineExecutionError(`There is NOT 1 BUT ${rawResponse[0].content.length} finalMessages content from OpenAI`);
17630
19473
  }
17631
- if (((_a = rawResponse[0].content[0]) === null || _a === void 0 ? void 0 : _a.type) !== 'text') {
17632
- throw new PipelineExecutionError(`There is NOT 'text' BUT ${(_b = rawResponse[0].content[0]) === null || _b === void 0 ? void 0 : _b.type} finalMessages content type from OpenAI`);
19474
+ if (((_b = rawResponse[0].content[0]) === null || _b === void 0 ? void 0 : _b.type) !== 'text') {
19475
+ throw new PipelineExecutionError(`There is NOT 'text' BUT ${(_c = rawResponse[0].content[0]) === null || _c === void 0 ? void 0 : _c.type} finalMessages content type from OpenAI`);
17633
19476
  }
17634
- const resultContent = (_c = rawResponse[0].content[0]) === null || _c === void 0 ? void 0 : _c.text.value;
19477
+ const resultContent = (_d = rawResponse[0].content[0]) === null || _d === void 0 ? void 0 : _d.text.value;
17635
19478
  // <- TODO: [🧠] There are also annotations, maybe use them
17636
19479
  // eslint-disable-next-line prefer-const
17637
19480
  complete = $getCurrentDate();
@@ -17699,7 +19542,7 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17699
19542
  throw new NotAllowed(`Creating new assistants is not allowed. Set \`isCreatingNewAssistantsAllowed: true\` in options to enable this feature.`);
17700
19543
  }
17701
19544
  // await this.playground();
17702
- const { name, instructions, knowledgeSources } = options;
19545
+ const { name, instructions, knowledgeSources, tools } = options;
17703
19546
  const client = await this.getClient();
17704
19547
  let vectorStoreId;
17705
19548
  // If knowledge sources are provided, create a vector store with them
@@ -17770,7 +19613,11 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17770
19613
  description: 'Assistant created via Promptbook',
17771
19614
  model: 'gpt-4o',
17772
19615
  instructions,
17773
- tools: [/* TODO: [🧠] Maybe add { type: 'code_interpreter' }, */ { type: 'file_search' }],
19616
+ tools: [
19617
+ /* TODO: [🧠] Maybe add { type: 'code_interpreter' }, */
19618
+ { type: 'file_search' },
19619
+ ...(tools === undefined ? [] : mapToolsToOpenAi(tools)),
19620
+ ],
17774
19621
  };
17775
19622
  // Attach vector store if created
17776
19623
  if (vectorStoreId) {
@@ -17795,7 +19642,7 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17795
19642
  if (!this.isCreatingNewAssistantsAllowed) {
17796
19643
  throw new NotAllowed(`Updating assistants is not allowed. Set \`isCreatingNewAssistantsAllowed: true\` in options to enable this feature.`);
17797
19644
  }
17798
- const { assistantId, name, instructions, knowledgeSources } = options;
19645
+ const { assistantId, name, instructions, knowledgeSources, tools } = options;
17799
19646
  const client = await this.getClient();
17800
19647
  let vectorStoreId;
17801
19648
  // If knowledge sources are provided, create a vector store with them
@@ -17864,7 +19711,11 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17864
19711
  const assistantUpdate = {
17865
19712
  name,
17866
19713
  instructions,
17867
- tools: [/* TODO: [🧠] Maybe add { type: 'code_interpreter' }, */ { type: 'file_search' }],
19714
+ tools: [
19715
+ /* TODO: [🧠] Maybe add { type: 'code_interpreter' }, */
19716
+ { type: 'file_search' },
19717
+ ...(tools === undefined ? [] : mapToolsToOpenAi(tools)),
19718
+ ],
17868
19719
  };
17869
19720
  if (vectorStoreId) {
17870
19721
  assistantUpdate.tool_resources = {
@@ -17905,6 +19756,7 @@ class OpenAiAssistantExecutionTools extends OpenAiExecutionTools {
17905
19756
  */
17906
19757
  const DISCRIMINANT = 'OPEN_AI_ASSISTANT_V1';
17907
19758
  /**
19759
+ * TODO: [🙎] In `OpenAiAssistantExecutionTools` Allow to create abstract assistants with `isCreatingNewAssistantsAllowed`
17908
19760
  * TODO: [🧠][🧙‍♂️] Maybe there can be some wizard for those who want to use just OpenAI
17909
19761
  * TODO: Maybe make custom OpenAiError
17910
19762
  * TODO: [🧠][🈁] Maybe use `isDeterministic` from options
@@ -17963,8 +19815,10 @@ class AgentLlmExecutionTools {
17963
19815
  }
17964
19816
  /**
17965
19817
  * Get cached or create agent model requirements
19818
+ *
19819
+ * Note: [🐤] This is names `getModelRequirements` *(not `getAgentModelRequirements`)* because in future these two will be united
17966
19820
  */
17967
- async getAgentModelRequirements() {
19821
+ async getModelRequirements() {
17968
19822
  if (this._cachedModelRequirements === null) {
17969
19823
  // Get available models from underlying LLM tools for best model selection
17970
19824
  const availableModels = await this.options.llmTools.listModels();
@@ -18034,9 +19888,25 @@ class AgentLlmExecutionTools {
18034
19888
  if (prompt.modelRequirements.modelVariant !== 'CHAT') {
18035
19889
  throw new Error('AgentLlmExecutionTools only supports chat prompts');
18036
19890
  }
18037
- const modelRequirements = await this.getAgentModelRequirements();
19891
+ const modelRequirements = await this.getModelRequirements();
18038
19892
  const chatPrompt = prompt;
18039
19893
  let underlyingLlmResult;
19894
+ // Create modified chat prompt with agent system message
19895
+ const promptWithAgentModelRequirements = {
19896
+ ...chatPrompt,
19897
+ modelRequirements: {
19898
+ ...chatPrompt.modelRequirements,
19899
+ ...modelRequirements,
19900
+ // Spread tools to convert readonly array to mutable
19901
+ tools: modelRequirements.tools ? [...modelRequirements.tools] : chatPrompt.modelRequirements.tools,
19902
+ // Prepend agent system message to existing system message
19903
+ systemMessage: modelRequirements.systemMessage +
19904
+ (chatPrompt.modelRequirements.systemMessage
19905
+ ? `\n\n${chatPrompt.modelRequirements.systemMessage}`
19906
+ : ''),
19907
+ },
19908
+ };
19909
+ console.log('!!!! promptWithAgentModelRequirements:', promptWithAgentModelRequirements);
18040
19910
  if (OpenAiAssistantExecutionTools.isOpenAiAssistantExecutionTools(this.options.llmTools)) {
18041
19911
  const requirementsHash = SHA256(JSON.stringify(modelRequirements)).toString();
18042
19912
  const cached = AgentLlmExecutionTools.assistantCache.get(this.title);
@@ -18057,6 +19927,7 @@ class AgentLlmExecutionTools {
18057
19927
  name: this.title,
18058
19928
  instructions: modelRequirements.systemMessage,
18059
19929
  knowledgeSources: modelRequirements.knowledgeSources,
19930
+ tools: modelRequirements.tools ? [...modelRequirements.tools] : undefined,
18060
19931
  });
18061
19932
  AgentLlmExecutionTools.assistantCache.set(this.title, {
18062
19933
  assistantId: assistant.assistantId,
@@ -18073,6 +19944,7 @@ class AgentLlmExecutionTools {
18073
19944
  name: this.title,
18074
19945
  instructions: modelRequirements.systemMessage,
18075
19946
  knowledgeSources: modelRequirements.knowledgeSources,
19947
+ tools: modelRequirements.tools ? [...modelRequirements.tools] : undefined,
18076
19948
  /*
18077
19949
  !!!
18078
19950
  metadata: {
@@ -18085,32 +19957,28 @@ class AgentLlmExecutionTools {
18085
19957
  requirementsHash,
18086
19958
  });
18087
19959
  }
18088
- underlyingLlmResult = await assistant.callChatModelStream(chatPrompt, onProgress);
19960
+ // Create modified chat prompt with agent system message specific to OpenAI Assistant
19961
+ const promptWithAgentModelRequirementsForOpenAiAssistantExecutionTools = {
19962
+ ...promptWithAgentModelRequirements,
19963
+ modelRequirements: {
19964
+ ...promptWithAgentModelRequirements.modelRequirements,
19965
+ modelName: undefined,
19966
+ systemMessage: undefined,
19967
+ temperature: undefined, // <- Note: Let the Assistant use its default temperature
19968
+ },
19969
+ };
19970
+ console.log('!!!! promptWithAgentModelRequirementsForOpenAiAssistantExecutionTools:', promptWithAgentModelRequirementsForOpenAiAssistantExecutionTools);
19971
+ underlyingLlmResult = await assistant.callChatModelStream(promptWithAgentModelRequirementsForOpenAiAssistantExecutionTools, onProgress);
18089
19972
  }
18090
19973
  else {
18091
19974
  if (this.options.isVerbose) {
18092
19975
  console.log(`2️⃣ Creating Assistant ${this.title} on generic LLM execution tools...`);
18093
19976
  }
18094
- // Create modified chat prompt with agent system message
18095
- const modifiedChatPrompt = {
18096
- ...chatPrompt,
18097
- modelRequirements: {
18098
- ...chatPrompt.modelRequirements,
18099
- ...modelRequirements,
18100
- // Spread tools to convert readonly array to mutable
18101
- tools: modelRequirements.tools ? [...modelRequirements.tools] : chatPrompt.modelRequirements.tools,
18102
- // Prepend agent system message to existing system message
18103
- systemMessage: modelRequirements.systemMessage +
18104
- (chatPrompt.modelRequirements.systemMessage
18105
- ? `\n\n${chatPrompt.modelRequirements.systemMessage}`
18106
- : ''),
18107
- },
18108
- };
18109
19977
  if (this.options.llmTools.callChatModelStream) {
18110
- underlyingLlmResult = await this.options.llmTools.callChatModelStream(modifiedChatPrompt, onProgress);
19978
+ underlyingLlmResult = await this.options.llmTools.callChatModelStream(promptWithAgentModelRequirements, onProgress);
18111
19979
  }
18112
19980
  else if (this.options.llmTools.callChatModel) {
18113
- underlyingLlmResult = await this.options.llmTools.callChatModel(modifiedChatPrompt);
19981
+ underlyingLlmResult = await this.options.llmTools.callChatModel(promptWithAgentModelRequirements);
18114
19982
  onProgress(underlyingLlmResult);
18115
19983
  }
18116
19984
  else {
@@ -18139,7 +20007,7 @@ AgentLlmExecutionTools.assistantCache = new Map();
18139
20007
  * TODO: [🧠] Adding parameter substitution support (here or should be responsibility of the underlying LLM Tools)
18140
20008
  */
18141
20009
 
18142
- var _Agent_instances, _Agent_selfLearn;
20010
+ var _Agent_instances, _Agent_selfLearnNonce, _Agent_selfLearnSamples, _Agent_selfLearnTeacher;
18143
20011
  /**
18144
20012
  * Represents one AI Agent
18145
20013
  *
@@ -18199,22 +20067,33 @@ class Agent extends AgentLlmExecutionTools {
18199
20067
  * This is parsed from commitments like USE BROWSER, USE SEARCH ENGINE, KNOWLEDGE, etc.
18200
20068
  */
18201
20069
  this.capabilities = [];
20070
+ /**
20071
+ * List of sample conversations (question/answer pairs)
20072
+ */
20073
+ this.samples = [];
18202
20074
  /**
18203
20075
  * Metadata like image or color
18204
20076
  */
18205
20077
  this.meta = {};
20078
+ /**
20079
+ * Human-readable titles for tool functions
20080
+ */
20081
+ this.toolTitles = {};
18206
20082
  // TODO: [🐱‍🚀] Add `Agent` simple "mocked" learning by appending to agent source
18207
20083
  // TODO: [🐱‍🚀] Add `Agent` learning by promptbookAgent
20084
+ this.teacherAgent = options.teacherAgent;
18208
20085
  this.agentSource = agentSource;
18209
20086
  this.agentSource.subscribe((source) => {
18210
20087
  this.updateAgentSource(source);
18211
- const { agentName, personaDescription, initialMessage, links, meta, capabilities } = parseAgentSource(source);
20088
+ const { agentName, personaDescription, initialMessage, links, meta, capabilities, samples } = parseAgentSource(source);
18212
20089
  this._agentName = agentName;
18213
20090
  this.personaDescription = personaDescription;
18214
20091
  this.initialMessage = initialMessage;
18215
20092
  this.links = links;
18216
20093
  this.capabilities = capabilities;
20094
+ this.samples = samples;
18217
20095
  this.meta = { ...this.meta, ...meta };
20096
+ this.toolTitles = getAllCommitmentsToolTitles();
18218
20097
  });
18219
20098
  }
18220
20099
  /**
@@ -18225,10 +20104,10 @@ class Agent extends AgentLlmExecutionTools {
18225
20104
  async callChatModelStream(prompt, onProgress) {
18226
20105
  var _a;
18227
20106
  // [1] Check if the user is asking the same thing as in the samples
18228
- const modelRequirements = await this.getAgentModelRequirements();
20107
+ const modelRequirements = await this.getModelRequirements();
18229
20108
  if (modelRequirements.samples) {
18230
20109
  const normalizedPrompt = normalizeMessageText(prompt.content);
18231
- const sample = modelRequirements.samples.find((sample) => normalizeMessageText(sample.question) === normalizedPrompt);
20110
+ const sample = modelRequirements.samples.find((sample) => sample.question !== null && normalizeMessageText(sample.question) === normalizedPrompt);
18232
20111
  if (sample) {
18233
20112
  const now = new Date().toISOString();
18234
20113
  const result = {
@@ -18274,21 +20153,42 @@ class Agent extends AgentLlmExecutionTools {
18274
20153
  if ((_a = modelRequirements.metadata) === null || _a === void 0 ? void 0 : _a.isClosed) {
18275
20154
  return result;
18276
20155
  }
18277
- await __classPrivateFieldGet(this, _Agent_instances, "m", _Agent_selfLearn).call(this, prompt, result);
18278
- // <- TODO: !!!! Do not await self-learn, run in background with error handling
20156
+ // TODO: !!!!! Return the answer and do the learning asynchronously
20157
+ // Note: [0] Asynchronously add nonce
20158
+ if (just(false)) {
20159
+ await __classPrivateFieldGet(this, _Agent_instances, "m", _Agent_selfLearnNonce).call(this);
20160
+ }
20161
+ // Note: [1] Do the append of the samples
20162
+ await __classPrivateFieldGet(this, _Agent_instances, "m", _Agent_selfLearnSamples).call(this, prompt, result);
20163
+ // Note: [2] Asynchronously call the teacher agent and invoke the silver link. When the teacher fails, keep just the samples
20164
+ await __classPrivateFieldGet(this, _Agent_instances, "m", _Agent_selfLearnTeacher).call(this, prompt, result).catch((error) => {
20165
+ // !!!!! if (this.options.isVerbose) {
20166
+ console.error(colors.bgCyan('[Self-learning]') + colors.red(' Failed to learn from teacher agent'));
20167
+ console.error(error);
20168
+ // }
20169
+ });
18279
20170
  return result;
18280
20171
  }
18281
20172
  }
18282
- _Agent_instances = new WeakSet(), _Agent_selfLearn =
20173
+ _Agent_instances = new WeakSet(), _Agent_selfLearnNonce =
18283
20174
  /**
18284
- * Self-learning: Appends the conversation and extracted knowledge to the agent source
20175
+ * Self-learning Step 0: Asynchronously with random timing add nonce to the agent source
18285
20176
  */
18286
- async function _Agent_selfLearn(prompt, result) {
18287
- // Learning: Append the conversation sample to the agent source
20177
+ async function _Agent_selfLearnNonce() {
20178
+ await forTime(Math.random() * 5000);
20179
+ // <- TODO: [🕓] `await forRandom(...)`
20180
+ console.info(colors.bgCyan('[Self-learning]') + colors.cyan(' Nonce'));
20181
+ const nonce = `NONCE ${await linguisticHash(Math.random().toString())}`;
20182
+ // Append to the current source
20183
+ const currentSource = this.agentSource.value;
20184
+ const newSource = padBook(validateBook(spaceTrim$2(currentSource) + '\n\n---\n\n' + nonce));
20185
+ // <- TODO: [🈲] Use some object-based way how to append on book (with sections `---`)
20186
+ // Update the source (which will trigger the subscription and update the underlying tools)
20187
+ this.agentSource.next(newSource);
20188
+ }, _Agent_selfLearnSamples = function _Agent_selfLearnSamples(prompt, result) {
20189
+ console.info(colors.bgCyan('[Self-learning]') + colors.cyan(' Sampling'));
18288
20190
  const learningExample = spaceTrim$2((block) => `
18289
20191
 
18290
- ---
18291
-
18292
20192
  USER MESSAGE
18293
20193
  ${block(prompt.content)}
18294
20194
 
@@ -18296,53 +20196,81 @@ async function _Agent_selfLearn(prompt, result) {
18296
20196
  ${block(result.content)}
18297
20197
 
18298
20198
  `);
18299
- // Extract knowledge
18300
- let knowledgeBlock = '';
18301
- try {
18302
- const extractionPrompt = {
18303
- title: 'Knowledge Extraction',
18304
- modelRequirements: {
18305
- modelVariant: 'CHAT',
18306
- },
18307
- content: spaceTrim$2((block) => `
18308
- You are an AI agent that is learning from a conversation.
20199
+ // Append to the current source
20200
+ const currentSource = this.agentSource.value;
20201
+ const newSource = padBook(validateBook(spaceTrim$2(currentSource) + '\n\n---\n\n' + learningExample));
20202
+ // <- TODO: [🈲] Use some object-based way how to append on book (with sections `---`)
20203
+ // Update the source (which will trigger the subscription and update the underlying tools)
20204
+ this.agentSource.next(newSource);
20205
+ }, _Agent_selfLearnTeacher =
20206
+ /**
20207
+ * Self-learning Step 2: Asynchronously call the teacher agent and invoke the silver link
20208
+ */
20209
+ async function _Agent_selfLearnTeacher(prompt, result) {
20210
+ // [1] Call the teacher agent // <- !!!!! Emojis
20211
+ if (this.teacherAgent === null) {
20212
+ return;
20213
+ }
20214
+ console.info(colors.bgCyan('[Self-learning]') + colors.cyan(' Teacher'));
20215
+ const teacherResult = await this.teacherAgent.callChatModel({
20216
+ title: 'Self-learning',
20217
+ modelRequirements: {
20218
+ modelVariant: 'CHAT',
20219
+ },
20220
+ // TODO: !!!! Use prompt notation
20221
+ content: spaceTrim$2((block) => `
18309
20222
 
18310
- Here is the conversation so far:
20223
+ You are a teacher agent helping another agent to learn from its interactions.
18311
20224
 
18312
- User: ${block(prompt.content)}
18313
- Agent: ${block(result.content)}
20225
+ Here is your current client which you are teaching:
18314
20226
 
18315
- Extract any new knowledge, facts, or important information that should be remembered for future interactions.
18316
- Format the output as a list of KNOWLEDGE blocks.
18317
- If there is no new knowledge, return nothing.
20227
+ \`\`\`book
20228
+ ${block(this.agentSource.value)}
20229
+ \`\`\`
18318
20230
 
18319
- Example output:
18320
- KNOWLEDGE The user's name is Alice.
18321
- KNOWLEDGE The project deadline is next Friday.
18322
- `),
18323
- pipelineUrl: 'https://github.com/webgptorg/promptbook/blob/main/prompts/knowledge-extraction.ptbk.md',
18324
- parameters: {},
18325
- };
18326
- if (this.options.llmTools.callChatModel) {
18327
- const extractionResult = await this.options.llmTools.callChatModel(extractionPrompt);
18328
- const extractedContent = extractionResult.content;
18329
- if (extractedContent.includes('KNOWLEDGE')) {
18330
- knowledgeBlock = '\n\n' + spaceTrim$2(extractedContent);
18331
- }
18332
- }
18333
- else {
18334
- // TODO: [🧠] Fallback to callChatModelStream if callChatModel is not available
18335
- }
18336
- }
18337
- catch (error) {
18338
- if (this.options.isVerbose) {
18339
- console.warn('Failed to extract knowledge', error);
18340
- }
20231
+ **And here is the latest interaction:**
20232
+
20233
+ **User:**
20234
+ ${block(prompt.content)}
20235
+
20236
+ **Agent:**
20237
+ ${block(result.content)}
20238
+
20239
+
20240
+ **Rules:**
20241
+
20242
+ - Decide what the agent should learn from this interaction.
20243
+ - Append new commitments at the end of the agent source.
20244
+ - Do not modify the current agent source, just return new commitments (KNOWLEDGE, RULE, etc.).
20245
+ - If there is nothing new to learn, return empty book code block
20246
+ - Wrap the commitments in a book code block.
20247
+ - Do not explain anything, just return the commitments wrapped in a book code block.
20248
+ - Write the learned commitments in the same style and language as in the original agent source.
20249
+
20250
+
20251
+ This is how book code block looks like:
20252
+
20253
+ \`\`\`book
20254
+ KNOWLEDGE The sky is blue.
20255
+ RULE Always be polite.
20256
+ \`\`\`
20257
+ `),
20258
+ // pipelineUrl: 'https://github.com/webgptorg/promptbook/blob/main/prompts/self-learning.ptbk.md',
20259
+ // <- TODO: !!!! Remove and `pipelineUrl` for agent purposes
20260
+ parameters: {},
20261
+ });
20262
+ console.log('!!!! teacherResult', teacherResult);
20263
+ const teacherCommitments = unwrapResult(teacherResult.content);
20264
+ if (teacherCommitments === '') {
20265
+ console.info(colors.bgCyan('[Self-learning]') +
20266
+ colors.cyan(' Teacher agent did not provide new commitments to learn'));
20267
+ return;
18341
20268
  }
18342
- // Append to the current source
20269
+ // [2] Append to the current source
18343
20270
  const currentSource = this.agentSource.value;
18344
- const newSource = padBook(validateBook(spaceTrim$2(currentSource) + '\n\n' + learningExample + knowledgeBlock));
18345
- // Update the source (which will trigger the subscription and update the underlying tools)
20271
+ const newSource = padBook(validateBook(spaceTrim$2(currentSource) + '\n\n' + teacherCommitments));
20272
+ // <- TODO: [🈲] Use some object-based way how to append on book (with sections `---`)
20273
+ // [3] Update the source
18346
20274
  this.agentSource.next(newSource);
18347
20275
  };
18348
20276
  /**
@@ -18363,10 +20291,25 @@ async function _Agent_selfLearn(prompt, result) {
18363
20291
  */
18364
20292
  class RemoteAgent extends Agent {
18365
20293
  static async connect(options) {
18366
- console.log('[🐱‍🚀]', `${options.agentUrl}/api/profile`);
18367
- const profileResponse = await fetch(`${options.agentUrl}/api/profile`);
20294
+ const agentProfileUrl = `${options.agentUrl}/api/profile`;
20295
+ const profileResponse = await fetch(agentProfileUrl);
18368
20296
  // <- TODO: [🐱‍🚀] What about closed-source agents?
18369
20297
  // <- TODO: [🐱‍🚀] Maybe use promptbookFetch
20298
+ if (!profileResponse.ok) {
20299
+ throw new Error(spaceTrim$2((block) => `
20300
+ Failed to fetch remote agent profile:
20301
+
20302
+ Agent URL:
20303
+ ${options.agentUrl}
20304
+
20305
+ Agent Profile URL:
20306
+ ${agentProfileUrl}
20307
+
20308
+ Http Error:
20309
+ ${block(profileResponse.statusText)}
20310
+
20311
+ `));
20312
+ }
18370
20313
  const profile = await profileResponse.json();
18371
20314
  // Note: We are creating dummy agent source because we don't have the source from the remote agent
18372
20315
  // But we populate the metadata from the profile
@@ -18393,6 +20336,7 @@ class RemoteAgent extends Agent {
18393
20336
  */
18394
20337
  },
18395
20338
  agentSource,
20339
+ teacherAgent: null, // <- Note:
18396
20340
  });
18397
20341
  remoteAgent._remoteAgentName = profile.agentName;
18398
20342
  remoteAgent._remoteAgentHash = profile.agentHash;
@@ -18400,11 +20344,13 @@ class RemoteAgent extends Agent {
18400
20344
  remoteAgent.initialMessage = profile.initialMessage;
18401
20345
  remoteAgent.links = profile.links;
18402
20346
  remoteAgent.meta = profile.meta;
20347
+ remoteAgent.toolTitles = profile.toolTitles || {};
18403
20348
  remoteAgent._isVoiceCallingEnabled = profile.isVoiceCallingEnabled === true; // [✨✷] Store voice calling status
18404
20349
  return remoteAgent;
18405
20350
  }
18406
20351
  constructor(options) {
18407
20352
  super(options);
20353
+ this.toolTitles = {};
18408
20354
  this._isVoiceCallingEnabled = false; // [✨✷] Track voice calling status
18409
20355
  this.agentUrl = options.agentUrl;
18410
20356
  }
@@ -18484,6 +20430,7 @@ class RemoteAgent extends Agent {
18484
20430
  // <- TODO: [🐱‍🚀] What about closed-source agents?
18485
20431
  // <- TODO: [🐱‍🚀] Maybe use promptbookFetch
18486
20432
  let content = '';
20433
+ const toolCalls = [];
18487
20434
  if (!bookResponse.body) {
18488
20435
  content = await bookResponse.text();
18489
20436
  }
@@ -18499,6 +20446,36 @@ class RemoteAgent extends Agent {
18499
20446
  doneReading = !!done;
18500
20447
  if (value) {
18501
20448
  const textChunk = decoder.decode(value, { stream: true });
20449
+ let isHandled = false;
20450
+ try {
20451
+ const lines = textChunk.split('\n');
20452
+ for (const line of lines) {
20453
+ const trimmedLine = line.trim();
20454
+ if (trimmedLine.startsWith('{') && trimmedLine.endsWith('}')) {
20455
+ const chunk = JSON.parse(trimmedLine);
20456
+ if (chunk.toolCalls) {
20457
+ toolCalls.push(...chunk.toolCalls);
20458
+ onProgress({
20459
+ content,
20460
+ modelName: this.modelName,
20461
+ timing: {},
20462
+ usage: {},
20463
+ rawPromptContent: {},
20464
+ rawRequest: {},
20465
+ rawResponse: {},
20466
+ toolCalls: chunk.toolCalls,
20467
+ });
20468
+ isHandled = true;
20469
+ }
20470
+ }
20471
+ }
20472
+ }
20473
+ catch (error) {
20474
+ // Ignore non-json chunks
20475
+ }
20476
+ if (isHandled) {
20477
+ continue;
20478
+ }
18502
20479
  // console.debug('RemoteAgent chunk:', textChunk);
18503
20480
  content += textChunk;
18504
20481
  onProgress({
@@ -18509,6 +20486,7 @@ class RemoteAgent extends Agent {
18509
20486
  rawPromptContent: {},
18510
20487
  rawRequest: {},
18511
20488
  rawResponse: {},
20489
+ toolCalls,
18512
20490
  });
18513
20491
  }
18514
20492
  }
@@ -18524,6 +20502,7 @@ class RemoteAgent extends Agent {
18524
20502
  rawPromptContent: {},
18525
20503
  rawRequest: {},
18526
20504
  rawResponse: {},
20505
+ toolCalls,
18527
20506
  });
18528
20507
  }
18529
20508
  }
@@ -18540,6 +20519,7 @@ class RemoteAgent extends Agent {
18540
20519
  rawPromptContent: {},
18541
20520
  rawRequest: {},
18542
20521
  rawResponse: {},
20522
+ toolCalls,
18543
20523
  // <- TODO: [🐱‍🚀] Transfer and proxy the metadata
18544
20524
  };
18545
20525
  return agentResult;
@@ -18628,7 +20608,7 @@ function PromptbookAgentSeamlessIntegration(props) {
18628
20608
  ? styles.PromptbookAgentSeamlessIntegrationStatusConnected
18629
20609
  : connectionStatus === 'error'
18630
20610
  ? styles.PromptbookAgentSeamlessIntegrationStatusError
18631
- : styles.PromptbookAgentSeamlessIntegrationStatusPending}` }), jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLabel, children: "CHAT" })] }), isOpen && (jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationWindow, children: [jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationHeader, style: { backgroundColor: color }, ref: setHeaderElement, children: [jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationTitle, children: (agent === null || agent === void 0 ? void 0 : agent.meta.fullname) || (meta === null || meta === void 0 ? void 0 : meta.fullname) || (agent === null || agent === void 0 ? void 0 : agent.agentName) || 'Chat with Agent' }), isIframeUsed && (jsx("button", { className: styles.PromptbookAgentSeamlessIntegrationClose, onClick: () => setIsOpen(false), title: "Close", children: jsx(CloseIcon, {}) }))] }), jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationContent, children: isIframeUsed ? (jsxs(Fragment, { children: [!isIframeLoaded && (jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLoading, children: "Loading chat..." })), jsx("iframe", { src: agentUrl + '/chat?headless', className: styles.PromptbookAgentSeamlessIntegrationIframe, style: { opacity: isIframeLoaded ? 1 : 0 }, tabIndex: -1, onLoad: () => setIsIframeLoaded(true) })] })) : agent ? (jsx(AgentChat, { agent: agent, actionsContainer: headerElement, isFocusedOnLoad: isFocusedOnLoad, extraActions: jsx("button", { className: styles.PromptbookAgentSeamlessIntegrationClose, onClick: () => setIsOpen(false), title: "Close", children: jsx(CloseIcon, {}) }) })) : error ? (jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationError, children: ["Failed to connect to agent: ", error.message] })) : (jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLoading, children: "Connecting to agent..." })) })] }))] }));
20611
+ : styles.PromptbookAgentSeamlessIntegrationStatusPending}` }), jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLabel, children: "CHAT" })] }), isOpen && (jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationWindow, children: [jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationHeader, style: { backgroundColor: color }, ref: setHeaderElement, children: [jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationTitle, children: (agent === null || agent === void 0 ? void 0 : agent.meta.fullname) || (meta === null || meta === void 0 ? void 0 : meta.fullname) || (agent === null || agent === void 0 ? void 0 : agent.agentName) || 'Chat with Agent' }), isIframeUsed && (jsx("button", { className: styles.PromptbookAgentSeamlessIntegrationClose, onClick: () => setIsOpen(false), title: "Close", children: jsx(CloseIcon, {}) }))] }), jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationContent, children: isIframeUsed ? (jsxs(Fragment, { children: [!isIframeLoaded && (jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLoading, children: "Loading chat..." })), jsx("iframe", { src: agentUrl + '/chat?headless', className: styles.PromptbookAgentSeamlessIntegrationIframe, style: { opacity: isIframeLoaded ? 1 : 0 }, tabIndex: -1, onLoad: () => setIsIframeLoaded(true) })] })) : agent ? (jsx(AgentChat, { agent: agent, actionsContainer: headerElement, isFocusedOnLoad: isFocusedOnLoad, extraActions: jsx("button", { className: styles.PromptbookAgentSeamlessIntegrationClose, onClick: () => setIsOpen(false), title: "Close", children: jsx(CloseIcon, {}) }), visual: "STANDALONE" })) : error ? (jsxs("div", { className: styles.PromptbookAgentSeamlessIntegrationError, children: ["Failed to connect to agent: ", error.message] })) : (jsx("div", { className: styles.PromptbookAgentSeamlessIntegrationLoading, children: "Connecting to agent..." })) })] }))] }));
18632
20612
  }
18633
20613
 
18634
20614
  /**