@copilotkitnext/angular 0.0.2 → 0.0.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -3
- package/dist/README.md +3 -3
- package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +10 -10
- package/dist/components/chat/copilot-chat-message-view.component.d.ts +42 -42
- package/dist/components/chat/copilot-chat-view.component.d.ts +14 -14
- package/dist/core/copilotkit.providers.d.ts +1 -1
- package/dist/core/copilotkit.service.d.ts +5 -5
- package/dist/core/copilotkit.types.d.ts +8 -10
- package/dist/directives/copilotkit-frontend-tool.directive.d.ts +1 -1
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
- package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
- package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
- package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
- package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
- package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
- package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
- package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
- package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
- package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
- package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
- package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
- package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
- package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
- package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
- package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
- package/dist/esm2022/core/copilotkit.service.mjs +426 -0
- package/dist/esm2022/core/copilotkit.types.mjs +13 -0
- package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
- package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
- package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
- package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
- package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +128 -0
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +265 -0
- package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
- package/dist/esm2022/index.mjs +70 -0
- package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
- package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
- package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
- package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
- package/dist/esm2022/lib/utils.mjs +10 -0
- package/dist/esm2022/services/resize-observer.service.mjs +152 -0
- package/dist/esm2022/services/scroll-position.service.mjs +124 -0
- package/dist/esm2022/types/frontend-tool.mjs +2 -0
- package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
- package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
- package/dist/esm2022/utils/agent.utils.mjs +204 -0
- package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
- package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
- package/dist/esm2022/utils/frontend-tool.utils.mjs +224 -0
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +293 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +174 -187
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/dist/utils/frontend-tool.utils.d.ts +1 -1
- package/package.json +23 -20
- package/vitest.config.mts +32 -21
- package/.turbo/turbo-build.log +0 -38
- package/.turbo/turbo-check-types.log +0 -0
- package/.turbo/turbo-test.log +0 -71
- package/ng-package.json +0 -19
- package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
- package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
- package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
- package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
- package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
- package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
- package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
- package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
- package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
- package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
- package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
- package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
- package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
- package/src/components/chat/copilot-chat-input.component.ts +0 -512
- package/src/components/chat/copilot-chat-input.types.ts +0 -148
- package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
- package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
- package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
- package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
- package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
- package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
- package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
- package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
- package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
- package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
- package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
- package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
- package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
- package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
- package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
- package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
- package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
- package/src/components/chat/copilot-chat-view.component.ts +0 -420
- package/src/components/chat/copilot-chat-view.types.ts +0 -52
- package/src/components/chat/copilot-chat.component.ts +0 -232
- package/src/components/copilotkit-tool-render.component.ts +0 -169
- package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
- package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
- package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
- package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
- package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
- package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
- package/src/core/copilotkit.providers.ts +0 -59
- package/src/core/copilotkit.service.ts +0 -542
- package/src/core/copilotkit.types.ts +0 -132
- package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
- package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
- package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
- package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
- package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
- package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
- package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
- package/src/directives/copilotkit-agent-context.directive.ts +0 -138
- package/src/directives/copilotkit-agent.directive.ts +0 -225
- package/src/directives/copilotkit-chat-config.directive.ts +0 -241
- package/src/directives/copilotkit-config.directive.ts +0 -81
- package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
- package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
- package/src/directives/stick-to-bottom.directive.ts +0 -204
- package/src/index.ts +0 -105
- package/src/lib/directives/tooltip.directive.ts +0 -292
- package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
- package/src/lib/slots/copilot-slot.component.ts +0 -135
- package/src/lib/slots/index.ts +0 -3
- package/src/lib/slots/slot.types.ts +0 -64
- package/src/lib/slots/slot.utils.ts +0 -289
- package/src/lib/utils.ts +0 -10
- package/src/public-api.ts +0 -1
- package/src/services/resize-observer.service.ts +0 -181
- package/src/services/scroll-position.service.ts +0 -169
- package/src/styles/globals.css +0 -266
- package/src/styles/index.css +0 -3
- package/src/test-setup.ts +0 -15
- package/src/testing/index.ts +0 -3
- package/src/testing/testing.utils.ts +0 -248
- package/src/types/frontend-tool.ts +0 -44
- package/src/types/human-in-the-loop.ts +0 -52
- package/src/utils/__tests__/agent.utils.spec.ts +0 -234
- package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
- package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
- package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
- package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
- package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
- package/src/utils/agent-context.utils.ts +0 -133
- package/src/utils/agent.utils.ts +0 -239
- package/src/utils/chat-config.utils.ts +0 -221
- package/src/utils/copilotkit.utils.ts +0 -20
- package/src/utils/frontend-tool.utils.ts +0 -266
- package/src/utils/human-in-the-loop.utils.ts +0 -359
- package/tsconfig.spec.json +0 -12
|
@@ -0,0 +1,286 @@
|
|
|
1
|
+
import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, signal, computed, 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 { CopilotChatConfigurationService } from "../../core/chat-configuration/chat-configuration.service";
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
export class CopilotChatAssistantMessageRendererComponent {
|
|
11
|
+
_content = "";
|
|
12
|
+
set content(value) {
|
|
13
|
+
this._content = value;
|
|
14
|
+
this.contentSignal.set(value);
|
|
15
|
+
}
|
|
16
|
+
get content() {
|
|
17
|
+
return this._content;
|
|
18
|
+
}
|
|
19
|
+
inputClass;
|
|
20
|
+
contentSignal = signal("");
|
|
21
|
+
markdownContainer;
|
|
22
|
+
chatConfig = inject(CopilotChatConfigurationService);
|
|
23
|
+
elementRef = inject(ElementRef);
|
|
24
|
+
// Track copy states for code blocks
|
|
25
|
+
copyStates = new Map();
|
|
26
|
+
copyStateSignal = signal(new Map());
|
|
27
|
+
renderedHtml = computed(() => {
|
|
28
|
+
const currentContent = this.contentSignal();
|
|
29
|
+
const completedMarkdown = completePartialMarkdown(currentContent);
|
|
30
|
+
return this.renderMarkdown(completedMarkdown);
|
|
31
|
+
});
|
|
32
|
+
get labels() {
|
|
33
|
+
return this.chatConfig.labels();
|
|
34
|
+
}
|
|
35
|
+
ngOnChanges(changes) {
|
|
36
|
+
if (changes["content"]) {
|
|
37
|
+
// Reset copy states when content changes
|
|
38
|
+
this.copyStates.clear();
|
|
39
|
+
this.copyStateSignal.set(new Map());
|
|
40
|
+
// Update content if container exists
|
|
41
|
+
if (this.markdownContainer) {
|
|
42
|
+
this.updateContent();
|
|
43
|
+
this.renderMathEquations();
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
ngAfterViewInit() {
|
|
48
|
+
this.updateContent();
|
|
49
|
+
this.renderMathEquations();
|
|
50
|
+
}
|
|
51
|
+
updateContent() {
|
|
52
|
+
if (!this.markdownContainer)
|
|
53
|
+
return;
|
|
54
|
+
const container = this.markdownContainer.nativeElement;
|
|
55
|
+
const html = this.renderedHtml();
|
|
56
|
+
container.innerHTML = html;
|
|
57
|
+
}
|
|
58
|
+
codeBlocksMap = new Map();
|
|
59
|
+
markedInstance = null;
|
|
60
|
+
initializeMarked() {
|
|
61
|
+
if (this.markedInstance)
|
|
62
|
+
return;
|
|
63
|
+
// Store highlighted code blocks temporarily
|
|
64
|
+
const highlightedBlocks = new Map();
|
|
65
|
+
// Create a new Marked instance
|
|
66
|
+
this.markedInstance = new Marked();
|
|
67
|
+
// Configure marked options
|
|
68
|
+
this.markedInstance.setOptions({
|
|
69
|
+
gfm: true,
|
|
70
|
+
breaks: true,
|
|
71
|
+
});
|
|
72
|
+
// Add a walkTokens function to process code tokens before rendering
|
|
73
|
+
this.markedInstance.use({
|
|
74
|
+
walkTokens: (token) => {
|
|
75
|
+
if (token.type === "code") {
|
|
76
|
+
const rawCode = token.text;
|
|
77
|
+
const lang = token.lang || "";
|
|
78
|
+
const blockId = this.generateBlockId(rawCode);
|
|
79
|
+
// Store the raw code in our map for copying
|
|
80
|
+
this.codeBlocksMap.set(blockId, rawCode);
|
|
81
|
+
const copied = this.copyStateSignal().get(blockId) || false;
|
|
82
|
+
const copyLabel = copied
|
|
83
|
+
? this.labels.assistantMessageToolbarCopyCodeCopiedLabel
|
|
84
|
+
: this.labels.assistantMessageToolbarCopyCodeLabel;
|
|
85
|
+
// Manually highlight the code
|
|
86
|
+
const language = hljs.getLanguage(lang) ? lang : "plaintext";
|
|
87
|
+
const highlighted = hljs.highlight(rawCode, { language }).value;
|
|
88
|
+
const codeClass = lang ? `hljs language-${lang}` : "hljs";
|
|
89
|
+
// Create the full HTML with header and highlighted code
|
|
90
|
+
const fullHtml = `
|
|
91
|
+
<div class="code-block-container">
|
|
92
|
+
<div class="code-block-header">
|
|
93
|
+
${lang ? `<span class="code-block-language">${lang}</span>` : "<span></span>"}
|
|
94
|
+
<button
|
|
95
|
+
class="code-block-copy-button"
|
|
96
|
+
data-code-block-id="${blockId}"
|
|
97
|
+
aria-label="${copyLabel} code">
|
|
98
|
+
${copied
|
|
99
|
+
? '<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>'
|
|
100
|
+
: '<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>'}
|
|
101
|
+
<span>${copyLabel}</span>
|
|
102
|
+
</button>
|
|
103
|
+
</div>
|
|
104
|
+
<pre><code class="${codeClass}">${highlighted}</code></pre>
|
|
105
|
+
</div>
|
|
106
|
+
`;
|
|
107
|
+
// Store the highlighted HTML
|
|
108
|
+
highlightedBlocks.set(blockId, fullHtml);
|
|
109
|
+
// Change the token to an html token to bypass marked's escaping
|
|
110
|
+
token.type = "html";
|
|
111
|
+
token.text = fullHtml;
|
|
112
|
+
}
|
|
113
|
+
},
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
renderMarkdown(content) {
|
|
117
|
+
// Initialize marked if not already done
|
|
118
|
+
this.initializeMarked();
|
|
119
|
+
// Clear the code blocks map for new render
|
|
120
|
+
this.codeBlocksMap.clear();
|
|
121
|
+
// Parse markdown
|
|
122
|
+
let html = this.markedInstance.parse(content);
|
|
123
|
+
// Process math equations
|
|
124
|
+
html = this.processMathEquations(html);
|
|
125
|
+
return html;
|
|
126
|
+
}
|
|
127
|
+
processMathEquations(html) {
|
|
128
|
+
// First, temporarily replace code blocks with placeholders to protect them from math processing
|
|
129
|
+
const codeBlocks = [];
|
|
130
|
+
const placeholder = "___CODE_BLOCK_PLACEHOLDER_";
|
|
131
|
+
// Store code blocks and replace with placeholders
|
|
132
|
+
html = html.replace(/<pre><code[\s\S]*?<\/code><\/pre>/g, (match) => {
|
|
133
|
+
const index = codeBlocks.length;
|
|
134
|
+
codeBlocks.push(match);
|
|
135
|
+
return `${placeholder}${index}___`;
|
|
136
|
+
});
|
|
137
|
+
// Also protect inline code
|
|
138
|
+
const inlineCode = [];
|
|
139
|
+
const inlinePlaceholder = "___INLINE_CODE_PLACEHOLDER_";
|
|
140
|
+
html = html.replace(/<code>[\s\S]*?<\/code>/g, (match) => {
|
|
141
|
+
const index = inlineCode.length;
|
|
142
|
+
inlineCode.push(match);
|
|
143
|
+
return `${inlinePlaceholder}${index}___`;
|
|
144
|
+
});
|
|
145
|
+
// Process display math $$ ... $$
|
|
146
|
+
html = html.replace(/\$\$([\s\S]*?)\$\$/g, (match, equation) => {
|
|
147
|
+
try {
|
|
148
|
+
return katex.renderToString(equation, {
|
|
149
|
+
displayMode: true,
|
|
150
|
+
throwOnError: false,
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
catch {
|
|
154
|
+
return match;
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
// Process inline math $ ... $
|
|
158
|
+
html = html.replace(/\$([^\$]+)\$/g, (match, equation) => {
|
|
159
|
+
try {
|
|
160
|
+
return katex.renderToString(equation, {
|
|
161
|
+
displayMode: false,
|
|
162
|
+
throwOnError: false,
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
catch {
|
|
166
|
+
return match;
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
// Restore code blocks
|
|
170
|
+
codeBlocks.forEach((block, index) => {
|
|
171
|
+
html = html.replace(`${placeholder}${index}___`, block);
|
|
172
|
+
});
|
|
173
|
+
// Restore inline code
|
|
174
|
+
inlineCode.forEach((code, index) => {
|
|
175
|
+
html = html.replace(`${inlinePlaceholder}${index}___`, code);
|
|
176
|
+
});
|
|
177
|
+
return html;
|
|
178
|
+
}
|
|
179
|
+
renderMathEquations() {
|
|
180
|
+
if (!this.markdownContainer)
|
|
181
|
+
return;
|
|
182
|
+
const container = this.markdownContainer.nativeElement;
|
|
183
|
+
// Find all math placeholders and render them
|
|
184
|
+
const mathElements = container.querySelectorAll(".math-placeholder");
|
|
185
|
+
mathElements.forEach((element) => {
|
|
186
|
+
const equation = element.getAttribute("data-equation");
|
|
187
|
+
const displayMode = element.getAttribute("data-display") === "true";
|
|
188
|
+
if (equation) {
|
|
189
|
+
try {
|
|
190
|
+
katex.render(equation, element, {
|
|
191
|
+
displayMode,
|
|
192
|
+
throwOnError: false,
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
catch (error) {
|
|
196
|
+
console.error("Failed to render math equation:", error);
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
});
|
|
200
|
+
}
|
|
201
|
+
handleClick(event) {
|
|
202
|
+
const target = event.target;
|
|
203
|
+
// Check if clicked on copy button or its children
|
|
204
|
+
const copyButton = target.closest(".code-block-copy-button");
|
|
205
|
+
if (copyButton) {
|
|
206
|
+
event.preventDefault();
|
|
207
|
+
const blockId = copyButton.getAttribute("data-code-block-id");
|
|
208
|
+
if (blockId) {
|
|
209
|
+
// Get the raw code from our map instead of from DOM
|
|
210
|
+
const code = this.codeBlocksMap.get(blockId);
|
|
211
|
+
if (code) {
|
|
212
|
+
this.copyCodeBlock(blockId, code);
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
copyCodeBlock(blockId, code) {
|
|
218
|
+
navigator.clipboard.writeText(code).then(() => {
|
|
219
|
+
// Update copy state
|
|
220
|
+
const newStates = new Map(this.copyStateSignal());
|
|
221
|
+
newStates.set(blockId, true);
|
|
222
|
+
this.copyStateSignal.set(newStates);
|
|
223
|
+
// Update the button in the DOM
|
|
224
|
+
const button = this.elementRef.nativeElement.querySelector(`[data-code-block-id="${blockId}"]`);
|
|
225
|
+
if (button) {
|
|
226
|
+
const originalHTML = button.innerHTML;
|
|
227
|
+
button.innerHTML = `
|
|
228
|
+
<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>
|
|
229
|
+
<span>${this.labels.assistantMessageToolbarCopyCodeCopiedLabel}</span>
|
|
230
|
+
`;
|
|
231
|
+
button.setAttribute("aria-label", `${this.labels.assistantMessageToolbarCopyCodeCopiedLabel} code`);
|
|
232
|
+
// Reset after 2 seconds
|
|
233
|
+
setTimeout(() => {
|
|
234
|
+
const states = new Map(this.copyStateSignal());
|
|
235
|
+
states.set(blockId, false);
|
|
236
|
+
this.copyStateSignal.set(states);
|
|
237
|
+
button.innerHTML = originalHTML;
|
|
238
|
+
button.setAttribute("aria-label", `${this.labels.assistantMessageToolbarCopyCodeLabel} code`);
|
|
239
|
+
}, 2000);
|
|
240
|
+
}
|
|
241
|
+
}, (err) => {
|
|
242
|
+
console.error("Failed to copy code:", err);
|
|
243
|
+
});
|
|
244
|
+
}
|
|
245
|
+
generateBlockId(code) {
|
|
246
|
+
// Simple hash function for generating unique IDs
|
|
247
|
+
let hash = 0;
|
|
248
|
+
for (let i = 0; i < code.length; i++) {
|
|
249
|
+
const char = code.charCodeAt(i);
|
|
250
|
+
hash = (hash << 5) - hash + char;
|
|
251
|
+
hash = hash & hash; // Convert to 32-bit integer
|
|
252
|
+
}
|
|
253
|
+
return `code-block-${hash}`;
|
|
254
|
+
}
|
|
255
|
+
escapeHtml(text) {
|
|
256
|
+
const div = document.createElement("div");
|
|
257
|
+
div.textContent = text;
|
|
258
|
+
return div.innerHTML;
|
|
259
|
+
}
|
|
260
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageRendererComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
261
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageRendererComponent, isStandalone: true, selector: "copilot-chat-assistant-message-renderer", inputs: { content: "content", inputClass: "inputClass" }, viewQueries: [{ propertyName: "markdownContainer", first: true, predicate: ["markdownContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
|
|
262
|
+
<div
|
|
263
|
+
#markdownContainer
|
|
264
|
+
[class]="inputClass"
|
|
265
|
+
(click)="handleClick($event)"
|
|
266
|
+
></div>
|
|
267
|
+
`, 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 });
|
|
268
|
+
}
|
|
269
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageRendererComponent, decorators: [{
|
|
270
|
+
type: Component,
|
|
271
|
+
args: [{ selector: "copilot-chat-assistant-message-renderer", standalone: true, imports: [CommonModule, LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: `
|
|
272
|
+
<div
|
|
273
|
+
#markdownContainer
|
|
274
|
+
[class]="inputClass"
|
|
275
|
+
(click)="handleClick($event)"
|
|
276
|
+
></div>
|
|
277
|
+
`, 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"] }]
|
|
278
|
+
}], propDecorators: { content: [{
|
|
279
|
+
type: Input
|
|
280
|
+
}], inputClass: [{
|
|
281
|
+
type: Input
|
|
282
|
+
}], markdownContainer: [{
|
|
283
|
+
type: ViewChild,
|
|
284
|
+
args: ["markdownContainer", { static: false }]
|
|
285
|
+
}] } });
|
|
286
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-assistant-message-renderer.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-assistant-message-renderer.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  OnChanges,\n  SimpleChanges,\n  signal,\n  computed,\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, Copy, Check } from \"lucide-angular\";\nimport { CopilotChatConfigurationService } from \"../../core/chat-configuration/chat-configuration.service\";\n\n@Component({\n  selector: \"copilot-chat-assistant-message-renderer\",\n  standalone: true,\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 CopilotChatAssistantMessageRendererComponent\n  implements OnChanges, AfterViewInit\n{\n  private _content = \"\";\n  @Input()\n  set content(value: string) {\n    this._content = value;\n    this.contentSignal.set(value);\n  }\n  get content(): string {\n    return this._content;\n  }\n\n  @Input() inputClass?: string;\n\n  private contentSignal = signal<string>(\"\");\n\n  @ViewChild(\"markdownContainer\", { static: false })\n  markdownContainer?: ElementRef<HTMLDivElement>;\n\n  private chatConfig = inject(CopilotChatConfigurationService);\n  private elementRef = inject(ElementRef);\n\n  // Track copy states for code blocks\n  private copyStates = new Map<string, boolean>();\n  private copyStateSignal = signal(new Map<string, boolean>());\n\n  renderedHtml = computed(() => {\n    const currentContent = this.contentSignal();\n    const completedMarkdown = completePartialMarkdown(currentContent);\n    return this.renderMarkdown(completedMarkdown);\n  });\n\n  get labels() {\n    return this.chatConfig.labels();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[\"content\"]) {\n      // Reset copy states when content changes\n      this.copyStates.clear();\n      this.copyStateSignal.set(new Map());\n      // Update content if container exists\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 copied = this.copyStateSignal().get(blockId) || false;\n          const copyLabel = copied\n            ? this.labels.assistantMessageToolbarCopyCodeCopiedLabel\n            : 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                  ${\n                    copied\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                      : '<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                  }\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 copy state\n        const newStates = new Map(this.copyStateSignal());\n        newStates.set(blockId, true);\n        this.copyStateSignal.set(newStates);\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            const states = new Map(this.copyStateSignal());\n            states.set(blockId, false);\n            this.copyStateSignal.set(states);\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"]}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { Directive, Input, signal, computed } from '@angular/core';
|
|
2
|
+
import { cn } from '../../lib/utils';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class CopilotChatAssistantMessageToolbarComponent {
|
|
5
|
+
set inputClass(value) {
|
|
6
|
+
this.customClass.set(value);
|
|
7
|
+
}
|
|
8
|
+
customClass = signal(undefined);
|
|
9
|
+
computedClass = computed(() => {
|
|
10
|
+
return cn('w-full bg-transparent flex items-center -ml-[5px] -mt-[0px]', this.customClass());
|
|
11
|
+
});
|
|
12
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
13
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageToolbarComponent, isStandalone: true, selector: "[copilotChatAssistantMessageToolbar]", inputs: { inputClass: "inputClass" }, host: { properties: { "class": "computedClass()" } }, ngImport: i0 });
|
|
14
|
+
}
|
|
15
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageToolbarComponent, decorators: [{
|
|
16
|
+
type: Directive,
|
|
17
|
+
args: [{
|
|
18
|
+
selector: '[copilotChatAssistantMessageToolbar]',
|
|
19
|
+
standalone: true,
|
|
20
|
+
host: {
|
|
21
|
+
'[class]': 'computedClass()'
|
|
22
|
+
}
|
|
23
|
+
}]
|
|
24
|
+
}], propDecorators: { inputClass: [{
|
|
25
|
+
type: Input
|
|
26
|
+
}] } });
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LWFzc2lzdGFudC1tZXNzYWdlLXRvb2xiYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtYXNzaXN0YW50LW1lc3NhZ2UtdG9vbGJhci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsTUFBTSxFQUNOLFFBQVEsRUFDVCxNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsRUFBRSxFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBU3JDLE1BQU0sT0FBTywyQ0FBMkM7SUFDdEQsSUFBYSxVQUFVLENBQUMsS0FBeUI7UUFDL0MsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDOUIsQ0FBQztJQUVPLFdBQVcsR0FBRyxNQUFNLENBQXFCLFNBQVMsQ0FBQyxDQUFDO0lBRTVELGFBQWEsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQzVCLE9BQU8sRUFBRSxDQUNQLDZEQUE2RCxFQUM3RCxJQUFJLENBQUMsV0FBVyxFQUFFLENBQ25CLENBQUM7SUFDSixDQUFDLENBQUMsQ0FBQzt3R0FaUSwyQ0FBMkM7NEZBQTNDLDJDQUEyQzs7NEZBQTNDLDJDQUEyQztrQkFQdkQsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsc0NBQXNDO29CQUNoRCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsSUFBSSxFQUFFO3dCQUNKLFNBQVMsRUFBRSxpQkFBaUI7cUJBQzdCO2lCQUNGOzhCQUVjLFVBQVU7c0JBQXRCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBEaXJlY3RpdmUsXG4gIElucHV0LFxuICBzaWduYWwsXG4gIGNvbXB1dGVkXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgY24gfSBmcm9tICcuLi8uLi9saWIvdXRpbHMnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbY29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhcl0nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBob3N0OiB7XG4gICAgJ1tjbGFzc10nOiAnY29tcHV0ZWRDbGFzcygpJ1xuICB9XG59KVxuZXhwb3J0IGNsYXNzIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXJDb21wb25lbnQge1xuICBASW5wdXQoKSBzZXQgaW5wdXRDbGFzcyh2YWx1ZTogc3RyaW5nIHwgdW5kZWZpbmVkKSB7XG4gICAgdGhpcy5jdXN0b21DbGFzcy5zZXQodmFsdWUpO1xuICB9XG4gIFxuICBwcml2YXRlIGN1c3RvbUNsYXNzID0gc2lnbmFsPHN0cmluZyB8IHVuZGVmaW5lZD4odW5kZWZpbmVkKTtcbiAgXG4gIGNvbXB1dGVkQ2xhc3MgPSBjb21wdXRlZCgoKSA9PiB7XG4gICAgcmV0dXJuIGNuKFxuICAgICAgJ3ctZnVsbCBiZy10cmFuc3BhcmVudCBmbGV4IGl0ZW1zLWNlbnRlciAtbWwtWzVweF0gLW10LVswcHhdJyxcbiAgICAgIHRoaXMuY3VzdG9tQ2xhc3MoKVxuICAgICk7XG4gIH0pO1xufSJdfQ==
|