|
73 | 73 | let timestampStart = new Date(2020, 7, 9).getTime();
|
74 | 74 | const timestamp = () => new Date(timestampStart += 100).toISOString();
|
75 | 75 |
|
76 |
| - const [leftTranscriptIndex = 0, rightTranscriptIndex = 1] = new URLSearchParams(location.search).getAll('transcript'); |
77 |
| - const [leftVariant = 'fluent', rightVariant = 'copilot'] = new URLSearchParams(location.search).getAll('variant'); |
| 76 | + const searchParams = new URLSearchParams(location.search); |
78 | 77 |
|
79 |
| - const sendBoxIndexes = new URLSearchParams(location.search).getAll('focus'); |
| 78 | + const [leftTranscriptIndex = 0, rightTranscriptIndex = 1] = searchParams.getAll('transcript'); |
| 79 | + const [leftVariant = 'fluent', rightVariant = 'copilot'] = searchParams.getAll('variant'); |
| 80 | + |
| 81 | + const sendBoxIndexes = searchParams.getAll('focus'); |
| 82 | + const activeFocusPresets = searchParams.getAll('focus-preset'); |
| 83 | + |
| 84 | + const adjustStyleOptions = Object.fromEntries(['codeBlockTheme'] |
| 85 | + .map(key => [key, searchParams.get(key)]) |
| 86 | + .filter(([key, value]) => value)); |
80 | 87 |
|
81 | 88 | const botIcon = '';
|
82 | 89 |
|
|
94 | 101 | }
|
95 | 102 | };
|
96 | 103 |
|
| 104 | + const waveSvg = `data:image/svg+xml;utf8,${encodeURIComponent(` |
| 105 | +<svg width="400" height="200" viewBox="0 0 400 200" xmlns="http://www.w3.org/2000/svg"> |
| 106 | + <!-- Primary Wave --> |
| 107 | + <path d="M0,100 C50,50 100,150 150,100 C200,50 250,150 300,100 C350,50 400,150 400,100" |
| 108 | + stroke="#3B82F6" fill="none" stroke-width="2" opacity="0.5"/> |
| 109 | + |
| 110 | + <!-- Second Harmonic --> |
| 111 | + <path d="M0,100 C25,75 50,125 75,100 C100,75 125,125 150,100 C175,75 200,125 225,100 C250,75 275,125 300,100 C325,75 350,125 375,100 C400,75 400,125 400,100" |
| 112 | + stroke="#10B981" fill="none" stroke-width="2" opacity="0.5"/> |
| 113 | + |
| 114 | + <!-- Combined Wave --> |
| 115 | + <path d="M0,100 C40,30 80,170 120,100 C160,30 200,170 240,100 C280,30 320,170 360,100 C380,65 400,135 400,100" |
| 116 | + stroke="#EF4444" fill="none" stroke-width="3"/> |
| 117 | + |
| 118 | + <!-- Grid Lines --> |
| 119 | + <line x1="0" y1="100" x2="400" y2="100" stroke="#CBD5E1" stroke-width="0.5" stroke-dasharray="4"/> |
| 120 | + <line x1="100" y1="0" x2="100" y2="200" stroke="#CBD5E1" stroke-width="0.5" stroke-dasharray="4"/> |
| 121 | + <line x1="200" y1="0" x2="200" y2="200" stroke="#CBD5E1" stroke-width="0.5" stroke-dasharray="4"/> |
| 122 | + <line x1="300" y1="0" x2="300" y2="200" stroke="#CBD5E1" stroke-width="0.5" stroke-dasharray="4"/> |
| 123 | +</svg>`)}`; |
| 124 | + |
97 | 125 | const transcripts = [[
|
98 | 126 | {
|
99 | 127 | timestamp: timestamp(),
|
|
511 | 539 | streamSequence: 99
|
512 | 540 | }
|
513 | 541 | }
|
| 542 | + ], [ |
| 543 | + { |
| 544 | + timestamp: timestamp(), |
| 545 | + from: { role: 'bot' }, |
| 546 | + entities: [{ |
| 547 | + ...aiMessageEntity, |
| 548 | + isBasedOn: { |
| 549 | + '@type': 'SoftwareSourceCode', |
| 550 | + 'programmingLanguage': 'python', |
| 551 | + "text": `import numpy as np |
| 552 | +import matplotlib.pyplot as plt |
| 553 | +
|
| 554 | +def plot_sine_waves(): |
| 555 | + """Create a beautiful visualization of sine waves with different frequencies.""" |
| 556 | + # Generate time points |
| 557 | + t = np.linspace(0, 10, 1000) |
| 558 | + |
| 559 | + # Create waves with different frequencies and phases |
| 560 | + wave1 = np.sin(t) |
| 561 | + wave2 = 0.5 * np.sin(2 * t + np.pi/4) |
| 562 | + wave3 = 0.3 * np.sin(3 * t + np.pi/3) |
| 563 | + |
| 564 | + # Combine waves |
| 565 | + combined = wave1 + wave2 + wave3 |
| 566 | + |
| 567 | + # Create a stylish plot |
| 568 | + plt.style.use('seaborn-darkgrid') |
| 569 | + plt.figure(figsize=(12, 8)) |
| 570 | + |
| 571 | + # Plot individual waves |
| 572 | + plt.plot(t, wave1, label='Primary Wave', alpha=0.5) |
| 573 | + plt.plot(t, wave2, label='Second Harmonic', alpha=0.5) |
| 574 | + plt.plot(t, wave3, label='Third Harmonic', alpha=0.5) |
| 575 | + |
| 576 | + # Plot combined wave with a thicker line |
| 577 | + plt.plot(t, combined, 'r-', |
| 578 | + label='Combined Wave', |
| 579 | + linewidth=2) |
| 580 | + |
| 581 | + plt.title('Harmonic Wave Composition', fontsize=14) |
| 582 | + plt.xlabel('Time', fontsize=12) |
| 583 | + plt.ylabel('Amplitude', fontsize=12) |
| 584 | + plt.legend() |
| 585 | + plt.grid(True, alpha=0.3) |
| 586 | + |
| 587 | + # Show the plot |
| 588 | + plt.tight_layout() |
| 589 | + plt.show() |
| 590 | +
|
| 591 | +# Generate the visualization |
| 592 | +plot_sine_waves()` |
| 593 | + } |
| 594 | + }], |
| 595 | + id: "a4c0c01d-c06e-4dde-9278-265c607b545b-82", |
| 596 | + type: "message", |
| 597 | + text: `This example demonstrates creating a beautiful visualization of harmonic waves using Python's Matplotlib library. The code generates three sine waves with different frequencies and phases, then combines them to show wave interference patterns.\n<img alt="wave plot" src="${waveSvg}">`, |
| 598 | + } |
514 | 599 | ]];
|
515 | 600 |
|
516 | 601 | const leftStore = testHelpers.createStore();
|
|
557 | 642 | TEXT_INPUT_PLACEHOLDER: 'Describe how your copilot should behave'
|
558 | 643 | }}
|
559 | 644 | styleOptions={{
|
560 |
| - maxMessageLength: null |
| 645 | + maxMessageLength: null, |
| 646 | + ...adjustStyleOptions |
561 | 647 | }}
|
562 | 648 | />
|
563 | 649 | </FluentThemeProvider>
|
|
568 | 654 | directLine={rightDirectLine}
|
569 | 655 | store={rightStore}
|
570 | 656 | overrideLocalizedStrings={{
|
571 |
| - TEXT_INPUT_PLACEHOLDER: 'Ask a work question or use / to reference people, files and more' |
| 657 | + TEXT_INPUT_PLACEHOLDER: 'Ask a work question or use / to reference people, files and more', |
| 658 | + ACTIVITY_CODE_CAUTION: 'AI-generated code. Review and use carefully. [More info on FAQ](https://aka.ms/AIandCopilotFAQs).' |
572 | 659 | }}
|
573 | 660 | styleOptions={{
|
574 | 661 | groupTimestamp: 1,
|
575 | 662 | timestampFormat: 'absolute',
|
576 |
| - botAvatarBackgroundColor: '#304E7A' |
| 663 | + botAvatarBackgroundColor: '#304E7A', |
| 664 | + ...adjustStyleOptions |
577 | 665 | }}
|
578 | 666 | />
|
579 | 667 | </FluentThemeProvider>
|
|
586 | 674 |
|
587 | 675 | await host.snapshot();
|
588 | 676 |
|
589 |
| - const sendboxes = Array.from(document.querySelectorAll(`[data-testid="${WebChat.testIds.sendBoxTextBox}"]`)).filter((_, index) => sendBoxIndexes.includes(String(index))); |
590 |
| - |
591 |
| - for (const sendbox of sendboxes) { |
592 |
| - sendbox.focus(); |
593 |
| - await host.sendKeys('Next message'); |
594 |
| - await host.snapshot(); |
595 |
| - await host.sendKeys('ENTER'); |
596 |
| - // hide timestamp to not fail snapshot if is not relative |
597 |
| - sendbox.parentElement.parentElement.parentElement.parentElement.querySelector('.webchat__basic-transcript__activity:last-child .webchat__stacked-layout__status').style.opacity = 0 |
598 |
| - await host.snapshot(); |
599 |
| - await host.sendShiftTab(); |
600 |
| - await host.snapshot(); |
601 |
| - await host.sendKeys('HOME'); |
602 |
| - await host.snapshot(); |
603 |
| - await host.sendKeys('END'); |
604 |
| - await host.snapshot(); |
605 |
| - await host.sendKeys('ARROW_UP'); |
606 |
| - await host.snapshot(); |
607 |
| - await host.sendKeys('ENTER'); |
608 |
| - await host.sendShiftTab(); |
609 |
| - await host.snapshot(); |
610 |
| - await host.sendKeys('ENTER'); |
611 |
| - await host.snapshot(); |
612 |
| - await host.sendKeys('ESCAPE'); |
| 677 | + const focusPresetsMap = new Map(Object.entries({ |
| 678 | + navigation: async sendbox => { |
| 679 | + sendbox.focus(); |
| 680 | + await host.sendKeys('Next message'); |
| 681 | + await host.snapshot(); |
| 682 | + await host.sendKeys('ENTER'); |
| 683 | + // hide timestamp to not fail snapshot if is not relative |
| 684 | + sendbox.parentElement.parentElement.parentElement.parentElement.querySelector('.webchat__basic-transcript__activity:last-child .webchat__stacked-layout__status').style.opacity = 0 |
| 685 | + await host.snapshot(); |
| 686 | + await host.sendShiftTab(); |
| 687 | + await host.snapshot(); |
| 688 | + await host.sendKeys('HOME'); |
| 689 | + await host.snapshot(); |
| 690 | + await host.sendKeys('END'); |
| 691 | + await host.snapshot(); |
| 692 | + await host.sendKeys('ARROW_UP'); |
| 693 | + await host.snapshot(); |
| 694 | + await host.sendKeys('ENTER'); |
| 695 | + await host.sendShiftTab(); |
| 696 | + await host.snapshot(); |
| 697 | + await host.sendKeys('ENTER'); |
| 698 | + await host.snapshot(); |
| 699 | + await host.sendKeys('ESCAPE'); |
| 700 | + }, |
| 701 | + viewCode: async sendbox => { |
| 702 | + sendbox.focus(); |
| 703 | + await host.sendShiftTab(); |
| 704 | + await host.sendKeys('ARROW_UP'); |
| 705 | + await host.sendKeys('ENTER'); |
| 706 | + await host.snapshot(); |
| 707 | + await host.sendKeys('ENTER'); |
| 708 | + await host.snapshot(); |
| 709 | + await host.sendKeys('ENTER'); |
| 710 | + await host.snapshot(); |
| 711 | + } |
| 712 | + })); |
| 713 | + |
| 714 | + const sendboxes = Array.from(document.querySelectorAll(`[data-testid="${WebChat.testIds.sendBoxTextBox}"]`)) |
| 715 | + .filter((_, index) => sendBoxIndexes.includes(String(index))); |
| 716 | + |
| 717 | + const focusPresets = activeFocusPresets |
| 718 | + .map(presetName => focusPresetsMap.get(presetName)) |
| 719 | + .filter(preset => preset) |
| 720 | + |
| 721 | + if (!focusPresets.length) focusPresets.push(focusPresetsMap.get('navigation')); |
| 722 | + |
| 723 | + for (const preset of focusPresets) { |
| 724 | + for (const sendbox of sendboxes) { |
| 725 | + await preset(sendbox) |
| 726 | + } |
613 | 727 | }
|
614 | 728 | });
|
615 | 729 | // TODO: unskip and try ShadowDOM when we get support
|
|
0 commit comments