@copilotkitnext/angular 0.0.1 → 0.0.4
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 +248 -0
- package/dist/README.md +248 -0
- 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/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 +430 -0
- package/dist/esm2022/core/copilotkit.types.mjs +12 -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 +130 -0
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +266 -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 +228 -0
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +296 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +163 -164
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/dist/styles.css +0 -27
- package/package.json +23 -20
- package/vitest.config.mts +32 -21
- package/.turbo/turbo-build.log +0 -39
- package/.turbo/turbo-check-types.log +0 -0
- package/.turbo/turbo-test.log +0 -71
- package/README-agent-context.md +0 -310
- package/ng-package.json +0 -19
- package/slots.md +0 -331
- 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,216 @@
|
|
|
1
|
+
import { Component, Input, ViewContainerRef, inject, ViewChild, TemplateRef, ChangeDetectionStrategy, signal, } from "@angular/core";
|
|
2
|
+
import { CommonModule } from "@angular/common";
|
|
3
|
+
import { ToolCallStatus } from "@copilotkitnext/core";
|
|
4
|
+
import { CopilotKitService } from "../../core/copilotkit.service";
|
|
5
|
+
import { partialJSONParse } from "@copilotkitnext/shared";
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
/**
|
|
9
|
+
* Component for rendering all tool calls for an assistant message.
|
|
10
|
+
* This component iterates through the message's tool calls and renders each one
|
|
11
|
+
* using the registered render functions in CopilotKitService.
|
|
12
|
+
*/
|
|
13
|
+
export class CopilotChatToolCallsViewComponent {
|
|
14
|
+
message;
|
|
15
|
+
messages = [];
|
|
16
|
+
isLoading = false;
|
|
17
|
+
container;
|
|
18
|
+
copilotkit = inject(CopilotKitService, {
|
|
19
|
+
optional: true,
|
|
20
|
+
});
|
|
21
|
+
componentRefs = new Map();
|
|
22
|
+
templateCache = new Map();
|
|
23
|
+
// Signals for reactive state
|
|
24
|
+
messageSignal = signal(null);
|
|
25
|
+
messagesSignal = signal([]);
|
|
26
|
+
isLoadingSignal = signal(false);
|
|
27
|
+
ngAfterViewInit() {
|
|
28
|
+
this.renderAllToolCalls();
|
|
29
|
+
}
|
|
30
|
+
ngOnChanges(changes) {
|
|
31
|
+
if (changes["message"]) {
|
|
32
|
+
this.messageSignal.set(this.message);
|
|
33
|
+
}
|
|
34
|
+
if (changes["messages"]) {
|
|
35
|
+
this.messagesSignal.set(this.messages);
|
|
36
|
+
}
|
|
37
|
+
if (changes["isLoading"]) {
|
|
38
|
+
this.isLoadingSignal.set(this.isLoading);
|
|
39
|
+
}
|
|
40
|
+
this.renderAllToolCalls();
|
|
41
|
+
}
|
|
42
|
+
ngOnDestroy() {
|
|
43
|
+
// Clean up all component refs
|
|
44
|
+
this.componentRefs.forEach((ref) => ref.destroy());
|
|
45
|
+
this.componentRefs.clear();
|
|
46
|
+
this.templateCache.clear();
|
|
47
|
+
}
|
|
48
|
+
getTemplateForToolCall(toolCall) {
|
|
49
|
+
return this.templateCache.get(toolCall.id) || null;
|
|
50
|
+
}
|
|
51
|
+
renderAllToolCalls() {
|
|
52
|
+
const message = this.messageSignal();
|
|
53
|
+
if (!message || !message.toolCalls || message.toolCalls.length === 0) {
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
// Clear existing renders
|
|
57
|
+
this.componentRefs.forEach((ref) => ref.destroy());
|
|
58
|
+
this.componentRefs.clear();
|
|
59
|
+
this.templateCache.clear();
|
|
60
|
+
if (!this.copilotkit || !this.container) {
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
const messages = this.messagesSignal();
|
|
64
|
+
const isLoading = this.isLoadingSignal();
|
|
65
|
+
// Render each tool call
|
|
66
|
+
message.toolCalls.forEach((toolCall) => {
|
|
67
|
+
const toolMessage = messages.find((m) => m.role === "tool" && m.toolCallId === toolCall.id);
|
|
68
|
+
this.renderSingleToolCall(toolCall, toolMessage, isLoading);
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
renderSingleToolCall(toolCall, toolMessage, isLoading) {
|
|
72
|
+
if (!this.copilotkit) {
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
// Get current render tool calls
|
|
76
|
+
const currentRenderToolCalls = this.copilotkit.currentRenderToolCalls();
|
|
77
|
+
// Find the render config for this tool call by name
|
|
78
|
+
// Also check for wildcard (*) renders if no exact match
|
|
79
|
+
const renderConfig = currentRenderToolCalls.find((rc) => rc.name === toolCall.function.name) || currentRenderToolCalls.find((rc) => rc.name === "*");
|
|
80
|
+
if (!renderConfig) {
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
// Parse the arguments if they're a string
|
|
84
|
+
const args = partialJSONParse(toolCall.function.arguments);
|
|
85
|
+
// Determine status
|
|
86
|
+
let status;
|
|
87
|
+
if (toolMessage) {
|
|
88
|
+
status = ToolCallStatus.Complete;
|
|
89
|
+
}
|
|
90
|
+
else if (isLoading) {
|
|
91
|
+
status = ToolCallStatus.InProgress;
|
|
92
|
+
}
|
|
93
|
+
else {
|
|
94
|
+
status = ToolCallStatus.Complete;
|
|
95
|
+
}
|
|
96
|
+
// Create props based on status - use discriminated union properly
|
|
97
|
+
let props;
|
|
98
|
+
if (status === ToolCallStatus.InProgress) {
|
|
99
|
+
props = {
|
|
100
|
+
name: toolCall.function.name,
|
|
101
|
+
description: "",
|
|
102
|
+
args: args, // Partial args for InProgress
|
|
103
|
+
status: ToolCallStatus.InProgress,
|
|
104
|
+
result: undefined,
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
else {
|
|
108
|
+
// Complete status
|
|
109
|
+
props = {
|
|
110
|
+
name: toolCall.function.name,
|
|
111
|
+
description: "",
|
|
112
|
+
args: args, // Full args for Complete
|
|
113
|
+
status: ToolCallStatus.Complete,
|
|
114
|
+
result: toolMessage?.content || "",
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
// Check if render is a Component class or TemplateRef
|
|
118
|
+
if (this.isComponentClass(renderConfig.render)) {
|
|
119
|
+
// Create component dynamically
|
|
120
|
+
this.renderComponent(toolCall.id, renderConfig.render, props);
|
|
121
|
+
}
|
|
122
|
+
else if (this.isTemplateRef(renderConfig.render)) {
|
|
123
|
+
// Use template
|
|
124
|
+
this.renderTemplate(toolCall.id, renderConfig.render, props);
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
renderComponent(toolCallId, componentClass, props) {
|
|
128
|
+
if (!this.container) {
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
// Create the component
|
|
132
|
+
const componentRef = this.container.createComponent(componentClass);
|
|
133
|
+
this.componentRefs.set(toolCallId, componentRef);
|
|
134
|
+
// Set inputs on the component
|
|
135
|
+
for (const [key, value] of Object.entries(props)) {
|
|
136
|
+
try {
|
|
137
|
+
componentRef.setInput(key, value);
|
|
138
|
+
}
|
|
139
|
+
catch (e) {
|
|
140
|
+
// Input might not exist on the component, which is fine
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
// Trigger change detection
|
|
144
|
+
componentRef.changeDetectorRef.detectChanges();
|
|
145
|
+
}
|
|
146
|
+
renderTemplate(toolCallId, template, props) {
|
|
147
|
+
this.templateCache.set(toolCallId, {
|
|
148
|
+
template,
|
|
149
|
+
context: {
|
|
150
|
+
$implicit: props,
|
|
151
|
+
name: props.name,
|
|
152
|
+
description: props.description,
|
|
153
|
+
args: props.args,
|
|
154
|
+
status: props.status,
|
|
155
|
+
result: props.result,
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
isComponentClass(value) {
|
|
160
|
+
return typeof value === "function" && value.prototype;
|
|
161
|
+
}
|
|
162
|
+
isTemplateRef(value) {
|
|
163
|
+
return value instanceof TemplateRef;
|
|
164
|
+
}
|
|
165
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolCallsViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
166
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotChatToolCallsViewComponent, isStandalone: true, selector: "copilot-chat-tool-calls-view", inputs: { message: "message", messages: "messages", isLoading: "isLoading" }, viewQueries: [{ propertyName: "container", first: true, predicate: ["dynamicContainer"], descendants: true, read: ViewContainerRef }], usesOnChanges: true, ngImport: i0, template: `
|
|
167
|
+
@for (toolCall of message?.toolCalls ?? []; track toolCall.id) {
|
|
168
|
+
<ng-container>
|
|
169
|
+
<ng-container #dynamicContainer></ng-container>
|
|
170
|
+
<ng-container *ngIf="getTemplateForToolCall(toolCall) as templateData">
|
|
171
|
+
<ng-container
|
|
172
|
+
*ngTemplateOutlet="
|
|
173
|
+
templateData.template;
|
|
174
|
+
context: templateData.context
|
|
175
|
+
"
|
|
176
|
+
></ng-container>
|
|
177
|
+
</ng-container>
|
|
178
|
+
</ng-container>
|
|
179
|
+
}
|
|
180
|
+
`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
181
|
+
}
|
|
182
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolCallsViewComponent, decorators: [{
|
|
183
|
+
type: Component,
|
|
184
|
+
args: [{
|
|
185
|
+
selector: "copilot-chat-tool-calls-view",
|
|
186
|
+
standalone: true,
|
|
187
|
+
imports: [CommonModule],
|
|
188
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
189
|
+
template: `
|
|
190
|
+
@for (toolCall of message?.toolCalls ?? []; track toolCall.id) {
|
|
191
|
+
<ng-container>
|
|
192
|
+
<ng-container #dynamicContainer></ng-container>
|
|
193
|
+
<ng-container *ngIf="getTemplateForToolCall(toolCall) as templateData">
|
|
194
|
+
<ng-container
|
|
195
|
+
*ngTemplateOutlet="
|
|
196
|
+
templateData.template;
|
|
197
|
+
context: templateData.context
|
|
198
|
+
"
|
|
199
|
+
></ng-container>
|
|
200
|
+
</ng-container>
|
|
201
|
+
</ng-container>
|
|
202
|
+
}
|
|
203
|
+
`,
|
|
204
|
+
}]
|
|
205
|
+
}], propDecorators: { message: [{
|
|
206
|
+
type: Input,
|
|
207
|
+
args: [{ required: true }]
|
|
208
|
+
}], messages: [{
|
|
209
|
+
type: Input
|
|
210
|
+
}], isLoading: [{
|
|
211
|
+
type: Input
|
|
212
|
+
}], container: [{
|
|
213
|
+
type: ViewChild,
|
|
214
|
+
args: ["dynamicContainer", { read: ViewContainerRef }]
|
|
215
|
+
}] } });
|
|
216
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-tool-calls-view.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-tool-calls-view.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  ViewContainerRef,\n  ComponentRef,\n  inject,\n  ViewChild,\n  AfterViewInit,\n  OnChanges,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  ChangeDetectionStrategy,\n  signal,\n  computed,\n  OnDestroy,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport type {\n  AssistantMessage,\n  Message,\n  ToolCall,\n  ToolMessage,\n} from \"@ag-ui/core\";\nimport { ToolCallStatus } from \"@copilotkitnext/core\";\nimport { CopilotKitService } from \"../../core/copilotkit.service\";\nimport { partialJSONParse } from \"@copilotkitnext/shared\";\nimport type {\n  ToolCallProps,\n  ToolCallRender,\n} from \"../../core/copilotkit.types\";\n\n/**\n * Component for rendering all tool calls for an assistant message.\n * This component iterates through the message's tool calls and renders each one\n * using the registered render functions in CopilotKitService.\n */\n@Component({\n  selector: \"copilot-chat-tool-calls-view\",\n  standalone: true,\n  imports: [CommonModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `\n    @for (toolCall of message?.toolCalls ?? []; track toolCall.id) {\n      <ng-container>\n        <ng-container #dynamicContainer></ng-container>\n        <ng-container *ngIf=\"getTemplateForToolCall(toolCall) as templateData\">\n          <ng-container\n            *ngTemplateOutlet=\"\n              templateData.template;\n              context: templateData.context\n            \"\n          ></ng-container>\n        </ng-container>\n      </ng-container>\n    }\n  `,\n})\nexport class CopilotChatToolCallsViewComponent\n  implements AfterViewInit, OnChanges, OnDestroy\n{\n  @Input({ required: true }) message!: AssistantMessage;\n  @Input() messages: Message[] = [];\n  @Input() isLoading = false;\n\n  @ViewChild(\"dynamicContainer\", { read: ViewContainerRef })\n  private container?: ViewContainerRef;\n\n  private copilotkit: CopilotKitService | null = inject(CopilotKitService, {\n    optional: true,\n  });\n  private componentRefs: Map<string, ComponentRef<any>> = new Map();\n  private templateCache: Map<\n    string,\n    { template: TemplateRef<any>; context: any }\n  > = new Map();\n\n  // Signals for reactive state\n  private messageSignal = signal<AssistantMessage | null>(null);\n  private messagesSignal = signal<Message[]>([]);\n  private isLoadingSignal = signal(false);\n\n  ngAfterViewInit(): void {\n    this.renderAllToolCalls();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[\"message\"]) {\n      this.messageSignal.set(this.message);\n    }\n    if (changes[\"messages\"]) {\n      this.messagesSignal.set(this.messages);\n    }\n    if (changes[\"isLoading\"]) {\n      this.isLoadingSignal.set(this.isLoading);\n    }\n\n    this.renderAllToolCalls();\n  }\n\n  ngOnDestroy(): void {\n    // Clean up all component refs\n    this.componentRefs.forEach((ref) => ref.destroy());\n    this.componentRefs.clear();\n    this.templateCache.clear();\n  }\n\n  getTemplateForToolCall(\n    toolCall: ToolCall\n  ): { template: TemplateRef<any>; context: any } | null {\n    return this.templateCache.get(toolCall.id) || null;\n  }\n\n  private renderAllToolCalls(): void {\n    const message = this.messageSignal();\n    if (!message || !message.toolCalls || message.toolCalls.length === 0) {\n      return;\n    }\n\n    // Clear existing renders\n    this.componentRefs.forEach((ref) => ref.destroy());\n    this.componentRefs.clear();\n    this.templateCache.clear();\n\n    if (!this.copilotkit || !this.container) {\n      return;\n    }\n\n    const messages = this.messagesSignal();\n    const isLoading = this.isLoadingSignal();\n\n    // Render each tool call\n    message.toolCalls.forEach((toolCall) => {\n      const toolMessage = messages.find(\n        (m) =>\n          m.role === \"tool\" && (m as ToolMessage).toolCallId === toolCall.id\n      ) as ToolMessage | undefined;\n\n      this.renderSingleToolCall(toolCall, toolMessage, isLoading);\n    });\n  }\n\n  private renderSingleToolCall(\n    toolCall: ToolCall,\n    toolMessage: ToolMessage | undefined,\n    isLoading: boolean\n  ): void {\n    if (!this.copilotkit) {\n      return;\n    }\n\n    // Get current render tool calls\n    const currentRenderToolCalls = this.copilotkit.currentRenderToolCalls();\n\n    // Find the render config for this tool call by name\n    // Also check for wildcard (*) renders if no exact match\n    const renderConfig =\n      currentRenderToolCalls.find(\n        (rc: ToolCallRender) => rc.name === toolCall.function.name\n      ) || currentRenderToolCalls.find((rc: ToolCallRender) => rc.name === \"*\");\n\n    if (!renderConfig) {\n      return;\n    }\n\n    // Parse the arguments if they're a string\n    const args = partialJSONParse(toolCall.function.arguments);\n\n    // Determine status\n    let status: ToolCallStatus;\n    if (toolMessage) {\n      status = ToolCallStatus.Complete;\n    } else if (isLoading) {\n      status = ToolCallStatus.InProgress;\n    } else {\n      status = ToolCallStatus.Complete;\n    }\n\n    // Create props based on status - use discriminated union properly\n    let props: ToolCallProps;\n    if (status === ToolCallStatus.InProgress) {\n      props = {\n        name: toolCall.function.name,\n        description: \"\",\n        args: args, // Partial args for InProgress\n        status: ToolCallStatus.InProgress,\n        result: undefined,\n      };\n    } else {\n      // Complete status\n      props = {\n        name: toolCall.function.name,\n        description: \"\",\n        args: args, // Full args for Complete\n        status: ToolCallStatus.Complete,\n        result: toolMessage?.content || \"\",\n      };\n    }\n\n    // Check if render is a Component class or TemplateRef\n    if (this.isComponentClass(renderConfig.render)) {\n      // Create component dynamically\n      this.renderComponent(toolCall.id, renderConfig.render, props);\n    } else if (this.isTemplateRef(renderConfig.render)) {\n      // Use template\n      this.renderTemplate(toolCall.id, renderConfig.render, props);\n    }\n  }\n\n  private renderComponent(\n    toolCallId: string,\n    componentClass: Type<any>,\n    props: ToolCallProps\n  ): void {\n    if (!this.container) {\n      return;\n    }\n\n    // Create the component\n    const componentRef = this.container.createComponent(componentClass);\n    this.componentRefs.set(toolCallId, componentRef);\n\n    // Set inputs on the component\n    for (const [key, value] of Object.entries(props)) {\n      try {\n        componentRef.setInput(key, value);\n      } catch (e) {\n        // Input might not exist on the component, which is fine\n      }\n    }\n\n    // Trigger change detection\n    componentRef.changeDetectorRef.detectChanges();\n  }\n\n  private renderTemplate(\n    toolCallId: string,\n    template: TemplateRef<any>,\n    props: ToolCallProps\n  ): void {\n    this.templateCache.set(toolCallId, {\n      template,\n      context: {\n        $implicit: props,\n        name: props.name,\n        description: props.description,\n        args: props.args,\n        status: props.status,\n        result: props.result,\n      },\n    });\n  }\n\n  private isComponentClass(value: any): value is Type<any> {\n    return typeof value === \"function\" && value.prototype;\n  }\n\n  private isTemplateRef(value: any): value is TemplateRef<any> {\n    return value instanceof TemplateRef;\n  }\n}\n"]}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { Component, Input, signal, computed, ChangeDetectionStrategy, ViewEncapsulation } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { cn } from '../../lib/utils';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export class CopilotChatToolbarComponent {
|
|
6
|
+
set inputClass(val) {
|
|
7
|
+
this.customClass.set(val);
|
|
8
|
+
}
|
|
9
|
+
customClass = signal(undefined);
|
|
10
|
+
computedClass = computed(() => {
|
|
11
|
+
const baseClasses = 'w-full h-[60px] bg-transparent flex items-center justify-between';
|
|
12
|
+
return cn(baseClasses, this.customClass());
|
|
13
|
+
});
|
|
14
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
15
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatToolbarComponent, isStandalone: true, selector: "div[copilotChatToolbar]", inputs: { inputClass: "inputClass" }, host: { properties: { "class": "computedClass()" } }, ngImport: i0, template: `<ng-content></ng-content>`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
16
|
+
}
|
|
17
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolbarComponent, decorators: [{
|
|
18
|
+
type: Component,
|
|
19
|
+
args: [{ selector: 'div[copilotChatToolbar]', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
|
|
20
|
+
'[class]': 'computedClass()'
|
|
21
|
+
}, template: `<ng-content></ng-content>` }]
|
|
22
|
+
}], propDecorators: { inputClass: [{
|
|
23
|
+
type: Input
|
|
24
|
+
}] } });
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LXRvb2xiYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtdG9vbGJhci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsTUFBTSxFQUNOLFFBQVEsRUFDUix1QkFBdUIsRUFDdkIsaUJBQWlCLEVBQ2xCLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsRUFBRSxFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBY3JDLE1BQU0sT0FBTywyQkFBMkI7SUFDdEMsSUFBYSxVQUFVLENBQUMsR0FBdUI7UUFDN0MsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDNUIsQ0FBQztJQUVELFdBQVcsR0FBRyxNQUFNLENBQXFCLFNBQVMsQ0FBQyxDQUFDO0lBRXBELGFBQWEsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQzVCLE1BQU0sV0FBVyxHQUFHLGtFQUFrRSxDQUFDO1FBQ3ZGLE9BQU8sRUFBRSxDQUFDLFdBQVcsRUFBRSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQztJQUM3QyxDQUFDLENBQUMsQ0FBQzt3R0FWUSwyQkFBMkI7NEZBQTNCLDJCQUEyQiwrS0FINUIsMkJBQTJCLDJEQU4zQixZQUFZOzs0RkFTWCwyQkFBMkI7a0JBWnZDLFNBQVM7K0JBQ0UseUJBQXlCLGNBQ3ZCLElBQUksV0FDUCxDQUFDLFlBQVksQ0FBQyxtQkFDTix1QkFBdUIsQ0FBQyxNQUFNLGlCQUNoQyxpQkFBaUIsQ0FBQyxJQUFJLFFBQy9CO3dCQUNKLFNBQVMsRUFBRSxpQkFBaUI7cUJBQzdCLFlBQ1MsMkJBQTJCOzhCQUl4QixVQUFVO3NCQUF0QixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBJbnB1dCxcbiAgc2lnbmFsLFxuICBjb21wdXRlZCxcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG4gIFZpZXdFbmNhcHN1bGF0aW9uXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IGNuIH0gZnJvbSAnLi4vLi4vbGliL3V0aWxzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnZGl2W2NvcGlsb3RDaGF0VG9vbGJhcl0nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIGVuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXG4gIGhvc3Q6IHtcbiAgICAnW2NsYXNzXSc6ICdjb21wdXRlZENsYXNzKCknXG4gIH0sXG4gIHRlbXBsYXRlOiBgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PmAsXG4gIHN0eWxlczogW11cbn0pXG5leHBvcnQgY2xhc3MgQ29waWxvdENoYXRUb29sYmFyQ29tcG9uZW50IHtcbiAgQElucHV0KCkgc2V0IGlucHV0Q2xhc3ModmFsOiBzdHJpbmcgfCB1bmRlZmluZWQpIHtcbiAgICB0aGlzLmN1c3RvbUNsYXNzLnNldCh2YWwpO1xuICB9XG4gIFxuICBjdXN0b21DbGFzcyA9IHNpZ25hbDxzdHJpbmcgfCB1bmRlZmluZWQ+KHVuZGVmaW5lZCk7XG4gIFxuICBjb21wdXRlZENsYXNzID0gY29tcHV0ZWQoKCkgPT4ge1xuICAgIGNvbnN0IGJhc2VDbGFzc2VzID0gJ3ctZnVsbCBoLVs2MHB4XSBiZy10cmFuc3BhcmVudCBmbGV4IGl0ZW1zLWNlbnRlciBqdXN0aWZ5LWJldHdlZW4nO1xuICAgIHJldHVybiBjbihiYXNlQ2xhc3NlcywgdGhpcy5jdXN0b21DbGFzcygpKTtcbiAgfSk7XG59Il19
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import { Component, Input, signal, computed, inject, ChangeDetectionStrategy, ViewEncapsulation } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { CdkMenuModule } from '@angular/cdk/menu';
|
|
4
|
+
import { OverlayModule } from '@angular/cdk/overlay';
|
|
5
|
+
import { LucideAngularModule, Settings2, ChevronRight } from 'lucide-angular';
|
|
6
|
+
import { CopilotChatConfigurationService } from '../../core/chat-configuration/chat-configuration.service';
|
|
7
|
+
import { cn } from '../../lib/utils';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/cdk/menu";
|
|
10
|
+
import * as i2 from "lucide-angular";
|
|
11
|
+
export class CopilotChatToolsMenuComponent {
|
|
12
|
+
Settings2Icon = Settings2;
|
|
13
|
+
ChevronRightIcon = ChevronRight;
|
|
14
|
+
set inputToolsMenu(val) {
|
|
15
|
+
this.toolsMenu.set(val || []);
|
|
16
|
+
}
|
|
17
|
+
set inputDisabled(val) {
|
|
18
|
+
this.disabled.set(val || false);
|
|
19
|
+
}
|
|
20
|
+
set inputClass(val) {
|
|
21
|
+
this.customClass.set(val);
|
|
22
|
+
}
|
|
23
|
+
chatConfig = inject(CopilotChatConfigurationService);
|
|
24
|
+
toolsMenu = signal([]);
|
|
25
|
+
disabled = signal(false);
|
|
26
|
+
customClass = signal(undefined);
|
|
27
|
+
hasItems = computed(() => this.toolsMenu().length > 0);
|
|
28
|
+
label = computed(() => this.chatConfig.labels().chatInputToolbarToolsButtonLabel);
|
|
29
|
+
buttonClass = computed(() => {
|
|
30
|
+
const baseClasses = cn(
|
|
31
|
+
// Base button styles
|
|
32
|
+
'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-full text-sm font-medium', 'transition-all disabled:pointer-events-none disabled:opacity-50', 'shrink-0 outline-none', 'focus-visible:ring-[3px]',
|
|
33
|
+
// chatInputToolbarSecondary variant
|
|
34
|
+
'cursor-pointer', 'bg-transparent text-[#444444]', 'dark:text-white dark:border-[#404040]', 'transition-colors', 'focus:outline-none', 'hover:bg-[#f8f8f8] hover:text-[#333333]', 'dark:hover:bg-[#404040] dark:hover:text-[#FFFFFF]', 'disabled:cursor-not-allowed disabled:opacity-50', 'disabled:hover:bg-transparent disabled:hover:text-[#444444]', 'dark:disabled:hover:bg-transparent dark:disabled:hover:text-[#CCCCCC]',
|
|
35
|
+
// Size
|
|
36
|
+
'h-9 px-3 gap-2 font-normal');
|
|
37
|
+
return cn(baseClasses, this.customClass());
|
|
38
|
+
});
|
|
39
|
+
isMenuItem(item) {
|
|
40
|
+
return item && typeof item === 'object' && 'label' in item;
|
|
41
|
+
}
|
|
42
|
+
handleItemClick(item) {
|
|
43
|
+
if (item.action) {
|
|
44
|
+
item.action();
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolsMenuComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
48
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotChatToolsMenuComponent, isStandalone: true, selector: "copilot-chat-tools-menu", inputs: { inputToolsMenu: "inputToolsMenu", inputDisabled: "inputDisabled", inputClass: "inputClass" }, ngImport: i0, template: `
|
|
49
|
+
@if (hasItems()) {
|
|
50
|
+
<button
|
|
51
|
+
type="button"
|
|
52
|
+
[disabled]="disabled()"
|
|
53
|
+
[class]="buttonClass()"
|
|
54
|
+
[cdkMenuTriggerFor]="menu"
|
|
55
|
+
>
|
|
56
|
+
<lucide-angular [img]="Settings2Icon" [size]="18"></lucide-angular>
|
|
57
|
+
<span class="text-sm font-normal">{{ label() }}</span>
|
|
58
|
+
</button>
|
|
59
|
+
|
|
60
|
+
<ng-template #menu>
|
|
61
|
+
<div class="min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1"
|
|
62
|
+
cdkMenu>
|
|
63
|
+
@for (item of toolsMenu(); track $index) {
|
|
64
|
+
@if (item === '-') {
|
|
65
|
+
<div class="h-px bg-gray-200 dark:bg-gray-700 my-1"></div>
|
|
66
|
+
} @else if (isMenuItem(item)) {
|
|
67
|
+
@if (item.items && item.items.length > 0) {
|
|
68
|
+
<!-- Submenu trigger -->
|
|
69
|
+
<button
|
|
70
|
+
type="button"
|
|
71
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm flex items-center justify-between"
|
|
72
|
+
[cdkMenuTriggerFor]="submenu"
|
|
73
|
+
cdkMenuItem
|
|
74
|
+
>
|
|
75
|
+
{{ item.label }}
|
|
76
|
+
<lucide-angular [img]="ChevronRightIcon" [size]="12" class="ml-auto"></lucide-angular>
|
|
77
|
+
</button>
|
|
78
|
+
|
|
79
|
+
<!-- Submenu template -->
|
|
80
|
+
<ng-template #submenu>
|
|
81
|
+
<div class="min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1"
|
|
82
|
+
cdkMenu>
|
|
83
|
+
@for (subItem of item.items; track $index) {
|
|
84
|
+
@if (subItem === '-') {
|
|
85
|
+
<div class="h-px bg-gray-200 dark:bg-gray-700 my-1"></div>
|
|
86
|
+
} @else if (isMenuItem(subItem)) {
|
|
87
|
+
<button
|
|
88
|
+
type="button"
|
|
89
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm"
|
|
90
|
+
(click)="handleItemClick(subItem)"
|
|
91
|
+
cdkMenuItem
|
|
92
|
+
>
|
|
93
|
+
{{ subItem.label }}
|
|
94
|
+
</button>
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
</div>
|
|
98
|
+
</ng-template>
|
|
99
|
+
} @else {
|
|
100
|
+
<!-- Regular menu item -->
|
|
101
|
+
<button
|
|
102
|
+
type="button"
|
|
103
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm"
|
|
104
|
+
(click)="handleItemClick(item)"
|
|
105
|
+
cdkMenuItem
|
|
106
|
+
>
|
|
107
|
+
{{ item.label }}
|
|
108
|
+
</button>
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
</div>
|
|
113
|
+
</ng-template>
|
|
114
|
+
}
|
|
115
|
+
`, isInline: true, styles: [".cdk-overlay-pane{position:absolute;pointer-events:auto;z-index:1000}.cdk-overlay-container{position:fixed;z-index:1000}[cdkMenu]{animation:menuFadeIn .15s ease-out}@keyframes menuFadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: CdkMenuModule }, { kind: "directive", type: i1.CdkMenu, selector: "[cdkMenu]", outputs: ["closed"], exportAs: ["cdkMenu"] }, { kind: "directive", type: i1.CdkMenuItem, selector: "[cdkMenuItem]", inputs: ["cdkMenuItemDisabled", "cdkMenuitemTypeaheadLabel"], outputs: ["cdkMenuItemTriggered"], exportAs: ["cdkMenuItem"] }, { kind: "directive", type: i1.CdkMenuTrigger, selector: "[cdkMenuTriggerFor]", inputs: ["cdkMenuTriggerFor", "cdkMenuPosition", "cdkMenuTriggerData"], outputs: ["cdkMenuOpened", "cdkMenuClosed"], exportAs: ["cdkMenuTriggerFor"] }, { kind: "ngmodule", type: OverlayModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
116
|
+
}
|
|
117
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatToolsMenuComponent, decorators: [{
|
|
118
|
+
type: Component,
|
|
119
|
+
args: [{ selector: 'copilot-chat-tools-menu', standalone: true, imports: [
|
|
120
|
+
CommonModule,
|
|
121
|
+
CdkMenuModule,
|
|
122
|
+
OverlayModule,
|
|
123
|
+
LucideAngularModule
|
|
124
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, template: `
|
|
125
|
+
@if (hasItems()) {
|
|
126
|
+
<button
|
|
127
|
+
type="button"
|
|
128
|
+
[disabled]="disabled()"
|
|
129
|
+
[class]="buttonClass()"
|
|
130
|
+
[cdkMenuTriggerFor]="menu"
|
|
131
|
+
>
|
|
132
|
+
<lucide-angular [img]="Settings2Icon" [size]="18"></lucide-angular>
|
|
133
|
+
<span class="text-sm font-normal">{{ label() }}</span>
|
|
134
|
+
</button>
|
|
135
|
+
|
|
136
|
+
<ng-template #menu>
|
|
137
|
+
<div class="min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1"
|
|
138
|
+
cdkMenu>
|
|
139
|
+
@for (item of toolsMenu(); track $index) {
|
|
140
|
+
@if (item === '-') {
|
|
141
|
+
<div class="h-px bg-gray-200 dark:bg-gray-700 my-1"></div>
|
|
142
|
+
} @else if (isMenuItem(item)) {
|
|
143
|
+
@if (item.items && item.items.length > 0) {
|
|
144
|
+
<!-- Submenu trigger -->
|
|
145
|
+
<button
|
|
146
|
+
type="button"
|
|
147
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm flex items-center justify-between"
|
|
148
|
+
[cdkMenuTriggerFor]="submenu"
|
|
149
|
+
cdkMenuItem
|
|
150
|
+
>
|
|
151
|
+
{{ item.label }}
|
|
152
|
+
<lucide-angular [img]="ChevronRightIcon" [size]="12" class="ml-auto"></lucide-angular>
|
|
153
|
+
</button>
|
|
154
|
+
|
|
155
|
+
<!-- Submenu template -->
|
|
156
|
+
<ng-template #submenu>
|
|
157
|
+
<div class="min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1"
|
|
158
|
+
cdkMenu>
|
|
159
|
+
@for (subItem of item.items; track $index) {
|
|
160
|
+
@if (subItem === '-') {
|
|
161
|
+
<div class="h-px bg-gray-200 dark:bg-gray-700 my-1"></div>
|
|
162
|
+
} @else if (isMenuItem(subItem)) {
|
|
163
|
+
<button
|
|
164
|
+
type="button"
|
|
165
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm"
|
|
166
|
+
(click)="handleItemClick(subItem)"
|
|
167
|
+
cdkMenuItem
|
|
168
|
+
>
|
|
169
|
+
{{ subItem.label }}
|
|
170
|
+
</button>
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
</div>
|
|
174
|
+
</ng-template>
|
|
175
|
+
} @else {
|
|
176
|
+
<!-- Regular menu item -->
|
|
177
|
+
<button
|
|
178
|
+
type="button"
|
|
179
|
+
class="w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm"
|
|
180
|
+
(click)="handleItemClick(item)"
|
|
181
|
+
cdkMenuItem
|
|
182
|
+
>
|
|
183
|
+
{{ item.label }}
|
|
184
|
+
</button>
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
</div>
|
|
189
|
+
</ng-template>
|
|
190
|
+
}
|
|
191
|
+
`, styles: [".cdk-overlay-pane{position:absolute;pointer-events:auto;z-index:1000}.cdk-overlay-container{position:fixed;z-index:1000}[cdkMenu]{animation:menuFadeIn .15s ease-out}@keyframes menuFadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}\n"] }]
|
|
192
|
+
}], propDecorators: { inputToolsMenu: [{
|
|
193
|
+
type: Input
|
|
194
|
+
}], inputDisabled: [{
|
|
195
|
+
type: Input
|
|
196
|
+
}], inputClass: [{
|
|
197
|
+
type: Input
|
|
198
|
+
}] } });
|
|
199
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-tools-menu.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-tools-menu.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  signal,\n  computed,\n  inject,\n  ChangeDetectionStrategy,\n  ViewEncapsulation\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { CdkMenuModule } from '@angular/cdk/menu';\nimport { OverlayModule } from '@angular/cdk/overlay';\nimport { LucideAngularModule, Settings2, ChevronRight } from 'lucide-angular';\nimport { CopilotChatConfigurationService } from '../../core/chat-configuration/chat-configuration.service';\nimport type { ToolsMenuItem } from './copilot-chat-input.types';\nimport { cn } from '../../lib/utils';\n\n@Component({\n  selector: 'copilot-chat-tools-menu',\n  standalone: true,\n  imports: [\n    CommonModule,\n    CdkMenuModule,\n    OverlayModule,\n    LucideAngularModule\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    @if (hasItems()) {\n      <button\n        type=\"button\"\n        [disabled]=\"disabled()\"\n        [class]=\"buttonClass()\"\n        [cdkMenuTriggerFor]=\"menu\"\n      >\n        <lucide-angular [img]=\"Settings2Icon\" [size]=\"18\"></lucide-angular>\n        <span class=\"text-sm font-normal\">{{ label() }}</span>\n      </button>\n      \n      <ng-template #menu>\n        <div class=\"min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1\"\n             cdkMenu>\n          @for (item of toolsMenu(); track $index) {\n            @if (item === '-') {\n              <div class=\"h-px bg-gray-200 dark:bg-gray-700 my-1\"></div>\n            } @else if (isMenuItem(item)) {\n              @if (item.items && item.items.length > 0) {\n                <!-- Submenu trigger -->\n                <button\n                  type=\"button\"\n                  class=\"w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm flex items-center justify-between\"\n                  [cdkMenuTriggerFor]=\"submenu\"\n                  cdkMenuItem\n                >\n                  {{ item.label }}\n                  <lucide-angular [img]=\"ChevronRightIcon\" [size]=\"12\" class=\"ml-auto\"></lucide-angular>\n                </button>\n                \n                <!-- Submenu template -->\n                <ng-template #submenu>\n                  <div class=\"min-w-[200px] bg-white dark:bg-[#1F1F1F] border border-gray-200 dark:border-gray-700 rounded-lg shadow-lg p-1\"\n                       cdkMenu>\n                    @for (subItem of item.items; track $index) {\n                      @if (subItem === '-') {\n                        <div class=\"h-px bg-gray-200 dark:bg-gray-700 my-1\"></div>\n                      } @else if (isMenuItem(subItem)) {\n                        <button\n                          type=\"button\"\n                          class=\"w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm\"\n                          (click)=\"handleItemClick(subItem)\"\n                          cdkMenuItem\n                        >\n                          {{ subItem.label }}\n                        </button>\n                      }\n                    }\n                  </div>\n                </ng-template>\n              } @else {\n                <!-- Regular menu item -->\n                <button\n                  type=\"button\"\n                  class=\"w-full px-3 py-2 text-left bg-transparent border-none rounded hover:bg-gray-100 dark:hover:bg-gray-800 cursor-pointer text-sm\"\n                  (click)=\"handleItemClick(item)\"\n                  cdkMenuItem\n                >\n                  {{ item.label }}\n                </button>\n              }\n            }\n          }\n        </div>\n      </ng-template>\n    }\n  `,\n  styles: [`\n    /* CDK Overlay styles for positioning */\n    .cdk-overlay-pane {\n      position: absolute;\n      pointer-events: auto;\n      z-index: 1000;\n    }\n    \n    /* Ensure menu appears above other content */\n    .cdk-overlay-container {\n      position: fixed;\n      z-index: 1000;\n    }\n    \n    /* Menu animation */\n    [cdkMenu] {\n      animation: menuFadeIn 0.15s ease-out;\n    }\n    \n    @keyframes menuFadeIn {\n      from {\n        opacity: 0;\n        transform: translateY(4px);\n      }\n      to {\n        opacity: 1;\n        transform: translateY(0);\n      }\n    }\n  `]\n})\nexport class CopilotChatToolsMenuComponent {\n  readonly Settings2Icon = Settings2;\n  readonly ChevronRightIcon = ChevronRight;\n  \n  @Input() set inputToolsMenu(val: (ToolsMenuItem | '-')[] | undefined) {\n    this.toolsMenu.set(val || []);\n  }\n  @Input() set inputDisabled(val: boolean | undefined) {\n    this.disabled.set(val || false);\n  }\n  @Input() set inputClass(val: string | undefined) {\n    this.customClass.set(val);\n  }\n  \n  private chatConfig = inject(CopilotChatConfigurationService);\n  \n  toolsMenu = signal<(ToolsMenuItem | '-')[]>([]);\n  disabled = signal<boolean>(false);\n  customClass = signal<string | undefined>(undefined);\n  \n  hasItems = computed(() => this.toolsMenu().length > 0);\n  \n  label = computed(() => this.chatConfig.labels().chatInputToolbarToolsButtonLabel);\n  \n  buttonClass = computed(() => {\n    const baseClasses = cn(\n      // Base button styles\n      'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-full text-sm font-medium',\n      'transition-all disabled:pointer-events-none disabled:opacity-50',\n      'shrink-0 outline-none',\n      'focus-visible:ring-[3px]',\n      // chatInputToolbarSecondary variant\n      'cursor-pointer',\n      'bg-transparent text-[#444444]',\n      'dark:text-white dark:border-[#404040]',\n      'transition-colors',\n      'focus:outline-none',\n      'hover:bg-[#f8f8f8] hover:text-[#333333]',\n      'dark:hover:bg-[#404040] dark:hover:text-[#FFFFFF]',\n      'disabled:cursor-not-allowed disabled:opacity-50',\n      'disabled:hover:bg-transparent disabled:hover:text-[#444444]',\n      'dark:disabled:hover:bg-transparent dark:disabled:hover:text-[#CCCCCC]',\n      // Size\n      'h-9 px-3 gap-2 font-normal'\n    );\n    return cn(baseClasses, this.customClass());\n  });\n  \n  isMenuItem(item: any): item is ToolsMenuItem {\n    return item && typeof item === 'object' && 'label' in item;\n  }\n  \n  handleItemClick(item: ToolsMenuItem): void {\n    if (item.action) {\n      item.action();\n    }\n  }\n}\n"]}
|