@copilotkitnext/angular 1.52.0-next.8 → 1.52.1-next.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/README.md +225 -432
  2. package/dist/README.md +225 -432
  3. package/dist/fesm2022/copilotkitnext-angular.mjs +322 -164
  4. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  5. package/dist/lib/agent.d.ts +20 -20
  6. package/dist/lib/components/chat/copilot-chat-assistant-message.d.ts +46 -46
  7. package/dist/lib/components/chat/copilot-chat-message-view.d.ts +126 -126
  8. package/dist/lib/components/chat/copilot-chat-tool-calls-view.d.ts +23 -23
  9. package/dist/lib/components/chat/copilot-chat-user-message-branch-navigation.d.ts +6 -6
  10. package/dist/lib/components/chat/copilot-chat-user-message.d.ts +6 -6
  11. package/dist/lib/components/chat/copilot-chat-view-scroll-view.d.ts +18 -18
  12. package/dist/lib/components/chat/copilot-chat-view.d.ts +54 -54
  13. package/dist/lib/components/chat/copilot-chat.d.ts +18 -18
  14. package/dist/lib/config.d.ts +2 -0
  15. package/dist/lib/copilotkit.d.ts +7 -1
  16. package/dist/lib/license-watermark.d.ts +1 -0
  17. package/dist/lib/render-tool-calls.d.ts +23 -23
  18. package/dist/styles.css +3 -0
  19. package/package.json +18 -18
  20. package/dist/esm2022/copilotkitnext-angular.mjs +0 -5
  21. package/dist/esm2022/index.mjs +0 -2
  22. package/dist/esm2022/lib/agent-context.mjs +0 -25
  23. package/dist/esm2022/lib/agent.mjs +0 -73
  24. package/dist/esm2022/lib/chat-config.mjs +0 -35
  25. package/dist/esm2022/lib/chat-state.mjs +0 -18
  26. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-buttons.mjs +0 -344
  27. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-renderer.mjs +0 -260
  28. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-toolbar.mjs +0 -22
  29. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.mjs +0 -415
  30. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.types.mjs +0 -2
  31. package/dist/esm2022/lib/components/chat/copilot-chat-audio-recorder.mjs +0 -196
  32. package/dist/esm2022/lib/components/chat/copilot-chat-buttons.mjs +0 -299
  33. package/dist/esm2022/lib/components/chat/copilot-chat-input-defaults.mjs +0 -39
  34. package/dist/esm2022/lib/components/chat/copilot-chat-input.mjs +0 -634
  35. package/dist/esm2022/lib/components/chat/copilot-chat-input.types.mjs +0 -10
  36. package/dist/esm2022/lib/components/chat/copilot-chat-message-view-cursor.mjs +0 -27
  37. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.mjs +0 -269
  38. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.types.mjs +0 -2
  39. package/dist/esm2022/lib/components/chat/copilot-chat-textarea.mjs +0 -139
  40. package/dist/esm2022/lib/components/chat/copilot-chat-tool-calls-view.mjs +0 -36
  41. package/dist/esm2022/lib/components/chat/copilot-chat-toolbar.mjs +0 -20
  42. package/dist/esm2022/lib/components/chat/copilot-chat-tools-menu.mjs +0 -203
  43. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-branch-navigation.mjs +0 -118
  44. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-buttons.mjs +0 -182
  45. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-renderer.mjs +0 -28
  46. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-toolbar.mjs +0 -25
  47. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.mjs +0 -328
  48. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.types.mjs +0 -2
  49. package/dist/esm2022/lib/components/chat/copilot-chat-view-disclaimer.mjs +0 -48
  50. package/dist/esm2022/lib/components/chat/copilot-chat-view-feather.mjs +0 -41
  51. package/dist/esm2022/lib/components/chat/copilot-chat-view-handlers.mjs +0 -19
  52. package/dist/esm2022/lib/components/chat/copilot-chat-view-input-container.mjs +0 -96
  53. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.mjs +0 -89
  54. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-view.mjs +0 -456
  55. package/dist/esm2022/lib/components/chat/copilot-chat-view.mjs +0 -404
  56. package/dist/esm2022/lib/components/chat/copilot-chat-view.types.mjs +0 -2
  57. package/dist/esm2022/lib/components/chat/copilot-chat.mjs +0 -165
  58. package/dist/esm2022/lib/config.mjs +0 -9
  59. package/dist/esm2022/lib/copilotkit.mjs +0 -125
  60. package/dist/esm2022/lib/directives/copilotkit-agent-context.mjs +0 -130
  61. package/dist/esm2022/lib/directives/stick-to-bottom.mjs +0 -170
  62. package/dist/esm2022/lib/directives/tooltip.mjs +0 -217
  63. package/dist/esm2022/lib/human-in-the-loop.mjs +0 -19
  64. package/dist/esm2022/lib/render-tool-calls.mjs +0 -167
  65. package/dist/esm2022/lib/resize-observer.mjs +0 -152
  66. package/dist/esm2022/lib/scroll-position.mjs +0 -123
  67. package/dist/esm2022/lib/slots/copilot-slot.mjs +0 -156
  68. package/dist/esm2022/lib/slots/index.mjs +0 -4
  69. package/dist/esm2022/lib/slots/slot.types.mjs +0 -6
  70. package/dist/esm2022/lib/slots/slot.utils.mjs +0 -235
  71. package/dist/esm2022/lib/tools.mjs +0 -31
  72. package/dist/esm2022/lib/utils.mjs +0 -10
  73. package/dist/esm2022/public-api.mjs +0 -48
@@ -1,260 +0,0 @@
1
- import { Component, input, ChangeDetectionStrategy, ViewEncapsulation, computed, effect, inject, ElementRef, ViewChild, } from "@angular/core";
2
- import { CommonModule } from "@angular/common";
3
- import { Marked } from "marked";
4
- import hljs from "highlight.js";
5
- import * as katex from "katex";
6
- import { completePartialMarkdown } from "@copilotkitnext/core";
7
- import { LucideAngularModule } from "lucide-angular";
8
- import { injectChatLabels } from "../../chat-config";
9
- import * as i0 from "@angular/core";
10
- export class CopilotChatAssistantMessageRenderer {
11
- content = input("");
12
- inputClass = input();
13
- labels = injectChatLabels();
14
- markdownContainer;
15
- elementRef = inject(ElementRef);
16
- // Track copy states for code blocks (DOM-updated; no signal needed)
17
- copyStates = new Map();
18
- renderedHtml = computed(() => {
19
- const currentContent = this.content();
20
- const completedMarkdown = completePartialMarkdown(currentContent);
21
- return this.renderMarkdown(completedMarkdown);
22
- });
23
- constructor() {
24
- // React to content changes using signals
25
- effect(() => {
26
- // Read content to establish dependency
27
- this.content();
28
- // Reset copy states when content changes
29
- this.copyStates.clear();
30
- // If view is ready, update DOM
31
- if (this.markdownContainer) {
32
- this.updateContent();
33
- this.renderMathEquations();
34
- }
35
- });
36
- }
37
- ngAfterViewInit() {
38
- this.updateContent();
39
- this.renderMathEquations();
40
- }
41
- updateContent() {
42
- if (!this.markdownContainer)
43
- return;
44
- const container = this.markdownContainer.nativeElement;
45
- const html = this.renderedHtml();
46
- container.innerHTML = html;
47
- }
48
- codeBlocksMap = new Map();
49
- markedInstance = null;
50
- initializeMarked() {
51
- if (this.markedInstance)
52
- return;
53
- // Store highlighted code blocks temporarily
54
- const highlightedBlocks = new Map();
55
- // Create a new Marked instance
56
- this.markedInstance = new Marked();
57
- // Configure marked options
58
- this.markedInstance.setOptions({
59
- gfm: true,
60
- breaks: true,
61
- });
62
- // Add a walkTokens function to process code tokens before rendering
63
- this.markedInstance.use({
64
- walkTokens: (token) => {
65
- if (token.type === "code") {
66
- const rawCode = token.text;
67
- const lang = token.lang || "";
68
- const blockId = this.generateBlockId(rawCode);
69
- // Store the raw code in our map for copying
70
- this.codeBlocksMap.set(blockId, rawCode);
71
- const copyLabel = this.labels.assistantMessageToolbarCopyCodeLabel;
72
- // Manually highlight the code
73
- const language = hljs.getLanguage(lang) ? lang : "plaintext";
74
- const highlighted = hljs.highlight(rawCode, { language }).value;
75
- const codeClass = lang ? `hljs language-${lang}` : "hljs";
76
- // Create the full HTML with header and highlighted code
77
- const fullHtml = `
78
- <div class="code-block-container">
79
- <div class="code-block-header">
80
- ${lang ? `<span class="code-block-language">${lang}</span>` : "<span></span>"}
81
- <button
82
- class="code-block-copy-button"
83
- data-code-block-id="${blockId}"
84
- aria-label="${copyLabel} code">
85
- <svg xmlns="http://www.w3.org/2000/svg" width="10" height="10" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect width="14" height="14" x="8" y="8" rx="2" ry="2"/><path d="M4 16c-1.11 0-2-.9-2-2V4c0-1.11.89-2 2-2h10c1.11 0 2 .89 2 2"/></svg>
86
- <span>${copyLabel}</span>
87
- </button>
88
- </div>
89
- <pre><code class="${codeClass}">${highlighted}</code></pre>
90
- </div>
91
- `;
92
- // Store the highlighted HTML
93
- highlightedBlocks.set(blockId, fullHtml);
94
- // Change the token to an html token to bypass marked's escaping
95
- token.type = "html";
96
- token.text = fullHtml;
97
- }
98
- },
99
- });
100
- }
101
- renderMarkdown(content) {
102
- // Initialize marked if not already done
103
- this.initializeMarked();
104
- // Clear the code blocks map for new render
105
- this.codeBlocksMap.clear();
106
- // Parse markdown
107
- let html = this.markedInstance.parse(content);
108
- // Process math equations
109
- html = this.processMathEquations(html);
110
- return html;
111
- }
112
- processMathEquations(html) {
113
- // First, temporarily replace code blocks with placeholders to protect them from math processing
114
- const codeBlocks = [];
115
- const placeholder = "___CODE_BLOCK_PLACEHOLDER_";
116
- // Store code blocks and replace with placeholders
117
- html = html.replace(/<pre><code[\s\S]*?<\/code><\/pre>/g, (match) => {
118
- const index = codeBlocks.length;
119
- codeBlocks.push(match);
120
- return `${placeholder}${index}___`;
121
- });
122
- // Also protect inline code
123
- const inlineCode = [];
124
- const inlinePlaceholder = "___INLINE_CODE_PLACEHOLDER_";
125
- html = html.replace(/<code>[\s\S]*?<\/code>/g, (match) => {
126
- const index = inlineCode.length;
127
- inlineCode.push(match);
128
- return `${inlinePlaceholder}${index}___`;
129
- });
130
- // Process display math $$ ... $$
131
- html = html.replace(/\$\$([\s\S]*?)\$\$/g, (match, equation) => {
132
- try {
133
- return katex.renderToString(equation, {
134
- displayMode: true,
135
- throwOnError: false,
136
- });
137
- }
138
- catch {
139
- return match;
140
- }
141
- });
142
- // Process inline math $ ... $
143
- html = html.replace(/\$([^$]+)\$/g, (match, equation) => {
144
- try {
145
- return katex.renderToString(equation, {
146
- displayMode: false,
147
- throwOnError: false,
148
- });
149
- }
150
- catch {
151
- return match;
152
- }
153
- });
154
- // Restore code blocks
155
- codeBlocks.forEach((block, index) => {
156
- html = html.replace(`${placeholder}${index}___`, block);
157
- });
158
- // Restore inline code
159
- inlineCode.forEach((code, index) => {
160
- html = html.replace(`${inlinePlaceholder}${index}___`, code);
161
- });
162
- return html;
163
- }
164
- renderMathEquations() {
165
- if (!this.markdownContainer)
166
- return;
167
- const container = this.markdownContainer.nativeElement;
168
- // Find all math placeholders and render them
169
- const mathElements = container.querySelectorAll(".math-placeholder");
170
- mathElements.forEach((element) => {
171
- const equation = element.getAttribute("data-equation");
172
- const displayMode = element.getAttribute("data-display") === "true";
173
- if (equation) {
174
- try {
175
- katex.render(equation, element, {
176
- displayMode,
177
- throwOnError: false,
178
- });
179
- }
180
- catch (error) {
181
- console.error("Failed to render math equation:", error);
182
- }
183
- }
184
- });
185
- }
186
- handleClick(event) {
187
- const target = event.target;
188
- // Check if clicked on copy button or its children
189
- const copyButton = target.closest(".code-block-copy-button");
190
- if (copyButton) {
191
- event.preventDefault();
192
- const blockId = copyButton.getAttribute("data-code-block-id");
193
- if (blockId) {
194
- // Get the raw code from our map instead of from DOM
195
- const code = this.codeBlocksMap.get(blockId);
196
- if (code) {
197
- this.copyCodeBlock(blockId, code);
198
- }
199
- }
200
- }
201
- }
202
- copyCodeBlock(blockId, code) {
203
- navigator.clipboard.writeText(code).then(() => {
204
- // Update the button in the DOM
205
- const button = this.elementRef.nativeElement.querySelector(`[data-code-block-id="${blockId}"]`);
206
- if (button) {
207
- const originalHTML = button.innerHTML;
208
- button.innerHTML = `
209
- <svg xmlns="http://www.w3.org/2000/svg" width="10" height="10" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M20 6 9 17l-5-5"/></svg>
210
- <span>${this.labels.assistantMessageToolbarCopyCodeCopiedLabel}</span>
211
- `;
212
- button.setAttribute("aria-label", `${this.labels.assistantMessageToolbarCopyCodeCopiedLabel} code`);
213
- // Reset after 2 seconds
214
- setTimeout(() => {
215
- button.innerHTML = originalHTML;
216
- button.setAttribute("aria-label", `${this.labels.assistantMessageToolbarCopyCodeLabel} code`);
217
- }, 2000);
218
- }
219
- }, (err) => {
220
- console.error("Failed to copy code:", err);
221
- });
222
- }
223
- generateBlockId(code) {
224
- // Simple hash function for generating unique IDs
225
- let hash = 0;
226
- for (let i = 0; i < code.length; i++) {
227
- const char = code.charCodeAt(i);
228
- hash = (hash << 5) - hash + char;
229
- hash = hash & hash; // Convert to 32-bit integer
230
- }
231
- return `code-block-${hash}`;
232
- }
233
- escapeHtml(text) {
234
- const div = document.createElement("div");
235
- div.textContent = text;
236
- return div.innerHTML;
237
- }
238
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChatAssistantMessageRenderer, deps: [], target: i0.ɵɵFactoryTarget.Component });
239
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: CopilotChatAssistantMessageRenderer, isStandalone: true, selector: "copilot-chat-assistant-message-renderer", inputs: { content: { classPropertyName: "content", publicName: "content", isSignal: true, isRequired: false, transformFunction: null }, inputClass: { classPropertyName: "inputClass", publicName: "inputClass", isSignal: true, isRequired: false, transformFunction: null } }, viewQueries: [{ propertyName: "markdownContainer", first: true, predicate: ["markdownContainer"], descendants: true }], ngImport: i0, template: `
240
- <div
241
- #markdownContainer
242
- [class]="inputClass()"
243
- (click)="handleClick($event)"
244
- ></div>
245
- `, isInline: true, styles: ["copilot-chat-assistant-message-renderer{display:block;width:100%}copilot-chat-assistant-message-renderer code:not(pre code){padding:2.5px 4.8px;background-color:#ececec;border-radius:.25rem;font-size:.875rem;font-family:ui-monospace,SFMono-Regular,SF Mono,Consolas,Liberation Mono,Menlo,monospace;font-weight:500;color:#000}.dark copilot-chat-assistant-message-renderer code:not(pre code){background-color:#171717;color:#f8fafc}copilot-chat-assistant-message-renderer .code-block-container{position:relative;margin:.25rem 0;background-color:#f9f9f9;border-radius:1rem}.dark copilot-chat-assistant-message-renderer .code-block-container{background-color:#171717}copilot-chat-assistant-message-renderer .code-block-header{display:flex;align-items:center;justify-content:space-between;padding:.75rem 1rem;font-size:.75rem;background-color:transparent}copilot-chat-assistant-message-renderer .code-block-language{font-weight:400;color:#737373}.dark copilot-chat-assistant-message-renderer .code-block-language{color:#fff}copilot-chat-assistant-message-renderer .code-block-copy-button{display:flex;align-items:center;gap:.125rem;padding:0 .5rem;font-size:.75rem;color:#737373;cursor:pointer;background:none;border:none;transition:opacity .2s}.dark copilot-chat-assistant-message-renderer .code-block-copy-button{color:#fff}copilot-chat-assistant-message-renderer .code-block-copy-button:hover{opacity:.8}copilot-chat-assistant-message-renderer .code-block-copy-button svg{width:10px;height:10px}copilot-chat-assistant-message-renderer .code-block-copy-button span{font-size:11px}copilot-chat-assistant-message-renderer pre{margin:0;padding:0 1rem 1rem;overflow-x:auto;background-color:transparent;border-radius:1rem}.dark copilot-chat-assistant-message-renderer pre{background-color:transparent}copilot-chat-assistant-message-renderer pre code{background-color:transparent;padding:0;font-size:.875rem;font-family:ui-monospace,SFMono-Regular,SF Mono,Consolas,Liberation Mono,Menlo,monospace}copilot-chat-assistant-message-renderer .hljs{background:transparent;color:#383a42}.dark copilot-chat-assistant-message-renderer .hljs{background:transparent;color:#abb2bf}copilot-chat-assistant-message-renderer .katex-display{overflow-x:auto;overflow-y:hidden;padding:1rem 0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
246
- }
247
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChatAssistantMessageRenderer, decorators: [{
248
- type: Component,
249
- args: [{ standalone: true, selector: "copilot-chat-assistant-message-renderer", imports: [CommonModule, LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: `
250
- <div
251
- #markdownContainer
252
- [class]="inputClass()"
253
- (click)="handleClick($event)"
254
- ></div>
255
- `, styles: ["copilot-chat-assistant-message-renderer{display:block;width:100%}copilot-chat-assistant-message-renderer code:not(pre code){padding:2.5px 4.8px;background-color:#ececec;border-radius:.25rem;font-size:.875rem;font-family:ui-monospace,SFMono-Regular,SF Mono,Consolas,Liberation Mono,Menlo,monospace;font-weight:500;color:#000}.dark copilot-chat-assistant-message-renderer code:not(pre code){background-color:#171717;color:#f8fafc}copilot-chat-assistant-message-renderer .code-block-container{position:relative;margin:.25rem 0;background-color:#f9f9f9;border-radius:1rem}.dark copilot-chat-assistant-message-renderer .code-block-container{background-color:#171717}copilot-chat-assistant-message-renderer .code-block-header{display:flex;align-items:center;justify-content:space-between;padding:.75rem 1rem;font-size:.75rem;background-color:transparent}copilot-chat-assistant-message-renderer .code-block-language{font-weight:400;color:#737373}.dark copilot-chat-assistant-message-renderer .code-block-language{color:#fff}copilot-chat-assistant-message-renderer .code-block-copy-button{display:flex;align-items:center;gap:.125rem;padding:0 .5rem;font-size:.75rem;color:#737373;cursor:pointer;background:none;border:none;transition:opacity .2s}.dark copilot-chat-assistant-message-renderer .code-block-copy-button{color:#fff}copilot-chat-assistant-message-renderer .code-block-copy-button:hover{opacity:.8}copilot-chat-assistant-message-renderer .code-block-copy-button svg{width:10px;height:10px}copilot-chat-assistant-message-renderer .code-block-copy-button span{font-size:11px}copilot-chat-assistant-message-renderer pre{margin:0;padding:0 1rem 1rem;overflow-x:auto;background-color:transparent;border-radius:1rem}.dark copilot-chat-assistant-message-renderer pre{background-color:transparent}copilot-chat-assistant-message-renderer pre code{background-color:transparent;padding:0;font-size:.875rem;font-family:ui-monospace,SFMono-Regular,SF Mono,Consolas,Liberation Mono,Menlo,monospace}copilot-chat-assistant-message-renderer .hljs{background:transparent;color:#383a42}.dark copilot-chat-assistant-message-renderer .hljs{background:transparent;color:#abb2bf}copilot-chat-assistant-message-renderer .katex-display{overflow-x:auto;overflow-y:hidden;padding:1rem 0}\n"] }]
256
- }], ctorParameters: () => [], propDecorators: { markdownContainer: [{
257
- type: ViewChild,
258
- args: ["markdownContainer", { static: false }]
259
- }] } });
260
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-assistant-message-renderer.js","sourceRoot":"","sources":["../../../../../src/lib/components/chat/copilot-chat-assistant-message-renderer.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  input,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  computed,\n  effect,\n  inject,\n  ElementRef,\n  AfterViewInit,\n  ViewChild,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { Marked } from \"marked\";\nimport hljs from \"highlight.js\";\nimport * as katex from \"katex\";\nimport { completePartialMarkdown } from \"@copilotkitnext/core\";\nimport { LucideAngularModule } from \"lucide-angular\";\nimport { injectChatLabels } from \"../../chat-config\";\n\n@Component({\n  standalone: true,\n  selector: \"copilot-chat-assistant-message-renderer\",\n  imports: [CommonModule, LucideAngularModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <div\n      #markdownContainer\n      [class]=\"inputClass()\"\n      (click)=\"handleClick($event)\"\n    ></div>\n  `,\n  styles: [\n    `\n      copilot-chat-assistant-message-renderer {\n        display: block;\n        width: 100%;\n      }\n\n      /* Inline code styling */\n      copilot-chat-assistant-message-renderer code:not(pre code) {\n        padding: 2.5px 4.8px;\n        background-color: rgb(236, 236, 236);\n        border-radius: 0.25rem;\n        font-size: 0.875rem;\n        font-family:\n          ui-monospace, SFMono-Regular, \"SF Mono\", Consolas, \"Liberation Mono\",\n          Menlo, monospace;\n        font-weight: 500;\n        color: #000000;\n      }\n\n      .dark copilot-chat-assistant-message-renderer code:not(pre code) {\n        background-color: #171717; /* same as code blocks */\n        color: rgb(248, 250, 252); /* text-foreground in dark mode */\n      }\n\n      /* Code block container */\n      copilot-chat-assistant-message-renderer .code-block-container {\n        position: relative;\n        margin: 0.25rem 0;\n        background-color: rgb(249, 249, 249);\n        border-radius: 1rem;\n      }\n\n      .dark copilot-chat-assistant-message-renderer .code-block-container {\n        background-color: #171717;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-header {\n        display: flex;\n        align-items: center;\n        justify-content: space-between;\n        padding: 0.75rem 1rem 0.75rem 1rem;\n        font-size: 0.75rem;\n        background-color: transparent;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-language {\n        font-weight: 400;\n        color: rgba(115, 115, 115, 1);\n      }\n\n      .dark copilot-chat-assistant-message-renderer .code-block-language {\n        color: white;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-copy-button {\n        display: flex;\n        align-items: center;\n        gap: 0.125rem;\n        padding: 0 0.5rem;\n        font-size: 0.75rem;\n        color: rgba(115, 115, 115, 1);\n        cursor: pointer;\n        background: none;\n        border: none;\n        transition: opacity 0.2s;\n      }\n\n      .dark copilot-chat-assistant-message-renderer .code-block-copy-button {\n        color: white;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-copy-button:hover {\n        opacity: 0.8;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-copy-button svg {\n        width: 10px;\n        height: 10px;\n      }\n\n      copilot-chat-assistant-message-renderer .code-block-copy-button span {\n        font-size: 11px;\n      }\n\n      copilot-chat-assistant-message-renderer pre {\n        margin: 0;\n        padding: 0 1rem 1rem 1rem;\n        overflow-x: auto;\n        background-color: transparent;\n        border-radius: 1rem;\n      }\n\n      .dark copilot-chat-assistant-message-renderer pre {\n        background-color: transparent;\n      }\n\n      copilot-chat-assistant-message-renderer pre code {\n        background-color: transparent;\n        padding: 0;\n        font-size: 0.875rem;\n        font-family:\n          ui-monospace, SFMono-Regular, \"SF Mono\", Consolas, \"Liberation Mono\",\n          Menlo, monospace;\n      }\n\n      /* Highlight.js theme adjustments */\n      copilot-chat-assistant-message-renderer .hljs {\n        background: transparent;\n        color: rgb(56, 58, 66);\n      }\n\n      .dark copilot-chat-assistant-message-renderer .hljs {\n        background: transparent;\n        color: #abb2bf;\n      }\n\n      /* Math equations */\n      copilot-chat-assistant-message-renderer .katex-display {\n        overflow-x: auto;\n        overflow-y: hidden;\n        padding: 1rem 0;\n      }\n    `,\n  ],\n})\nexport class CopilotChatAssistantMessageRenderer implements AfterViewInit {\n  readonly content = input<string>(\"\");\n  readonly inputClass = input<string | undefined>();\n  readonly labels = injectChatLabels();\n\n  @ViewChild(\"markdownContainer\", { static: false })\n  markdownContainer?: ElementRef<HTMLDivElement>;\n\n  private elementRef = inject(ElementRef);\n\n  // Track copy states for code blocks (DOM-updated; no signal needed)\n  private copyStates = new Map<string, boolean>();\n\n  readonly renderedHtml = computed(() => {\n    const currentContent = this.content();\n    const completedMarkdown = completePartialMarkdown(currentContent);\n    return this.renderMarkdown(completedMarkdown);\n  });\n\n  constructor() {\n    // React to content changes using signals\n    effect(() => {\n      // Read content to establish dependency\n      this.content();\n      // Reset copy states when content changes\n      this.copyStates.clear();\n      // If view is ready, update DOM\n      if (this.markdownContainer) {\n        this.updateContent();\n        this.renderMathEquations();\n      }\n    });\n  }\n\n  ngAfterViewInit(): void {\n    this.updateContent();\n    this.renderMathEquations();\n  }\n\n  private updateContent(): void {\n    if (!this.markdownContainer) return;\n    const container = this.markdownContainer.nativeElement;\n    const html = this.renderedHtml();\n    container.innerHTML = html;\n  }\n\n  private codeBlocksMap = new Map<string, string>();\n  private markedInstance: Marked | null = null;\n\n  private initializeMarked(): void {\n    if (this.markedInstance) return;\n\n    // Store highlighted code blocks temporarily\n    const highlightedBlocks = new Map<string, string>();\n\n    // Create a new Marked instance\n    this.markedInstance = new Marked();\n\n    // Configure marked options\n    this.markedInstance.setOptions({\n      gfm: true,\n      breaks: true,\n    });\n\n    // Add a walkTokens function to process code tokens before rendering\n    this.markedInstance.use({\n      walkTokens: (token: any) => {\n        if (token.type === \"code\") {\n          const rawCode = token.text;\n          const lang = token.lang || \"\";\n\n          const blockId = this.generateBlockId(rawCode);\n          // Store the raw code in our map for copying\n          this.codeBlocksMap.set(blockId, rawCode);\n\n          const copyLabel = this.labels.assistantMessageToolbarCopyCodeLabel;\n\n          // Manually highlight the code\n          const language = hljs.getLanguage(lang) ? lang : \"plaintext\";\n          const highlighted = hljs.highlight(rawCode, { language }).value;\n          const codeClass = lang ? `hljs language-${lang}` : \"hljs\";\n\n          // Create the full HTML with header and highlighted code\n          const fullHtml = `\n            <div class=\"code-block-container\">\n              <div class=\"code-block-header\">\n                ${lang ? `<span class=\"code-block-language\">${lang}</span>` : \"<span></span>\"}\n                <button \n                  class=\"code-block-copy-button\" \n                  data-code-block-id=\"${blockId}\"\n                  aria-label=\"${copyLabel} code\">\n                  <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><rect width=\"14\" height=\"14\" x=\"8\" y=\"8\" rx=\"2\" ry=\"2\"/><path d=\"M4 16c-1.11 0-2-.9-2-2V4c0-1.11.89-2 2-2h10c1.11 0 2 .89 2 2\"/></svg>\n                  <span>${copyLabel}</span>\n                </button>\n              </div>\n              <pre><code class=\"${codeClass}\">${highlighted}</code></pre>\n            </div>\n          `;\n\n          // Store the highlighted HTML\n          highlightedBlocks.set(blockId, fullHtml);\n\n          // Change the token to an html token to bypass marked's escaping\n          token.type = \"html\";\n          token.text = fullHtml;\n        }\n      },\n    });\n  }\n\n  private renderMarkdown(content: string): string {\n    // Initialize marked if not already done\n    this.initializeMarked();\n\n    // Clear the code blocks map for new render\n    this.codeBlocksMap.clear();\n\n    // Parse markdown\n    let html = this.markedInstance!.parse(content) as string;\n\n    // Process math equations\n    html = this.processMathEquations(html);\n\n    return html;\n  }\n\n  private processMathEquations(html: string): string {\n    // First, temporarily replace code blocks with placeholders to protect them from math processing\n    const codeBlocks: string[] = [];\n    const placeholder = \"___CODE_BLOCK_PLACEHOLDER_\";\n\n    // Store code blocks and replace with placeholders\n    html = html.replace(/<pre><code[\\s\\S]*?<\\/code><\\/pre>/g, (match) => {\n      const index = codeBlocks.length;\n      codeBlocks.push(match);\n      return `${placeholder}${index}___`;\n    });\n\n    // Also protect inline code\n    const inlineCode: string[] = [];\n    const inlinePlaceholder = \"___INLINE_CODE_PLACEHOLDER_\";\n    html = html.replace(/<code>[\\s\\S]*?<\\/code>/g, (match) => {\n      const index = inlineCode.length;\n      inlineCode.push(match);\n      return `${inlinePlaceholder}${index}___`;\n    });\n\n    // Process display math $$ ... $$\n    html = html.replace(/\\$\\$([\\s\\S]*?)\\$\\$/g, (match, equation) => {\n      try {\n        return katex.renderToString(equation, {\n          displayMode: true,\n          throwOnError: false,\n        });\n      } catch {\n        return match;\n      }\n    });\n\n    // Process inline math $ ... $\n    html = html.replace(/\\$([^$]+)\\$/g, (match, equation) => {\n      try {\n        return katex.renderToString(equation, {\n          displayMode: false,\n          throwOnError: false,\n        });\n      } catch {\n        return match;\n      }\n    });\n\n    // Restore code blocks\n    codeBlocks.forEach((block, index) => {\n      html = html.replace(`${placeholder}${index}___`, block);\n    });\n\n    // Restore inline code\n    inlineCode.forEach((code, index) => {\n      html = html.replace(`${inlinePlaceholder}${index}___`, code);\n    });\n\n    return html;\n  }\n\n  private renderMathEquations(): void {\n    if (!this.markdownContainer) return;\n\n    const container = this.markdownContainer.nativeElement;\n\n    // Find all math placeholders and render them\n    const mathElements = container.querySelectorAll(\".math-placeholder\");\n    mathElements.forEach((element) => {\n      const equation = element.getAttribute(\"data-equation\");\n      const displayMode = element.getAttribute(\"data-display\") === \"true\";\n\n      if (equation) {\n        try {\n          katex.render(equation, element as HTMLElement, {\n            displayMode,\n            throwOnError: false,\n          });\n        } catch (error) {\n          console.error(\"Failed to render math equation:\", error);\n        }\n      }\n    });\n  }\n\n  handleClick(event: MouseEvent): void {\n    const target = event.target as HTMLElement;\n\n    // Check if clicked on copy button or its children\n    const copyButton = target.closest(\n      \".code-block-copy-button\",\n    ) as HTMLButtonElement;\n    if (copyButton) {\n      event.preventDefault();\n      const blockId = copyButton.getAttribute(\"data-code-block-id\");\n\n      if (blockId) {\n        // Get the raw code from our map instead of from DOM\n        const code = this.codeBlocksMap.get(blockId);\n        if (code) {\n          this.copyCodeBlock(blockId, code);\n        }\n      }\n    }\n  }\n\n  private copyCodeBlock(blockId: string, code: string): void {\n    navigator.clipboard.writeText(code).then(\n      () => {\n        // Update the button in the DOM\n        const button = this.elementRef.nativeElement.querySelector(\n          `[data-code-block-id=\"${blockId}\"]`,\n        );\n        if (button) {\n          const originalHTML = button.innerHTML;\n          button.innerHTML = `\n            <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"10\" height=\"10\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M20 6 9 17l-5-5\"/></svg>\n            <span>${this.labels.assistantMessageToolbarCopyCodeCopiedLabel}</span>\n          `;\n          button.setAttribute(\n            \"aria-label\",\n            `${this.labels.assistantMessageToolbarCopyCodeCopiedLabel} code`,\n          );\n\n          // Reset after 2 seconds\n          setTimeout(() => {\n            button.innerHTML = originalHTML;\n            button.setAttribute(\n              \"aria-label\",\n              `${this.labels.assistantMessageToolbarCopyCodeLabel} code`,\n            );\n          }, 2000);\n        }\n      },\n      (err) => {\n        console.error(\"Failed to copy code:\", err);\n      },\n    );\n  }\n\n  private generateBlockId(code: string): string {\n    // Simple hash function for generating unique IDs\n    let hash = 0;\n    for (let i = 0; i < code.length; i++) {\n      const char = code.charCodeAt(i);\n      hash = (hash << 5) - hash + char;\n      hash = hash & hash; // Convert to 32-bit integer\n    }\n    return `code-block-${hash}`;\n  }\n\n  private escapeHtml(text: string): string {\n    const div = document.createElement(\"div\");\n    div.textContent = text;\n    return div.innerHTML;\n  }\n}\n"]}
@@ -1,22 +0,0 @@
1
- import { Directive, input, computed } from "@angular/core";
2
- import { cn } from "../../utils";
3
- import * as i0 from "@angular/core";
4
- export class CopilotChatAssistantMessageToolbar {
5
- inputClass = input();
6
- computedClass = computed(() => {
7
- return cn("w-full bg-transparent flex items-center -ml-[5px] -mt-[0px]", this.inputClass());
8
- });
9
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChatAssistantMessageToolbar, deps: [], target: i0.ɵɵFactoryTarget.Directive });
10
- static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.14", type: CopilotChatAssistantMessageToolbar, isStandalone: true, selector: "[copilotChatAssistantMessageToolbar]", inputs: { inputClass: { classPropertyName: "inputClass", publicName: "inputClass", isSignal: true, isRequired: false, transformFunction: null } }, host: { properties: { "class": "computedClass()" } }, ngImport: i0 });
11
- }
12
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChatAssistantMessageToolbar, decorators: [{
13
- type: Directive,
14
- args: [{
15
- standalone: true,
16
- selector: "[copilotChatAssistantMessageToolbar]",
17
- host: {
18
- "[class]": "computedClass()",
19
- },
20
- }]
21
- }] });
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LWFzc2lzdGFudC1tZXNzYWdlLXRvb2xiYXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtYXNzaXN0YW50LW1lc3NhZ2UtdG9vbGJhci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0QsT0FBTyxFQUFFLEVBQUUsRUFBRSxNQUFNLGFBQWEsQ0FBQzs7QUFTakMsTUFBTSxPQUFPLGtDQUFrQztJQUNwQyxVQUFVLEdBQUcsS0FBSyxFQUFzQixDQUFDO0lBRXpDLGFBQWEsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQ3JDLE9BQU8sRUFBRSxDQUNQLDZEQUE2RCxFQUM3RCxJQUFJLENBQUMsVUFBVSxFQUFFLENBQ2xCLENBQUM7SUFDSixDQUFDLENBQUMsQ0FBQzt3R0FSUSxrQ0FBa0M7NEZBQWxDLGtDQUFrQzs7NEZBQWxDLGtDQUFrQztrQkFQOUMsU0FBUzttQkFBQztvQkFDVCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLHNDQUFzQztvQkFDaEQsSUFBSSxFQUFFO3dCQUNKLFNBQVMsRUFBRSxpQkFBaUI7cUJBQzdCO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBpbnB1dCwgY29tcHV0ZWQgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuaW1wb3J0IHsgY24gfSBmcm9tIFwiLi4vLi4vdXRpbHNcIjtcblxuQERpcmVjdGl2ZSh7XG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIHNlbGVjdG9yOiBcIltjb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VUb29sYmFyXVwiLFxuICBob3N0OiB7XG4gICAgXCJbY2xhc3NdXCI6IFwiY29tcHV0ZWRDbGFzcygpXCIsXG4gIH0sXG59KVxuZXhwb3J0IGNsYXNzIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXIge1xuICByZWFkb25seSBpbnB1dENsYXNzID0gaW5wdXQ8c3RyaW5nIHwgdW5kZWZpbmVkPigpO1xuXG4gIHJlYWRvbmx5IGNvbXB1dGVkQ2xhc3MgPSBjb21wdXRlZCgoKSA9PiB7XG4gICAgcmV0dXJuIGNuKFxuICAgICAgXCJ3LWZ1bGwgYmctdHJhbnNwYXJlbnQgZmxleCBpdGVtcy1jZW50ZXIgLW1sLVs1cHhdIC1tdC1bMHB4XVwiLFxuICAgICAgdGhpcy5pbnB1dENsYXNzKCksXG4gICAgKTtcbiAgfSk7XG59XG4iXX0=