@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,214 @@
|
|
|
1
|
+
import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, signal, effect, runInInjectionContext, Optional, SkipSelf, } from "@angular/core";
|
|
2
|
+
import { CommonModule } from "@angular/common";
|
|
3
|
+
import { CopilotChatViewComponent } from "./copilot-chat-view.component";
|
|
4
|
+
import { CopilotChatConfigurationService } from "../../core/chat-configuration/chat-configuration.service";
|
|
5
|
+
import { COPILOT_CHAT_INITIAL_CONFIG, } from "../../core/chat-configuration/chat-configuration.types";
|
|
6
|
+
import { watchAgent } from "../../utils/agent.utils";
|
|
7
|
+
import { DEFAULT_AGENT_ID, randomUUID } from "@copilotkitnext/shared";
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "../../core/chat-configuration/chat-configuration.service";
|
|
10
|
+
/**
|
|
11
|
+
* CopilotChat component - Angular equivalent of React's <CopilotChat>
|
|
12
|
+
* Provides a complete chat interface that wires an agent to the chat view
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```html
|
|
16
|
+
* <copilot-chat [agentId]="'default'" [threadId]="'abc123'"></copilot-chat>
|
|
17
|
+
* ```
|
|
18
|
+
*/
|
|
19
|
+
export class CopilotChatComponent {
|
|
20
|
+
chatConfig;
|
|
21
|
+
cdr;
|
|
22
|
+
injector;
|
|
23
|
+
agentId;
|
|
24
|
+
threadId;
|
|
25
|
+
constructor(chatConfig, cdr, injector) {
|
|
26
|
+
this.chatConfig = chatConfig;
|
|
27
|
+
this.cdr = cdr;
|
|
28
|
+
this.injector = injector;
|
|
29
|
+
// Create initial watcher once (constructor is a safe injection context)
|
|
30
|
+
const initialId = this.agentId ?? DEFAULT_AGENT_ID;
|
|
31
|
+
this.createWatcher(initialId);
|
|
32
|
+
// Connect once when agent becomes available
|
|
33
|
+
effect(() => {
|
|
34
|
+
const a = this.agent();
|
|
35
|
+
if (!a)
|
|
36
|
+
return;
|
|
37
|
+
// Apply thread id when agent is available
|
|
38
|
+
a.threadId = this.threadId || this.generatedThreadId;
|
|
39
|
+
if (!this.hasConnectedOnce) {
|
|
40
|
+
this.hasConnectedOnce = true;
|
|
41
|
+
this.connectToAgent(a);
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
// Signals from watchAgent - using direct references instead of assignment
|
|
46
|
+
agent = signal(undefined).asReadonly();
|
|
47
|
+
messages = signal([]).asReadonly();
|
|
48
|
+
isRunning = signal(false).asReadonly();
|
|
49
|
+
showCursor = signal(false);
|
|
50
|
+
generatedThreadId = randomUUID();
|
|
51
|
+
agentWatcher;
|
|
52
|
+
hasConnectedOnce = false;
|
|
53
|
+
lastAgentId;
|
|
54
|
+
ngOnInit() {
|
|
55
|
+
this.setupChatHandlers();
|
|
56
|
+
}
|
|
57
|
+
ngOnChanges(changes) {
|
|
58
|
+
if (changes["agentId"] && !changes["agentId"].firstChange) {
|
|
59
|
+
const newId = this.agentId ?? DEFAULT_AGENT_ID;
|
|
60
|
+
this.createWatcher(newId);
|
|
61
|
+
}
|
|
62
|
+
if (changes["threadId"] && !changes["threadId"].firstChange) {
|
|
63
|
+
const a = this.agent();
|
|
64
|
+
if (a) {
|
|
65
|
+
a.threadId = this.threadId || this.generatedThreadId;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
async connectToAgent(agent) {
|
|
70
|
+
if (!agent)
|
|
71
|
+
return;
|
|
72
|
+
this.showCursor.set(true);
|
|
73
|
+
this.cdr.markForCheck();
|
|
74
|
+
try {
|
|
75
|
+
await agent.runAgent({ forwardedProps: { __copilotkitConnect: true } }, {
|
|
76
|
+
onTextMessageStartEvent: () => {
|
|
77
|
+
this.showCursor.set(false);
|
|
78
|
+
this.cdr.detectChanges();
|
|
79
|
+
},
|
|
80
|
+
onToolCallStartEvent: () => {
|
|
81
|
+
this.showCursor.set(false);
|
|
82
|
+
this.cdr.detectChanges();
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
this.showCursor.set(false);
|
|
86
|
+
this.cdr.markForCheck();
|
|
87
|
+
}
|
|
88
|
+
catch (error) {
|
|
89
|
+
console.error("Failed to connect to agent:", error);
|
|
90
|
+
this.showCursor.set(false);
|
|
91
|
+
this.cdr.markForCheck();
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
setupChatHandlers() {
|
|
95
|
+
if (!this.chatConfig)
|
|
96
|
+
return;
|
|
97
|
+
// Handle input submission
|
|
98
|
+
this.chatConfig.setSubmitHandler(async (value) => {
|
|
99
|
+
const agent = this.agent();
|
|
100
|
+
if (!agent || !value.trim())
|
|
101
|
+
return;
|
|
102
|
+
// Add user message
|
|
103
|
+
const userMessage = {
|
|
104
|
+
id: randomUUID(),
|
|
105
|
+
role: "user",
|
|
106
|
+
content: value,
|
|
107
|
+
};
|
|
108
|
+
agent.addMessage(userMessage);
|
|
109
|
+
// Clear the input
|
|
110
|
+
this.chatConfig.setInputValue("");
|
|
111
|
+
// Show cursor while processing
|
|
112
|
+
this.showCursor.set(true);
|
|
113
|
+
this.cdr.markForCheck();
|
|
114
|
+
// Run the agent with named subscriber callbacks
|
|
115
|
+
try {
|
|
116
|
+
await agent.runAgent({}, {
|
|
117
|
+
onTextMessageStartEvent: () => {
|
|
118
|
+
this.showCursor.set(false);
|
|
119
|
+
this.cdr.detectChanges();
|
|
120
|
+
},
|
|
121
|
+
onToolCallStartEvent: () => {
|
|
122
|
+
this.showCursor.set(false);
|
|
123
|
+
this.cdr.detectChanges();
|
|
124
|
+
},
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
catch (error) {
|
|
128
|
+
console.error("Agent run error:", error);
|
|
129
|
+
}
|
|
130
|
+
finally {
|
|
131
|
+
this.showCursor.set(false);
|
|
132
|
+
this.cdr.markForCheck();
|
|
133
|
+
}
|
|
134
|
+
});
|
|
135
|
+
// Handle input value changes (optional)
|
|
136
|
+
this.chatConfig.setChangeHandler(() => {
|
|
137
|
+
// Keep input state if needed
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
ngOnDestroy() {
|
|
141
|
+
if (this.agentWatcher?.unsubscribe) {
|
|
142
|
+
this.agentWatcher.unsubscribe();
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
createWatcher(desiredAgentId) {
|
|
146
|
+
// Tear down previous watcher if it exists
|
|
147
|
+
if (this.agentWatcher?.unsubscribe) {
|
|
148
|
+
this.agentWatcher.unsubscribe();
|
|
149
|
+
this.agentWatcher = undefined;
|
|
150
|
+
}
|
|
151
|
+
// Setup watcher for desired agent - ensure injection context
|
|
152
|
+
this.agentWatcher = runInInjectionContext(this.injector, () => watchAgent({ agentId: desiredAgentId }));
|
|
153
|
+
this.agent = this.agentWatcher.agent;
|
|
154
|
+
this.messages = this.agentWatcher.messages;
|
|
155
|
+
this.isRunning = this.agentWatcher.isRunning;
|
|
156
|
+
this.hasConnectedOnce = false;
|
|
157
|
+
this.lastAgentId = desiredAgentId;
|
|
158
|
+
}
|
|
159
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatComponent, deps: [{ token: i1.CopilotChatConfigurationService, optional: true }, { token: i0.ChangeDetectorRef }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Component });
|
|
160
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatComponent, isStandalone: true, selector: "copilot-chat", inputs: { agentId: "agentId", threadId: "threadId" }, providers: [
|
|
161
|
+
{
|
|
162
|
+
provide: CopilotChatConfigurationService,
|
|
163
|
+
deps: [
|
|
164
|
+
[new Optional(), new SkipSelf(), CopilotChatConfigurationService],
|
|
165
|
+
[new Optional(), COPILOT_CHAT_INITIAL_CONFIG],
|
|
166
|
+
],
|
|
167
|
+
useFactory: (parent, initial) => parent ?? new CopilotChatConfigurationService(initial ?? null),
|
|
168
|
+
},
|
|
169
|
+
], usesOnChanges: true, ngImport: i0, template: `
|
|
170
|
+
<copilot-chat-view
|
|
171
|
+
[messages]="messages()"
|
|
172
|
+
[autoScroll]="true"
|
|
173
|
+
[messageViewClass]="'w-full'"
|
|
174
|
+
[showCursor]="showCursor()"
|
|
175
|
+
>
|
|
176
|
+
</copilot-chat-view>
|
|
177
|
+
`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: CopilotChatViewComponent, selector: "copilot-chat-view", inputs: ["messages", "autoScroll", "showCursor", "messageViewComponent", "messageViewTemplate", "messageViewClass", "scrollViewComponent", "scrollViewTemplate", "scrollViewClass", "scrollToBottomButtonComponent", "scrollToBottomButtonTemplate", "scrollToBottomButtonClass", "inputComponent", "inputTemplate", "inputContainerComponent", "inputContainerTemplate", "inputContainerClass", "featherComponent", "featherTemplate", "featherClass", "disclaimerComponent", "disclaimerTemplate", "disclaimerClass", "disclaimerText"], outputs: ["assistantMessageThumbsUp", "assistantMessageThumbsDown", "assistantMessageReadAloud", "assistantMessageRegenerate", "userMessageCopy", "userMessageEdit"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
178
|
+
}
|
|
179
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatComponent, decorators: [{
|
|
180
|
+
type: Component,
|
|
181
|
+
args: [{
|
|
182
|
+
selector: "copilot-chat",
|
|
183
|
+
standalone: true,
|
|
184
|
+
imports: [CommonModule, CopilotChatViewComponent],
|
|
185
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
186
|
+
encapsulation: ViewEncapsulation.None,
|
|
187
|
+
providers: [
|
|
188
|
+
{
|
|
189
|
+
provide: CopilotChatConfigurationService,
|
|
190
|
+
deps: [
|
|
191
|
+
[new Optional(), new SkipSelf(), CopilotChatConfigurationService],
|
|
192
|
+
[new Optional(), COPILOT_CHAT_INITIAL_CONFIG],
|
|
193
|
+
],
|
|
194
|
+
useFactory: (parent, initial) => parent ?? new CopilotChatConfigurationService(initial ?? null),
|
|
195
|
+
},
|
|
196
|
+
],
|
|
197
|
+
template: `
|
|
198
|
+
<copilot-chat-view
|
|
199
|
+
[messages]="messages()"
|
|
200
|
+
[autoScroll]="true"
|
|
201
|
+
[messageViewClass]="'w-full'"
|
|
202
|
+
[showCursor]="showCursor()"
|
|
203
|
+
>
|
|
204
|
+
</copilot-chat-view>
|
|
205
|
+
`,
|
|
206
|
+
}]
|
|
207
|
+
}], ctorParameters: () => [{ type: i1.CopilotChatConfigurationService, decorators: [{
|
|
208
|
+
type: Optional
|
|
209
|
+
}] }, { type: i0.ChangeDetectorRef }, { type: i0.Injector }], propDecorators: { agentId: [{
|
|
210
|
+
type: Input
|
|
211
|
+
}], threadId: [{
|
|
212
|
+
type: Input
|
|
213
|
+
}] } });
|
|
214
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  OnInit,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  signal,\n  effect,\n  ChangeDetectorRef,\n  Signal,\n  Injector,\n  runInInjectionContext,\n  Optional,\n  SkipSelf,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { CopilotChatViewComponent } from \"./copilot-chat-view.component\";\nimport { CopilotChatConfigurationService } from \"../../core/chat-configuration/chat-configuration.service\";\nimport {\n  COPILOT_CHAT_INITIAL_CONFIG,\n  CopilotChatConfiguration,\n} from \"../../core/chat-configuration/chat-configuration.types\";\nimport { watchAgent } from \"../../utils/agent.utils\";\nimport { AgentWatchResult } from \"../../core/copilotkit.types\";\nimport { DEFAULT_AGENT_ID, randomUUID } from \"@copilotkitnext/shared\";\nimport { Message, AbstractAgent } from \"@ag-ui/client\";\n\n/**\n * CopilotChat component - Angular equivalent of React's <CopilotChat>\n * Provides a complete chat interface that wires an agent to the chat view\n *\n * @example\n * ```html\n * <copilot-chat [agentId]=\"'default'\" [threadId]=\"'abc123'\"></copilot-chat>\n * ```\n */\n@Component({\n  selector: \"copilot-chat\",\n  standalone: true,\n  imports: [CommonModule, CopilotChatViewComponent],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: CopilotChatConfigurationService,\n      deps: [\n        [new Optional(), new SkipSelf(), CopilotChatConfigurationService],\n        [new Optional(), COPILOT_CHAT_INITIAL_CONFIG],\n      ],\n      useFactory: (\n        parent: CopilotChatConfigurationService | null,\n        initial: CopilotChatConfiguration | null\n      ) => parent ?? new CopilotChatConfigurationService(initial ?? null),\n    },\n  ],\n  template: `\n    <copilot-chat-view\n      [messages]=\"messages()\"\n      [autoScroll]=\"true\"\n      [messageViewClass]=\"'w-full'\"\n      [showCursor]=\"showCursor()\"\n    >\n    </copilot-chat-view>\n  `,\n})\nexport class CopilotChatComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() agentId?: string;\n  @Input() threadId?: string;\n\n  constructor(\n    @Optional() private chatConfig: CopilotChatConfigurationService | null,\n    private cdr: ChangeDetectorRef,\n    private injector: Injector\n  ) {\n    // Create initial watcher once (constructor is a safe injection context)\n    const initialId = this.agentId ?? DEFAULT_AGENT_ID;\n    this.createWatcher(initialId);\n\n    // Connect once when agent becomes available\n    effect(() => {\n      const a = this.agent();\n      if (!a) return;\n      // Apply thread id when agent is available\n      a.threadId = this.threadId || this.generatedThreadId;\n      if (!this.hasConnectedOnce) {\n        this.hasConnectedOnce = true;\n        this.connectToAgent(a);\n      }\n    });\n  }\n\n  // Signals from watchAgent - using direct references instead of assignment\n  protected agent: Signal<AbstractAgent | undefined> = signal<\n    AbstractAgent | undefined\n  >(undefined).asReadonly() as unknown as Signal<AbstractAgent | undefined>;\n  protected messages: Signal<Message[]> = signal<Message[]>(\n    []\n  ).asReadonly() as unknown as Signal<Message[]>;\n  protected isRunning: Signal<boolean> = signal<boolean>(\n    false\n  ).asReadonly() as unknown as Signal<boolean>;\n  protected showCursor = signal<boolean>(false);\n\n  private generatedThreadId: string = randomUUID();\n  private agentWatcher?: AgentWatchResult;\n  private hasConnectedOnce = false;\n  private lastAgentId?: string;\n\n  ngOnInit(): void {\n    this.setupChatHandlers();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[\"agentId\"] && !changes[\"agentId\"].firstChange) {\n      const newId = this.agentId ?? DEFAULT_AGENT_ID;\n      this.createWatcher(newId);\n    }\n    if (changes[\"threadId\"] && !changes[\"threadId\"].firstChange) {\n      const a = this.agent();\n      if (a) {\n        a.threadId = this.threadId || this.generatedThreadId;\n      }\n    }\n  }\n\n  private async connectToAgent(agent: AbstractAgent): Promise<void> {\n    if (!agent) return;\n\n    this.showCursor.set(true);\n    this.cdr.markForCheck();\n\n    try {\n      await agent.runAgent(\n        { forwardedProps: { __copilotkitConnect: true } },\n        {\n          onTextMessageStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n          onToolCallStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n        }\n      );\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    } catch (error) {\n      console.error(\"Failed to connect to agent:\", error);\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    }\n  }\n\n  private setupChatHandlers(): void {\n    if (!this.chatConfig) return;\n\n    // Handle input submission\n    this.chatConfig.setSubmitHandler(async (value: string) => {\n      const agent = this.agent();\n      if (!agent || !value.trim()) return;\n\n      // Add user message\n      const userMessage: Message = {\n        id: randomUUID(),\n        role: \"user\",\n        content: value,\n      };\n      agent.addMessage(userMessage);\n\n      // Clear the input\n      this.chatConfig!.setInputValue(\"\");\n\n      // Show cursor while processing\n      this.showCursor.set(true);\n      this.cdr.markForCheck();\n\n      // Run the agent with named subscriber callbacks\n      try {\n        await agent.runAgent(\n          {},\n          {\n            onTextMessageStartEvent: () => {\n              this.showCursor.set(false);\n              this.cdr.detectChanges();\n            },\n            onToolCallStartEvent: () => {\n              this.showCursor.set(false);\n              this.cdr.detectChanges();\n            },\n          }\n        );\n      } catch (error) {\n        console.error(\"Agent run error:\", error);\n      } finally {\n        this.showCursor.set(false);\n        this.cdr.markForCheck();\n      }\n    });\n\n    // Handle input value changes (optional)\n    this.chatConfig.setChangeHandler(() => {\n      // Keep input state if needed\n    });\n  }\n\n  ngOnDestroy(): void {\n    if (this.agentWatcher?.unsubscribe) {\n      this.agentWatcher.unsubscribe();\n    }\n  }\n\n  private createWatcher(desiredAgentId: string) {\n    // Tear down previous watcher if it exists\n    if (this.agentWatcher?.unsubscribe) {\n      this.agentWatcher.unsubscribe();\n      this.agentWatcher = undefined;\n    }\n    // Setup watcher for desired agent - ensure injection context\n    this.agentWatcher = runInInjectionContext(this.injector, () =>\n      watchAgent({ agentId: desiredAgentId })\n    );\n    this.agent = this.agentWatcher.agent;\n    this.messages = this.agentWatcher.messages;\n    this.isRunning = this.agentWatcher.isRunning;\n    this.hasConnectedOnce = false;\n    this.lastAgentId = desiredAgentId;\n  }\n}\n"]}
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import { Component, Input, ViewContainerRef, TemplateRef, inject, ViewChild, } from "@angular/core";
|
|
2
|
+
import { CommonModule } from "@angular/common";
|
|
3
|
+
import { CopilotKitService } from "../core/copilotkit.service";
|
|
4
|
+
import { ToolCallStatus } from "../core/copilotkit.types";
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/common";
|
|
7
|
+
export class CopilotKitToolRenderComponent {
|
|
8
|
+
toolName;
|
|
9
|
+
args;
|
|
10
|
+
status = ToolCallStatus.InProgress;
|
|
11
|
+
result;
|
|
12
|
+
description;
|
|
13
|
+
container;
|
|
14
|
+
copilotkit = inject(CopilotKitService);
|
|
15
|
+
componentRef;
|
|
16
|
+
templateRef;
|
|
17
|
+
templateContext;
|
|
18
|
+
ngAfterViewInit() {
|
|
19
|
+
this.renderTool();
|
|
20
|
+
}
|
|
21
|
+
ngOnChanges(changes) {
|
|
22
|
+
if (changes["toolName"] ||
|
|
23
|
+
changes["args"] ||
|
|
24
|
+
changes["status"] ||
|
|
25
|
+
changes["result"]) {
|
|
26
|
+
this.renderTool();
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
renderTool() {
|
|
30
|
+
// Clear existing component
|
|
31
|
+
if (this.componentRef) {
|
|
32
|
+
this.componentRef.destroy();
|
|
33
|
+
this.componentRef = undefined;
|
|
34
|
+
}
|
|
35
|
+
// Clear template
|
|
36
|
+
this.templateRef = undefined;
|
|
37
|
+
this.templateContext = undefined;
|
|
38
|
+
if (!this.toolName) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
// Get the tool render configuration
|
|
42
|
+
const toolRender = this.copilotkit.getToolRender(this.toolName);
|
|
43
|
+
if (!toolRender) {
|
|
44
|
+
console.warn(`No render found for tool: ${this.toolName}`);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
// Prepare props to pass to the component
|
|
48
|
+
const props = {
|
|
49
|
+
name: this.toolName,
|
|
50
|
+
description: this.description || "",
|
|
51
|
+
args: this.args,
|
|
52
|
+
status: this.status,
|
|
53
|
+
result: this.result,
|
|
54
|
+
};
|
|
55
|
+
// Check if render is a Component class or TemplateRef
|
|
56
|
+
if (this.isComponentClass(toolRender.render)) {
|
|
57
|
+
// Create component dynamically
|
|
58
|
+
this.renderComponent(toolRender.render, props);
|
|
59
|
+
}
|
|
60
|
+
else if (this.isTemplateRef(toolRender.render)) {
|
|
61
|
+
// Use template
|
|
62
|
+
this.renderTemplate(toolRender.render, props);
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
console.error(`Invalid render type for tool: ${this.toolName}`);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
renderComponent(componentClass, props) {
|
|
69
|
+
// Clear the container
|
|
70
|
+
this.container.clear();
|
|
71
|
+
// Create the component
|
|
72
|
+
this.componentRef = this.container.createComponent(componentClass);
|
|
73
|
+
// Set inputs on the component using setInput
|
|
74
|
+
// Try setting a single 'props' input first
|
|
75
|
+
try {
|
|
76
|
+
this.componentRef.setInput('props', props);
|
|
77
|
+
}
|
|
78
|
+
catch (e) {
|
|
79
|
+
// If props input doesn't exist, try setting individual inputs
|
|
80
|
+
for (const [key, value] of Object.entries(props)) {
|
|
81
|
+
try {
|
|
82
|
+
this.componentRef.setInput(key, value);
|
|
83
|
+
}
|
|
84
|
+
catch (inputError) {
|
|
85
|
+
// Input might not exist on the component, skip it
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
// Trigger change detection
|
|
90
|
+
this.componentRef.changeDetectorRef.detectChanges();
|
|
91
|
+
}
|
|
92
|
+
renderTemplate(template, props) {
|
|
93
|
+
this.templateRef = template;
|
|
94
|
+
this.templateContext = {
|
|
95
|
+
$implicit: props,
|
|
96
|
+
name: props.name,
|
|
97
|
+
description: props.description,
|
|
98
|
+
args: props.args,
|
|
99
|
+
status: props.status,
|
|
100
|
+
result: props.result,
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
isComponentClass(value) {
|
|
104
|
+
return typeof value === "function" && value.prototype;
|
|
105
|
+
}
|
|
106
|
+
isTemplateRef(value) {
|
|
107
|
+
return value instanceof TemplateRef;
|
|
108
|
+
}
|
|
109
|
+
ngOnDestroy() {
|
|
110
|
+
if (this.componentRef) {
|
|
111
|
+
this.componentRef.destroy();
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitToolRenderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
115
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitToolRenderComponent, isStandalone: true, selector: "copilotkit-tool-render", inputs: { toolName: "toolName", args: "args", status: "status", result: "result", description: "description" }, viewQueries: [{ propertyName: "container", first: true, predicate: ["dynamicContainer"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: `
|
|
116
|
+
<ng-container #dynamicContainer></ng-container>
|
|
117
|
+
<ng-container *ngIf="templateRef && templateContext">
|
|
118
|
+
<ng-container
|
|
119
|
+
*ngTemplateOutlet="templateRef; context: templateContext"
|
|
120
|
+
></ng-container>
|
|
121
|
+
</ng-container>
|
|
122
|
+
`, 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"] }] });
|
|
123
|
+
}
|
|
124
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitToolRenderComponent, decorators: [{
|
|
125
|
+
type: Component,
|
|
126
|
+
args: [{
|
|
127
|
+
selector: "copilotkit-tool-render",
|
|
128
|
+
standalone: true,
|
|
129
|
+
imports: [CommonModule],
|
|
130
|
+
template: `
|
|
131
|
+
<ng-container #dynamicContainer></ng-container>
|
|
132
|
+
<ng-container *ngIf="templateRef && templateContext">
|
|
133
|
+
<ng-container
|
|
134
|
+
*ngTemplateOutlet="templateRef; context: templateContext"
|
|
135
|
+
></ng-container>
|
|
136
|
+
</ng-container>
|
|
137
|
+
`,
|
|
138
|
+
}]
|
|
139
|
+
}], propDecorators: { toolName: [{
|
|
140
|
+
type: Input
|
|
141
|
+
}], args: [{
|
|
142
|
+
type: Input
|
|
143
|
+
}], status: [{
|
|
144
|
+
type: Input
|
|
145
|
+
}], result: [{
|
|
146
|
+
type: Input
|
|
147
|
+
}], description: [{
|
|
148
|
+
type: Input
|
|
149
|
+
}], container: [{
|
|
150
|
+
type: ViewChild,
|
|
151
|
+
args: ["dynamicContainer", { read: ViewContainerRef, static: true }]
|
|
152
|
+
}] } });
|
|
153
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilotkit-tool-render.component.js","sourceRoot":"","sources":["../../../src/components/copilotkit-tool-render.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  ViewContainerRef,\n  TemplateRef,\n  Type,\n  OnChanges,\n  SimpleChanges,\n  ComponentRef,\n  inject,\n  ViewChild,\n  AfterViewInit,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { CopilotKitService } from \"../core/copilotkit.service\";\nimport type {\n  ToolCallProps,\n  AngularToolCallRender,\n} from \"../core/copilotkit.types\";\nimport { ToolCallStatus } from \"../core/copilotkit.types\";\n\n@Component({\n  selector: \"copilotkit-tool-render\",\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <ng-container #dynamicContainer></ng-container>\n    <ng-container *ngIf=\"templateRef && templateContext\">\n      <ng-container\n        *ngTemplateOutlet=\"templateRef; context: templateContext\"\n      ></ng-container>\n    </ng-container>\n  `,\n})\nexport class CopilotKitToolRenderComponent implements OnChanges, AfterViewInit {\n  @Input() toolName!: string;\n  @Input() args: any;\n  @Input() status: ToolCallStatus = ToolCallStatus.InProgress;\n  @Input() result?: any;\n  @Input() description?: string;\n\n  @ViewChild(\"dynamicContainer\", { read: ViewContainerRef, static: true })\n  private container!: ViewContainerRef;\n\n  private copilotkit = inject(CopilotKitService);\n  private componentRef?: ComponentRef<any>;\n\n  templateRef?: TemplateRef<any>;\n  templateContext?: any;\n\n  ngAfterViewInit(): void {\n    this.renderTool();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (\n      changes[\"toolName\"] ||\n      changes[\"args\"] ||\n      changes[\"status\"] ||\n      changes[\"result\"]\n    ) {\n      this.renderTool();\n    }\n  }\n\n  private renderTool(): void {\n    // Clear existing component\n    if (this.componentRef) {\n      this.componentRef.destroy();\n      this.componentRef = undefined;\n    }\n\n    // Clear template\n    this.templateRef = undefined;\n    this.templateContext = undefined;\n\n    if (!this.toolName) {\n      return;\n    }\n\n    // Get the tool render configuration\n    const toolRender = this.copilotkit.getToolRender(this.toolName) as\n      | AngularToolCallRender\n      | undefined;\n\n    if (!toolRender) {\n      console.warn(`No render found for tool: ${this.toolName}`);\n      return;\n    }\n\n    // Prepare props to pass to the component\n    const props: ToolCallProps<any> = {\n      name: this.toolName,\n      description: this.description || \"\",\n      args: this.args,\n      status: this.status,\n      result: this.result,\n    };\n\n    // Check if render is a Component class or TemplateRef\n    if (this.isComponentClass(toolRender.render)) {\n      // Create component dynamically\n      this.renderComponent(toolRender.render, props);\n    } else if (this.isTemplateRef(toolRender.render)) {\n      // Use template\n      this.renderTemplate(toolRender.render, props);\n    } else {\n      console.error(`Invalid render type for tool: ${this.toolName}`);\n    }\n  }\n\n  private renderComponent(\n    componentClass: Type<any>,\n    props: ToolCallProps\n  ): void {\n    // Clear the container\n    this.container.clear();\n\n    // Create the component\n    this.componentRef = this.container.createComponent(componentClass);\n\n    // Set inputs on the component using setInput\n    // Try setting a single 'props' input first\n    try {\n      this.componentRef.setInput('props', props);\n    } catch (e) {\n      // If props input doesn't exist, try setting individual inputs\n      for (const [key, value] of Object.entries(props)) {\n        try {\n          this.componentRef.setInput(key, value);\n        } catch (inputError) {\n          // Input might not exist on the component, skip it\n        }\n      }\n    }\n\n    // Trigger change detection\n    this.componentRef.changeDetectorRef.detectChanges();\n  }\n\n  private renderTemplate(\n    template: TemplateRef<any>,\n    props: ToolCallProps\n  ): void {\n    this.templateRef = template;\n    this.templateContext = {\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  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  ngOnDestroy(): void {\n    if (this.componentRef) {\n      this.componentRef.destroy();\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './index';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdGtpdG5leHQtYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb3BpbG90a2l0bmV4dC1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vaW5kZXgnO1xuIl19
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { CopilotChatConfigurationService } from './chat-configuration.service';
|
|
2
|
+
import { COPILOT_CHAT_INITIAL_CONFIG } from './chat-configuration.types';
|
|
3
|
+
/**
|
|
4
|
+
* Provides CopilotKit chat configuration at a specific component level.
|
|
5
|
+
* This allows for scoped configuration where different parts of the app
|
|
6
|
+
* can have different chat configurations.
|
|
7
|
+
*
|
|
8
|
+
* @param config - Optional initial configuration
|
|
9
|
+
* @returns Array of providers
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```typescript
|
|
13
|
+
* // Global configuration in app.config.ts
|
|
14
|
+
* export const appConfig: ApplicationConfig = {
|
|
15
|
+
* providers: [
|
|
16
|
+
* provideCopilotChatConfiguration({
|
|
17
|
+
* labels: {
|
|
18
|
+
* chatInputPlaceholder: "How can I help you today?"
|
|
19
|
+
* }
|
|
20
|
+
* })
|
|
21
|
+
* ]
|
|
22
|
+
* };
|
|
23
|
+
*
|
|
24
|
+
* // Component-scoped configuration
|
|
25
|
+
* @Component({
|
|
26
|
+
* selector: 'customer-support-chat',
|
|
27
|
+
* providers: [
|
|
28
|
+
* provideCopilotChatConfiguration({
|
|
29
|
+
* labels: {
|
|
30
|
+
* chatInputPlaceholder: "Describe your issue..."
|
|
31
|
+
* },
|
|
32
|
+
* onSubmitInput: (value) => console.log('Support message:', value)
|
|
33
|
+
* })
|
|
34
|
+
* ],
|
|
35
|
+
* template: `...`
|
|
36
|
+
* })
|
|
37
|
+
* export class CustomerSupportChatComponent {}
|
|
38
|
+
*
|
|
39
|
+
* // Multiple independent chats
|
|
40
|
+
* @Component({
|
|
41
|
+
* selector: 'sales-chat',
|
|
42
|
+
* providers: [
|
|
43
|
+
* provideCopilotChatConfiguration({
|
|
44
|
+
* labels: {
|
|
45
|
+
* chatInputPlaceholder: "Ask about our products..."
|
|
46
|
+
* }
|
|
47
|
+
* })
|
|
48
|
+
* ],
|
|
49
|
+
* template: `...`
|
|
50
|
+
* })
|
|
51
|
+
* export class SalesChatComponent {}
|
|
52
|
+
* ```
|
|
53
|
+
*/
|
|
54
|
+
export function provideCopilotChatConfiguration(config) {
|
|
55
|
+
return [
|
|
56
|
+
// Provide the service
|
|
57
|
+
CopilotChatConfigurationService,
|
|
58
|
+
// Provide the initial configuration
|
|
59
|
+
{
|
|
60
|
+
provide: COPILOT_CHAT_INITIAL_CONFIG,
|
|
61
|
+
useValue: config || {}
|
|
62
|
+
}
|
|
63
|
+
];
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1jb25maWd1cmF0aW9uLnByb3ZpZGVycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb3JlL2NoYXQtY29uZmlndXJhdGlvbi9jaGF0LWNvbmZpZ3VyYXRpb24ucHJvdmlkZXJzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSwrQkFBK0IsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQy9FLE9BQU8sRUFFTCwyQkFBMkIsRUFDNUIsTUFBTSw0QkFBNEIsQ0FBQztBQUVwQzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FrREc7QUFDSCxNQUFNLFVBQVUsK0JBQStCLENBQzdDLE1BQWlDO0lBRWpDLE9BQU87UUFDTCxzQkFBc0I7UUFDdEIsK0JBQStCO1FBQy9CLG9DQUFvQztRQUNwQztZQUNFLE9BQU8sRUFBRSwyQkFBMkI7WUFDcEMsUUFBUSxFQUFFLE1BQU0sSUFBSSxFQUFFO1NBQ3ZCO0tBQ0YsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQcm92aWRlciB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSB9IGZyb20gJy4vY2hhdC1jb25maWd1cmF0aW9uLnNlcnZpY2UnO1xuaW1wb3J0IHsgXG4gIENvcGlsb3RDaGF0Q29uZmlndXJhdGlvbixcbiAgQ09QSUxPVF9DSEFUX0lOSVRJQUxfQ09ORklHIFxufSBmcm9tICcuL2NoYXQtY29uZmlndXJhdGlvbi50eXBlcyc7XG5cbi8qKlxuICogUHJvdmlkZXMgQ29waWxvdEtpdCBjaGF0IGNvbmZpZ3VyYXRpb24gYXQgYSBzcGVjaWZpYyBjb21wb25lbnQgbGV2ZWwuXG4gKiBUaGlzIGFsbG93cyBmb3Igc2NvcGVkIGNvbmZpZ3VyYXRpb24gd2hlcmUgZGlmZmVyZW50IHBhcnRzIG9mIHRoZSBhcHBcbiAqIGNhbiBoYXZlIGRpZmZlcmVudCBjaGF0IGNvbmZpZ3VyYXRpb25zLlxuICogXG4gKiBAcGFyYW0gY29uZmlnIC0gT3B0aW9uYWwgaW5pdGlhbCBjb25maWd1cmF0aW9uXG4gKiBAcmV0dXJucyBBcnJheSBvZiBwcm92aWRlcnNcbiAqIFxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIEdsb2JhbCBjb25maWd1cmF0aW9uIGluIGFwcC5jb25maWcudHNcbiAqIGV4cG9ydCBjb25zdCBhcHBDb25maWc6IEFwcGxpY2F0aW9uQ29uZmlnID0ge1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICBwcm92aWRlQ29waWxvdENoYXRDb25maWd1cmF0aW9uKHtcbiAqICAgICAgIGxhYmVsczoge1xuICogICAgICAgICBjaGF0SW5wdXRQbGFjZWhvbGRlcjogXCJIb3cgY2FuIEkgaGVscCB5b3UgdG9kYXk/XCJcbiAqICAgICAgIH1cbiAqICAgICB9KVxuICogICBdXG4gKiB9O1xuICogXG4gKiAvLyBDb21wb25lbnQtc2NvcGVkIGNvbmZpZ3VyYXRpb25cbiAqIEBDb21wb25lbnQoe1xuICogICBzZWxlY3RvcjogJ2N1c3RvbWVyLXN1cHBvcnQtY2hhdCcsXG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVDb3BpbG90Q2hhdENvbmZpZ3VyYXRpb24oe1xuICogICAgICAgbGFiZWxzOiB7XG4gKiAgICAgICAgIGNoYXRJbnB1dFBsYWNlaG9sZGVyOiBcIkRlc2NyaWJlIHlvdXIgaXNzdWUuLi5cIlxuICogICAgICAgfSxcbiAqICAgICAgIG9uU3VibWl0SW5wdXQ6ICh2YWx1ZSkgPT4gY29uc29sZS5sb2coJ1N1cHBvcnQgbWVzc2FnZTonLCB2YWx1ZSlcbiAqICAgICB9KVxuICogICBdLFxuICogICB0ZW1wbGF0ZTogYC4uLmBcbiAqIH0pXG4gKiBleHBvcnQgY2xhc3MgQ3VzdG9tZXJTdXBwb3J0Q2hhdENvbXBvbmVudCB7fVxuICogXG4gKiAvLyBNdWx0aXBsZSBpbmRlcGVuZGVudCBjaGF0c1xuICogQENvbXBvbmVudCh7XG4gKiAgIHNlbGVjdG9yOiAnc2FsZXMtY2hhdCcsXG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVDb3BpbG90Q2hhdENvbmZpZ3VyYXRpb24oe1xuICogICAgICAgbGFiZWxzOiB7XG4gKiAgICAgICAgIGNoYXRJbnB1dFBsYWNlaG9sZGVyOiBcIkFzayBhYm91dCBvdXIgcHJvZHVjdHMuLi5cIlxuICogICAgICAgfVxuICogICAgIH0pXG4gKiAgIF0sXG4gKiAgIHRlbXBsYXRlOiBgLi4uYFxuICogfSlcbiAqIGV4cG9ydCBjbGFzcyBTYWxlc0NoYXRDb21wb25lbnQge31cbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZUNvcGlsb3RDaGF0Q29uZmlndXJhdGlvbihcbiAgY29uZmlnPzogQ29waWxvdENoYXRDb25maWd1cmF0aW9uXG4pOiBQcm92aWRlcltdIHtcbiAgcmV0dXJuIFtcbiAgICAvLyBQcm92aWRlIHRoZSBzZXJ2aWNlXG4gICAgQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSxcbiAgICAvLyBQcm92aWRlIHRoZSBpbml0aWFsIGNvbmZpZ3VyYXRpb25cbiAgICB7XG4gICAgICBwcm92aWRlOiBDT1BJTE9UX0NIQVRfSU5JVElBTF9DT05GSUcsXG4gICAgICB1c2VWYWx1ZTogY29uZmlnIHx8IHt9XG4gICAgfVxuICBdO1xufSJdfQ==
|